io/amf-serializer/index.js

  1. /*
  2. JSCAD Object to AMF (XML) Format Serialization
  3. ## License
  4. Copyright (c) 2018 JSCAD Organization https://github.com/jscad
  5. All code released under MIT license
  6. Notes:
  7. 1) geom2 conversion to:
  8. none
  9. 2) geom3 conversion to:
  10. mesh
  11. 3) path2 conversion to:
  12. none
  13. TBD
  14. 1) support zip output
  15. */
  16. /**
  17. * Serializer of JSCAD geometries to AMF source data (XML)
  18. *
  19. * The serialization of the following geometries are possible.
  20. * - serialization of 3D geometry (geom3) to AMF object (a unique mesh containing both vertices and volumes)
  21. *
  22. * Colors are added to volumes when found on the 3D geometry.
  23. * Colors are added to triangles when found on individual polygons.
  24. *
  25. * @module io/amf-serializer
  26. * @example
  27. * const { serializer, mimeType } = require('@jscad/amf-serializer')
  28. */
  29. const stringify = require('onml/lib/stringify')
  30. const { geometries, modifiers } = require('@jscad/modeling')
  31. const { flatten, toArray } = require('@jscad/array-utils')
  32. const mimeType = 'application/amf+xml'
  33. /**
  34. * Serialize the give objects (geometry) to AMF source data (XML).
  35. * @param {Object} options - options for serialization
  36. * @param {String} [options.unit='millimeter'] - unit of design; millimeter, inch, feet, meter or micrometer
  37. * @param {Function} [options.statusCallback] - call back function for progress ({ progress: 0-100 })
  38. * @param {...Object} objects - objects to serialize into AMF source data
  39. * @returns {Array} serialized contents, AMF source data(XML)
  40. * @alias module:io/amf-serializer.serialize
  41. * @example
  42. * const geometry = primitives.cube()
  43. * const amfData = serializer({unit: 'meter'}, geometry)
  44. */
  45. const serialize = (options, ...objects) => {
  46. const defaults = {
  47. statusCallback: null,
  48. unit: 'millimeter' // millimeter, inch, feet, meter or micrometer
  49. }
  50. options = Object.assign({}, defaults, options)
  51. objects = flatten(objects)
  52. // convert only 3D geometries
  53. let objects3d = objects.filter((object) => geometries.geom3.isA(object))
  54. if (objects3d.length === 0) throw new Error('only 3D geometries can be serialized to AMF')
  55. if (objects.length !== objects3d.length) console.warn('some objects could not be serialized to AMF')
  56. // convert to triangles
  57. objects3d = toArray(modifiers.generalize({ snap: true, triangulate: true }, objects3d))
  58. options.statusCallback && options.statusCallback({ progress: 0 })
  59. // construct the contents of the XML
  60. let body = ['amf',
  61. {
  62. unit: options.unit,
  63. version: '1.1'
  64. },
  65. ['metadata', { type: 'author' }, 'Created by JSCAD']
  66. ]
  67. body = body.concat(translateObjects(objects3d, options))
  68. // convert the contents to AMF (XML) format
  69. const amf = `<?xml version="1.0" encoding="UTF-8"?>
  70. ${stringify(body, 2)}`
  71. options && options.statusCallback && options.statusCallback({ progress: 100 })
  72. return [amf]
  73. }
  74. const translateObjects = (objects, options) => {
  75. const contents = []
  76. objects.forEach((object, i) => {
  77. const polygons = geometries.geom3.toPolygons(object)
  78. if (polygons.length > 0) {
  79. options.id = i
  80. contents.push(convertToObject(object, options))
  81. }
  82. })
  83. return contents
  84. }
  85. const convertToObject = (object, options) => {
  86. const contents = ['object', { id: options.id }, convertToMesh(object, options)]
  87. return contents
  88. }
  89. const convertToMesh = (object, options) => {
  90. let contents = ['mesh', {}, convertToVertices(object, options)]
  91. contents = contents.concat(convertToVolumes(object, options))
  92. return contents
  93. }
  94. /*
  95. * This section converts each 3D geometry to a list of vertex / coordinates
  96. */
  97. const convertToVertices = (object, options) => {
  98. const contents = ['vertices', {}]
  99. const vertices = []
  100. const polygons = geometries.geom3.toPolygons(object)
  101. polygons.forEach((polygon) => {
  102. for (let i = 0; i < polygon.vertices.length; i++) {
  103. vertices.push(convertToVertex(polygon.vertices[i], options))
  104. }
  105. })
  106. return contents.concat(vertices)
  107. }
  108. const convertToVertex = (vertex, options) => {
  109. const contents = ['vertex', {}, convertToCoordinates(vertex, options)]
  110. return contents
  111. }
  112. const convertToCoordinates = (vertex, options) => {
  113. const contents = ['coordinates', {}, ['x', {}, vertex[0]], ['y', {}, vertex[1]], ['z', {}, vertex[2]]]
  114. return contents
  115. }
  116. /*
  117. * This section converts each 3D geometry to a list of volumes consisting of indexes into the list of vertices
  118. */
  119. const convertToVolumes = (object, options) => {
  120. const objectcolor = convertColor(object.color)
  121. const polygons = geometries.geom3.toPolygons(object)
  122. const contents = []
  123. let volume = ['volume', {}]
  124. // add color specification if available
  125. if (objectcolor) {
  126. volume.push(objectcolor)
  127. }
  128. let vcount = 0
  129. polygons.forEach((polygon) => {
  130. if (polygon.vertices.length < 3) {
  131. return
  132. }
  133. const triangles = convertToTriangles(polygon, vcount, options)
  134. volume = volume.concat(triangles)
  135. vcount += polygon.vertices.length
  136. })
  137. contents.push(volume)
  138. return contents
  139. }
  140. const convertColor = (color) => {
  141. if (color) {
  142. if (color.length < 4) color.push(1.0)
  143. return ['color', {}, ['r', {}, color[0]], ['g', {}, color[1]], ['b', {}, color[2]], ['a', {}, color[3]]]
  144. }
  145. return null
  146. }
  147. const convertToColor = (polygon, options) => {
  148. const color = polygon.color
  149. return convertColor(color)
  150. }
  151. const convertToTriangles = (polygon, index, options) => {
  152. const polycolor = convertToColor(polygon, options)
  153. // making sure they are all triangles (triangular polygons)
  154. const contents = []
  155. for (let i = 0; i < polygon.vertices.length - 2; i++) {
  156. if (polycolor) {
  157. contents.push(['triangle', {}, polycolor, ['v1', {}, index], ['v2', {}, (index + i + 1)], ['v3', {}, (index + i + 2)]])
  158. } else {
  159. contents.push(['triangle', {}, ['v1', {}, index], ['v2', {}, (index + i + 1)], ['v3', {}, (index + i + 2)]])
  160. }
  161. }
  162. return contents
  163. }
  164. module.exports = {
  165. serialize,
  166. mimeType
  167. }