export_obj.ts 3.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697
  1. function export_obj_write_string(out: i32[], str: string) {
  2. for (let i: i32 = 0; i < str.length; ++i) {
  3. array_push(out, char_code_at(str, i));
  4. }
  5. }
  6. function export_obj_run(path: string, paint_objects: mesh_object_t[], apply_displacement = false) {
  7. let o: i32[] = [];
  8. export_obj_write_string(o, "# armorsculpt.org\n");
  9. let texpaint = project_layers[0].texpaint;
  10. let pixels = image_get_pixels(texpaint);
  11. let pixelsView = buffer_view_create(pixels);
  12. let mesh = paint_objects[0].data;
  13. let inda = mesh.index_arrays[0].values;
  14. let posa: i16_array_t = i16_array_create(inda.length * 4);
  15. for (let i: i32 = 0; i < inda.length; ++i) {
  16. let index = inda[i];
  17. posa[index * 4 ] = math_floor(buffer_view_get_f32(pixelsView, i * 16 ) * 32767);
  18. posa[index * 4 + 1] = math_floor(buffer_view_get_f32(pixelsView, i * 16 + 4) * 32767);
  19. posa[index * 4 + 2] = math_floor(buffer_view_get_f32(pixelsView, i * 16 + 8) * 32767);
  20. }
  21. let poff = 0;
  22. // for (let p of paint_objects) {
  23. let p = paint_objects[0];
  24. // let mesh = p.data.raw;
  25. let inv = 1 / 32767;
  26. let sc = p.data.scale_pos * inv;
  27. // let posa = mesh.vertex_arrays[0].values;
  28. let len = math_floor(posa.length / 4);
  29. // let len = math_floor(inda.length);
  30. // Merge shared vertices and remap indices
  31. let posa2: i16_array_t = i16_array_create(len * 3);
  32. let posmap: map_t<i32, i32> = map_create();
  33. let pi = 0;
  34. for (let i: i32 = 0; i < len; ++i) {
  35. let found = false;
  36. for (let j: i32 = 0; j < pi; ++j) {
  37. if (posa2[j * 3 ] == posa[i * 4 ] &&
  38. posa2[j * 3 + 1] == posa[i * 4 + 1] &&
  39. posa2[j * 3 + 2] == posa[i * 4 + 2]) {
  40. map_set(posmap, i, j);
  41. found = true;
  42. break;
  43. }
  44. }
  45. if (!found) {
  46. map_set(posmap, i, pi);
  47. posa2[pi * 3 ] = posa[i * 4 ];
  48. posa2[pi * 3 + 1] = posa[i * 4 + 1];
  49. posa2[pi * 3 + 2] = posa[i * 4 + 2];
  50. pi++;
  51. }
  52. }
  53. export_obj_write_string(o, "o " + p.base.name + "\n");
  54. for (let i: i32 = 0; i < pi; ++i) {
  55. export_obj_write_string(o, "v ");
  56. let vx = posa2[i * 3] * sc + "";
  57. export_obj_write_string(o, substring(vx, 0, string_index_of(vx, ".") + 7));
  58. export_obj_write_string(o, " ");
  59. let vy = posa2[i * 3 + 2] * sc + "";
  60. export_obj_write_string(o, substring(vy, 0, string_index_of(vy, ".") + 7));
  61. export_obj_write_string(o, " ");
  62. let vz = -posa2[i * 3 + 1] * sc + "";
  63. export_obj_write_string(o, substring(vz, 0, string_index_of(vz, ".") + 7));
  64. export_obj_write_string(o, "\n");
  65. }
  66. // let inda = mesh.index_arrays[0].values;
  67. for (let i: i32 = 0; i < math_floor(inda.length / 3); ++i) {
  68. let pi1 = map_get(posmap, inda[i * 3 ]) + 1 + poff;
  69. let pi2 = map_get(posmap, inda[i * 3 + 1]) + 1 + poff;
  70. let pi3 = map_get(posmap, inda[i * 3 + 2]) + 1 + poff;
  71. export_obj_write_string(o, "f ");
  72. export_obj_write_string(o, pi1 + "");
  73. export_obj_write_string(o, " ");
  74. export_obj_write_string(o, pi2 + "");
  75. export_obj_write_string(o, " ");
  76. export_obj_write_string(o, pi3 + "");
  77. export_obj_write_string(o, "\n");
  78. }
  79. poff += pi;
  80. // }
  81. if (!ends_with(path, ".obj")) {
  82. path += ".obj";
  83. }
  84. let b = u8_array_t.from(o).buffer;
  85. krom_file_save_bytes(path, b, buffer_size(b));
  86. }