diff --git a/assets/dummy.txt b/assets/.gitkeep similarity index 100% rename from assets/dummy.txt rename to assets/.gitkeep diff --git a/assets/models/suzanne.dae b/assets/models/suzanne.dae new file mode 100644 index 0000000..6ae15a5 --- /dev/null +++ b/assets/models/suzanne.dae @@ -0,0 +1,69 @@ + + + + + Blender User + Blender 2.90.0 commit date:2020-08-31, commit time:11:26, hash:0330d1af29c0 + + 2020-09-23T23:04:35 + 2020-09-23T23:04:35 + + Z_UP + + + + + + + 0.4375 -0.765625 0.1640625 -0.4375 -0.765625 0.1640625 0.5 -0.6875 0.09375 -0.5 -0.6875 0.09375 0.546875 -0.578125 0.0546875 -0.546875 -0.578125 0.0546875 0.3515625 -0.6171875 -0.0234375 -0.3515625 -0.6171875 -0.0234375 0.3515625 -0.71875 0.03125 -0.3515625 -0.71875 0.03125 0.3515625 -0.78125 0.1328125 -0.3515625 -0.78125 0.1328125 0.2734375 -0.796875 0.1640625 -0.2734375 -0.796875 0.1640625 0.203125 -0.7421875 0.09375 -0.203125 -0.7421875 0.09375 0.15625 -0.6484375 0.0546875 -0.15625 -0.6484375 0.0546875 0.078125 -0.65625 0.2421875 -0.078125 -0.65625 0.2421875 0.140625 -0.7421875 0.2421875 -0.140625 -0.7421875 0.2421875 0.2421875 -0.796875 0.2421875 -0.2421875 -0.796875 0.2421875 0.2734375 -0.796875 0.328125 -0.2734375 -0.796875 0.328125 0.203125 -0.7421875 0.390625 -0.203125 -0.7421875 0.390625 0.15625 -0.6484375 0.4375 -0.15625 -0.6484375 0.4375 0.3515625 -0.6171875 0.515625 -0.3515625 -0.6171875 0.515625 0.3515625 -0.71875 0.453125 -0.3515625 -0.71875 0.453125 0.3515625 -0.78125 0.359375 -0.3515625 -0.78125 0.359375 0.4375 -0.765625 0.328125 -0.4375 -0.765625 0.328125 0.5 -0.6875 0.390625 -0.5 -0.6875 0.390625 0.546875 -0.578125 0.4375 -0.546875 -0.578125 0.4375 0.625 -0.5625 0.2421875 -0.625 -0.5625 0.2421875 0.5625 -0.671875 0.2421875 -0.5625 -0.671875 0.2421875 0.46875 -0.7578125 0.2421875 -0.46875 -0.7578125 0.2421875 0.4765625 -0.7734375 0.2421875 -0.4765625 -0.7734375 0.2421875 0.4453125 -0.78125 0.3359375 -0.4453125 -0.78125 0.3359375 0.3515625 -0.8046875 0.375 -0.3515625 -0.8046875 0.375 0.265625 -0.8203125 0.3359375 -0.265625 -0.8203125 0.3359375 0.2265625 -0.8203125 0.2421875 -0.2265625 -0.8203125 0.2421875 0.265625 -0.8203125 0.15625 -0.265625 -0.8203125 0.15625 0.3515625 -0.828125 0.2421875 -0.3515625 -0.828125 0.2421875 0.3515625 -0.8046875 0.1171875 -0.3515625 -0.8046875 0.1171875 0.4453125 -0.78125 0.15625 -0.4453125 -0.78125 0.15625 0 -0.7421875 0.4296875 0 -0.8203125 0.3515625 0 -0.734375 -0.6796875 0 -0.78125 -0.3203125 0 -0.796875 -0.1875 0 -0.71875 -0.7734375 0 -0.6015625 0.40625 0 -0.5703125 0.5703125 0 0.546875 0.8984375 0 0.8515625 0.5625 0 0.828125 0.0703125 0 0.3515625 -0.3828125 0.203125 -0.5625 -0.1875 -0.203125 -0.5625 -0.1875 0.3125 -0.5703125 -0.4375 -0.3125 -0.5703125 -0.4375 0.3515625 -0.5703125 -0.6953125 -0.3515625 -0.5703125 -0.6953125 0.3671875 -0.53125 -0.890625 -0.3671875 -0.53125 -0.890625 0.328125 -0.5234375 -0.9453125 -0.328125 -0.5234375 -0.9453125 0.1796875 -0.5546875 -0.96875 -0.1796875 -0.5546875 -0.96875 0 -0.578125 -0.984375 0.4375 -0.53125 -0.140625 -0.4375 -0.53125 -0.140625 0.6328125 -0.5390625 -0.0390625 -0.6328125 -0.5390625 -0.0390625 0.828125 -0.4453125 0.1484375 -0.828125 -0.4453125 0.1484375 0.859375 -0.59375 0.4296875 -0.859375 -0.59375 0.4296875 0.7109375 -0.625 0.484375 -0.7109375 -0.625 0.484375 0.4921875 -0.6875 0.6015625 -0.4921875 -0.6875 0.6015625 0.3203125 -0.734375 0.7578125 -0.3203125 -0.734375 0.7578125 0.15625 -0.7578125 0.71875 -0.15625 -0.7578125 0.71875 0.0625 -0.75 0.4921875 -0.0625 -0.75 0.4921875 0.1640625 -0.7734375 0.4140625 -0.1640625 -0.7734375 0.4140625 0.125 -0.765625 0.3046875 -0.125 -0.765625 0.3046875 0.203125 -0.7421875 0.09375 -0.203125 -0.7421875 0.09375 0.375 -0.703125 0.015625 -0.375 -0.703125 0.015625 0.4921875 -0.671875 0.0625 -0.4921875 -0.671875 0.0625 0.625 -0.6484375 0.1875 -0.625 -0.6484375 0.1875 0.640625 -0.6484375 0.296875 -0.640625 -0.6484375 0.296875 0.6015625 -0.6640625 0.375 -0.6015625 -0.6640625 0.375 0.4296875 -0.71875 0.4375 -0.4296875 -0.71875 0.4375 0.25 -0.7578125 0.46875 -0.25 -0.7578125 0.46875 0 -0.734375 -0.765625 0.109375 -0.734375 -0.71875 -0.109375 -0.734375 -0.71875 0.1171875 -0.7109375 -0.8359375 -0.1171875 -0.7109375 -0.8359375 0.0625 -0.6953125 -0.8828125 -0.0625 -0.6953125 -0.8828125 0 -0.6875 -0.890625 0 -0.75 -0.1953125 0 -0.7421875 -0.140625 0.1015625 -0.7421875 -0.1484375 -0.1015625 -0.7421875 -0.1484375 0.125 -0.75 -0.2265625 -0.125 -0.75 -0.2265625 0.0859375 -0.7421875 -0.2890625 -0.0859375 -0.7421875 -0.2890625 0.3984375 -0.671875 -0.046875 -0.3984375 -0.671875 -0.046875 0.6171875 -0.625 0.0546875 -0.6171875 -0.625 0.0546875 0.7265625 -0.6015625 0.203125 -0.7265625 -0.6015625 0.203125 0.7421875 -0.65625 0.375 -0.7421875 -0.65625 0.375 0.6875 -0.7265625 0.4140625 -0.6875 -0.7265625 0.4140625 0.4375 -0.796875 0.546875 -0.4375 -0.796875 0.546875 0.3125 -0.8359375 0.640625 -0.3125 -0.8359375 0.640625 0.203125 -0.8515625 0.6171875 -0.203125 -0.8515625 0.6171875 0.1015625 -0.84375 0.4296875 -0.1015625 -0.84375 0.4296875 0.125 -0.8125 -0.1015625 -0.125 -0.8125 -0.1015625 0.2109375 -0.7109375 -0.4453125 -0.2109375 -0.7109375 -0.4453125 0.25 -0.6875 -0.703125 -0.25 -0.6875 -0.703125 0.265625 -0.6640625 -0.8203125 -0.265625 -0.6640625 -0.8203125 0.234375 -0.6328125 -0.9140625 -0.234375 -0.6328125 -0.9140625 0.1640625 -0.6328125 -0.9296875 -0.1640625 -0.6328125 -0.9296875 0 -0.640625 -0.9453125 0 -0.7265625 0.046875 0 -0.765625 0.2109375 0.328125 -0.7421875 0.4765625 -0.328125 -0.7421875 0.4765625 0.1640625 -0.75 0.140625 -0.1640625 -0.75 0.140625 0.1328125 -0.7578125 0.2109375 -0.1328125 -0.7578125 0.2109375 0.1171875 -0.734375 -0.6875 -0.1171875 -0.734375 -0.6875 0.078125 -0.75 -0.4453125 -0.078125 -0.75 -0.4453125 0 -0.75 -0.4453125 0 -0.7421875 -0.328125 0.09375 -0.78125 -0.2734375 -0.09375 -0.78125 -0.2734375 0.1328125 -0.796875 -0.2265625 -0.1328125 -0.796875 -0.2265625 0.109375 -0.78125 -0.1328125 -0.109375 -0.78125 -0.1328125 0.0390625 -0.78125 -0.125 -0.0390625 -0.78125 -0.125 0 -0.828125 -0.203125 0.046875 -0.8125 -0.1484375 -0.046875 -0.8125 -0.1484375 0.09375 -0.8125 -0.15625 -0.09375 -0.8125 -0.15625 0.109375 -0.828125 -0.2265625 -0.109375 -0.828125 -0.2265625 0.078125 -0.8046875 -0.25 -0.078125 -0.8046875 -0.25 0 -0.8046875 -0.2890625 0.2578125 -0.5546875 -0.3125 -0.2578125 -0.5546875 -0.3125 0.1640625 -0.7109375 -0.2421875 -0.1640625 -0.7109375 -0.2421875 0.1796875 -0.7109375 -0.3125 -0.1796875 -0.7109375 -0.3125 0.234375 -0.5546875 -0.25 -0.234375 -0.5546875 -0.25 0 -0.6875 -0.875 0.046875 -0.6875 -0.8671875 -0.046875 -0.6875 -0.8671875 0.09375 -0.7109375 -0.8203125 -0.09375 -0.7109375 -0.8203125 0.09375 -0.7265625 -0.7421875 -0.09375 -0.7265625 -0.7421875 0 -0.65625 -0.78125 0.09375 -0.6640625 -0.75 -0.09375 -0.6640625 -0.75 0.09375 -0.640625 -0.8125 -0.09375 -0.640625 -0.8125 0.046875 -0.6328125 -0.8515625 -0.046875 -0.6328125 -0.8515625 0 -0.6328125 -0.859375 0.171875 -0.78125 0.21875 -0.171875 -0.78125 0.21875 0.1875 -0.7734375 0.15625 -0.1875 -0.7734375 0.15625 0.3359375 -0.7578125 0.4296875 -0.3359375 -0.7578125 0.4296875 0.2734375 -0.7734375 0.421875 -0.2734375 -0.7734375 0.421875 0.421875 -0.7734375 0.3984375 -0.421875 -0.7734375 0.3984375 0.5625 -0.6953125 0.3515625 -0.5625 -0.6953125 0.3515625 0.5859375 -0.6875 0.2890625 -0.5859375 -0.6875 0.2890625 0.578125 -0.6796875 0.1953125 -0.578125 -0.6796875 0.1953125 0.4765625 -0.71875 0.1015625 -0.4765625 -0.71875 0.1015625 0.375 -0.7421875 0.0625 -0.375 -0.7421875 0.0625 0.2265625 -0.78125 0.109375 -0.2265625 -0.78125 0.109375 0.1796875 -0.78125 0.296875 -0.1796875 -0.78125 0.296875 0.2109375 -0.78125 0.375 -0.2109375 -0.78125 0.375 0.234375 -0.7578125 0.359375 -0.234375 -0.7578125 0.359375 0.1953125 -0.7578125 0.296875 -0.1953125 -0.7578125 0.296875 0.2421875 -0.7578125 0.125 -0.2421875 -0.7578125 0.125 0.375 -0.7265625 0.0859375 -0.375 -0.7265625 0.0859375 0.4609375 -0.703125 0.1171875 -0.4609375 -0.703125 0.1171875 0.546875 -0.671875 0.2109375 -0.546875 -0.671875 0.2109375 0.5546875 -0.671875 0.28125 -0.5546875 -0.671875 0.28125 0.53125 -0.6796875 0.3359375 -0.53125 -0.6796875 0.3359375 0.4140625 -0.75 0.390625 -0.4140625 -0.75 0.390625 0.28125 -0.765625 0.3984375 -0.28125 -0.765625 0.3984375 0.3359375 -0.75 0.40625 -0.3359375 -0.75 0.40625 0.203125 -0.75 0.171875 -0.203125 -0.75 0.171875 0.1953125 -0.75 0.2265625 -0.1953125 -0.75 0.2265625 0.109375 -0.609375 0.4609375 -0.109375 -0.609375 0.4609375 0.1953125 -0.6171875 0.6640625 -0.1953125 -0.6171875 0.6640625 0.3359375 -0.59375 0.6875 -0.3359375 -0.59375 0.6875 0.484375 -0.5546875 0.5546875 -0.484375 -0.5546875 0.5546875 0.6796875 -0.4921875 0.453125 -0.6796875 -0.4921875 0.453125 0.796875 -0.4609375 0.40625 -0.796875 -0.4609375 0.40625 0.7734375 -0.375 0.1640625 -0.7734375 -0.375 0.1640625 0.6015625 -0.4140625 0 -0.6015625 -0.4140625 0 0.4375 -0.46875 -0.09375 -0.4375 -0.46875 -0.09375 0 -0.2890625 0.8984375 0 0.078125 0.984375 0 0.671875 -0.1953125 0 -0.1875 -0.4609375 0 -0.4609375 -0.9765625 0 -0.34375 -0.8046875 0 -0.3203125 -0.5703125 0 -0.28125 -0.484375 0.8515625 -0.0546875 0.234375 -0.8515625 -0.0546875 0.234375 0.859375 0.046875 0.3203125 -0.859375 0.046875 0.3203125 0.7734375 0.4375 0.265625 -0.7734375 0.4375 0.265625 0.4609375 0.703125 0.4375 -0.4609375 0.703125 0.4375 0.734375 -0.0703125 -0.046875 -0.734375 -0.0703125 -0.046875 0.59375 0.1640625 -0.125 -0.59375 0.1640625 -0.125 0.640625 0.4296875 -0.0078125 -0.640625 0.4296875 -0.0078125 0.3359375 0.6640625 0.0546875 -0.3359375 0.6640625 0.0546875 0.234375 -0.40625 -0.3515625 -0.234375 -0.40625 -0.3515625 0.1796875 -0.2578125 -0.4140625 -0.1796875 -0.2578125 -0.4140625 0.2890625 -0.3828125 -0.7109375 -0.2890625 -0.3828125 -0.7109375 0.25 -0.390625 -0.5 -0.25 -0.390625 -0.5 0.328125 -0.3984375 -0.9140625 -0.328125 -0.3984375 -0.9140625 0.140625 -0.3671875 -0.7578125 -0.140625 -0.3671875 -0.7578125 0.125 -0.359375 -0.5390625 -0.125 -0.359375 -0.5390625 0.1640625 -0.4375 -0.9453125 -0.1640625 -0.4375 -0.9453125 0.21875 -0.4296875 -0.28125 -0.21875 -0.4296875 -0.28125 0.2109375 -0.46875 -0.2265625 -0.2109375 -0.46875 -0.2265625 0.203125 -0.5 -0.171875 -0.203125 -0.5 -0.171875 0.2109375 -0.1640625 -0.390625 -0.2109375 -0.1640625 -0.390625 0.296875 0.265625 -0.3125 -0.296875 0.265625 -0.3125 0.34375 0.5390625 -0.1484375 -0.34375 0.5390625 -0.1484375 0.453125 0.3828125 0.8671875 -0.453125 0.3828125 0.8671875 0.453125 0.0703125 0.9296875 -0.453125 0.0703125 0.9296875 0.453125 -0.234375 0.8515625 -0.453125 -0.234375 0.8515625 0.4609375 -0.4296875 0.5234375 -0.4609375 -0.4296875 0.5234375 0.7265625 -0.3359375 0.40625 -0.7265625 -0.3359375 0.40625 0.6328125 -0.28125 0.453125 -0.6328125 -0.28125 0.453125 0.640625 -0.0546875 0.703125 -0.640625 -0.0546875 0.703125 0.796875 -0.125 0.5625 -0.796875 -0.125 0.5625 0.796875 0.1171875 0.6171875 -0.796875 0.1171875 0.6171875 0.640625 0.1953125 0.75 -0.640625 0.1953125 0.75 0.640625 0.4453125 0.6796875 -0.640625 0.4453125 0.6796875 0.796875 0.359375 0.5390625 -0.796875 0.359375 0.5390625 0.6171875 0.5859375 0.328125 -0.6171875 0.5859375 0.328125 0.484375 0.546875 0.0234375 -0.484375 0.546875 0.0234375 0.8203125 0.203125 0.328125 -0.8203125 0.203125 0.328125 0.40625 -0.1484375 -0.171875 -0.40625 -0.1484375 -0.171875 0.4296875 0.2109375 -0.1953125 -0.4296875 0.2109375 -0.1953125 0.890625 0.234375 0.40625 -0.890625 0.234375 0.40625 0.7734375 0.125 -0.140625 -0.7734375 0.125 -0.140625 1.039062 0.328125 -0.1015625 -1.039062 0.328125 -0.1015625 1.28125 0.4296875 0.0546875 -1.28125 0.4296875 0.0546875 1.351562 0.421875 0.3203125 -1.351562 0.421875 0.3203125 1.234375 0.421875 0.5078125 -1.234375 0.421875 0.5078125 1.023437 0.3125 0.4765625 -1.023437 0.3125 0.4765625 1.015625 0.2890625 0.4140625 -1.015625 0.2890625 0.4140625 1.1875 0.390625 0.4375 -1.1875 0.390625 0.4375 1.265625 0.40625 0.2890625 -1.265625 0.40625 0.2890625 1.210937 0.40625 0.078125 -1.210937 0.40625 0.078125 1.03125 0.3046875 -0.0390625 -1.03125 0.3046875 -0.0390625 0.828125 0.1328125 -0.0703125 -0.828125 0.1328125 -0.0703125 0.921875 0.21875 0.359375 -0.921875 0.21875 0.359375 0.9453125 0.2890625 0.3046875 -0.9453125 0.2890625 0.3046875 0.8828125 0.2109375 -0.0234375 -0.8828125 0.2109375 -0.0234375 1.039062 0.3671875 0 -1.039062 0.3671875 0 1.1875 0.4453125 0.09375 -1.1875 0.4453125 0.09375 1.234375 0.4453125 0.25 -1.234375 0.4453125 0.25 1.171875 0.4375 0.359375 -1.171875 0.4375 0.359375 1.023437 0.359375 0.34375 -1.023437 0.359375 0.34375 0.84375 0.2109375 0.2890625 -0.84375 0.2109375 0.2890625 0.8359375 0.2734375 0.171875 -0.8359375 0.2734375 0.171875 0.7578125 0.2734375 0.09375 -0.7578125 0.2734375 0.09375 0.8203125 0.2734375 0.0859375 -0.8203125 0.2734375 0.0859375 0.84375 0.2734375 0.015625 -0.84375 0.2734375 0.015625 0.8125 0.2734375 -0.015625 -0.8125 0.2734375 -0.015625 0.7265625 0.0703125 0 -0.7265625 0.0703125 0 0.71875 0.171875 -0.0234375 -0.71875 0.171875 -0.0234375 0.71875 0.1875 0.0390625 -0.71875 0.1875 0.0390625 0.796875 0.2109375 0.203125 -0.796875 0.2109375 0.203125 0.890625 0.265625 0.2421875 -0.890625 0.265625 0.2421875 0.890625 0.3203125 0.234375 -0.890625 0.3203125 0.234375 0.8125 0.3203125 -0.015625 -0.8125 0.3203125 -0.015625 0.8515625 0.3203125 0.015625 -0.8515625 0.3203125 0.015625 0.828125 0.3203125 0.078125 -0.828125 0.3203125 0.078125 0.765625 0.3203125 0.09375 -0.765625 0.3203125 0.09375 0.84375 0.3203125 0.171875 -0.84375 0.3203125 0.171875 1.039062 0.4140625 0.328125 -1.039062 0.4140625 0.328125 1.1875 0.484375 0.34375 -1.1875 0.484375 0.34375 1.257812 0.4921875 0.2421875 -1.257812 0.4921875 0.2421875 1.210937 0.484375 0.0859375 -1.210937 0.484375 0.0859375 1.046875 0.421875 0 -1.046875 0.421875 0 0.8828125 0.265625 -0.015625 -0.8828125 0.265625 -0.015625 0.953125 0.34375 0.2890625 -0.953125 0.34375 0.2890625 0.890625 0.328125 0.109375 -0.890625 0.328125 0.109375 0.9375 0.3359375 0.0625 -0.9375 0.3359375 0.0625 1 0.3671875 0.125 -1 0.3671875 0.125 0.9609375 0.3515625 0.171875 -0.9609375 0.3515625 0.171875 1.015625 0.375 0.234375 -1.015625 0.375 0.234375 1.054687 0.3828125 0.1875 -1.054687 0.3828125 0.1875 1.109375 0.390625 0.2109375 -1.109375 0.390625 0.2109375 1.085937 0.390625 0.2734375 -1.085937 0.390625 0.2734375 1.023437 0.484375 0.4375 -1.023437 0.484375 0.4375 1.25 0.546875 0.46875 -1.25 0.546875 0.46875 1.367187 0.5 0.296875 -1.367187 0.5 0.296875 1.3125 0.53125 0.0546875 -1.3125 0.53125 0.0546875 1.039062 0.4921875 -0.0859375 -1.039062 0.4921875 -0.0859375 0.7890625 0.328125 -0.125 -0.7890625 0.328125 -0.125 0.859375 0.3828125 0.3828125 -0.859375 0.3828125 0.3828125 + + + + + + + + + + 0.9693107 -0.2455551 -0.01181077 0.6076383 -0.6084929 -0.5104039 0.8001103 -0.5998463 -0.002838194 -0.6076383 -0.6084929 -0.5104039 -0.9693107 -0.2455551 -0.01181077 -0.8001103 -0.5998463 -0.002838194 0.6801811 -0.4888258 -0.5462629 0.868241 -0.4961202 -0.004730463 -0.6801811 -0.4888258 -0.5462629 -0.868241 -0.4961202 -0.004730463 0.1193287 -0.4762772 -0.8711606 -0.1193287 -0.4762772 -0.8711606 0.7290178 -0.1934322 -0.6565952 0.09949201 -0.6522223 -0.7514702 -0.09949201 -0.6522223 -0.7514702 -0.7290178 -0.1934322 -0.6565952 0.03140449 -0.2529148 -0.9669788 -0.4562917 -0.7107604 -0.5353668 0.4562917 -0.7107604 -0.5353668 -0.03140449 -0.2529148 -0.9669788 -0.5538566 -0.5406419 -0.6332056 0.5538566 -0.5406419 -0.6332056 -0.6893172 -0.7244454 -0.004577934 0.6893172 -0.7244454 -0.004577934 0.8097314 -0.5867592 -0.006988823 -0.9525825 -0.3039975 -0.01312303 -0.6530899 -0.3193271 -0.6866615 0.9525825 -0.3039975 -0.01312303 -0.4559605 -0.7206861 0.5222181 0.4559605 -0.7206861 0.5222181 -0.8097314 -0.5867592 -0.006988823 0.5306106 -0.5717203 0.6257702 0.103065 -0.6644136 0.7402245 -0.5306106 -0.5717203 0.6257702 -0.103065 -0.6644136 0.7402245 -0.125739 -0.5252657 0.8415971 0.02572745 -0.2311505 0.9725778 -0.6643726 -0.3055889 0.6820738 -0.02572745 -0.2311505 0.9725778 0.7363902 -0.180275 0.652097 -0.7363902 -0.180275 0.652097 -0.6102255 -0.6180689 0.4955965 0.6102255 -0.6180689 0.4955965 0.125739 -0.5252657 0.8415971 -0.6682124 -0.5147937 0.537103 0.6682124 -0.5147937 0.537103 0.9644749 -0.263871 -0.01266551 -0.9644749 -0.263871 -0.01266551 -0.7216292 -0.2223942 0.6555854 0.7216292 -0.2223942 0.6555854 -0.04315358 -0.3414443 0.9389109 0.04315358 -0.3414443 0.9389109 0.6643726 -0.3055889 0.6820738 0.623694 -0.4646881 0.6285466 -0.623694 -0.4646881 0.6285466 0.9269885 -0.3748666 -0.0129401 -0.6158783 -0.4641061 -0.6366314 -0.9269885 -0.3748666 -0.0129401 0.6158783 -0.4641061 -0.6366314 0.04254424 -0.3374854 -0.9403689 -0.04254424 -0.3374854 -0.9403689 0.6530899 -0.3193271 -0.6866615 0.7151939 -0.2227312 -0.6624867 -0.7151939 -0.2227312 -0.6624867 0.1836023 -0.9829863 -0.005310297 -0.1836023 -0.9829863 -0.005310297 0.1553737 -0.6322988 -0.7589844 0 -0.2522678 -0.9676575 0.1595551 -0.1529018 -0.975276 -0.1553737 -0.6322988 -0.7589844 0 -0.6315561 -0.7753303 0.3502368 -0.6846953 -0.6391608 0.5266669 -0.1611099 -0.834665 -0.3502368 -0.6846953 -0.6391608 -0.1595551 -0.1529018 -0.975276 0.9457384 -0.1977053 -0.2578593 -0.9457384 -0.1977053 -0.2578593 -0.5266669 -0.1611099 -0.834665 0.9728174 -0.2087181 0.1003153 0.555663 -0.7999997 -0.2263609 -0.555663 -0.7999997 -0.2263609 -0.9728174 -0.2087181 0.1003153 0.9557315 -0.1564707 0.2491872 0.5651913 -0.8244242 -0.02972602 -0.5651913 -0.8244242 -0.02972602 -0.9557315 -0.1564707 0.2491872 0.8915622 -0.3094664 -0.3306773 0.3442891 -0.7633202 -0.5466326 0.05609339 -0.9629272 -0.263865 -0.3442891 -0.7633202 -0.5466326 -0.8915622 -0.3094664 -0.3306773 -0.05609339 -0.9629272 -0.263865 0.5874934 -0.1969705 -0.7848913 0.3488331 0.00814855 -0.9371495 -0.5874934 -0.1969705 -0.7848913 -0.4991354 -0.7806716 -0.376053 0.5665653 -0.7598458 -0.3188074 0.4991354 -0.7806716 -0.376053 -0.5665653 -0.7598458 -0.3188074 0.845146 -0.2985112 0.4434178 0.9069906 0.1289423 -0.4009265 -0.845146 -0.2985112 0.4434178 -0.4607163 -0.8756539 -0.1448131 0.5170561 -0.2125356 0.8291452 0.4607163 -0.8756539 -0.1448131 -0.5170561 -0.2125356 0.8291452 -0.4801308 -0.8578386 -0.1832688 0.5975683 -0.1645907 0.7847433 0.4801308 -0.8578386 -0.1832688 -0.5975683 -0.1645907 0.7847433 -0.3084599 -0.9512296 0.003845393 0.266583 -0.9391598 0.216593 0.3084599 -0.9512296 0.003845393 -0.266583 -0.9391598 0.216593 -0.6050714 -0.2097576 0.7680433 0.2312719 -0.1750561 0.9570103 0.6050714 -0.2097576 0.7680433 0.157389 -0.9734805 0.1660261 -0.8242095 -0.1473176 0.5467873 -0.157389 -0.9734805 0.1660261 0.8242095 -0.1473176 0.5467873 0.06109863 -0.9978127 -0.02523905 0 -0.2672881 0.9636167 -0.06109863 -0.9978127 -0.02523905 0 -0.9965715 -0.08273732 0.2581619 -0.9577838 -0.126502 0.3678491 -0.8855785 -0.2836158 -0.2581619 -0.9577838 -0.126502 0.1490263 -0.9767435 -0.1541535 0.2190034 -0.975017 0.03714144 -0.1490263 -0.9767435 -0.1541535 0.2254126 -0.9050075 -0.3607639 -0.2190034 -0.975017 0.03714144 0.3588142 -0.9257695 -0.1191775 -0.2254126 -0.9050075 -0.3607639 0.4602344 -0.872309 -0.1651106 -0.3588142 -0.9257695 -0.1191775 0.4279043 -0.8155848 -0.3895117 -0.4602344 -0.872309 -0.1651106 0.158944 -0.8519436 -0.498928 -0.4279043 -0.8155848 -0.3895117 -0.1281823 -0.9477862 -0.2920115 0.1281823 -0.9477862 -0.2920115 -0.158944 -0.8519436 -0.498928 0 -0.9979325 0.06427246 0.03158771 -0.9834921 -0.178173 -0.03158771 -0.9834921 -0.178173 0 -0.975048 -0.2219944 -0.2005744 -0.9703384 -0.1349574 0.2005744 -0.9703384 -0.1349574 -0.2392998 -0.9230486 -0.3011925 0.2392998 -0.9230486 -0.3011925 -0.05887138 -0.923753 -0.3784372 0.05887138 -0.923753 -0.3784372 0.1307118 -0.9388179 -0.3186463 -0.1307118 -0.9388179 -0.3186463 0.1459442 -0.9819653 -0.1201858 0.5936834 -0.7973968 0.1081588 0.1815272 -0.9823468 -0.04519867 -0.1815272 -0.9823468 -0.04519867 -0.5936834 -0.7973968 0.1081588 -0.1459442 -0.9819653 -0.1201858 0 -0.8794736 -0.4759479 0.1341 -0.9909482 0.006256341 0.5002767 -0.7519716 -0.4292577 -0.1341 -0.9909482 0.006256341 0 -1 0 0 -0.9994199 -0.03405964 0 -0.8096075 -0.5869717 0.932177 -0.3347646 -0.1377635 0.5835901 -0.4234858 -0.6928797 -0.5835901 -0.4234858 -0.6928797 -0.932177 -0.3347646 -0.1377635 -0.5002767 -0.7519716 -0.4292577 0.6186609 -0.1247942 0.7756837 0.5014046 -0.8017162 -0.3253377 -0.6186609 -0.1247942 0.7756837 -0.9262677 -0.2880128 -0.2430575 -0.2405224 -0.2035636 0.9490579 -0.1769512 -0.842731 0.508422 0.2405224 -0.2035636 0.9490579 -0.5014046 -0.8017162 -0.3253377 -0.1949574 -0.5489205 0.812821 -1 0 0 0 -0.5010676 -0.8654083 0 -0.8764336 -0.4815227 -0.1832972 -0.7890386 -0.5863618 -0.1857395 -0.7815036 0.5956115 0.1857395 -0.7815036 0.5956115 0.3611049 -0.8046731 0.4712797 -0.3611049 -0.8046731 0.4712797 0.9262677 -0.2880128 -0.2430575 -0.4488133 -0.8363748 -0.3147125 0.1832972 -0.7890386 -0.5863618 0.4488133 -0.8363748 -0.3147125 0 -0.9874642 0.1578429 0.7751554 -0.6305862 0.03866773 -0.7751554 -0.6305862 0.03866773 -0.6506601 -0.7446579 0.1487485 0.6506601 -0.7446579 0.1487485 0.9277763 -0.1208856 0.3530128 -0.9277763 -0.1208856 0.3530128 0.9306079 -0.1263476 0.3435189 -0.9306079 -0.1263476 0.3435189 -0.1368472 -0.8386013 -0.5272769 0.1368472 -0.8386013 -0.5272769 0 -0.2732335 -0.9619477 -0.6351334 -0.7712182 0.04275727 0.6351334 -0.7712182 0.04275727 -0.4141166 -0.701639 0.5798364 0.4141166 -0.701639 0.5798364 0 -0.9380437 -0.3465171 0 -0.8293219 0.5587714 0 -0.8458586 0.5334073 0.295853 -0.8287789 0.47497 -0.673771 -0.7298651 0.1154538 -0.295853 -0.8287789 0.47497 0.673771 -0.7298651 0.1154538 -0.5176656 -0.4860173 -0.7041376 0.5176656 -0.4860173 -0.7041376 0 -0.7151783 -0.6989423 -0.01010191 -0.9974972 -0.06998103 0.1580563 -0.9838309 -0.08423155 0.01010191 -0.9974972 -0.06998103 -0.1580563 -0.9838309 -0.08423155 0.2934117 -0.9540918 -0.06015336 -0.2934117 -0.9540918 -0.06015336 0.1829642 -0.9777216 -0.1028811 -0.1829642 -0.9777216 -0.1028811 0.03170949 -0.9750295 -0.2197997 0.1844582 -0.9650182 -0.1863199 -0.03170949 -0.9750295 -0.2197997 -0.1844582 -0.9650182 -0.1863199 0.2989927 -0.9535905 -0.03561544 -0.2989927 -0.9535905 -0.03561544 0.2942985 -0.9502521 -0.1020264 -0.2942985 -0.9502521 -0.1020264 0.1775906 -0.9822268 -0.06076353 -0.1775906 -0.9822268 -0.06076353 -0.2943593 -0.9556836 0.00463891 0.2943593 -0.9556836 0.00463891 -0.08868885 -0.9879075 -0.1271736 0.2036247 -0.9735962 0.1031857 0.08868885 -0.9879075 -0.1271736 -0.2036247 -0.9735962 0.1031857 0.1315066 -0.9891837 0.06497508 -0.1315066 -0.9891837 0.06497508 0.4634375 -0.7932609 -0.3949214 0.1743876 -0.9474123 -0.2683262 -0.4634375 -0.7932609 -0.3949214 -0.1743876 -0.9474123 -0.2683262 0.1133162 -0.9425728 -0.3141909 -0.1133162 -0.9425728 -0.3141909 -0.2740567 -0.4391012 -0.8556185 0.2740567 -0.4391012 -0.8556185 -0.1423098 -0.8001762 -0.5826371 0.1423098 -0.8001762 -0.5826371 -0.4229323 -0.8997308 -0.1077626 0.4229323 -0.8997308 -0.1077626 -0.1921459 -0.9625302 0.1913524 0.1921459 -0.9625302 0.1913524 -0.1652916 -0.7751252 0.609803 0.1652916 -0.7751252 0.609803 0.143075 -0.8169436 0.5586884 -0.143075 -0.8169436 0.5586884 0.4322762 -0.6876621 0.5833165 -0.4322762 -0.6876621 0.5833165 0.6880639 -0.6614204 0.2984815 -0.6880639 -0.6614204 0.2984815 0.7893733 -0.5793112 -0.2031953 -0.7893733 -0.5793112 -0.2031953 0.8016515 -0.5976908 0.01098692 -0.8016515 -0.5976908 0.01098692 -0.459308 -0.212594 0.8624615 0 -0.511625 0.8592089 0.459308 -0.212594 0.8624615 -0.4766535 0.7164146 0.5094619 0.4766535 0.7164146 0.5094619 -0.2312719 -0.1750561 0.9570103 -0.1192997 0.7538256 0.6461537 0.1192997 0.7538256 0.6461537 -0.2266951 0.4281204 0.874827 0.2266951 0.4281204 0.874827 -0.3455379 0.2191275 0.9124619 0.6957222 0.4218403 0.5813962 0.3455379 0.2191275 0.9124619 -0.6957222 0.4218403 0.5813962 -0.9069906 0.1289423 -0.4009265 -0.9302163 0.2023403 -0.3061962 0.5443387 0.05331683 -0.8371695 0.9302163 0.2023403 -0.3061962 -0.5443387 0.05331683 -0.8371695 0.4719823 0.1767988 -0.8636986 -0.4719823 0.1767988 -0.8636986 0 0.6367198 -0.7710954 0.277086 0.9078313 -0.314747 0 0.9769868 -0.2133 -0.277086 0.9078313 -0.314747 -0.6893451 0.278613 -0.6687138 0.1513736 0.9768787 -0.1509768 0 0.9547685 -0.2973504 -0.1513736 0.9768787 -0.1509768 0.06747716 0.6181291 -0.7831752 0 0.4716064 -0.8818091 -0.06747716 0.6181291 -0.7831752 0.5550822 0.6819807 -0.4762207 -0.5550822 0.6819807 -0.4762207 0.6203987 0.7798321 0.08347058 -0.6203987 0.7798321 0.08347058 0.7798632 0.6258621 -0.01049864 -0.7798632 0.6258621 -0.01049864 0.6893451 0.278613 -0.6687138 0.8956729 0.3624137 0.2577334 -0.8956729 0.3624137 0.2577334 0.978706 -0.06149524 -0.1958388 -0.978706 -0.06149524 -0.1958388 -0.887188 -0.4336143 -0.1577223 0.7857264 -0.2367403 -0.571479 -0.7857264 -0.2367403 -0.571479 -0.3488331 0.00814855 -0.9371495 0.4454968 0.820432 -0.3583629 0 0.7225895 -0.6912774 0 0.9523703 -0.304944 -0.4454968 0.820432 -0.3583629 -0.5222814 0.5477042 -0.6536377 0.5222814 0.5477042 -0.6536377 0 0.3365356 -0.9416708 -0.5070669 0.203315 -0.8375835 0.5726917 -0.011994 -0.8196831 0 0.1832989 -0.9830573 -0.5726917 -0.011994 -0.8196831 0.7210462 -0.06506687 -0.6898251 0.9850252 -0.06305181 -0.1604679 -0.7210462 -0.06506687 -0.6898251 -0.9850252 -0.06305181 -0.1604679 0.4730157 0.8632332 0.1763089 0 0.9309914 0.3650414 -0.4730157 0.8632332 0.1763089 0.4442412 -0.5271015 0.7244403 0 -0.02258414 0.999745 0 -0.5568183 0.8306344 -0.4442412 -0.5271015 0.7244403 -0.4135013 -0.03946095 0.9096481 0.3912615 0.4268474 0.8153011 0 0.5513927 0.8342458 -0.3912615 0.4268474 0.8153011 0.7717392 -0.07849538 0.6310762 0.4444231 -0.4250434 0.7885596 -0.7717392 -0.07849538 0.6310762 -0.4444231 -0.4250434 0.7885596 0.7417721 -0.4278504 0.5164479 0.6681936 -0.3195085 0.6718865 -0.7417721 -0.4278504 0.5164479 -0.6681936 -0.3195085 0.6718865 0.8486464 0.01400828 0.528775 0.6784078 0.06952238 0.7313889 -0.8486464 0.01400828 0.528775 -0.6784078 0.06952238 0.7313889 0.8721742 0.3746522 0.3145601 0.6075481 0.5535899 0.5695821 -0.8721742 0.3746522 0.3145601 -0.6075481 0.5535899 0.5695821 0.619659 0.7825392 -0.06045824 0.6707801 0.7402722 -0.04529035 -0.619659 0.7825392 -0.06045824 0.4135013 -0.03946095 0.9096481 0.3406521 -0.3223102 0.8832172 -0.3406521 -0.3223102 0.8832172 0 -0.8484593 0.5292608 0.9970246 0.0695222 -0.03329616 -0.9970246 0.0695222 -0.03329616 0.9085541 -0.3524958 0.2242239 -0.9085541 -0.3524958 0.2242239 0.5070669 0.203315 -0.8375835 0.5790389 -0.1427073 -0.8027133 -0.5790389 -0.1427073 -0.8027133 -0.5632923 0.1213137 -0.8173034 0.3123062 -0.001159727 -0.9499809 -0.3123062 -0.001159727 -0.9499809 0.887188 -0.4336143 -0.1577223 0.3255146 0.7283641 -0.6029314 -0.3255146 0.7283641 -0.6029314 -0.5292324 0.6817365 -0.5051223 0.5632923 0.1213137 -0.8173034 0.5292324 0.6817365 -0.5051223 -0.2792569 -0.5759101 0.7683379 0.5511864 -0.8306556 -0.07877141 0.01880002 -0.488648 0.8722785 0.2792569 -0.5759101 0.7683379 -0.5511864 -0.8306556 -0.07877141 -0.4492394 -0.8925887 -0.0383318 0.3214618 -0.9424114 -0.09232145 0.3835701 -0.3287875 0.8630023 -0.3214618 -0.9424114 -0.09232145 -0.01880002 -0.488648 0.8722785 -0.3835701 -0.3287875 0.8630023 0.778813 -0.6043972 0.1677932 -0.778813 -0.6043972 0.1677932 0.1544557 -0.9802032 -0.1238758 -0.1544557 -0.9802032 -0.1238758 0.6526296 -0.5888133 -0.4768371 -0.6526296 -0.5888133 -0.4768371 0.04110932 -0.9495736 0.3108378 -0.04110932 -0.9495736 0.3108378 0.5028883 -0.3703148 -0.781006 -0.5028883 -0.3703148 -0.781006 -0.5383957 -0.7892665 0.2952768 0.3299418 -0.8896617 0.3156588 0.02954286 -0.7719306 -0.63502 -0.3299418 -0.8896617 0.3156588 -0.02954286 -0.7719306 -0.63502 0.5383957 -0.7892665 0.2952768 0.1628835 -0.487033 0.8580606 -0.1628835 -0.487033 0.8580606 -0.1868091 -0.2350907 0.9538527 0.1868091 -0.2350907 0.9538527 -0.9847559 -0.1426151 -0.09958338 0.9847559 -0.1426151 -0.09958338 0.7621592 0.01931869 0.6471014 -0.1495745 -0.649479 -0.745523 0.1495745 -0.649479 -0.745523 0.5604543 -0.4991108 -0.660893 -0.5604543 -0.4991108 -0.660893 0.6842077 -0.4721915 -0.5557833 -0.6842077 -0.4721915 -0.5557833 0.8572238 0.1482928 -0.4931295 -0.8572238 0.1482928 -0.4931295 -0.7312088 -0.67249 0.1144164 0.7312088 -0.67249 0.1144164 0.4492394 -0.8925887 -0.0383318 0.5998507 -0.6138894 0.5131462 -0.5998507 -0.6138894 0.5131462 0.9609609 -0.2498614 -0.1188421 0.8420515 -0.5097904 -0.1762475 -0.9609609 -0.2498614 -0.1188421 0.8514774 -0.5227583 0.0413531 0.4813766 -0.6048261 0.634399 -0.8420515 -0.5097904 -0.1762475 -0.8514774 -0.5227583 0.0413531 -0.4813766 -0.6048261 0.634399 0.8483738 -0.3011639 -0.4353875 0.6863703 -0.3745586 -0.6233792 -0.8483738 -0.3011639 -0.4353875 0.7260877 -0.4731426 -0.4989316 0.3727269 -0.9004591 -0.224161 -0.7260877 -0.4731426 -0.4989316 -0.6863703 -0.3745586 -0.6233792 -0.3727269 -0.9004591 -0.224161 0.6593034 -0.5880719 -0.4684982 0.6482331 -0.6347435 -0.4205885 -0.6593034 -0.5880719 -0.4684982 -0.6482331 -0.6347435 -0.4205885 -0.5725453 -0.7047557 -0.4189408 0.758382 -0.5948274 0.2665278 0.5725453 -0.7047557 -0.4189408 -0.758382 -0.5948274 0.2665278 -0.4492391 -0.8086305 0.3798695 0.4492391 -0.8086305 0.3798695 -0.2929261 -0.8812505 0.370934 0.6450042 -0.6984139 0.3101416 0.2929261 -0.8812505 0.370934 -0.6450042 -0.6984139 0.3101416 -0.03311294 -0.3256059 0.9449256 0.03311294 -0.3256059 0.9449256 0.461777 -0.8236667 -0.3291432 -0.461777 -0.8236667 -0.3291432 -0.2624001 -0.8043518 -0.5330708 0.2624001 -0.8043518 -0.5330708 -0.7528783 -0.6572921 -0.03378474 0.7528783 -0.6572921 -0.03378474 -0.5831308 -0.6403543 0.4999049 -0.7621592 0.01931869 0.6471014 0.5831308 -0.6403543 0.4999049 0.06500494 -0.7073639 0.7038543 -0.06500494 -0.7073639 0.7038543 0.1951104 -0.9800079 0.03894269 -0.1951104 -0.9800079 0.03894269 -0.4084435 -0.9038679 0.1272666 0.4084435 -0.9038679 0.1272666 0.3346774 -0.9423217 -0.004577875 -0.3346774 -0.9423217 -0.004577875 -0.444819 -0.8907062 -0.09369426 0.3143753 -0.9436142 -0.1037337 0.3343088 -0.9363942 0.1067872 -0.3143753 -0.9436142 -0.1037337 -0.3343088 -0.9363942 0.1067872 0.2896593 -0.9035258 0.3158144 -0.2896593 -0.9035258 0.3158144 -0.3831329 -0.9211488 -0.0685147 0.3831329 -0.9211488 -0.0685147 -0.09885054 0.5321571 -0.8408552 -0.0252701 0.7331398 -0.6796085 0.09885054 0.5321571 -0.8408552 0.0252701 0.7331398 -0.6796085 0.6366097 0.583414 -0.5043375 -0.6366097 0.583414 -0.5043375 0.9252799 0.368 0.09183216 -0.9252799 0.368 0.09183216 0.2870049 0.74852 0.5977844 -0.2870049 0.74852 0.5977844 -0.4142415 0.724503 0.5509079 0.4142415 0.724503 0.5509079 -0.6500716 0.4853563 0.5846677 0.6500716 0.4853563 0.5846677 -0.6707801 0.7402722 -0.04529035 -0.3678491 -0.8855785 -0.2836158 0.444819 -0.8907062 -0.09369426 + + + + + + + + + + 0.890955 0.590063 0.860081 0.560115 0.904571 0.559404 0.856226 0.850547 0.888398 0.821999 0.90064 0.853232 0.904571 0.559404 0.853018 0.521562 0.920166 0.524546 0.847458 0.888748 0.90064 0.853232 0.914672 0.888748 0.860081 0.560115 0.798481 0.569535 0.853018 0.521562 0.795104 0.838402 0.856226 0.850547 0.847458 0.888748 0.870622 0.589649 0.8289 0.590771 0.860081 0.560115 0.826436 0.818537 0.868067 0.82151 0.856226 0.850547 0.854402 0.604754 0.828171 0.633354 0.8289 0.590771 0.827598 0.775964 0.852534 0.8057 0.826436 0.818537 0.8289 0.590771 0.791018 0.645443 0.798481 0.569535 0.791018 0.762238 0.826436 0.818537 0.795104 0.838402 0.855181 0.668527 0.791018 0.645443 0.828171 0.633354 0.856142 0.742025 0.791018 0.762238 0.844839 0.707525 0.867508 0.642291 0.828171 0.633354 0.854107 0.625459 0.867293 0.768782 0.827598 0.775964 0.856142 0.742025 0.867508 0.642291 0.900375 0.666964 0.855181 0.668527 0.901223 0.745592 0.867293 0.768782 0.856142 0.742025 0.900375 0.666964 0.842358 0.702491 0.855181 0.668527 0.901223 0.745592 0.844839 0.707525 0.92118 0.713713 0.931889 0.636832 0.918898 0.699697 0.900375 0.666964 0.931368 0.777093 0.92118 0.713713 0.968213 0.77022 0.905882 0.627902 0.900375 0.666964 0.890474 0.641909 0.90499 0.78486 0.901223 0.745592 0.931368 0.777093 0.906232 0.605742 0.931889 0.636832 0.905882 0.627902 0.904357 0.807013 0.931368 0.777093 0.93125 0.820926 0.933717 0.593037 0.968392 0.645333 0.931889 0.636832 0.93125 0.820926 0.968213 0.77022 0.965038 0.841671 0.904571 0.559404 0.968392 0.573812 0.933717 0.593037 0.90064 0.853232 0.965038 0.841671 0.914672 0.888748 0.890955 0.590063 0.933717 0.593037 0.906232 0.605742 0.888398 0.821999 0.93125 0.820926 0.90064 0.853232 0.906232 0.605742 0.889591 0.593275 0.890955 0.590063 0.904357 0.807013 0.887178 0.818729 0.900583 0.804677 0.905882 0.627902 0.902359 0.607909 0.906232 0.605742 0.90499 0.78486 0.900583 0.804677 0.898822 0.786233 0.890474 0.641909 0.899781 0.626257 0.905882 0.627902 0.890219 0.770183 0.898822 0.786233 0.887351 0.775442 0.867508 0.642291 0.887842 0.636527 0.890474 0.641909 0.867293 0.768782 0.887351 0.775442 0.870376 0.775972 0.867508 0.642291 0.859881 0.623942 0.870908 0.635245 0.858859 0.786774 0.867293 0.768782 0.870376 0.775972 0.854107 0.625459 0.859664 0.608186 0.859881 0.623942 0.857942 0.802505 0.853157 0.785002 0.858859 0.786774 0.854402 0.604754 0.871664 0.593961 0.859664 0.608186 0.869299 0.817249 0.852534 0.8057 0.857942 0.802505 0.870622 0.589649 0.889591 0.593275 0.871664 0.593961 0.887178 0.818729 0.868067 0.82151 0.869299 0.817249 0.8794 0.616512 0.871664 0.593961 0.889591 0.593275 0.887178 0.818729 0.869299 0.817249 0.878029 0.795063 0.859664 0.608186 0.871664 0.593961 0.8794 0.616512 0.878029 0.795063 0.869299 0.817249 0.857942 0.802505 0.8794 0.616512 0.859881 0.623942 0.859664 0.608186 0.857942 0.802505 0.858859 0.786774 0.878029 0.795063 0.8794 0.616512 0.870908 0.635245 0.859881 0.623942 0.858859 0.786774 0.870376 0.775972 0.878029 0.795063 0.8794 0.616512 0.887842 0.636527 0.870908 0.635245 0.870376 0.775972 0.887351 0.775442 0.878029 0.795063 0.8794 0.616512 0.899781 0.626257 0.887842 0.636527 0.887351 0.775442 0.898822 0.786233 0.878029 0.795063 0.8794 0.616512 0.902359 0.607909 0.899781 0.626257 0.898822 0.786233 0.900583 0.804677 0.878029 0.795063 0.8794 0.616512 0.889591 0.593275 0.902359 0.607909 0.900583 0.804677 0.887178 0.818729 0.878029 0.795063 0.536419 0.06207197 0.518916 0.05029398 0.54026 0.05380499 0.501452 0.06204295 0.518916 0.05029398 0.518925 0.05968099 0.542788 0.064089 0.54026 0.05380499 0.55193 0.05833798 0.495083 0.06404697 0.497626 0.05376994 0.501452 0.06204295 0.555073 0.06189996 0.542788 0.064089 0.55193 0.05833798 0.495083 0.06404697 0.482805 0.06182897 0.485955 0.05827295 0.563812 0.076586 0.54629 0.07266896 0.555073 0.06189996 0.491565 0.07262498 0.474014 0.07651096 0.482805 0.06182897 0.583135 0.1084949 0.548333 0.08489298 0.563812 0.076586 0.489507 0.084858 0.454527 0.108481 0.474014 0.07651096 0.605512 0.165134 0.621513 0.227818 0.553118 0.209599 0.416514 0.22949 0.432024 0.165644 0.485339 0.210053 0.676379 0.233241 0.621513 0.227818 0.647395 0.200502 0.360308 0.235899 0.416514 0.22949 0.372747 0.256357 0.676379 0.233241 0.683908 0.279995 0.664761 0.253225 0.353696 0.284606 0.360308 0.235899 0.372747 0.256357 0.707254 0.310054 0.683908 0.279995 0.715342 0.265392 0.330721 0.316853 0.353696 0.284606 0.351187 0.31744 0.697446 0.332673 0.687515 0.311539 0.707254 0.310054 0.341964 0.339667 0.351187 0.31744 0.362723 0.329722 0.662817 0.372521 0.676824 0.323937 0.697446 0.332673 0.379297 0.378686 0.362723 0.329722 0.402772 0.362131 0.662817 0.372521 0.618316 0.375151 0.63905 0.35733 0.424583 0.379267 0.379297 0.378686 0.402772 0.362131 0.604826 0.397804 0.618316 0.375151 0.626842 0.395792 0.439252 0.40154 0.424583 0.379267 0.442396 0.381222 0.553095 0.390512 0.600808 0.377857 0.604826 0.397804 0.490934 0.391862 0.442396 0.381222 0.482938 0.358497 0.521923 0.386009 0.559674 0.357011 0.553095 0.390512 0.521923 0.386009 0.482938 0.358497 0.521086 0.343868 0.599845 0.344815 0.559674 0.357011 0.577279 0.340156 0.441977 0.347815 0.482938 0.358497 0.442396 0.381222 0.599845 0.344815 0.618316 0.375151 0.600808 0.377857 0.424583 0.379267 0.441977 0.347815 0.442396 0.381222 0.63905 0.35733 0.615546 0.342005 0.634472 0.332311 0.402772 0.362131 0.425972 0.345582 0.424583 0.379267 0.676824 0.323937 0.634472 0.332311 0.662406 0.312804 0.362723 0.329722 0.406362 0.33648 0.402772 0.362131 0.687515 0.311539 0.662406 0.312804 0.66844 0.297958 0.351187 0.31744 0.377061 0.317685 0.362723 0.329722 0.683908 0.279995 0.66844 0.297958 0.664101 0.277872 0.353696 0.284606 0.370304 0.302644 0.351187 0.31744 0.664761 0.253225 0.664101 0.277872 0.639236 0.253047 0.372747 0.256357 0.3741 0.281778 0.353696 0.284606 0.621513 0.227818 0.639236 0.253047 0.613992 0.242662 0.416514 0.22949 0.398938 0.255633 0.372747 0.256357 0.572941 0.258564 0.621513 0.227818 0.613992 0.242662 0.416514 0.22949 0.466409 0.259709 0.424464 0.244473 0.572941 0.258564 0.51976 0.2488639 0.553118 0.209599 0.51976 0.2488639 0.466409 0.259709 0.485339 0.210053 0.559674 0.357011 0.558527 0.316594 0.577279 0.340156 0.482938 0.358497 0.482619 0.317843 0.521086 0.343868 0.558527 0.316594 0.520277 0.294764 0.556923 0.291214 0.520277 0.294764 0.482619 0.317843 0.483433 0.292249 0.563905 0.272007 0.520277 0.294764 0.51976 0.2488639 0.475886 0.273078 0.520277 0.294764 0.483433 0.292249 0.525483 0.06896698 0.518925 0.05968099 0.536419 0.06207197 0.518925 0.05968099 0.512375 0.06895595 0.501452 0.06204295 0.531231 0.07382899 0.536419 0.06207197 0.542788 0.064089 0.501452 0.06204295 0.506626 0.07381099 0.495083 0.06404697 0.531231 0.07382899 0.54629 0.07266896 0.531019 0.08743095 0.506626 0.07381099 0.491565 0.07262498 0.495083 0.06404697 0.555621 0.121749 0.532669 0.09091997 0.548333 0.08489298 0.505177 0.09090799 0.482177 0.121781 0.489507 0.084858 0.531019 0.08743095 0.548333 0.08489298 0.532669 0.09091997 0.489507 0.084858 0.506827 0.08741599 0.505177 0.09090799 0.518981 0.151749 0.532042 0.127713 0.538112 0.1583819 0.518981 0.151749 0.505828 0.127728 0.518941 0.128358 0.532042 0.127713 0.518925 0.093952 0.532669 0.09091997 0.505828 0.127728 0.518925 0.093952 0.518941 0.128358 0.531019 0.08743095 0.518925 0.093952 0.518927 0.08517998 0.506827 0.08741599 0.518925 0.093952 0.505177 0.09090799 0.548362 0.17356 0.535214 0.166808 0.538112 0.1583819 0.502799 0.166857 0.489683 0.173693 0.499851 0.158434 0.537248 0.187577 0.548362 0.17356 0.544281 0.193366 0.50089 0.187571 0.489683 0.173693 0.5001 0.176033 0.528757 0.191785 0.544281 0.193366 0.519841 0.200843 0.509219 0.191626 0.493996 0.193428 0.50089 0.187571 0.519132 0.185382 0.519841 0.200843 0.517577 0.190607 0.519132 0.185382 0.519841 0.200843 0.509219 0.191626 0.538112 0.1583819 0.518998 0.1590279 0.518981 0.151749 0.499851 0.158434 0.518998 0.1590279 0.502799 0.166857 0.535214 0.166808 0.519016 0.165599 0.518998 0.1590279 0.502799 0.166857 0.519016 0.165599 0.50691 0.171667 0.519132 0.185382 0.528222 0.186316 0.528757 0.191785 0.509787 0.18626 0.519132 0.185382 0.509219 0.191626 0.528757 0.191785 0.533528 0.184215 0.537248 0.187577 0.504547 0.184206 0.509219 0.191626 0.50089 0.187571 0.533528 0.184215 0.537959 0.175966 0.537248 0.187577 0.504547 0.184206 0.5001 0.176033 0.504604 0.176791 0.537959 0.175966 0.531131 0.171631 0.535214 0.166808 0.50691 0.171667 0.5001 0.176033 0.502799 0.166857 0.533449 0.176739 0.528222 0.186316 0.519099 0.179457 0.504604 0.176791 0.509787 0.18626 0.504547 0.184206 0.519099 0.179457 0.531131 0.171631 0.533449 0.176739 0.50691 0.171667 0.519099 0.179457 0.504604 0.176791 0.519841 0.200843 0.553118 0.209599 0.51976 0.2488639 0.485339 0.210053 0.519841 0.200843 0.51976 0.2488639 0.544281 0.193366 0.561572 0.167779 0.553118 0.209599 0.476363 0.167996 0.493996 0.193428 0.485339 0.210053 0.538112 0.1583819 0.561572 0.167779 0.548362 0.17356 0.499851 0.158434 0.476363 0.167996 0.478371 0.149447 0.532042 0.127713 0.559475 0.1493189 0.538112 0.1583819 0.505828 0.127728 0.478371 0.149447 0.482177 0.121781 0.596138 0.133426 0.555621 0.121749 0.583135 0.1084949 0.441395 0.133592 0.482177 0.121781 0.478371 0.149447 0.601169 0.147885 0.559475 0.1493189 0.596138 0.133426 0.436337 0.148194 0.478371 0.149447 0.476363 0.167996 0.605512 0.165134 0.561572 0.167779 0.601169 0.147885 0.476363 0.167996 0.432024 0.165644 0.436337 0.148194 0.518927 0.08517998 0.528933 0.084957 0.531019 0.08743095 0.518927 0.08517998 0.508915 0.08494496 0.518925 0.08386498 0.531231 0.07382899 0.528933 0.084957 0.529036 0.07542896 0.508915 0.08494496 0.506626 0.07381099 0.50882 0.07541495 0.525483 0.06896698 0.529036 0.07542896 0.523751 0.070508 0.50882 0.07541495 0.512375 0.06895595 0.514106 0.07050096 0.518928 0.06789898 0.523751 0.070508 0.518929 0.06946796 0.514106 0.07050096 0.518928 0.06789898 0.518929 0.06946796 0.523751 0.070508 0.518928 0.07425898 0.518929 0.06946796 0.514106 0.07050096 0.518928 0.07425898 0.516297 0.07496595 0.523751 0.070508 0.524236 0.07669097 0.52156 0.07497 0.513619 0.07668399 0.514106 0.07050096 0.516297 0.07496595 0.529036 0.07542896 0.524601 0.07988595 0.524236 0.07669097 0.513252 0.07987898 0.50882 0.07541495 0.513619 0.07668399 0.518925 0.08386498 0.524601 0.07988595 0.528933 0.084957 0.518925 0.08386498 0.513252 0.07987898 0.518926 0.07933098 0.518926 0.07933098 0.52156 0.07497 0.524601 0.07988595 0.516297 0.07496595 0.518926 0.07933098 0.513252 0.07987898 0.524601 0.07988595 0.52156 0.07497 0.524236 0.07669097 0.513619 0.07668399 0.516297 0.07496595 0.513252 0.07987898 0.556923 0.291214 0.571787 0.277295 0.568351 0.292904 0.46807 0.278617 0.483433 0.292249 0.471978 0.294282 0.558527 0.316594 0.568351 0.292904 0.573085 0.311386 0.471978 0.294282 0.482619 0.317843 0.46779 0.313081 0.558527 0.316594 0.584855 0.327708 0.577279 0.340156 0.482619 0.317843 0.456477 0.329961 0.46779 0.313081 0.572941 0.258564 0.571787 0.277295 0.563905 0.272007 0.466409 0.259709 0.46807 0.278617 0.458737 0.268049 0.572941 0.258564 0.61172 0.255725 0.580734 0.26662 0.427062 0.257728 0.466409 0.259709 0.458737 0.268049 0.613992 0.242662 0.632494 0.262853 0.61172 0.255725 0.406068 0.265508 0.424464 0.244473 0.427062 0.257728 0.639236 0.253047 0.653658 0.279971 0.632494 0.262853 0.384904 0.283634 0.398938 0.255633 0.406068 0.265508 0.664101 0.277872 0.656064 0.297636 0.653658 0.279971 0.383015 0.301864 0.3741 0.281778 0.384904 0.283634 0.662406 0.312804 0.656064 0.297636 0.66844 0.297958 0.377061 0.317685 0.383015 0.301864 0.386858 0.314615 0.634472 0.332311 0.652752 0.310186 0.662406 0.312804 0.406362 0.33648 0.386858 0.314615 0.411556 0.327673 0.634472 0.332311 0.614408 0.331972 0.62904 0.323864 0.426727 0.335361 0.406362 0.33648 0.411556 0.327673 0.615546 0.342005 0.601033 0.333624 0.614408 0.331972 0.440344 0.336537 0.425972 0.345582 0.426727 0.335361 0.599845 0.344815 0.584855 0.327708 0.601033 0.333624 0.456477 0.329961 0.441977 0.347815 0.440344 0.336537 0.601033 0.333624 0.590644 0.321516 0.601799 0.328453 0.450408 0.323919 0.440344 0.336537 0.439372 0.331331 0.614408 0.331972 0.601799 0.328453 0.613335 0.327083 0.439372 0.331331 0.426727 0.335361 0.427623 0.330358 0.614408 0.331972 0.626851 0.320513 0.62904 0.323864 0.426727 0.335361 0.413648 0.324175 0.427623 0.330358 0.62904 0.323864 0.646248 0.306421 0.652752 0.310186 0.411556 0.327673 0.393381 0.31051 0.413648 0.324175 0.656064 0.297636 0.646248 0.306421 0.649541 0.296225 0.393381 0.31051 0.383015 0.301864 0.389662 0.300183 0.656064 0.297636 0.647785 0.283486 0.653658 0.279971 0.383015 0.301864 0.39104 0.287071 0.389662 0.300183 0.632494 0.262853 0.647785 0.283486 0.629829 0.267263 0.39104 0.287071 0.406068 0.265508 0.408893 0.269959 0.632494 0.262853 0.612641 0.26156 0.61172 0.255725 0.406068 0.265508 0.426254 0.263693 0.408893 0.269959 0.61172 0.255725 0.585166 0.270991 0.580734 0.26662 0.427062 0.257728 0.454369 0.272583 0.426254 0.263693 0.571787 0.277295 0.585166 0.270991 0.578124 0.2819 0.454369 0.272583 0.46807 0.278617 0.461798 0.283441 0.584855 0.327708 0.579548 0.30934 0.590644 0.321516 0.461204 0.311233 0.456477 0.329961 0.450408 0.323919 0.573085 0.311386 0.577524 0.293776 0.579548 0.30934 0.462754 0.295432 0.46779 0.313081 0.461204 0.311233 0.568351 0.292904 0.578124 0.2819 0.577524 0.293776 0.461798 0.283441 0.471978 0.294282 0.462754 0.295432 0.521923 0.386009 0.553209 0.433063 0.523031 0.433628 0.492809 0.434538 0.521923 0.386009 0.523031 0.433628 0.553095 0.390512 0.609819 0.431516 0.553209 0.433063 0.43586 0.43574 0.490934 0.391862 0.492809 0.434538 0.626842 0.395792 0.609819 0.431516 0.604826 0.397804 0.416915 0.400552 0.43586 0.43574 0.396518 0.425416 0.662817 0.372521 0.648174 0.419316 0.626842 0.395792 0.379297 0.378686 0.396518 0.425416 0.350292 0.396229 0.697446 0.332673 0.692106 0.388274 0.662817 0.372521 0.341964 0.339667 0.350292 0.396229 0.312756 0.350588 0.697446 0.332673 0.735879 0.312112 0.726332 0.341754 0.301067 0.320593 0.341964 0.339667 0.312756 0.350588 0.715342 0.265392 0.735879 0.312112 0.707254 0.310054 0.320452 0.270303 0.301067 0.320593 0.304876 0.261087 0.715342 0.265392 0.698172 0.216906 0.7299 0.256393 0.337414 0.219179 0.320452 0.270303 0.304876 0.261087 0.676379 0.233241 0.663103 0.190671 0.698172 0.216906 0.373474 0.191872 0.360308 0.235899 0.337414 0.219179 0.649444 0.02237796 0.62144 0.04808896 0.626908 0.01560795 0.388827 0.021586 0.416419 0.04763096 0.376796 0.07529598 0.626908 0.01560795 0.577206 0.03280097 0.56746 1.44e-4 0.411318 0.01513099 0.460782 0.03265595 0.416419 0.04763096 0.56746 1.44e-4 0.547413 0.04172396 0.518922 0.02488595 0.470636 1.44e-4 0.490511 0.04166895 0.460782 0.03265595 0.54026 0.05380499 0.518922 0.02488595 0.547413 0.04172396 0.518922 0.02488595 0.497626 0.05376994 0.490511 0.04166895 0.55193 0.05833798 0.547413 0.04172396 0.558059 0.05387097 0.490511 0.04166895 0.485955 0.05827295 0.479842 0.05378496 0.555073 0.06189996 0.558059 0.05387097 0.576951 0.057998 0.479842 0.05378496 0.482805 0.06182897 0.46092 0.05784499 0.563812 0.076586 0.576951 0.057998 0.611687 0.07826799 0.46092 0.05784499 0.474014 0.07651096 0.425932 0.07798498 0.576951 0.057998 0.62144 0.04808896 0.611687 0.07826799 0.416419 0.04763096 0.46092 0.05784499 0.425932 0.07798498 0.558059 0.05387097 0.577206 0.03280097 0.576951 0.057998 0.479842 0.05378496 0.460782 0.03265595 0.490511 0.04166895 0.611687 0.07826799 0.660451 0.07608395 0.626663 0.111357 0.425932 0.07798498 0.376796 0.07529598 0.416419 0.04763096 0.583135 0.1084949 0.611687 0.07826799 0.626663 0.111357 0.425932 0.07798498 0.454527 0.108481 0.410618 0.111244 0.629482 0.130456 0.601169 0.147885 0.596138 0.133426 0.407648 0.130594 0.436337 0.148194 0.413741 0.147158 0.626663 0.111357 0.596138 0.133426 0.583135 0.1084949 0.410618 0.111244 0.441395 0.133592 0.407648 0.130594 0.601169 0.147885 0.619303 0.159841 0.605512 0.165134 0.436337 0.148194 0.418035 0.160361 0.413741 0.147158 0.619303 0.159841 0.647395 0.200502 0.605512 0.165134 0.418035 0.160361 0.389677 0.20189 0.373474 0.191872 0.886245 0.1217769 0.89178 0.03691595 0.9459 0.07956898 0.141314 0.112482 0.142277 0.02146697 0.183115 0.09212696 0.849114 0.09973198 0.805584 0.01078599 0.89178 0.03691595 0.183115 0.09212696 0.232648 0.003483951 0.246353 0.07650995 0.805584 0.01078599 0.687018 0.07720398 0.672384 0.022201 0.349875 0.07595497 0.232648 0.003483951 0.365979 0.02099096 0.672384 0.022201 0.660451 0.07608395 0.649444 0.02237796 0.376796 0.07529598 0.365979 0.02099096 0.388827 0.021586 0.660451 0.07608395 0.629482 0.130456 0.626663 0.111357 0.376796 0.07529598 0.407648 0.130594 0.349875 0.07595497 0.7299 0.256393 0.760215 0.193244 0.789046 0.233323 0.271553 0.193871 0.304876 0.261087 0.241255 0.236977 0.909112 0.183261 0.9459 0.07956898 0.994525 0.1677049 0.107928 0.179083 0.07896095 0.06071895 0.141314 0.112482 0.862868 0.338556 0.962901 0.344752 0.911671 0.402429 0.160557 0.356821 0.04396796 0.367038 0.123776 0.315519 0.962901 0.344752 0.91536 0.259804 0.999856 0.25464 0.09896498 0.266968 0.04396796 0.367038 1.44e-4 0.259113 0.999856 0.25464 0.909112 0.183261 0.994525 0.1677049 0.107928 0.179083 1.44e-4 0.259113 0.01182895 0.155367 0.749542 0.334683 0.766337 0.300809 0.789162 0.313727 0.267408 0.310142 0.288183 0.346496 0.242992 0.325552 0.789162 0.313727 0.815314 0.276388 0.846174 0.293397 0.213065 0.285164 0.242992 0.325552 0.178537 0.304983 0.846174 0.293397 0.845007 0.256352 0.873517 0.265922 0.179662 0.263312 0.178537 0.304983 0.147089 0.274284 0.873517 0.265922 0.859075 0.228168 0.886999 0.2337689 0.1628029 0.23172 0.147089 0.274284 0.131514 0.237587 0.87503 0.184705 0.859075 0.228168 0.842355 0.19516 0.145224 0.182749 0.1628029 0.23172 0.131514 0.237587 0.909112 0.183261 0.886999 0.2337689 0.87503 0.184705 0.131514 0.237587 0.107928 0.179083 0.145224 0.182749 0.91536 0.259804 0.873517 0.265922 0.886999 0.2337689 0.147089 0.274284 0.09896498 0.266968 0.131514 0.237587 0.894128 0.301884 0.846174 0.293397 0.873517 0.265922 0.178537 0.304983 0.123776 0.315519 0.147089 0.274284 0.862868 0.338556 0.789162 0.313727 0.846174 0.293397 0.242992 0.325552 0.160557 0.356821 0.178537 0.304983 0.749542 0.334683 0.794286 0.364062 0.770185 0.379538 0.288183 0.346496 0.239776 0.382592 0.242992 0.325552 0.794286 0.364062 0.911671 0.402429 0.845499 0.449967 0.239776 0.382592 0.1064 0.432652 0.160557 0.356821 0.770572 0.444261 0.770185 0.379538 0.815858 0.445381 0.271364 0.473316 0.268122 0.398737 0.287033 0.442912 0.815858 0.445381 0.794286 0.364062 0.845499 0.449967 0.239776 0.382592 0.21926 0.477186 0.185281 0.484099 0.819845 0.468071 0.815858 0.445381 0.845499 0.449967 0.185281 0.484099 0.21926 0.477186 0.215894 0.503605 0.7299 0.256393 0.766337 0.300809 0.735879 0.312112 0.304876 0.261087 0.267408 0.310142 0.241255 0.236977 0.789046 0.233323 0.815314 0.276388 0.766337 0.300809 0.213065 0.285164 0.241255 0.236977 0.267408 0.310142 0.809631 0.233887 0.845007 0.256352 0.815314 0.276388 0.179662 0.263312 0.219168 0.237388 0.213065 0.285164 0.859075 0.228168 0.829287 0.2195619 0.842355 0.19516 0.1628029 0.23172 0.1990669 0.222464 0.179662 0.263312 0.788458 0.08082598 0.715482 0.1397269 0.687018 0.07720398 0.246353 0.07650995 0.319538 0.139409 0.246666 0.11485 0.715482 0.1397269 0.785486 0.15233 0.760215 0.193244 0.319538 0.139409 0.2459689 0.151002 0.246666 0.11485 0.698172 0.216906 0.715482 0.1397269 0.760215 0.193244 0.319538 0.139409 0.337414 0.219179 0.271553 0.193871 0.663103 0.190671 0.629482 0.130456 0.715482 0.1397269 0.407648 0.130594 0.373474 0.191872 0.319538 0.139409 0.629482 0.130456 0.687018 0.07720398 0.715482 0.1397269 0.319538 0.139409 0.349875 0.07595497 0.407648 0.130594 0.663103 0.190671 0.619303 0.159841 0.623495 0.146796 0.413741 0.147158 0.418035 0.160361 0.373474 0.191872 0.837382 0.156361 0.87503 0.184705 0.842355 0.19516 0.196622 0.155241 0.145224 0.182749 0.171653 0.1322939 0.87503 0.184705 0.886245 0.1217769 0.909112 0.183261 0.145224 0.182749 0.141314 0.112482 0.171653 0.1322939 0.78648 0.117591 0.837382 0.156361 0.785486 0.15233 0.246666 0.11485 0.196622 0.155241 0.171653 0.1322939 0.849114 0.09973198 0.78648 0.117591 0.788458 0.08082598 0.183115 0.09212696 0.246666 0.11485 0.171653 0.1322939 0.886245 0.1217769 0.858171 0.137775 0.849114 0.09973198 0.183115 0.09212696 0.171653 0.1322939 0.141314 0.112482 0.432388 0.894943 0.491058 0.881714 0.506166 0.904851 0.321637 0.893225 0.263032 0.878321 0.315867 0.868209 0.506166 0.904851 0.572792 0.860484 0.604825 0.879946 0.181486 0.854693 0.247207 0.901159 0.148729 0.873349 0.572792 0.860484 0.619962 0.791615 0.604825 0.879946 0.181486 0.854693 0.136063 0.784093 0.169745 0.787474 0.586396 0.793977 0.563786 0.739211 0.619962 0.791615 0.169745 0.787474 0.194086 0.733241 0.208656 0.740879 0.549027 0.746412 0.50827 0.697693 0.563786 0.739211 0.208656 0.740879 0.250811 0.693249 0.258399 0.707497 0.50827 0.697693 0.438641 0.680683 0.434803 0.658882 0.320962 0.677959 0.250811 0.693249 0.325318 0.656224 0.500314 0.711729 0.452955 0.700023 0.438641 0.680683 0.306136 0.696976 0.258399 0.707497 0.320962 0.677959 0.549027 0.746412 0.505666 0.730944 0.500314 0.711729 0.252524 0.726592 0.208656 0.740879 0.258399 0.707497 0.568148 0.787367 0.549027 0.746412 0.586396 0.793977 0.188269 0.781375 0.208656 0.740879 0.2145749 0.750414 0.555495 0.826352 0.586396 0.793977 0.572792 0.860484 0.19985 0.820889 0.169745 0.787474 0.188269 0.781375 0.501231 0.844356 0.572792 0.860484 0.491058 0.881714 0.253846 0.840502 0.181486 0.854693 0.19985 0.820889 0.491058 0.881714 0.457832 0.84004 0.501231 0.844356 0.297562 0.837358 0.263032 0.878321 0.253846 0.840502 0.785486 0.15233 0.783193 0.187449 0.760215 0.193244 0.2459689 0.151002 0.246955 0.187075 0.2336249 0.17562 0.434803 0.658882 0.394766 0.686125 0.391039 0.611891 0.325318 0.656224 0.364838 0.684445 0.320962 0.677959 0.789046 0.233323 0.783193 0.187449 0.809631 0.233887 0.246955 0.187075 0.241255 0.236977 0.219168 0.237388 0.391747 0.862097 0.438797 0.870229 0.432388 0.894943 0.315867 0.868209 0.363377 0.861308 0.321637 0.893225 0.438641 0.680683 0.435018 0.71828 0.394766 0.686125 0.323658 0.715731 0.320962 0.677959 0.364838 0.684445 0.384658 0.710299 0.435018 0.71828 0.433669 0.729661 0.3744 0.708969 0.323658 0.715731 0.364838 0.684445 0.410995 0.747662 0.433669 0.729661 0.427812 0.742828 0.324726 0.727177 0.347028 0.745816 0.33027 0.740536 0.384657 0.795423 0.410995 0.747662 0.418086 0.784946 0.37227 0.794472 0.347028 0.745816 0.3744 0.708969 0.384657 0.795423 0.431333 0.817535 0.401605 0.84146 0.37227 0.794472 0.32479 0.81546 0.338952 0.783073 0.401605 0.84146 0.457832 0.84004 0.438797 0.870229 0.354026 0.840297 0.297562 0.837358 0.32479 0.81546 0.816266 0.203086 0.829287 0.2195619 0.809631 0.233887 0.209828 0.206161 0.1990669 0.222464 0.1997669 0.214827 0.783193 0.187449 0.816266 0.203086 0.809631 0.233887 0.246955 0.187075 0.209828 0.206161 0.226485 0.183086 0.783193 0.187449 0.796021 0.176969 0.802192 0.1846089 0.226485 0.183086 0.2336249 0.17562 0.246955 0.187075 0.457832 0.84004 0.448505 0.804621 0.473386 0.8247 0.307886 0.802031 0.297562 0.837358 0.282357 0.821525 0.418086 0.784946 0.448505 0.804621 0.431333 0.817535 0.338952 0.783073 0.307886 0.802031 0.321237 0.777208 0.418086 0.784946 0.423718 0.754191 0.435868 0.779569 0.334089 0.752045 0.338952 0.783073 0.321237 0.777208 0.427812 0.742828 0.423718 0.754191 0.410995 0.747662 0.33027 0.740536 0.334089 0.752045 0.319919 0.74725 0.433669 0.729661 0.43795 0.749777 0.427812 0.742828 0.324726 0.727177 0.319919 0.74725 0.312907 0.729222 0.433669 0.729661 0.440995 0.724383 0.445392 0.731997 0.31751 0.721697 0.324726 0.727177 0.312907 0.729222 0.435018 0.71828 0.455277 0.713731 0.440995 0.724383 0.30346 0.710657 0.323658 0.715731 0.31751 0.721697 0.501231 0.844356 0.473386 0.8247 0.512485 0.828811 0.282357 0.821525 0.253846 0.840502 0.2429749 0.824574 0.555495 0.826352 0.512485 0.828811 0.550942 0.811814 0.2429749 0.824574 0.19985 0.820889 0.204839 0.806417 0.568148 0.787367 0.550942 0.811814 0.552139 0.787682 0.204839 0.806417 0.188269 0.781375 0.204331 0.782156 0.568148 0.787367 0.539407 0.764539 0.54285 0.755753 0.188269 0.781375 0.217774 0.759319 0.204331 0.782156 0.54285 0.755753 0.508439 0.743135 0.505666 0.730944 0.2145749 0.750414 0.249419 0.738732 0.217774 0.759319 0.505666 0.730944 0.455277 0.713731 0.452955 0.700023 0.252524 0.726592 0.30346 0.710657 0.249419 0.738732 0.445392 0.731997 0.454776 0.761665 0.43795 0.749777 0.312907 0.729222 0.302729 0.758742 0.28696 0.74502 0.470841 0.748408 0.475403 0.783904 0.454776 0.761665 0.28696 0.74502 0.281439 0.780511 0.268291 0.766661 0.475403 0.783904 0.503673 0.787562 0.494476 0.80247 0.252972 0.78341 0.281439 0.780511 0.26179 0.798626 0.503673 0.787562 0.516802 0.807339 0.494476 0.80247 0.252972 0.78341 0.239243 0.802891 0.23792 0.787045 0.512485 0.828811 0.494476 0.80247 0.516802 0.807339 0.26179 0.798626 0.2429749 0.824574 0.239243 0.802891 0.475403 0.783904 0.473386 0.8247 0.448505 0.804621 0.281439 0.780511 0.282357 0.821525 0.26179 0.798626 0.435868 0.779569 0.475403 0.783904 0.448505 0.804621 0.321237 0.777208 0.281439 0.780511 0.302729 0.758742 0.43795 0.749777 0.435868 0.779569 0.423718 0.754191 0.321237 0.777208 0.319919 0.74725 0.334089 0.752045 0.455277 0.713731 0.445392 0.731997 0.440995 0.724383 0.30346 0.710657 0.312907 0.729222 0.28696 0.74502 0.508439 0.743135 0.470841 0.748408 0.455277 0.713731 0.28696 0.74502 0.249419 0.738732 0.30346 0.710657 0.503673 0.787562 0.508439 0.743135 0.539407 0.764539 0.252972 0.78341 0.249419 0.738732 0.268291 0.766661 0.518562 0.791602 0.539407 0.764539 0.552139 0.787682 0.23792 0.787045 0.217774 0.759319 0.252972 0.78341 0.550942 0.811814 0.518562 0.791602 0.552139 0.787682 0.23792 0.787045 0.204839 0.806417 0.204331 0.782156 0.512485 0.828811 0.516802 0.807339 0.550942 0.811814 0.204839 0.806417 0.239243 0.802891 0.2429749 0.824574 0.50827 0.697693 0.484068 0.628776 0.543385 0.683538 0.276936 0.625067 0.250811 0.693249 0.216123 0.67812 0.563786 0.739211 0.543385 0.683538 0.581052 0.726933 0.216123 0.67812 0.194086 0.733241 0.1771759 0.720426 0.563786 0.739211 0.616701 0.759965 0.619962 0.791615 0.194086 0.733241 0.140379 0.752377 0.1771759 0.720426 0.619962 0.791615 0.660647 0.741167 0.707492 0.759884 0.136063 0.784093 0.09703797 0.732052 0.140379 0.752377 0.707492 0.759884 0.677256 0.670436 0.745511 0.6521 0.04952597 0.748824 0.08356398 0.662038 0.09703797 0.732052 0.745511 0.6521 0.671403 0.592656 0.740843 0.572428 0.019409 0.639749 0.09281998 0.589862 0.08356398 0.662038 0.543385 0.683538 0.671403 0.592656 0.677256 0.670436 0.216123 0.67812 0.09281998 0.589862 0.276936 0.625067 0.677256 0.670436 0.581052 0.726933 0.543385 0.683538 0.1771759 0.720426 0.08356398 0.662038 0.216123 0.67812 0.660647 0.741167 0.616701 0.759965 0.581052 0.726933 0.1771759 0.720426 0.140379 0.752377 0.09703797 0.732052 0.829287 0.2195619 0.834705 0.206959 0.842355 0.19516 0.05121594 0.522659 0.09281998 0.589862 0.03366398 0.564403 0.62042 0.565675 0.484068 0.628776 0.498072 0.552315 0.276936 0.625067 0.145041 0.562595 0.264218 0.55014 0.391039 0.611891 0.484068 0.628776 0.434803 0.658882 0.276936 0.625067 0.369913 0.610196 0.325318 0.656224 0.890955 0.590063 0.870622 0.589649 0.860081 0.560115 0.856226 0.850547 0.868067 0.82151 0.888398 0.821999 0.904571 0.559404 0.860081 0.560115 0.853018 0.521562 0.847458 0.888748 0.856226 0.850547 0.90064 0.853232 0.860081 0.560115 0.8289 0.590771 0.798481 0.569535 0.795104 0.838402 0.826436 0.818537 0.856226 0.850547 0.870622 0.589649 0.854402 0.604754 0.8289 0.590771 0.826436 0.818537 0.852534 0.8057 0.868067 0.82151 0.854402 0.604754 0.854107 0.625459 0.828171 0.633354 0.827598 0.775964 0.853157 0.785002 0.852534 0.8057 0.8289 0.590771 0.828171 0.633354 0.791018 0.645443 0.791018 0.762238 0.827598 0.775964 0.826436 0.818537 0.855181 0.668527 0.842358 0.702491 0.791018 0.645443 0.856142 0.742025 0.827598 0.775964 0.791018 0.762238 0.867508 0.642291 0.855181 0.668527 0.828171 0.633354 0.867293 0.768782 0.853157 0.785002 0.827598 0.775964 0.867508 0.642291 0.890474 0.641909 0.900375 0.666964 0.901223 0.745592 0.890219 0.770183 0.867293 0.768782 0.900375 0.666964 0.918898 0.699697 0.842358 0.702491 0.901223 0.745592 0.856142 0.742025 0.844839 0.707525 0.931889 0.636832 0.968392 0.645333 0.918898 0.699697 0.931368 0.777093 0.901223 0.745592 0.92118 0.713713 0.905882 0.627902 0.931889 0.636832 0.900375 0.666964 0.90499 0.78486 0.890219 0.770183 0.901223 0.745592 0.906232 0.605742 0.933717 0.593037 0.931889 0.636832 0.904357 0.807013 0.90499 0.78486 0.931368 0.777093 0.933717 0.593037 0.968392 0.573812 0.968392 0.645333 0.93125 0.820926 0.931368 0.777093 0.968213 0.77022 0.904571 0.559404 0.920166 0.524546 0.968392 0.573812 0.90064 0.853232 0.93125 0.820926 0.965038 0.841671 0.890955 0.590063 0.904571 0.559404 0.933717 0.593037 0.888398 0.821999 0.904357 0.807013 0.93125 0.820926 0.906232 0.605742 0.902359 0.607909 0.889591 0.593275 0.904357 0.807013 0.888398 0.821999 0.887178 0.818729 0.905882 0.627902 0.899781 0.626257 0.902359 0.607909 0.90499 0.78486 0.904357 0.807013 0.900583 0.804677 0.890474 0.641909 0.887842 0.636527 0.899781 0.626257 0.890219 0.770183 0.90499 0.78486 0.898822 0.786233 0.867508 0.642291 0.870908 0.635245 0.887842 0.636527 0.867293 0.768782 0.890219 0.770183 0.887351 0.775442 0.867508 0.642291 0.854107 0.625459 0.859881 0.623942 0.858859 0.786774 0.853157 0.785002 0.867293 0.768782 0.854107 0.625459 0.854402 0.604754 0.859664 0.608186 0.857942 0.802505 0.852534 0.8057 0.853157 0.785002 0.854402 0.604754 0.870622 0.589649 0.871664 0.593961 0.869299 0.817249 0.868067 0.82151 0.852534 0.8057 0.870622 0.589649 0.890955 0.590063 0.889591 0.593275 0.887178 0.818729 0.888398 0.821999 0.868067 0.82151 0.536419 0.06207197 0.518925 0.05968099 0.518916 0.05029398 0.501452 0.06204295 0.497626 0.05376994 0.518916 0.05029398 0.542788 0.064089 0.536419 0.06207197 0.54026 0.05380499 0.495083 0.06404697 0.485955 0.05827295 0.497626 0.05376994 0.555073 0.06189996 0.54629 0.07266896 0.542788 0.064089 0.495083 0.06404697 0.491565 0.07262498 0.482805 0.06182897 0.563812 0.076586 0.548333 0.08489298 0.54629 0.07266896 0.491565 0.07262498 0.489507 0.084858 0.474014 0.07651096 0.583135 0.1084949 0.555621 0.121749 0.548333 0.08489298 0.489507 0.084858 0.482177 0.121781 0.454527 0.108481 0.605512 0.165134 0.647395 0.200502 0.621513 0.227818 0.416514 0.22949 0.389677 0.20189 0.432024 0.165644 0.676379 0.233241 0.664761 0.253225 0.621513 0.227818 0.360308 0.235899 0.389677 0.20189 0.416514 0.22949 0.676379 0.233241 0.715342 0.265392 0.683908 0.279995 0.353696 0.284606 0.320452 0.270303 0.360308 0.235899 0.707254 0.310054 0.687515 0.311539 0.683908 0.279995 0.330721 0.316853 0.320452 0.270303 0.353696 0.284606 0.697446 0.332673 0.676824 0.323937 0.687515 0.311539 0.341964 0.339667 0.330721 0.316853 0.351187 0.31744 0.662817 0.372521 0.63905 0.35733 0.676824 0.323937 0.379297 0.378686 0.341964 0.339667 0.362723 0.329722 0.662817 0.372521 0.626842 0.395792 0.618316 0.375151 0.424583 0.379267 0.416915 0.400552 0.379297 0.378686 0.604826 0.397804 0.600808 0.377857 0.618316 0.375151 0.439252 0.40154 0.416915 0.400552 0.424583 0.379267 0.553095 0.390512 0.559674 0.357011 0.600808 0.377857 0.490934 0.391862 0.439252 0.40154 0.442396 0.381222 0.521923 0.386009 0.521086 0.343868 0.559674 0.357011 0.521923 0.386009 0.490934 0.391862 0.482938 0.358497 0.599845 0.344815 0.600808 0.377857 0.559674 0.357011 0.441977 0.347815 0.464579 0.34223 0.482938 0.358497 0.599845 0.344815 0.615546 0.342005 0.618316 0.375151 0.424583 0.379267 0.425972 0.345582 0.441977 0.347815 0.63905 0.35733 0.618316 0.375151 0.615546 0.342005 0.402772 0.362131 0.406362 0.33648 0.425972 0.345582 0.676824 0.323937 0.63905 0.35733 0.634472 0.332311 0.362723 0.329722 0.377061 0.317685 0.406362 0.33648 0.687515 0.311539 0.676824 0.323937 0.662406 0.312804 0.351187 0.31744 0.370304 0.302644 0.377061 0.317685 0.683908 0.279995 0.687515 0.311539 0.66844 0.297958 0.353696 0.284606 0.3741 0.281778 0.370304 0.302644 0.664761 0.253225 0.683908 0.279995 0.664101 0.277872 0.372747 0.256357 0.398938 0.255633 0.3741 0.281778 0.621513 0.227818 0.664761 0.253225 0.639236 0.253047 0.416514 0.22949 0.424464 0.244473 0.398938 0.255633 0.572941 0.258564 0.553118 0.209599 0.621513 0.227818 0.416514 0.22949 0.485339 0.210053 0.466409 0.259709 0.572941 0.258564 0.563905 0.272007 0.51976 0.2488639 0.51976 0.2488639 0.475886 0.273078 0.466409 0.259709 0.559674 0.357011 0.521086 0.343868 0.558527 0.316594 0.482938 0.358497 0.464579 0.34223 0.482619 0.317843 0.558527 0.316594 0.521086 0.343868 0.520277 0.294764 0.520277 0.294764 0.521086 0.343868 0.482619 0.317843 0.563905 0.272007 0.556923 0.291214 0.520277 0.294764 0.475886 0.273078 0.51976 0.2488639 0.520277 0.294764 0.525483 0.06896698 0.518928 0.06789898 0.518925 0.05968099 0.518925 0.05968099 0.518928 0.06789898 0.512375 0.06895595 0.531231 0.07382899 0.525483 0.06896698 0.536419 0.06207197 0.501452 0.06204295 0.512375 0.06895595 0.506626 0.07381099 0.531231 0.07382899 0.542788 0.064089 0.54629 0.07266896 0.506626 0.07381099 0.506827 0.08741599 0.491565 0.07262498 0.555621 0.121749 0.532042 0.127713 0.532669 0.09091997 0.505177 0.09090799 0.505828 0.127728 0.482177 0.121781 0.531019 0.08743095 0.54629 0.07266896 0.548333 0.08489298 0.489507 0.084858 0.491565 0.07262498 0.506827 0.08741599 0.518981 0.151749 0.518941 0.128358 0.532042 0.127713 0.518981 0.151749 0.499851 0.158434 0.505828 0.127728 0.532042 0.127713 0.518941 0.128358 0.518925 0.093952 0.505828 0.127728 0.505177 0.09090799 0.518925 0.093952 0.531019 0.08743095 0.532669 0.09091997 0.518925 0.093952 0.506827 0.08741599 0.518927 0.08517998 0.518925 0.093952 0.548362 0.17356 0.537959 0.175966 0.535214 0.166808 0.502799 0.166857 0.5001 0.176033 0.489683 0.173693 0.537248 0.187577 0.537959 0.175966 0.548362 0.17356 0.50089 0.187571 0.493996 0.193428 0.489683 0.173693 0.528757 0.191785 0.537248 0.187577 0.544281 0.193366 0.509219 0.191626 0.519841 0.200843 0.493996 0.193428 0.519132 0.185382 0.528757 0.191785 0.519841 0.200843 0.538112 0.1583819 0.535214 0.166808 0.518998 0.1590279 0.499851 0.158434 0.518981 0.151749 0.518998 0.1590279 0.535214 0.166808 0.531131 0.171631 0.519016 0.165599 0.502799 0.166857 0.518998 0.1590279 0.519016 0.165599 0.519132 0.185382 0.519099 0.179457 0.528222 0.186316 0.509787 0.18626 0.519099 0.179457 0.519132 0.185382 0.528757 0.191785 0.528222 0.186316 0.533528 0.184215 0.504547 0.184206 0.509787 0.18626 0.509219 0.191626 0.533528 0.184215 0.533449 0.176739 0.537959 0.175966 0.504547 0.184206 0.50089 0.187571 0.5001 0.176033 0.537959 0.175966 0.533449 0.176739 0.531131 0.171631 0.50691 0.171667 0.504604 0.176791 0.5001 0.176033 0.533449 0.176739 0.533528 0.184215 0.528222 0.186316 0.504604 0.176791 0.519099 0.179457 0.509787 0.18626 0.519099 0.179457 0.519016 0.165599 0.531131 0.171631 0.50691 0.171667 0.519016 0.165599 0.519099 0.179457 0.519841 0.200843 0.544281 0.193366 0.553118 0.209599 0.485339 0.210053 0.493996 0.193428 0.519841 0.200843 0.544281 0.193366 0.548362 0.17356 0.561572 0.167779 0.476363 0.167996 0.489683 0.173693 0.493996 0.193428 0.538112 0.1583819 0.559475 0.1493189 0.561572 0.167779 0.499851 0.158434 0.489683 0.173693 0.476363 0.167996 0.532042 0.127713 0.555621 0.121749 0.559475 0.1493189 0.505828 0.127728 0.499851 0.158434 0.478371 0.149447 0.596138 0.133426 0.559475 0.1493189 0.555621 0.121749 0.441395 0.133592 0.454527 0.108481 0.482177 0.121781 0.601169 0.147885 0.561572 0.167779 0.559475 0.1493189 0.436337 0.148194 0.441395 0.133592 0.478371 0.149447 0.605512 0.165134 0.553118 0.209599 0.561572 0.167779 0.476363 0.167996 0.485339 0.210053 0.432024 0.165644 0.518927 0.08517998 0.518925 0.08386498 0.528933 0.084957 0.518927 0.08517998 0.506827 0.08741599 0.508915 0.08494496 0.531231 0.07382899 0.531019 0.08743095 0.528933 0.084957 0.508915 0.08494496 0.506827 0.08741599 0.506626 0.07381099 0.525483 0.06896698 0.531231 0.07382899 0.529036 0.07542896 0.50882 0.07541495 0.506626 0.07381099 0.512375 0.06895595 0.518928 0.06789898 0.525483 0.06896698 0.523751 0.070508 0.514106 0.07050096 0.512375 0.06895595 0.518928 0.06789898 0.523751 0.070508 0.52156 0.07497 0.518928 0.07425898 0.514106 0.07050096 0.518929 0.06946796 0.518928 0.07425898 0.523751 0.070508 0.529036 0.07542896 0.524236 0.07669097 0.513619 0.07668399 0.50882 0.07541495 0.514106 0.07050096 0.529036 0.07542896 0.528933 0.084957 0.524601 0.07988595 0.513252 0.07987898 0.508915 0.08494496 0.50882 0.07541495 0.518925 0.08386498 0.518926 0.07933098 0.524601 0.07988595 0.518925 0.08386498 0.508915 0.08494496 0.513252 0.07987898 0.518926 0.07933098 0.518928 0.07425898 0.52156 0.07497 0.516297 0.07496595 0.518928 0.07425898 0.518926 0.07933098 0.556923 0.291214 0.563905 0.272007 0.571787 0.277295 0.46807 0.278617 0.475886 0.273078 0.483433 0.292249 0.558527 0.316594 0.556923 0.291214 0.568351 0.292904 0.471978 0.294282 0.483433 0.292249 0.482619 0.317843 0.558527 0.316594 0.573085 0.311386 0.584855 0.327708 0.482619 0.317843 0.464579 0.34223 0.456477 0.329961 0.572941 0.258564 0.580734 0.26662 0.571787 0.277295 0.466409 0.259709 0.475886 0.273078 0.46807 0.278617 0.572941 0.258564 0.613992 0.242662 0.61172 0.255725 0.427062 0.257728 0.424464 0.244473 0.466409 0.259709 0.613992 0.242662 0.639236 0.253047 0.632494 0.262853 0.406068 0.265508 0.398938 0.255633 0.424464 0.244473 0.639236 0.253047 0.664101 0.277872 0.653658 0.279971 0.384904 0.283634 0.3741 0.281778 0.398938 0.255633 0.664101 0.277872 0.66844 0.297958 0.656064 0.297636 0.383015 0.301864 0.370304 0.302644 0.3741 0.281778 0.662406 0.312804 0.652752 0.310186 0.656064 0.297636 0.377061 0.317685 0.370304 0.302644 0.383015 0.301864 0.634472 0.332311 0.62904 0.323864 0.652752 0.310186 0.406362 0.33648 0.377061 0.317685 0.386858 0.314615 0.634472 0.332311 0.615546 0.342005 0.614408 0.331972 0.426727 0.335361 0.425972 0.345582 0.406362 0.33648 0.615546 0.342005 0.599845 0.344815 0.601033 0.333624 0.440344 0.336537 0.441977 0.347815 0.425972 0.345582 0.599845 0.344815 0.577279 0.340156 0.584855 0.327708 0.456477 0.329961 0.464579 0.34223 0.441977 0.347815 0.601033 0.333624 0.584855 0.327708 0.590644 0.321516 0.450408 0.323919 0.456477 0.329961 0.440344 0.336537 0.614408 0.331972 0.601033 0.333624 0.601799 0.328453 0.439372 0.331331 0.440344 0.336537 0.426727 0.335361 0.614408 0.331972 0.613335 0.327083 0.626851 0.320513 0.426727 0.335361 0.411556 0.327673 0.413648 0.324175 0.62904 0.323864 0.626851 0.320513 0.646248 0.306421 0.411556 0.327673 0.386858 0.314615 0.393381 0.31051 0.656064 0.297636 0.652752 0.310186 0.646248 0.306421 0.393381 0.31051 0.386858 0.314615 0.383015 0.301864 0.656064 0.297636 0.649541 0.296225 0.647785 0.283486 0.383015 0.301864 0.384904 0.283634 0.39104 0.287071 0.632494 0.262853 0.653658 0.279971 0.647785 0.283486 0.39104 0.287071 0.384904 0.283634 0.406068 0.265508 0.632494 0.262853 0.629829 0.267263 0.612641 0.26156 0.406068 0.265508 0.427062 0.257728 0.426254 0.263693 0.61172 0.255725 0.612641 0.26156 0.585166 0.270991 0.427062 0.257728 0.458737 0.268049 0.454369 0.272583 0.571787 0.277295 0.580734 0.26662 0.585166 0.270991 0.454369 0.272583 0.458737 0.268049 0.46807 0.278617 0.584855 0.327708 0.573085 0.311386 0.579548 0.30934 0.461204 0.311233 0.46779 0.313081 0.456477 0.329961 0.573085 0.311386 0.568351 0.292904 0.577524 0.293776 0.462754 0.295432 0.471978 0.294282 0.46779 0.313081 0.568351 0.292904 0.571787 0.277295 0.578124 0.2819 0.461798 0.283441 0.46807 0.278617 0.471978 0.294282 0.521923 0.386009 0.553095 0.390512 0.553209 0.433063 0.492809 0.434538 0.490934 0.391862 0.521923 0.386009 0.553095 0.390512 0.604826 0.397804 0.609819 0.431516 0.43586 0.43574 0.439252 0.40154 0.490934 0.391862 0.626842 0.395792 0.648174 0.419316 0.609819 0.431516 0.416915 0.400552 0.439252 0.40154 0.43586 0.43574 0.662817 0.372521 0.692106 0.388274 0.648174 0.419316 0.379297 0.378686 0.416915 0.400552 0.396518 0.425416 0.697446 0.332673 0.726332 0.341754 0.692106 0.388274 0.341964 0.339667 0.379297 0.378686 0.350292 0.396229 0.697446 0.332673 0.707254 0.310054 0.735879 0.312112 0.301067 0.320593 0.330721 0.316853 0.341964 0.339667 0.715342 0.265392 0.7299 0.256393 0.735879 0.312112 0.320452 0.270303 0.330721 0.316853 0.301067 0.320593 0.715342 0.265392 0.676379 0.233241 0.698172 0.216906 0.337414 0.219179 0.360308 0.235899 0.320452 0.270303 0.676379 0.233241 0.647395 0.200502 0.663103 0.190671 0.373474 0.191872 0.389677 0.20189 0.360308 0.235899 0.649444 0.02237796 0.660451 0.07608395 0.62144 0.04808896 0.388827 0.021586 0.411318 0.01513099 0.416419 0.04763096 0.626908 0.01560795 0.62144 0.04808896 0.577206 0.03280097 0.411318 0.01513099 0.470636 1.44e-4 0.460782 0.03265595 0.56746 1.44e-4 0.577206 0.03280097 0.547413 0.04172396 0.470636 1.44e-4 0.518922 0.02488595 0.490511 0.04166895 0.54026 0.05380499 0.518916 0.05029398 0.518922 0.02488595 0.518922 0.02488595 0.518916 0.05029398 0.497626 0.05376994 0.55193 0.05833798 0.54026 0.05380499 0.547413 0.04172396 0.490511 0.04166895 0.497626 0.05376994 0.485955 0.05827295 0.555073 0.06189996 0.55193 0.05833798 0.558059 0.05387097 0.479842 0.05378496 0.485955 0.05827295 0.482805 0.06182897 0.563812 0.076586 0.555073 0.06189996 0.576951 0.057998 0.46092 0.05784499 0.482805 0.06182897 0.474014 0.07651096 0.576951 0.057998 0.577206 0.03280097 0.62144 0.04808896 0.416419 0.04763096 0.460782 0.03265595 0.46092 0.05784499 0.558059 0.05387097 0.547413 0.04172396 0.577206 0.03280097 0.479842 0.05378496 0.46092 0.05784499 0.460782 0.03265595 0.611687 0.07826799 0.62144 0.04808896 0.660451 0.07608395 0.425932 0.07798498 0.410618 0.111244 0.376796 0.07529598 0.583135 0.1084949 0.563812 0.076586 0.611687 0.07826799 0.425932 0.07798498 0.474014 0.07651096 0.454527 0.108481 0.629482 0.130456 0.623495 0.146796 0.601169 0.147885 0.407648 0.130594 0.441395 0.133592 0.436337 0.148194 0.626663 0.111357 0.629482 0.130456 0.596138 0.133426 0.410618 0.111244 0.454527 0.108481 0.441395 0.133592 0.601169 0.147885 0.623495 0.146796 0.619303 0.159841 0.436337 0.148194 0.432024 0.165644 0.418035 0.160361 0.619303 0.159841 0.663103 0.190671 0.647395 0.200502 0.418035 0.160361 0.432024 0.165644 0.389677 0.20189 0.886245 0.1217769 0.849114 0.09973198 0.89178 0.03691595 0.141314 0.112482 0.07896095 0.06071895 0.142277 0.02146697 0.849114 0.09973198 0.788458 0.08082598 0.805584 0.01078599 0.183115 0.09212696 0.142277 0.02146697 0.232648 0.003483951 0.805584 0.01078599 0.788458 0.08082598 0.687018 0.07720398 0.349875 0.07595497 0.246353 0.07650995 0.232648 0.003483951 0.672384 0.022201 0.687018 0.07720398 0.660451 0.07608395 0.376796 0.07529598 0.349875 0.07595497 0.365979 0.02099096 0.660451 0.07608395 0.687018 0.07720398 0.629482 0.130456 0.376796 0.07529598 0.410618 0.111244 0.407648 0.130594 0.7299 0.256393 0.698172 0.216906 0.760215 0.193244 0.271553 0.193871 0.337414 0.219179 0.304876 0.261087 0.909112 0.183261 0.886245 0.1217769 0.9459 0.07956898 0.107928 0.179083 0.01182895 0.155367 0.07896095 0.06071895 0.862868 0.338556 0.894128 0.301884 0.962901 0.344752 0.160557 0.356821 0.1064 0.432652 0.04396796 0.367038 0.962901 0.344752 0.894128 0.301884 0.91536 0.259804 0.09896498 0.266968 0.123776 0.315519 0.04396796 0.367038 0.999856 0.25464 0.91536 0.259804 0.909112 0.183261 0.107928 0.179083 0.09896498 0.266968 1.44e-4 0.259113 0.749542 0.334683 0.735879 0.312112 0.766337 0.300809 0.267408 0.310142 0.301067 0.320593 0.288183 0.346496 0.789162 0.313727 0.766337 0.300809 0.815314 0.276388 0.213065 0.285164 0.267408 0.310142 0.242992 0.325552 0.846174 0.293397 0.815314 0.276388 0.845007 0.256352 0.179662 0.263312 0.213065 0.285164 0.178537 0.304983 0.873517 0.265922 0.845007 0.256352 0.859075 0.228168 0.1628029 0.23172 0.179662 0.263312 0.147089 0.274284 0.87503 0.184705 0.886999 0.2337689 0.859075 0.228168 0.145224 0.182749 0.176788 0.196179 0.1628029 0.23172 0.909112 0.183261 0.91536 0.259804 0.886999 0.2337689 0.131514 0.237587 0.09896498 0.266968 0.107928 0.179083 0.91536 0.259804 0.894128 0.301884 0.873517 0.265922 0.147089 0.274284 0.123776 0.315519 0.09896498 0.266968 0.894128 0.301884 0.862868 0.338556 0.846174 0.293397 0.178537 0.304983 0.160557 0.356821 0.123776 0.315519 0.862868 0.338556 0.794286 0.364062 0.789162 0.313727 0.242992 0.325552 0.239776 0.382592 0.160557 0.356821 0.749542 0.334683 0.789162 0.313727 0.794286 0.364062 0.288183 0.346496 0.268122 0.398737 0.239776 0.382592 0.794286 0.364062 0.862868 0.338556 0.911671 0.402429 0.239776 0.382592 0.185281 0.484099 0.1064 0.432652 0.770572 0.444261 0.7557 0.418603 0.770185 0.379538 0.271364 0.473316 0.21926 0.477186 0.268122 0.398737 0.815858 0.445381 0.770185 0.379538 0.794286 0.364062 0.239776 0.382592 0.268122 0.398737 0.21926 0.477186 0.7299 0.256393 0.789046 0.233323 0.766337 0.300809 0.304876 0.261087 0.301067 0.320593 0.267408 0.310142 0.789046 0.233323 0.809631 0.233887 0.815314 0.276388 0.213065 0.285164 0.219168 0.237388 0.241255 0.236977 0.809631 0.233887 0.829287 0.2195619 0.845007 0.256352 0.179662 0.263312 0.1990669 0.222464 0.219168 0.237388 0.859075 0.228168 0.845007 0.256352 0.829287 0.2195619 0.1628029 0.23172 0.176788 0.196179 0.1990669 0.222464 0.788458 0.08082598 0.78648 0.117591 0.715482 0.1397269 0.246353 0.07650995 0.349875 0.07595497 0.319538 0.139409 0.715482 0.1397269 0.78648 0.117591 0.785486 0.15233 0.319538 0.139409 0.271553 0.193871 0.2459689 0.151002 0.698172 0.216906 0.663103 0.190671 0.715482 0.1397269 0.319538 0.139409 0.373474 0.191872 0.337414 0.219179 0.663103 0.190671 0.623495 0.146796 0.629482 0.130456 0.407648 0.130594 0.413741 0.147158 0.373474 0.191872 0.837382 0.156361 0.858171 0.137775 0.87503 0.184705 0.196622 0.155241 0.176788 0.196179 0.145224 0.182749 0.87503 0.184705 0.858171 0.137775 0.886245 0.1217769 0.145224 0.182749 0.107928 0.179083 0.141314 0.112482 0.78648 0.117591 0.858171 0.137775 0.837382 0.156361 0.246666 0.11485 0.2459689 0.151002 0.196622 0.155241 0.849114 0.09973198 0.858171 0.137775 0.78648 0.117591 0.183115 0.09212696 0.246353 0.07650995 0.246666 0.11485 0.432388 0.894943 0.438797 0.870229 0.491058 0.881714 0.321637 0.893225 0.247207 0.901159 0.263032 0.878321 0.506166 0.904851 0.491058 0.881714 0.572792 0.860484 0.181486 0.854693 0.263032 0.878321 0.247207 0.901159 0.572792 0.860484 0.586396 0.793977 0.619962 0.791615 0.181486 0.854693 0.148729 0.873349 0.136063 0.784093 0.586396 0.793977 0.549027 0.746412 0.563786 0.739211 0.169745 0.787474 0.136063 0.784093 0.194086 0.733241 0.549027 0.746412 0.500314 0.711729 0.50827 0.697693 0.208656 0.740879 0.194086 0.733241 0.250811 0.693249 0.50827 0.697693 0.500314 0.711729 0.438641 0.680683 0.320962 0.677959 0.258399 0.707497 0.250811 0.693249 0.500314 0.711729 0.505666 0.730944 0.452955 0.700023 0.306136 0.696976 0.252524 0.726592 0.258399 0.707497 0.549027 0.746412 0.54285 0.755753 0.505666 0.730944 0.252524 0.726592 0.2145749 0.750414 0.208656 0.740879 0.568148 0.787367 0.54285 0.755753 0.549027 0.746412 0.188269 0.781375 0.169745 0.787474 0.208656 0.740879 0.555495 0.826352 0.568148 0.787367 0.586396 0.793977 0.19985 0.820889 0.181486 0.854693 0.169745 0.787474 0.501231 0.844356 0.555495 0.826352 0.572792 0.860484 0.253846 0.840502 0.263032 0.878321 0.181486 0.854693 0.491058 0.881714 0.438797 0.870229 0.457832 0.84004 0.297562 0.837358 0.315867 0.868209 0.263032 0.878321 0.785486 0.15233 0.796021 0.176969 0.783193 0.187449 0.2459689 0.151002 0.271553 0.193871 0.246955 0.187075 0.434803 0.658882 0.438641 0.680683 0.394766 0.686125 0.325318 0.656224 0.369913 0.610196 0.364838 0.684445 0.789046 0.233323 0.760215 0.193244 0.783193 0.187449 0.246955 0.187075 0.271553 0.193871 0.241255 0.236977 0.391747 0.862097 0.401605 0.84146 0.438797 0.870229 0.315867 0.868209 0.354026 0.840297 0.363377 0.861308 0.438641 0.680683 0.452955 0.700023 0.435018 0.71828 0.323658 0.715731 0.306136 0.696976 0.320962 0.677959 0.384658 0.710299 0.394766 0.686125 0.435018 0.71828 0.3744 0.708969 0.324726 0.727177 0.323658 0.715731 0.410995 0.747662 0.384658 0.710299 0.433669 0.729661 0.324726 0.727177 0.3744 0.708969 0.347028 0.745816 0.384657 0.795423 0.384658 0.710299 0.410995 0.747662 0.37227 0.794472 0.338952 0.783073 0.347028 0.745816 0.384657 0.795423 0.418086 0.784946 0.431333 0.817535 0.37227 0.794472 0.354026 0.840297 0.32479 0.81546 0.401605 0.84146 0.431333 0.817535 0.457832 0.84004 0.354026 0.840297 0.315867 0.868209 0.297562 0.837358 0.816266 0.203086 0.825107 0.209762 0.829287 0.2195619 0.209828 0.206161 0.219168 0.237388 0.1990669 0.222464 0.783193 0.187449 0.802192 0.1846089 0.816266 0.203086 0.246955 0.187075 0.219168 0.237388 0.209828 0.206161 0.457832 0.84004 0.431333 0.817535 0.448505 0.804621 0.307886 0.802031 0.32479 0.81546 0.297562 0.837358 0.418086 0.784946 0.435868 0.779569 0.448505 0.804621 0.338952 0.783073 0.32479 0.81546 0.307886 0.802031 0.418086 0.784946 0.410995 0.747662 0.423718 0.754191 0.334089 0.752045 0.347028 0.745816 0.338952 0.783073 0.427812 0.742828 0.43795 0.749777 0.423718 0.754191 0.33027 0.740536 0.347028 0.745816 0.334089 0.752045 0.433669 0.729661 0.445392 0.731997 0.43795 0.749777 0.324726 0.727177 0.33027 0.740536 0.319919 0.74725 0.433669 0.729661 0.435018 0.71828 0.440995 0.724383 0.31751 0.721697 0.323658 0.715731 0.324726 0.727177 0.435018 0.71828 0.452955 0.700023 0.455277 0.713731 0.30346 0.710657 0.306136 0.696976 0.323658 0.715731 0.501231 0.844356 0.457832 0.84004 0.473386 0.8247 0.282357 0.821525 0.297562 0.837358 0.253846 0.840502 0.555495 0.826352 0.501231 0.844356 0.512485 0.828811 0.2429749 0.824574 0.253846 0.840502 0.19985 0.820889 0.568148 0.787367 0.555495 0.826352 0.550942 0.811814 0.204839 0.806417 0.19985 0.820889 0.188269 0.781375 0.568148 0.787367 0.552139 0.787682 0.539407 0.764539 0.188269 0.781375 0.2145749 0.750414 0.217774 0.759319 0.54285 0.755753 0.539407 0.764539 0.508439 0.743135 0.2145749 0.750414 0.252524 0.726592 0.249419 0.738732 0.505666 0.730944 0.508439 0.743135 0.455277 0.713731 0.252524 0.726592 0.306136 0.696976 0.30346 0.710657 0.445392 0.731997 0.470841 0.748408 0.454776 0.761665 0.312907 0.729222 0.319919 0.74725 0.302729 0.758742 0.470841 0.748408 0.48887 0.770464 0.475403 0.783904 0.28696 0.74502 0.302729 0.758742 0.281439 0.780511 0.475403 0.783904 0.48887 0.770464 0.503673 0.787562 0.252972 0.78341 0.268291 0.766661 0.281439 0.780511 0.503673 0.787562 0.518562 0.791602 0.516802 0.807339 0.252972 0.78341 0.26179 0.798626 0.239243 0.802891 0.512485 0.828811 0.473386 0.8247 0.494476 0.80247 0.26179 0.798626 0.282357 0.821525 0.2429749 0.824574 0.475403 0.783904 0.494476 0.80247 0.473386 0.8247 0.281439 0.780511 0.307886 0.802031 0.282357 0.821525 0.435868 0.779569 0.454776 0.761665 0.475403 0.783904 0.321237 0.777208 0.307886 0.802031 0.281439 0.780511 0.43795 0.749777 0.454776 0.761665 0.435868 0.779569 0.321237 0.777208 0.302729 0.758742 0.319919 0.74725 0.455277 0.713731 0.470841 0.748408 0.445392 0.731997 0.30346 0.710657 0.31751 0.721697 0.312907 0.729222 0.508439 0.743135 0.48887 0.770464 0.470841 0.748408 0.28696 0.74502 0.268291 0.766661 0.249419 0.738732 0.503673 0.787562 0.48887 0.770464 0.508439 0.743135 0.252972 0.78341 0.217774 0.759319 0.249419 0.738732 0.518562 0.791602 0.503673 0.787562 0.539407 0.764539 0.23792 0.787045 0.204331 0.782156 0.217774 0.759319 0.550942 0.811814 0.516802 0.807339 0.518562 0.791602 0.23792 0.787045 0.239243 0.802891 0.204839 0.806417 0.50827 0.697693 0.434803 0.658882 0.484068 0.628776 0.276936 0.625067 0.325318 0.656224 0.250811 0.693249 0.563786 0.739211 0.50827 0.697693 0.543385 0.683538 0.216123 0.67812 0.250811 0.693249 0.194086 0.733241 0.563786 0.739211 0.581052 0.726933 0.616701 0.759965 0.194086 0.733241 0.136063 0.784093 0.140379 0.752377 0.619962 0.791615 0.616701 0.759965 0.660647 0.741167 0.136063 0.784093 0.04952597 0.748824 0.09703797 0.732052 0.707492 0.759884 0.660647 0.741167 0.677256 0.670436 0.04952597 0.748824 0.019409 0.639749 0.08356398 0.662038 0.745511 0.6521 0.677256 0.670436 0.671403 0.592656 0.019409 0.639749 0.03366398 0.564403 0.09281998 0.589862 0.543385 0.683538 0.484068 0.628776 0.671403 0.592656 0.216123 0.67812 0.08356398 0.662038 0.09281998 0.589862 0.677256 0.670436 0.660647 0.741167 0.581052 0.726933 0.1771759 0.720426 0.09703797 0.732052 0.08356398 0.662038 0.829287 0.2195619 0.834578 0.206879 0.834705 0.206959 0.05121594 0.522659 0.145041 0.562595 0.09281998 0.589862 0.62042 0.565675 0.671403 0.592656 0.484068 0.628776 0.276936 0.625067 0.09281998 0.589862 0.145041 0.562595 0.391039 0.611891 0.498072 0.552315 0.484068 0.628776 0.276936 0.625067 0.264218 0.55014 0.369913 0.610196 + + + + + + + + + + + + + + +

46 0 0 2 1 1 44 2 2 3 3 3 47 4 4 45 5 5 44 2 6 4 6 7 42 7 8 5 8 9 45 5 10 43 9 11 2 1 12 6 10 13 4 6 14 7 11 15 3 3 16 5 8 17 0 12 18 8 13 19 2 1 20 9 14 21 1 15 22 3 3 23 10 16 24 14 17 25 8 13 26 15 18 27 11 19 28 9 14 29 8 13 30 16 20 31 6 10 32 17 21 33 9 14 34 7 11 35 20 22 36 16 20 37 14 17 38 21 23 39 17 21 40 19 24 41 22 25 42 14 17 43 12 26 44 23 27 45 15 18 46 21 23 47 22 25 48 26 28 49 20 22 50 27 29 51 23 27 52 21 23 53 26 28 54 18 30 55 20 22 56 27 29 57 19 24 58 29 31 59 32 32 60 28 33 61 26 28 62 33 34 63 29 31 64 31 35 65 34 36 66 26 28 67 24 37 68 35 38 69 27 29 70 33 34 71 36 39 72 32 32 73 34 36 74 37 40 75 33 34 76 39 41 77 38 42 78 30 43 79 32 32 80 39 41 81 31 35 82 41 44 83 44 2 84 40 45 85 38 42 86 45 5 87 41 44 88 43 9 89 46 0 90 38 42 91 36 39 92 47 4 93 39 41 94 45 5 95 36 39 96 48 46 97 46 0 98 37 40 99 49 47 100 51 48 101 34 36 102 50 49 103 36 39 104 35 38 105 51 48 106 53 50 107 24 37 108 52 51 109 34 36 110 25 52 111 53 50 112 55 53 113 22 25 114 54 54 115 24 37 116 23 27 117 55 53 118 57 55 119 22 25 120 58 56 121 56 57 122 59 58 123 23 27 124 57 55 125 12 26 126 62 59 127 58 56 128 63 60 129 13 61 130 59 58 131 10 16 132 64 62 133 62 59 134 65 63 135 11 19 136 63 60 137 0 12 138 48 46 139 64 62 140 49 47 141 1 15 142 65 63 143 60 64 144 64 62 145 48 46 146 49 47 147 65 63 148 61 65 149 62 59 150 64 62 151 60 64 152 61 65 153 65 63 154 63 60 155 60 64 156 58 56 157 62 59 158 63 60 159 59 58 160 61 65 161 60 64 162 56 57 163 58 56 164 59 58 165 57 55 166 61 65 167 60 64 168 54 54 169 56 57 170 57 55 171 55 53 172 61 65 173 60 64 174 52 51 175 54 54 176 55 53 177 53 50 178 61 65 179 60 64 180 50 49 181 52 51 182 53 50 183 51 48 184 61 65 185 60 64 186 48 46 187 50 49 188 51 48 189 49 47 190 61 65 191 173 66 192 90 67 193 88 68 194 174 69 195 90 67 196 175 70 197 171 71 198 88 68 199 86 72 200 172 73 201 89 74 202 174 69 203 84 75 204 171 71 205 86 72 206 172 73 207 85 76 208 87 77 209 82 78 210 169 79 211 84 75 212 170 80 213 83 81 214 85 76 215 80 82 216 167 83 217 82 78 218 168 84 219 81 85 220 83 81 221 78 86 222 145 87 223 163 88 224 146 89 225 79 90 226 164 91 227 93 92 228 145 87 229 91 93 230 94 94 231 146 89 232 148 95 233 93 92 234 149 96 235 147 97 236 150 98 237 94 94 238 148 95 239 97 99 240 149 96 241 95 100 242 98 101 243 150 98 244 152 102 245 99 103 246 151 104 247 97 99 248 100 105 249 152 102 250 154 106 251 101 107 252 153 108 253 99 103 254 102 109 255 154 106 256 156 110 257 101 107 258 157 111 259 155 112 260 158 113 261 102 109 262 156 110 263 105 114 264 157 111 265 103 115 266 106 116 267 158 113 268 160 117 269 107 118 270 159 119 271 105 114 272 108 120 273 160 117 274 162 121 275 66 122 276 161 123 277 107 118 278 66 122 279 162 121 280 67 124 281 127 125 282 161 123 283 109 126 284 128 127 285 162 121 286 160 117 287 127 125 288 157 111 289 159 119 290 158 113 291 128 127 292 160 117 293 155 112 294 178 128 295 125 129 296 156 110 297 179 130 298 158 113 299 153 108 300 125 129 301 123 131 302 154 106 303 126 132 304 156 110 305 151 104 306 123 131 307 121 133 308 152 102 309 124 134 310 154 106 311 149 96 312 121 133 313 119 135 314 150 98 315 122 136 316 152 102 317 147 97 318 119 135 319 117 137 320 148 95 321 120 138 322 150 98 323 145 87 324 117 137 325 115 139 326 146 89 327 118 140 328 148 95 329 113 141 330 145 87 331 115 139 332 146 89 333 114 142 334 116 143 335 113 141 336 176 144 337 163 88 338 176 144 339 114 142 340 164 91 341 161 123 342 111 145 343 109 126 344 162 121 345 112 146 346 67 124 347 111 145 348 177 147 349 182 148 350 177 147 351 112 146 352 183 149 353 180 150 354 177 147 355 176 144 356 181 151 357 177 147 358 183 149 359 134 152 360 175 70 361 173 66 362 175 70 363 135 153 364 174 69 365 132 154 366 173 66 367 171 71 368 174 69 369 133 155 370 172 73 371 132 154 372 169 79 373 130 156 374 133 155 375 170 80 376 172 73 377 165 157 378 184 158 379 167 83 380 185 159 381 166 160 382 168 84 383 130 156 384 167 83 385 184 158 386 168 84 387 131 161 388 185 159 389 189 162 390 186 163 391 143 164 392 189 162 393 187 165 394 188 166 395 186 163 396 68 167 397 184 158 398 187 165 399 68 167 400 188 166 401 130 156 402 68 167 403 129 168 404 131 161 405 68 167 406 185 159 407 141 169 408 190 170 409 143 164 410 191 171 411 142 172 412 144 173 413 194 174 414 141 169 415 139 175 416 195 176 417 142 172 418 193 177 419 196 178 420 139 175 421 138 179 422 197 180 423 140 181 424 195 176 425 70 182 426 138 179 427 137 183 428 70 182 429 138 179 430 197 180 431 143 164 432 69 184 433 189 162 434 144 173 435 69 184 436 191 171 437 190 170 438 207 185 439 69 184 440 191 171 441 207 185 442 206 186 443 70 182 444 199 187 445 196 178 446 200 188 447 70 182 448 197 180 449 196 178 450 201 189 451 194 174 452 202 190 453 197 180 454 195 176 455 201 189 456 192 191 457 194 174 458 202 190 459 193 177 460 204 192 461 192 191 462 205 193 463 190 170 464 206 186 465 193 177 466 191 171 467 203 194 468 199 187 469 198 195 470 204 192 471 200 188 472 202 190 473 198 195 474 205 193 475 203 194 476 206 186 477 198 195 478 204 192 479 138 179 480 163 88 481 176 144 482 164 91 483 138 179 484 176 144 485 139 175 486 210 196 487 163 88 488 211 197 489 140 181 490 164 91 491 143 164 492 210 196 493 141 169 494 144 173 495 211 197 496 213 198 497 186 163 498 212 199 499 143 164 500 187 165 501 213 198 502 166 160 503 208 200 504 165 157 505 80 82 506 209 201 507 166 160 508 213 198 509 214 202 510 212 199 511 208 200 512 215 203 513 213 198 514 211 197 515 78 86 516 210 196 517 214 202 518 211 197 519 79 90 520 215 203 521 129 168 522 221 204 523 130 156 524 129 168 525 222 205 526 71 206 527 132 154 528 221 204 529 219 207 530 222 205 531 133 155 532 220 208 533 134 152 534 219 207 535 217 209 536 220 208 537 135 153 538 218 210 539 136 211 540 217 209 541 216 212 542 218 210 543 136 211 544 216 212 545 217 209 546 230 213 547 216 212 548 218 210 549 230 213 550 229 214 551 217 209 552 226 215 553 228 216 554 227 217 555 218 210 556 229 214 557 219 207 558 224 218 559 226 215 560 225 219 561 220 208 562 227 217 563 71 206 564 224 218 565 221 204 566 71 206 567 225 219 568 223 220 569 223 220 570 228 216 571 224 218 572 229 214 573 223 220 574 225 219 575 224 218 576 228 216 577 226 215 578 227 217 579 229 214 580 225 219 581 182 148 582 233 221 583 231 222 584 234 223 585 183 149 586 232 224 587 111 145 588 231 222 589 253 225 590 232 224 591 112 146 592 254 226 593 111 145 594 255 227 595 109 126 596 112 146 597 256 228 598 254 226 599 113 141 600 233 221 601 180 150 602 114 142 603 234 223 604 252 229 605 113 141 606 249 230 607 251 231 608 250 232 609 114 142 610 252 229 611 115 139 612 247 233 613 249 230 614 248 234 615 116 143 616 250 232 617 117 137 618 245 235 619 247 233 620 246 236 621 118 140 622 248 234 623 119 135 624 243 237 625 245 235 626 244 238 627 120 138 628 246 236 629 123 131 630 243 237 631 121 133 632 124 134 633 244 238 634 242 239 635 125 129 636 241 240 637 123 131 638 126 132 639 242 239 640 240 241 641 125 129 642 235 242 643 239 243 644 236 244 645 126 132 646 240 241 647 178 128 648 237 245 649 235 242 650 238 246 651 179 130 652 236 244 653 127 125 654 255 227 655 237 245 656 256 228 657 128 127 658 238 246 659 237 245 660 257 247 661 275 248 662 258 249 663 238 246 664 276 250 665 235 242 666 275 248 667 277 251 668 276 250 669 236 244 670 278 252 671 235 242 672 273 253 673 239 243 674 236 244 675 274 254 676 278 252 677 239 243 678 271 255 679 241 240 680 240 241 681 272 256 682 274 254 683 243 237 684 271 255 685 269 257 686 272 256 687 244 238 688 270 258 689 243 237 690 267 259 691 245 235 692 244 238 693 268 260 694 270 258 695 247 233 696 267 259 697 265 261 698 268 260 699 248 234 700 266 262 701 247 233 702 263 263 703 249 230 704 248 234 705 264 264 706 266 262 707 249 230 708 261 265 709 251 231 710 250 232 711 262 266 712 264 264 713 233 221 714 261 265 715 279 267 716 262 266 717 234 223 718 280 268 719 255 227 720 259 269 721 257 247 722 260 270 723 256 228 724 258 249 725 253 225 726 281 271 727 259 269 728 282 272 729 254 226 730 260 270 731 231 222 732 279 267 733 281 271 734 280 268 735 232 224 736 282 272 737 66 122 738 283 273 739 72 274 740 284 275 741 66 122 742 72 274 743 107 118 744 285 276 745 283 273 746 286 277 747 108 120 748 284 275 749 103 115 750 285 276 751 105 114 752 104 278 753 286 277 754 288 279 755 101 107 756 287 280 757 103 115 758 102 109 759 288 279 760 290 281 761 99 103 762 289 282 763 101 107 764 100 105 765 290 281 766 292 283 767 99 103 768 293 284 769 291 285 770 294 286 771 100 105 772 292 283 773 95 100 774 293 284 775 97 99 776 96 287 777 294 286 778 296 288 779 95 100 780 297 289 781 295 290 782 298 291 783 96 287 784 296 288 785 93 92 786 299 292 787 297 289 788 300 293 789 94 94 790 298 291 791 308 294 792 337 295 793 307 296 794 308 294 795 338 297 796 328 298 797 307 296 798 335 299 799 306 300 800 307 296 801 336 301 802 338 297 803 306 300 804 339 302 805 305 303 806 306 300 807 340 304 808 336 301 809 88 68 810 305 303 811 339 302 812 305 303 813 89 74 814 340 304 815 86 72 816 339 302 817 333 305 818 340 304 819 87 77 820 334 306 821 84 75 822 333 305 823 329 307 824 334 306 825 85 76 826 330 308 827 82 78 828 329 307 829 331 309 830 330 308 831 83 81 832 332 310 833 329 307 834 337 295 835 331 309 836 338 297 837 330 308 838 332 310 839 333 305 840 335 299 841 329 307 842 334 306 843 336 301 844 340 304 845 331 309 846 327 311 847 325 312 848 332 310 849 328 298 850 338 297 851 80 82 852 331 309 853 325 312 854 332 310 855 81 85 856 326 313 857 341 314 858 214 202 859 208 200 860 342 315 861 215 203 862 344 316 863 325 312 864 208 200 865 80 82 866 326 313 867 209 201 868 342 315 869 214 202 870 345 317 871 78 86 872 215 203 873 346 318 874 344 316 875 345 317 876 91 93 877 78 86 878 346 318 879 92 319 880 300 293 881 323 320 882 303 321 883 76 322 884 324 323 885 303 321 886 352 324 887 351 325 888 77 326 889 303 321 890 352 324 891 77 326 892 350 327 893 77 326 894 347 328 895 304 329 896 348 330 897 77 326 898 304 329 899 304 329 900 327 311 901 308 294 902 328 298 903 304 329 904 308 294 905 327 311 906 341 314 907 325 312 908 328 298 909 342 315 910 348 330 911 295 290 912 317 331 913 309 332 914 318 333 915 296 288 916 310 334 917 315 335 918 76 322 919 75 336 920 316 337 921 76 322 922 324 323 923 357 338 924 302 339 925 301 340 926 358 341 927 302 339 928 356 342 929 302 339 930 353 343 931 74 344 932 354 345 933 302 339 934 74 344 935 74 344 936 315 335 937 75 336 938 316 337 939 74 344 940 75 336 941 291 285 942 361 346 943 363 347 944 362 348 945 292 283 946 364 349 947 363 347 948 367 350 949 365 351 950 368 352 951 364 349 952 366 353 953 365 351 954 369 354 955 371 355 956 370 356 957 366 353 958 372 357 959 371 355 960 375 358 961 373 359 962 376 360 963 372 357 964 374 361 965 377 362 966 375 358 967 313 363 968 378 364 969 376 360 970 374 361 971 315 335 972 373 359 973 377 362 974 374 361 975 316 337 976 378 364 977 353 343 978 371 355 979 373 359 980 372 357 981 354 345 982 374 361 983 355 365 984 365 351 985 371 355 986 366 353 987 356 342 988 372 357 989 357 338 990 363 347 991 365 351 992 364 349 993 358 341 994 366 353 995 291 285 996 359 366 997 289 282 998 292 283 999 360 367 1000 364 349 1001 359 366 1002 301 340 1003 73 368 1004 360 367 1005 301 340 1006 358 341 1007 285 276 1008 289 282 1009 283 273 1010 286 277 1011 290 281 1012 288 279 1013 283 273 1014 359 366 1015 73 368 1016 360 367 1017 284 275 1018 73 368 1019 72 274 1020 283 273 1021 73 368 1022 73 368 1023 284 275 1024 72 274 1025 295 290 1026 361 346 1027 293 284 1028 296 288 1029 362 348 1030 310 334 1031 309 332 1032 367 350 1033 361 346 1034 368 352 1035 310 334 1036 362 348 1037 311 369 1038 369 354 1039 367 350 1040 370 356 1041 312 370 1042 368 352 1043 375 358 1044 381 371 1045 313 363 1046 376 360 1047 382 372 1048 370 356 1049 349 373 1050 383 374 1051 347 328 1052 350 327 1053 384 375 1054 386 376 1055 383 374 1056 319 377 1057 317 331 1058 384 375 1059 320 378 1060 386 376 1061 297 289 1062 383 374 1063 317 331 1064 384 375 1065 298 291 1066 318 333 1067 299 292 1068 341 314 1069 383 374 1070 342 315 1071 300 293 1072 384 375 1073 341 314 1074 347 328 1075 383 374 1076 384 375 1077 348 330 1078 342 315 1079 299 292 1080 345 317 1081 343 379 1082 344 316 1083 346 318 1084 300 293 1085 321 380 1086 377 362 1087 313 363 1088 322 381 1089 378 364 1090 380 382 1091 377 362 1092 323 320 1093 315 335 1094 378 364 1095 324 323 1096 380 382 1097 385 383 1098 321 380 1099 319 377 1100 386 376 1101 322 381 1102 380 382 1103 351 325 1104 385 383 1105 349 373 1106 352 324 1107 386 376 1108 380 382 1109 323 320 1110 379 384 1111 351 325 1112 352 324 1113 380 382 1114 324 323 1115 387 385 1116 401 386 1117 399 387 1118 388 388 1119 402 389 1120 414 390 1121 399 387 1122 403 391 1123 397 392 1124 404 393 1125 400 394 1126 398 395 1127 403 391 1128 395 396 1129 397 392 1130 404 393 1131 396 397 1132 406 398 1133 405 399 1134 393 400 1135 395 396 1136 406 398 1137 394 401 1138 408 402 1139 407 403 1140 391 404 1141 393 400 1142 408 402 1143 392 405 1144 410 406 1145 391 404 1146 411 407 1147 389 408 1148 412 409 1149 392 405 1150 390 410 1151 409 411 1152 417 412 1153 411 407 1154 418 413 1155 410 406 1156 412 409 1157 407 403 1158 419 414 1159 409 411 1160 420 415 1161 408 402 1162 410 406 1163 423 416 1164 407 403 1165 405 399 1166 424 417 1167 408 402 1168 422 418 1169 425 419 1170 405 399 1171 403 391 1172 426 420 1173 406 398 1174 424 417 1175 427 421 1176 403 391 1177 401 386 1178 428 422 1179 404 393 1180 426 420 1181 401 386 1182 415 423 1183 427 421 1184 416 424 1185 402 389 1186 428 422 1187 319 377 1188 441 425 1189 317 331 1190 320 378 1191 442 426 1192 444 427 1193 389 408 1194 443 428 1195 319 377 1196 390 410 1197 444 427 1198 412 409 1199 309 332 1200 441 425 1201 311 369 1202 442 426 1203 310 334 1204 312 370 1205 381 371 1206 413 429 1207 387 385 1208 414 390 1209 382 372 1210 388 388 1211 411 407 1212 439 430 1213 443 428 1214 440 431 1215 412 409 1216 444 427 1217 445 432 1218 439 430 1219 437 433 1220 446 434 1221 440 431 1222 444 427 1223 433 435 1224 437 433 1225 435 436 1226 438 437 1227 434 438 1228 436 439 1229 447 440 1230 433 435 1231 431 441 1232 448 442 1233 434 438 1234 446 434 1235 447 440 1236 449 443 1237 429 444 1238 448 442 1239 450 445 1240 432 446 1241 429 444 1242 415 423 1243 413 429 1244 430 447 1245 416 424 1246 450 445 1247 447 440 1248 381 371 1249 311 369 1250 448 442 1251 382 372 1252 430 447 1253 441 425 1254 447 440 1255 311 369 1256 442 426 1257 448 442 1258 446 434 1259 441 425 1260 443 428 1261 445 432 1262 446 434 1263 444 427 1264 442 426 1265 415 423 1266 451 448 1267 475 449 1268 452 450 1269 416 424 1270 476 451 1271 431 441 1272 451 448 1273 449 443 1274 432 446 1275 452 450 1276 462 452 1277 431 441 1278 459 453 1279 461 454 1280 460 455 1281 432 446 1282 462 452 1283 435 436 1284 459 453 1285 433 435 1286 436 439 1287 460 455 1288 458 456 1289 437 433 1290 457 457 1291 435 436 1292 438 437 1293 458 456 1294 456 458 1295 437 433 1296 453 459 1297 455 460 1298 454 461 1299 438 437 1300 456 458 1301 439 430 1302 473 462 1303 453 459 1304 474 463 1305 440 431 1306 454 461 1307 427 421 1308 475 449 1309 463 464 1310 476 451 1311 428 422 1312 464 465 1313 425 419 1314 463 464 1315 465 466 1316 464 465 1317 426 420 1318 466 467 1319 423 416 1320 465 466 1321 467 468 1322 466 467 1323 424 417 1324 468 469 1325 423 416 1326 469 470 1327 421 471 1328 424 417 1329 470 472 1330 468 469 1331 421 471 1332 471 473 1333 419 414 1334 422 418 1335 472 474 1336 470 472 1337 419 414 1338 473 462 1339 417 412 1340 420 415 1341 474 463 1342 472 474 1343 455 460 1344 477 475 1345 457 457 1346 456 458 1347 478 476 1348 480 477 1349 479 478 1350 483 479 1351 477 475 1352 480 477 1353 484 480 1354 482 481 1355 483 479 1356 487 482 1357 485 483 1358 488 484 1359 484 480 1360 486 485 1361 487 482 1362 491 486 1363 485 483 1364 488 484 1365 492 487 1366 490 488 1367 463 464 1368 485 483 1369 491 486 1370 486 485 1371 464 465 1372 492 487 1373 483 479 1374 475 449 1375 451 448 1376 484 480 1377 476 451 1378 486 485 1379 461 454 1380 483 479 1381 451 448 1382 462 452 1383 484 480 1384 478 476 1385 457 457 1386 461 454 1387 459 453 1388 462 452 1389 458 456 1390 460 455 1391 473 462 1392 455 460 1393 453 459 1394 474 463 1395 456 458 1396 480 477 1397 471 473 1398 479 478 1399 473 462 1400 480 477 1401 472 474 1402 474 463 1403 487 482 1404 471 473 1405 469 470 1406 488 484 1407 472 474 1408 482 481 1409 489 489 1410 469 470 1411 467 468 1412 490 488 1413 470 472 1414 488 484 1415 465 466 1416 489 489 1417 467 468 1418 490 488 1419 466 467 1420 468 469 1421 463 464 1422 491 486 1423 465 466 1424 466 467 1425 492 487 1426 464 465 1427 391 404 1428 503 490 1429 501 491 1430 504 492 1431 392 405 1432 502 493 1433 393 400 1434 501 491 1435 499 494 1436 502 493 1437 394 401 1438 500 495 1439 393 400 1440 497 496 1441 395 396 1442 394 401 1443 498 497 1444 500 495 1445 395 396 1446 495 498 1447 397 392 1448 396 397 1449 496 499 1450 498 497 1451 397 392 1452 493 500 1453 399 387 1454 398 395 1455 494 501 1456 496 499 1457 399 387 1458 505 502 1459 387 385 1460 400 394 1461 506 503 1462 494 501 1463 501 491 1464 505 502 1465 493 500 1466 502 493 1467 506 503 1468 504 492 1469 493 500 1470 499 494 1471 501 491 1472 500 495 1473 494 501 1474 502 493 1475 495 498 1476 497 496 1477 499 494 1478 500 495 1479 498 497 1480 496 499 1481 381 371 1482 505 502 1483 313 363 1484 382 372 1485 506 503 1486 388 388 1487 313 363 1488 503 490 1489 321 380 1490 504 492 1491 314 504 1492 322 381 1493 319 377 1494 503 490 1495 389 408 1496 504 492 1497 320 378 1498 390 410 1499 46 0 1500 0 12 1501 2 1 1502 3 3 1503 1 15 1504 47 4 1505 44 2 1506 2 1 1507 4 6 1508 5 8 1509 3 3 1510 45 5 1511 2 1 1512 8 13 1513 6 10 1514 7 11 1515 9 14 1516 3 3 1517 0 12 1518 10 16 1519 8 13 1520 9 14 1521 11 19 1522 1 15 1523 10 16 1524 12 26 1525 14 17 1526 15 18 1527 13 61 1528 11 19 1529 8 13 1530 14 17 1531 16 20 1532 17 21 1533 15 18 1534 9 14 1535 20 22 1536 18 30 1537 16 20 1538 21 23 1539 15 18 1540 17 21 1541 22 25 1542 20 22 1543 14 17 1544 23 27 1545 13 61 1546 15 18 1547 22 25 1548 24 37 1549 26 28 1550 27 29 1551 25 52 1552 23 27 1553 26 28 1554 28 33 1555 18 30 1556 27 29 1557 21 23 1558 19 24 1559 32 32 1560 30 43 1561 28 33 1562 33 34 1563 27 29 1564 29 31 1565 34 36 1566 32 32 1567 26 28 1568 35 38 1569 25 52 1570 27 29 1571 36 39 1572 38 42 1573 32 32 1574 37 40 1575 35 38 1576 33 34 1577 38 42 1578 40 45 1579 30 43 1580 39 41 1581 33 34 1582 31 35 1583 44 2 1584 42 7 1585 40 45 1586 45 5 1587 39 41 1588 41 44 1589 46 0 1590 44 2 1591 38 42 1592 47 4 1593 37 40 1594 39 41 1595 36 39 1596 50 49 1597 48 46 1598 37 40 1599 47 4 1600 49 47 1601 34 36 1602 52 51 1603 50 49 1604 35 38 1605 37 40 1606 51 48 1607 24 37 1608 54 54 1609 52 51 1610 25 52 1611 35 38 1612 53 50 1613 22 25 1614 56 57 1615 54 54 1616 23 27 1617 25 52 1618 55 53 1619 22 25 1620 12 26 1621 58 56 1622 59 58 1623 13 61 1624 23 27 1625 12 26 1626 10 16 1627 62 59 1628 63 60 1629 11 19 1630 13 61 1631 10 16 1632 0 12 1633 64 62 1634 65 63 1635 1 15 1636 11 19 1637 0 12 1638 46 0 1639 48 46 1640 49 47 1641 47 4 1642 1 15 1643 173 66 1644 175 70 1645 90 67 1646 174 69 1647 89 74 1648 90 67 1649 171 71 1650 173 66 1651 88 68 1652 172 73 1653 87 77 1654 89 74 1655 84 75 1656 169 79 1657 171 71 1658 172 73 1659 170 80 1660 85 76 1661 82 78 1662 167 83 1663 169 79 1664 170 80 1665 168 84 1666 83 81 1667 80 82 1668 165 157 1669 167 83 1670 168 84 1671 166 160 1672 81 85 1673 78 86 1674 91 93 1675 145 87 1676 146 89 1677 92 319 1678 79 90 1679 93 92 1680 147 97 1681 145 87 1682 94 94 1683 92 319 1684 146 89 1685 93 92 1686 95 100 1687 149 96 1688 150 98 1689 96 287 1690 94 94 1691 97 99 1692 151 104 1693 149 96 1694 98 101 1695 96 287 1696 150 98 1697 99 103 1698 153 108 1699 151 104 1700 100 105 1701 98 101 1702 152 102 1703 101 107 1704 155 112 1705 153 108 1706 102 109 1707 100 105 1708 154 106 1709 101 107 1710 103 115 1711 157 111 1712 158 113 1713 104 278 1714 102 109 1715 105 114 1716 159 119 1717 157 111 1718 106 116 1719 104 278 1720 158 113 1721 107 118 1722 161 123 1723 159 119 1724 108 120 1725 106 116 1726 160 117 1727 66 122 1728 67 124 1729 161 123 1730 66 122 1731 108 120 1732 162 121 1733 127 125 1734 159 119 1735 161 123 1736 128 127 1737 110 505 1738 162 121 1739 127 125 1740 178 128 1741 157 111 1742 158 113 1743 179 130 1744 128 127 1745 155 112 1746 157 111 1747 178 128 1748 156 110 1749 126 132 1750 179 130 1751 153 108 1752 155 112 1753 125 129 1754 154 106 1755 124 134 1756 126 132 1757 151 104 1758 153 108 1759 123 131 1760 152 102 1761 122 136 1762 124 134 1763 149 96 1764 151 104 1765 121 133 1766 150 98 1767 120 138 1768 122 136 1769 147 97 1770 149 96 1771 119 135 1772 148 95 1773 118 140 1774 120 138 1775 145 87 1776 147 97 1777 117 137 1778 146 89 1779 116 143 1780 118 140 1781 113 141 1782 163 88 1783 145 87 1784 146 89 1785 164 91 1786 114 142 1787 113 141 1788 180 150 1789 176 144 1790 176 144 1791 181 151 1792 114 142 1793 161 123 1794 67 124 1795 111 145 1796 162 121 1797 110 505 1798 112 146 1799 111 145 1800 67 124 1801 177 147 1802 177 147 1803 67 124 1804 112 146 1805 180 150 1806 182 148 1807 177 147 1808 181 151 1809 176 144 1810 177 147 1811 134 152 1812 136 211 1813 175 70 1814 175 70 1815 136 211 1816 135 153 1817 132 154 1818 134 152 1819 173 66 1820 174 69 1821 135 153 1822 133 155 1823 132 154 1824 171 71 1825 169 79 1826 133 155 1827 131 161 1828 170 80 1829 165 157 1830 186 163 1831 184 158 1832 185 159 1833 187 165 1834 166 160 1835 130 156 1836 169 79 1837 167 83 1838 168 84 1839 170 80 1840 131 161 1841 189 162 1842 188 166 1843 186 163 1844 189 162 1845 144 173 1846 187 165 1847 186 163 1848 188 166 1849 68 167 1850 187 165 1851 185 159 1852 68 167 1853 130 156 1854 184 158 1855 68 167 1856 131 161 1857 129 168 1858 68 167 1859 141 169 1860 192 191 1861 190 170 1862 191 171 1863 193 177 1864 142 172 1865 194 174 1866 192 191 1867 141 169 1868 195 176 1869 140 181 1870 142 172 1871 196 178 1872 194 174 1873 139 175 1874 197 180 1875 138 179 1876 140 181 1877 70 182 1878 196 178 1879 138 179 1880 143 164 1881 190 170 1882 69 184 1883 144 173 1884 189 162 1885 69 184 1886 190 170 1887 205 193 1888 207 185 1889 191 171 1890 69 184 1891 207 185 1892 70 182 1893 198 195 1894 199 187 1895 200 188 1896 198 195 1897 70 182 1898 196 178 1899 199 187 1900 201 189 1901 202 190 1902 200 188 1903 197 180 1904 201 189 1905 203 194 1906 192 191 1907 202 190 1908 195 176 1909 193 177 1910 192 191 1911 203 194 1912 205 193 1913 206 186 1914 204 192 1915 193 177 1916 203 194 1917 201 189 1918 199 187 1919 204 192 1920 198 195 1921 200 188 1922 198 195 1923 207 185 1924 205 193 1925 206 186 1926 207 185 1927 198 195 1928 138 179 1929 139 175 1930 163 88 1931 164 91 1932 140 181 1933 138 179 1934 139 175 1935 141 169 1936 210 196 1937 211 197 1938 142 172 1939 140 181 1940 143 164 1941 212 199 1942 210 196 1943 144 173 1944 142 172 1945 211 197 1946 186 163 1947 165 157 1948 212 199 1949 187 165 1950 144 173 1951 213 198 1952 208 200 1953 212 199 1954 165 157 1955 209 201 1956 81 85 1957 166 160 1958 214 202 1959 210 196 1960 212 199 1961 215 203 1962 209 201 1963 213 198 1964 78 86 1965 163 88 1966 210 196 1967 211 197 1968 164 91 1969 79 90 1970 129 168 1971 71 206 1972 221 204 1973 129 168 1974 131 161 1975 222 205 1976 132 154 1977 130 156 1978 221 204 1979 222 205 1980 131 161 1981 133 155 1982 134 152 1983 132 154 1984 219 207 1985 220 208 1986 133 155 1987 135 153 1988 136 211 1989 134 152 1990 217 209 1991 218 210 1992 135 153 1993 136 211 1994 217 209 1995 228 216 1996 230 213 1997 218 210 1998 216 212 1999 230 213 2000 217 209 2001 219 207 2002 226 215 2003 227 217 2004 220 208 2005 218 210 2006 219 207 2007 221 204 2008 224 218 2009 225 219 2010 222 205 2011 220 208 2012 71 206 2013 223 220 2014 224 218 2015 71 206 2016 222 205 2017 225 219 2018 223 220 2019 230 213 2020 228 216 2021 229 214 2022 230 213 2023 223 220 2024 182 148 2025 180 150 2026 233 221 2027 234 223 2028 181 151 2029 183 149 2030 111 145 2031 182 148 2032 231 222 2033 232 224 2034 183 149 2035 112 146 2036 111 145 2037 253 225 2038 255 227 2039 112 146 2040 110 505 2041 256 228 2042 113 141 2043 251 231 2044 233 221 2045 114 142 2046 181 151 2047 234 223 2048 113 141 2049 115 139 2050 249 230 2051 250 232 2052 116 143 2053 114 142 2054 115 139 2055 117 137 2056 247 233 2057 248 234 2058 118 140 2059 116 143 2060 117 137 2061 119 135 2062 245 235 2063 246 236 2064 120 138 2065 118 140 2066 119 135 2067 121 133 2068 243 237 2069 244 238 2070 122 136 2071 120 138 2072 123 131 2073 241 240 2074 243 237 2075 124 134 2076 122 136 2077 244 238 2078 125 129 2079 239 243 2080 241 240 2081 126 132 2082 124 134 2083 242 239 2084 125 129 2085 178 128 2086 235 242 2087 236 244 2088 179 130 2089 126 132 2090 178 128 2091 127 125 2092 237 245 2093 238 246 2094 128 127 2095 179 130 2096 127 125 2097 109 126 2098 255 227 2099 256 228 2100 110 505 2101 128 127 2102 237 245 2103 255 227 2104 257 247 2105 258 249 2106 256 228 2107 238 246 2108 235 242 2109 237 245 2110 275 248 2111 276 250 2112 238 246 2113 236 244 2114 235 242 2115 277 251 2116 273 253 2117 236 244 2118 240 241 2119 274 254 2120 239 243 2121 273 253 2122 271 255 2123 240 241 2124 242 239 2125 272 256 2126 243 237 2127 241 240 2128 271 255 2129 272 256 2130 242 239 2131 244 238 2132 243 237 2133 269 257 2134 267 259 2135 244 238 2136 246 236 2137 268 260 2138 247 233 2139 245 235 2140 267 259 2141 268 260 2142 246 236 2143 248 234 2144 247 233 2145 265 261 2146 263 263 2147 248 234 2148 250 232 2149 264 264 2150 249 230 2151 263 263 2152 261 265 2153 250 232 2154 252 229 2155 262 266 2156 233 221 2157 251 231 2158 261 265 2159 262 266 2160 252 229 2161 234 223 2162 255 227 2163 253 225 2164 259 269 2165 260 270 2166 254 226 2167 256 228 2168 253 225 2169 231 222 2170 281 271 2171 282 272 2172 232 224 2173 254 226 2174 231 222 2175 233 221 2176 279 267 2177 280 268 2178 234 223 2179 232 224 2180 66 122 2181 107 118 2182 283 273 2183 284 275 2184 108 120 2185 66 122 2186 107 118 2187 105 114 2188 285 276 2189 286 277 2190 106 116 2191 108 120 2192 103 115 2193 287 280 2194 285 276 2195 104 278 2196 106 116 2197 286 277 2198 101 107 2199 289 282 2200 287 280 2201 102 109 2202 104 278 2203 288 279 2204 99 103 2205 291 285 2206 289 282 2207 100 105 2208 102 109 2209 290 281 2210 99 103 2211 97 99 2212 293 284 2213 294 286 2214 98 101 2215 100 105 2216 95 100 2217 295 290 2218 293 284 2219 96 287 2220 98 101 2221 294 286 2222 95 100 2223 93 92 2224 297 289 2225 298 291 2226 94 94 2227 96 287 2228 93 92 2229 91 93 2230 299 292 2231 300 293 2232 92 319 2233 94 94 2234 308 294 2235 327 311 2236 337 295 2237 308 294 2238 307 296 2239 338 297 2240 307 296 2241 337 295 2242 335 299 2243 307 296 2244 306 300 2245 336 301 2246 306 300 2247 335 299 2248 339 302 2249 306 300 2250 305 303 2251 340 304 2252 88 68 2253 90 67 2254 305 303 2255 305 303 2256 90 67 2257 89 74 2258 86 72 2259 88 68 2260 339 302 2261 340 304 2262 89 74 2263 87 77 2264 84 75 2265 86 72 2266 333 305 2267 334 306 2268 87 77 2269 85 76 2270 82 78 2271 84 75 2272 329 307 2273 330 308 2274 85 76 2275 83 81 2276 329 307 2277 335 299 2278 337 295 2279 338 297 2280 336 301 2281 330 308 2282 333 305 2283 339 302 2284 335 299 2285 334 306 2286 330 308 2287 336 301 2288 331 309 2289 337 295 2290 327 311 2291 332 310 2292 326 313 2293 328 298 2294 80 82 2295 82 78 2296 331 309 2297 332 310 2298 83 81 2299 81 85 2300 341 314 2301 343 379 2302 214 202 2303 342 315 2304 209 201 2305 215 203 2306 325 312 2307 341 314 2308 208 200 2309 326 313 2310 81 85 2311 209 201 2312 214 202 2313 343 379 2314 345 317 2315 215 203 2316 79 90 2317 346 318 2318 345 317 2319 299 292 2320 91 93 2321 346 318 2322 79 90 2323 92 319 2324 323 320 2325 351 325 2326 303 321 2327 324 323 2328 76 322 2329 303 321 2330 351 325 2331 349 373 2332 77 326 2333 352 324 2334 303 321 2335 77 326 2336 77 326 2337 349 373 2338 347 328 2339 348 330 2340 350 327 2341 77 326 2342 304 329 2343 347 328 2344 327 311 2345 328 298 2346 348 330 2347 304 329 2348 327 311 2349 347 328 2350 341 314 2351 328 298 2352 326 313 2353 342 315 2354 295 290 2355 297 289 2356 317 331 2357 318 333 2358 298 291 2359 296 288 2360 315 335 2361 323 320 2362 76 322 2363 316 337 2364 75 336 2365 76 322 2366 357 338 2367 355 365 2368 302 339 2369 358 341 2370 301 340 2371 302 339 2372 302 339 2373 355 365 2374 353 343 2375 354 345 2376 356 342 2377 302 339 2378 74 344 2379 353 343 2380 315 335 2381 316 337 2382 354 345 2383 74 344 2384 291 285 2385 293 284 2386 361 346 2387 362 348 2388 294 286 2389 292 283 2390 363 347 2391 361 346 2392 367 350 2393 368 352 2394 362 348 2395 364 349 2396 365 351 2397 367 350 2398 369 354 2399 370 356 2400 368 352 2401 366 353 2402 371 355 2403 369 354 2404 375 358 2405 376 360 2406 370 356 2407 372 357 2408 377 362 2409 373 359 2410 375 358 2411 378 364 2412 314 504 2413 376 360 2414 315 335 2415 353 343 2416 373 359 2417 374 361 2418 354 345 2419 316 337 2420 353 343 2421 355 365 2422 371 355 2423 372 357 2424 356 342 2425 354 345 2426 355 365 2427 357 338 2428 365 351 2429 366 353 2430 358 341 2431 356 342 2432 357 338 2433 359 366 2434 363 347 2435 364 349 2436 360 367 2437 358 341 2438 291 285 2439 363 347 2440 359 366 2441 292 283 2442 290 281 2443 360 367 2444 359 366 2445 357 338 2446 301 340 2447 360 367 2448 73 368 2449 301 340 2450 285 276 2451 287 280 2452 289 282 2453 286 277 2454 284 275 2455 290 281 2456 283 273 2457 289 282 2458 359 366 2459 360 367 2460 290 281 2461 284 275 2462 295 290 2463 309 332 2464 361 346 2465 296 288 2466 294 286 2467 362 348 2468 309 332 2469 311 369 2470 367 350 2471 368 352 2472 312 370 2473 310 334 2474 311 369 2475 381 371 2476 369 354 2477 370 356 2478 382 372 2479 312 370 2480 375 358 2481 369 354 2482 381 371 2483 376 360 2484 314 504 2485 382 372 2486 349 373 2487 385 383 2488 383 374 2489 350 327 2490 348 330 2491 384 375 2492 383 374 2493 385 383 2494 319 377 2495 384 375 2496 318 333 2497 320 378 2498 297 289 2499 299 292 2500 383 374 2501 384 375 2502 300 293 2503 298 291 2504 299 292 2505 343 379 2506 341 314 2507 342 315 2508 344 316 2509 300 293 2510 321 380 2511 379 384 2512 377 362 2513 322 381 2514 314 504 2515 378 364 2516 377 362 2517 379 384 2518 323 320 2519 378 364 2520 316 337 2521 324 323 2522 385 383 2523 379 384 2524 321 380 2525 386 376 2526 320 378 2527 322 381 2528 351 325 2529 379 384 2530 385 383 2531 352 324 2532 350 327 2533 386 376 2534 387 385 2535 413 429 2536 401 386 2537 388 388 2538 400 394 2539 402 389 2540 399 387 2541 401 386 2542 403 391 2543 404 393 2544 402 389 2545 400 394 2546 403 391 2547 405 399 2548 395 396 2549 404 393 2550 398 395 2551 396 397 2552 405 399 2553 407 403 2554 393 400 2555 406 398 2556 396 397 2557 394 401 2558 407 403 2559 409 411 2560 391 404 2561 408 402 2562 394 401 2563 392 405 2564 391 404 2565 409 411 2566 411 407 2567 412 409 2568 410 406 2569 392 405 2570 409 411 2571 419 414 2572 417 412 2573 418 413 2574 420 415 2575 410 406 2576 407 403 2577 421 471 2578 419 414 2579 420 415 2580 422 418 2581 408 402 2582 423 416 2583 421 471 2584 407 403 2585 424 417 2586 406 398 2587 408 402 2588 425 419 2589 423 416 2590 405 399 2591 426 420 2592 404 393 2593 406 398 2594 427 421 2595 425 419 2596 403 391 2597 428 422 2598 402 389 2599 404 393 2600 401 386 2601 413 429 2602 415 423 2603 416 424 2604 414 390 2605 402 389 2606 319 377 2607 443 428 2608 441 425 2609 320 378 2610 318 333 2611 442 426 2612 389 408 2613 411 407 2614 443 428 2615 390 410 2616 320 378 2617 444 427 2618 309 332 2619 317 331 2620 441 425 2621 442 426 2622 318 333 2623 310 334 2624 381 371 2625 429 444 2626 413 429 2627 414 390 2628 430 447 2629 382 372 2630 411 407 2631 417 412 2632 439 430 2633 440 431 2634 418 413 2635 412 409 2636 445 432 2637 443 428 2638 439 430 2639 446 434 2640 438 437 2641 440 431 2642 433 435 2643 445 432 2644 437 433 2645 438 437 2646 446 434 2647 434 438 2648 447 440 2649 445 432 2650 433 435 2651 448 442 2652 432 446 2653 434 438 2654 447 440 2655 431 441 2656 449 443 2657 448 442 2658 430 447 2659 450 445 2660 429 444 2661 449 443 2662 415 423 2663 430 447 2664 414 390 2665 416 424 2666 447 440 2667 429 444 2668 381 371 2669 448 442 2670 312 370 2671 382 372 2672 441 425 2673 445 432 2674 447 440 2675 442 426 2676 312 370 2677 448 442 2678 415 423 2679 449 443 2680 451 448 2681 452 450 2682 450 445 2683 416 424 2684 431 441 2685 461 454 2686 451 448 2687 432 446 2688 450 445 2689 452 450 2690 431 441 2691 433 435 2692 459 453 2693 460 455 2694 434 438 2695 432 446 2696 435 436 2697 457 457 2698 459 453 2699 436 439 2700 434 438 2701 460 455 2702 437 433 2703 455 460 2704 457 457 2705 438 437 2706 436 439 2707 458 456 2708 437 433 2709 439 430 2710 453 459 2711 454 461 2712 440 431 2713 438 437 2714 439 430 2715 417 412 2716 473 462 2717 474 463 2718 418 413 2719 440 431 2720 427 421 2721 415 423 2722 475 449 2723 476 451 2724 416 424 2725 428 422 2726 425 419 2727 427 421 2728 463 464 2729 464 465 2730 428 422 2731 426 420 2732 423 416 2733 425 419 2734 465 466 2735 466 467 2736 426 420 2737 424 417 2738 423 416 2739 467 468 2740 469 470 2741 424 417 2742 422 418 2743 470 472 2744 421 471 2745 469 470 2746 471 473 2747 422 418 2748 420 415 2749 472 474 2750 419 414 2751 471 473 2752 473 462 2753 420 415 2754 418 413 2755 474 463 2756 455 460 2757 479 478 2758 477 475 2759 456 458 2760 458 456 2761 478 476 2762 479 478 2763 481 506 2764 483 479 2765 480 477 2766 478 476 2767 484 480 2768 483 479 2769 481 506 2770 487 482 2771 488 484 2772 482 481 2773 484 480 2774 487 482 2775 489 489 2776 491 486 2777 488 484 2778 486 485 2779 492 487 2780 463 464 2781 475 449 2782 485 483 2783 486 485 2784 476 451 2785 464 465 2786 483 479 2787 485 483 2788 475 449 2789 484 480 2790 452 450 2791 476 451 2792 461 454 2793 477 475 2794 483 479 2795 462 452 2796 452 450 2797 484 480 2798 457 457 2799 477 475 2800 461 454 2801 462 452 2802 478 476 2803 458 456 2804 473 462 2805 479 478 2806 455 460 2807 474 463 2808 454 461 2809 456 458 2810 471 473 2811 481 506 2812 479 478 2813 480 477 2814 482 481 2815 472 474 2816 487 482 2817 481 506 2818 471 473 2819 488 484 2820 470 472 2821 472 474 2822 489 489 2823 487 482 2824 469 470 2825 490 488 2826 468 469 2827 470 472 2828 465 466 2829 491 486 2830 489 489 2831 490 488 2832 492 487 2833 466 467 2834 391 404 2835 389 408 2836 503 490 2837 504 492 2838 390 410 2839 392 405 2840 393 400 2841 391 404 2842 501 491 2843 502 493 2844 392 405 2845 394 401 2846 393 400 2847 499 494 2848 497 496 2849 394 401 2850 396 397 2851 498 497 2852 395 396 2853 497 496 2854 495 498 2855 396 397 2856 398 395 2857 496 499 2858 397 392 2859 495 498 2860 493 500 2861 398 395 2862 400 394 2863 494 501 2864 399 387 2865 493 500 2866 505 502 2867 400 394 2868 388 388 2869 506 503 2870 501 491 2871 503 490 2872 505 502 2873 502 493 2874 494 501 2875 506 503 2876 493 500 2877 495 498 2878 499 494 2879 500 495 2880 496 499 2881 494 501 2882 381 371 2883 387 385 2884 505 502 2885 382 372 2886 314 504 2887 506 503 2888 313 363 2889 505 502 2890 503 490 2891 504 492 2892 506 503 2893 314 504 2894 319 377 2895 321 380 2896 503 490 2897 504 492 2898 322 381 2899 320 378 2900

+
+
+
+
+ + + + -1 0 0 0 0 0 1 0 0 1 0 0 0 0 0 1 + + + + + + + +
\ No newline at end of file diff --git a/src/Orbit/Core/IO/Parser/ITextParser.h b/src/Orbit/Core/Color/RGB.h similarity index 69% rename from src/Orbit/Core/IO/Parser/ITextParser.h rename to src/Orbit/Core/Color/RGB.h index 82a641b..b63485c 100644 --- a/src/Orbit/Core/IO/Parser/ITextParser.h +++ b/src/Orbit/Core/Color/RGB.h @@ -16,30 +16,41 @@ */ #pragma once -#include "Orbit/Core/IO/Parser/IParser.h" +#include "Orbit/Core/Core.h" + +// Fix macro interference on Windows +#if defined( ORB_OS_WINDOWS ) +# include +# undef RGB +#endif // ORB_OS_WINDOWS ORB_NAMESPACE_BEGIN -class ORB_API_CORE ITextParser : public IParser +class RGB { public: - explicit ITextParser( ByteSpan data ); - -protected: + constexpr RGB( void ) = default; - void SkipWhitespace ( void ); - std::string ReadAlphaNumeric( void ); - std::string ReadPrintable ( void ); - std::string ReadLiteral ( void ); + constexpr explicit RGB( float grey ) + : r( grey ) + , g( grey ) + , b( grey ) + { + } -protected: + constexpr RGB( float red, float green, float blue ) + : r( red ) + , g( green ) + , b( blue ) + { + } - std::string Peek( size_t length ) const; - -protected: +public: - bool ExpectString( std::string_view str ); + float r = 0.0f; + float g = 0.0f; + float b = 0.0f; }; diff --git a/src/Orbit/Core/IO/Parser/IParser.h b/src/Orbit/Core/Color/RGBA.h similarity index 66% rename from src/Orbit/Core/IO/Parser/IParser.h rename to src/Orbit/Core/Color/RGBA.h index 0c2f964..cf69220 100644 --- a/src/Orbit/Core/IO/Parser/IParser.h +++ b/src/Orbit/Core/Color/RGBA.h @@ -16,41 +16,46 @@ */ #pragma once -#include "Orbit/Core/Utility/Span.h" - -#include -#include +#include "Orbit/Core/Color/RGB.h" ORB_NAMESPACE_BEGIN -class ORB_API_CORE IParser +class RGBA { public: - explicit IParser( ByteSpan data ); - virtual ~IParser( void ) = default; + constexpr RGBA( void ) = default; + + constexpr explicit RGBA( float grey ) + : r( grey ) + , g( grey ) + , b( grey ) + , a( 1.0f ) + { + } + + constexpr explicit RGBA( const RGB& rgb, float alpha = 1.0f ) + : r( rgb.r ) + , g( rgb.g ) + , b( rgb.b ) + , a( alpha ) + { + } + + constexpr RGBA( float red, float green, float blue, float alpha = 1.0f ) + : r( red ) + , g( green ) + , b( blue ) + , a( alpha ) + { + } public: - bool IsGood( void ) const { return good_; } - -protected: - - void Skip ( size_t size ); - void ReadBytes( void* dst, size_t count ); - -protected: - - bool IsEOF( void ) const; - -protected: - - std::unique_ptr< uint8_t[] > data_; - - size_t size_; - size_t offset_; - - bool good_; + float r = 0.0f; + float g = 0.0f; + float b = 0.0f; + float a = 1.0f; }; diff --git a/src/Orbit/Core/Core.h b/src/Orbit/Core/Core.h index dff37f7..0267806 100644 --- a/src/Orbit/Core/Core.h +++ b/src/Orbit/Core/Core.h @@ -18,8 +18,36 @@ #pragma once #include "Orbit/Orbit.h" +#include + #if defined( ORB_BUILD_CORE ) # define ORB_API_CORE ORB_DLL_EXPORT #else // ORB_BUILD_CORE # define ORB_API_CORE ORB_DLL_IMPORT #endif // !ORB_BUILD_CORE + +ORB_NAMESPACE_BEGIN + +template< typename T, typename... RemainingTypes > +constexpr T Or( T first, RemainingTypes... remaining ) +{ + static_assert( ( std::is_convertible_v< T, RemainingTypes > && ... ) ); + + if( !!first ) + { + return first; + } + else + { + for( T var : { remaining... } ) + { + if( !!var ) + return var; + } + } + + // None of them evaluated to true. Just return the first one. + return first; +} + +ORB_NAMESPACE_END diff --git a/src/Orbit/Core/IO/Asset.cpp b/src/Orbit/Core/IO/Asset.cpp index c775a23..43158c4 100644 --- a/src/Orbit/Core/IO/Asset.cpp +++ b/src/Orbit/Core/IO/Asset.cpp @@ -37,7 +37,7 @@ ORB_NAMESPACE_BEGIN Asset::Asset( std::string_view path ) { - ORB_TRACE( "Loading asset: %s", path.data() ); + ORB_TRACE( "Loading asset: %.*s", path.size(), path.data() ); #if defined( ORB_OS_WINDOWS ) diff --git a/src/Orbit/Core/IO/Parser/IParser.cpp b/src/Orbit/Core/IO/File/BinaryFile.cpp similarity index 56% rename from src/Orbit/Core/IO/Parser/IParser.cpp rename to src/Orbit/Core/IO/File/BinaryFile.cpp index f88f623..bff3b0e 100644 --- a/src/Orbit/Core/IO/Parser/IParser.cpp +++ b/src/Orbit/Core/IO/File/BinaryFile.cpp @@ -15,43 +15,39 @@ * 3. This notice may not be removed or altered from any source distribution. */ -#include "IParser.h" +#include "BinaryFile.h" #include ORB_NAMESPACE_BEGIN -IParser::IParser( ByteSpan data ) - : data_ { data.Copy() } - , size_ { data.Size() } - , offset_{ 0 } - , good_ { false } +void BinaryFile::Init( size_t total_size ) { + total_size_ = total_size; + current_offset_ = 0; } -void IParser::Skip( size_t size ) +void BinaryFile::Skip( size_t size ) { - if( ( offset_ + size ) < size_ ) offset_ += size; - else offset_ = size_; + // Increment offset, but clamp it so it doesn't exceed the total size + current_offset_ = std::min( current_offset_ + size, total_size_ ); } -void IParser::ReadBytes( void* dst, size_t count ) +void BinaryFile::ReadBytes( const void* src, void* dst, size_t size ) { - if( ( offset_ + count ) < size_ ) - { - std::memcpy( dst, &data_[ offset_ ], count ); - offset_ += count; - } - else - { - std::memcpy( dst, &data_[ offset_ ], ( offset_ - size_ ) ); - offset_ = size_; - } + // Make sure we don't read more bytes than we have + const size_t bytes_to_write = std::min( size, total_size_ - current_offset_ ); + + // Copy bytes from src to dst + std::memcpy( dst, static_cast< const uint8_t* >( src ) + current_offset_, bytes_to_write ); + + // Increment offset + current_offset_ += bytes_to_write; } -bool IParser::IsEOF( void ) const +bool BinaryFile::IsEOF( void ) const { - return ( offset_ == size_ ); + return ( current_offset_ == total_size_ ); } ORB_NAMESPACE_END diff --git a/src/Orbit/Core/IO/Parser/XML/XMLElement.h b/src/Orbit/Core/IO/File/BinaryFile.h similarity index 54% rename from src/Orbit/Core/IO/Parser/XML/XMLElement.h rename to src/Orbit/Core/IO/File/BinaryFile.h index 7b703e1..97fb155 100644 --- a/src/Orbit/Core/IO/Parser/XML/XMLElement.h +++ b/src/Orbit/Core/IO/File/BinaryFile.h @@ -16,41 +16,42 @@ */ #pragma once -#include "Orbit/Core/IO/Parser/XML/XMLAttribute.h" +#include "Orbit/Core/Utility/Span.h" -#include -#include +#include +#include ORB_NAMESPACE_BEGIN -class ORB_API_CORE XMLElement +/** Base class for file format parsers. + * Does not store the data by default. This is by design in case the output can be processed in chunks rather than + * needing to wait until the entire file has been parsed. + */ +class ORB_API_CORE BinaryFile { public: - std::string_view Attribute( std::string_view key ) const; + BinaryFile( void ) = default; + virtual ~BinaryFile( void ) = default; -public: +protected: - const XMLElement& ChildWithAttribute( std::string_view element, std::string_view attribute, std::string_view value ) const; - size_t CountChildren ( std::string_view element ) const; - bool IsValid ( void ) const; + /** Initializes the metadata */ + void Init( size_t total_size ); -public: + /** Skips forward @size amount of bytes */ + void Skip( size_t size ); - auto begin( void ) const { return children.begin(); } - auto end ( void ) const { return children.end(); } + /** Reads @size amount of bytes from @src and writes them to @dst */ + void ReadBytes( const void* src, void* dst, size_t size ); -public: - - const XMLElement& operator[]( std::string_view key ) const; - -public: + /** Returns whether or not the file has reached the end */ + bool IsEOF( void ) const; - std::string name; - std::string content; +protected: - std::vector< XMLAttribute > attributes; - std::vector< XMLElement > children; + size_t total_size_ = 0; + size_t current_offset_ = 0; }; diff --git a/src/Orbit/Core/IO/Parser/TGA/TGAParser.cpp b/src/Orbit/Core/IO/File/Image/TARGA/TARGAFile.cpp similarity index 72% rename from src/Orbit/Core/IO/Parser/TGA/TGAParser.cpp rename to src/Orbit/Core/IO/File/Image/TARGA/TARGAFile.cpp index bc93646..c10c27a 100644 --- a/src/Orbit/Core/IO/Parser/TGA/TGAParser.cpp +++ b/src/Orbit/Core/IO/File/Image/TARGA/TARGAFile.cpp @@ -15,7 +15,7 @@ * 3. This notice may not be removed or altered from any source distribution. */ -#include "TGAParser.h" +#include "TARGAFile.h" #include @@ -71,23 +71,24 @@ struct Header ImageSpecification image_specification; }; -TGAParser::TGAParser( ByteSpan data ) - : IParser( data ) +TARGAFile::TARGAFile( ByteSpan data ) { + Init( data.Size() ); + Header header; - ReadBytes( &header.id_length, 1 ); - ReadBytes( &header.color_map_type, 1 ); - ReadBytes( &header.image_type, 1 ); - ReadBytes( &header.color_map_specification.first_entry_index, 2 ); - ReadBytes( &header.color_map_specification.color_map_length, 2 ); - ReadBytes( &header.color_map_specification.color_map_entry_size, 1 ); - ReadBytes( &header.image_specification.x, 2 ); - ReadBytes( &header.image_specification.y, 2 ); - ReadBytes( &header.image_specification.width, 2 ); - ReadBytes( &header.image_specification.height, 2 ); - ReadBytes( &header.image_specification.depth, 1 ); - ReadBytes( &header.image_specification.descriptor, 1 ); + ReadBytes( data.Ptr(), &header.id_length, 1 ); + ReadBytes( data.Ptr(), &header.color_map_type, 1 ); + ReadBytes( data.Ptr(), &header.image_type, 1 ); + ReadBytes( data.Ptr(), &header.color_map_specification.first_entry_index, 2 ); + ReadBytes( data.Ptr(), &header.color_map_specification.color_map_length, 2 ); + ReadBytes( data.Ptr(), &header.color_map_specification.color_map_entry_size, 1 ); + ReadBytes( data.Ptr(), &header.image_specification.x, 2 ); + ReadBytes( data.Ptr(), &header.image_specification.y, 2 ); + ReadBytes( data.Ptr(), &header.image_specification.width, 2 ); + ReadBytes( data.Ptr(), &header.image_specification.height, 2 ); + ReadBytes( data.Ptr(), &header.image_specification.depth, 1 ); + ReadBytes( data.Ptr(), &header.image_specification.descriptor, 1 ); if( ( header.image_specification.width == 0 ) || ( header.image_specification.height == 0 ) ) return; @@ -111,19 +112,19 @@ TGAParser::TGAParser( ByteSpan data ) { case ImageType::UncompressedTrueColor: { - image_data_.reset( new uint32_t[ pixel_count ] ); + image_data_ = std::make_unique< uint32_t[] >( pixel_count ); for( size_t i = 0; i < pixel_count; ++i ) - image_data_[ i ] = ReadTrueColor(); + image_data_[ i ] = ReadTrueColor( data.Ptr() ); } break; case ImageType::RunLengthEncodedTrueColor: { - image_data_.reset( new uint32_t[ pixel_count ] ); + image_data_ = std::make_unique< uint32_t[] >( pixel_count ); for( size_t pixels_read = 0; pixels_read < pixel_count; ) - pixels_read += ReadNextRLEPacket( &image_data_[ pixels_read ] ); + pixels_read += ReadNextRLEPacket( data.Ptr(), &image_data_[ pixels_read ] ); } break; @@ -135,22 +136,21 @@ TGAParser::TGAParser( ByteSpan data ) width_ = header.image_specification.width; height_ = header.image_specification.height; - good_ = true; } -uint32_t TGAParser::ReadTrueColor( void ) +uint32_t TARGAFile::ReadTrueColor( const void* src ) { uint32_t argb = 0xFF000000; - ReadBytes( &argb, bytes_per_pixel_ ); + ReadBytes( src, &argb, bytes_per_pixel_ ); const uint32_t abgr = ( ( argb & 0xFF00FF00 ) | ( ( argb & 0x00FF0000 ) >> 16 ) | ( ( argb & 0x000000FF ) << 16 ) ); return abgr; } -size_t TGAParser::ReadNextRLEPacket( uint32_t* dst ) +size_t TARGAFile::ReadNextRLEPacket( const void* src, uint32_t* dst ) { uint8_t repetition_count_and_packet_type = 0; - ReadBytes( &repetition_count_and_packet_type, 1 ); + ReadBytes( src, &repetition_count_and_packet_type, 1 ); const PacketType packet_type = static_cast< PacketType >( ( repetition_count_and_packet_type & 0x80 ) >> 7 ); const uint8_t repetition_count = ( ( repetition_count_and_packet_type & 0x7f ) + 1 ); @@ -160,13 +160,13 @@ size_t TGAParser::ReadNextRLEPacket( uint32_t* dst ) case PacketType::Raw: { for( size_t i = 0; i < repetition_count; ++i ) - dst[ i ] = ReadTrueColor(); + dst[ i ] = ReadTrueColor( src ); } break; case PacketType::RunLength: { - const uint32_t pixel_value = ReadTrueColor(); + const uint32_t pixel_value = ReadTrueColor( src ); for( size_t i = 0; i < repetition_count; ++i ) dst[ i ] = pixel_value; diff --git a/src/Orbit/Core/IO/Parser/TGA/TGAParser.h b/src/Orbit/Core/IO/File/Image/TARGA/TARGAFile.h similarity index 73% rename from src/Orbit/Core/IO/Parser/TGA/TGAParser.h rename to src/Orbit/Core/IO/File/Image/TARGA/TARGAFile.h index 06756fc..4d15138 100644 --- a/src/Orbit/Core/IO/Parser/TGA/TGAParser.h +++ b/src/Orbit/Core/IO/File/Image/TARGA/TARGAFile.h @@ -16,7 +16,7 @@ */ #pragma once -#include "Orbit/Core/IO/Parser/IParser.h" +#include "Orbit/Core/IO/File/BinaryFile.h" #include #include @@ -24,22 +24,22 @@ ORB_NAMESPACE_BEGIN -class ORB_API_CORE TGAParser : public IParser +class ORB_API_CORE TARGAFile : public BinaryFile { public: - explicit TGAParser( ByteSpan data ); + explicit TARGAFile( ByteSpan data ); public: - const uint32_t* ImageData( void ) const { return image_data_.get(); } - uint16_t Width ( void ) const { return width_; } - uint16_t Height ( void ) const { return height_; } + const uint32_t* ImageData ( void ) const { return image_data_.get(); } + uint16_t Width ( void ) const { return width_; } + uint16_t Height ( void ) const { return height_; } private: - uint32_t ReadTrueColor ( void ); - size_t ReadNextRLEPacket( uint32_t* dst ); + uint32_t ReadTrueColor ( const void* src ); + size_t ReadNextRLEPacket ( const void* src, uint32_t* dst ); private: diff --git a/src/Orbit/Core/IO/Parser/XML/XMLAttribute.h b/src/Orbit/Core/IO/File/Markup/XML/XMLAttribute.h similarity index 91% rename from src/Orbit/Core/IO/Parser/XML/XMLAttribute.h rename to src/Orbit/Core/IO/File/Markup/XML/XMLAttribute.h index 0eb3947..f5ca1d2 100644 --- a/src/Orbit/Core/IO/Parser/XML/XMLAttribute.h +++ b/src/Orbit/Core/IO/File/Markup/XML/XMLAttribute.h @@ -19,6 +19,7 @@ #include "Orbit/Core/Core.h" #include +#include ORB_NAMESPACE_BEGIN @@ -28,4 +29,10 @@ struct XMLAttribute std::string value; }; +struct XMLAttributeView +{ + std::string_view name; + std::string_view value; +}; + ORB_NAMESPACE_END diff --git a/src/Orbit/Core/IO/Parser/XML/XMLElement.cpp b/src/Orbit/Core/IO/File/Markup/XML/XMLElement.cpp similarity index 61% rename from src/Orbit/Core/IO/Parser/XML/XMLElement.cpp rename to src/Orbit/Core/IO/File/Markup/XML/XMLElement.cpp index 3ae03b4..c335058 100644 --- a/src/Orbit/Core/IO/Parser/XML/XMLElement.cpp +++ b/src/Orbit/Core/IO/File/Markup/XML/XMLElement.cpp @@ -19,60 +19,63 @@ ORB_NAMESPACE_BEGIN -std::string_view XMLElement::Attribute( std::string_view key ) const +std::string_view XMLElement::FindAttribute( std::string_view attribute_name ) const { for( const XMLAttribute& attribute : attributes ) { - if( attribute.name == key ) + if( attribute.name == attribute_name ) return attribute.value; } return { }; } -const XMLElement& XMLElement::ChildWithAttribute( std::string_view element, std::string_view attribute, std::string_view value ) const +const XMLElement* XMLElement::FindChild( std::string_view child_name ) const { for( const XMLElement& child : children ) { - if( child.name != element ) - continue; + if( child.name == child_name ) + return &child; + } - for( const XMLAttribute& attrib : child.attributes ) - { - if( attrib.name != attribute || attrib.value != value ) - continue; + return nullptr; +} - return child; +const XMLElement* XMLElement::FindChildWithAttribute( std::string_view child_name, XMLAttributeView attribute ) const +{ + for( const XMLElement& child : children ) + { + if( child.name == child_name ) + { + for( const XMLAttribute& attrib : child.attributes ) + { + if( attrib.name == attribute.name && attrib.value == attribute.value ) + return &child; + } } } - static XMLElement dummy; - return dummy; + return nullptr; } -size_t XMLElement::CountChildren( std::string_view element ) const +size_t XMLElement::CountChildrenWithName( std::string_view child_name ) const { size_t count = 0; for( const XMLElement& child : children ) { - if( child.name == element ) + if( child.name == child_name ) ++count; } return count; } -bool XMLElement::IsValid( void ) const -{ - return !( name.empty() && content.empty() && children.empty() && attributes.empty() ); -} - -const XMLElement& XMLElement::operator[]( std::string_view key ) const +const XMLElement& XMLElement::operator[]( std::string_view child_name ) const { for( const XMLElement& child : children ) { - if( child.name == key ) + if( child.name == child_name ) return child; } diff --git a/src/Orbit/Core/IO/File/Markup/XML/XMLElement.h b/src/Orbit/Core/IO/File/Markup/XML/XMLElement.h new file mode 100644 index 0000000..263318b --- /dev/null +++ b/src/Orbit/Core/IO/File/Markup/XML/XMLElement.h @@ -0,0 +1,69 @@ +/* + * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ + * + * This software is provided 'as-is', without any express or implied warranty. In no event will + * the authors be held liable for any damages arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, including commercial + * applications, and to alter it and redistribute it freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the + * original software. If you use this software in a product, an acknowledgment in the product + * documentation would be appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as + * being the original software. + * 3. This notice may not be removed or altered from any source distribution. + */ + +#pragma once +#include "Orbit/Core/IO/File/Markup/XML/XMLAttribute.h" + +#include +#include + +ORB_NAMESPACE_BEGIN + +class ORB_API_CORE XMLElement +{ +public: + + /** Searches for an attribute with the name @name and returns its value */ + std::string_view FindAttribute( std::string_view attribute_name ) const; + + /** Searched for a child by the name @name */ + const XMLElement* FindChild( std::string_view child_name ) const; + +public: + + /** Searches all children and returns a pointer to the first one with an element by the name @name that + * contains the attribute @attribute. + * Returns nullptr if no child was found. + */ + const XMLElement* FindChildWithAttribute( std::string_view child_name, XMLAttributeView attribute ) const; + + /** Returns the number of children called @name */ + size_t CountChildrenWithName( std::string_view child_name ) const; + +public: + + /** Enable range-based looping over this element's children */ + auto begin ( void ) const { return children.begin(); } + auto end ( void ) const { return children.end(); } + +public: + + /** Returns a reference to a child with the name @key. + * If no child was found, returns a reference to an empty dummy. + */ + const XMLElement& operator[]( std::string_view child_name ) const; + +public: + + std::string name; + std::string content; + std::vector< XMLAttribute > attributes; + std::vector< XMLElement > children; + +}; + +ORB_NAMESPACE_END diff --git a/src/Orbit/Core/IO/File/Markup/XML/XMLFile.cpp b/src/Orbit/Core/IO/File/Markup/XML/XMLFile.cpp new file mode 100644 index 0000000..7f7d56e --- /dev/null +++ b/src/Orbit/Core/IO/File/Markup/XML/XMLFile.cpp @@ -0,0 +1,126 @@ +/* + * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ + * + * This software is provided 'as-is', without any express or implied warranty. In no event will + * the authors be held liable for any damages arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, including commercial + * applications, and to alter it and redistribute it freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the + * original software. If you use this software in a product, an acknowledgment in the product + * documentation would be appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as + * being the original software. + * 3. This notice may not be removed or altered from any source distribution. + */ + +#include "XMLFile.h" + +#include + +ORB_NAMESPACE_BEGIN + +XMLFile::XMLFile( ByteSpan data ) +{ + Init( data.Size() ); + + const char* src = reinterpret_cast< const char* >( data.Ptr() ); + + if( !ExpectString( src, R"()" ) ) + return; + + SkipWhitespace( src ); + + while( !IsEOF() ) + { + if( !ParseElement( src, &root_element_ ) ) + return; + } +} + +std::string XMLFile::ReadName( const char* src ) +{ + // Search for digits, letters, periods, hyphens, underscores and colons + constexpr std::string_view valid_characters = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz.-_:"; + const std::string_view characters_to_be_searched = std::string_view( src + current_offset_, total_size_ - current_offset_ ); + const size_t end_of_name = characters_to_be_searched.find_first_not_of( valid_characters ); + const std::string_view name = characters_to_be_searched.substr( 0, end_of_name ); + + Skip( end_of_name ); + + return std::string( name ); +} + +std::string XMLFile::ReadContent( const char* src ) +{ + // Search for left-hand angular brackets + const std::string_view characters_to_be_searched = std::string_view( src + current_offset_, total_size_ - current_offset_ ); + const size_t end_of_content = characters_to_be_searched.find_first_of( '<' ); + const std::string_view content = characters_to_be_searched.substr( 0, end_of_content ); + + Skip( end_of_content ); + + return std::string( content ); +} + +bool XMLFile::ParseElement( const char* src, XMLElement* parent ) +{ + XMLElement element; + + if( !ExpectString( src, "<" ) ) + return false; + + element.name = ReadName( src ); + SkipWhitespace( src ); + + // Attributes + while( Peek( src, 1 ) != ">" && Peek( src, 2 ) != "/>" ) + { + XMLAttribute attribute; + attribute.name = ReadName( src ); + + if( !ExpectString( src, "=" ) ) + return false; + + attribute.value = ReadCapturedStringLiteral( src ); + element.attributes.push_back( attribute ); + + SkipWhitespace( src ); + } + + if( ExpectString( src, ">" ) ) + { + SkipWhitespace( src ); + + /* Does element have character data instead of child elements? */ + if( Peek( src, 1 ) != "<" ) + { + element.content = ReadContent( src ); + + SkipWhitespace( src ); + if( !ExpectString( src, "" ) ) + return false; + } + else + { + while( !ExpectString( src, "" ) ) + { + if( !ParseElement( src, &element ) ) + return false; + } + } + } + else if( !ExpectString( src, "/>" ) ) + { + return false; + } + + parent->children.push_back( element ); + + SkipWhitespace( src ); + + return true; +} + +ORB_NAMESPACE_END diff --git a/src/Orbit/Core/IO/File/Markup/XML/XMLFile.h b/src/Orbit/Core/IO/File/Markup/XML/XMLFile.h new file mode 100644 index 0000000..dfac9cc --- /dev/null +++ b/src/Orbit/Core/IO/File/Markup/XML/XMLFile.h @@ -0,0 +1,56 @@ +/* + * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ + * + * This software is provided 'as-is', without any express or implied warranty. In no event will + * the authors be held liable for any damages arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, including commercial + * applications, and to alter it and redistribute it freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the + * original software. If you use this software in a product, an acknowledgment in the product + * documentation would be appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as + * being the original software. + * 3. This notice may not be removed or altered from any source distribution. + */ + +#pragma once +#include "Orbit/Core/IO/File/Markup/XML/XMLElement.h" +#include "Orbit/Core/IO/File/TextFile.h" + +#include + +ORB_NAMESPACE_BEGIN + +class ORB_API_CORE XMLFile : public TextFile +{ +public: + + /** Parses @data into an element hierarchy */ + explicit XMLFile( ByteSpan data ); + ~XMLFile( void ) = default; + +public: + + const XMLElement& GetRootElement( void ) const { return root_element_; } + +protected: + + XMLElement root_element_; + +private: + + /** Reads a string from @src, compatible as an XML element name, and returns the string */ + std::string ReadName( const char* src ); + + /** Reads a string from @src, compatible as XML element content, and returns the string */ + std::string ReadContent( const char* src ); + + /** Recursively parse elements from @src and insert them into @parent + * Returns false if the data was ill-formed, and true if the parser succeeded to parse the data into @parent */ + bool ParseElement( const char* src, XMLElement* parent ); + +}; + +ORB_NAMESPACE_END diff --git a/src/Orbit/Core/IO/File/TextFile.cpp b/src/Orbit/Core/IO/File/TextFile.cpp new file mode 100644 index 0000000..4db4e1c --- /dev/null +++ b/src/Orbit/Core/IO/File/TextFile.cpp @@ -0,0 +1,142 @@ +/* + * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ + * + * This software is provided 'as-is', without any express or implied warranty. In no event will + * the authors be held liable for any damages arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, including commercial + * applications, and to alter it and redistribute it freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the + * original software. If you use this software in a product, an acknowledgment in the product + * documentation would be appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as + * being the original software. + * 3. This notice may not be removed or altered from any source distribution. + */ + +#include "TextFile.h" + +#include +#include +#include + +ORB_NAMESPACE_BEGIN + +void TextFile::SkipWhitespace( const char* src ) +{ + while( current_offset_ < total_size_ && std::isspace( src[ current_offset_ ] ) ) + ++current_offset_; +} + +bool TextFile::ExpectString( const char* src, std::string_view str ) +{ + if( ( current_offset_ + str.length() ) > total_size_ ) + return false; + + // Compare characters + if( std::strncmp( src + current_offset_, str.data(), str.length() ) == 0 ) + { + current_offset_ += str.length(); + return true; + } + + return false; +} + +std::string TextFile::ReadAlphaNumeric( const char* src ) +{ + // Find end of alphanumeric string + size_t end = current_offset_; + while( end < total_size_ && std::isalnum( src[ end ] ) ) + ++end; + + // Construct string from alphanumeric characters + const std::string string( src + current_offset_, end - current_offset_ ); + + // Move offset to end of string + current_offset_ = end; + + return string; +} + +std::string TextFile::ReadPrintable( const char* src ) +{ + // Find end of printable string + size_t end = current_offset_; + while( end < total_size_ && std::isprint( src[ end ] ) ) + ++end; + + // Construct string from printable characters + const std::string printable( src + current_offset_, end - current_offset_ ); + + // Move offset to end of string + current_offset_ = end; + + return printable; +} + +std::string TextFile::ReadCapturedStringLiteral( const char* src ) +{ + // Expect quotation mark + if( !ExpectString( src, "\"" ) ) + return std::string(); + + // Find matching quotation mark + const size_t end = FindCharacter( src, '"' ); + + // Construct captured string + const std::string string( src + current_offset_, end ); + + // Move offset to end of string + current_offset_ += end; + + // Skip last quotation mark + Skip( 1 ); + + return string; +} + +std::string_view TextFile::ReadLine( const char* src ) +{ + const size_t line_length = FindCharacter( src, '\n' ); + std::string_view line = std::string_view( src + current_offset_, line_length ); + + if( ( current_offset_ + line_length ) == total_size_ ) + { + // Set EOF + current_offset_ = total_size_; + } + else + { + if( !line.empty() && line.back() == '\r' ) + line.remove_suffix( 1 ); + + // Move offset to after line feed character + current_offset_ += ( line_length + 1 ); + } + + return line; +} + +std::string TextFile::Peek( const char* src, size_t length ) const +{ + // Clamp length so we don't read more characters than we have available + length = std::min( length, current_offset_ - total_size_ ); + + return std::string( src + current_offset_, length ); +} + +size_t TextFile::FindCharacter( const char* src, char c ) const +{ + const char* begin = src + current_offset_; + const char* end = src + total_size_; + const char* it = begin; + + // Increment offset until character is uncovered + for( ; ( it < end && *it != c ); ++it ); + + return it - begin; +} + +ORB_NAMESPACE_END diff --git a/src/Orbit/Core/IO/File/TextFile.h b/src/Orbit/Core/IO/File/TextFile.h new file mode 100644 index 0000000..ca7db82 --- /dev/null +++ b/src/Orbit/Core/IO/File/TextFile.h @@ -0,0 +1,58 @@ +/* + * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ + * + * This software is provided 'as-is', without any express or implied warranty. In no event will + * the authors be held liable for any damages arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, including commercial + * applications, and to alter it and redistribute it freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the + * original software. If you use this software in a product, an acknowledgment in the product + * documentation would be appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as + * being the original software. + * 3. This notice may not be removed or altered from any source distribution. + */ + +#pragma once +#include "Orbit/Core/IO/File/BinaryFile.h" + +ORB_NAMESPACE_BEGIN + +class ORB_API_CORE TextFile : public BinaryFile +{ +protected: + + /** Skips any character considered whitespace */ + void SkipWhitespace( const char* src ); + + /** Peeks the characters in @src and returns whether they match @str and increments the offset */ + bool ExpectString( const char* src, std::string_view str ); + + /** Reads characters from @src until it uncovers one that isn't alphanumeric (see @std::isalnum), at which point + * it returns all characters read as one string + */ + std::string ReadAlphaNumeric( const char* src ); + + /** Reads characters from @src until it uncovers one that isn't printable (see @std::isprint), at which point + * it returns all characters read as one string + */ + std::string ReadPrintable( const char* src ); + + /** Expects a leading quotation mark and reads characters until a matching trailing quotation mark is uncovered and + * returns all characters between the two quotation marks as one string + */ + std::string ReadCapturedStringLiteral( const char* src ); + + /** Reads characters from @src until it reaches a newline */ + std::string_view ReadLine( const char* src ); + + /** Reads @length amount of characters from @src without moving the offset and returns them as one string */ + std::string Peek( const char* src, size_t length ) const; + + /** Search in @src for the character @c and return the number of characters traversed */ + size_t FindCharacter( const char* src, char c ) const; +}; + +ORB_NAMESPACE_END diff --git a/src/Orbit/Core/IO/Parser/ITextParser.cpp b/src/Orbit/Core/IO/Parser/ITextParser.cpp deleted file mode 100644 index d977de1..0000000 --- a/src/Orbit/Core/IO/Parser/ITextParser.cpp +++ /dev/null @@ -1,109 +0,0 @@ -/* - * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ - * - * This software is provided 'as-is', without any express or implied warranty. In no event will - * the authors be held liable for any damages arising from the use of this software. - * - * Permission is granted to anyone to use this software for any purpose, including commercial - * applications, and to alter it and redistribute it freely, subject to the following restrictions: - * - * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the - * original software. If you use this software in a product, an acknowledgment in the product - * documentation would be appreciated but is not required. - * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as - * being the original software. - * 3. This notice may not be removed or altered from any source distribution. - */ - -#include "ITextParser.h" - -#include -#include -#include - -ORB_NAMESPACE_BEGIN - -ITextParser::ITextParser( ByteSpan data ) - : IParser( data ) -{ -} - -void ITextParser::SkipWhitespace( void ) -{ - while( offset_ < size_ && std::isspace( data_[ offset_ ] ) ) - ++offset_; -} - -std::string ITextParser::ReadAlphaNumeric( void ) -{ - size_t off = offset_; - - while( off < size_ && std::isalnum( data_[ off ] ) ) - ++off; - - const std::string alphanumeric( reinterpret_cast< const char* >( &data_[ offset_ ] ), ( off - offset_ ) ); - - offset_ = off; - - return alphanumeric; -} - -std::string ITextParser::ReadPrintable( void ) -{ - size_t off = offset_; - - while( off < size_ && std::isprint( data_[ off ] ) ) - ++off; - - const std::string printable( reinterpret_cast< const char* >( &data_[ offset_ ] ), ( off - offset_ ) ); - - offset_ = off; - - return printable; -} - -std::string ITextParser::ReadLiteral( void ) -{ - if( data_[ offset_ ] != '"' ) - return std::string(); - - size_t off = ( ++offset_ ); - - while( off < size_ && data_[ off ] != '"' ) - ++off; - - const std::string literal( reinterpret_cast< const char* >( &data_[ offset_ ] ), off - offset_ ); - - offset_ = off; - - if( offset_ < size_ ) - ++offset_; - - return literal; -} - -std::string ITextParser::Peek( size_t length ) const -{ - if( ( offset_ + length ) > size_ ) - length = ( size_ - offset_ ); - - return std::string( reinterpret_cast< const char* >( &data_[ offset_ ] ), length ); -} - -bool ITextParser::ExpectString( std::string_view str ) -{ - if( ( offset_ + str.length() ) > size_ ) - return false; - - const char* data = reinterpret_cast< const char* >( &data_[ offset_ ] ); - - if( std::strncmp( data, str.data(), str.length() ) == 0 ) - { - offset_ += str.length(); - return true; - } - - return false; -} - -ORB_NAMESPACE_END diff --git a/src/Orbit/Core/IO/Parser/XML/XMLParser.cpp b/src/Orbit/Core/IO/Parser/XML/XMLParser.cpp deleted file mode 100644 index 8cb2bc3..0000000 --- a/src/Orbit/Core/IO/Parser/XML/XMLParser.cpp +++ /dev/null @@ -1,128 +0,0 @@ -/* - * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ - * - * This software is provided 'as-is', without any express or implied warranty. In no event will - * the authors be held liable for any damages arising from the use of this software. - * - * Permission is granted to anyone to use this software for any purpose, including commercial - * applications, and to alter it and redistribute it freely, subject to the following restrictions: - * - * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the - * original software. If you use this software in a product, an acknowledgment in the product - * documentation would be appreciated but is not required. - * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as - * being the original software. - * 3. This notice may not be removed or altered from any source distribution. - */ - -#include "XMLParser.h" - -#include - -ORB_NAMESPACE_BEGIN - -XMLParser::XMLParser( ByteSpan data ) - : ITextParser( data ) -{ - if( !ExpectString( R"()" ) ) - return; - - SkipWhitespace(); - - while( !IsEOF() ) - { - if( !ParseElement( &root_element_ ) ) - return; - } - - good_ = true; -} - -std::string XMLParser::ReadName( void ) -{ - size_t off = offset_; - - while( off < size_ && ( std::isalnum( data_[ off ] ) || data_[ off ] == '_' || data_[ off ] == ':' ) ) - ++off; - - const std::string name( reinterpret_cast< const char* >( &data_[ offset_ ] ), ( off - offset_ ) ); - - offset_ = off; - - return name; -} - -std::string XMLParser::ReadContent( void ) -{ - size_t off = offset_; - - while( off < size_ && data_[ off ] != '<' && ( std::isprint( data_[ off ] ) || std::isspace( data_[ off ] ) ) ) - ++off; - - const std::string name( reinterpret_cast< const char* >( &data_[ offset_ ] ), ( off - offset_ ) ); - - offset_ = off; - - return name; -} - -bool XMLParser::ParseElement( XMLElement* parent ) -{ - XMLElement element; - - if( !ExpectString( "<" ) ) - return false; - - element.name = ReadName(); - SkipWhitespace(); - - /* Attributes */ - while( Peek( 1 ) != ">" && Peek( 2 ) != "/>" ) - { - XMLAttribute attribute; - attribute.name = ReadName(); - - if( !ExpectString( "=" ) ) - return false; - - attribute.value = ReadLiteral(); - element.attributes.push_back( attribute ); - - SkipWhitespace(); - } - - if( ExpectString( ">" ) ) - { - SkipWhitespace(); - - /* Does element have character data instead of child elements? */ - if( Peek( 1 ) != "<" ) - { - element.content = ReadContent(); - - SkipWhitespace(); - if( !ExpectString( "" ) ) - return false; - } - else - { - while( !ExpectString( "" ) ) - { - if( !ParseElement( &element ) ) - return false; - } - } - } - else if( !ExpectString( "/>" ) ) - { - return false; - } - - parent->children.push_back( element ); - - SkipWhitespace(); - - return true; -} - -ORB_NAMESPACE_END diff --git a/src/Orbit/Core/Shape/SphereShape.cpp b/src/Orbit/Core/Shape/SphereShape.cpp index e69de29..93ce9f7 100644 --- a/src/Orbit/Core/Shape/SphereShape.cpp +++ b/src/Orbit/Core/Shape/SphereShape.cpp @@ -0,0 +1,27 @@ +/* + * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ + * + * This software is provided 'as-is', without any express or implied warranty. In no event will + * the authors be held liable for any damages arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, including commercial + * applications, and to alter it and redistribute it freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the + * original software. If you use this software in a product, an acknowledgment in the product + * documentation would be appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as + * being the original software. + * 3. This notice may not be removed or altered from any source distribution. + */ + +#include "SphereShape.h" + +ORB_NAMESPACE_BEGIN + +SphereShape::SphereShape( float radius ) + : radius( radius ) +{ +} + +ORB_NAMESPACE_END diff --git a/src/Orbit/Core/Utility/Color.h b/src/Orbit/Core/Utility/Color.h deleted file mode 100644 index 75f2593..0000000 --- a/src/Orbit/Core/Utility/Color.h +++ /dev/null @@ -1,71 +0,0 @@ -/* - * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ - * - * This software is provided 'as-is', without any express or implied warranty. In no event will - * the authors be held liable for any damages arising from the use of this software. - * - * Permission is granted to anyone to use this software for any purpose, including commercial - * applications, and to alter it and redistribute it freely, subject to the following restrictions: - * - * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the - * original software. If you use this software in a product, an acknowledgment in the product - * documentation would be appreciated but is not required. - * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as - * being the original software. - * 3. This notice may not be removed or altered from any source distribution. - */ - -#pragma once -#include "Orbit/Core/Core.h" - -#include - -ORB_NAMESPACE_BEGIN - -class Color -{ -public: - - constexpr Color( void ) - : r( 0.0f ) - , g( 0.0f ) - , b( 0.0f ) - , a( 1.0f ) - { - } - - constexpr Color( float r, float g, float b, float a = 1.0f ) - : r( r ) - , g( g ) - , b( b ) - , a( a ) - { - } - -public: - - constexpr float& operator[]( size_t i ) { return ( &r )[ i ]; } - constexpr const float& operator[]( size_t i ) const { return ( &r )[ i ]; } - -public: - - static Color Random( void ) - { - Color color; - color.r = rand() / static_cast< float >( RAND_MAX ); - color.g = rand() / static_cast< float >( RAND_MAX ); - color.b = rand() / static_cast< float >( RAND_MAX ); - - return color; - } - -public: - - float r; - float g; - float b; - float a; - -}; - -ORB_NAMESPACE_END diff --git a/src/Orbit/Graphics/Animation/Animation.cpp b/src/Orbit/Graphics/Animation/Animation.cpp index bb55da2..2e354c8 100644 --- a/src/Orbit/Graphics/Animation/Animation.cpp +++ b/src/Orbit/Graphics/Animation/Animation.cpp @@ -17,7 +17,7 @@ #include "Animation.h" -#include "Orbit/Core/IO/Parser/XML/XMLParser.h" +#include "Orbit/Core/IO/File/Markup/XML/XMLFile.h" #include "Orbit/Core/IO/Log.h" #include "Orbit/Math/Matrix/Matrix4.h" @@ -27,29 +27,29 @@ ORB_NAMESPACE_BEGIN -static bool SortKeyFrames( const KeyFrame& a, const KeyFrame& b ) +Animation::Animation( JointKeyFrameMap keyframes ) + : keyframes_( std::move( keyframes ) ) { - return ( a.time < b.time ); -} - -Animation::Animation( ByteSpan data ) - : duration_{ 0.0 } -{ - if( !ParseCollada( data ) ) + // Figure out duration by finding the last keyframe + for( auto& it : keyframes_ ) { - LogErrorString( "Failed to load animation. Unrecognized format." ); + for( const KeyFrame& keyframe : it.second ) + { + if( keyframe.time > duration_ ) + duration_ = keyframe.time; + } } } Matrix4 Animation::JointPoseAtTime( std::string_view joint, float time ) const { - if( auto it = joint_key_frames_.find( std::string( joint ) ); it != joint_key_frames_.end() ) + if( auto it = keyframes_.find( std::string( joint ) ); it != keyframes_.end() ) { for( const KeyFrame& kf : it->second ) { if( kf.time >= time ) { - /* TODO: Interpolate poses */ + // TODO: Interpolate poses return kf.transform; } } @@ -61,108 +61,4 @@ Matrix4 Animation::JointPoseAtTime( std::string_view joint, float time ) const return Matrix4(); } -bool Animation::ParseCollada( ByteSpan data ) -{ - XMLParser parser( data ); - - const XMLElement& collada = parser.GetRootElement()[ "COLLADA" ]; - const XMLElement& library_animations = collada[ "library_animations" ]; - - for( const XMLElement& animation : library_animations ) - { - if( animation.name != "animation" ) - continue; - - const XMLElement& sampler = animation[ "sampler" ]; - - std::string input_source_id; - { - std::istringstream ss( std::string( sampler.ChildWithAttribute( "input", "semantic", "INPUT" ).Attribute( "source" ) ) ); - ss.ignore( 1 ); - ss >> input_source_id; - } - - std::string output_source_id; - { - std::istringstream ss( std::string( sampler.ChildWithAttribute( "input", "semantic", "OUTPUT" ).Attribute( "source" ) ) ); - ss.ignore( 1 ); - ss >> output_source_id; - } - - std::string interpolation_source_id; - { - std::istringstream ss( std::string( sampler.ChildWithAttribute( "input", "semantic", "INTERPOLATION" ).Attribute( "source" ) ) ); - ss.ignore( 1 ); - ss >> interpolation_source_id; - } - - std::string target_joint; - { - std::istringstream ss( std::string( animation[ "channel" ].Attribute( "target" ) ) ); - ss >> target_joint; - target_joint.erase( target_joint.rfind( "/" ) ); - } - - std::vector< KeyFrame > key_frames; - for( const XMLElement& source : animation ) - { - if( source.name != "source" ) - continue; - - if( key_frames.empty() ) - { - size_t count = 0; - { - std::istringstream ss( std::string( source[ "technique_common" ][ "accessor" ].Attribute( "count" ) ) ); - ss >> count; - } - - key_frames.resize( count ); - } - - const std::string source_id( source.Attribute( "id" ) ); - - if( source_id == input_source_id ) - { - std::istringstream ss( source[ "float_array" ].content ); - - for( KeyFrame& kf : key_frames ) - ss >> kf.time; - } - else if( source_id == output_source_id ) - { - std::istringstream ss( source[ "float_array" ].content ); - - for( KeyFrame& kf : key_frames ) - { - for( size_t e = 0; e < 16; ++e ) - ss >> kf.transform[ e ]; - } - } - else if( source_id == interpolation_source_id ) - { - std::istringstream ss( source[ "Name_array" ].content ); - - /* One of: LINEAR, BEZIER, CARDINAL, HERMITE, BSPLINE and STEP */ - for( KeyFrame& kf : key_frames ) - ss >> kf.interpolation_type; - } - } - - std::sort( key_frames.begin(), key_frames.end(), SortKeyFrames ); - - if( !key_frames.empty() ) - { - const KeyFrame& last_frame = key_frames.back(); - - if( last_frame.time > duration_ ) - duration_ = last_frame.time; - } - - joint_key_frames_.try_emplace( std::move( target_joint ), std::move( key_frames ) ); - } - - return true; -} - ORB_NAMESPACE_END diff --git a/src/Orbit/Graphics/Animation/Animation.h b/src/Orbit/Graphics/Animation/Animation.h index 4ec6a30..ec4e99a 100644 --- a/src/Orbit/Graphics/Animation/Animation.h +++ b/src/Orbit/Graphics/Animation/Animation.h @@ -25,11 +25,13 @@ ORB_NAMESPACE_BEGIN +using JointKeyFrameMap = std::map< std::string, std::vector< KeyFrame > >; + class ORB_API_GRAPHICS Animation { public: - explicit Animation( ByteSpan data ); + explicit Animation( JointKeyFrameMap keyframes ); public: @@ -37,17 +39,13 @@ class ORB_API_GRAPHICS Animation public: - float GetDuration( void ) const { return duration_; } - -private: - - bool ParseCollada( ByteSpan data ); + double GetDuration( void ) const { return duration_; } private: - std::map< std::string, std::vector< KeyFrame > > joint_key_frames_; + JointKeyFrameMap keyframes_; - float duration_; + double duration_ = 0.0; }; diff --git a/src/Orbit/Graphics/Animation/Joint.h b/src/Orbit/Graphics/Animation/Joint.h index 461fb93..65c0a24 100644 --- a/src/Orbit/Graphics/Animation/Joint.h +++ b/src/Orbit/Graphics/Animation/Joint.h @@ -32,7 +32,7 @@ struct Joint Matrix4 inverse_bind_transform; - int id; + int id = -1; }; ORB_NAMESPACE_END diff --git a/src/Orbit/Graphics/Animation/KeyFrame.h b/src/Orbit/Graphics/Animation/KeyFrame.h index cd04f7d..cc2b6a1 100644 --- a/src/Orbit/Graphics/Animation/KeyFrame.h +++ b/src/Orbit/Graphics/Animation/KeyFrame.h @@ -30,7 +30,7 @@ struct KeyFrame Matrix4 transform; - float time; + double time = 0.0; }; ORB_NAMESPACE_END diff --git a/src/Orbit/Graphics/Context/RenderContext.cpp b/src/Orbit/Graphics/Context/RenderContext.cpp index 0f9a104..b5e6645 100644 --- a/src/Orbit/Graphics/Context/RenderContext.cpp +++ b/src/Orbit/Graphics/Context/RenderContext.cpp @@ -865,10 +865,15 @@ void RenderContext::Clear( BufferMask mask ) auto& details = std::get< Private::_RenderContextDetailsD3D11 >( details_ ); if( !!( mask & BufferMask::Color ) ) - details.device_context->ClearRenderTargetView( details.render_target_view.ptr_, &details.clear_color[ 0 ] ); + { + const RGBA rgba( details.clear_color, 1.0f ); + details.device_context->ClearRenderTargetView( details.render_target_view.ptr_, &rgba.r ); + } if( !!( mask & BufferMask::Depth ) ) + { details.device_context->ClearDepthStencilView( details.depth_stencil_view.ptr_, D3D11_CLEAR_DEPTH, 1.0f, 0 ); + } break; } @@ -902,7 +907,6 @@ void RenderContext::SetClearColor( float r, float g, float b ) details.clear_color.r = r; details.clear_color.g = g; details.clear_color.b = b; - details.clear_color.a = 1.0f; break; } diff --git a/src/Orbit/Graphics/Debug/DebugManager.cpp b/src/Orbit/Graphics/Debug/DebugManager.cpp index a9502b4..a06ab51 100644 --- a/src/Orbit/Graphics/Debug/DebugManager.cpp +++ b/src/Orbit/Graphics/Debug/DebugManager.cpp @@ -17,20 +17,20 @@ #include "DebugManager.h" -#include "Orbit/Core/Utility/Color.h" #include "Orbit/Graphics/Geometry/MeshFactory.h" #include "Orbit/Graphics/Renderer/DefaultRenderer.h" #include "Orbit/Graphics/Renderer/RenderCommand.h" #include "Orbit/Math/Matrix/Matrix4.h" #include "Orbit/Math/Vector/Vector4.h" #include "Orbit/Core/IO/Log.h" +#include "Orbit/Core/Shape/SphereShape.h" ORB_NAMESPACE_BEGIN struct DebugVertex { Vector4 position; - Color color; + RGBA color; }; constexpr std::string_view shader_source = R"( @@ -38,6 +38,8 @@ constexpr std::string_view shader_source = R"( cbuffer VertexUniforms { + float4 u_color; + matrix u_model; matrix u_view_projection; }; @@ -56,8 +58,8 @@ struct PixelData PixelData VSMain( VertexData input ) { PixelData output; - output.position = mul( input.position, u_view_projection ); - output.color = input.color; + output.position = mul( mul( input.position, u_model ), u_view_projection ); + output.color = input.color * u_color; return output; } @@ -72,6 +74,8 @@ float4 PSMain( PixelData input ) : SV_TARGET #if defined( VERTEX ) ORB_CONSTANTS_BEGIN( VertexUniforms ) + ORB_CONSTANT( vec4, u_color ); + ORB_CONSTANT( mat4, u_model ); ORB_CONSTANT( mat4, u_view_projection ); ORB_CONSTANTS_END @@ -83,8 +87,8 @@ ORB_VARYING vec4 v_color; void main() { - v_position = u_view_projection * a_position; - v_color = a_color; + v_position = u_view_projection * u_model * a_position; + v_color = a_color * u_color; gl_Position = v_position; } @@ -111,16 +115,15 @@ static VertexLayout vertex_layout }; DebugManager::DebugManager( void ) - : shader_ ( shader_source, vertex_layout ) - , line_segments_ { } - , spheres_ { } - , lines_vertex_buffer_ ( nullptr, 0, vertex_layout.GetStride(), false ) - , spheres_vertex_buffer_( nullptr, 0, vertex_layout.GetStride(), false ) - , sphere_geometry_ ( MeshFactory::GetInstance().CreateGeometryFromShape( ShapeType::Sphere, vertex_layout, MeshFactory::DetailLevel::Low ) ) + : shader_ ( shader_source, vertex_layout ) + , line_segments_ { } + , spheres_ { } + , lines_vertex_buffer_( nullptr, 0, vertex_layout.GetStride(), false ) + , sphere_mesh_ ( MeshFactory::GetInstance().CreateMeshFromShape( SphereShape( 1.0f ), vertex_layout, MeshFactory::DetailLevel::Low ) ) { } -void DebugManager::PushLineSegment( const LineSegment& line_segment, Color color, double duration ) +void DebugManager::PushLineSegment( const LineSegment& line_segment, RGBA color, double duration ) { DebugLineSegment obj; obj.birth = Clock::now(); @@ -131,13 +134,14 @@ void DebugManager::PushLineSegment( const LineSegment& line_segment, Color color line_segments_.emplace_back( std::move( obj ) ); } -void DebugManager::PushSphere( Vector3 center, Color color, double duration ) +void DebugManager::PushSphere( Vector3 center, float radius, RGBA color, double duration ) { DebugSphere obj; - obj.birth = Clock::now(); - obj.death = obj.birth + std::chrono::duration_cast< Clock::duration >( std::chrono::duration< double >( duration ) ); - obj.color = color; - obj.position = center; + obj.birth = Clock::now(); + obj.death = obj.birth + std::chrono::duration_cast< Clock::duration >( std::chrono::duration< double >( duration ) ); + obj.color = color; + obj.center = center; + obj.radius = radius; spheres_.emplace_back( std::move( obj ) ); } @@ -156,10 +160,10 @@ void DebugManager::Render( IRenderer& renderer, const Matrix4& view_projection ) for( const DebugLineSegment& obj : line_segments_ ) { - Color color = obj.color; + RGBA color = obj.color; color.a = CalcAlphaForObject( obj, now ); - dst->position = Vector4( obj.line_segment.start, 1.0f ); + dst->position = Vector4( obj.line_segment.start, 1.0f ); dst->color = color; ++dst; @@ -181,57 +185,33 @@ void DebugManager::Render( IRenderer& renderer, const Matrix4& view_projection ) if( !spheres_.empty() ) { - spheres_vertex_buffer_.Update( nullptr, spheres_.size() * sphere_geometry_.GetFaceCount() * 6 ); - - DebugVertex* dst = static_cast< DebugVertex* >( spheres_vertex_buffer_.Map() ); - for( const DebugSphere& obj : spheres_ ) { - Color color = obj.color; - color.a = CalcAlphaForObject( obj, now ); + Matrix4 scale; + scale.Scale( Vector3( obj.radius ) ); - for( Face face : sphere_geometry_.GetFaces() ) - { - Vertex v1 = sphere_geometry_.GetVertex( face.indices[ 0 ] ); - Vertex v2 = sphere_geometry_.GetVertex( face.indices[ 1 ] ); - Vertex v3 = sphere_geometry_.GetVertex( face.indices[ 2 ] ); - - v1.position += Vector4( obj.position, 0.0f ); - v2.position += Vector4( obj.position, 0.0f ); - v3.position += Vector4( obj.position, 0.0f ); - - dst->position = v1.position; - dst->color = color; - ++dst; - dst->position = v2.position; - dst->color = color; - ++dst; - - dst->position = v1.position; - dst->color = color; - ++dst; - dst->position = v3.position; - dst->color = color; - ++dst; - - dst->position = v2.position; - dst->color = color; - ++dst; - dst->position = v3.position; - dst->color = color; - ++dst; - } - } + Matrix4 translation; + translation.Translate( obj.center ); - spheres_vertex_buffer_.Unmap(); + const Matrix4 transform = scale * translation; - // Create render command - RenderCommand render_command; - render_command.shader = shader_; - render_command.vertex_buffer = spheres_vertex_buffer_; - render_command.topology = Topology::Lines; + RGBA color = obj.color; + color.a = CalcAlphaForObject( obj, now ); + + // Create render command + RenderCommand render_command; + render_command.vertex_buffer = sphere_mesh_->GetVertexBuffer(); + render_command.index_buffer = sphere_mesh_->GetIndexBuffer(); + render_command.shader = shader_; + render_command.topology = Topology::Lines; + render_command.before_draw_callback = [ transform, color ]( const RenderCommand& command ) + { + command.shader->SetVertexUniform( "u_model", &transform, sizeof( Matrix4 ) ); + command.shader->SetVertexUniform( "u_color", &color, sizeof( RGBA ) ); + }; - renderer.PushCommand( std::move( render_command ) ); + renderer.PushCommand( std::move( render_command ) ); + } } } @@ -265,7 +245,7 @@ float DebugManager::CalcAlphaForObject( const DebugObjectBase& object, Clock::ti float time_alive = std::chrono::duration_cast< std::chrono::duration< float > >( now - object.birth ).count(); float time_left_to_live = std::chrono::duration_cast< std::chrono::duration< float > >( object.death - now ).count(); - return ( time_left_to_live / std::min( time_alive + time_left_to_live, 1.0f ) ); + return std::min( 1.0f, ( time_left_to_live / std::min( time_alive + time_left_to_live, 1.0f ) ) ); } ORB_NAMESPACE_END diff --git a/src/Orbit/Graphics/Debug/DebugManager.h b/src/Orbit/Graphics/Debug/DebugManager.h index 6b65c61..e74a64b 100644 --- a/src/Orbit/Graphics/Debug/DebugManager.h +++ b/src/Orbit/Graphics/Debug/DebugManager.h @@ -19,11 +19,14 @@ #include "Orbit/Core/Utility/Singleton.h" #include "Orbit/Graphics/Buffer/VertexBuffer.h" #include "Orbit/Graphics/Geometry/Geometry.h" +#include "Orbit/Graphics/Geometry/Mesh.h" #include "Orbit/Graphics/Shader/Shader.h" #include "Orbit/Math/Geometry/LineSegment.h" +#include "Orbit/Math/Matrix/Matrix4.h" #include "Orbit/Math/Vector/Vector3.h" #include +#include #include ORB_NAMESPACE_BEGIN @@ -41,7 +44,7 @@ class ORB_API_GRAPHICS DebugManager : public Singleton< DebugManager > { Clock::time_point birth; Clock::time_point death; - Color color; + RGBA color; }; struct DebugLineSegment : DebugObjectBase @@ -51,7 +54,8 @@ class ORB_API_GRAPHICS DebugManager : public Singleton< DebugManager > struct DebugSphere : DebugObjectBase { - Vector3 position; + Vector3 center; + float radius = 0.5f; }; using LineSegmentVector = std::vector< DebugLineSegment >; @@ -63,8 +67,8 @@ class ORB_API_GRAPHICS DebugManager : public Singleton< DebugManager > public: - void PushLineSegment( const LineSegment& line_segment, Color color, double duration = 0.0 ); - void PushSphere ( Vector3 center, Color color, double duration = 0.0 ); + void PushLineSegment( const LineSegment& line_segment, RGBA color, double duration = 0.0 ); + void PushSphere ( Vector3 center, float radius, RGBA color, double duration = 0.0 ); void Render ( IRenderer& renderer, const Matrix4& view_projection ); void Flush ( void ); @@ -74,12 +78,11 @@ class ORB_API_GRAPHICS DebugManager : public Singleton< DebugManager > private: - Shader shader_; - LineSegmentVector line_segments_; - SphereVector spheres_; - VertexBuffer lines_vertex_buffer_; - VertexBuffer spheres_vertex_buffer_; - Geometry sphere_geometry_; + Shader shader_; + LineSegmentVector line_segments_; + SphereVector spheres_; + VertexBuffer lines_vertex_buffer_; + std::shared_ptr< Mesh > sphere_mesh_; }; diff --git a/src/Orbit/Graphics/Geometry/Geometry.cpp b/src/Orbit/Graphics/Geometry/Geometry.cpp index 8837492..c218b7a 100644 --- a/src/Orbit/Graphics/Geometry/Geometry.cpp +++ b/src/Orbit/Graphics/Geometry/Geometry.cpp @@ -144,33 +144,44 @@ void Geometry::SetVertex( size_t index, const Vertex& vertex ) { uint8_t* dst = &vertex_data_[ index * vertex_layout_.GetStride() ]; - if( vertex_layout_.Contains( VertexComponent::Position ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::Position ), &vertex.position, sizeof( Vector4 ) ); - if( vertex_layout_.Contains( VertexComponent::Normal ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::Normal ), &vertex.normal, sizeof( Vector3 ) ); - if( vertex_layout_.Contains( VertexComponent::Color ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::Color ), &vertex.color, sizeof( Color ) ); - if( vertex_layout_.Contains( VertexComponent::TexCoord ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::TexCoord ), &vertex.tex_coord, sizeof( Vector2 ) ); - if( vertex_layout_.Contains( VertexComponent::JointIDs ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::JointIDs ), &vertex.joint_ids, sizeof( int ) * 4 ); - if( vertex_layout_.Contains( VertexComponent::Weights ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::Weights ), &vertex.weights, sizeof( float ) * 4 ); + if( vertex_layout_.Contains( VertexComponent::Position ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::Position ), &vertex.position, sizeof( Vector4 ) ); + if( vertex_layout_.Contains( VertexComponent::Binormal ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::Binormal ), &vertex.binormal, sizeof( Vector3 ) ); + if( vertex_layout_.Contains( VertexComponent::Tangent ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::Tangent ), &vertex.tangent, sizeof( Vector3 ) ); + if( vertex_layout_.Contains( VertexComponent::Normal ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::Normal ), &vertex.normal, sizeof( Vector3 ) ); + if( vertex_layout_.Contains( VertexComponent::Color ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::Color ), &vertex.color, sizeof( RGBA ) ); + if( vertex_layout_.Contains( VertexComponent::TexCoord ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::TexCoord ), &vertex.tex_coord, sizeof( Vector2 ) ); + if( vertex_layout_.Contains( VertexComponent::BlendIndices ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::BlendIndices ), &vertex.blend_indices, sizeof( int ) * 4 ); + if( vertex_layout_.Contains( VertexComponent::BlendWeights ) ) memcpy( dst + vertex_layout_.OffsetOf( VertexComponent::BlendWeights ), &vertex.blend_weights, sizeof( float ) * 4 ); } void Geometry::GenerateNormals( void ) { - for( Face face : GetFaces() ) + for( size_t i = 0; i < GetVertexCount(); ++i ) { - const Vertex triangle_vertices[ 3 ] - { - GetVertex( face.indices[ 0 ] ), - GetVertex( face.indices[ 1 ] ), - GetVertex( face.indices[ 2 ] ), - }; + Vertex vertex = GetVertex( i ); + vertex.normal = Vector3( vertex.position ).Normalized(); + SetVertex( i, vertex ); + } +} - const Orbit::Vector3 pos0_to_pos1 = Vector3( triangle_vertices[ 1 ].position - triangle_vertices[ 0 ].position ); - const Orbit::Vector3 pos0_to_pos2 = Vector3( triangle_vertices[ 2 ].position - triangle_vertices[ 0 ].position ); - const Orbit::Vector3 normal = ( pos0_to_pos1.CrossProduct( pos0_to_pos2 ) ).Normalized(); +void Geometry::GenerateTexCoords( void ) +{ + // Tri-planar mapping. Courtesy of https://www.martinpalko.com/triplanar-mapping/ + for( Face face : GetFaces() ) + { for( size_t i = 0; i < 3; ++i ) { - Vertex vertex = triangle_vertices[ i ]; - vertex.normal = normal; + constexpr float blend_sharpness = 1.0f; + Vertex vertex = GetVertex( face.indices[ i ] ); + Vector2 x_uv = Vector2( vertex.position.x, vertex.position.z ); + Vector2 y_uv = Vector2( vertex.position.z, vertex.position.y ); + Vector2 z_uv = Vector2( vertex.position.x, vertex.position.y ); + Vector3 blend_weights = vertex.normal.Abs().Pow( blend_sharpness ); + blend_weights = blend_weights / ( blend_weights.x + blend_weights.y + blend_weights.z ); + + // #Note: Blending the texture coordinates may look a bit strange. Consider picking the one with the largest weight instead. + vertex.tex_coord = ( x_uv * blend_weights.x ) + ( y_uv * blend_weights.y ) + ( z_uv * blend_weights.z ); SetVertex( face.indices[ i ], vertex ); } @@ -221,12 +232,14 @@ Vertex Geometry::GetVertex( size_t index ) const const uint8_t* src = &vertex_data_[ index * vertex_layout_.GetStride() ]; Vertex vertex; - if( vertex_layout_.Contains( VertexComponent::Position ) ) memcpy( &vertex.position, src + vertex_layout_.OffsetOf( VertexComponent::Position ), sizeof( Vector4 ) ); - if( vertex_layout_.Contains( VertexComponent::Normal ) ) memcpy( &vertex.normal, src + vertex_layout_.OffsetOf( VertexComponent::Normal ), sizeof( Vector3 ) ); - if( vertex_layout_.Contains( VertexComponent::Color ) ) memcpy( &vertex.color, src + vertex_layout_.OffsetOf( VertexComponent::Color ), sizeof( Color ) ); - if( vertex_layout_.Contains( VertexComponent::TexCoord ) ) memcpy( &vertex.tex_coord, src + vertex_layout_.OffsetOf( VertexComponent::TexCoord ), sizeof( Vector2 ) ); - if( vertex_layout_.Contains( VertexComponent::JointIDs ) ) memcpy( &vertex.joint_ids, src + vertex_layout_.OffsetOf( VertexComponent::JointIDs ), sizeof( int ) * 4 ); - if( vertex_layout_.Contains( VertexComponent::Weights ) ) memcpy( &vertex.weights, src + vertex_layout_.OffsetOf( VertexComponent::Weights ), sizeof( float ) * 4 ); + if( vertex_layout_.Contains( VertexComponent::Position ) ) memcpy( &vertex.position, src + vertex_layout_.OffsetOf( VertexComponent::Position ), sizeof( Vector4 ) ); + if( vertex_layout_.Contains( VertexComponent::Binormal ) ) memcpy( &vertex.normal, src + vertex_layout_.OffsetOf( VertexComponent::Binormal ), sizeof( Vector3 ) ); + if( vertex_layout_.Contains( VertexComponent::Tangent ) ) memcpy( &vertex.normal, src + vertex_layout_.OffsetOf( VertexComponent::Tangent ), sizeof( Vector3 ) ); + if( vertex_layout_.Contains( VertexComponent::Normal ) ) memcpy( &vertex.normal, src + vertex_layout_.OffsetOf( VertexComponent::Normal ), sizeof( Vector3 ) ); + if( vertex_layout_.Contains( VertexComponent::Color ) ) memcpy( &vertex.color, src + vertex_layout_.OffsetOf( VertexComponent::Color ), sizeof( RGBA ) ); + if( vertex_layout_.Contains( VertexComponent::TexCoord ) ) memcpy( &vertex.tex_coord, src + vertex_layout_.OffsetOf( VertexComponent::TexCoord ), sizeof( Vector2 ) ); + if( vertex_layout_.Contains( VertexComponent::BlendIndices ) ) memcpy( &vertex.blend_indices, src + vertex_layout_.OffsetOf( VertexComponent::BlendIndices ), sizeof( int ) * 4 ); + if( vertex_layout_.Contains( VertexComponent::BlendWeights ) ) memcpy( &vertex.blend_weights, src + vertex_layout_.OffsetOf( VertexComponent::BlendWeights ), sizeof( float ) * 4 ); return vertex; } @@ -251,21 +264,24 @@ VertexRange Geometry::GetVertices( void ) const return VertexRange( this ); } -Mesh Geometry::ToMesh( std::string_view name ) const +std::shared_ptr< Mesh > Geometry::ToMesh( std::string_view name ) const { - const size_t vertex_stride = vertex_layout_.GetStride(); - const size_t vertex_count = GetVertexCount(); - const size_t index_size = EvalIndexSize( vertex_count ); - const size_t index_count = ( face_data_.size() / index_size ); - Mesh mesh( name ); + if( vertex_data_.empty() ) + return nullptr; - mesh.vertex_layout_ = vertex_layout_; +////////////////////////////////////////////////////////////////////////// + + const size_t vertex_count = GetVertexCount(); + const size_t vertex_stride = vertex_layout_.GetStride(); + const size_t index_count = ( face_data_.size() / EvalIndexSize( vertex_count ) ); + auto mesh = std::make_shared< Mesh >( name ); - if( !vertex_data_.empty() ) - mesh.vertex_buffer_ = std::make_unique< VertexBuffer >( vertex_data_.data(), vertex_count, vertex_stride ); + mesh->vertex_layout_ = vertex_layout_; + mesh->vertex_buffer_ = std::make_unique< VertexBuffer >( vertex_data_.data(), vertex_count, vertex_stride ); + // Index buffer is optional if( !face_data_.empty() ) - mesh.index_buffer_ = std::make_unique< IndexBuffer >( GetIndexFormat(), face_data_.data(), index_count ); + mesh->index_buffer_ = std::make_unique< IndexBuffer >( GetIndexFormat(), face_data_.data(), index_count ); return mesh; } diff --git a/src/Orbit/Graphics/Geometry/Geometry.h b/src/Orbit/Graphics/Geometry/Geometry.h index 4ecf33c..d49a5d2 100644 --- a/src/Orbit/Graphics/Geometry/Geometry.h +++ b/src/Orbit/Graphics/Geometry/Geometry.h @@ -42,26 +42,27 @@ class ORB_API_GRAPHICS Geometry public: - void SetFromData ( ByteSpan vertex_data ); - void SetFromData ( ByteSpan vertex_data, ByteSpan face_data, IndexFormat index_format ); - void Reserve ( size_t vertex_count, size_t face_count ); - size_t AddFace ( const Face& face ); - size_t AddVertex ( const Vertex& vertex ); - void SetFace ( size_t index, const Face& face ); - void SetVertex ( size_t index, const Vertex& vertex ); - void GenerateNormals( void ); - void FlipFaceTowards( size_t index, const Vector3& direction ); + void SetFromData ( ByteSpan vertex_data ); + void SetFromData ( ByteSpan vertex_data, ByteSpan face_data, IndexFormat index_format ); + void Reserve ( size_t vertex_count, size_t face_count ); + size_t AddFace ( const Face& face ); + size_t AddVertex ( const Vertex& vertex ); + void SetFace ( size_t index, const Face& face ); + void SetVertex ( size_t index, const Vertex& vertex ); + void GenerateNormals ( void ); + void GenerateTexCoords ( void ); + void FlipFaceTowards ( size_t index, const Vector3& direction ); public: - VertexLayout GetVertexLayout( void ) const { return vertex_layout_; } - size_t GetVertexCount ( void ) const; - size_t GetFaceCount ( void ) const; - Vertex GetVertex ( size_t index ) const; - Face GetFace ( size_t index ) const; - FaceRange GetFaces ( void ) const; - VertexRange GetVertices ( void ) const; - Mesh ToMesh ( std::string_view name ) const; + VertexLayout GetVertexLayout( void ) const { return vertex_layout_; } + size_t GetVertexCount ( void ) const; + size_t GetFaceCount ( void ) const; + Vertex GetVertex ( size_t index ) const; + Face GetFace ( size_t index ) const; + FaceRange GetFaces ( void ) const; + VertexRange GetVertices ( void ) const; + std::shared_ptr< Mesh > ToMesh ( std::string_view name ) const; public: diff --git a/src/Orbit/Graphics/Geometry/Mesh.h b/src/Orbit/Graphics/Geometry/Mesh.h index 020435c..13b05f3 100644 --- a/src/Orbit/Graphics/Geometry/Mesh.h +++ b/src/Orbit/Graphics/Geometry/Mesh.h @@ -54,12 +54,13 @@ class ORB_API_GRAPHICS Mesh private: - VertexLayout vertex_layout_; + VertexLayout vertex_layout_; - std::string name_; + std::string name_; std::unique_ptr< VertexBuffer > vertex_buffer_; std::unique_ptr< IndexBuffer > index_buffer_; + }; ORB_NAMESPACE_END diff --git a/src/Orbit/Graphics/Geometry/MeshFactory.cpp b/src/Orbit/Graphics/Geometry/MeshFactory.cpp index d21638f..c953261 100644 --- a/src/Orbit/Graphics/Geometry/MeshFactory.cpp +++ b/src/Orbit/Graphics/Geometry/MeshFactory.cpp @@ -17,10 +17,10 @@ #include "MeshFactory.h" +#include "Orbit/Core/Color/RGBA.h" #include "Orbit/Core/Shape/CubeShape.h" #include "Orbit/Core/Shape/EquilateralTriangleShape.h" #include "Orbit/Core/Shape/SphereShape.h" -#include "Orbit/Core/Utility/Color.h" #include "Orbit/Graphics/Geometry/Geometry.h" #include "Orbit/Graphics/Geometry/Mesh.h" #include "Orbit/Graphics/Geometry/VertexLayout.h" @@ -46,49 +46,46 @@ Geometry MeshFactory::CreateGeometryFromShape( ShapeType shape_type, const Verte return geometry_data; } -Mesh MeshFactory::CreateMeshFromShape( const IShape& shape, const VertexLayout& vertex_layout, DetailLevel detail_level ) const +std::shared_ptr< Mesh > MeshFactory::CreateMeshFromShape( const IShape& shape, const VertexLayout& vertex_layout, DetailLevel detail_level ) const { Geometry geometry = CreateGeometryFromShape( shape.GetType(), vertex_layout, detail_level ); - Mesh mesh = geometry.ToMesh( EvalShapeName( shape.GetType() ) ); switch( shape.GetType() ) { case ShapeType::EquilateralTriangle: { const EquilateralTriangleShape& triangle_shape = static_cast< const EquilateralTriangleShape& >( shape ); + auto mesh = geometry.ToMesh( "EquilateralTriangleShape" ); - mesh.transform_.Scale( Vector3( triangle_shape.scale ) ); + mesh->transform_.Scale( Vector3( triangle_shape.scale ) ); - } break; + return mesh; + } case ShapeType::Cube: { const CubeShape& cube_shape = static_cast< const CubeShape& >( shape ); + auto mesh = geometry.ToMesh( "Cube" ); - mesh.transform_.Scale( Vector3( cube_shape.half_extent ) ); + mesh->transform_.Scale( Vector3( cube_shape.half_extent ) ); - } break; + return mesh; + } case ShapeType::Sphere: { const SphereShape& cube_shape = static_cast< const SphereShape& >( shape ); + auto mesh = geometry.ToMesh( "Sphere" ); - mesh.transform_.Scale( Vector3( cube_shape.radius ) ); + mesh->transform_.Scale( Vector3( cube_shape.radius ) ); - } break; - } - - return mesh; -} + return mesh; + } -std::string_view MeshFactory::EvalShapeName( ShapeType type ) const -{ - switch( type ) - { - case ShapeType::EquilateralTriangle: return "EquilateralTriangle"; - case ShapeType::Cube: return "Cube"; - case ShapeType::Sphere: return "Sphere"; - default: return "UnknownShape"; + default: + { + return nullptr; + } } } @@ -141,8 +138,8 @@ void MeshFactory::GenerateCubeData( Geometry& geometry_data ) const -1.0f + ( 2.0f * ( ( position_indices[ i ] & 2 ) != 0 ) ), -1.0f + ( 2.0f * ( ( position_indices[ i ] & 4 ) != 0 ) ), 1.0f ); - const Color color( 0.75f, 0.75f, 0.75f, 1.0f ); - Vertex vertex; + const RGBA color( 0.75f, 0.75f, 0.75f, 1.0f ); + Vertex vertex; vertex.position = pos; vertex.normal = normals[ i / 4 ]; @@ -215,19 +212,19 @@ void MeshFactory::GenerateSphereData( Geometry& geometry, DetailLevel detail_lev ////////////////////////////////////////////////////////////////////////// - // Position Normal Color TexCoord - geometry.AddVertex( { Vector4( Vector3( -1.0f, GoldenRatio, 0.0f ).Normalized(), 1.0f ), Vector3( 0.0f, 0.0f, 1.0f ), Color( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 0.0f, 1.0f ) } ); - geometry.AddVertex( { Vector4( Vector3( 1.0f, GoldenRatio, 0.0f ).Normalized(), 1.0f ), Vector3( 0.0f, 0.0f, 1.0f ), Color( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 1.0f, 1.0f ) } ); - geometry.AddVertex( { Vector4( Vector3( -1.0f, -GoldenRatio, 0.0f ).Normalized(), 1.0f ), Vector3( 0.0f, 0.0f, 1.0f ), Color( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 0.0f, 0.0f ) } ); - geometry.AddVertex( { Vector4( Vector3( 1.0f, -GoldenRatio, 0.0f ).Normalized(), 1.0f ), Vector3( 0.0f, 0.0f, 1.0f ), Color( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 1.0f, 0.0f ) } ); - geometry.AddVertex( { Vector4( Vector3( 0.0f, -1.0f, GoldenRatio ).Normalized(), 1.0f ), Vector3( 1.0f, 0.0f, 0.0f ), Color( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 0.0f, 1.0f ) } ); - geometry.AddVertex( { Vector4( Vector3( 0.0f, 1.0f, GoldenRatio ).Normalized(), 1.0f ), Vector3( 1.0f, 0.0f, 0.0f ), Color( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 1.0f, 1.0f ) } ); - geometry.AddVertex( { Vector4( Vector3( 0.0f, -1.0f, -GoldenRatio ).Normalized(), 1.0f ), Vector3( 1.0f, 0.0f, 0.0f ), Color( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 0.0f, 0.0f ) } ); - geometry.AddVertex( { Vector4( Vector3( 0.0f, 1.0f, -GoldenRatio ).Normalized(), 1.0f ), Vector3( 1.0f, 0.0f, 0.0f ), Color( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 1.0f, 0.0f ) } ); - geometry.AddVertex( { Vector4( Vector3( GoldenRatio, 0.0f, -1.0f ).Normalized(), 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), Color( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 0.0f, 1.0f ) } ); - geometry.AddVertex( { Vector4( Vector3( GoldenRatio, 0.0f, 1.0f ).Normalized(), 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), Color( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 1.0f, 1.0f ) } ); - geometry.AddVertex( { Vector4( Vector3( -GoldenRatio, 0.0f, -1.0f ).Normalized(), 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), Color( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 0.0f, 0.0f ) } ); - geometry.AddVertex( { Vector4( Vector3( -GoldenRatio, 0.0f, 1.0f ).Normalized(), 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), Color( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 1.0f, 0.0f ) } ); + // Position Binormal Tangent Normal Color TexCoord + geometry.AddVertex( { Vector4( Vector3( -1.0f, GoldenRatio, 0.0f ).Normalized(), 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), Vector3( 1.0f, 0.0f, 1.0f ), Vector3( 0.0f, 0.0f, 1.0f ), RGBA( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 0.0f, 1.0f ) } ); + geometry.AddVertex( { Vector4( Vector3( 1.0f, GoldenRatio, 0.0f ).Normalized(), 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), Vector3( 1.0f, 0.0f, 1.0f ), Vector3( 0.0f, 0.0f, 1.0f ), RGBA( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 1.0f, 1.0f ) } ); + geometry.AddVertex( { Vector4( Vector3( -1.0f, -GoldenRatio, 0.0f ).Normalized(), 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), Vector3( 1.0f, 0.0f, 1.0f ), Vector3( 0.0f, 0.0f, 1.0f ), RGBA( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 0.0f, 0.0f ) } ); + geometry.AddVertex( { Vector4( Vector3( 1.0f, -GoldenRatio, 0.0f ).Normalized(), 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), Vector3( 1.0f, 0.0f, 1.0f ), Vector3( 0.0f, 0.0f, 1.0f ), RGBA( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 1.0f, 0.0f ) } ); + geometry.AddVertex( { Vector4( Vector3( 0.0f, -1.0f, GoldenRatio ).Normalized(), 1.0f ), Vector3( 0.0f, 0.0f, 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), Vector3( 1.0f, 0.0f, 0.0f ), RGBA( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 0.0f, 1.0f ) } ); + geometry.AddVertex( { Vector4( Vector3( 0.0f, 1.0f, GoldenRatio ).Normalized(), 1.0f ), Vector3( 0.0f, 0.0f, 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), Vector3( 1.0f, 0.0f, 0.0f ), RGBA( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 1.0f, 1.0f ) } ); + geometry.AddVertex( { Vector4( Vector3( 0.0f, -1.0f, -GoldenRatio ).Normalized(), 1.0f ), Vector3( 0.0f, 0.0f, 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), Vector3( 1.0f, 0.0f, 0.0f ), RGBA( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 0.0f, 0.0f ) } ); + geometry.AddVertex( { Vector4( Vector3( 0.0f, 1.0f, -GoldenRatio ).Normalized(), 1.0f ), Vector3( 0.0f, 0.0f, 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), Vector3( 1.0f, 0.0f, 0.0f ), RGBA( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 1.0f, 0.0f ) } ); + geometry.AddVertex( { Vector4( Vector3( GoldenRatio, 0.0f, -1.0f ).Normalized(), 1.0f ), Vector3( 1.0f, 0.0f, 0.0f ), Vector3( 0.0f, 0.0f, 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), RGBA( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 0.0f, 1.0f ) } ); + geometry.AddVertex( { Vector4( Vector3( GoldenRatio, 0.0f, 1.0f ).Normalized(), 1.0f ), Vector3( 1.0f, 0.0f, 0.0f ), Vector3( 0.0f, 0.0f, 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), RGBA( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 1.0f, 1.0f ) } ); + geometry.AddVertex( { Vector4( Vector3( -GoldenRatio, 0.0f, -1.0f ).Normalized(), 1.0f ), Vector3( 1.0f, 0.0f, 0.0f ), Vector3( 0.0f, 0.0f, 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), RGBA( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 0.0f, 0.0f ) } ); + geometry.AddVertex( { Vector4( Vector3( -GoldenRatio, 0.0f, 1.0f ).Normalized(), 1.0f ), Vector3( 1.0f, 0.0f, 0.0f ), Vector3( 0.0f, 0.0f, 1.0f ), Vector3( 0.0f, 1.0f, 0.0f ), RGBA( 0.75f, 0.75f, 0.75f, 1.0f ), Vector2( 1.0f, 0.0f ) } ); ////////////////////////////////////////////////////////////////////////// @@ -299,27 +296,27 @@ void MeshFactory::GenerateEquilateralTriangleData( Geometry& geometry ) const // Bottom left corner { Vertex vertex; - vertex.position = Orbit::Vector4( -1.0f / Orbit::PythagorasConstant, -1.0f / Orbit::PythagorasConstant, 0.0f, 1.0f ); - vertex.color = Orbit::Color( 1.0f, 0.0f, 1.0f, 1.0f ); - vertex.tex_coord = Orbit::Vector2( 0.0f, 0.0f ); + vertex.position = Vector4( -1.0f / Orbit::PythagorasConstant, -1.0f / Orbit::PythagorasConstant, 0.0f, 1.0f ); + vertex.color = RGBA( 1.0f, 0.0f, 1.0f, 1.0f ); + vertex.tex_coord = Vector2( 0.0f, 0.0f ); face.indices[ 0 ] = geometry.AddVertex( vertex ); } // Top center corner { Vertex vertex; - vertex.position = Orbit::Vector4( 0.0f, 1.0f / Orbit::PythagorasConstant, 0.0f, 1.0f ); - vertex.color = Orbit::Color( 0.0f, 1.0f, 1.0f, 1.0f ); - vertex.tex_coord = Orbit::Vector2( 0.5f, 1.0f ); + vertex.position = Vector4( 0.0f, 1.0f / Orbit::PythagorasConstant, 0.0f, 1.0f ); + vertex.color = RGBA( 0.0f, 1.0f, 1.0f, 1.0f ); + vertex.tex_coord = Vector2( 0.5f, 1.0f ); face.indices[ 1 ] = geometry.AddVertex( vertex ); } // Bottom right corner { Vertex vertex; - vertex.position = Orbit::Vector4( 1.0f / Orbit::PythagorasConstant, -1.0f / Orbit::PythagorasConstant, 0.0f, 1.0f ); - vertex.color = Orbit::Color( 1.0f, 1.0f, 0.0f, 1.0f ); - vertex.tex_coord = Orbit::Vector2( 1.0f, 0.0f ); + vertex.position = Vector4( 1.0f / Orbit::PythagorasConstant, -1.0f / Orbit::PythagorasConstant, 0.0f, 1.0f ); + vertex.color = RGBA( 1.0f, 1.0f, 0.0f, 1.0f ); + vertex.tex_coord = Vector2( 1.0f, 0.0f ); face.indices[ 2 ] = geometry.AddVertex( vertex ); } diff --git a/src/Orbit/Graphics/Geometry/MeshFactory.h b/src/Orbit/Graphics/Geometry/MeshFactory.h index 774775c..b8d39f3 100644 --- a/src/Orbit/Graphics/Geometry/MeshFactory.h +++ b/src/Orbit/Graphics/Geometry/MeshFactory.h @@ -44,15 +44,14 @@ class ORB_API_GRAPHICS MeshFactory final : public Singleton< MeshFactory > public: - Geometry CreateGeometryFromShape( ShapeType shape_type, const VertexLayout& vertex_layout, DetailLevel detail_level = DetailLevel::Medium ) const; - Mesh CreateMeshFromShape ( const IShape& shape, const VertexLayout& vertex_layout, DetailLevel detail_level = DetailLevel::Medium ) const; + Geometry CreateGeometryFromShape ( ShapeType shape_type, const VertexLayout& vertex_layout, DetailLevel detail_level = DetailLevel::Medium ) const; + std::shared_ptr< Mesh > CreateMeshFromShape ( const IShape& shape, const VertexLayout& vertex_layout, DetailLevel detail_level = DetailLevel::Medium ) const; private: - std::string_view EvalShapeName ( ShapeType type ) const; - void GenerateCubeData ( Geometry& geometry ) const; - void GenerateSphereData ( Geometry& geometry, DetailLevel detail_level ) const; - void GenerateEquilateralTriangleData ( Geometry& geometry ) const; + void GenerateCubeData ( Geometry& geometry ) const; + void GenerateSphereData ( Geometry& geometry, DetailLevel detail_level ) const; + void GenerateEquilateralTriangleData ( Geometry& geometry ) const; }; diff --git a/src/Orbit/Graphics/Geometry/Model.cpp b/src/Orbit/Graphics/Geometry/Model.cpp deleted file mode 100644 index 5853525..0000000 --- a/src/Orbit/Graphics/Geometry/Model.cpp +++ /dev/null @@ -1,573 +0,0 @@ -/* - * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ - * - * This software is provided 'as-is', without any express or implied warranty. In no event will - * the authors be held liable for any damages arising from the use of this software. - * - * Permission is granted to anyone to use this software for any purpose, including commercial - * applications, and to alter it and redistribute it freely, subject to the following restrictions: - * - * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the - * original software. If you use this software in a product, an acknowledgment in the product - * documentation would be appreciated but is not required. - * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as - * being the original software. - * 3. This notice may not be removed or altered from any source distribution. - */ - -#include "Model.h" - -#include "Orbit/Core/IO/Parser/XML/XMLParser.h" -#include "Orbit/Core/IO/Log.h" -#include "Orbit/Core/Utility/Color.h" -#include "Orbit/Core/Utility/StringConverting.h" -#include "Orbit/Graphics/Buffer/IndexBuffer.h" -#include "Orbit/Graphics/Buffer/VertexBuffer.h" -#include "Orbit/Graphics/Geometry/Geometry.h" -#include "Orbit/Math/Vector/Vector2.h" -#include "Orbit/Math/Vector/Vector3.h" -#include "Orbit/Math/Vector/Vector4.h" - -#include -#include -#include -#include -#include -#include -#include - -ORB_NAMESPACE_BEGIN - -Model::Model( ByteSpan data, const VertexLayout& layout ) -{ - if( !( ParseCollada( data, layout ) || - ParseOBJ( data, layout ) ) ) - { - LogError( "Failed to load model. Unsupported format." ); - } -} - -static Joint ColladaParseNodeRecursive( const XMLElement& node, const Matrix4& parent_inverse_bind_transform, const std::vector< std::string >& all_joint_names, const std::vector< Matrix4 >& all_joint_transforms ) -{ - Joint joint; - joint.name = node.Attribute( "id" ); - - auto find_joint = [ &joint ]( const std::string& str ) { return str == joint.name; }; - - if( auto it = std::find_if( all_joint_names.begin(), all_joint_names.end(), find_joint ); it != all_joint_names.end() ) - { - joint.id = static_cast< int >( it - all_joint_names.begin() ); - joint.inverse_bind_transform = all_joint_transforms[ joint.id ]; - } - else - { - std::istringstream ss( node[ "matrix" ].content ); - Matrix4 local_bind_transform; - - joint.id = -1; - - for( size_t i = 0; i < 16; ++i ) - ss >> local_bind_transform[ i ]; - - const Matrix4 parent_bind_transform = parent_inverse_bind_transform.Inverted(); - const Matrix4 bind_transform = ( parent_bind_transform * local_bind_transform ); - - joint.inverse_bind_transform = bind_transform.Inverted(); - } - - for( const XMLElement& child : node ) - { - if( child.name != "node" ) - continue; - - joint.children.push_back( ColladaParseNodeRecursive( child, joint.inverse_bind_transform, all_joint_names, all_joint_transforms ) ); - } - - return joint; -} - -bool Model::ParseCollada( ByteSpan data, const VertexLayout& layout ) -{ - const XMLParser xml_parser( data ); - - if( !xml_parser.IsGood() ) - return false; - -////////////////////////////////////////////////////////////////////////// - - const XMLElement& collada = xml_parser.GetRootElement()[ "COLLADA" ]; - std::vector< std::string > all_joint_names; - std::vector< Matrix4 > all_joint_transforms; - std::map< std::string, size_t > mesh_id_table; - - for( const XMLElement& geometry : collada[ "library_geometries" ] ) - { - if( geometry.name != "geometry" ) - continue; - - const std::string geometry_id( geometry.Attribute( "id" ) ); - const XMLElement& polylist = geometry[ "mesh" ][ "polylist" ]; - - size_t vertex_count = 0; - /* Peek the number of vertices */ - { - std::string positions_source( geometry[ "mesh" ][ "vertices" ].ChildWithAttribute( "input", "semantic", "POSITION" ).Attribute( "source" ) ); - positions_source.erase( positions_source.begin() ); - - const XMLElement& source = geometry[ "mesh" ].ChildWithAttribute( "source", "id", positions_source ); - std::istringstream ss( std::string( source[ "float_array" ].Attribute( "count" ) ) ); - ss >> vertex_count; - assert( vertex_count % 3 == 0 ); - vertex_count /= 3; - } - - const size_t face_count = FromString< size_t >( polylist.Attribute( "count" ) ); - - /* Opt out if not COLLADA */ - if( vertex_count == 0 || face_count == 0 ) - return false; - - Geometry geometry_data( layout ); - std::vector< Vector4 > positions; - std::vector< Vector3 > normals; - std::vector< Vector2 > tex_coords; - - geometry_data.Reserve( vertex_count, face_count ); - - if( layout.Contains( VertexComponent::Position ) ) - { - std::string source_id( geometry[ "mesh" ][ "vertices" ].ChildWithAttribute( "input", "semantic", "POSITION" ).Attribute( "source" ) ); - - if( !source_id.empty() ) - { - source_id.erase( source_id.begin() ); - - const XMLElement& source = geometry[ "mesh" ].ChildWithAttribute( "source", "id", source_id ); - const size_t stride = FromString< size_t >( source[ "technique_common" ][ "accessor" ].Attribute( "stride" ) ); - std::istringstream ss( source[ "float_array" ].content ); - Vector4 vec( 0.0f, 0.0f, 0.0f, 1.0f ); - size_t i = 0; - - while( ss >> vec[ i++ ] ) - { - if( i == stride ) - { - positions.push_back( vec ); - i = 0; - } - } - } - } - - if( layout.Contains( VertexComponent::Normal ) ) - { - const size_t offset = layout.OffsetOf( VertexComponent::Normal ); - std::string source_id( polylist.ChildWithAttribute( "input", "semantic", "NORMAL" ).Attribute( "source" ) ); - - if( !source_id.empty() ) - { - source_id.erase( source_id.begin() ); - - const XMLElement& source = geometry[ "mesh" ].ChildWithAttribute( "source", "id", source_id ); - const size_t stride = FromString< size_t >( source[ "technique_common" ][ "accessor" ].Attribute( "stride" ) ); - std::istringstream ss( source[ "float_array" ].content ); - Vector3 vec( 0.0f, 0.0f, 0.0f ); - size_t i = 0; - - while( ss >> vec[ i++ ] ) - { - if( i == stride ) - { - normals.push_back( vec ); - i = 0; - } - } - } - } - - if( layout.Contains( VertexComponent::TexCoord ) ) - { - const size_t offset = layout.OffsetOf( VertexComponent::TexCoord ); - std::string source_id( polylist.ChildWithAttribute( "input", "semantic", "TEXCOORD" ).Attribute( "source" ) ); - - if( !source_id.empty() ) - { - source_id.erase( source_id.begin() ); - - const XMLElement& source = geometry[ "mesh" ].ChildWithAttribute( "source", "id", source_id ); - const size_t stride = FromString< size_t >( source[ "technique_common" ][ "accessor" ].Attribute( "stride" ) ); - std::istringstream ss( source[ "float_array" ].content ); - Vector2 vec( 0.0f, 0.0f ); - size_t i = 0; - - while( ss >> vec[ i++ ] ) - { - if( i == stride ) - { - tex_coords.push_back( vec ); - i = 0; - } - } - } - } - - /* Write to index data */ - { - const size_t input_count = polylist.CountChildren( "input" ); - - std::vector< size_t > all_indices; - { - std::istringstream ss( polylist[ "p" ].content ); - size_t index; - - while( ss >> index ) - all_indices.push_back( index ); - } - - std::vector< size_t > vcounts; - { - std::istringstream ss( polylist[ "vcount" ].content ); - size_t index; - - while( ss >> index ) - vcounts.push_back( index ); - } - - { - const XMLElement& input = polylist.ChildWithAttribute( "input", "semantic", "VERTEX" ); - size_t index = FromString< size_t >( input.Attribute( "offset" ) ); - - for( size_t f = 0; f < face_count; ++f ) - { - Face face{ }; - - assert( vcounts[ f ] == 3 ); - - for( size_t v = 0; v < vcounts[ f ]; ++v ) - { - face.indices[ v ] = all_indices[ index ]; - index += input_count; - } - - geometry_data.AddFace( face ); - } - } - - for( size_t i = 0; i < vertex_count; ++i ) - { - Vertex vertex{ }; - vertex.position = positions[ i ]; - - geometry_data.AddVertex( vertex ); - } - - if( auto& input = polylist.ChildWithAttribute( "input", "semantic", "NORMAL" ); input.IsValid() ) - { - size_t normal_index = FromString< size_t >( input.Attribute( "offset" ) ); - - for( Face face : geometry_data.GetFaces() ) - { - for( size_t index : face.indices ) - { - Vertex vertex = geometry_data.GetVertex( index ); - vertex.normal = normals[ all_indices[ normal_index ] ]; - normal_index += input_count; - } - } - } - - if( auto& input = polylist.ChildWithAttribute( "input", "semantic", "TEXCOORD" ); input.IsValid() ) - { - size_t tex_coord_index = FromString< size_t >( input.Attribute( "offset" ) ); - - for( Face face : geometry_data.GetFaces() ) - { - for( size_t index : face.indices ) - { - Vertex vertex = geometry_data.GetVertex( index ); - vertex.tex_coord = tex_coords[ all_indices[ index ] ]; - tex_coord_index += input_count; - } - } - } - } - - if( layout.Contains( VertexComponent::JointIDs ) || layout.Contains( VertexComponent::Weights ) ) - { - for( const XMLElement& controller : collada[ "library_controllers" ] ) - { - const std::string controller_id( controller.Attribute( "id" ) ); - const XMLElement& skin = controller[ "skin" ]; - std::string skin_source_id( skin.Attribute( "source" ) ); - skin_source_id.erase( skin_source_id.begin() ); - - if( skin_source_id != geometry_id ) - continue; - - const XMLElement& vertex_weights = skin[ "vertex_weights" ]; - std::string weight_source_id( vertex_weights.ChildWithAttribute( "input", "semantic", "WEIGHT" ).Attribute( "source" ) ); - std::string joints_source_id( skin[ "joints" ].ChildWithAttribute( "input", "semantic", "JOINT" ).Attribute( "source" ) ); - std::string matrices_source_id( skin[ "joints" ].ChildWithAttribute( "input", "semantic", "INV_BIND_MATRIX" ).Attribute( "source" ) ); - weight_source_id.erase( weight_source_id.begin() ); - joints_source_id.erase( joints_source_id.begin() ); - matrices_source_id.erase( matrices_source_id.begin() ); - - Matrix4 bind_shape_matrix; - { - std::istringstream ss( skin[ "bind_shape_matrix" ].content ); - - for( size_t i = 0; i < 16; ++i ) - ss >> bind_shape_matrix[ i ]; - } - - std::vector< float > weights; - for( const XMLElement& source : skin ) - { - const std::string source_id( source.Attribute( "id" ) ); - - if( source_id == weight_source_id ) - { - const XMLElement& float_array = source[ "float_array" ]; - - size_t count = 0; - { - std::istringstream ss( std::string( float_array.Attribute( "count" ) ) ); - ss >> count; - } - - std::istringstream ss( float_array.content ); - - weights.reserve( count ); - - for( size_t i = 0; i < count; ++i ) - { - float weight = 0.0f; - ss >> weight; - weights.push_back( weight ); - } - } - else if( source_id == joints_source_id ) - { - const XMLElement& name_array = source[ "Name_array" ]; - - size_t count = 0; - { - std::istringstream ss( std::string( name_array.Attribute( "count" ) ) ); - ss >> count; - } - - if( all_joint_names.size() != count ) - { - std::istringstream ss( name_array.content ); - - all_joint_names.clear(); - all_joint_names.reserve( count ); - - for( size_t i = 0; i < count; ++i ) - { - std::string joint_name; - ss >> joint_name; - all_joint_names.emplace_back( std::move( joint_name ) ); - } - } - } - else if( source_id == matrices_source_id ) - { - const XMLElement& float_array = source[ "float_array" ]; - - size_t count = 0; - { - std::istringstream ss( std::string( float_array.Attribute( "count" ) ) ); - ss >> count; - assert( count % 16 == 0 ); - count /= 16; - } - - if( all_joint_transforms.size() != count ) - { - std::istringstream ss( float_array.content ); - - all_joint_transforms.clear(); - all_joint_transforms.reserve( count ); - - for( size_t i = 0; i < count; ++i ) - { - Matrix4 joint_transform; - for( size_t e = 0; e < 16; ++e ) - ss >> joint_transform[ e ]; - - all_joint_transforms.push_back( joint_transform * bind_shape_matrix ); - } - } - } - } - - size_t vertex_weight_count = 0; - { - std::istringstream ss( std::string( vertex_weights.Attribute( "count" ) ) ); - ss >> vertex_weight_count; - } - - std::vector< size_t > vcounts; - vcounts.reserve( vertex_weight_count ); - { - std::istringstream ss( vertex_weights[ "vcount" ].content ); - - for( size_t i = 0; i < vertex_weight_count; ++i ) - { - size_t count = 0; - ss >> count; - vcounts.push_back( count ); - } - } - - std::istringstream ss( vertex_weights[ "v" ].content ); - - for( size_t i = 0; i < vcounts.size(); ++i ) - { - using WeightPair = std::pair< int, float >; - - size_t vcount = vcounts[ i ]; - - std::vector< WeightPair > weight_pairs; - for( size_t v = 0; v < vcount; ++v ) - { - int joint_index = 0; - size_t weight_index = 0; - - ss >> joint_index; - ss >> weight_index; - - weight_pairs.push_back( { joint_index, weights[ weight_index ] } ); - } - - if( vcount > 4 ) - { - std::sort( weight_pairs.begin(), weight_pairs.end(), []( const WeightPair& a, const WeightPair& b ) { return ( a.second > b.second ); } ); - - float weight_to_redistribute = 0.0f; - for( size_t v = 4; v < vcount; ++v ) - weight_to_redistribute += weight_pairs[ v ].second; - - for( size_t v = 0; v < 4; ++v ) - weight_pairs[ v ].second += ( weight_to_redistribute / 4 ); - - weight_pairs.resize( 4 ); - vcount = 4; - } - - Vertex vertex = geometry_data.GetVertex( i ); - - for( size_t v = 0; v < vcount; ++v ) - { - vertex.joint_ids[ v ] = weight_pairs[ v ].first; - vertex.weights [ v ] = weight_pairs[ v ].second; - } - - geometry_data.SetVertex( i, vertex ); - } - - break; - } - } - - geometry_data.GenerateNormals(); - -////////////////////////////////////////////////////////////////////////// - - Mesh mesh = geometry_data.ToMesh( geometry.Attribute( "name" ) ); - meshes_.emplace_back( std::move( mesh ) ); - - mesh_id_table[ "#" + geometry_id ] = meshes_.size(); - } - -////////////////////////////////////////////////////////////////////////// - - const XMLElement& visual_scene = collada[ "library_visual_scenes" ][ "visual_scene" ]; - bool has_joint_data = false; - - for( const XMLElement& node : visual_scene ) - { - if( const XMLElement& instance_geometry = node[ "instance_geometry" ]; instance_geometry.IsValid() ) - { - if( auto it = mesh_id_table.find( std::string( instance_geometry.Attribute( "url" ) ) ); it != mesh_id_table.end() ) - { - std::istringstream ss( node[ "matrix" ].content ); - - for( size_t e = 0; e < 16; ++e ) - ss >> meshes_[ it->second ].transform_[ e ]; - } - else - { - // Failed to find mesh in ID table - assert( false ); - } - } - else if( node.children.size() == 1 ) - { - // Treat the lack of an "instance_xxx" element as being a joint node - has_joint_data = true; - } - } - - if( has_joint_data ) - root_joint_ = std::make_unique< Joint >( ColladaParseNodeRecursive( visual_scene, Matrix4(), all_joint_names, all_joint_transforms ) ); - - return true; -} - -bool Model::ParseOBJ( ByteSpan data, const VertexLayout& layout ) -{ - const char* begin = reinterpret_cast< const char* >( data.begin() ); - const char* end = reinterpret_cast< const char* >( data.end() ); - const char* it = begin; - size_t vertex_count = 0; - size_t face_count = 0; - - while( it < end ) - { - /**/ if( it[ 0 ] == 'v' ) { ++vertex_count; } - else if( it[ 0 ] == 'f' ) { ++face_count; } - - /* Seek to next line */ - while( it < end && *( it++ ) != '\n' ); - } - it = begin; - - /* Opt out if not OBJ */ - if( vertex_count == 0 && face_count == 0 ) - return false; - -////////////////////////////////////////////////////////////////////////// - - Geometry geometry( layout ); - - geometry.Reserve( vertex_count, face_count ); - - /* Parse the vertex and index data */ - while( it < end ) - { - Vertex vertex; - Face face; - - if( std::sscanf( it, "v %f %f %f\n", &vertex.position[ 0 ], &vertex.position[ 1 ], &vertex.position[ 2 ] ) == 3 ) - geometry.AddVertex( vertex ); - - else if( std::sscanf( it, "f %zd %zd %zd\n", &face.indices[ 0 ], &face.indices[ 1 ], &face.indices[ 2 ] ) == 3 ) - geometry.AddFace( Face{ ( face.indices[ 0 ] - 1 ), ( face.indices[ 1 ] - 1 ), ( face.indices[ 2 ] - 1 ) } ); - - /* Seek to next line */ - while( it < end && *( it++ ) != '\n' ); - } - - geometry.GenerateNormals(); - -////////////////////////////////////////////////////////////////////////// - - Mesh mesh = geometry.ToMesh( "OBJRoot" ); - meshes_.emplace_back( std::move( mesh ) ); - - return true; -} - -ORB_NAMESPACE_END diff --git a/src/Orbit/Graphics/Geometry/Model.h b/src/Orbit/Graphics/Geometry/Model.h deleted file mode 100644 index 05eff68..0000000 --- a/src/Orbit/Graphics/Geometry/Model.h +++ /dev/null @@ -1,60 +0,0 @@ -/* - * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ - * - * This software is provided 'as-is', without any express or implied warranty. In no event will - * the authors be held liable for any damages arising from the use of this software. - * - * Permission is granted to anyone to use this software for any purpose, including commercial - * applications, and to alter it and redistribute it freely, subject to the following restrictions: - * - * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the - * original software. If you use this software in a product, an acknowledgment in the product - * documentation would be appreciated but is not required. - * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as - * being the original software. - * 3. This notice may not be removed or altered from any source distribution. - */ - -#pragma once -#include "Orbit/Core/Utility/Span.h" -#include "Orbit/Graphics/Animation/Joint.h" -#include "Orbit/Graphics/Buffer/IndexBuffer.h" -#include "Orbit/Graphics/Buffer/VertexBuffer.h" -#include "Orbit/Graphics/Geometry/Mesh.h" -#include "Orbit/Graphics/Geometry/VertexLayout.h" -#include "Orbit/Graphics/Renderer/RenderCommand.h" - -ORB_NAMESPACE_BEGIN - -class ORB_API_GRAPHICS Model -{ - ORB_DISABLE_COPY( Model ); - -public: - - explicit Model( ByteSpan data, const VertexLayout& layout ); - -public: - - bool HasJoints ( void ) const { return root_joint_ != nullptr; } - const Joint& GetRootJoint( void ) const { return *root_joint_; } - -public: - - auto begin( void ) const { return meshes_.begin(); } - auto end ( void ) const { return meshes_.end(); } - -private: - - bool ParseCollada( ByteSpan data, const VertexLayout& layout ); - bool ParseOBJ ( ByteSpan data, const VertexLayout& layout ); - -private: - - std::vector< Mesh > meshes_; - - std::unique_ptr< Joint > root_joint_; - -}; - -ORB_NAMESPACE_END diff --git a/src/Orbit/Graphics/Geometry/Vertex.h b/src/Orbit/Graphics/Geometry/Vertex.h index 1ebe162..ab36543 100644 --- a/src/Orbit/Graphics/Geometry/Vertex.h +++ b/src/Orbit/Graphics/Geometry/Vertex.h @@ -16,7 +16,7 @@ */ #pragma once -#include "Orbit/Core/Utility/Color.h" +#include "Orbit/Core/Color/RGBA.h" #include "Orbit/Graphics/Graphics.h" #include "Orbit/Math/Vector/Vector2.h" #include "Orbit/Math/Vector/Vector3.h" @@ -28,12 +28,14 @@ ORB_NAMESPACE_BEGIN struct Vertex { - Vector4 position { 0.0f, 0.0f, 0.0f, 1.0f }; - Vector3 normal { 0.0f, 0.0f, 1.0f }; - Color color { 1.0f, 1.0f, 1.0f, 1.0f }; - Vector2 tex_coord{ 0.0f, 0.0f }; - std::array< int, 4 > joint_ids{ 0, 0, 0, 0 }; - std::array< float, 4 > weights { 1.0f, 0.0f, 0.0f, 0.0f }; + Vector4 position { 0.0f, 0.0f, 0.0f, 1.0f }; + Vector3 binormal { 1.0f, 0.0f, 0.0f }; + Vector3 tangent { 0.0f, 1.0f, 0.0f }; + Vector3 normal { 0.0f, 0.0f, 1.0f }; + RGBA color { 1.0f, 1.0f, 1.0f, 1.0f }; + Vector2 tex_coord { 0.0f, 0.0f }; + std::array< int, 4 > blend_indices { 0, 0, 0, 0 }; + std::array< float, 4 > blend_weights { 1.0f, 0.0f, 0.0f, 0.0f }; }; diff --git a/src/Orbit/Graphics/Geometry/VertexLayout.cpp b/src/Orbit/Graphics/Geometry/VertexLayout.cpp index 2a850d2..6f407b6 100644 --- a/src/Orbit/Graphics/Geometry/VertexLayout.cpp +++ b/src/Orbit/Graphics/Geometry/VertexLayout.cpp @@ -30,12 +30,14 @@ static size_t DataCountOf( VertexComponent component ) { default: { assert( false ); return 0; } - case VertexComponent::Position: return 4; - case VertexComponent::Normal: return 3; - case VertexComponent::Color: return 4; - case VertexComponent::TexCoord: return 2; - case VertexComponent::JointIDs: return 4; - case VertexComponent::Weights: return 4; + case VertexComponent::Position: return 4; + case VertexComponent::Binormal: return 3; + case VertexComponent::Tangent: return 3; + case VertexComponent::Normal: return 3; + case VertexComponent::Color: return 4; + case VertexComponent::TexCoord: return 2; + case VertexComponent::BlendIndices: return 4; + case VertexComponent::BlendWeights: return 4; } } @@ -46,13 +48,15 @@ static PrimitiveDataType DataTypeOf( VertexComponent component ) default: { assert( false ); return static_cast< PrimitiveDataType >( ~0 ); } case Orbit::VertexComponent::Position: + case Orbit::VertexComponent::Binormal: + case Orbit::VertexComponent::Tangent: case Orbit::VertexComponent::Normal: case Orbit::VertexComponent::Color: case Orbit::VertexComponent::TexCoord: - case Orbit::VertexComponent::Weights: + case Orbit::VertexComponent::BlendWeights: return PrimitiveDataType::Float; - case Orbit::VertexComponent::JointIDs: + case Orbit::VertexComponent::BlendIndices: return PrimitiveDataType::Int; } } @@ -82,7 +86,7 @@ bool VertexComponentIterator::operator!=( const VertexComponentIterator& other ) /* Trying to compare iterator from another layout */ assert( layout == other.layout ); - return ( indexed_component.index != other.indexed_component.index ); + return ( indexed_component.layout_index != other.indexed_component.layout_index ); } IndexedVertexComponent VertexComponentIterator::operator*( void ) const @@ -92,10 +96,19 @@ IndexedVertexComponent VertexComponentIterator::operator*( void ) const VertexComponentIterator& VertexComponentIterator::operator++( void ) { - ++indexed_component.index; + ++indexed_component.layout_index; - if( indexed_component.index < layout->components_.size() ) - indexed_component.type = layout->components_[ indexed_component.index ]; + if( indexed_component.layout_index < layout->components_.size() ) + { + indexed_component.type = layout->components_[ indexed_component.layout_index ]; + + // Find out the semantic index + for( size_t i = 0; i < indexed_component.layout_index; ++i ) + { + if( layout->components_[ i ] == indexed_component.type ) + ++indexed_component.semantic_index; + } + } return *this; } @@ -160,7 +173,7 @@ VertexComponentIterator VertexLayout::begin( void ) const { if( !components_.empty() ) { - IndexedVertexComponent indexed_component{ components_.front(), 0 }; + IndexedVertexComponent indexed_component{ components_.front(), 0, 0 }; return { this, indexed_component }; } @@ -170,7 +183,7 @@ VertexComponentIterator VertexLayout::begin( void ) const VertexComponentIterator VertexLayout::end( void ) const { - IndexedVertexComponent indexed_component{ std::numeric_limits< VertexComponent >::max(), components_.size() }; + IndexedVertexComponent indexed_component{ std::numeric_limits< VertexComponent >::max(), components_.size(), 0 }; return { this, indexed_component }; } diff --git a/src/Orbit/Graphics/Geometry/VertexLayout.h b/src/Orbit/Graphics/Geometry/VertexLayout.h index 9f9d0ed..fc16634 100644 --- a/src/Orbit/Graphics/Geometry/VertexLayout.h +++ b/src/Orbit/Graphics/Geometry/VertexLayout.h @@ -26,10 +26,12 @@ enum class VertexComponent : uint8_t { Position, Normal, + Binormal, + Tangent, Color, TexCoord, - JointIDs, - Weights, + BlendIndices, + BlendWeights, }; struct ORB_API_GRAPHICS IndexedVertexComponent @@ -39,7 +41,8 @@ struct ORB_API_GRAPHICS IndexedVertexComponent PrimitiveDataType GetDataType ( void ) const; VertexComponent type; - size_t index; + size_t layout_index; + size_t semantic_index; }; class VertexLayout; diff --git a/src/Orbit/Graphics/ModelFormats/COLLADAFile.cpp b/src/Orbit/Graphics/ModelFormats/COLLADAFile.cpp new file mode 100644 index 0000000..9540d38 --- /dev/null +++ b/src/Orbit/Graphics/ModelFormats/COLLADAFile.cpp @@ -0,0 +1,396 @@ +/* + * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ + * + * This software is provided 'as-is', without any express or implied warranty. In no event will + * the authors be held liable for any damages arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, including commercial + * applications, and to alter it and redistribute it freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the + * original software. If you use this software in a product, an acknowledgment in the product + * documentation would be appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as + * being the original software. + * 3. This notice may not be removed or altered from any source distribution. + */ + +#include "COLLADAFile.h" + +#include "Orbit/Core/Utility/StringConverting.h" +#include "Orbit/Graphics/Animation/KeyFrame.h" +#include "Orbit/Graphics/Geometry/Geometry.h" +#include "Orbit/Math/Matrix/Matrix4.h" + +#include +#include +#include +#include + +ORB_NAMESPACE_BEGIN + +COLLADAFile::COLLADAFile( ByteSpan data, const VertexLayout& vertex_layout ) + : XMLFile( data ) +{ + Asset(); + LibraryEffects(); + LibraryImages(); + LibraryGeometries( vertex_layout ); + LibraryAnimations(); + LibraryVisualScenes(); +} + +void COLLADAFile::Asset( void ) +{ + const XMLElement& elm_asset = root_element_[ "COLLADA" ][ "asset" ]; + const std::string_view up_axis = elm_asset[ "up_axis" ].content; + + // Set up correction matrix based on which axis is upwards + /**/ if( up_axis == "X_UP" ) correction_matrix_.RotateZ( -Pi / 2 ); + else if( up_axis == "Y_UP" ) correction_matrix_.SetIdentity(); + else if( up_axis == "Z_UP" ) correction_matrix_.RotateX( -Pi / 2 ); +} + +void COLLADAFile::LibraryEffects( void ) +{ + for( const XMLElement& elm_effect : root_element_[ "COLLADA" ][ "library_effects" ] ) + { + } +} + +void COLLADAFile::LibraryImages( void ) +{ + for( const XMLElement& elm_image : root_element_[ "COLLADA" ][ "library_images" ] ) + { + } +} + +void COLLADAFile::LibraryMaterials( void ) +{ + for( const XMLElement& elm_material : root_element_[ "COLLADA" ][ "library_materials" ] ) + { + } +} + +void COLLADAFile::LibraryGeometries( const VertexLayout& vertex_layout ) +{ + for( const XMLElement& elm_geometry : root_element_[ "COLLADA" ][ "library_geometries" ] ) + { + Geometry geometry = Geometry( vertex_layout ); + const XMLElement& elm_mesh = elm_geometry[ "mesh" ]; + const XMLElement& elm_triangles = *Or( elm_mesh.FindChild( "triangles" ), elm_mesh.FindChild( "polylist" ) ); + const uint32_t num_triangles = FromString< uint32_t >( elm_triangles.FindAttribute( "count" ) ); + const uint32_t num_inputs = elm_triangles.CountChildrenWithName( "input" ); + const XMLElement* elm_input_vertex = elm_triangles.FindChildWithAttribute( "input", { "semantic", "VERTEX" } ); + const XMLElement* elm_input_normal = elm_triangles.FindChildWithAttribute( "input", { "semantic", "NORMAL" } ); + const XMLElement* elm_input_texcoord = elm_triangles.FindChildWithAttribute( "input", { "semantic", "TEXCOORD" } ); + +////////////////////////////////////////////////////////////////////////// + + // Find the largest amount of vertices required + uint32_t minimum_vertices_required = 0; + std::string_view chosen_id; + + for( const XMLElement& source : elm_mesh ) + { + if( source.name == "source" ) + { + const XMLElement& accessor = source[ "technique_common" ][ "accessor" ]; + const uint32_t count = FromString< uint32_t >( accessor.FindAttribute( "count" ) ); + + if( count > minimum_vertices_required ) + { + minimum_vertices_required = count; + chosen_id = source.FindAttribute( "id" ); + } + } + } + +////////////////////////////////////////////////////////////////////////// + + std::vector< Vector4 > positions; + + if( elm_input_vertex && vertex_layout.Contains( VertexComponent::Position ) ) + { + const std::string_view id_vertices = SourceID( *elm_input_vertex ); + const XMLElement& elm_vertices = *elm_mesh.FindChildWithAttribute( "vertices", { "id", id_vertices } ); + + if( const XMLElement* elm_input_position = elm_vertices.FindChildWithAttribute( "input", { "semantic", "POSITION" } ) ) + { + const std::string_view id_positions = SourceID( *elm_input_position ); + const XMLElement& elm_source_positions = *elm_mesh.FindChildWithAttribute( "source", { "id", id_positions } ); + const XMLElement& elm_accessor = elm_source_positions[ "technique_common" ][ "accessor" ]; + const uint32_t count = FromString< uint32_t >( elm_accessor.FindAttribute( "count" ) ); + const uint32_t stride = FromString< uint32_t >( elm_accessor.FindAttribute( "stride" ) ); + const std::string_view id_positions_array = SourceID( elm_accessor ); + const XMLElement& elm_float_array = *elm_source_positions.FindChildWithAttribute( "float_array", { "id", id_positions_array } ); + std::string_view stream = elm_float_array.content; + + assert( stride <= 4 ); // Maximum 4 components + + for( uint32_t position_index = 0; position_index < count; ++position_index ) + { + Vector4 position( 0, 0, 0, 1 ); + + for( uint32_t component_index = 0; component_index < stride; ++component_index ) + { + const size_t space = stream.find_first_of( ' ' ); + const std::string_view value_string = stream.substr( 0, space ); + const float value = FromString< float >( value_string ); + + // Store values in position + position[ component_index ] = value; + + // Chip away from the content string + stream.remove_prefix( space + 1 ); + } + + // Correct point vector + position = correction_matrix_ * position; + + // Add position to vector + positions.push_back( position ); + } + } + } + +////////////////////////////////////////////////////////////////////////// + + std::vector< Vector3 > normals; + + if( elm_input_normal && vertex_layout.Contains( VertexComponent::Normal ) ) + { + const std::string_view id_normals = SourceID( *elm_input_normal ); + const XMLElement& elm_source = *elm_mesh.FindChildWithAttribute( "source", { "id", id_normals } ); + const XMLElement& elm_accessor = elm_source[ "technique_common" ][ "accessor" ]; + const std::string_view id_normals_array = SourceID( elm_accessor ); + const uint32_t count = FromString< uint32_t >( elm_accessor.FindAttribute( "count" ) ); + const uint32_t stride = FromString< uint32_t >( elm_accessor.FindAttribute( "stride" ) ); + const XMLElement& elm_float_array = *elm_source.FindChildWithAttribute( "float_array", { "id", id_normals_array } ); + std::string_view stream = elm_float_array.content; + + assert( stride <= 3 ); // Maximum 3 components + + for( uint32_t normal_index = 0; normal_index < count; ++normal_index ) + { + Vector3 normal; + + for( uint32_t component_index = 0; component_index < stride; ++component_index ) + { + const size_t space = stream.find_first_of( ' ' ); + const std::string_view value_string = stream.substr( 0, space ); + const float value = FromString< float >( value_string ); + + // Store values in normal + normal[ component_index ] = value; + + // Chip away from the content string + stream.remove_prefix( space + 1 ); + } + + // Correct unit vector + normal = Vector3( correction_matrix_ * Vector4( normal, 0 ) ); + + // Add normal to vector + normals.emplace_back( std::move( normal ) ); + } + } + +////////////////////////////////////////////////////////////////////////// + + std::vector< Vector2 > texcoords; + + if( elm_input_texcoord && vertex_layout.Contains( VertexComponent::TexCoord ) ) + { + const std::string_view id_texcoords_0 = SourceID( *elm_input_texcoord ); + const XMLElement& elm_source = *elm_mesh.FindChildWithAttribute( "source", { "id", id_texcoords_0 } ); + const XMLElement& elm_accessor = elm_source[ "technique_common" ][ "accessor" ]; + const std::string_view id_map_0_array = SourceID( elm_accessor ); + const uint32_t count = FromString< uint32_t >( elm_accessor.FindAttribute( "count" ) ); + const uint32_t stride = FromString< uint32_t >( elm_accessor.FindAttribute( "stride" ) ); + const XMLElement& elm_float_array = *elm_source.FindChildWithAttribute( "float_array", { "id", id_map_0_array } ); + std::string_view stream = elm_float_array.content; + + assert( stride <= 2 ); // Maximum 2 components + + for( uint32_t texcoord_index = 0; texcoord_index < count; ++texcoord_index ) + { + Vector2 texcoord; + + for( uint32_t component_index = 0; component_index < stride; ++component_index ) + { + const size_t space = stream.find_first_of( ' ' ); + const std::string_view value_string = stream.substr( 0, space ); + const float value = FromString< float >( value_string ); + + // Store values in normal + texcoord[ component_index ] = value; + + // Chip away from the content string + stream.remove_prefix( space + 1 ); + } + + // Add texture coordinate to vector + texcoords.emplace_back( std::move( texcoord ) ); + } + } + +////////////////////////////////////////////////////////////////////////// + + std::string_view content_p = elm_triangles[ "p" ].content; + + while( !content_p.empty() ) + { + Vertex vertex; + + for( uint32_t input_offset = 0; input_offset < num_inputs; ++input_offset ) + { + const size_t space = content_p.find_first_of( ' ' ); + const uint32_t index = FromString< uint32_t >( content_p.substr( 0, space ) ); + const std::string current_input_offset_string = std::to_string( input_offset ); + const XMLElement& elm_input_for_current_offset = *elm_triangles.FindChildWithAttribute( "input", { "offset", current_input_offset_string } ); + const std::string_view semantic = elm_input_for_current_offset.FindAttribute( "semantic" ); + + /**/ if( semantic == "VERTEX" ) { vertex.position = positions.at( index ); } + else if( semantic == "NORMAL" ) { vertex.normal = normals .at( index ); } + else if( semantic == "TEXCOORD" ) { vertex.tex_coord = texcoords.at( index ); } + + // Chip away on the content string until it reaches the end + if( space == std::string_view::npos ) content_p.remove_prefix( content_p.length() ); + else content_p.remove_prefix( space + 1 ); + } + + // Create vertex + geometry.AddVertex( vertex ); + } + +////////////////////////////////////////////////////////////////////////// + + const std::string_view geometry_name = elm_geometry.FindAttribute( "name" ); + + // Create mesh + model_data_.meshes.emplace_back( geometry.ToMesh( geometry_name ) ); + } +} + +void COLLADAFile::LibraryControllers( void ) +{ + for( const XMLElement& elm_controller : root_element_[ "COLLADA" ][ "library_controllers" ] ) + { + } +} + +void COLLADAFile::LibraryAnimations( void ) +{ + for( const XMLElement& elm_animation : root_element_[ "COLLADA" ][ "library_animations" ] ) + { + const std::string_view id = elm_animation.FindAttribute( "id" ); + const std::string_view name = elm_animation.FindAttribute( "name" ); + const std::string target = std::string( name ) + "/matrix"; + std::map< uint32_t, KeyFrame > key_frames; + + if( const XMLElement* elm_channel = elm_animation.FindChildWithAttribute( "channel", { "target", target } ) ) + { + const std::string_view matrix_id = SourceID( *elm_channel ); + const XMLElement& elm_sampler = *elm_animation.FindChildWithAttribute( "sampler", { "id", matrix_id } ); + + // INPUT + if( const XMLElement* elm_input = elm_sampler.FindChildWithAttribute( "input", { "semantic", "INPUT" } ) ) + { + const std::string_view input_id = SourceID( *elm_input ); + const XMLElement& elm_source = *elm_animation.FindChildWithAttribute( "source", { "id", input_id } ); + const XMLElement& elm_accessor = elm_source[ "technique_common" ][ "accessor" ]; + const std::string_view input_array_id = SourceID( elm_accessor ); + const XMLElement& elm_float_array = *elm_source.FindChildWithAttribute( "float_array", { "id", input_array_id } ); + const uint32_t count = FromString< uint32_t >( elm_accessor.FindAttribute( "count" ) ); + std::string_view content = elm_float_array.content; + + for( uint32_t i = 0; i < count; ++i ) + { + const size_t space = content.find_first_of( ' ' ); + const double time = FromString< double >( content.substr( 0, space ) ); + + // Give time to key frame + key_frames[ i ].time = time; + + // Chip away on content string until it is empty + content.remove_prefix( space + 1 ); + } + } + + // OUTPUT + if( const XMLElement* elm_input = elm_sampler.FindChildWithAttribute( "input", { "semantic", "OUTPUT" } ) ) + { + const std::string_view input_id = SourceID( *elm_input ); + const XMLElement& elm_source = *elm_animation.FindChildWithAttribute( "source", { "id", input_id } ); + const XMLElement& elm_accessor = elm_source[ "technique_common" ][ "accessor" ]; + const std::string_view input_array_id = SourceID( elm_accessor ); + const XMLElement& elm_float_array = *elm_source.FindChildWithAttribute( "float_array", { "id", input_array_id } ); + const uint32_t count = FromString< uint32_t >( elm_accessor.FindAttribute( "count" ) ); + const uint32_t stride = FromString< uint32_t >( elm_accessor.FindAttribute( "stride" ) ); + std::string_view content = elm_float_array.content; + + for( uint32_t i = 0; i < count; ++i ) + { + for( uint32_t j = 0; j < stride; ++j ) + { + const size_t space = content.find_first_of( ' ' ); + const float element = FromString< float >( content.substr( 0, space ) ); + + // Apply element to transform + key_frames[ i ].transform[ j ] = element; + + // Chip away on content string until it is empty + content.remove_prefix( space + 1 ); + } + + // Correct transform + key_frames[ i ].transform *= correction_matrix_; + } + } + + // INTERPOLATION + if( const XMLElement* elm_input = elm_sampler.FindChildWithAttribute( "input", { "semantic", "INTERPOLATION" } ) ) + { + const std::string_view input_id = SourceID( *elm_input ); + const XMLElement& elm_source = *elm_animation.FindChildWithAttribute( "source", { "id", input_id } ); + const XMLElement& elm_accessor = elm_source[ "technique_common" ][ "accessor" ]; + const std::string_view input_array_id = SourceID( elm_accessor ); + const XMLElement& elm_name_array = *elm_source.FindChildWithAttribute( "Name_array", { "id", input_array_id } ); + const uint32_t count = FromString< uint32_t >( elm_accessor.FindAttribute( "count" ) ); + std::string_view content = elm_name_array.content; + + for( uint32_t i = 0; i < count; ++i ) + { + const size_t space = content.find_first_of( ' ' ); + const std::string_view interpolation_type = content.substr( 0, space ); + + // Set interpolation type of key frame + key_frames[ i ].interpolation_type = interpolation_type; + + // Chip away on content string until it is empty + content.remove_prefix( space + 1 ); + } + } + } + } +} + +void COLLADAFile::LibraryVisualScenes( void ) +{ + for( const XMLElement& elm_visual_scene : root_element_[ "COLLADA" ][ "library_visual_scenes" ] ) + { + } +} + +std::string_view COLLADAFile::SourceID( const XMLElement& element ) +{ + std::string_view source_id = element.FindAttribute( "source" ); + + // Remove leading '#' + source_id.remove_prefix( 1 ); + + return source_id; +} + +ORB_NAMESPACE_END diff --git a/src/Orbit/Graphics/ModelFormats/COLLADAFile.h b/src/Orbit/Graphics/ModelFormats/COLLADAFile.h new file mode 100644 index 0000000..b3a4d8c --- /dev/null +++ b/src/Orbit/Graphics/ModelFormats/COLLADAFile.h @@ -0,0 +1,81 @@ +/* + * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ + * + * This software is provided 'as-is', without any express or implied warranty. In no event will + * the authors be held liable for any damages arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, including commercial + * applications, and to alter it and redistribute it freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the + * original software. If you use this software in a product, an acknowledgment in the product + * documentation would be appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as + * being the original software. + * 3. This notice may not be removed or altered from any source distribution. + */ + +#pragma once +#include "Orbit/Core/IO/File/Markup/XML/XMLFile.h" +#include "Orbit/Graphics/Animation/KeyFrame.h" +#include "Orbit/Graphics/Animation/Joint.h" +#include "Orbit/Graphics/Geometry/Geometry.h" + +#include +#include +#include + +ORB_NAMESPACE_BEGIN + +class VertexLayout; +struct KeyFrame; + +class ORB_API_GRAPHICS COLLADAFile : public XMLFile +{ +public: + + struct ModelData + { + std::vector< std::shared_ptr< Mesh > > meshes; + + Joint root_joint; + }; + + struct AnimationData + { + std::map< std::string, std::vector< KeyFrame > > keyframes; + }; + +public: + + explicit COLLADAFile( ByteSpan data, const VertexLayout& vertex_layout ); + +public: + + ModelData GetModelData ( void ) const { return model_data_; } + AnimationData GetAnimationData ( void ) const { return animation_data_; } + +private: + + void Asset ( void ); + void LibraryEffects ( void ); + void LibraryImages ( void ); + void LibraryMaterials ( void ); + void LibraryGeometries ( const VertexLayout& vertex_layout ); + void LibraryControllers ( void ); + void LibraryAnimations ( void ); + void LibraryVisualScenes ( void ); + +private: + + std::string_view SourceID( const XMLElement& element ); + +private: + + ModelData model_data_; + AnimationData animation_data_; + Matrix4 correction_matrix_; + +}; + +ORB_NAMESPACE_END diff --git a/src/Orbit/Graphics/ModelFormats/WavefrontMTLFile.cpp b/src/Orbit/Graphics/ModelFormats/WavefrontMTLFile.cpp new file mode 100644 index 0000000..c4f41bf --- /dev/null +++ b/src/Orbit/Graphics/ModelFormats/WavefrontMTLFile.cpp @@ -0,0 +1,85 @@ +/* + * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ + * + * This software is provided 'as-is', without any express or implied warranty. In no event will + * the authors be held liable for any damages arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, including commercial + * applications, and to alter it and redistribute it freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the + * original software. If you use this software in a product, an acknowledgment in the product + * documentation would be appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as + * being the original software. + * 3. This notice may not be removed or altered from any source distribution. + */ + +#include "WavefrontMTLFile.h" + +#include +#include + +ORB_NAMESPACE_BEGIN + +WavefrontMTLFile::WavefrontMTLFile( ByteSpan data ) +{ + Init( data.Size() ); + + const char* src = reinterpret_cast< const char* >( data.Ptr() ); + std::optional< Material > current_material; + + while( !IsEOF() ) + { + const std::string_view line = ReadLine( src ); + + // Skip empty lines + if( line.empty() ) + continue; + + // Skip comments + if( line[ 0 ] == '#' ) + continue; + +////////////////////////////////////////////////////////////////////////// + + if( char new_material_name_buf[ 64 ]; std::sscanf( line.data(), "newtml %s", new_material_name_buf ) > 0 ) + { + if( current_material ) + materials_.push_back( *current_material ); + + current_material = std::make_optional< Material >(); + } + else if( std::sscanf( line.data(), "Ns %f", ¤t_material->specular_exponent ) > 0 ) + { + } + else if( auto& Ka = current_material->ambient_color; std::sscanf( line.data(), "Ka %f %f %f", &Ka.r, &Ka.g, &Ka.b ) > 0 ) + { + } + else if( auto& Kd = current_material->diffuse_color; std::sscanf( line.data(), "Kd %f %f %f", &Kd.r, &Kd.g, &Kd.b ) > 0 ) + { + } + else if( auto& Ks = current_material->specular_color; std::sscanf( line.data(), "Ks %f %f %f", &Ks.r, &Ks.g, &Ks.b ) > 0 ) + { + } + else if( std::sscanf( line.data(), "d %f", ¤t_material->dissolved ) > 0 ) + { + } + else if( float Tr; std::sscanf( line.data(), "Tr %f", &Tr ) > 0 ) + { + current_material->dissolved = 1.0f - Tr; + } + else if( std::sscanf( line.data(), "Ni %f", ¤t_material->optical_density ) > 0 ) + { + } + else if( int illum; std::sscanf( line.data(), "illum %d", &illum ) > 0 ) + { + current_material->illumination_model = static_cast< IlluminationModel >( illum ); + } + } + + if( current_material ) + materials_.push_back( *current_material ); +} + +ORB_NAMESPACE_END diff --git a/src/Orbit/Graphics/ModelFormats/WavefrontMTLFile.h b/src/Orbit/Graphics/ModelFormats/WavefrontMTLFile.h new file mode 100644 index 0000000..35e8e46 --- /dev/null +++ b/src/Orbit/Graphics/ModelFormats/WavefrontMTLFile.h @@ -0,0 +1,73 @@ +/* + * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ + * + * This software is provided 'as-is', without any express or implied warranty. In no event will + * the authors be held liable for any damages arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, including commercial + * applications, and to alter it and redistribute it freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the + * original software. If you use this software in a product, an acknowledgment in the product + * documentation would be appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as + * being the original software. + * 3. This notice may not be removed or altered from any source distribution. + */ + +#pragma once +#include "Orbit/Core/IO/File/TextFile.h" +#include "Orbit/Core/Color/RGB.h" +#include "Orbit/Graphics/Graphics.h" + +#include + +ORB_NAMESPACE_BEGIN + +class ORB_API_GRAPHICS WavefrontMTLFile : public TextFile +{ +public: + + enum class IlluminationModel + { + ColorOn_AmbientOff = 0, + ColorOn_AmbientOn = 1, + HighlightOn = 2, + ReflectionOn_RayTraceOn = 3, + TransparencyGlassOn_ReflectionRayTraceOn = 4, + ReflectionFresnelOn_RayTraceOn = 5, + TransparencyRefractionOn_ReflectionFresnelOff_RayTraceOn = 6, + TransparencyRefractionOn_ReflectionFresnelOn_RayTraceOn = 7, + ReflectionOn_RayTraceOff = 8, + TransparencyGlassOn_ReflectionRayTraceOff = 9, + CastsShadowsOntoInvisibleSurfaces = 10, + }; + + struct Material + { + IlluminationModel illumination_model = IlluminationModel::ColorOn_AmbientOff; + + RGB ambient_color; + RGB diffuse_color; + RGB specular_color; + + float specular_exponent = 0.0f; + float dissolved = 1.0f; + float optical_density = 1.0f; + }; + +public: + + explicit WavefrontMTLFile( ByteSpan data ); + +public: + + auto GetMaterials( void ) const { return materials_; } + +private: + + std::vector< Material > materials_; + +}; + +ORB_NAMESPACE_END diff --git a/src/Orbit/Graphics/ModelFormats/WavefrontOBJFile.cpp b/src/Orbit/Graphics/ModelFormats/WavefrontOBJFile.cpp new file mode 100644 index 0000000..43374ad --- /dev/null +++ b/src/Orbit/Graphics/ModelFormats/WavefrontOBJFile.cpp @@ -0,0 +1,222 @@ +/* + * Copyright (c) 2020 Sebastian Kylander https://gaztin.com/ + * + * This software is provided 'as-is', without any express or implied warranty. In no event will + * the authors be held liable for any damages arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, including commercial + * applications, and to alter it and redistribute it freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not claim that you wrote the + * original software. If you use this software in a product, an acknowledgment in the product + * documentation would be appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be misrepresented as + * being the original software. + * 3. This notice may not be removed or altered from any source distribution. + */ + +#include "WavefrontOBJFile.h" + +#include "Orbit/Graphics/Geometry/Geometry.h" +#include "Orbit/Graphics/ModelFormats/WavefrontMTLFile.h" + +#include +#include +#include + +ORB_NAMESPACE_BEGIN + +WavefrontOBJFile::WavefrontOBJFile( ByteSpan data, const VertexLayout& vertex_layout ) +{ + Init( data.Size() ); + + const char* src = reinterpret_cast< const char* >( data.Ptr() ); + std::optional< Geometry > current_geometry; + std::string current_mesh_name = "Unnamed Wavefront Model"; + std::vector< Vector4 > positions; + std::vector< Vector2 > tex_coords; + std::vector< Vector3 > normals; + std::vector< WavefrontMTLFile::Material > materials; + + while( !IsEOF() ) + { + const std::string_view line = ReadLine( src ); + + // Skip empty lines + if( line.empty() ) + continue; + + // Skip comments + if( line[ 0 ] == '#' ) + continue; + +////////////////////////////////////////////////////////////////////////// + + float x, y, z, w; + float u, v; + int v1, v2, v3; + int vt1, vt2, vt3; + int vn1, vn2, vn3; + + if( line.substr( 0, 7 ) == "mtllib " ) + { + const std::string_view mtllib = line.substr( 7 ); + const Asset mtl_asset = Asset( mtllib ); + const WavefrontMTLFile mtl_file = WavefrontMTLFile( { mtl_asset.GetData(), mtl_asset.GetSize() } ); + auto new_materials = mtl_file.GetMaterials(); + + // Store materials + materials.insert( materials.end(), new_materials.begin(), new_materials.end() ); + } + else if( line.substr( 0, 2 ) == "o " ) + { + const std::string_view o = line.substr( 2 ); + + if( current_geometry ) + ProduceMesh( *current_geometry, current_mesh_name, tex_coords.empty(), normals.empty() ); + + current_geometry = std::make_optional< Geometry >( vertex_layout ); + current_mesh_name = o; + + tex_coords.clear(); + normals.clear(); + } + else if( w = 1.0f; std::sscanf( line.data(), "v %f %f %f %f", &x, &y, &z, &w ) >= 3 ) + { + positions.emplace_back( x, y, z, w ); + } + else if( v = 0.0f, w = 0.0f; std::sscanf( line.data(), "vt %f %f %f", &u, &v, &w ) >= 1 ) + { + tex_coords.emplace_back( u, v ); + } + else if( std::sscanf( line.data(), "vn %f %f %f", &x, &y, &z ) == 3 ) + { + normals.emplace_back( x, y, z ); + } + else if( std::sscanf( line.data(), "f %d %d %d", &v1, &v2, &v3 ) == 3 ) + { + assert( tex_coords.empty() ); + assert( normals.empty() ); + + // Backup geometry if 'o' tag was missing + if( !current_geometry ) + current_geometry = std::make_optional< Geometry >( vertex_layout ); + + Vertex vertex1; + vertex1.position = positions.at( v1 - 1 ); + current_geometry->AddVertex( vertex1 ); + + Vertex vertex2; + vertex2.position = positions.at( v2 - 1 ); + current_geometry->AddVertex( vertex2 ); + + Vertex vertex3; + vertex3.position = positions.at( v3 - 1 ); + current_geometry->AddVertex( vertex3 ); + } + else if( std::sscanf( line.data(), "f %d/%d %d/%d %d/%d", &v1, &vt1, &v2, &vt2, &v3, &vt3 ) == 6 ) + { + assert( normals.empty() ); + + // Backup geometry if 'o' tag was missing + if( !current_geometry ) + current_geometry = std::make_optional< Geometry >( vertex_layout ); + + Vertex vertex1; + vertex1.position = positions.at( v1 - 1 ); + vertex1.tex_coord = tex_coords.at( vt1 - 1 ); + current_geometry->AddVertex( vertex1 ); + + Vertex vertex2; + vertex2.position = positions.at( v2 - 1 ); + vertex2.tex_coord = tex_coords.at( vt2 - 1 ); + current_geometry->AddVertex( vertex2 ); + + Vertex vertex3; + vertex3.position = positions.at( v3 - 1 ); + vertex3.tex_coord = tex_coords.at( vt3 - 1 ); + current_geometry->AddVertex( vertex3 ); + } + else if( std::sscanf( line.data(), "f %d/%d/%d %d/%d/%d %d/%d/%d", &v1, &vt1, &vn1, &v2, &vt2, &vn2, &v3, &vt3, &vn3 ) == 9 ) + { + // Backup geometry if 'o' tag was missing + if( !current_geometry ) + current_geometry = std::make_optional< Geometry >( vertex_layout ); + + Vertex vertex1; + vertex1.position = positions.at( v1 - 1 ); + vertex1.tex_coord = tex_coords.at( vt1 - 1 ); + vertex1.normal = normals.at( vn1 - 1 ); + current_geometry->AddVertex( vertex1 ); + + Vertex vertex2; + vertex2.position = positions.at( v2 - 1 ); + vertex2.tex_coord = tex_coords.at( vt2 - 1 ); + vertex2.normal = normals.at( vn2 - 1 ); + current_geometry->AddVertex( vertex2 ); + + Vertex vertex3; + vertex3.position = positions.at( v3 - 1 ); + vertex3.tex_coord = tex_coords.at( vt3 - 1 ); + vertex3.normal = normals.at( vn3 - 1 ); + current_geometry->AddVertex( vertex3 ); + } + else if( std::sscanf( line.data(), "f %d//%d %d//%d %d//%d", &v1, &vn1, &v2, &vn2, &v3, &vn3 ) == 6 ) + { + assert( tex_coords.empty() ); + + // Backup geometry if 'o' tag was missing + if( !current_geometry ) + current_geometry = std::make_optional< Geometry >( vertex_layout ); + + Vertex vertex1; + vertex1.position = positions.at( v1 - 1 ); + vertex1.normal = normals.at( vn1 - 1 ); + current_geometry->AddVertex( vertex1 ); + + Vertex vertex2; + vertex2.position = positions.at( v2 - 1 ); + vertex2.normal = normals.at( vn2 - 1 ); + current_geometry->AddVertex( vertex2 ); + + Vertex vertex3; + vertex3.position = positions.at( v3 - 1 ); + vertex3.normal = normals.at( vn3 - 1 ); + current_geometry->AddVertex( vertex3 ); + } + else if( line.substr( 0, 7 ) == "usemtl " ) + { + const std::string_view usemtl = line.substr( 7 ); + + // #TODO: Use material in @materials + } + else if( line == "s 1" ) + { + // #TODO: Turn on smooth shading + } + else if( line == "s off" ) + { + // #TODO: Turn off smooth shading + } + else if( line.substr( 0, 2 ) == "g " ) + { + assert( false && "Groups are not yet supported" ); + } + } + + if( current_geometry ) + ProduceMesh( *current_geometry, current_mesh_name, tex_coords.empty(), normals.empty() ); +} + +void WavefrontOBJFile::ProduceMesh( Geometry& geometry, std::string_view mesh_name, bool generate_tex_coords, bool generate_normals ) +{ + if( generate_tex_coords ) + geometry.GenerateTexCoords(); + + if( generate_normals ) + geometry.GenerateNormals(); + + meshes_.push_back( geometry.ToMesh( mesh_name ) ); +} + +ORB_NAMESPACE_END diff --git a/src/Orbit/Core/IO/Parser/XML/XMLParser.h b/src/Orbit/Graphics/ModelFormats/WavefrontOBJFile.h similarity index 69% rename from src/Orbit/Core/IO/Parser/XML/XMLParser.h rename to src/Orbit/Graphics/ModelFormats/WavefrontOBJFile.h index 7f1a3ea..b129332 100644 --- a/src/Orbit/Core/IO/Parser/XML/XMLParser.h +++ b/src/Orbit/Graphics/ModelFormats/WavefrontOBJFile.h @@ -16,33 +16,33 @@ */ #pragma once -#include "Orbit/Core/IO/Parser/XML/XMLElement.h" -#include "Orbit/Core/IO/Parser/ITextParser.h" +#include "Orbit/Core/IO/File/TextFile.h" +#include "Orbit/Graphics/Geometry/Mesh.h" +#include #include ORB_NAMESPACE_BEGIN -class ORB_API_CORE XMLParser : public ITextParser +class VertexLayout; + +class ORB_API_GRAPHICS WavefrontOBJFile : public TextFile { public: - explicit XMLParser( ByteSpan data ); - ~XMLParser( void ) = default; + WavefrontOBJFile( ByteSpan data, const VertexLayout& vertex_layout ); public: - const XMLElement& GetRootElement( void ) const { return root_element_; } + auto GetMeshes( void ) { return meshes_; } private: - std::string ReadName ( void ); - std::string ReadContent ( void ); - bool ParseElement( XMLElement* parent ); + void ProduceMesh( Geometry& geometry, std::string_view mesh_name, bool generate_tex_coords, bool generate_normals ); private: - XMLElement root_element_; + std::vector< std::shared_ptr< Mesh > > meshes_; }; diff --git a/src/Orbit/Graphics/Private/RenderContextDetails.h b/src/Orbit/Graphics/Private/RenderContextDetails.h index 6d7d799..bd1540a 100644 --- a/src/Orbit/Graphics/Private/RenderContextDetails.h +++ b/src/Orbit/Graphics/Private/RenderContextDetails.h @@ -16,9 +16,9 @@ */ #pragma once +#include "Orbit/Core/Color/RGB.h" #include "Orbit/Core/Platform/Windows/ComPtr.h" #include "Orbit/Core/Private/WindowDetails.h" -#include "Orbit/Core/Utility/Color.h" #include "Orbit/Graphics/API/OpenGL/OpenGLVersion.h" #include "Orbit/Graphics/API/OpenGL/OpenGL.h" #include "Orbit/Graphics/Renderer/BlendEquation.h" @@ -93,7 +93,7 @@ namespace Private std::map< BlendEquation, ComPtr< ID3D11BlendState > > blend_states; - Color clear_color; + RGB clear_color; }; #endif // ORB_HAS_D3D11 diff --git a/src/Orbit/Graphics/Private/ShaderDetails.h b/src/Orbit/Graphics/Private/ShaderDetails.h index 2e50688..fa7df88 100644 --- a/src/Orbit/Graphics/Private/ShaderDetails.h +++ b/src/Orbit/Graphics/Private/ShaderDetails.h @@ -56,8 +56,10 @@ namespace Private ComPtr< ID3D11InputLayout > input_layout; ComPtr< ID3D11SamplerState > sampler_state; - std::vector< ComPtr< ID3D11Buffer > > vertex_constant_buffers; - std::vector< ComPtr< ID3D11Buffer > > pixel_constant_buffers; + std::vector< ComPtr< ID3D11Buffer > > vertex_constant_cpu_buffers; + std::vector< ComPtr< ID3D11Buffer > > vertex_constant_gpu_buffers; + std::vector< ComPtr< ID3D11Buffer > > pixel_constant_cpu_buffers; + std::vector< ComPtr< ID3D11Buffer > > pixel_constant_gpu_buffers; }; #endif // ORB_HAS_D3D11 diff --git a/src/Orbit/Graphics/Renderer/BlendEquation.h b/src/Orbit/Graphics/Renderer/BlendEquation.h index d686b00..9ec8c3d 100644 --- a/src/Orbit/Graphics/Renderer/BlendEquation.h +++ b/src/Orbit/Graphics/Renderer/BlendEquation.h @@ -16,7 +16,7 @@ */ #pragma once -#include "Orbit/Core/Utility/Color.h" +#include "Orbit/Core/Color/RGBA.h" #include "Orbit/Graphics/Graphics.h" ORB_NAMESPACE_BEGIN @@ -71,7 +71,7 @@ class ORB_API_GRAPHICS BlendEquation BlendFactor dst_factor_alpha; BlendOp op_color; BlendOp op_alpha; - Color constant; + RGBA constant; }; diff --git a/src/Orbit/Graphics/Renderer/IRenderer.cpp b/src/Orbit/Graphics/Renderer/IRenderer.cpp index 074e1f9..81967eb 100644 --- a/src/Orbit/Graphics/Renderer/IRenderer.cpp +++ b/src/Orbit/Graphics/Renderer/IRenderer.cpp @@ -118,6 +118,10 @@ void IRenderer::APIDraw( const RenderCommand& command ) { auto& context_details = RenderContext::GetInstance().GetPrivateDetails(); + // Invoke the before-draw callback + if( command.before_draw_callback ) + command.before_draw_callback( command ); + switch( context_details.index() ) { default: break; @@ -157,7 +161,7 @@ void IRenderer::APIDraw( const RenderCommand& command ) it = d3d11.blend_states.try_emplace( d3d11.blend_states.end(), command.blend_equation, std::move( blend_state ) ); } - d3d11.device_context->OMSetBlendState( it->second.ptr_, &command.blend_equation.constant[ 0 ], 0xFFFFFFFF ); + d3d11.device_context->OMSetBlendState( it->second.ptr_, &command.blend_equation.constant.r, 0xFFFFFFFF ); } else { diff --git a/src/Orbit/Graphics/Renderer/RenderCommand.h b/src/Orbit/Graphics/Renderer/RenderCommand.h index 086aa85..208ddc1 100644 --- a/src/Orbit/Graphics/Renderer/RenderCommand.h +++ b/src/Orbit/Graphics/Renderer/RenderCommand.h @@ -19,6 +19,7 @@ #include "Orbit/Core/Utility/Ref.h" #include "Orbit/Graphics/Renderer/BlendEquation.h" +#include #include #include @@ -33,7 +34,8 @@ class VertexBuffer; struct ORB_API_GRAPHICS RenderCommand { - std::vector< Ref< Texture2D > > textures; + std::function< void( const RenderCommand& ) > before_draw_callback; + std::vector< Ref< Texture2D > > textures; Ref< VertexBuffer > vertex_buffer; Ref< IndexBuffer > index_buffer; diff --git a/src/Orbit/Graphics/Shader/Shader.cpp b/src/Orbit/Graphics/Shader/Shader.cpp index 8d385ed..b1f3e95 100644 --- a/src/Orbit/Graphics/Shader/Shader.cpp +++ b/src/Orbit/Graphics/Shader/Shader.cpp @@ -113,7 +113,24 @@ Shader::Shader( std::string_view source, const VertexLayout& vertex_layout ) D3D11_SHADER_BUFFER_DESC buffer_desc; shader_buffer->GetDesc( &buffer_desc ); - // Create constant buffer + // Create constant CPU buffer + { + D3D11_BUFFER_DESC desc; + desc.ByteWidth = buffer_desc.Size; + desc.Usage = D3D11_USAGE_STAGING; + desc.BindFlags = 0; + desc.CPUAccessFlags = D3D11_CPU_ACCESS_WRITE; + desc.MiscFlags = 0; + desc.StructureByteStride = 0; + + ComPtr< ID3D11Buffer > buffer; + if( SUCCEEDED( d3d11.device->CreateBuffer( &desc, nullptr, &buffer.ptr_ ) ) ) + { + details.vertex_constant_cpu_buffers.emplace_back( std::move( buffer ) ); + } + } + + // Create constant GPU buffer { D3D11_BUFFER_DESC desc; desc.ByteWidth = buffer_desc.Size; @@ -124,8 +141,10 @@ Shader::Shader( std::string_view source, const VertexLayout& vertex_layout ) desc.StructureByteStride = 0; ComPtr< ID3D11Buffer > buffer; - d3d11.device->CreateBuffer( &desc, nullptr, &buffer.ptr_ ); - details.vertex_constant_buffers.emplace_back( std::move( buffer ) ); + if( SUCCEEDED( d3d11.device->CreateBuffer( &desc, nullptr, &buffer.ptr_ ) ) ) + { + details.vertex_constant_gpu_buffers.emplace_back( std::move( buffer ) ); + } } // Iterate variables @@ -168,7 +187,24 @@ Shader::Shader( std::string_view source, const VertexLayout& vertex_layout ) D3D11_SHADER_BUFFER_DESC buffer_desc; shader_buffer->GetDesc( &buffer_desc ); - // Create constant buffer + // Create constant CPU buffer + { + D3D11_BUFFER_DESC desc; + desc.ByteWidth = buffer_desc.Size; + desc.Usage = D3D11_USAGE_STAGING; + desc.BindFlags = 0; + desc.CPUAccessFlags = D3D11_CPU_ACCESS_WRITE; + desc.MiscFlags = 0; + desc.StructureByteStride = 0; + + ComPtr< ID3D11Buffer > buffer; + if( SUCCEEDED( d3d11.device->CreateBuffer( &desc, nullptr, &buffer.ptr_ ) ) ) + { + details.pixel_constant_cpu_buffers.emplace_back( std::move( buffer ) ); + } + } + + // Create constant GPU buffer { D3D11_BUFFER_DESC desc; desc.ByteWidth = buffer_desc.Size; @@ -179,8 +215,10 @@ Shader::Shader( std::string_view source, const VertexLayout& vertex_layout ) desc.StructureByteStride = 0; ComPtr< ID3D11Buffer > buffer; - d3d11.device->CreateBuffer( &desc, nullptr, &buffer.ptr_ ); - details.pixel_constant_buffers.emplace_back( std::move( buffer ) ); + if( SUCCEEDED( d3d11.device->CreateBuffer( &desc, nullptr, &buffer.ptr_ ) ) ) + { + details.pixel_constant_gpu_buffers.emplace_back( std::move( buffer ) ); + } } // Iterate variables @@ -214,16 +252,19 @@ Shader::Shader( std::string_view source, const VertexLayout& vertex_layout ) D3D11_INPUT_ELEMENT_DESC desc { }; desc.AlignedByteOffset = descriptors.empty() ? 0 : D3D11_APPEND_ALIGNED_ELEMENT; desc.InputSlotClass = D3D11_INPUT_PER_VERTEX_DATA; + desc.SemanticIndex = component.semantic_index; switch( component.type ) { - default: { assert( false ); } break; - case VertexComponent::Position: { desc.SemanticName = "POSITION"; } break; - case VertexComponent::Normal: { desc.SemanticName = "NORMAL"; } break; - case VertexComponent::Color: { desc.SemanticName = "COLOR"; } break; - case VertexComponent::TexCoord: { desc.SemanticName = "TEXCOORD"; } break; - case VertexComponent::JointIDs: { desc.SemanticName = "JOINTIDS"; } break; - case VertexComponent::Weights: { desc.SemanticName = "WEIGHTS"; } break; + default: { assert( false ); } break; + case VertexComponent::Position: { desc.SemanticName = "POSITION"; } break; + case VertexComponent::Binormal: { desc.SemanticName = "BINORMAL"; } break; + case VertexComponent::Tangent: { desc.SemanticName = "TANGENT"; } break; + case VertexComponent::Normal: { desc.SemanticName = "NORMAL"; } break; + case VertexComponent::Color: { desc.SemanticName = "COLOR"; } break; + case VertexComponent::TexCoord: { desc.SemanticName = "TEXCOORD"; } break; + case VertexComponent::BlendIndices: { desc.SemanticName = "BLENDINDICES"; } break; + case VertexComponent::BlendWeights: { desc.SemanticName = "BLENDWEIGHT"; } break; } switch( component.GetDataType() ) @@ -536,11 +577,17 @@ void Shader::Bind( void ) if( details.sampler_state ) d3d11.device_context->PSSetSamplers( 0, 1, &details.sampler_state.ptr_ ); - for( size_t i = 0; i < details.vertex_constant_buffers.size(); ++i ) - d3d11.device_context->VSSetConstantBuffers( i, 1, &details.vertex_constant_buffers[ i ].ptr_ ); + for( size_t i = 0; i < details.vertex_constant_gpu_buffers.size(); ++i ) + { + d3d11.device_context->CopyResource( details.vertex_constant_gpu_buffers[ i ].ptr_, details.vertex_constant_cpu_buffers[ i ].ptr_ ); + d3d11.device_context->VSSetConstantBuffers( i, 1, &details.vertex_constant_gpu_buffers[ i ].ptr_ ); + } - for( size_t i = 0; i < details.pixel_constant_buffers.size(); ++i ) - d3d11.device_context->PSSetConstantBuffers( i, 1, &details.pixel_constant_buffers[ i ].ptr_ ); + for( size_t i = 0; i < details.pixel_constant_gpu_buffers.size(); ++i ) + { + d3d11.device_context->CopyResource( details.pixel_constant_gpu_buffers[ i ].ptr_, details.pixel_constant_cpu_buffers[ i ].ptr_ ); + d3d11.device_context->PSSetConstantBuffers( i, 1, &details.pixel_constant_gpu_buffers[ i ].ptr_ ); + } break; } @@ -559,18 +606,18 @@ void Shader::Bind( void ) for( IndexedVertexComponent component : details.layout ) { - glEnableVertexAttribArray( static_cast< GLuint >( component.index ) ); + glEnableVertexAttribArray( static_cast< GLuint >( component.layout_index ) ); switch( component.GetDataType() ) { case PrimitiveDataType::Float: { - glVertexAttribPointer( static_cast< GLuint >( component.index ), static_cast< GLint >( component.GetDataCount() ), OpenGLVertexAttribDataType::Float, GL_FALSE, static_cast< GLsizei >( details.layout.GetStride() ), ptr ); + glVertexAttribPointer( static_cast< GLuint >( component.layout_index ), static_cast< GLint >( component.GetDataCount() ), OpenGLVertexAttribDataType::Float, GL_FALSE, static_cast< GLsizei >( details.layout.GetStride() ), ptr ); } break; case PrimitiveDataType::Int: { - glVertexAttribIPointer( static_cast< GLuint >( component.index ), static_cast< GLint >( component.GetDataCount() ), OpenGLVertexAttribDataType::Int, static_cast< GLsizei >( details.layout.GetStride() ), ptr ); + glVertexAttribIPointer( static_cast< GLuint >( component.layout_index ), static_cast< GLint >( component.GetDataCount() ), OpenGLVertexAttribDataType::Int, static_cast< GLsizei >( details.layout.GetStride() ), ptr ); } break; } @@ -658,7 +705,7 @@ void Shader::Unbind( void ) auto& details = std::get< Private::_ShaderDetailsOpenGL >( details_ ); for( IndexedVertexComponent component : details.layout ) - glDisableVertexAttribArray( static_cast< GLuint >( component.index ) ); + glDisableVertexAttribArray( static_cast< GLuint >( component.layout_index ) ); glUseProgram( 0 ); @@ -673,11 +720,13 @@ void Shader::Unbind( void ) } } -void Shader::SetVertexUniform( std::string_view name, const void* data, size_t size ) +void Shader::SetVertexUniform( std::string_view name, const void* data, size_t size ) const { // Find uniform among registered uniforms auto uniform = std::find_if( vertex_uniforms_.begin(), vertex_uniforms_.end(), [ name ]( const Uniform& u ) { return u.name == name; } ); - assert( uniform != vertex_uniforms_.end() ); + if( uniform == vertex_uniforms_.end() ) + return; + assert( uniform->size >= size ); switch( details_.index() ) @@ -689,10 +738,10 @@ void Shader::SetVertexUniform( std::string_view name, const void* data, size_t s { auto& d3d11 = std::get< Private::_RenderContextDetailsD3D11 >( RenderContext::GetInstance().GetPrivateDetails() ); auto& details = std::get< Private::_ShaderDetailsD3D11 >( details_ ); - auto& buffer = details.vertex_constant_buffers[ uniform->buffer_index ]; + auto& buffer = details.vertex_constant_cpu_buffers[ uniform->buffer_index ]; D3D11_MAPPED_SUBRESOURCE subresource; - if( SUCCEEDED( d3d11.device_context->Map( buffer.ptr_, 0, D3D11_MAP_WRITE_NO_OVERWRITE, 0, &subresource ) ) ) + if( SUCCEEDED( d3d11.device_context->Map( buffer.ptr_, 0, D3D11_MAP_WRITE, 0, &subresource ) ) ) { assert( subresource.RowPitch >= uniform->size ); @@ -722,6 +771,7 @@ void Shader::SetVertexUniform( std::string_view name, const void* data, size_t s memcpy( dst, data, size ); glUnmapBuffer( OpenGLBufferTarget::Uniform ); + glBindBuffer( OpenGLBufferTarget::Uniform, 0 ); } else { @@ -738,11 +788,13 @@ void Shader::SetVertexUniform( std::string_view name, const void* data, size_t s } } -void Shader::SetPixelUniform( std::string_view name, const void* data, size_t size ) +void Shader::SetPixelUniform( std::string_view name, const void* data, size_t size ) const { // Find uniform among registered uniforms auto uniform = std::find_if( pixel_uniforms_.begin(), pixel_uniforms_.end(), [ name ]( const Uniform& u ) { return u.name == name; } ); - assert( uniform != pixel_uniforms_.end() ); + if( uniform == pixel_uniforms_.end() ) + return; + assert( uniform->size >= size ); switch( details_.index() ) @@ -754,10 +806,10 @@ void Shader::SetPixelUniform( std::string_view name, const void* data, size_t si { auto& d3d11 = std::get< Private::_RenderContextDetailsD3D11 >( RenderContext::GetInstance().GetPrivateDetails() ); auto& details = std::get< Private::_ShaderDetailsD3D11 >( details_ ); - auto& buffer = details.pixel_constant_buffers[ uniform->buffer_index ]; + auto& buffer = details.pixel_constant_cpu_buffers[ uniform->buffer_index ]; D3D11_MAPPED_SUBRESOURCE subresource; - if( SUCCEEDED( d3d11.device_context->Map( buffer.ptr_, 0, D3D11_MAP_WRITE_NO_OVERWRITE, 0, &subresource ) ) ) + if( SUCCEEDED( d3d11.device_context->Map( buffer.ptr_, 0, D3D11_MAP_WRITE, 0, &subresource ) ) ) { assert( subresource.RowPitch >= uniform->size ); diff --git a/src/Orbit/Graphics/Shader/Shader.h b/src/Orbit/Graphics/Shader/Shader.h index 0be1365..b7bcffd 100644 --- a/src/Orbit/Graphics/Shader/Shader.h +++ b/src/Orbit/Graphics/Shader/Shader.h @@ -37,17 +37,17 @@ class ORB_API_GRAPHICS Shader void Bind ( void ); void Unbind ( void ); - void SetVertexUniform ( std::string_view name, const void* data, size_t size ); - void SetPixelUniform ( std::string_view name, const void* data, size_t size ); + void SetVertexUniform ( std::string_view name, const void* data, size_t size ) const; + void SetPixelUniform ( std::string_view name, const void* data, size_t size ) const; template< typename T > - void SetVertexUniform( const ShaderGen::UniformBase& uniform, const T& data ) + void SetVertexUniform( const ShaderGen::UniformBase& uniform, const T& data ) const { SetVertexUniform( uniform.GetName(), &data, sizeof( T ) ); } template< typename T > - void SetPixelUniform( const ShaderGen::UniformBase& uniform, const T& data ) + void SetPixelUniform( const ShaderGen::UniformBase& uniform, const T& data ) const { SetPixelUniform( uniform.GetName(), &data, sizeof( T ) ); } diff --git a/src/Orbit/Graphics/Texture/Texture.cpp b/src/Orbit/Graphics/Texture/Texture.cpp index 597a463..2262aadd 100644 --- a/src/Orbit/Graphics/Texture/Texture.cpp +++ b/src/Orbit/Graphics/Texture/Texture.cpp @@ -17,18 +17,16 @@ #include "Texture.h" -#include "Orbit/Core/IO/Parser/TGA/TGAParser.h" +#include "Orbit/Core/IO/File/Image/TARGA/TARGAFile.h" ORB_NAMESPACE_BEGIN Texture::Texture( ByteSpan data ) { - TGAParser tga_parser( data ); - if( tga_parser.IsGood() ) - { - texture2d_.emplace( tga_parser.Width(), tga_parser.Height(), tga_parser.ImageData(), PixelFormat::RGBA ); - return; - } + // #TODO: Take generic image data as parameter so we can load any kind of image file + const TARGAFile file( data ); + + texture2d_.emplace( file.Width(), file.Height(), file.ImageData(), PixelFormat::RGBA ); } ORB_NAMESPACE_END diff --git a/src/Orbit/Math/Vector/VectorBase.h b/src/Orbit/Math/Vector/VectorBase.h index 1c69c11..653c018 100644 --- a/src/Orbit/Math/Vector/VectorBase.h +++ b/src/Orbit/Math/Vector/VectorBase.h @@ -59,6 +59,24 @@ class VectorBase return ( *this / Length() ); } + Derived Abs( void ) const + { + Derived result; + for( size_t i = 0; i < Size; ++i ) + result[ i ] = std::fabsf( ( *this )[ i ] ); + + return result; + } + + Derived Pow( float exponent ) const + { + Derived result; + for( size_t i = 0; i < Size; ++i ) + result[ i ] = std::powf( ( *this )[ i ], exponent ); + + return result; + } + bool IsZero( void ) const { return IsZero( std::numeric_limits< float >::epsilon() ); diff --git a/src/Orbit/ShaderGen/Generator/IShader.cpp b/src/Orbit/ShaderGen/Generator/IShader.cpp index 7d71197..741c291 100644 --- a/src/Orbit/ShaderGen/Generator/IShader.cpp +++ b/src/Orbit/ShaderGen/Generator/IShader.cpp @@ -63,17 +63,43 @@ namespace ShaderGen return "error_type"; } - static std::string_view VertexComponentSemanticName( VertexComponent component, ShaderType shader_type ) + static std::string_view AttributeSemanticName( VertexComponent component ) { switch( component ) { - case VertexComponent::Position: return ( ( shader_type == ShaderType::Fragment ) ? "SV_POSITION" : "POSITION" ); - case VertexComponent::Normal: return "NORMAL"; - case VertexComponent::Color: return "COLOR"; - case VertexComponent::TexCoord: return "TEXCOORD"; - case VertexComponent::JointIDs: return "JOINTIDS"; - case VertexComponent::Weights: return "WEIGHTS"; - default: return "ERROR"; + case VertexComponent::Position: return "POSITION"; + case VertexComponent::Binormal: return "BINORMAL"; + case VertexComponent::Tangent: return "TANGENT"; + case VertexComponent::Normal: return "NORMAL"; + case VertexComponent::Color: return "COLOR"; + case VertexComponent::TexCoord: return "TEXCOORD"; + case VertexComponent::BlendIndices: return "BLENDINDICES"; + case VertexComponent::BlendWeights: return "BLENDWEIGHT"; + default: return "ERROR"; + } + }; + + static std::string VaryingSemanticName( VertexComponent component, size_t resource_index ) + { + using namespace std::literals::string_literals; + + assert( resource_index <= 9 ); // Single digit + +////////////////////////////////////////////////////////////////////////// + + const char digit_char = static_cast< char >( 0x30 + resource_index ); + + switch( component ) + { + case VertexComponent::Position: return ( resource_index == 0 ? "SV_POSITION"s : ( "POSITION"s + digit_char ) ); + case VertexComponent::Binormal: return "BINORMAL"s + digit_char; + case VertexComponent::Tangent: return "TANGENT"s + digit_char; + case VertexComponent::Normal: return "NORMAL"s + digit_char; + case VertexComponent::Color: return "COLOR"s + digit_char; + case VertexComponent::TexCoord: return "TEXCOORD"s + digit_char; + case VertexComponent::BlendIndices: return "BLENDINDICES"s + digit_char; + case VertexComponent::BlendWeights: return "BLENDWEIGHT"s + digit_char; + default: return "ERROR"s + digit_char; } }; @@ -200,6 +226,20 @@ namespace ShaderGen } } + Variable IShader::Max( const Variable& lhs, const Variable& rhs ) + { + assert( lhs.GetDataType() == rhs.GetDataType() ); + + return Variable( "max( " + lhs.GetValue() + ", " + rhs.GetValue() + " )", lhs.GetDataType() ); + } + + Variable IShader::Min( const Variable& lhs, const Variable& rhs ) + { + assert( lhs.GetDataType() == rhs.GetDataType() ); + + return Variable( "min( " + lhs.GetValue() + ", " + rhs.GetValue() + " )", lhs.GetDataType() ); + } + Variable IShader::Dot( const Variable& lhs, const Variable& rhs ) { return Variable( "dot( " + lhs.GetValue() + ", " + rhs.GetValue() + " )", DataType::Float ); @@ -260,10 +300,10 @@ namespace ShaderGen for( auto it : attribute_layout_ ) { auto type_string = VertexComponentTypeString( it ); - auto semantic_name = VertexComponentSemanticName( it.type, ShaderType::Vertex ); + auto semantic_name = AttributeSemanticName( it.type ); std::ostringstream ss; - ss << "\t" << type_string << " attribute_" << it.index << " : " << semantic_name << ";\n"; + ss << "\t" << type_string << " attribute_" << it.layout_index << " : " << semantic_name << ";\n"; full_source_code.append( ss.str() ); } @@ -273,10 +313,10 @@ namespace ShaderGen for( auto it : varying_layout_ ) { auto type_string = VertexComponentTypeString( it ); - auto semantic_name = VertexComponentSemanticName( it.type, ShaderType::Fragment ); + auto semantic_name = VaryingSemanticName( it.type, it.semantic_index ); std::ostringstream ss; - ss << "\t" << type_string << " varying_" << it.index << " : " << semantic_name << ";\n"; + ss << "\t" << type_string << " varying_" << it.layout_index << " : " << semantic_name << ";\n"; full_source_code.append( ss.str() ); } @@ -399,7 +439,7 @@ namespace ShaderGen for( auto it : attribute_layout_ ) { std::ostringstream ss; - ss << "ORB_ATTRIBUTE( " << it.index << " ) " << VertexComponentTypeString( it ) << " attribute_" << it.index << ";\n"; + ss << "ORB_ATTRIBUTE( " << it.layout_index << " ) " << VertexComponentTypeString( it ) << " attribute_" << it.layout_index << ";\n"; full_source_code.append( ss.str() ); } @@ -408,7 +448,7 @@ namespace ShaderGen for( auto it : varying_layout_ ) { std::ostringstream ss; - ss << "ORB_VARYING " << VertexComponentTypeString( it ) << " varying_" << it.index << ";\n"; + ss << "ORB_VARYING " << VertexComponentTypeString( it ) << " varying_" << it.layout_index << ";\n"; full_source_code.append( ss.str() ); } @@ -483,7 +523,7 @@ namespace ShaderGen for( auto it : varying_layout_ ) { std::ostringstream ss; - ss << "ORB_VARYING " << VertexComponentTypeString( it ) << " varying_" << it.index << ";\n"; + ss << "ORB_VARYING " << VertexComponentTypeString( it ) << " varying_" << it.layout_index << ";\n"; full_source_code.append( ss.str() ); } diff --git a/src/Orbit/ShaderGen/Generator/IShader.h b/src/Orbit/ShaderGen/Generator/IShader.h index 24b2ef6..9a39ef9 100644 --- a/src/Orbit/ShaderGen/Generator/IShader.h +++ b/src/Orbit/ShaderGen/Generator/IShader.h @@ -75,6 +75,8 @@ namespace ShaderGen Variable CanonicalScreenPos( const Variable& pos ); Variable Transpose ( const Variable& matrix ); Variable Sample ( const Variable& sampler, const Variable& texcoord ); + Variable Max ( const Variable& lhs, const Variable& rhs ); + Variable Min ( const Variable& lhs, const Variable& rhs ); Variable Dot ( const Variable& lhs, const Variable& rhs ); Variable Normalize ( const Variable& vec ); Variable Cos ( const Variable& radians ); diff --git a/src/Orbit/ShaderGen/Variables/Attribute.h b/src/Orbit/ShaderGen/Variables/Attribute.h index 5259a7c..353a6c0 100644 --- a/src/Orbit/ShaderGen/Variables/Attribute.h +++ b/src/Orbit/ShaderGen/Variables/Attribute.h @@ -30,12 +30,14 @@ namespace ShaderGen { public: - using Position = AttributeHelper< VertexComponent::Position >; - using Normal = AttributeHelper< VertexComponent::Normal >; - using Color = AttributeHelper< VertexComponent::Color >; - using TexCoord = AttributeHelper< VertexComponent::TexCoord >; - using JointIDs = AttributeHelper< VertexComponent::JointIDs >; - using Weights = AttributeHelper< VertexComponent::Weights >; + using Position = AttributeHelper< VertexComponent::Position >; + using Normal = AttributeHelper< VertexComponent::Normal >; + using Binormal = AttributeHelper< VertexComponent::Binormal >; + using Tangent = AttributeHelper< VertexComponent::Tangent >; + using Color = AttributeHelper< VertexComponent::Color >; + using TexCoord = AttributeHelper< VertexComponent::TexCoord >; + using BlendIndices = AttributeHelper< VertexComponent::BlendIndices >; + using BlendWeights = AttributeHelper< VertexComponent::BlendWeights >; using Variable::operator=; public: diff --git a/src/Orbit/ShaderGen/Variables/Variable.cpp b/src/Orbit/ShaderGen/Variables/Variable.cpp index a7c77bd..4565be5 100644 --- a/src/Orbit/ShaderGen/Variables/Variable.cpp +++ b/src/Orbit/ShaderGen/Variables/Variable.cpp @@ -84,58 +84,114 @@ namespace ShaderGen } } + std::string Variable::GetValue( void ) const + { + used_ = true; + + return GetValueDerived(); + } + Variable Variable::operator*( const Variable& rhs ) const { - assert( ( data_type_ == rhs.data_type_ ) || - ( data_type_ == DataType::Mat4 && ( rhs.data_type_ == DataType::FVec4 ) ) || - ( data_type_ == DataType::FVec4 && ( rhs.data_type_ == DataType::Mat4 || rhs.data_type_ == DataType::Float ) ) || - ( data_type_ == DataType::FVec3 && ( rhs.data_type_ == DataType::Float ) ) || - ( data_type_ == DataType::FVec2 && ( rhs.data_type_ == DataType::Float ) ) || - ( data_type_ == DataType::Float && ( rhs.data_type_ == DataType::FVec4 || rhs.data_type_ == DataType::FVec3 || rhs.data_type_ == DataType::FVec2 ) ) ); + assert( ( ( data_type_ == DataType::Int ) && ( rhs.data_type_ == DataType::Int ) ) || + ( ( data_type_ == DataType::Float ) && ( rhs.data_type_ == DataType::Float ) ) || + ( ( data_type_ == DataType::Float ) && ( rhs.data_type_ == DataType::Float ) ) || + ( ( data_type_ == DataType::FVec2 ) && ( rhs.data_type_ == DataType::FVec2 ) ) || + ( ( data_type_ == DataType::FVec3 ) && ( rhs.data_type_ == DataType::FVec3 ) ) || + ( ( data_type_ == DataType::FVec4 ) && ( rhs.data_type_ == DataType::FVec4 ) ) || + ( ( data_type_ == DataType::FVec2 ) && ( rhs.data_type_ == DataType::Float ) ) || + ( ( data_type_ == DataType::FVec3 ) && ( rhs.data_type_ == DataType::Float ) ) || + ( ( data_type_ == DataType::FVec4 ) && ( rhs.data_type_ == DataType::Float ) ) || + ( ( data_type_ == DataType::IVec2 ) && ( rhs.data_type_ == DataType::IVec2 ) ) || + ( ( data_type_ == DataType::IVec3 ) && ( rhs.data_type_ == DataType::IVec3 ) ) || + ( ( data_type_ == DataType::IVec4 ) && ( rhs.data_type_ == DataType::IVec4 ) ) || + ( ( data_type_ == DataType::IVec2 ) && ( rhs.data_type_ == DataType::Int ) ) || + ( ( data_type_ == DataType::IVec3 ) && ( rhs.data_type_ == DataType::Int ) ) || + ( ( data_type_ == DataType::IVec4 ) && ( rhs.data_type_ == DataType::Int ) ) || + ( ( data_type_ == DataType::Int ) && ( rhs.data_type_ == DataType::IVec2 ) ) || + ( ( data_type_ == DataType::Int ) && ( rhs.data_type_ == DataType::IVec3 ) ) || + ( ( data_type_ == DataType::Int ) && ( rhs.data_type_ == DataType::IVec4 ) ) || + ( ( data_type_ == DataType::Float ) && ( rhs.data_type_ == DataType::FVec2 ) ) || + ( ( data_type_ == DataType::Float ) && ( rhs.data_type_ == DataType::FVec3 ) ) || + ( ( data_type_ == DataType::Float ) && ( rhs.data_type_ == DataType::FVec4 ) ) || + ( ( data_type_ == DataType::Float ) && ( rhs.data_type_ == DataType::Mat4 ) ) || + ( ( data_type_ == DataType::FVec4 ) && ( rhs.data_type_ == DataType::Mat4 ) ) || + ( ( data_type_ == DataType::Mat4 ) && ( rhs.data_type_ == DataType::FVec4 ) ) || + ( ( data_type_ == DataType::Mat4 ) && ( rhs.data_type_ == DataType::Mat4 ) ) ); DataType result_type = DataType::Unknown; switch( data_type_ ) { case DataType::Float: case DataType::Mat4: - { result_type = rhs.data_type_; - } break; + break; case DataType::FVec2: case DataType::FVec3: case DataType::FVec4: - { result_type = data_type_; - } break; + break; - default: break; + default: + break; } - if( ShaderManager::GetInstance().GetLanguage() == ShaderLanguage::HLSL && - ( data_type_ == DataType::Mat4 || rhs.data_type_ == DataType::Mat4 ) ) - { + // For HLSL, if either lhs or rhs are of matrix type, use `mul()` + if( ShaderManager::GetInstance().GetLanguage() == ShaderLanguage::HLSL && ( data_type_ == DataType::Mat4 || rhs.data_type_ == DataType::Mat4 ) ) return Variable( "mul( " + rhs.GetValue() + ", " + GetValue() + " )", result_type ); - } return Variable( "( " + GetValue() + " * " + rhs.GetValue() + " )", result_type ); } Variable Variable::operator/( const Variable& rhs ) const { - assert( ( ( data_type_ == DataType::Float ) || ( data_type_ == DataType::FVec2 ) || ( data_type_ == DataType::FVec3 ) || ( data_type_ == DataType::FVec4 ) ) && - ( rhs.data_type_ == DataType::Float ) ); + assert( ( ( data_type_ == DataType::Int ) && ( rhs.data_type_ == DataType::Int ) ) || + ( ( data_type_ == DataType::Float ) && ( rhs.data_type_ == DataType::Float ) ) || + ( ( data_type_ == DataType::FVec2 ) && ( rhs.data_type_ == DataType::FVec2 ) ) || + ( ( data_type_ == DataType::FVec3 ) && ( rhs.data_type_ == DataType::FVec3 ) ) || + ( ( data_type_ == DataType::FVec4 ) && ( rhs.data_type_ == DataType::FVec4 ) ) || + ( ( data_type_ == DataType::IVec2 ) && ( rhs.data_type_ == DataType::IVec2 ) ) || + ( ( data_type_ == DataType::IVec3 ) && ( rhs.data_type_ == DataType::IVec3 ) ) || + ( ( data_type_ == DataType::IVec4 ) && ( rhs.data_type_ == DataType::IVec4 ) ) || + ( ( data_type_ == DataType::IVec2 ) && ( rhs.data_type_ == DataType::Int ) ) || + ( ( data_type_ == DataType::IVec3 ) && ( rhs.data_type_ == DataType::Int ) ) || + ( ( data_type_ == DataType::IVec4 ) && ( rhs.data_type_ == DataType::Int ) ) || + ( ( data_type_ == DataType::FVec2 ) && ( rhs.data_type_ == DataType::Float ) ) || + ( ( data_type_ == DataType::FVec3 ) && ( rhs.data_type_ == DataType::Float ) ) || + ( ( data_type_ == DataType::FVec4 ) && ( rhs.data_type_ == DataType::Float ) ) || + ( ( data_type_ == DataType::Mat4 ) && ( rhs.data_type_ == DataType::Float ) ) ); return Variable( "( " + GetValue() + " / " + rhs.GetValue() + " )", data_type_ ); } Variable Variable::operator+( const Variable& rhs ) const { + assert( ( ( data_type_ == DataType::Int ) && ( rhs.data_type_ == DataType::Int ) ) || + ( ( data_type_ == DataType::Float ) && ( rhs.data_type_ == DataType::Float ) ) || + ( ( data_type_ == DataType::IVec2 ) && ( rhs.data_type_ == DataType::IVec2 ) ) || + ( ( data_type_ == DataType::IVec3 ) && ( rhs.data_type_ == DataType::IVec3 ) ) || + ( ( data_type_ == DataType::IVec4 ) && ( rhs.data_type_ == DataType::IVec4 ) ) || + ( ( data_type_ == DataType::FVec2 ) && ( rhs.data_type_ == DataType::FVec2 ) ) || + ( ( data_type_ == DataType::FVec3 ) && ( rhs.data_type_ == DataType::FVec3 ) ) || + ( ( data_type_ == DataType::FVec4 ) && ( rhs.data_type_ == DataType::FVec4 ) ) || + ( ( data_type_ == DataType::Mat4 ) && ( rhs.data_type_ == DataType::Mat4 ) ) ); + return Variable( "( " + GetValue() + " + " + rhs.GetValue() + " )", data_type_ ); } Variable Variable::operator-( const Variable& rhs ) const { + assert( ( ( data_type_ == DataType::Int ) && ( rhs.data_type_ == DataType::Int ) ) || + ( ( data_type_ == DataType::Float ) && ( rhs.data_type_ == DataType::Float ) ) || + ( ( data_type_ == DataType::IVec2 ) && ( rhs.data_type_ == DataType::IVec2 ) ) || + ( ( data_type_ == DataType::IVec3 ) && ( rhs.data_type_ == DataType::IVec3 ) ) || + ( ( data_type_ == DataType::IVec4 ) && ( rhs.data_type_ == DataType::IVec4 ) ) || + ( ( data_type_ == DataType::FVec2 ) && ( rhs.data_type_ == DataType::FVec2 ) ) || + ( ( data_type_ == DataType::FVec3 ) && ( rhs.data_type_ == DataType::FVec3 ) ) || + ( ( data_type_ == DataType::FVec4 ) && ( rhs.data_type_ == DataType::FVec4 ) ) || + ( ( data_type_ == DataType::Mat4 ) && ( rhs.data_type_ == DataType::Mat4 ) ) ); + return Variable( "( " + GetValue() + " - " + rhs.GetValue() + " )", data_type_ ); } @@ -144,9 +200,9 @@ namespace ShaderGen return Variable( "( -" + GetValue() + " )", data_type_ ); } - Variable Variable::operator[]( size_t index ) const + Variable Variable::operator[]( int index ) const { - return ( *this )[ Variable( static_cast< int >( index ) ) ]; + return ( *this )[ Variable( index ) ]; } SwizzlePermutations* Variable::operator->( void ) const diff --git a/src/Orbit/ShaderGen/Variables/Variable.h b/src/Orbit/ShaderGen/Variables/Variable.h index e2dfd13..e503a5d 100644 --- a/src/Orbit/ShaderGen/Variables/Variable.h +++ b/src/Orbit/ShaderGen/Variables/Variable.h @@ -41,18 +41,21 @@ namespace ShaderGen public: - /* Stores the value in a local variable. Useful when we want to manipulate swizzles within - * a variable, since `Vec2(1.0, 0.5).g *= 2.0;` is ill-behaved. */ + /** Stores the value in a local variable. Useful when we want to manipulate swizzles within a variable, since + * `Vec2(1.0, 0.5).g *= 2.0;` is ill-behaved. + */ void StoreValue( void ); + /** Fetch the value of this variable. Marks this variable as used. */ + std::string GetValue( void ) const; + public: - DataType GetDataType( void ) const { return data_type_; }; - bool IsUsed ( void ) const { return used_; } - void SetUsed ( bool used ) const { used_ = used; } - bool IsStored ( void ) const { return stored_; }; - void SetStored ( bool stored ) { stored_ = stored; } - std::string GetValue ( void ) const { used_ = true; return GetValueDerived(); } + DataType GetDataType( void ) const { return data_type_; }; + bool IsUsed ( void ) const { return used_; } + void SetUsed ( bool used ) const { used_ = used; } + bool IsStored ( void ) const { return stored_; }; + void SetStored ( bool stored ) { stored_ = stored; } public: @@ -60,8 +63,8 @@ namespace ShaderGen Variable operator/ ( const Variable& rhs ) const; Variable operator+ ( const Variable& rhs ) const; Variable operator- ( const Variable& rhs ) const; - Variable operator- ( void ) const; - Variable operator[]( size_t index ) const; + Variable operator- ( void ) const; + Variable operator[]( int index ) const; SwizzlePermutations* operator->( void ) const; diff --git a/src/Orbit/ShaderGen/Variables/Varying.cpp b/src/Orbit/ShaderGen/Variables/Varying.cpp index 491372a..1f66f1b 100644 --- a/src/Orbit/ShaderGen/Variables/Varying.cpp +++ b/src/Orbit/ShaderGen/Variables/Varying.cpp @@ -28,8 +28,9 @@ ORB_NAMESPACE_BEGIN namespace ShaderGen { - Varying::Varying( VertexComponent component ) - : Variable( ShaderManager::GetInstance().NewVarying( component ), DataTypeFromVertexComponent( component ) ) + Varying::Varying( VertexComponent component, size_t resource_index ) + : Variable ( ShaderManager::GetInstance().NewVarying( component ), DataTypeFromVertexComponent( component ) ) + , resource_index_( resource_index ) { stored_ = true; } diff --git a/src/Orbit/ShaderGen/Variables/Varying.h b/src/Orbit/ShaderGen/Variables/Varying.h index acadbf8..e294475 100644 --- a/src/Orbit/ShaderGen/Variables/Varying.h +++ b/src/Orbit/ShaderGen/Variables/Varying.h @@ -23,30 +23,38 @@ ORB_NAMESPACE_BEGIN namespace ShaderGen { - template< VertexComponent VC > + template< VertexComponent VC, size_t RI > class VaryingHelper; - + class ORB_API_SHADERGEN Varying : public Variable { public: - - using Position = VaryingHelper< VertexComponent::Position >; - using Normal = VaryingHelper< VertexComponent::Normal >; - using Color = VaryingHelper< VertexComponent::Color >; - using TexCoord = VaryingHelper< VertexComponent::TexCoord >; + + template< size_t RI > using Position = VaryingHelper< VertexComponent::Position, RI >; + template< size_t RI > using Normal = VaryingHelper< VertexComponent::Normal, RI >; + template< size_t RI > using Binormal = VaryingHelper< VertexComponent::Binormal, RI >; + template< size_t RI > using Tangent = VaryingHelper< VertexComponent::Tangent, RI >; + template< size_t RI > using Color = VaryingHelper< VertexComponent::Color, RI >; + template< size_t RI > using TexCoord = VaryingHelper< VertexComponent::TexCoord, RI >; + template< size_t RI > using BlendIndices = VaryingHelper< VertexComponent::BlendIndices, RI >; + template< size_t RI > using BlendWeights = VaryingHelper< VertexComponent::BlendWeights, RI >; using Variable::operator=; public: - Varying( VertexComponent component ); + Varying( VertexComponent component, size_t resource_index ); private: std::string GetValueDerived( void ) const override; + + private: + + size_t resource_index_; }; - template< VertexComponent VC > + template< VertexComponent VC, size_t RI > class VaryingHelper : public Varying { public: @@ -56,7 +64,7 @@ namespace ShaderGen public: VaryingHelper( void ) - : Varying( VC ) + : Varying( VC, RI ) { } diff --git a/src/Samples/01-Triangle/TriangleShader.h b/src/Samples/01-Triangle/TriangleShader.h index 4fcc02e..b40231a 100644 --- a/src/Samples/01-Triangle/TriangleShader.h +++ b/src/Samples/01-Triangle/TriangleShader.h @@ -40,8 +40,8 @@ class TriangleShader final : public Orbit::ShaderGen::IShader Attribute::Color a_color; Attribute::TexCoord a_texcoord; - Varying::Position v_position; - Varying::Color v_color; - Varying::TexCoord v_texcoord; + Varying::Position< 0 > v_position; + Varying::Color < 0 > v_color; + Varying::TexCoord< 0 > v_texcoord; }; diff --git a/src/Samples/01-Triangle/main.cpp b/src/Samples/01-Triangle/main.cpp index 614f988..f497f8f 100644 --- a/src/Samples/01-Triangle/main.cpp +++ b/src/Samples/01-Triangle/main.cpp @@ -46,12 +46,15 @@ class SampleApp final : public Orbit::Application< SampleApp > render_context_.Clear( Orbit::BufferMask::Color | Orbit::BufferMask::Depth ); // Push mesh to render queue - Orbit::RenderCommand command; - command.vertex_buffer = mesh_.GetVertexBuffer(); - command.index_buffer = mesh_.GetIndexBuffer(); - command.shader = shader_; - command.textures.emplace_back( texture_.GetTexture2D() ); - Orbit::DefaultRenderer::GetInstance().PushCommand( std::move( command ) ); + if( mesh_ ) + { + Orbit::RenderCommand command; + command.vertex_buffer = mesh_->GetVertexBuffer(); + command.index_buffer = mesh_->GetIndexBuffer(); + command.shader = shader_; + command.textures.emplace_back( texture_.GetTexture2D() ); + Orbit::DefaultRenderer::GetInstance().PushCommand( std::move( command ) ); + } // Render scene Orbit::DefaultRenderer::GetInstance().Render(); @@ -62,10 +65,10 @@ class SampleApp final : public Orbit::Application< SampleApp > private: - Orbit::RenderContext render_context_; - TriangleShader shader_source_; - Orbit::Shader shader_; - Orbit::Mesh mesh_; - Orbit::Texture texture_; + Orbit::RenderContext render_context_; + TriangleShader shader_source_; + Orbit::Shader shader_; + std::shared_ptr< Orbit::Mesh > mesh_; + Orbit::Texture texture_; }; diff --git a/src/Samples/02-Cube/CubeShader.h b/src/Samples/02-Cube/CubeShader.h index 29c0cb6..1f0ac24 100644 --- a/src/Samples/02-Cube/CubeShader.h +++ b/src/Samples/02-Cube/CubeShader.h @@ -42,10 +42,10 @@ class CubeShader final : public Orbit::ShaderGen::IShader Attribute::TexCoord a_texcoord; Attribute::Color a_color; - Varying::Position v_position; - Varying::Normal v_normal; - Varying::TexCoord v_texcoord; - Varying::Color v_color; + Varying::Position< 0 > v_position; + Varying::Normal < 0 > v_normal; + Varying::TexCoord< 0 > v_texcoord; + Varying::Color < 0 > v_color; public: diff --git a/src/Samples/02-Cube/main.cpp b/src/Samples/02-Cube/main.cpp index 2f01d01..b0d5046 100644 --- a/src/Samples/02-Cube/main.cpp +++ b/src/Samples/02-Cube/main.cpp @@ -59,19 +59,23 @@ class SampleApp final : public Orbit::Application< SampleApp > // Update camera camera_.Update( delta_time ); - // Update uniforms - const Orbit::Matrix4 model = mesh_.transform_ * model_matrix_; - shader_.SetVertexUniform( shader_source_.u_view_projection, camera_.GetViewProjection() ); - shader_.SetVertexUniform( shader_source_.u_model, model ); - shader_.SetVertexUniform( shader_source_.u_model_inverse, model.Inverted() ); - - // Push cube mesh to render queue - Orbit::RenderCommand command; - command.vertex_buffer = mesh_.GetVertexBuffer(); - command.index_buffer = mesh_.GetIndexBuffer(); - command.shader = shader_; - command.textures.emplace_back( texture_.GetTexture2D() ); - Orbit::DefaultRenderer::GetInstance().PushCommand( std::move( command ) ); + if( mesh_ ) + { + const Orbit::Matrix4 model = mesh_->transform_ * model_matrix_; + + // Update uniforms + shader_.SetVertexUniform( shader_source_.u_view_projection, camera_.GetViewProjection() ); + shader_.SetVertexUniform( shader_source_.u_model, model ); + shader_.SetVertexUniform( shader_source_.u_model_inverse, model.Inverted() ); + + // Push cube mesh to render queue + Orbit::RenderCommand command; + command.vertex_buffer = mesh_->GetVertexBuffer(); + command.index_buffer = mesh_->GetIndexBuffer(); + command.shader = shader_; + command.textures.emplace_back( texture_.GetTexture2D() ); + Orbit::DefaultRenderer::GetInstance().PushCommand( std::move( command ) ); + } // Render scene Orbit::DefaultRenderer::GetInstance().Render(); @@ -82,12 +86,12 @@ class SampleApp final : public Orbit::Application< SampleApp > private: - Orbit::RenderContext render_context_; - CubeShader shader_source_; - Orbit::Shader shader_; - Orbit::Mesh mesh_; - Orbit::Texture texture_; - Orbit::Matrix4 model_matrix_; - Camera camera_; + Orbit::RenderContext render_context_; + CubeShader shader_source_; + Orbit::Shader shader_; + std::shared_ptr< Orbit::Mesh > mesh_; + Orbit::Texture texture_; + Orbit::Matrix4 model_matrix_; + Camera camera_; }; diff --git a/src/Samples/03-Model/ModelShader.cpp b/src/Samples/03-Model/ModelShader.cpp index 9e473a5..c865afa 100644 --- a/src/Samples/03-Model/ModelShader.cpp +++ b/src/Samples/03-Model/ModelShader.cpp @@ -18,25 +18,31 @@ #include "ModelShader.h" #include +#include #include ModelShader::Vec4 ModelShader::VSMain( void ) { - v_position = u_view_projection * u_model * a_position; - v_color = a_color; - v_texcoord = a_texcoord; - v_normal = ( Transpose( u_model_inverse ) * Vec4( a_normal, 1.0 ) )->xyz; + v_position = u_view_projection * u_model * a_position; + v_color = a_color; + v_texcoord = a_texcoord; + v_normal = ( u_model * Vec4( a_normal, 0.0f ) )->xyz; + v_world_position = u_model * a_position; return v_position; } ModelShader::Vec4 ModelShader::PSMain( void ) { - Vec4 tex_color = Sample( diffuse_texture, v_texcoord ); - Vec4 out_color = tex_color + v_color; + constexpr float ambient_luminance = 0.1f; - Float diffuse = ( -Dot( v_normal, u_light_dir ) * 0.5 ); - out_color->rgb *= diffuse; + Vec4 albedo = Sample( diffuse_texture, v_texcoord ); + Vec3 ambient = u_light_color * ambient_luminance; + Vec3 norm = Normalize( v_normal ); + Vec3 light_dir = Normalize( u_light_pos - v_world_position->xyz ); + Float NdotL = Max( Dot( norm, light_dir ), 0.0f ); + Vec3 diffuse = u_light_color * NdotL; + Vec3 phong = ( ambient + diffuse ) * v_color->rgb; - return out_color; + return Vec4( phong, 1.0f ); } diff --git a/src/Samples/03-Model/ModelShader.h b/src/Samples/03-Model/ModelShader.h index 0a083f8..47b4bbd 100644 --- a/src/Samples/03-Model/ModelShader.h +++ b/src/Samples/03-Model/ModelShader.h @@ -42,17 +42,19 @@ class ModelShader final : public Orbit::ShaderGen::IShader Attribute::TexCoord a_texcoord; Attribute::Normal a_normal; - Varying::Position v_position; - Varying::Color v_color; - Varying::TexCoord v_texcoord; - Varying::Normal v_normal; + Varying::Position< 0 > v_position; + Varying::Color < 0 > v_color; + Varying::TexCoord< 0 > v_texcoord; + Varying::Normal < 0 > v_normal; + Varying::Position< 1 > v_world_position; public: Uniform< Mat4 > u_view_projection; Uniform< Mat4 > u_model; - Uniform< Mat4 > u_model_inverse; - Uniform< Vec3 > u_light_dir; + Uniform< Vec3 > u_light_pos; + Uniform< Vec3 > u_light_color; + Uniform< Vec3 > u_cam_pos; }; diff --git a/src/Samples/03-Model/main.cpp b/src/Samples/03-Model/main.cpp index e4e4bf0..b7a1593 100644 --- a/src/Samples/03-Model/main.cpp +++ b/src/Samples/03-Model/main.cpp @@ -23,7 +23,9 @@ #include #include #include -#include +#include +#include +#include #include #include #include @@ -34,11 +36,10 @@ class SampleApp final : public Orbit::Application< SampleApp > SampleApp( void ) : shader_ ( shader_source_.Generate(), shader_source_.GetVertexLayout() ) - , model_ ( Orbit::Asset( "models/teapot.obj" ), shader_source_.GetVertexLayout() ) + , meshes_ ( Orbit::COLLADAFile( Orbit::Asset( "models/suzanne.dae" ), shader_source_.GetVertexLayout() ).GetModelData().meshes ) , texture_( Orbit::Asset( "textures/checkerboard.tga" ) ) { render_context_.SetClearColor( 0.0f, 0.0f, 0.5f ); - model_matrix_.Translate( Orbit::Vector3( 0.0f, -2.0f, 0.0f ) ); } public: @@ -46,6 +47,7 @@ class SampleApp final : public Orbit::Application< SampleApp > void OnFrame( void ) override { const float delta_time = Orbit::Clock::GetDelta(); + const float life_time = Orbit::Clock::GetLife(); // Clear context render_context_.Clear( Orbit::BufferMask::Color | Orbit::BufferMask::Depth ); @@ -53,28 +55,38 @@ class SampleApp final : public Orbit::Application< SampleApp > // Update camera camera_.Update( delta_time ); - // Rotate model - model_matrix_.Rotate( Orbit::Vector3( 0.0f, 0.5f * Orbit::Pi * delta_time, 0.0f ) ); - // Update vertex uniforms shader_.SetVertexUniform( shader_source_.u_view_projection, camera_.GetViewProjection() ); shader_.SetVertexUniform( shader_source_.u_model, model_matrix_ ); - shader_.SetVertexUniform( shader_source_.u_model_inverse, model_matrix_.Inverted() ); + shader_.SetVertexUniform( shader_source_.u_cam_pos, camera_.position ); + + Orbit::Vector3 light_pos; + light_pos.x = cosf( life_time * Orbit::Pi / 2 ) * 2.0f; + light_pos.y = 0.5f; + light_pos.z = sinf( life_time * Orbit::Pi / 2 ) * 2.0f; + + // Add debug sphere at light pos + Orbit::DebugManager::GetInstance().PushSphere( light_pos, Orbit::RGBA( 1, 1, 0 ) ); // Update pixel uniforms - shader_.SetPixelUniform( shader_source_.u_light_dir, Orbit::Vector3( 1.0f, -1.0f, 1.0f ).Normalized() ); + shader_.SetPixelUniform( shader_source_.u_light_pos, light_pos ); + shader_.SetPixelUniform( shader_source_.u_light_color, Orbit::RGB( 0.7f, 0.4f, 0.2f ) ); // Push meshes to render queue - for( const Orbit::Mesh& mesh : model_ ) + for( auto& mesh : meshes_ ) { Orbit::RenderCommand command; - command.vertex_buffer = mesh.GetVertexBuffer(); - command.index_buffer = mesh.GetIndexBuffer(); + command.vertex_buffer = mesh->GetVertexBuffer(); + command.index_buffer = mesh->GetIndexBuffer(); command.shader = shader_; command.textures.emplace_back( texture_.GetTexture2D() ); Orbit::DefaultRenderer::GetInstance().PushCommand( std::move( command ) ); } + // Render debug objects + Orbit::DebugManager::GetInstance().Render( Orbit::DefaultRenderer::GetInstance(), camera_.GetViewProjection() ); + Orbit::DebugManager::GetInstance().Flush(); + // Render scene Orbit::DefaultRenderer::GetInstance().Render(); @@ -84,12 +96,12 @@ class SampleApp final : public Orbit::Application< SampleApp > private: - Orbit::RenderContext render_context_; - ModelShader shader_source_; - Orbit::Shader shader_; - Orbit::Model model_; - Orbit::Texture texture_; - Orbit::Matrix4 model_matrix_; - Camera camera_; + Orbit::RenderContext render_context_; + ModelShader shader_source_; + Orbit::Shader shader_; + std::vector< std::shared_ptr< Orbit::Mesh > > meshes_; + Orbit::Texture texture_; + Orbit::Matrix4 model_matrix_; + Camera camera_; }; diff --git a/src/Samples/04-Animation/AnimationShader.cpp b/src/Samples/04-Animation/AnimationShader.cpp index b8a0f8b..cb8e1a5 100644 --- a/src/Samples/04-Animation/AnimationShader.cpp +++ b/src/Samples/04-Animation/AnimationShader.cpp @@ -28,11 +28,11 @@ AnimationShader::Vec4 AnimationShader::VSMain( void ) for( size_t i = 0; i < 4; ++i ) { - Vec4 local_position = u_joint_transforms[ a_joint_ids[ i ] ] * a_position; - total_local_pos += local_position * a_weights[ i ]; + Vec4 local_position = u_joint_transforms[ a_blend_indices[ i ] ] * a_position; + total_local_pos += local_position * a_blend_weights[ i ]; - Vec4 world_normal = u_joint_transforms[ a_joint_ids[ i ] ] * Vec4( a_normal, 0.0 ); - total_normal += world_normal * a_weights[ i ]; + Vec4 world_normal = u_joint_transforms[ a_blend_indices[ i ] ] * Vec4( a_normal, 0.0 ); + total_normal += world_normal * a_blend_weights[ i ]; } total_normal = Normalize( total_normal ); diff --git a/src/Samples/04-Animation/AnimationShader.h b/src/Samples/04-Animation/AnimationShader.h index 5752ac1..151b52f 100644 --- a/src/Samples/04-Animation/AnimationShader.h +++ b/src/Samples/04-Animation/AnimationShader.h @@ -41,17 +41,17 @@ class AnimationShader final : public Orbit::ShaderGen::IShader Sampler diffuse_texture; - Attribute::Position a_position; - Attribute::Color a_color; - Attribute::TexCoord a_texcoord; - Attribute::Normal a_normal; - Attribute::JointIDs a_joint_ids; - Attribute::Weights a_weights; - - Varying::Position v_position; - Varying::Color v_color; - Varying::TexCoord v_texcoord; - Varying::Normal v_normal; + Attribute::Position a_position; + Attribute::Color a_color; + Attribute::TexCoord a_texcoord; + Attribute::Normal a_normal; + Attribute::BlendIndices a_blend_indices; + Attribute::BlendWeights a_blend_weights; + + Varying::Position< 0 > v_position; + Varying::Color < 0 > v_color; + Varying::TexCoord< 0 > v_texcoord; + Varying::Normal < 0 > v_normal; public: diff --git a/src/Samples/04-Animation/main.cpp b/src/Samples/04-Animation/main.cpp index 115ad60..0b527d9 100644 --- a/src/Samples/04-Animation/main.cpp +++ b/src/Samples/04-Animation/main.cpp @@ -24,7 +24,8 @@ #include #include #include -#include +#include +#include #include #include @@ -33,9 +34,9 @@ class SampleApp final : public Orbit::Application< SampleApp > public: SampleApp( void ) - : shader_ ( shader_source_.Generate(), shader_source_.GetVertexLayout() ) - , model_ ( Orbit::Asset( "models/mannequin.dae" ), shader_source_.GetVertexLayout() ) - , animation_( Orbit::Asset( "animations/jump.dae" ) ) + : shader_ ( shader_source_.Generate(), shader_source_.GetVertexLayout() ) + , model_data_( Orbit::COLLADAFile( Orbit::Asset( "models/mannequin.dae" ), shader_source_.GetVertexLayout() ).GetModelData() ) + , animation_ ( Orbit::COLLADAFile( Orbit::Asset( "animations/jump.dae" ), shader_source_.GetVertexLayout() ).GetAnimationData().keyframes ) { render_context_.SetClearColor( 0.0f, 0.0f, 0.5f ); model_matrix_.Translate( Orbit::Vector3( 0.0f, -2.0f, 0.0f ) ); @@ -51,7 +52,7 @@ class SampleApp final : public Orbit::Application< SampleApp > void UpdateJointTransformsRecursive( const Orbit::Joint& joint, const Orbit::Matrix4& parent_pose ) { const float life_time = Orbit::Clock::GetLife(); - const float animation_time = std::fmod( life_time, animation_.GetDuration() ); + const float animation_time = std::fmod( life_time, static_cast< float >( animation_.GetDuration() ) ); const Orbit::Matrix4 local_pose = animation_.JointPoseAtTime( joint.name, animation_time ); const Orbit::Matrix4 pose = ( parent_pose * local_pose ); @@ -73,19 +74,18 @@ class SampleApp final : public Orbit::Application< SampleApp > camera_.Update( delta_time ); // Update joint transforms - if( model_.HasJoints() ) - UpdateJointTransformsRecursive( model_.GetRootJoint(), Orbit::Matrix4() ); + UpdateJointTransformsRecursive( model_data_.root_joint, Orbit::Matrix4() ); // Update uniforms shader_.SetVertexUniform( shader_source_.u_view_projection, camera_.GetViewProjection() ); shader_.SetVertexUniform( shader_source_.u_joint_transforms, joint_transforms_ ); // Push meshes to render queue - for( const Orbit::Mesh& mesh : model_ ) + for( auto& mesh : model_data_.meshes ) { Orbit::RenderCommand command; - command.vertex_buffer = mesh.GetVertexBuffer(); - command.index_buffer = mesh.GetIndexBuffer(); + command.vertex_buffer = mesh->GetVertexBuffer(); + command.index_buffer = mesh->GetIndexBuffer(); command.shader = shader_; Orbit::DefaultRenderer::GetInstance().PushCommand( std::move( command ) ); } @@ -103,13 +103,13 @@ class SampleApp final : public Orbit::Application< SampleApp > private: - Orbit::RenderContext render_context_; - AnimationShader shader_source_; - Orbit::Shader shader_; - Orbit::Model model_; - Orbit::Animation animation_; - Orbit::Matrix4 model_matrix_; - Camera camera_; - JointTransformArray joint_transforms_; + Orbit::RenderContext render_context_; + AnimationShader shader_source_; + Orbit::Shader shader_; + Orbit::COLLADAFile::ModelData model_data_; + Orbit::Animation animation_; + Orbit::Matrix4 model_matrix_; + Camera camera_; + JointTransformArray joint_transforms_; }; diff --git a/src/Samples/05-PostFX/PostFXShader.h b/src/Samples/05-PostFX/PostFXShader.h index 32c9464..8e57d1f 100644 --- a/src/Samples/05-PostFX/PostFXShader.h +++ b/src/Samples/05-PostFX/PostFXShader.h @@ -32,8 +32,8 @@ class PostFXShader final : public Orbit::ShaderGen::IShader Attribute::Position a_position; - Varying::Position v_position; - Varying::TexCoord v_texcoord; + Varying::Position< 0 > v_position; + Varying::TexCoord< 0 > v_texcoord; public: diff --git a/src/Samples/05-PostFX/SceneShader.h b/src/Samples/05-PostFX/SceneShader.h index c87f07e..d78138d 100644 --- a/src/Samples/05-PostFX/SceneShader.h +++ b/src/Samples/05-PostFX/SceneShader.h @@ -42,10 +42,10 @@ class SceneShader final : public Orbit::ShaderGen::IShader Attribute::TexCoord a_texcoord; Attribute::Normal a_normal; - Varying::Position v_position; - Varying::Color v_color; - Varying::TexCoord v_texcoord; - Varying::Normal v_normal; + Varying::Position< 0 > v_position; + Varying::Color < 0 > v_color; + Varying::TexCoord< 0 > v_texcoord; + Varying::Normal < 0 > v_normal; public: diff --git a/src/Samples/05-PostFX/main.cpp b/src/Samples/05-PostFX/main.cpp index 12ea30a..0791a64 100644 --- a/src/Samples/05-PostFX/main.cpp +++ b/src/Samples/05-PostFX/main.cpp @@ -27,7 +27,8 @@ #include #include #include -#include +#include +#include #include #include @@ -38,7 +39,7 @@ class SampleApp final : public Orbit::Application< SampleApp > SampleApp( void ) : scene_shader_ ( scene_shader_source_.Generate(), scene_shader_source_.GetVertexLayout() ) , post_fx_shader_( post_fx_shader_source_.Generate(), post_fx_shader_source_.GetVertexLayout() ) - , model_ ( Orbit::Asset( "models/bunny.obj" ), scene_shader_source_.GetVertexLayout() ) + , meshes_ ( Orbit::WavefrontOBJFile( Orbit::Asset( "models/bunny.obj" ), scene_shader_source_.GetVertexLayout() ).GetMeshes() ) { render_context_.SetClearColor( 0.0f, 0.0f, 0.5f ); model_matrix_.Rotate( Orbit::Vector3( 0.0f, Orbit::Pi * 1.0f, 0.0f ) ); @@ -67,24 +68,26 @@ class SampleApp final : public Orbit::Application< SampleApp > // Update post-fx shader uniforms post_fx_shader_.SetPixelUniform( post_fx_shader_source_.u_time, life_time ); - // Push meshes to render queue - for( const Orbit::Mesh& mesh : model_ ) + // Push meshe to render queue + for( auto& mesh : meshes_ ) { Orbit::RenderCommand command; - command.vertex_buffer = mesh.GetVertexBuffer(); - command.index_buffer = mesh.GetIndexBuffer(); + command.vertex_buffer = mesh->GetVertexBuffer(); + command.index_buffer = mesh->GetIndexBuffer(); command.shader = scene_shader_; command.frame_buffer = frame_buffer_; Orbit::DefaultRenderer::GetInstance().PushCommand( std::move( command ) ); } // Push render quad to render queue - Orbit::RenderCommand command; - command.vertex_buffer = render_quad_.vertex_buffer_; - command.index_buffer = render_quad_.index_buffer_; - command.shader = post_fx_shader_; - command.textures.emplace_back( frame_buffer_.GetTexture2D() ); - Orbit::DefaultRenderer::GetInstance().PushCommand( std::move( command ) ); + { + Orbit::RenderCommand command; + command.vertex_buffer = render_quad_.vertex_buffer_; + command.index_buffer = render_quad_.index_buffer_; + command.shader = post_fx_shader_; + command.textures.emplace_back( frame_buffer_.GetTexture2D() ); + Orbit::DefaultRenderer::GetInstance().PushCommand( std::move( command ) ); + } // Render scene Orbit::DefaultRenderer::GetInstance().Render(); @@ -95,15 +98,15 @@ class SampleApp final : public Orbit::Application< SampleApp > private: - Orbit::RenderContext render_context_; - SceneShader scene_shader_source_; - PostFXShader post_fx_shader_source_; - Orbit::Shader scene_shader_; - Orbit::Shader post_fx_shader_; - Orbit::Model model_; - Orbit::FrameBuffer frame_buffer_; - Orbit::Matrix4 model_matrix_; - Camera camera_; - RenderQuad render_quad_; + Orbit::RenderContext render_context_; + SceneShader scene_shader_source_; + PostFXShader post_fx_shader_source_; + Orbit::Shader scene_shader_; + Orbit::Shader post_fx_shader_; + std::vector< std::shared_ptr< Orbit::Mesh > > meshes_; + Orbit::FrameBuffer frame_buffer_; + Orbit::Matrix4 model_matrix_; + Camera camera_; + RenderQuad render_quad_; };