UNPKG

@acransac/vtk.js

Version:

Visualization Toolkit for the Web

228 lines (182 loc) 8.1 kB
import macro from 'vtk.js/Sources/macro'; import vtkActor from 'vtk.js/Sources/Rendering/Core/Actor'; import vtkCircleSource from 'vtk.js/Sources/Filters/Sources/CircleSource'; import vtkDataArray from 'vtk.js/Sources/Common/Core/DataArray'; import vtkGlyph3DMapper from 'vtk.js/Sources/Rendering/Core/Glyph3DMapper'; import vtkHandleRepresentation from 'vtk.js/Sources/Widgets/Representations/HandleRepresentation'; import vtkMatrixBuilder from 'vtk.js/Sources/Common/Core/MatrixBuilder'; import vtkPolyData from 'vtk.js/Sources/Common/DataModel/PolyData'; import vtkWidgetRepresentation from 'vtk.js/Sources/Widgets/Representations/WidgetRepresentation'; import { ScalarMode } from 'vtk.js/Sources/Rendering/Core/Mapper/Constants'; import { vec3, mat3 } from 'gl-matrix'; // ---------------------------------------------------------------------------- // vtkCircleContextRepresentation methods // ---------------------------------------------------------------------------- function vtkCircleContextRepresentation(publicAPI, model) { // Set our className model.classHierarchy.push('vtkCircleContextRepresentation'); // -------------------------------------------------------------------------- // Internal polydata dataset // -------------------------------------------------------------------------- model.internalPolyData = vtkPolyData.newInstance({ mtime: 0 }); model.internalArrays = { points: model.internalPolyData.getPoints(), scale: vtkDataArray.newInstance({ name: 'scale', numberOfComponents: 3, empty: true, }), color: vtkDataArray.newInstance({ name: 'color', numberOfComponents: 1, empty: true, }), direction: vtkDataArray.newInstance({ name: 'direction', numberOfComponents: 9, empty: true, }), }; model.internalPolyData.getPointData().addArray(model.internalArrays.scale); model.internalPolyData.getPointData().addArray(model.internalArrays.color); model.internalPolyData .getPointData() .addArray(model.internalArrays.direction); // -------------------------------------------------------------------------- // Generic rendering pipeline // -------------------------------------------------------------------------- model.pipelines = { circle: { source: publicAPI, glyph: vtkCircleSource.newInstance({ resolution: model.glyphResolution, radius: 1, lines: model.drawBorder, face: model.drawFace, }), mapper: vtkGlyph3DMapper.newInstance({ orientationArray: 'direction', scaleArray: 'scale', scaleMode: vtkGlyph3DMapper.ScaleModes.SCALE_BY_COMPONENTS, colorByArrayName: 'color', scalarMode: ScalarMode.USE_POINT_FIELD_DATA, }), actor: vtkActor.newInstance({ pickable: false }), }, }; model.pipelines.circle.actor.getProperty().setOpacity(0.2); model.pipelines.circle.mapper.setOrientationModeToMatrix(); model.pipelines.circle.mapper.setResolveCoincidentTopology(true); model.pipelines.circle.mapper.setResolveCoincidentTopologyPolygonOffsetParameters( -1, -1 ); vtkWidgetRepresentation.connectPipeline(model.pipelines.circle); publicAPI.addActor(model.pipelines.circle.actor); model.transform = vtkMatrixBuilder.buildFromDegree(); // -------------------------------------------------------------------------- publicAPI.setGlyphResolution = macro.chain( publicAPI.setGlyphResolution, (r) => model.glyph.setResolution(r) ); // -------------------------------------------------------------------------- publicAPI.setDrawBorder = (draw) => { model.pipelines.circle.glyph.setLines(draw); }; // -------------------------------------------------------------------------- publicAPI.setDrawFace = (draw) => { model.pipelines.circle.glyph.setFace(draw); }; // -------------------------------------------------------------------------- publicAPI.setOpacity = (opacity) => { model.pipelines.circle.actor.getProperty().setOpacity(opacity); }; // -------------------------------------------------------------------------- publicAPI.requestData = (inData, outData) => { const { points, scale, color, direction } = model.internalArrays; const list = publicAPI.getRepresentationStates(inData[0]); const totalCount = list.length; if (color.getNumberOfValues() !== totalCount) { // Need to resize dataset points.setData(new Float32Array(3 * totalCount)); scale.setData(new Float32Array(3 * totalCount)); direction.setData(new Float32Array(9 * totalCount)); color.setData(new Float32Array(totalCount)); } const typedArray = { points: points.getData(), scale: scale.getData(), color: color.getData(), direction: direction.getData(), }; for (let i = 0; i < list.length; i++) { const state = list[i]; const isActive = state.getActive(); const scaleFactor = isActive ? model.activeScaleFactor : 1; const coord = state.getOrigin(); typedArray.points[i * 3 + 0] = coord[0]; typedArray.points[i * 3 + 1] = coord[1]; typedArray.points[i * 3 + 2] = coord[2]; const right = state.getRight ? state.getRight() : [1, 0, 0]; const up = state.getUp ? state.getUp() : [0, 1, 0]; const dir = state.getDirection ? state.getDirection() : [0, 0, 1]; const rotation = [...right, ...up, ...dir]; let scale3 = state.getScale3 ? state.getScale3() : [1, 1, 1]; scale3 = scale3.map((x) => (x === 0 ? 2 * model.defaultScale : 2 * x)); // Reorient rotation and scale3 since the circle source faces X instead of Z const reorientCircleSource4 = vtkMatrixBuilder .buildFromDegree() .rotateFromDirections([1, 0, 0], [0, 0, 1]) // from X to Z .getMatrix(); const reorientCircleSource3 = []; mat3.fromMat4(reorientCircleSource3, reorientCircleSource4); vec3.transformMat4(scale3, scale3, reorientCircleSource4); mat3.multiply(rotation, rotation, reorientCircleSource3); for (let j = 0; j < 9; j += 1) { typedArray.direction[i * 9 + j] = rotation[j]; } const scale1 = (state.getScale1 ? state.getScale1() : model.defaultScale) / 2; let sFactor = scaleFactor; if (state.getVisible && !state.getVisible()) { sFactor = 0; } typedArray.scale[i * 3 + 0] = scale1 * sFactor * scale3[0]; typedArray.scale[i * 3 + 1] = scale1 * sFactor * scale3[1]; typedArray.scale[i * 3 + 2] = scale1 * sFactor * scale3[2]; typedArray.color[i] = model.useActiveColor && isActive ? model.activeColor : state.getColor(); } model.internalPolyData.modified(); outData[0] = model.internalPolyData; }; // -------------------------------------------------------------------------- // Initialization // -------------------------------------------------------------------------- publicAPI.setActiveScaleFactor(1); } // ---------------------------------------------------------------------------- // Object factory // ---------------------------------------------------------------------------- const DEFAULT_VALUES = { glyphResolution: 32, defaultScale: 1, drawBorder: false, drawFace: true, }; // ---------------------------------------------------------------------------- export function extend(publicAPI, model, initialValues = {}) { Object.assign(model, DEFAULT_VALUES, initialValues); vtkHandleRepresentation.extend(publicAPI, model, initialValues); macro.setGet(publicAPI, model, ['glyphResolution', 'defaultScale']); macro.get(publicAPI, model, ['glyph', 'mapper', 'actor']); // Object specific methods vtkCircleContextRepresentation(publicAPI, model); } // ---------------------------------------------------------------------------- export const newInstance = macro.newInstance( extend, 'vtkCircleContextRepresentation' ); // ---------------------------------------------------------------------------- export default { newInstance, extend };