From f4ab928789c6ff9682d15cc9514a58eb26c8e0f1 Mon Sep 17 00:00:00 2001 From: Francois Date: Tue, 19 Sep 2023 17:39:55 -0400 Subject: [PATCH 01/20] Some cleanup and the start of rendering cylinders the proper way --- docs/test.tldr | 383 ++++++++++++++++++++ src/index.ts | 1 - src/meshes/CylinderMesh.ts | 20 - src/react/Axii.txt | 48 --- src/react/Box.txt | 124 ------- src/react/Cylinder.txt | 109 ------ src/react/CylinderMesh.txt | 120 ------ src/react/Grid.txt | 87 ----- src/renderer/Renderer.ts | 17 +- src/renderer/renderCylinder.ts | 81 +++++ src/renderer/{sphere.ts => renderSphere.ts} | 6 - src/shapes/Cylinder.ts | 36 +- src/shapes/Shape.ts | 13 +- workbench/main.ts | 6 +- workbench/scenes/Cylinders.ts | 114 ++++++ workbench/scenes/KitchenSink.ts | 3 - workbench/scenes/SingleCylinder.ts | 145 ++++++++ 17 files changed, 765 insertions(+), 548 deletions(-) create mode 100644 docs/test.tldr delete mode 100644 src/meshes/CylinderMesh.ts delete mode 100644 src/react/Axii.txt delete mode 100644 src/react/Box.txt delete mode 100644 src/react/Cylinder.txt delete mode 100644 src/react/CylinderMesh.txt delete mode 100644 src/react/Grid.txt create mode 100644 src/renderer/renderCylinder.ts rename src/renderer/{sphere.ts => renderSphere.ts} (97%) create mode 100644 workbench/scenes/Cylinders.ts create mode 100644 workbench/scenes/SingleCylinder.ts diff --git a/docs/test.tldr b/docs/test.tldr new file mode 100644 index 0000000..4e4fad5 --- /dev/null +++ b/docs/test.tldr @@ -0,0 +1,383 @@ +{ + "tldrawFileFormatVersion": 1, + "schema": { + "schemaVersion": 1, + "storeVersion": 4, + "recordVersions": { + "asset": { + "version": 1, + "subTypeKey": "type", + "subTypeVersions": { + "image": 2, + "video": 2, + "bookmark": 0 + } + }, + "camera": { + "version": 1 + }, + "document": { + "version": 2 + }, + "instance": { + "version": 17 + }, + "instance_page_state": { + "version": 3 + }, + "page": { + "version": 1 + }, + "shape": { + "version": 3, + "subTypeKey": "type", + "subTypeVersions": { + "group": 0, + "embed": 4, + "bookmark": 1, + "image": 2, + "text": 1, + "draw": 1, + "geo": 7, + "line": 0, + "note": 4, + "frame": 0, + "arrow": 1, + "highlight": 0, + "video": 1 + } + }, + "instance_presence": { + "version": 4 + }, + "pointer": { + "version": 1 + } + } + }, + "records": [ + { + "gridSize": 10, + "name": "", + "meta": {}, + "id": "document:document", + "typeName": "document" + }, + { + "id": "pointer:pointer", + "typeName": "pointer", + "x": 1323.078125, + "y": 743.875, + "lastActivityTimestamp": 1694612082026, + "meta": {} + }, + { + "meta": {}, + "id": "page:BOCl88s28IpD2VrJVSgCf", + "name": "Page 1", + "index": "a1", + "typeName": "page" + }, + { + "x": 0, + "y": 0, + "z": 1, + "meta": {}, + "id": "camera:page:BOCl88s28IpD2VrJVSgCf", + "typeName": "camera" + }, + { + "editingId": null, + "croppingId": null, + "selectedIds": [], + "hoveredId": null, + "erasingIds": [], + "hintingIds": [], + "focusLayerId": null, + "meta": {}, + "id": "instance_page_state:page:BOCl88s28IpD2VrJVSgCf", + "pageId": "page:BOCl88s28IpD2VrJVSgCf", + "typeName": "instance_page_state" + }, + { + "followingUserId": null, + "opacityForNextShape": 1, + "stylesForNextShape": { + "tldraw:geo": "ellipse", + "tldraw:fill": "semi" + }, + "brush": null, + "scribble": null, + "cursor": { + "type": "default", + "color": "black", + "rotation": 0 + }, + "isFocusMode": false, + "exportBackground": true, + "isDebugMode": false, + "isToolLocked": false, + "screenBounds": { + "x": 0, + "y": 0, + "w": 2265, + "h": 1308 + }, + "zoomBrush": null, + "isGridMode": false, + "isPenMode": false, + "chatMessage": "", + "isChatting": false, + "highlightedUserIds": [], + "meta": {}, + "id": "instance:instance", + "currentPageId": "page:BOCl88s28IpD2VrJVSgCf", + "typeName": "instance" + }, + { + "x": 650.796875, + "y": 386.6328125, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:Z96x88IKDjkhGxn0NXPDW", + "type": "geo", + "props": { + "w": 184.109375, + "h": 146.57421875, + "geo": "rectangle", + "color": "black", + "labelColor": "black", + "fill": "semi", + "dash": "draw", + "size": "m", + "font": "draw", + "text": "Start", + "align": "middle", + "verticalAlign": "middle", + "growY": 0, + "url": "" + }, + "parentId": "page:BOCl88s28IpD2VrJVSgCf", + "index": "a1", + "typeName": "shape" + }, + { + "x": 995.734375, + "y": 352.6640625, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:of1Kr9dzBYlOrolSGLjNN", + "type": "geo", + "props": { + "w": 278.625, + "h": 240.74609375, + "geo": "rectangle", + "color": "black", + "labelColor": "black", + "fill": "semi", + "dash": "draw", + "size": "m", + "font": "draw", + "text": "Middle", + "align": "middle", + "verticalAlign": "middle", + "growY": 0, + "url": "" + }, + "parentId": "page:BOCl88s28IpD2VrJVSgCf", + "index": "a2", + "typeName": "shape" + }, + { + "x": 744.515625, + "y": 552.5625, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:QjsyMlTJy279E7lRS01sa", + "type": "arrow", + "parentId": "page:BOCl88s28IpD2VrJVSgCf", + "index": "a3", + "props": { + "dash": "draw", + "size": "m", + "fill": "none", + "color": "black", + "labelColor": "black", + "bend": 0, + "start": { + "type": "binding", + "boundShapeId": "shape:Z96x88IKDjkhGxn0NXPDW", + "normalizedAnchor": { + "x": 0.5, + "y": 0.5 + }, + "isExact": false + }, + "end": { + "type": "binding", + "boundShapeId": "shape:of1Kr9dzBYlOrolSGLjNN", + "normalizedAnchor": { + "x": 0.5, + "y": 0.5 + }, + "isExact": false + }, + "arrowheadStart": "none", + "arrowheadEnd": "arrow", + "text": "", + "font": "draw" + }, + "typeName": "shape" + }, + { + "x": 1468.46875, + "y": 437.19921875, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:cXBMkFoK7U62Dbcsj9fw7", + "type": "geo", + "props": { + "w": 278.625, + "h": 240.74609375, + "geo": "rectangle", + "color": "black", + "labelColor": "black", + "fill": "semi", + "dash": "draw", + "size": "m", + "font": "draw", + "text": "End", + "align": "middle", + "verticalAlign": "middle", + "growY": 0, + "url": "" + }, + "parentId": "page:BOCl88s28IpD2VrJVSgCf", + "index": "a2V", + "typeName": "shape" + }, + { + "x": 1167.6328125, + "y": 420.6953125, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:9O1KB_ryC5dL9ggheEi9g", + "type": "arrow", + "parentId": "page:BOCl88s28IpD2VrJVSgCf", + "index": "a2l", + "props": { + "dash": "draw", + "size": "m", + "fill": "semi", + "color": "black", + "labelColor": "black", + "bend": 0, + "start": { + "type": "binding", + "boundShapeId": "shape:of1Kr9dzBYlOrolSGLjNN", + "normalizedAnchor": { + "x": 0.5, + "y": 0.5 + }, + "isExact": false + }, + "end": { + "type": "binding", + "boundShapeId": "shape:cXBMkFoK7U62Dbcsj9fw7", + "normalizedAnchor": { + "x": 0.5, + "y": 0.5 + }, + "isExact": false + }, + "arrowheadStart": "none", + "arrowheadEnd": "arrow", + "text": "", + "font": "draw" + }, + "typeName": "shape" + }, + { + "x": 994.8359375, + "y": 769.7578125, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:xc8hARLSRwGfnJ3yUzY91", + "type": "geo", + "props": { + "w": 249, + "h": 248.8671875, + "geo": "ellipse", + "color": "black", + "labelColor": "black", + "fill": "semi", + "dash": "draw", + "size": "m", + "font": "draw", + "text": "Off the beaten path", + "align": "middle", + "verticalAlign": "middle", + "growY": 0, + "url": "" + }, + "parentId": "page:BOCl88s28IpD2VrJVSgCf", + "index": "a4", + "typeName": "shape" + }, + { + "x": 1124.296875, + "y": 522.0078125, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:VmU06a1JHpgle52-BBMLR", + "type": "arrow", + "parentId": "page:BOCl88s28IpD2VrJVSgCf", + "index": "a5", + "props": { + "dash": "draw", + "size": "m", + "fill": "semi", + "color": "black", + "labelColor": "black", + "bend": 0, + "start": { + "type": "binding", + "boundShapeId": "shape:of1Kr9dzBYlOrolSGLjNN", + "normalizedAnchor": { + "x": 0.5, + "y": 0.5 + }, + "isExact": false + }, + "end": { + "type": "binding", + "boundShapeId": "shape:xc8hARLSRwGfnJ3yUzY91", + "normalizedAnchor": { + "x": 0.5, + "y": 0.5 + }, + "isExact": false + }, + "arrowheadStart": "none", + "arrowheadEnd": "arrow", + "text": "", + "font": "draw" + }, + "typeName": "shape" + } + ] +} \ No newline at end of file diff --git a/src/index.ts b/src/index.ts index dabc09f..8807392 100644 --- a/src/index.ts +++ b/src/index.ts @@ -13,6 +13,5 @@ export * from "./shapes/Cylinder"; export * from "./shapes/Sphere"; export * from "./meshes/Mesh"; export * from "./meshes/BoxMesh"; -export * from "./meshes/CylinderMesh"; export * from "./math/Vector3"; export * from "./math/Matrix4x4"; diff --git a/src/meshes/CylinderMesh.ts b/src/meshes/CylinderMesh.ts deleted file mode 100644 index 32d3167..0000000 --- a/src/meshes/CylinderMesh.ts +++ /dev/null @@ -1,20 +0,0 @@ -import { Vector3 } from "../math/Vector3"; -import { extrude } from "./Extrude"; - -function circlePolyline(radius: number, segments: number): Vector3[] { - const vertices: Vector3[] = []; - for (let i = 0; i < segments; i++) { - vertices.push( - Vector3( - Math.cos((i / segments) * 2 * Math.PI) * radius, - 0, - Math.sin((i / segments) * 2 * Math.PI) * radius - ) - ); - } - return vertices; -} - -export function CylinderMesh(radius: number, height: number, segments: number) { - return extrude(circlePolyline(radius, segments), height); -} diff --git a/src/react/Axii.txt b/src/react/Axii.txt deleted file mode 100644 index 44d007d..0000000 --- a/src/react/Axii.txt +++ /dev/null @@ -1,48 +0,0 @@ -import React from "react"; -import { Box } from "./Box"; - -export type AxiiProps = { - x: number; - y: number; - z: number; - scale: number; -}; - -export function Axii(props: AxiiProps) { - const thickness = props.scale / 20.0; - - return ( - - - - - - ); -} diff --git a/src/react/Box.txt b/src/react/Box.txt deleted file mode 100644 index 21be079..0000000 --- a/src/react/Box.txt +++ /dev/null @@ -1,124 +0,0 @@ -import React from "react"; -import { point3DToIsometric } from "./Camera"; - -type BoxProps = { - x: number; - y: number; - z: number; - width: number; - height: number; - depth: number; - fill: string; - stroke: string; -}; - -/* - - Coordinate System - Y - | - | - | - | - / \ - / \ - Z \ X - - Vertices Indices - - 0 - /- - /- \- - /-- \- - /-- \- - 3 /- \- 1 - -- --| - |-\ --/ | - | --\ -/ | - | --\ 2 --/ | - | ---/ | - | | | - | | | - | | | - | | | - 6 - | | - \- | / 4 - \- | /- - \- | /- - \- | /-- - -- - 5 - - - */ - -function generatePolygons({ - width, - height, - depth, -}: BoxProps): Array>>> { - const vertices = [ - // 0,1,2,3 - [-width / 2, +height / 2, -depth / 2], - [+width / 2, +height / 2, -depth / 2], - [+width / 2, +height / 2, +depth / 2], - [-width / 2, +height / 2, +depth / 2], - // 4,5,6 - [+width / 2, -height / 2, -depth / 2], - [+width / 2, -height / 2, depth / 2], - [-width / 2, -height / 2, depth / 2], - ]; - return [ - // Faces - [ - // Right Face (2,1,4,5) - [vertices[3], vertices[2], vertices[5], vertices[6]], - // Front Face - [vertices[2], vertices[1], vertices[4], vertices[5]], - - // Top Face - [vertices[0], vertices[1], vertices[2], vertices[3]], - ], - // Outline - [[]], - ]; -} - -export function Box(props: BoxProps) { - const polygons = generatePolygons(props); - const faces = polygons[0]; - const brightness = [160 / 255, 210 / 255, 255 / 255]; - - return ( - - { - // All faces - faces.map((face, i) => { - let points = ""; - // A face - face.forEach((vertex) => { - const { x, y } = point3DToIsometric( - vertex[0] + props.x, - vertex[1] + props.y, - vertex[2] + props.z - ); - points += `${Math.floor(x)},${Math.floor(y)} `; - }); - // "0,100 50,25 50,75 100,0"; - return ( - - ); - }) - } - - ); -} diff --git a/src/react/Cylinder.txt b/src/react/Cylinder.txt deleted file mode 100644 index 9444de8..0000000 --- a/src/react/Cylinder.txt +++ /dev/null @@ -1,109 +0,0 @@ -import React from "react"; -import { point3DToIsometric } from "./Camera"; -import { generateCylinder } from "./CylinderMesh"; -import { Vector3 } from "./Vector3"; - -type CylinderProps = { - x: number; - y: number; - z: number; - height: number; - radius: number; - fill: string; - stroke: string; - segments: number; -}; - -const directLight = Vector3(1, 1, 0).normalize(); -const ambientLight = 0.4; -const cameraDirection = Vector3(1, 1, 1).normalize(); - -export function Cylinder(props: CylinderProps) { - const mesh = generateCylinder({ - x: props.x, - y: props.y, - z: props.z, - height: props.height, - radius: props.radius, - segments: props.segments, - fill: props.fill, - stroke: props.stroke, - }); - - const bottomFace = mesh.faces[1]; - - const transformedVertices = mesh.vertices.map((vertex) => { - return point3DToIsometric(vertex.x, vertex.y, vertex.z); - }); - - const maxDotPerFace = mesh.faces.map((face) => { - let maxDot = cameraDirection.dotProduct(mesh.vertices[face.indices[0]]); - for (let i = 1; i < face.indices.length; i++) { - maxDot = Math.max( - cameraDirection.dotProduct(mesh.vertices[face.indices[i]]), - maxDot - ); - } - return maxDot; - }); - - // Sort faces by distance from camera - const faceIndices = mesh.faces.map((_, i) => i); - const sortedFaces = faceIndices.sort((faceIndexA, faceIndexB) => { - return maxDotPerFace[faceIndexA] - maxDotPerFace[faceIndexB]; - }); - - return ( - - { - // All faces - sortedFaces.map((faceIndex) => { - const face = mesh.faces[faceIndex]; - let points = ""; - - const brightness = Math.min( - Math.max( - face.normal.dotProduct(directLight) + ambientLight, - ambientLight - ), - 1.0 - ); - - const filter = - bottomFace === face - ? "drop-shadow(0px 0px 7px black)" - : `brightness(${brightness})`; - - face.indices.forEach((index) => { - console.log(faceIndex, index); - points += `${transformedVertices[index].x},${transformedVertices[index].y} `; - }); - return ( - - ); - }) - } - {/* {transformedVertices.map((vertex, i) => ( - - {i} - - ))} */} - - ); -} diff --git a/src/react/CylinderMesh.txt b/src/react/CylinderMesh.txt deleted file mode 100644 index a0bb118..0000000 --- a/src/react/CylinderMesh.txt +++ /dev/null @@ -1,120 +0,0 @@ -import { Mesh, Face } from "./Mesh"; -import { Vector3 } from "./Vector3"; - -/* - - Coordinate System - Y - | - | - | - | - / \ - / \ - Z \ X - - ^ - \_ 0 - \ / - <----/ - - - - - */ - -export function generateCylinder(options: { - x: number; - y: number; - z: number; - radius: number; - height: number; - segments: number; - fill: string; - stroke: string; -}): Mesh { - const mesh: Mesh = { - vertices: [], - faces: [], - }; - - let sideFacesNormals = []; - - // Top face vertices - for (let i = 0; i < options.segments; i++) { - const percent = i / options.segments; - const theta = percent * (Math.PI * 2); - - const normalizedX = Math.cos(theta); - const normalizedZ = Math.sin(theta); - - mesh.vertices.push( - Vector3( - options.x + normalizedX * options.radius, - options.y + options.height / 2, - options.z + normalizedZ * options.radius - ) - ); - } - - // Calculate side faces normals - for (let i = 0; i < options.segments; i++) { - const nextI = (i + 1) % options.segments; - const average = Vector3( - (mesh.vertices[i].x + mesh.vertices[nextI].x) / 2 - options.x, - 0, - (mesh.vertices[i].z + mesh.vertices[nextI].z) / 2 - options.z - ); - average.normalize(); - sideFacesNormals.push(average); - } - - // Bottom face vertices are derived from top face vertices but reversed and z is flipped - const bottomFaceVertices = mesh.vertices.map((vertex) => - Vector3(vertex.x, options.y - options.height / 2.0, vertex.z) - ); - - // Make bottom face vertices clockwise - bottomFaceVertices.reverse(); - mesh.vertices = mesh.vertices.concat(bottomFaceVertices); - - // Generate faces - const topFace: Face = { - indices: [], - fill: options.fill, - stroke: options.stroke, - normal: Vector3(0, 1, 0), - }; - const bottomFace: Face = { - indices: [], - fill: options.fill, - stroke: options.stroke, - normal: Vector3(0, -1, 0), - }; - mesh.faces.push(topFace); - mesh.faces.push(bottomFace); - for (let i = 0; i < options.segments; i++) { - // Top face - topFace.indices.push(i); - // Bottom face - bottomFace.indices.push(i + options.segments); - - // Side faces - const nextI = (i + 1) % options.segments; - const sideFace: Face = { - indices: [ - nextI, - i, - options.segments - 1 - i + options.segments, - options.segments - 1 - nextI + options.segments, - ], - fill: options.fill, - stroke: options.stroke, - normal: sideFacesNormals[i], - }; - - mesh.faces.push(sideFace); - } - - return mesh; -} diff --git a/src/react/Grid.txt b/src/react/Grid.txt deleted file mode 100644 index 2616050..0000000 --- a/src/react/Grid.txt +++ /dev/null @@ -1,87 +0,0 @@ -import React from "react"; -import { point3DToIsometric } from "./Camera"; - -export type GridProps = { - x: number; - y: number; - z: number; - width: number; - depth: number; -}; - -/* - - Coordinate System - Y - | - | - | - | - / \ - / \ - Z \ X - - Vertices Indices - - 0 - /- - /- \- - /-- \- - /-- \- - 3 /- \- 1 - -- --| - -\ --/ - --\ -/ - --\ 2 --/ - ---/ - - */ - -export const GridColor = "#f1f1f1"; - -export function Grid(props: GridProps) { - const gridPoints = [ - [-props.width / 2, 0, -props.depth / 2], - [+props.width / 2, 0, -props.depth / 2], - [+props.width / 2, 0, +props.depth / 2], - [-props.width / 2, 0, +props.depth / 2], - ]; - - let points = ""; - // A face - gridPoints.forEach((vertex) => { - const { x, y } = point3DToIsometric( - vertex[0] + props.x, - vertex[1] + props.y, - vertex[2] + props.z - ); - points += `${Math.floor(x)},${Math.floor(y)} `; - }); - - const floorPolygon = ( - - ); - - return ( - - - - - - - - - {floorPolygon} - - ); -} diff --git a/src/renderer/Renderer.ts b/src/renderer/Renderer.ts index c6829bd..6ad559f 100644 --- a/src/renderer/Renderer.ts +++ b/src/renderer/Renderer.ts @@ -5,8 +5,9 @@ import { Viewport } from "./Viewport"; import { MeshShape, Shape, TransformProperties } from "../shapes/Shape"; import { Matrix4x4 } from "../math/Matrix4x4"; import { applyLighting } from "../lighting/LightingModel"; -import { renderSphere } from "./sphere"; +import { renderSphere } from "./renderSphere"; import { ColorToCSS } from "../colors/Color"; +import { renderCylinder } from "./renderCylinder"; const CrackFillingStrokeWidth = 0.5; @@ -111,6 +112,20 @@ export function render( inverseAndProjectionMatrix ); break; + case "cylinder": + renderCylinder( + scene, + svg, + defs, + shape, + viewport, + worldTransform, + cameraZoom, + cameraDirection, + inverseCameraMatrix, + inverseAndProjectionMatrix + ); + break; default: throw new Error(`Unknown shape type: ${(shape as Shape).type}`); } diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts new file mode 100644 index 0000000..1ae2c07 --- /dev/null +++ b/src/renderer/renderCylinder.ts @@ -0,0 +1,81 @@ +// All of this code is based on this early prototype: +// https://codesandbox.io/s/directionally-lit-sphere-using-svg-radial-gradients-c32ncz?file=/src/Sphere.tsx:2108-4852 +// and this Observable Notebook +// https://observablehq.com/d/011f054fc7eaf966 + +import { projectToScreenCoordinate } from "../cameras/Camera"; +import { ColorToCSS } from "../colors/Color"; +import { applyLighting } from "../lighting/LightingModel"; +import { Matrix4x4 } from "../math/Matrix4x4"; +import { Vector3 } from "../math/Vector3"; +import { CylinderShape } from "../shapes/Shape"; +import { Scene } from "./Scene"; +import { Viewport } from "./Viewport"; + +export function renderCylinder( + _scene: Scene, + svg: SVGElement, + _defs: SVGDefsElement, + cylinder: CylinderShape, + viewport: Viewport, + worldTransform: Matrix4x4, + _cameraZoom: number, + cameraDirection: Vector3, + _inverseCameraMatrix: Matrix4x4, + inverseAndProjectionMatrix: Matrix4x4 +) { + const points: Vector3[] = [ + // Top + Vector3(0, cylinder.height / 2, 0), + // Bottom + Vector3(0, -cylinder.height / 2, 0), + ].map((point) => { + worldTransform.applyToVector3(point); + + return projectToScreenCoordinate( + point, + inverseAndProjectionMatrix, + viewport + ); + }); + + const yAxis = Vector3(0, 1, 0); + worldTransform.extractBasis(Vector3(0, 0, 0), yAxis, Vector3(0, 0, 0)); + + // Top === -1 + // Bottom === 1 + // Front === 0 + console.log(yAxis.dotProduct(cameraDirection)); + + // Scenarios we can view the cylinder from: + // 1. From the top/bottom (can't see the tube) + // 2. From the side (can't see the top or bottom) + // 3. From a diagonal (can see the tube and either the top or bottom) + + // Are we viewing the cylinder from the top or bottom? + + points.forEach(({ x, y }) => { + // Create a 'circle' element + const circle = document.createElementNS( + "http://www.w3.org/2000/svg", + "circle" + ); + + circle.id = "sphere"; + circle.setAttribute("cx", x.toString()); + circle.setAttribute("cy", y.toString()); + + // TODO: Factor in camera projection matrix, this currectly + // ignores all zoom factors. Can we even handle skew with sphere?! + // I don't think we can. + circle.setAttribute("r", "10"); + + circle.setAttribute("fill", ColorToCSS(cylinder.fill)); + + svg.appendChild(circle); + }); + + // Get the center of the cylinder's top face + + // Get the center of the cylinder's bottom face +} diff --git a/src/renderer/sphere.ts b/src/renderer/renderSphere.ts similarity index 97% rename from src/renderer/sphere.ts rename to src/renderer/renderSphere.ts index c91f906..966dc69 100644 --- a/src/renderer/sphere.ts +++ b/src/renderer/renderSphere.ts @@ -242,9 +242,6 @@ function sphereLightSide( circle.setAttribute("cx", x.toString()); circle.setAttribute("cy", y.toString()); - // TODO: Factor in camera projection matrix, this currectly - // ignores all zoom factors. Can we even handle skew with sphere?! - // I don't think we can. circle.setAttribute("r", Radius.toString()); circle.setAttribute("fill", fillUrl); @@ -424,9 +421,6 @@ function sphereDarkSide( circle.setAttribute("cx", x.toString()); circle.setAttribute("cy", y.toString()); - // TODO: Factor in camera projection matrix, this currectly - // ignores all zoom factors. Can we even handle skew with sphere?! - // I don't think we can. circle.setAttribute("r", Radius.toString()); circle.setAttribute("fill", fillUrl); diff --git a/src/shapes/Cylinder.ts b/src/shapes/Cylinder.ts index b86e5c6..54de89f 100644 --- a/src/shapes/Cylinder.ts +++ b/src/shapes/Cylinder.ts @@ -1,44 +1,26 @@ import { BasicShapeProperties, - DefaultShapeDimension, DefaultBasicShapeProperties, - Shape, + DefaultShapeDimension, + CylinderShape, } from "./Shape"; -import { CylinderMesh } from "../meshes/CylinderMesh"; export type CylinderProperties = { - segments: number; radius: number; height: number; -}; +} & BasicShapeProperties; -const DefaultCylinderProperties: CylinderProperties & { id: string } = { - segments: 32, +const DefaultSphereProperties: CylinderProperties = { radius: DefaultShapeDimension / 2, height: DefaultShapeDimension, - id: "cylinder", + ...DefaultBasicShapeProperties(), }; -export function Cylinder( - props: Partial -): Shape { - const meshParams: CylinderProperties = { - segments: props.segments || DefaultCylinderProperties.segments, - radius: props.radius || DefaultCylinderProperties.radius, - height: props.height || DefaultCylinderProperties.height, - }; - - const cylinder: Shape = { - type: "mesh", - mesh: CylinderMesh( - meshParams.radius, - meshParams.height, - meshParams.segments - ), - ...DefaultBasicShapeProperties(), - id: props.id || DefaultCylinderProperties.id, +export function Cylinder(props: Partial): CylinderShape { + const cylinder: CylinderShape = { + type: "cylinder", + ...DefaultSphereProperties, ...props, }; - return cylinder; } diff --git a/src/shapes/Shape.ts b/src/shapes/Shape.ts index a891ac6..2d06df0 100644 --- a/src/shapes/Shape.ts +++ b/src/shapes/Shape.ts @@ -41,6 +41,12 @@ export type SphereShape = { radius: number; } & BasicShapeProperties; +export type CylinderShape = { + type: "cylinder"; + radius: number; + height: number; +} & BasicShapeProperties; + export type GroupShape = TransformProperties & { type: "group"; id: string; @@ -55,4 +61,9 @@ export type GridShape = BasicShapeProperties & { cellSize: number; }; -export type Shape = MeshShape | SphereShape | GroupShape | GridShape; +export type Shape = + | MeshShape + | SphereShape + | CylinderShape + | GroupShape + | GridShape; diff --git a/workbench/main.ts b/workbench/main.ts index edd6a62..1ebb03f 100644 --- a/workbench/main.ts +++ b/workbench/main.ts @@ -6,12 +6,16 @@ import Spheres from "./scenes/Spheres"; import SingleSphere from "./scenes/SingleSphere"; import Worm from "./scenes/Worm"; import { getPaused, setPaused } from "./Settings"; +import Cylinders from "./scenes/Cylinders"; +import SingleCylinder from "./scenes/SingleCylinder"; // KitchenSink(); // Transforms(); // Octopus(); -Spheres(); +// Spheres(); +// Cylinders(); // SingleSphere(); +SingleCylinder(); // Worm(); document diff --git a/workbench/scenes/Cylinders.ts b/workbench/scenes/Cylinders.ts new file mode 100644 index 0000000..b989934 --- /dev/null +++ b/workbench/scenes/Cylinders.ts @@ -0,0 +1,114 @@ +import { + Scene, + Vector3, + Box, + Sphere, + Cylinder, + render, + Group, + Grid, + Color, +} from "../../src/index"; +import { Axii } from "../Axii"; +import { + getCamera, + getEnvironment, + getLighting, + getPaused, + onUpdate, +} from "../Settings"; + +const scenarios: [Vector3, string][] = [ + [Vector3(-1, -1, -3), "From Up, Right, and in Front"], + [Vector3(-1, -1, 3), "From Up, Right, and in Back"], + [Vector3(0, 0, -1), "From Front"], + [Vector3(0, 0, 1), "From Back"], + [Vector3(-1, 0, 0), "From Right"], + [Vector3(1, 0, 0), "From Left"], + [Vector3(-1, 0, 1), "From Right Behind"], + [Vector3(1, 0, 1), "From Left Behind"], + [Vector3(0, -1, 1), "From Top Behind"], + [Vector3(0, 1, -1), "From Bottom Front"], + [Vector3(0, 1, 1), "From Bottom Behind"], + [Vector3(-1, 0, 0.0001), "From Right Just Behind"], +]; + +export default function () { + const viewportScale = 1 / Math.floor(Math.sqrt(scenarios.length)); + + scenarios.forEach(([lightDirection, title]) => { + addCylinderScene(lightDirection, title); + }); +} + +function addCylinderScene(lightDirection: Vector3, title: string) { + const container = document.createElement("div"); + container.innerHTML = `

${title}

`; + container.style.display = "inline-block"; + const svgContainer = document.createElement("div"); + // container.style.transform = "translate(-50%, -50%) scale(2)"; //`translate(${scale},${scale}) scale(${scale})`; + svgContainer.style.display = "inline-block"; + + const { viewport, camera, updateCamera } = getCamera("front", 1); + + function resize() { + const viewportScale = 1 / Math.ceil(Math.sqrt(scenarios.length) * 1.5); + + svgContainer.style.width = + Math.floor(window.innerWidth * viewportScale - 1) + "px"; + svgContainer.style.height = + Math.floor(window.innerWidth * viewportScale - 1) + "px"; + + viewport.height = viewport.width; + + const cameraZoom = (viewport.width / 100) * 0.9; + camera.projectionMatrix.makeOrthographic( + 0, + viewport.width / cameraZoom, + 0, + viewport.height / cameraZoom, + 0, + 10000 + ); + } + resize(); + window.addEventListener("resize", resize); + + svgContainer.style.transform = `translate(0%,0%)`; //`translate(${scale},${scale}) scale(${scale})`; + svgContainer.className = "scene 2"; + container.appendChild(svgContainer); + + lightDirection.normalize(); + + const lightRadius = 15; + + const lightBall = Sphere({ + radius: lightRadius / 4, + fill: Color(255, 255, 0, 0), + stroke: Color(255, 255, 0), + strokeWidth: lightRadius / 2, + }); + lightBall.position = lightDirection.clone().multiply(-70); + + const cylinder = Cylinder({ id: title, strokeWidth: 0, height: 100 }); + const scene: Scene = { + ...getLighting("moonlit"), + shapes: [ + getEnvironment(), + // Axii(Vector3(-sphere.radius * 2, 0, 0)), + cylinder, + lightBall, + ], + }; + + scene.directionalLight.direction = lightDirection; + + onUpdate(({ now, deltaTime }) => { + const cameraSpeed = 0.0; + + updateCamera(now * cameraSpeed * 360 + 45, 20); + + render(svgContainer, scene, viewport, camera); + document.getElementById("root")!.appendChild(container); + }); +} diff --git a/workbench/scenes/KitchenSink.ts b/workbench/scenes/KitchenSink.ts index 71e59da..17027c5 100644 --- a/workbench/scenes/KitchenSink.ts +++ b/workbench/scenes/KitchenSink.ts @@ -104,7 +104,6 @@ export default function () { scale: 1.0, radius: 50, height: 300, - segments: 180, fill: Color(255, 0, 255), stroke: Color(0, 0, 0), strokeWidth: 0, @@ -171,7 +170,6 @@ export default function () { scale: 1.0, radius: 55, height: 1, - segments: 180, fill: Color(0, 0, 0, 0.5), stroke: Color(0, 0, 0, 0), strokeWidth: 0.0, @@ -241,7 +239,6 @@ export default function () { // scale: 1.0, // radius: 55, // height: 1, - // segments: 180, // fill: Color(0, 0, 0, 0.5), // stroke: Color(0, 0, 0, 0), // strokeWidth: 0.0, diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts new file mode 100644 index 0000000..f23e834 --- /dev/null +++ b/workbench/scenes/SingleCylinder.ts @@ -0,0 +1,145 @@ +import { + Scene, + Vector3, + Box, + Sphere, + Cylinder, + render, + Group, + Grid, + Color, +} from "../../src/index"; +import { + getCamera, + getEnvironment, + getLighting, + getPaused, + onUpdate, +} from "../Settings"; +import { Axii } from "../Axii"; +import { type } from "os"; + +export default function () { + const referenceRadius = 75; + + const lightSpeed = 0.3; + const lightDistance = 100; + const lightSphere = Sphere({ + // id: "light", + radius: 5, + fill: Color(255, 255, 0, 0), + stroke: Color(255, 255, 0), + strokeWidth: 10, + }); + + const position = Vector3(0, referenceRadius / 2, 0); + + const scene: Scene = { + ...getLighting("reference"), + shapes: [ + getEnvironment("grid"), + // Axii(Vector3(-referenceRadius * 3, 0, 0)), + // Group({ + // position: Vector3(0, 0, 0), + // rotation: Vector3(45, 0, 0), + // scale: 3, + // children: [ + Cylinder({ + id: "reference", + position, + radius: referenceRadius, + fill: Color(255, 0, 0), + stroke: Color(0, 0, 0), + strokeWidth: 0, + }), + // ], + // }), + lightSphere, + ], + }; + + lightSphere.position = Vector3(1, 1, -1); + + const { viewport, camera, updateCamera } = getCamera("front"); + + const onPointerEvent = (event: PointerEvent) => { + // return; + event.preventDefault(); + event.stopPropagation(); + + const centerX = window.innerWidth / 2; + const centerY = window.innerHeight / 2; + const diffX = event.clientX - centerX; + const diffY = event.clientY - centerY; + const distance = Math.sqrt(diffX * diffX + diffY * diffY); + + const distanceNormalized = distance / referenceRadius; + let degrees = distanceNormalized * 90; + + //Math.cos(((distanceNormalized * 90) / 180) * Math.PI); + // console.log( + // degrees, + // event.clientX, + // event.clientY, + // centerX, + // centerY, + // diffX, + // diffY, + // distance, + // distanceNormalized + // ); + + const spinMode: string = "z"; + + if (spinMode === "y") { + if (diffX < 0) { + degrees *= -1; + } + lightSphere.position.x = Math.sin((degrees / 180) * Math.PI); + // lightSphere.position.y = 0.0; + lightSphere.position.y = 0.5; + lightSphere.position.z = Math.cos((degrees / 180) * Math.PI); + + if (event.buttons === 1) { + console.log(); + lightSphere.position.z *= -1; + } + } else if (spinMode === "z") { + lightSphere.position.x = Math.sin((degrees / 180) * Math.PI); + lightSphere.position.y = Math.cos((degrees / 180) * Math.PI); + lightSphere.position.z = 0; // + // lightSphere.position.z = -0.5; // + } + + lightSphere.position.normalize().multiply(lightDistance).add(position); + + // const x = event.clientX; + // const z = event.clientY; + // lightSphere.position.y = 0; + // const x = event.clientX; + // const y = event.clientY; + }; + document.addEventListener("pointerdown", onPointerEvent); + document.addEventListener("pointermove", onPointerEvent); + document.addEventListener("pointerup", onPointerEvent); + + onUpdate(({ now, deltaTime }) => { + const cameraSpeed = 0.1; + // const cameraSpeed = 0.0; + updateCamera(now * cameraSpeed * 360 + 45, 20); + // updateCamera(45, 20); + + // lightSphere.position.x = + // Math.sin(now * Math.PI * 2 * lightSpeed) * lightDistance; + // lightSphere.position.y = 0; + // lightSphere.position.z = + // Math.cos(now * Math.PI * 2 * lightSpeed) * lightDistance; + + scene.directionalLight.direction = lightSphere.position + .clone() + .normalize() + .multiply(-1); + + render(document.getElementById("root")!, scene, viewport, camera); + }); +} From 4827d341bac70f7d6ac2954e66fa84bf80463a15 Mon Sep 17 00:00:00 2001 From: Francois Date: Wed, 20 Sep 2023 15:44:20 -0400 Subject: [PATCH 02/20] More cylinder with linear gradients work. Drew up diagrams of what the work is --- docs/cylinders.tldr | 802 +++++++++++++++++++++++++++ docs/{overview.tldr => spheres.tldr} | 475 +--------------- docs/test.tldr | 383 ------------- src/renderer/renderCylinder.ts | 5 + workbench/scenes/SingleCylinder.ts | 2 +- 5 files changed, 820 insertions(+), 847 deletions(-) create mode 100644 docs/cylinders.tldr rename docs/{overview.tldr => spheres.tldr} (95%) delete mode 100644 docs/test.tldr diff --git a/docs/cylinders.tldr b/docs/cylinders.tldr new file mode 100644 index 0000000..d67b319 --- /dev/null +++ b/docs/cylinders.tldr @@ -0,0 +1,802 @@ +{ + "tldrawFileFormatVersion": 1, + "schema": { + "schemaVersion": 1, + "storeVersion": 4, + "recordVersions": { + "asset": { + "version": 1, + "subTypeKey": "type", + "subTypeVersions": { + "image": 2, + "video": 2, + "bookmark": 0 + } + }, + "camera": { + "version": 1 + }, + "document": { + "version": 2 + }, + "instance": { + "version": 17 + }, + "instance_page_state": { + "version": 3 + }, + "page": { + "version": 1 + }, + "shape": { + "version": 3, + "subTypeKey": "type", + "subTypeVersions": { + "group": 0, + "embed": 4, + "bookmark": 1, + "image": 2, + "text": 1, + "draw": 1, + "geo": 7, + "line": 0, + "note": 4, + "frame": 0, + "arrow": 1, + "highlight": 0, + "video": 1 + } + }, + "instance_presence": { + "version": 4 + }, + "pointer": { + "version": 1 + } + } + }, + "records": [ + { + "gridSize": 10, + "name": "", + "meta": {}, + "id": "document:document", + "typeName": "document" + }, + { + "meta": {}, + "id": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "name": "Page 1", + "index": "a1", + "typeName": "page" + }, + { + "id": "pointer:pointer", + "typeName": "pointer", + "x": 773.1827019043085, + "y": -2307.7631842089286, + "lastActivityTimestamp": 1695235993865, + "meta": {} + }, + { + "followingUserId": null, + "opacityForNextShape": 1, + "stylesForNextShape": { + "tldraw:geo": "ellipse", + "tldraw:fill": "solid", + "tldraw:dash": "draw", + "tldraw:color": "grey", + "tldraw:size": "xl" + }, + "brush": null, + "scribble": null, + "cursor": { + "type": "default", + "color": "black", + "rotation": 0 + }, + "isFocusMode": false, + "exportBackground": true, + "isDebugMode": false, + "isToolLocked": false, + "screenBounds": { + "x": 0, + "y": 0, + "w": 1720, + "h": 1065 + }, + "zoomBrush": null, + "isGridMode": false, + "isPenMode": false, + "chatMessage": "", + "isChatting": false, + "highlightedUserIds": [], + "meta": {}, + "id": "instance:instance", + "currentPageId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "typeName": "instance" + }, + { + "editingId": null, + "croppingId": null, + "selectedIds": [], + "hoveredId": null, + "erasingIds": [], + "hintingIds": [], + "focusLayerId": null, + "meta": {}, + "id": "instance_page_state:page:uEZU1HSz7SqZ2Uxqq6Zuo", + "pageId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "typeName": "instance_page_state" + }, + { + "x": 3296.1532355956915, + "y": 4784.585999150335, + "z": 0.1, + "meta": {}, + "id": "camera:page:uEZU1HSz7SqZ2Uxqq6Zuo", + "typeName": "camera" + }, + { + "x": 3858.5470331978095, + "y": -1372.462245270935, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "type": "text", + "props": { + "color": "black", + "size": "xl", + "w": 215.8984375, + "text": "Scenarios", + "font": "draw", + "align": "middle", + "autoSize": true, + "scale": 3.252738929090746 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "aP", + "id": "shape:7RolEC3m4rJk4jn5-CAE4", + "typeName": "shape" + }, + { + "x": 2366.5432332143587, + "y": -332.016114953992, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "type": "text", + "props": { + "color": "black", + "size": "m", + "w": 25.1015625, + "text": "-X", + "font": "sans", + "align": "middle", + "autoSize": true, + "scale": 1.2830224378840631 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "aI", + "id": "shape:tu0K2uUZxn5A2oxnAuC_f", + "typeName": "shape" + }, + { + "x": 87.48047865185384, + "y": -1358.852479615269, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "type": "text", + "props": { + "color": "black", + "size": "xl", + "w": 1446.350694438507, + "text": "Coordinate Systems", + "font": "draw", + "align": "middle", + "autoSize": true, + "scale": 3.252738929090746 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "aPV", + "id": "shape:S928xs5O2UINqRJ4QevDP", + "typeName": "shape" + }, + { + "x": 7591.861850225016, + "y": -42.01651732141181, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "type": "image", + "props": { + "w": 6866.369673402156, + "h": 731.0203454572638, + "assetId": "asset:814849268", + "playing": true, + "url": "", + "crop": null + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "au", + "id": "shape:LQ7VYHfT_59lkzS-mOGnD", + "typeName": "shape" + }, + { + "x": 348.75694983770563, + "y": -873.4659494540831, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "type": "image", + "props": { + "w": 952.8005305002758, + "h": 844.8660954045414, + "assetId": "asset:8475689", + "playing": true, + "url": "", + "crop": null + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "aw", + "id": "shape:AQzu5e64eKuhuVxH4SdfJ", + "typeName": "shape" + }, + { + "x": 2980.7525610453717, + "y": -550.3093653822234, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:IjmZGGAMExMSzcUWacvS0", + "type": "geo", + "props": { + "w": 444.36654882128096, + "h": 1164.652194524181, + "geo": "rectangle", + "color": "black", + "labelColor": "black", + "fill": "semi", + "dash": "draw", + "size": "m", + "font": "draw", + "text": "", + "align": "middle", + "verticalAlign": "middle", + "growY": 0, + "url": "" + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "ax", + "typeName": "shape" + }, + { + "x": 3690.8819292540693, + "y": -331.73819635991754, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:Fc6bD2du9uFks5l3nqnaL", + "type": "geo", + "props": { + "w": 488.7984291145008, + "h": 488.7984291145008, + "geo": "ellipse", + "color": "black", + "labelColor": "black", + "fill": "solid", + "dash": "draw", + "size": "m", + "font": "draw", + "text": "", + "align": "middle", + "verticalAlign": "middle", + "growY": 0, + "url": "" + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "ay", + "typeName": "shape" + }, + { + "x": 0.23878404765400774, + "y": 158.11281483310268, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:vKGg_ZO4T0xqrJwEfYV7l", + "type": "geo", + "props": { + "w": 473.52385375144695, + "h": 1032.5372595488209, + "geo": "rectangle", + "color": "black", + "labelColor": "black", + "fill": "semi", + "dash": "draw", + "size": "m", + "font": "draw", + "text": "", + "align": "middle", + "verticalAlign": "middle", + "growY": 0, + "url": "" + }, + "parentId": "shape:w5gNAqCLUoJVqtvGBogvl", + "index": "a1", + "typeName": "shape" + }, + { + "x": -9.094947017729282e-13, + "y": -4.547473508864641e-13, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:2cdT5p8JrrciZOjXCzX8M", + "type": "geo", + "props": { + "w": 475.4490015797892, + "h": 282.081309634055, + "geo": "ellipse", + "color": "black", + "labelColor": "black", + "fill": "solid", + "dash": "dashed", + "size": "m", + "font": "draw", + "text": "", + "align": "middle", + "verticalAlign": "middle", + "growY": 0, + "url": "" + }, + "parentId": "shape:w5gNAqCLUoJVqtvGBogvl", + "index": "a2", + "typeName": "shape" + }, + { + "x": 2.1588002942544335, + "y": 727.6686127339904, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:sOMQZZel9vnX4Etlset7o", + "type": "image", + "props": { + "w": 468.53327820317276, + "h": 468.53327820317276, + "assetId": "asset:1151615746", + "playing": true, + "url": "", + "crop": null + }, + "parentId": "shape:w5gNAqCLUoJVqtvGBogvl", + "index": "a4", + "typeName": "shape" + }, + { + "x": 0.39714217349956016, + "y": 1054.8683325412799, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:wOhIGOyU6S9doZjnO7cVU", + "type": "geo", + "props": { + "w": 475.4490015797892, + "h": 282.081309634055, + "geo": "ellipse", + "color": "black", + "labelColor": "black", + "fill": "semi", + "dash": "draw", + "size": "m", + "font": "draw", + "text": "", + "align": "middle", + "verticalAlign": "middle", + "growY": 0, + "url": "" + }, + "parentId": "shape:w5gNAqCLUoJVqtvGBogvl", + "index": "a3", + "typeName": "shape" + }, + { + "x": 5884.552384773063, + "y": 490.1435920196451, + "rotation": 3.124139361069851, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:fzPp09k2Qt7VHNYKth32W", + "type": "geo", + "props": { + "w": 473.52385375144695, + "h": 1032.5372595488209, + "geo": "rectangle", + "color": "black", + "labelColor": "black", + "fill": "semi", + "dash": "draw", + "size": "m", + "font": "draw", + "text": "", + "align": "middle", + "verticalAlign": "middle", + "growY": 0, + "url": "" + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b038", + "typeName": "shape" + }, + { + "x": 5887.550581560161, + "y": 648.2281581489623, + "rotation": 3.124139361069851, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:APrjsUKRFoccLczrzHJcW", + "type": "geo", + "props": { + "w": 475.4490015797892, + "h": 282.081309634055, + "geo": "ellipse", + "color": "black", + "labelColor": "black", + "fill": "solid", + "dash": "dashed", + "size": "m", + "font": "draw", + "text": "", + "align": "middle", + "verticalAlign": "middle", + "growY": 0, + "url": "" + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b03G", + "typeName": "shape" + }, + { + "x": 5872.692541680559, + "y": -79.29195087061487, + "rotation": 3.124139361069851, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:sdZx3nc_EwfFUaA8095CX", + "type": "image", + "props": { + "w": 468.53327820317276, + "h": 468.53327820317276, + "assetId": "asset:1151615746", + "playing": true, + "url": "", + "crop": null + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b03l", + "typeName": "shape" + }, + { + "x": 5868.743508996007, + "y": -406.47258174927737, + "rotation": 3.124139361069851, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:eXL3GnmxAt9eoxQvo-U88", + "type": "geo", + "props": { + "w": 475.4490015797892, + "h": 282.081309634055, + "geo": "ellipse", + "color": "black", + "labelColor": "black", + "fill": "semi", + "dash": "draw", + "size": "m", + "font": "draw", + "text": "", + "align": "middle", + "verticalAlign": "middle", + "growY": 0, + "url": "" + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b03V", + "typeName": "shape" + }, + { + "x": 2982.2895389886444, + "y": -782.2091382395772, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:vaHxHh6pSpFP6tdcPitPg", + "type": "text", + "props": { + "color": "black", + "size": "xl", + "w": 221.2669219970703, + "text": "From Side", + "font": "draw", + "align": "middle", + "autoSize": true, + "scale": 1.8297611929228117 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b04", + "typeName": "shape" + }, + { + "x": 3722.1463530613637, + "y": -797.32841073072, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:34HMB1Cv3dxBFhGVgdKrH", + "type": "text", + "props": { + "color": "black", + "size": "xl", + "w": 375.07245453053696, + "text": "From Top", + "font": "draw", + "align": "middle", + "autoSize": true, + "scale": 1.8297611929228117 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b05", + "typeName": "shape" + }, + { + "x": 4454.33027664319, + "y": -963.3078028504594, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:741R1AM4WwU_yVoM0Bn-P", + "type": "text", + "props": { + "color": "black", + "size": "xl", + "w": 488.431878435833, + "text": "Angled \nFrom Above", + "font": "draw", + "align": "middle", + "autoSize": true, + "scale": 1.8297611929228117 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b06", + "typeName": "shape" + }, + { + "x": 4455.743382039469, + "y": -677.1986768240918, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:w5gNAqCLUoJVqtvGBogvl", + "type": "group", + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b02", + "props": {}, + "typeName": "shape" + }, + { + "x": 5388.4972957386335, + "y": -981.1381280595376, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:4ZX1sFKLXVKzXqv1t38no", + "type": "text", + "props": { + "color": "black", + "size": "xl", + "w": 478.68742846960436, + "text": "Angled \nFrom Below", + "font": "draw", + "align": "middle", + "autoSize": true, + "scale": 1.8297611929228117 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b07", + "typeName": "shape" + }, + { + "x": 7722.063997502218, + "y": -1050.236113218834, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:G_47b0sfuUUf5_vjQWdVe", + "type": "image", + "props": { + "w": 1352.2116903743579, + "h": 614.3381506885971, + "assetId": "asset:1338420211", + "playing": true, + "url": "", + "crop": null + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b08", + "typeName": "shape" + }, + { + "x": 9247.632871314963, + "y": -985.7024215698013, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:fQCgjcXEwXUPcY2HuJpCF", + "type": "image", + "props": { + "w": 1618.01170459689, + "h": 625.7694840428144, + "assetId": "asset:-1438593003", + "playing": true, + "url": "", + "crop": null + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b09", + "typeName": "shape" + }, + { + "x": 11127.309714090754, + "y": -1104.7579472038265, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:lpSC9yEOtgOKGwmROePFh", + "type": "image", + "props": { + "w": 720.145143970468, + "h": 686.0206191744817, + "assetId": "asset:43572479", + "playing": true, + "url": "", + "crop": null + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b0A", + "typeName": "shape" + }, + { + "x": 8610.659962213733, + "y": -1428.3187271738625, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:BE4q-5sIZwZkndABcLO3T", + "type": "text", + "props": { + "color": "black", + "size": "xl", + "w": 793.1430838033111, + "text": "References", + "font": "draw", + "align": "middle", + "autoSize": true, + "scale": 3.252738929090746 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "aQ", + "typeName": "shape" + }, + { + "type": "image", + "props": { + "name": "sine-cosine-graph.svg", + "src": "", + "w": 405, + "h": 184, + "mimeType": "image/svg+xml", + "isAnimated": false + }, + "meta": {}, + "id": "asset:1338420211", + "typeName": "asset" + }, + { + "type": "image", + "props": { + "name": "sin-cos-tan.svg", + "src": "", + "w": 468, + "h": 181, + "mimeType": "image/svg+xml", + "isAnimated": false + }, + "meta": {}, + "id": "asset:-1438593003", + "typeName": "asset" + }, + { + "type": "image", + "props": { + "name": "istockphoto-1411518273-612x612.jpg", + "src": "", + "w": 612, + "h": 583, + "mimeType": "image/jpeg", + "isAnimated": false + }, + "meta": {}, + "id": "asset:43572479", + "typeName": "asset" + }, + { + "type": "image", + "props": { + "name": "Screenshot 2023-08-10 at 9.54.06 AM.png", + "src": "", + "w": 1052, + "h": 112, + "mimeType": "image/png", + "isAnimated": false + }, + "meta": {}, + "id": "asset:814849268", + "typeName": "asset" + }, + { + "type": "image", + "props": { + "name": "coordinate-system.png", + "src": "", + "w": 256, + "h": 227, + "mimeType": "image/png", + "isAnimated": false + }, + "meta": {}, + "id": "asset:8475689", + "typeName": "asset" + }, + { + "id": "asset:1151615746", + "type": "image", + "typeName": "asset", + "props": { + "name": "pixel.png", + "src": "", + "w": 1, + "h": 1, + "mimeType": "image/png", + "isAnimated": false + }, + "meta": {} + } + ] +} \ No newline at end of file diff --git a/docs/overview.tldr b/docs/spheres.tldr similarity index 95% rename from docs/overview.tldr rename to docs/spheres.tldr index 8e9d60d..cacf97b 100644 --- a/docs/overview.tldr +++ b/docs/spheres.tldr @@ -73,9 +73,9 @@ { "id": "pointer:pointer", "typeName": "pointer", - "x": 2380.717073098672, - "y": -1882.200858564179, - "lastActivityTimestamp": 1693753541680, + "x": 8563.253390968892, + "y": -1486.0380517182352, + "lastActivityTimestamp": 1695236002633, "meta": {} }, { @@ -98,8 +98,8 @@ "screenBounds": { "x": 0, "y": 0, - "w": 1335, - "h": 972 + "w": 1720, + "h": 1065 }, "zoomBrush": null, "isGridMode": false, @@ -126,9 +126,9 @@ "typeName": "instance_page_state" }, { - "x": 1981.2627423281028, - "y": 2656.8088573779582, - "z": 0.1551187828992279, + "x": 1650.6137965311086, + "y": 4079.5927392182352, + "z": 0.1, "meta": {}, "id": "camera:page:uEZU1HSz7SqZ2Uxqq6Zuo", "typeName": "camera" @@ -4186,443 +4186,6 @@ "id": "shape:kS_xGSRxHq6kqgvcOoO_V", "typeName": "shape" }, - { - "x": 8217.057162725016, - "y": 1473.5108264285882, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "image", - "props": { - "w": 6866.369673402156, - "h": 731.0203454572638, - "assetId": "asset:814849268", - "playing": true, - "url": "", - "crop": null - }, - "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", - "index": "au", - "id": "shape:LQ7VYHfT_59lkzS-mOGnD", - "typeName": "shape" - }, - { - "x": 8272.167074723866, - "y": -231.22657301491654, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "geo", - "props": { - "w": 454.8896361744155, - "h": 1022.0345912535959, - "geo": "rectangle", - "color": "black", - "labelColor": "black", - "fill": "none", - "dash": "draw", - "size": "m", - "font": "draw", - "text": "", - "align": "middle", - "verticalAlign": "middle", - "growY": 0, - "url": "" - }, - "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", - "index": "av", - "id": "shape:CnwDcmII_t296x3lkKtFP", - "typeName": "shape" - }, - { - "x": 8904.07069829703, - "y": -817.6850953418113, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "text", - "props": { - "color": "black", - "size": "xl", - "w": 159.890625, - "text": "90 deg", - "font": "draw", - "align": "middle", - "autoSize": true, - "scale": 1 - }, - "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", - "index": "aZ8", - "id": "shape:PFe-0RRWKAKsdkGT4mKqr", - "typeName": "shape" - }, - { - "x": 8345.957313540965, - "y": -1049.8087810434195, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "text", - "props": { - "color": "black", - "size": "xl", - "w": 179.0703125, - "text": "180 deg", - "font": "draw", - "align": "middle", - "autoSize": true, - "scale": 1.0175342344114682 - }, - "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", - "index": "abf", - "id": "shape:o2Ft4PGowrSOr47-UamY9", - "typeName": "shape" - }, - { - "x": 67.8754837299798, - "y": 47.21755527336745, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "geo", - "props": { - "w": 696.8285122982894, - "h": 696.8285122982894, - "geo": "ellipse", - "color": "black", - "labelColor": "black", - "fill": "solid", - "dash": "draw", - "size": "m", - "font": "draw", - "text": "", - "align": "middle", - "verticalAlign": "middle", - "growY": 0, - "url": "" - }, - "parentId": "shape:WgCT0wkzM1R-BbbSbN6Ir", - "index": "a1V", - "id": "shape:TXO1JaRhTgSRoeYgx6MJh", - "typeName": "shape" - }, - { - "x": 46.28643898099608, - "y": 407.41411678032, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "arrow", - "parentId": "shape:WgCT0wkzM1R-BbbSbN6Ir", - "index": "a2V", - "props": { - "dash": "draw", - "size": "m", - "fill": "solid", - "color": "black", - "labelColor": "black", - "bend": 0, - "start": { - "type": "point", - "x": 0, - "y": 0 - }, - "end": { - "type": "point", - "x": 748.456731641716, - "y": 0.8493324275289126 - }, - "arrowheadStart": "arrow", - "arrowheadEnd": "arrow", - "text": "", - "font": "sans" - }, - "id": "shape:le363Xgw9ILO0dnCjvTq1", - "typeName": "shape" - }, - { - "x": 406.78981186913046, - "y": 19.967466381366876, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "arrow", - "parentId": "shape:WgCT0wkzM1R-BbbSbN6Ir", - "index": "a3V", - "props": { - "dash": "draw", - "size": "m", - "fill": "solid", - "color": "black", - "labelColor": "black", - "bend": 0, - "start": { - "type": "point", - "x": 0, - "y": 0 - }, - "end": { - "type": "point", - "x": -0.6459194614203337, - "y": 764.5199360694071 - }, - "arrowheadStart": "arrow", - "arrowheadEnd": "arrow", - "text": "", - "font": "draw" - }, - "id": "shape:QuPzuPNF8BNYYaWMI4AJk", - "typeName": "shape" - }, - { - "x": 429.885116346953, - "y": 760.4997480093755, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "text", - "props": { - "color": "black", - "size": "m", - "w": 20.52835900614501, - "text": "Y", - "font": "sans", - "align": "middle", - "autoSize": true, - "scale": 1.2830224378840631 - }, - "parentId": "shape:WgCT0wkzM1R-BbbSbN6Ir", - "index": "a4V", - "id": "shape:fqrkrap26fTpbnwoKkEoJ", - "typeName": "shape" - }, - { - "x": 418.7933362357916, - "y": 0, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "text", - "props": { - "color": "black", - "size": "m", - "w": 31.09324689309659, - "text": "-Y", - "font": "sans", - "align": "middle", - "autoSize": true, - "scale": 1.2830224378840631 - }, - "parentId": "shape:WgCT0wkzM1R-BbbSbN6Ir", - "index": "a5V", - "id": "shape:yLjtNV8toXx15TwY9GFiy", - "typeName": "shape" - }, - { - "x": 0, - "y": 388.3400678576322, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "text", - "props": { - "color": "black", - "size": "m", - "w": 25.1015625, - "text": "-X", - "font": "sans", - "align": "middle", - "autoSize": true, - "scale": 1.2830224378840631 - }, - "parentId": "shape:WgCT0wkzM1R-BbbSbN6Ir", - "index": "a6V", - "id": "shape:VITSKZNi6pCbkPITvWAjr", - "typeName": "shape" - }, - { - "x": 798.9936461431382, - "y": 387.8216090690489, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "text", - "props": { - "color": "black", - "size": "m", - "w": 20.969397969167655, - "text": "X", - "font": "sans", - "align": "middle", - "autoSize": true, - "scale": 1.2830224378840631 - }, - "parentId": "shape:WgCT0wkzM1R-BbbSbN6Ir", - "index": "a8", - "id": "shape:y5Z2bjJ_BaKo8LpviVLU0", - "typeName": "shape" - }, - { - "x": 8037.104238963189, - "y": -1182.1769350402412, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "group", - "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", - "index": "aIt4", - "props": {}, - "id": "shape:WgCT0wkzM1R-BbbSbN6Ir", - "typeName": "shape" - }, - { - "x": 7800.474579333158, - "y": -834.4879771612282, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "text", - "props": { - "color": "black", - "size": "xl", - "w": 188.2265625, - "text": "270 deg", - "font": "draw", - "align": "middle", - "autoSize": true, - "scale": 1.0175342344114682 - }, - "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", - "index": "abt", - "id": "shape:Xo4Cx1pX8FYQJjypdQFki", - "typeName": "shape" - }, - { - "x": 9349.520072241889, - "y": -258.163687206718, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "arrow", - "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", - "index": "aIt6", - "props": { - "dash": "draw", - "size": "xl", - "fill": "solid", - "color": "black", - "labelColor": "black", - "bend": 58.28815129448117, - "start": { - "type": "binding", - "boundShapeId": "shape:TXO1JaRhTgSRoeYgx6MJh", - "normalizedAnchor": { - "x": 0.49789846091471335, - "y": 0.7868841502035268 - }, - "isExact": true - }, - "end": { - "type": "point", - "x": -455.3650259982719, - "y": -523.5691446449982 - }, - "arrowheadStart": "bar", - "arrowheadEnd": "none", - "text": "", - "font": "mono" - }, - "id": "shape:2khfmpp7zg12MiCqN3No-", - "typeName": "shape" - }, - { - "x": 8876.887013340933, - "y": -811.6521537047781, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "arrow", - "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", - "index": "aItd", - "props": { - "dash": "draw", - "size": "xl", - "fill": "solid", - "color": "black", - "labelColor": "black", - "bend": 167.03745946457875, - "start": { - "type": "point", - "x": 15.542857136722432, - "y": 23.17594613894488 - }, - "end": { - "type": "point", - "x": -882.0964743464142, - "y": 19.747294931123463 - }, - "arrowheadStart": "none", - "arrowheadEnd": "none", - "text": "", - "font": "mono" - }, - "id": "shape:8teW8SFoyI0rL4OAlhmWR", - "typeName": "shape" - }, - { - "x": 7990.656627253442, - "y": -787.0494112562494, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "arrow", - "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", - "index": "aIt5", - "props": { - "dash": "draw", - "size": "xl", - "fill": "solid", - "color": "black", - "labelColor": "black", - "bend": 39.897221530040014, - "start": { - "type": "point", - "x": 0, - "y": 0 - }, - "end": { - "type": "binding", - "boundShapeId": "shape:TXO1JaRhTgSRoeYgx6MJh", - "normalizedAnchor": { - "x": 0.4721443941478094, - "y": 0.784641958175453 - }, - "isExact": true - }, - "arrowheadStart": "none", - "arrowheadEnd": "arrow", - "text": "", - "font": "mono" - }, - "id": "shape:79xux0QQ-sIUZ466x0Exj", - "typeName": "shape" - }, { "x": 1006.5364908533305, "y": 622.1724782802919, @@ -4630,7 +4193,6 @@ "isLocked": false, "opacity": 1, "meta": {}, - "id": "shape:AQzu5e64eKuhuVxH4SdfJ", "type": "image", "props": { "w": 952.8005305002758, @@ -4642,6 +4204,7 @@ }, "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", "index": "aw", + "id": "shape:AQzu5e64eKuhuVxH4SdfJ", "typeName": "shape" }, { @@ -4716,22 +4279,6 @@ }, { "type": "image", - "props": { - "name": "Screenshot 2023-08-10 at 9.54.06 AM.png", - "src": "", - "w": 1052, - "h": 112, - "mimeType": "image/png", - "isAnimated": false - }, - "meta": {}, - "id": "asset:814849268", - "typeName": "asset" - }, - { - "id": "asset:8475689", - "type": "image", - "typeName": "asset", "props": { "name": "coordinate-system.png", "src": "", @@ -4740,7 +4287,9 @@ "mimeType": "image/png", "isAnimated": false }, - "meta": {} + "meta": {}, + "id": "asset:8475689", + "typeName": "asset" } ] } \ No newline at end of file diff --git a/docs/test.tldr b/docs/test.tldr deleted file mode 100644 index 4e4fad5..0000000 --- a/docs/test.tldr +++ /dev/null @@ -1,383 +0,0 @@ -{ - "tldrawFileFormatVersion": 1, - "schema": { - "schemaVersion": 1, - "storeVersion": 4, - "recordVersions": { - "asset": { - "version": 1, - "subTypeKey": "type", - "subTypeVersions": { - "image": 2, - "video": 2, - "bookmark": 0 - } - }, - "camera": { - "version": 1 - }, - "document": { - "version": 2 - }, - "instance": { - "version": 17 - }, - "instance_page_state": { - "version": 3 - }, - "page": { - "version": 1 - }, - "shape": { - "version": 3, - "subTypeKey": "type", - "subTypeVersions": { - "group": 0, - "embed": 4, - "bookmark": 1, - "image": 2, - "text": 1, - "draw": 1, - "geo": 7, - "line": 0, - "note": 4, - "frame": 0, - "arrow": 1, - "highlight": 0, - "video": 1 - } - }, - "instance_presence": { - "version": 4 - }, - "pointer": { - "version": 1 - } - } - }, - "records": [ - { - "gridSize": 10, - "name": "", - "meta": {}, - "id": "document:document", - "typeName": "document" - }, - { - "id": "pointer:pointer", - "typeName": "pointer", - "x": 1323.078125, - "y": 743.875, - "lastActivityTimestamp": 1694612082026, - "meta": {} - }, - { - "meta": {}, - "id": "page:BOCl88s28IpD2VrJVSgCf", - "name": "Page 1", - "index": "a1", - "typeName": "page" - }, - { - "x": 0, - "y": 0, - "z": 1, - "meta": {}, - "id": "camera:page:BOCl88s28IpD2VrJVSgCf", - "typeName": "camera" - }, - { - "editingId": null, - "croppingId": null, - "selectedIds": [], - "hoveredId": null, - "erasingIds": [], - "hintingIds": [], - "focusLayerId": null, - "meta": {}, - "id": "instance_page_state:page:BOCl88s28IpD2VrJVSgCf", - "pageId": "page:BOCl88s28IpD2VrJVSgCf", - "typeName": "instance_page_state" - }, - { - "followingUserId": null, - "opacityForNextShape": 1, - "stylesForNextShape": { - "tldraw:geo": "ellipse", - "tldraw:fill": "semi" - }, - "brush": null, - "scribble": null, - "cursor": { - "type": "default", - "color": "black", - "rotation": 0 - }, - "isFocusMode": false, - "exportBackground": true, - "isDebugMode": false, - "isToolLocked": false, - "screenBounds": { - "x": 0, - "y": 0, - "w": 2265, - "h": 1308 - }, - "zoomBrush": null, - "isGridMode": false, - "isPenMode": false, - "chatMessage": "", - "isChatting": false, - "highlightedUserIds": [], - "meta": {}, - "id": "instance:instance", - "currentPageId": "page:BOCl88s28IpD2VrJVSgCf", - "typeName": "instance" - }, - { - "x": 650.796875, - "y": 386.6328125, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "id": "shape:Z96x88IKDjkhGxn0NXPDW", - "type": "geo", - "props": { - "w": 184.109375, - "h": 146.57421875, - "geo": "rectangle", - "color": "black", - "labelColor": "black", - "fill": "semi", - "dash": "draw", - "size": "m", - "font": "draw", - "text": "Start", - "align": "middle", - "verticalAlign": "middle", - "growY": 0, - "url": "" - }, - "parentId": "page:BOCl88s28IpD2VrJVSgCf", - "index": "a1", - "typeName": "shape" - }, - { - "x": 995.734375, - "y": 352.6640625, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "id": "shape:of1Kr9dzBYlOrolSGLjNN", - "type": "geo", - "props": { - "w": 278.625, - "h": 240.74609375, - "geo": "rectangle", - "color": "black", - "labelColor": "black", - "fill": "semi", - "dash": "draw", - "size": "m", - "font": "draw", - "text": "Middle", - "align": "middle", - "verticalAlign": "middle", - "growY": 0, - "url": "" - }, - "parentId": "page:BOCl88s28IpD2VrJVSgCf", - "index": "a2", - "typeName": "shape" - }, - { - "x": 744.515625, - "y": 552.5625, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "id": "shape:QjsyMlTJy279E7lRS01sa", - "type": "arrow", - "parentId": "page:BOCl88s28IpD2VrJVSgCf", - "index": "a3", - "props": { - "dash": "draw", - "size": "m", - "fill": "none", - "color": "black", - "labelColor": "black", - "bend": 0, - "start": { - "type": "binding", - "boundShapeId": "shape:Z96x88IKDjkhGxn0NXPDW", - "normalizedAnchor": { - "x": 0.5, - "y": 0.5 - }, - "isExact": false - }, - "end": { - "type": "binding", - "boundShapeId": "shape:of1Kr9dzBYlOrolSGLjNN", - "normalizedAnchor": { - "x": 0.5, - "y": 0.5 - }, - "isExact": false - }, - "arrowheadStart": "none", - "arrowheadEnd": "arrow", - "text": "", - "font": "draw" - }, - "typeName": "shape" - }, - { - "x": 1468.46875, - "y": 437.19921875, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "id": "shape:cXBMkFoK7U62Dbcsj9fw7", - "type": "geo", - "props": { - "w": 278.625, - "h": 240.74609375, - "geo": "rectangle", - "color": "black", - "labelColor": "black", - "fill": "semi", - "dash": "draw", - "size": "m", - "font": "draw", - "text": "End", - "align": "middle", - "verticalAlign": "middle", - "growY": 0, - "url": "" - }, - "parentId": "page:BOCl88s28IpD2VrJVSgCf", - "index": "a2V", - "typeName": "shape" - }, - { - "x": 1167.6328125, - "y": 420.6953125, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "id": "shape:9O1KB_ryC5dL9ggheEi9g", - "type": "arrow", - "parentId": "page:BOCl88s28IpD2VrJVSgCf", - "index": "a2l", - "props": { - "dash": "draw", - "size": "m", - "fill": "semi", - "color": "black", - "labelColor": "black", - "bend": 0, - "start": { - "type": "binding", - "boundShapeId": "shape:of1Kr9dzBYlOrolSGLjNN", - "normalizedAnchor": { - "x": 0.5, - "y": 0.5 - }, - "isExact": false - }, - "end": { - "type": "binding", - "boundShapeId": "shape:cXBMkFoK7U62Dbcsj9fw7", - "normalizedAnchor": { - "x": 0.5, - "y": 0.5 - }, - "isExact": false - }, - "arrowheadStart": "none", - "arrowheadEnd": "arrow", - "text": "", - "font": "draw" - }, - "typeName": "shape" - }, - { - "x": 994.8359375, - "y": 769.7578125, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "id": "shape:xc8hARLSRwGfnJ3yUzY91", - "type": "geo", - "props": { - "w": 249, - "h": 248.8671875, - "geo": "ellipse", - "color": "black", - "labelColor": "black", - "fill": "semi", - "dash": "draw", - "size": "m", - "font": "draw", - "text": "Off the beaten path", - "align": "middle", - "verticalAlign": "middle", - "growY": 0, - "url": "" - }, - "parentId": "page:BOCl88s28IpD2VrJVSgCf", - "index": "a4", - "typeName": "shape" - }, - { - "x": 1124.296875, - "y": 522.0078125, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "id": "shape:VmU06a1JHpgle52-BBMLR", - "type": "arrow", - "parentId": "page:BOCl88s28IpD2VrJVSgCf", - "index": "a5", - "props": { - "dash": "draw", - "size": "m", - "fill": "semi", - "color": "black", - "labelColor": "black", - "bend": 0, - "start": { - "type": "binding", - "boundShapeId": "shape:of1Kr9dzBYlOrolSGLjNN", - "normalizedAnchor": { - "x": 0.5, - "y": 0.5 - }, - "isExact": false - }, - "end": { - "type": "binding", - "boundShapeId": "shape:xc8hARLSRwGfnJ3yUzY91", - "normalizedAnchor": { - "x": 0.5, - "y": 0.5 - }, - "isExact": false - }, - "arrowheadStart": "none", - "arrowheadEnd": "arrow", - "text": "", - "font": "draw" - }, - "typeName": "shape" - } - ] -} \ No newline at end of file diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index 1ae2c07..f47923b 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -12,6 +12,11 @@ import { CylinderShape } from "../shapes/Shape"; import { Scene } from "./Scene"; import { Viewport } from "./Viewport"; +enum CylinderEnds { + Top = 0, + Bottom = 1, +} + export function renderCylinder( _scene: Scene, svg: SVGElement, diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index f23e834..c7a756b 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -60,7 +60,7 @@ export default function () { lightSphere.position = Vector3(1, 1, -1); - const { viewport, camera, updateCamera } = getCamera("front"); + const { viewport, camera, updateCamera } = getCamera("isometric"); const onPointerEvent = (event: PointerEvent) => { // return; From bd18416f036a3e85ada95cd6c5d3f65da05dc973 Mon Sep 17 00:00:00 2001 From: Francois Date: Wed, 20 Sep 2023 16:00:18 -0400 Subject: [PATCH 03/20] More progress on cylinders. Adding some debugging visualization --- src/renderer/renderCylinder.ts | 59 ++++++++++++++++++++++++++---- workbench/scenes/SingleCylinder.ts | 26 ++++++++----- 2 files changed, 67 insertions(+), 18 deletions(-) diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index f47923b..63c6ade 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -4,7 +4,7 @@ // https://observablehq.com/d/011f054fc7eaf966 import { projectToScreenCoordinate } from "../cameras/Camera"; -import { ColorToCSS } from "../colors/Color"; +import { Color, ColorToCSS } from "../colors/Color"; import { applyLighting } from "../lighting/LightingModel"; import { Matrix4x4 } from "../math/Matrix4x4"; import { Vector3 } from "../math/Vector3"; @@ -24,7 +24,7 @@ export function renderCylinder( cylinder: CylinderShape, viewport: Viewport, worldTransform: Matrix4x4, - _cameraZoom: number, + cameraZoom: number, cameraDirection: Vector3, _inverseCameraMatrix: Matrix4x4, inverseAndProjectionMatrix: Matrix4x4 @@ -47,10 +47,51 @@ export function renderCylinder( const yAxis = Vector3(0, 1, 0); worldTransform.extractBasis(Vector3(0, 0, 0), yAxis, Vector3(0, 0, 0)); + const addCylinderEnd = ( + { x, y }: Vector3, + radius: number, + dotProduct: number, + fill: Color + ) => { + const dotProductAbsolute = Math.abs(dotProduct); + // Create a 'circle' element + const circle = document.createElementNS( + "http://www.w3.org/2000/svg", + "ellipse" + ); + + circle.id = "sphere"; + circle.setAttribute("cx", x.toString()); + circle.setAttribute("cy", y.toString()); + + // TODO: Factor in camera projection matrix, this currectly + // ignores all zoom factors. Can we even handle skew with sphere?! + // I don't think we can. + circle.setAttribute("rx", radius.toString()); + circle.setAttribute("ry", (radius * dotProductAbsolute).toString()); + + circle.setAttribute("fill", ColorToCSS(fill)); + + svg.appendChild(circle); + }; + + const cylinderScale = worldTransform.getScale().x; + const cylinderScaleFactor = cylinderScale * cameraZoom; + const Radius = cylinder.radius * cylinderScaleFactor; + // Top === -1 - // Bottom === 1 // Front === 0 - console.log(yAxis.dotProduct(cameraDirection)); + // Bottom === 1 + const dotProduct = yAxis.dotProduct(cameraDirection); + console.log(dotProduct); + const isTopVisible = yAxis.dotProduct(cameraDirection) > 0; + + addCylinderEnd( + isTopVisible ? points[CylinderEnds.Top] : points[CylinderEnds.Bottom], + cylinder.radius, + dotProduct, + isTopVisible ? Color(255, 0, 0) : Color(0, 0, 255) + ); // Scenarios we can view the cylinder from: // 1. From the top/bottom (can't see the tube) @@ -59,8 +100,7 @@ export function renderCylinder( // Are we viewing the cylinder from the top or bottom? - points.forEach(({ x, y }) => { - // Create a 'circle' element + points.forEach(({ x, y }, index) => { const circle = document.createElementNS( "http://www.w3.org/2000/svg", "circle" @@ -73,9 +113,12 @@ export function renderCylinder( // TODO: Factor in camera projection matrix, this currectly // ignores all zoom factors. Can we even handle skew with sphere?! // I don't think we can. - circle.setAttribute("r", "10"); + circle.setAttribute("r", (5).toString()); - circle.setAttribute("fill", ColorToCSS(cylinder.fill)); + circle.setAttribute( + "fill", + index === CylinderEnds.Top ? "rgb(128,0,0)" : "rgb(0,0,128)" + ); svg.appendChild(circle); }); diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index c7a756b..e550b47 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -20,7 +20,7 @@ import { Axii } from "../Axii"; import { type } from "os"; export default function () { - const referenceRadius = 75; + const referenceRadius = 50; const lightSpeed = 0.3; const lightDistance = 100; @@ -33,6 +33,16 @@ export default function () { }); const position = Vector3(0, referenceRadius / 2, 0); + const cylinder = Cylinder({ + id: "reference", + position, + radius: referenceRadius, + height: referenceRadius * 4, + // radius: referenceRadius, + fill: Color(255, 0, 0), + stroke: Color(0, 0, 0), + strokeWidth: 0, + }); const scene: Scene = { ...getLighting("reference"), @@ -44,14 +54,7 @@ export default function () { // rotation: Vector3(45, 0, 0), // scale: 3, // children: [ - Cylinder({ - id: "reference", - position, - radius: referenceRadius, - fill: Color(255, 0, 0), - stroke: Color(0, 0, 0), - strokeWidth: 0, - }), + cylinder, // ], // }), lightSphere, @@ -60,7 +63,7 @@ export default function () { lightSphere.position = Vector3(1, 1, -1); - const { viewport, camera, updateCamera } = getCamera("isometric"); + const { viewport, camera, updateCamera } = getCamera("top"); const onPointerEvent = (event: PointerEvent) => { // return; @@ -127,8 +130,11 @@ export default function () { const cameraSpeed = 0.1; // const cameraSpeed = 0.0; updateCamera(now * cameraSpeed * 360 + 45, 20); + // updateCamera(45, 20); + cylinder.rotation.x = now * 90; + // lightSphere.position.x = // Math.sin(now * Math.PI * 2 * lightSpeed) * lightDistance; // lightSphere.position.y = 0; From ba0d0ce12521718c64def5a4ff3cb7ec5b20c543 Mon Sep 17 00:00:00 2001 From: Francois Date: Thu, 21 Sep 2023 08:48:08 -0400 Subject: [PATCH 04/20] Added more debug logging for cylinder rendering --- src/renderer/renderCylinder.ts | 25 +++++++++++++++++++------ workbench/scenes/SingleCylinder.ts | 9 ++++++--- 2 files changed, 25 insertions(+), 9 deletions(-) diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index 63c6ade..7d6b8ff 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -26,7 +26,7 @@ export function renderCylinder( worldTransform: Matrix4x4, cameraZoom: number, cameraDirection: Vector3, - _inverseCameraMatrix: Matrix4x4, + inverseCameraMatrix: Matrix4x4, inverseAndProjectionMatrix: Matrix4x4 ) { const points: Vector3[] = [ @@ -44,8 +44,14 @@ export function renderCylinder( ); }); - const yAxis = Vector3(0, 1, 0); - worldTransform.extractBasis(Vector3(0, 0, 0), yAxis, Vector3(0, 0, 0)); + const yAxisWorldSpace = Vector3(0, 0, 0); + worldTransform.extractBasis( + Vector3(0, 0, 0), + yAxisWorldSpace, + Vector3(0, 0, 0) + ); + const yAxisCameraSpace = yAxisWorldSpace.clone(); + inverseCameraMatrix.applyToVector3(yAxisCameraSpace); const addCylinderEnd = ( { x, y }: Vector3, @@ -82,9 +88,16 @@ export function renderCylinder( // Top === -1 // Front === 0 // Bottom === 1 - const dotProduct = yAxis.dotProduct(cameraDirection); - console.log(dotProduct); - const isTopVisible = yAxis.dotProduct(cameraDirection) > 0; + const dotProduct = yAxisWorldSpace.dotProduct( + cameraDirection.clone().multiply(-1) + ); + console.log( + `dotProduct: ${dotProduct.toFixed(3)} + yAxisCameraSpace: ${yAxisCameraSpace.x.toFixed( + 2 + )}, ${yAxisCameraSpace.y.toFixed(2)}, ${yAxisCameraSpace.z.toFixed(2)}` + ); + const isTopVisible = dotProduct > 0; addCylinderEnd( isTopVisible ? points[CylinderEnds.Top] : points[CylinderEnds.Bottom], diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index e550b47..e3636b6 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -32,12 +32,13 @@ export default function () { strokeWidth: 10, }); - const position = Vector3(0, referenceRadius / 2, 0); + const height = referenceRadius * 4; + const position = Vector3(0, height / 2, 0); const cylinder = Cylinder({ id: "reference", position, radius: referenceRadius, - height: referenceRadius * 4, + height, // radius: referenceRadius, fill: Color(255, 0, 0), stroke: Color(0, 0, 0), @@ -133,7 +134,9 @@ export default function () { // updateCamera(45, 20); - cylinder.rotation.x = now * 90; + // cylinder.rotation.x = now * 90; + cylinder.rotation.x = 0; + cylinder.rotation.z = 0; // lightSphere.position.x = // Math.sin(now * Math.PI * 2 * lightSpeed) * lightDistance; From a193dafe4865164c8bfc36071030de0f2d118cc3 Mon Sep 17 00:00:00 2001 From: Francois Date: Thu, 21 Sep 2023 08:58:39 -0400 Subject: [PATCH 05/20] ... --- docs/cylinders.tldr | 281 +++++++++++++++++++++++++++++++++----------- 1 file changed, 213 insertions(+), 68 deletions(-) diff --git a/docs/cylinders.tldr b/docs/cylinders.tldr index d67b319..ecb930f 100644 --- a/docs/cylinders.tldr +++ b/docs/cylinders.tldr @@ -73,9 +73,9 @@ { "id": "pointer:pointer", "typeName": "pointer", - "x": 773.1827019043085, - "y": -2307.7631842089286, - "lastActivityTimestamp": 1695235993865, + "x": 3300.3676672089873, + "y": -440.83875562093954, + "lastActivityTimestamp": 1695301096298, "meta": {} }, { @@ -85,8 +85,9 @@ "tldraw:geo": "ellipse", "tldraw:fill": "solid", "tldraw:dash": "draw", - "tldraw:color": "grey", - "tldraw:size": "xl" + "tldraw:color": "red", + "tldraw:size": "xl", + "tldraw:font": "sans" }, "brush": null, "scribble": null, @@ -102,7 +103,7 @@ "screenBounds": { "x": 0, "y": 0, - "w": 1720, + "w": 653.3333129882812, "h": 1065 }, "zoomBrush": null, @@ -130,16 +131,16 @@ "typeName": "instance_page_state" }, { - "x": 3296.1532355956915, - "y": 4784.585999150335, - "z": 0.1, + "x": -1255.7672860927873, + "y": 2041.450273116584, + "z": 0.1590253146762286, "meta": {}, "id": "camera:page:uEZU1HSz7SqZ2Uxqq6Zuo", "typeName": "camera" }, { - "x": 3858.5470331978095, - "y": -1372.462245270935, + "x": 6181.257219304578, + "y": -1673.0204261600752, "rotation": 0, "isLocked": false, "opacity": 1, @@ -160,29 +161,6 @@ "id": "shape:7RolEC3m4rJk4jn5-CAE4", "typeName": "shape" }, - { - "x": 2366.5432332143587, - "y": -332.016114953992, - "rotation": 0, - "isLocked": false, - "opacity": 1, - "meta": {}, - "type": "text", - "props": { - "color": "black", - "size": "m", - "w": 25.1015625, - "text": "-X", - "font": "sans", - "align": "middle", - "autoSize": true, - "scale": 1.2830224378840631 - }, - "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", - "index": "aI", - "id": "shape:tu0K2uUZxn5A2oxnAuC_f", - "typeName": "shape" - }, { "x": 87.48047865185384, "y": -1358.852479615269, @@ -207,8 +185,8 @@ "typeName": "shape" }, { - "x": 7591.861850225016, - "y": -42.01651732141181, + "x": 3135.3461131690756, + "y": 877.1664221652654, "rotation": 0, "isLocked": false, "opacity": 1, @@ -249,8 +227,8 @@ "typeName": "shape" }, { - "x": 2980.7525610453717, - "y": -550.3093653822234, + "x": 6767.541785702926, + "y": -831.4130302999579, "rotation": 0, "isLocked": false, "opacity": 1, @@ -278,8 +256,8 @@ "typeName": "shape" }, { - "x": 3690.8819292540693, - "y": -331.73819635991754, + "x": 7477.671153911623, + "y": -612.841861277652, "rotation": 0, "isLocked": false, "opacity": 1, @@ -415,8 +393,8 @@ "typeName": "shape" }, { - "x": 5884.552384773063, - "y": 490.1435920196451, + "x": 6345.161887298109, + "y": 274.58703131214236, "rotation": 3.124139361069851, "isLocked": false, "opacity": 1, @@ -444,8 +422,8 @@ "typeName": "shape" }, { - "x": 5887.550581560161, - "y": 648.2281581489623, + "x": 6348.160084085207, + "y": 432.67159744145954, "rotation": 3.124139361069851, "isLocked": false, "opacity": 1, @@ -473,8 +451,8 @@ "typeName": "shape" }, { - "x": 5872.692541680559, - "y": -79.29195087061487, + "x": 6333.302044205605, + "y": -294.8485115781176, "rotation": 3.124139361069851, "isLocked": false, "opacity": 1, @@ -494,8 +472,8 @@ "typeName": "shape" }, { - "x": 5868.743508996007, - "y": -406.47258174927737, + "x": 6329.353011521053, + "y": -622.0291424567802, "rotation": 3.124139361069851, "isLocked": false, "opacity": 1, @@ -523,8 +501,8 @@ "typeName": "shape" }, { - "x": 2982.2895389886444, - "y": -782.2091382395772, + "x": 6775.411857360511, + "y": -1180.8589049499938, "rotation": 0, "isLocked": false, "opacity": 1, @@ -546,8 +524,8 @@ "typeName": "shape" }, { - "x": 3722.1463530613637, - "y": -797.32841073072, + "x": 7386.153369477026, + "y": -1235.7827448681776, "rotation": 0, "isLocked": false, "opacity": 1, @@ -557,8 +535,8 @@ "props": { "color": "black", "size": "xl", - "w": 375.07245453053696, - "text": "From Top", + "w": 620.6368710143194, + "text": "From \nStraight Above", "font": "draw", "align": "middle", "autoSize": true, @@ -569,8 +547,8 @@ "typeName": "shape" }, { - "x": 4454.33027664319, - "y": -963.3078028504594, + "x": 4914.939779168236, + "y": -1196.6946887670404, "rotation": 0, "isLocked": false, "opacity": 1, @@ -592,8 +570,8 @@ "typeName": "shape" }, { - "x": 4455.743382039469, - "y": -677.1986768240918, + "x": 4916.352884564515, + "y": -892.7552375315945, "rotation": 0, "isLocked": false, "opacity": 1, @@ -606,8 +584,8 @@ "typeName": "shape" }, { - "x": 5388.4972957386335, - "y": -981.1381280595376, + "x": 5849.10679826368, + "y": -1196.6946887670404, "rotation": 0, "isLocked": false, "opacity": 1, @@ -629,8 +607,8 @@ "typeName": "shape" }, { - "x": 7722.063997502218, - "y": -1050.236113218834, + "x": -26.210843806376033, + "y": 2351.423737855385, "rotation": 0, "isLocked": false, "opacity": 1, @@ -650,8 +628,8 @@ "typeName": "shape" }, { - "x": 9247.632871314963, - "y": -985.7024215698013, + "x": 1499.3580300063695, + "y": 2415.9574295044176, "rotation": 0, "isLocked": false, "opacity": 1, @@ -671,8 +649,8 @@ "typeName": "shape" }, { - "x": 11127.309714090754, - "y": -1104.7579472038265, + "x": 3379.03487278216, + "y": 2296.9019038703923, "rotation": 0, "isLocked": false, "opacity": 1, @@ -692,8 +670,8 @@ "typeName": "shape" }, { - "x": 8610.659962213733, - "y": -1428.3187271738625, + "x": 862.385120905139, + "y": 1973.3411239003563, "rotation": 0, "isLocked": false, "opacity": 1, @@ -714,6 +692,173 @@ "index": "aQ", "typeName": "shape" }, + { + "x": 8203.76888615956, + "y": -1235.7827448681776, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:bgh63thztH2bYAvA6UZ3C", + "type": "text", + "props": { + "color": "black", + "size": "xl", + "w": 610.8924210480907, + "text": "From \nStraight Below", + "font": "draw", + "align": "middle", + "autoSize": true, + "scale": 1.8297611929228117 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b05V", + "typeName": "shape" + }, + { + "x": 8268.846148538236, + "y": -604.9700883978435, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:i-6lP7EMT-IeqVRwbASox", + "type": "geo", + "props": { + "w": 488.7984291145008, + "h": 488.7984291145008, + "geo": "ellipse", + "color": "black", + "labelColor": "black", + "fill": "solid", + "dash": "draw", + "size": "m", + "font": "draw", + "text": "", + "align": "middle", + "verticalAlign": "middle", + "growY": 0, + "url": "" + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "az", + "typeName": "shape" + }, + { + "x": 5058.126092713332, + "y": 534.0002813253899, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:oHSpWqScCyP0Pda-kaDk7", + "type": "text", + "props": { + "color": "grey", + "size": "xl", + "w": 24.16666603088379, + "text": "✅", + "font": "sans", + "align": "middle", + "autoSize": true, + "scale": 3.548562932484002 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b0C", + "typeName": "shape" + }, + { + "x": 6883.078480824545, + "y": 379.4034835066593, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:wSxS6zWO_zXVOpKC-MG2l", + "type": "text", + "props": { + "color": "red", + "size": "xl", + "w": 24.16666603088379, + "text": "❌", + "font": "sans", + "align": "middle", + "autoSize": true, + "scale": 4.389068440319015 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b0D", + "typeName": "shape" + }, + { + "x": 7635.694000188957, + "y": -113.57575017599834, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:uiKCOBK_9QM_BKFsn2uwa", + "type": "text", + "props": { + "color": "red", + "size": "xl", + "w": 24.16666603088379, + "text": "❌", + "font": "sans", + "align": "middle", + "autoSize": true, + "scale": 4.389068440319015 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b0E", + "typeName": "shape" + }, + { + "x": 8420.972095253725, + "y": -118.31345965366813, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:mEbKCOTFK6zknDlox3qgo", + "type": "text", + "props": { + "color": "red", + "size": "xl", + "w": 24.16666603088379, + "text": "❌", + "font": "sans", + "align": "middle", + "autoSize": true, + "scale": 4.389068440319015 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b0F", + "typeName": "shape" + }, + { + "x": 6031.655272338277, + "y": 536.3090911255997, + "rotation": 0, + "isLocked": false, + "opacity": 1, + "meta": {}, + "id": "shape:PmXIjAiao6Knx6WtLrXZE", + "type": "text", + "props": { + "color": "grey", + "size": "xl", + "w": 24.16666603088379, + "text": "✅", + "font": "sans", + "align": "middle", + "autoSize": true, + "scale": 3.548562932484002 + }, + "parentId": "page:uEZU1HSz7SqZ2Uxqq6Zuo", + "index": "b0CV", + "typeName": "shape" + }, { "type": "image", "props": { From bede4449093690b6ee23edb4ffe6b567a6d22f0b Mon Sep 17 00:00:00 2001 From: Francois Date: Sun, 24 Sep 2023 15:04:22 -0400 Subject: [PATCH 06/20] Start of Cylinder working --- src/math/Vector3.ts | 18 +++ src/renderer/renderCylinder.ts | 190 +++++++++++++++++++++-------- workbench/scenes/SingleCylinder.ts | 6 +- 3 files changed, 160 insertions(+), 54 deletions(-) diff --git a/src/math/Vector3.ts b/src/math/Vector3.ts index 8a64487..f7aa63c 100644 --- a/src/math/Vector3.ts +++ b/src/math/Vector3.ts @@ -3,6 +3,9 @@ export interface Vector3 { y: number; z: number; set(vec: Vector3): Vector3; + setX(x: number): Vector3; + setY(y: number): Vector3; + setZ(z: number): Vector3; add(vec: Vector3): Vector3; subtract(vec: Vector3): Vector3; multiply(scalar: number): Vector3; @@ -26,6 +29,21 @@ const Vector3Proto = { return this; }, + setX(this: Vector3, x: number): Vector3 { + this.x = x; + return this; + }, + + setY(this: Vector3, y: number): Vector3 { + this.y = y; + return this; + }, + + setZ(this: Vector3, z: number): Vector3 { + this.z = z; + return this; + }, + /** * Adds a vector to this vector, mutating it in place. It * returns this vector, so that API chaining is possible (ie: `v.add(v2).add(v3)`) diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index 7d6b8ff..d7f28ec 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -18,7 +18,7 @@ enum CylinderEnds { } export function renderCylinder( - _scene: Scene, + scene: Scene, svg: SVGElement, _defs: SVGDefsElement, cylinder: CylinderShape, @@ -53,34 +53,6 @@ export function renderCylinder( const yAxisCameraSpace = yAxisWorldSpace.clone(); inverseCameraMatrix.applyToVector3(yAxisCameraSpace); - const addCylinderEnd = ( - { x, y }: Vector3, - radius: number, - dotProduct: number, - fill: Color - ) => { - const dotProductAbsolute = Math.abs(dotProduct); - // Create a 'circle' element - const circle = document.createElementNS( - "http://www.w3.org/2000/svg", - "ellipse" - ); - - circle.id = "sphere"; - circle.setAttribute("cx", x.toString()); - circle.setAttribute("cy", y.toString()); - - // TODO: Factor in camera projection matrix, this currectly - // ignores all zoom factors. Can we even handle skew with sphere?! - // I don't think we can. - circle.setAttribute("rx", radius.toString()); - circle.setAttribute("ry", (radius * dotProductAbsolute).toString()); - - circle.setAttribute("fill", ColorToCSS(fill)); - - svg.appendChild(circle); - }; - const cylinderScale = worldTransform.getScale().x; const cylinderScaleFactor = cylinderScale * cameraZoom; const Radius = cylinder.radius * cylinderScaleFactor; @@ -91,6 +63,10 @@ export function renderCylinder( const dotProduct = yAxisWorldSpace.dotProduct( cameraDirection.clone().multiply(-1) ); + const dotProductAbsolute = Math.abs(dotProduct); + + const ShortRadius = Radius * dotProductAbsolute; + console.log( `dotProduct: ${dotProduct.toFixed(3)} yAxisCameraSpace: ${yAxisCameraSpace.x.toFixed( @@ -99,12 +75,90 @@ export function renderCylinder( ); const isTopVisible = dotProduct > 0; - addCylinderEnd( - isTopVisible ? points[CylinderEnds.Top] : points[CylinderEnds.Bottom], - cylinder.radius, - dotProduct, - isTopVisible ? Color(255, 0, 0) : Color(0, 0, 255) + const yAxisScreenSpace = Vector3(yAxisCameraSpace.x, -yAxisCameraSpace.y, 0) + .normalize() + .multiply(isTopVisible ? 1 : -1); + + const capCenter = isTopVisible + ? points[CylinderEnds.Top] + : points[CylinderEnds.Bottom]; + + const tailCenter = isTopVisible + ? points[CylinderEnds.Bottom] + : points[CylinderEnds.Top]; + + // addCylinderEnd( + // capCenter, + // cylinder.radius, + // dotProduct, + // isTopVisible ? Color(255, 0, 0) : Color(0, 0, 255), + // svg + // ); + + const leftNormal = Vector3(-yAxisScreenSpace.y, yAxisScreenSpace.x, 0); + const rightNormal = Vector3(yAxisScreenSpace.y, -yAxisScreenSpace.x, 0); + const topLeftPoint = leftNormal.clone().multiply(Radius).add(capCenter); + const topRightPoint = rightNormal.clone().multiply(Radius).add(capCenter); + const bottomLeftPoint = leftNormal.clone().multiply(Radius).add(tailCenter); + const bottomRightPoint = rightNormal.clone().multiply(Radius).add(tailCenter); + + const xAxisRotation = normalToXAxisDegrees(rightNormal.x, rightNormal.y); + const largeArcFlag = 0; + const sweepFlag = 0; + + const reversedLightDirection = scene.directionalLight.direction + .clone() + .multiply(-1); + + const capPath = document.createElementNS( + "http://www.w3.org/2000/svg", + "path" + ); + capPath.setAttribute("id", "cylinder-top"); + capPath.setAttribute( + "fill", + applyLighting( + scene.directionalLight.color, + cylinder.fill, + scene.ambientLightColor, + isTopVisible + ? reversedLightDirection.dotProduct(yAxisWorldSpace) + : reversedLightDirection.dotProduct( + yAxisWorldSpace.clone().multiply(-1) + ) + ) + ); + capPath.setAttribute( + "d", + ` + M ${topLeftPoint.x} ${ + topLeftPoint.y + } A ${Radius} ${ShortRadius} ${xAxisRotation} ${largeArcFlag} ${sweepFlag} ${ + topRightPoint.x + } ${topRightPoint.y} + A ${Radius} ${ShortRadius} ${xAxisRotation} ${1} ${sweepFlag} ${ + topLeftPoint.x + } ${topLeftPoint.y}` + ); + + svg.appendChild(capPath); + + const tubePath = document.createElementNS( + "http://www.w3.org/2000/svg", + "path" + ); + tubePath.setAttribute("id", "cylinder-tube"); + tubePath.setAttribute("fill", "purple"); + tubePath.setAttribute( + "d", + ` + M ${topLeftPoint.x} ${topLeftPoint.y} + A ${Radius} ${ShortRadius} ${xAxisRotation} 0 1 ${topRightPoint.x} ${topRightPoint.y} + L ${bottomRightPoint.x} ${bottomRightPoint.y} + A ${Radius} ${ShortRadius} ${xAxisRotation} 0 0 ${bottomLeftPoint.x} ${bottomLeftPoint.y} + ` ); + svg.appendChild(tubePath); // Scenarios we can view the cylinder from: // 1. From the top/bottom (can't see the tube) @@ -113,30 +167,62 @@ export function renderCylinder( // Are we viewing the cylinder from the top or bottom? - points.forEach(({ x, y }, index) => { - const circle = document.createElementNS( - "http://www.w3.org/2000/svg", - "circle" - ); + // points.forEach(({ x, y }, index) => { + // const circle = document.createElementNS( + // "http://www.w3.org/2000/svg", + // "circle" + // ); - circle.id = "sphere"; - circle.setAttribute("cx", x.toString()); - circle.setAttribute("cy", y.toString()); + // circle.id = "sphere"; + // circle.setAttribute("cx", x.toString()); + // circle.setAttribute("cy", y.toString()); - // TODO: Factor in camera projection matrix, this currectly - // ignores all zoom factors. Can we even handle skew with sphere?! - // I don't think we can. - circle.setAttribute("r", (5).toString()); + // // TODO: Factor in camera projection matrix, this currectly + // // ignores all zoom factors. Can we even handle skew with sphere?! + // // I don't think we can. + // circle.setAttribute("r", (5).toString()); - circle.setAttribute( - "fill", - index === CylinderEnds.Top ? "rgb(128,0,0)" : "rgb(0,0,128)" - ); + // circle.setAttribute( + // "fill", + // index === CylinderEnds.Top ? "rgb(128,0,0)" : "rgb(0,0,128)" + // ); - svg.appendChild(circle); - }); + // svg.appendChild(circle); + // }); // Get the center of the cylinder's top face // Get the center of the cylinder's bottom face } + +function addCylinderEnd( + { x, y }: Vector3, + radius: number, + dotProductAbsolute: number, + fill: Color, + svg: SVGElement +) { + // Create a 'circle' element + const circle = document.createElementNS( + "http://www.w3.org/2000/svg", + "ellipse" + ); + + circle.id = "sphere"; + circle.setAttribute("cx", x.toString()); + circle.setAttribute("cy", y.toString()); + + // TODO: Factor in camera projection matrix, this currectly + // ignores all zoom factors. Can we even handle skew with sphere?! + // I don't think we can. + circle.setAttribute("rx", radius.toString()); + circle.setAttribute("ry", (radius * dotProductAbsolute).toString()); + + circle.setAttribute("fill", ColorToCSS(fill)); + + svg.appendChild(circle); +} + +function normalToXAxisDegrees(x: number, y: number) { + return (Math.atan2(y, x) / Math.PI) * 180; +} diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index e3636b6..4e36489 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -135,8 +135,10 @@ export default function () { // updateCamera(45, 20); // cylinder.rotation.x = now * 90; - cylinder.rotation.x = 0; - cylinder.rotation.z = 0; + cylinder.rotation.x = 45; + cylinder.rotation.y = now * 90; + + // cylinder.rotation.x = now * 90; // lightSphere.position.x = // Math.sin(now * Math.PI * 2 * lightSpeed) * lightDistance; From 2adaa2038654e7b27e979a38b40f01624312b26a Mon Sep 17 00:00:00 2001 From: Francois Date: Sun, 24 Sep 2023 15:08:02 -0400 Subject: [PATCH 07/20] Started to add stroke support to and cylinder is starting to work using just a simple tube and cap path --- src/renderer/renderCylinder.ts | 15 +++++++++++++-- workbench/scenes/SingleCylinder.ts | 2 +- 2 files changed, 14 insertions(+), 3 deletions(-) diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index d7f28ec..a8f05d4 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -128,6 +128,18 @@ export function renderCylinder( ) ) ); + + if (cylinder.strokeWidth && cylinder.stroke.a > 0.0) { + capPath.setAttribute("stroke", ColorToCSS(cylinder.stroke)); + + if (cylinder.strokeWidth !== 1.0) { + capPath.setAttribute( + "stroke-width", + (cylinder.strokeWidth * cylinderScaleFactor).toString() + ); + } + } + capPath.setAttribute( "d", ` @@ -141,8 +153,6 @@ export function renderCylinder( } ${topLeftPoint.y}` ); - svg.appendChild(capPath); - const tubePath = document.createElementNS( "http://www.w3.org/2000/svg", "path" @@ -159,6 +169,7 @@ export function renderCylinder( ` ); svg.appendChild(tubePath); + svg.appendChild(capPath); // Scenarios we can view the cylinder from: // 1. From the top/bottom (can't see the tube) diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index 4e36489..5db1d28 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -42,7 +42,7 @@ export default function () { // radius: referenceRadius, fill: Color(255, 0, 0), stroke: Color(0, 0, 0), - strokeWidth: 0, + strokeWidth: 4, }); const scene: Scene = { From e3c0880f941bdb3d7e54f39e8057a491bf201952 Mon Sep 17 00:00:00 2001 From: Francois Date: Sun, 24 Sep 2023 15:12:16 -0400 Subject: [PATCH 08/20] Cylinder strokes working well --- src/renderer/renderCylinder.ts | 35 +++++++++++++++++++++++----------- 1 file changed, 24 insertions(+), 11 deletions(-) diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index a8f05d4..d71014f 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -129,16 +129,8 @@ export function renderCylinder( ) ); - if (cylinder.strokeWidth && cylinder.stroke.a > 0.0) { - capPath.setAttribute("stroke", ColorToCSS(cylinder.stroke)); - - if (cylinder.strokeWidth !== 1.0) { - capPath.setAttribute( - "stroke-width", - (cylinder.strokeWidth * cylinderScaleFactor).toString() - ); - } - } + addStrokeAttribute(capPath, cylinder, cylinderScaleFactor); + svg.appendChild(capPath); capPath.setAttribute( "d", @@ -166,10 +158,14 @@ export function renderCylinder( A ${Radius} ${ShortRadius} ${xAxisRotation} 0 1 ${topRightPoint.x} ${topRightPoint.y} L ${bottomRightPoint.x} ${bottomRightPoint.y} A ${Radius} ${ShortRadius} ${xAxisRotation} 0 0 ${bottomLeftPoint.x} ${bottomLeftPoint.y} + Z ` ); + + addStrokeAttribute(tubePath, cylinder, cylinderScaleFactor); + svg.appendChild(tubePath); - svg.appendChild(capPath); + // Add Cap last // Scenarios we can view the cylinder from: // 1. From the top/bottom (can't see the tube) @@ -206,6 +202,23 @@ export function renderCylinder( // Get the center of the cylinder's bottom face } +function addStrokeAttribute( + svgShape: SVGElement, + cylinderShape: CylinderShape, + scaleFactor: number +) { + if (cylinderShape.strokeWidth && cylinderShape.stroke.a > 0.0) { + svgShape.setAttribute("stroke", ColorToCSS(cylinderShape.stroke)); + + if (cylinderShape.strokeWidth !== 1.0) { + svgShape.setAttribute( + "stroke-width", + (cylinderShape.strokeWidth * scaleFactor).toString() + ); + } + } +} + function addCylinderEnd( { x, y }: Vector3, radius: number, From e4eb0f4355470846abb02eaeed13a44e8388a1fc Mon Sep 17 00:00:00 2001 From: Francois Date: Sun, 24 Sep 2023 15:33:30 -0400 Subject: [PATCH 09/20] Added crack filling work around to Cylinders --- src/renderer/renderCylinder.ts | 86 ++++++++++++++++++++++++------ workbench/scenes/SingleCylinder.ts | 2 +- 2 files changed, 72 insertions(+), 16 deletions(-) diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index d71014f..58b2a12 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -17,10 +17,12 @@ enum CylinderEnds { Bottom = 1, } +const CrackFillingStrokeWidth = 0.5; + export function renderCylinder( scene: Scene, svg: SVGElement, - _defs: SVGDefsElement, + defs: SVGDefsElement, cylinder: CylinderShape, viewport: Viewport, worldTransform: Matrix4x4, @@ -115,21 +117,18 @@ export function renderCylinder( "path" ); capPath.setAttribute("id", "cylinder-top"); - capPath.setAttribute( - "fill", - applyLighting( - scene.directionalLight.color, - cylinder.fill, - scene.ambientLightColor, - isTopVisible - ? reversedLightDirection.dotProduct(yAxisWorldSpace) - : reversedLightDirection.dotProduct( - yAxisWorldSpace.clone().multiply(-1) - ) - ) + + const capFill = applyLighting( + scene.directionalLight.color, + cylinder.fill, + scene.ambientLightColor, + isTopVisible + ? reversedLightDirection.dotProduct(yAxisWorldSpace) + : reversedLightDirection.dotProduct(yAxisWorldSpace.clone().multiply(-1)) ); + capPath.setAttribute("fill", capFill); - addStrokeAttribute(capPath, cylinder, cylinderScaleFactor); + addStrokeAttribute(capPath, cylinder, cylinderScaleFactor, capFill); svg.appendChild(capPath); capPath.setAttribute( @@ -165,6 +164,56 @@ export function renderCylinder( addStrokeAttribute(tubePath, cylinder, cylinderScaleFactor); svg.appendChild(tubePath); + + const uuid = crypto.randomUUID(); + const fillUuid = uuid + "-fill"; + const fillUrl = `url(#${fillUuid})`; + + tubePath.setAttribute("fill", fillUrl); + + // Create the 'radialGradient' element + const linearGradient = document.createElementNS( + "http://www.w3.org/2000/svg", + "linearGradient" + ); + + linearGradient.setAttribute("id", fillUuid); + linearGradient.setAttribute("gradientUnits", "userSpaceOnUse"); + + // Make the control points of the gradient the center of the cylinder's + // just to keep it nice and organized when editing in Figma and such later + const leftOfTubeCenter = topLeftPoint + .clone() + .add(bottomLeftPoint) + .multiply(0.5); + const rightOfTubeCenter = topRightPoint + .clone() + .add(bottomRightPoint) + .multiply(0.5); + + linearGradient.setAttribute("x1", leftOfTubeCenter.x.toString()); + linearGradient.setAttribute("y1", leftOfTubeCenter.y.toString()); + linearGradient.setAttribute("x2", rightOfTubeCenter.x.toString()); + linearGradient.setAttribute("y2", rightOfTubeCenter.y.toString()); + + const gradientStops = [ + { offset: 0, stopColor: ColorToCSS(cylinder.fill) }, + { offset: 1.0, stopColor: ColorToCSS(Color(0, 0, 0)) }, + ]; + + for (let stop of gradientStops) { + const stopElement = document.createElementNS( + "http://www.w3.org/2000/svg", + "stop" + ); + + stopElement.setAttribute("offset", stop.offset.toString()); + stopElement.setAttribute("stop-color", stop.stopColor); + linearGradient.appendChild(stopElement); + } + + defs.appendChild(linearGradient); + // Add Cap last // Scenarios we can view the cylinder from: @@ -205,7 +254,8 @@ export function renderCylinder( function addStrokeAttribute( svgShape: SVGElement, cylinderShape: CylinderShape, - scaleFactor: number + scaleFactor: number, + fillColor?: string ) { if (cylinderShape.strokeWidth && cylinderShape.stroke.a > 0.0) { svgShape.setAttribute("stroke", ColorToCSS(cylinderShape.stroke)); @@ -216,6 +266,12 @@ function addStrokeAttribute( (cylinderShape.strokeWidth * scaleFactor).toString() ); } + } else if (fillColor !== undefined) { + svgShape.setAttribute("stroke", fillColor); + svgShape.setAttribute( + "stroke-width", + (CrackFillingStrokeWidth * scaleFactor).toString() + ); } } diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index 5db1d28..4e36489 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -42,7 +42,7 @@ export default function () { // radius: referenceRadius, fill: Color(255, 0, 0), stroke: Color(0, 0, 0), - strokeWidth: 4, + strokeWidth: 0, }); const scene: Scene = { From 8a932f6155e98d6fe07de5c9c8ab2474efefb812 Mon Sep 17 00:00:00 2001 From: Francois Date: Sun, 24 Sep 2023 16:02:35 -0400 Subject: [PATCH 10/20] Some more fixes --- src/renderer/renderCylinder.ts | 12 +++++++----- workbench/Settings.ts | 2 +- workbench/scenes/SingleCylinder.ts | 6 +++--- 3 files changed, 11 insertions(+), 9 deletions(-) diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index 58b2a12..bc2f9b6 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -53,7 +53,7 @@ export function renderCylinder( Vector3(0, 0, 0) ); const yAxisCameraSpace = yAxisWorldSpace.clone(); - inverseCameraMatrix.applyToVector3(yAxisCameraSpace); + inverseCameraMatrix.extractRotation().applyToVector3(yAxisCameraSpace); const cylinderScale = worldTransform.getScale().x; const cylinderScaleFactor = cylinderScale * cameraZoom; @@ -62,20 +62,22 @@ export function renderCylinder( // Top === -1 // Front === 0 // Bottom === 1 - const dotProduct = yAxisWorldSpace.dotProduct( - cameraDirection.clone().multiply(-1) + const dotProduct = yAxisCameraSpace.dotProduct( + // cameraDirection.clone().multiply(-1) + Vector3(0, 0, 1) ); const dotProductAbsolute = Math.abs(dotProduct); const ShortRadius = Radius * dotProductAbsolute; + const isTopVisible = dotProduct > 0; console.log( - `dotProduct: ${dotProduct.toFixed(3)} + `scenario: ${isTopVisible ? "top" : "bottom"} + dotProduct: ${dotProduct.toFixed(3)} yAxisCameraSpace: ${yAxisCameraSpace.x.toFixed( 2 )}, ${yAxisCameraSpace.y.toFixed(2)}, ${yAxisCameraSpace.z.toFixed(2)}` ); - const isTopVisible = dotProduct > 0; const yAxisScreenSpace = Vector3(yAxisCameraSpace.x, -yAxisCameraSpace.y, 0) .normalize() diff --git a/workbench/Settings.ts b/workbench/Settings.ts index f6e9ee4..b628ed3 100644 --- a/workbench/Settings.ts +++ b/workbench/Settings.ts @@ -162,7 +162,7 @@ export function getCamera(choice: CameraChoice, zoom: number = 1) { case "top": camera.matrix.makeTranslation(0, 0, 0); - camera.matrix.makeRotationX(Math.PI / 2); + camera.matrix.makeRotationX(-Math.PI / 2); break; case "isometric": { diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index 4e36489..98d6ec2 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -128,14 +128,14 @@ export default function () { document.addEventListener("pointerup", onPointerEvent); onUpdate(({ now, deltaTime }) => { - const cameraSpeed = 0.1; - // const cameraSpeed = 0.0; + // const cameraSpeed = 0.1; + const cameraSpeed = 0.0; updateCamera(now * cameraSpeed * 360 + 45, 20); // updateCamera(45, 20); // cylinder.rotation.x = now * 90; - cylinder.rotation.x = 45; + // cylinder.rotation.x = 45; cylinder.rotation.y = now * 90; // cylinder.rotation.x = now * 90; From ad37d7a82390e5c0f659853edfa284713ba64e33 Mon Sep 17 00:00:00 2001 From: Francois Date: Sun, 24 Sep 2023 16:03:32 -0400 Subject: [PATCH 11/20] Fixed 'top' camera mode which turned out to be bottom camera mode --- workbench/scenes/SingleCylinder.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index 98d6ec2..6233aa8 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -64,7 +64,7 @@ export default function () { lightSphere.position = Vector3(1, 1, -1); - const { viewport, camera, updateCamera } = getCamera("top"); + const { viewport, camera, updateCamera } = getCamera("isometric"); const onPointerEvent = (event: PointerEvent) => { // return; From fd78db348db96b6152187bed46035d05fbee8614 Mon Sep 17 00:00:00 2001 From: Francois Date: Sun, 24 Sep 2023 18:09:06 -0400 Subject: [PATCH 12/20] Sorta getting there, math isn't quite right --- src/renderer/renderCylinder.ts | 52 +++++++++++++++++++----------- workbench/scenes/SingleCylinder.ts | 26 ++------------- 2 files changed, 37 insertions(+), 41 deletions(-) diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index bc2f9b6..306973c 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -55,21 +55,17 @@ export function renderCylinder( const yAxisCameraSpace = yAxisWorldSpace.clone(); inverseCameraMatrix.extractRotation().applyToVector3(yAxisCameraSpace); - const cylinderScale = worldTransform.getScale().x; - const cylinderScaleFactor = cylinderScale * cameraZoom; - const Radius = cylinder.radius * cylinderScaleFactor; - // Top === -1 // Front === 0 // Bottom === 1 - const dotProduct = yAxisCameraSpace.dotProduct( - // cameraDirection.clone().multiply(-1) - Vector3(0, 0, 1) - ); + const dotProduct = yAxisCameraSpace.dotProduct(Vector3(0, 0, 1)); // This boils down to just taking the z component const dotProductAbsolute = Math.abs(dotProduct); + const isTopVisible = dotProduct > 0; + const cylinderScale = worldTransform.getScale().x; + const cylinderScaleFactor = cylinderScale * cameraZoom; + const Radius = cylinder.radius * cylinderScaleFactor; const ShortRadius = Radius * dotProductAbsolute; - const isTopVisible = dotProduct > 0; console.log( `scenario: ${isTopVisible ? "top" : "bottom"} @@ -164,9 +160,20 @@ export function renderCylinder( ); addStrokeAttribute(tubePath, cylinder, cylinderScaleFactor); - svg.appendChild(tubePath); + // Convert the light direction into camera space (not projected into screen space) + const directionalLightInCameraSpace = scene.directionalLight.direction + .clone() + .multiply(-1); + inverseCameraMatrix + .extractRotation() + .applyToVector3(directionalLightInCameraSpace); + // Then convert it into screen aligned cylinder space + Matrix4x4() + .makeRotationZ((xAxisRotation / 180) * Math.PI) + .applyToVector3(directionalLightInCameraSpace); + const uuid = crypto.randomUUID(); const fillUuid = uuid + "-fill"; const fillUrl = `url(#${fillUuid})`; @@ -198,19 +205,28 @@ export function renderCylinder( linearGradient.setAttribute("x2", rightOfTubeCenter.x.toString()); linearGradient.setAttribute("y2", rightOfTubeCenter.y.toString()); - const gradientStops = [ - { offset: 0, stopColor: ColorToCSS(cylinder.fill) }, - { offset: 1.0, stopColor: ColorToCSS(Color(0, 0, 0)) }, - ]; + // Add the gradient stops + const GradientSteps = Math.min(2, Math.max(255, Math.floor(Radius))); + + for (let i = 0; i < GradientSteps; i++) { + const normalized = i / (GradientSteps - 1); + const x = Math.sin(normalized * Math.PI); + const z = Math.cos(normalized * Math.PI + Math.PI); - for (let stop of gradientStops) { const stopElement = document.createElementNS( "http://www.w3.org/2000/svg", "stop" ); - - stopElement.setAttribute("offset", stop.offset.toString()); - stopElement.setAttribute("stop-color", stop.stopColor); + stopElement.setAttribute("offset", normalized.toFixed(3)); + stopElement.setAttribute( + "stop-color", + applyLighting( + scene.directionalLight.color, + cylinder.fill, + scene.ambientLightColor, + directionalLightInCameraSpace.dotProduct(Vector3(x, 0, z)) + ) + ); linearGradient.appendChild(stopElement); } diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index 6233aa8..f3cfa55 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -80,19 +80,6 @@ export default function () { const distanceNormalized = distance / referenceRadius; let degrees = distanceNormalized * 90; - //Math.cos(((distanceNormalized * 90) / 180) * Math.PI); - // console.log( - // degrees, - // event.clientX, - // event.clientY, - // centerX, - // centerY, - // diffX, - // diffY, - // distance, - // distanceNormalized - // ); - const spinMode: string = "z"; if (spinMode === "y") { @@ -100,7 +87,6 @@ export default function () { degrees *= -1; } lightSphere.position.x = Math.sin((degrees / 180) * Math.PI); - // lightSphere.position.y = 0.0; lightSphere.position.y = 0.5; lightSphere.position.z = Math.cos((degrees / 180) * Math.PI); @@ -111,17 +97,10 @@ export default function () { } else if (spinMode === "z") { lightSphere.position.x = Math.sin((degrees / 180) * Math.PI); lightSphere.position.y = Math.cos((degrees / 180) * Math.PI); - lightSphere.position.z = 0; // - // lightSphere.position.z = -0.5; // + lightSphere.position.z = 0; } lightSphere.position.normalize().multiply(lightDistance).add(position); - - // const x = event.clientX; - // const z = event.clientY; - // lightSphere.position.y = 0; - // const x = event.clientX; - // const y = event.clientY; }; document.addEventListener("pointerdown", onPointerEvent); document.addEventListener("pointermove", onPointerEvent); @@ -136,7 +115,8 @@ export default function () { // cylinder.rotation.x = now * 90; // cylinder.rotation.x = 45; - cylinder.rotation.y = now * 90; + cylinder.rotation.x = 90; + // cylinder.rotation.y = now * 90; // cylinder.rotation.x = now * 90; From b97513b9cdc642349b73552c6a099c98b8f20058 Mon Sep 17 00:00:00 2001 From: Francois Date: Mon, 25 Sep 2023 13:01:12 -0400 Subject: [PATCH 13/20] Getting closer to cylinder rendering --- src/renderer/renderCylinder.ts | 19 ++++++++++++++++--- workbench/Settings.ts | 18 +++++++++++++++++- workbench/scenes/SingleCylinder.ts | 21 ++++++++++++++------- 3 files changed, 47 insertions(+), 11 deletions(-) diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index 306973c..136dc6f 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -205,13 +205,26 @@ export function renderCylinder( linearGradient.setAttribute("x2", rightOfTubeCenter.x.toString()); linearGradient.setAttribute("y2", rightOfTubeCenter.y.toString()); + const leftEdgeNormal = Vector3(0, 0, 1) + .crossProduct(yAxisCameraSpace) + .normalize(); + + const lightingSpace = Matrix4x4().lookAt( + Vector3(0, 0, 0), + leftEdgeNormal, + yAxisCameraSpace + ); + // Add the gradient stops - const GradientSteps = Math.min(2, Math.max(255, Math.floor(Radius))); + const GradientSteps = Math.max(2, Math.min(255, Math.floor(Radius))); for (let i = 0; i < GradientSteps; i++) { const normalized = i / (GradientSteps - 1); - const x = Math.sin(normalized * Math.PI); + const x = Math.cos(normalized * Math.PI + Math.PI / 2); const z = Math.cos(normalized * Math.PI + Math.PI); + const normal = Vector3(x, 0, z); + + lightingSpace.applyToVector3(normal); const stopElement = document.createElementNS( "http://www.w3.org/2000/svg", @@ -224,7 +237,7 @@ export function renderCylinder( scene.directionalLight.color, cylinder.fill, scene.ambientLightColor, - directionalLightInCameraSpace.dotProduct(Vector3(x, 0, z)) + directionalLightInCameraSpace.dotProduct(normal) ) ); linearGradient.appendChild(stopElement); diff --git a/workbench/Settings.ts b/workbench/Settings.ts index b628ed3..28574c8 100644 --- a/workbench/Settings.ts +++ b/workbench/Settings.ts @@ -11,7 +11,12 @@ import { Viewport, } from "../src"; -export type LightingChoice = "reference" | "moonlit" | "underwater" | "none"; +export type LightingChoice = + | "reference" + | "black and white" + | "moonlit" + | "underwater" + | "none"; export function getLighting(lighting: LightingChoice): { directionalLight: DirectionalLight; @@ -38,6 +43,17 @@ export function getLighting(lighting: LightingChoice): { g: 252, b: 255, }; + } else if (lighting === "black and white") { + ambientLightColor = { + r: 0, + g: 0, + b: 0, + }; + directionalLightColor = { + r: 255, + g: 252, + b: 255, + }; } else if (lighting === "moonlit") { // Bluish white ambientLightColor = { diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index f3cfa55..f88f445 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -40,13 +40,13 @@ export default function () { radius: referenceRadius, height, // radius: referenceRadius, - fill: Color(255, 0, 0), + fill: Color(255, 255, 255), stroke: Color(0, 0, 0), strokeWidth: 0, }); const scene: Scene = { - ...getLighting("reference"), + ...getLighting("black and white"), shapes: [ getEnvironment("grid"), // Axii(Vector3(-referenceRadius * 3, 0, 0)), @@ -76,6 +76,7 @@ export default function () { const diffX = event.clientX - centerX; const diffY = event.clientY - centerY; const distance = Math.sqrt(diffX * diffX + diffY * diffY); + const lightSpeed = 2; const distanceNormalized = distance / referenceRadius; let degrees = distanceNormalized * 90; @@ -86,17 +87,17 @@ export default function () { if (diffX < 0) { degrees *= -1; } - lightSphere.position.x = Math.sin((degrees / 180) * Math.PI); - lightSphere.position.y = 0.5; - lightSphere.position.z = Math.cos((degrees / 180) * Math.PI); + lightSphere.position.x = Math.sin((degrees / 180) * Math.PI * lightSpeed); + lightSphere.position.y = 0.0; + lightSphere.position.z = Math.cos((degrees / 180) * Math.PI * lightSpeed); if (event.buttons === 1) { console.log(); lightSphere.position.z *= -1; } } else if (spinMode === "z") { - lightSphere.position.x = Math.sin((degrees / 180) * Math.PI); - lightSphere.position.y = Math.cos((degrees / 180) * Math.PI); + lightSphere.position.x = Math.sin((degrees / 180) * Math.PI * lightSpeed); + lightSphere.position.y = Math.cos((degrees / 180) * Math.PI * lightSpeed); lightSphere.position.z = 0; } @@ -126,11 +127,17 @@ export default function () { // lightSphere.position.z = // Math.cos(now * Math.PI * 2 * lightSpeed) * lightDistance; + // lightSphere.position = Vector3(1, 0, 1).multiply(lightDistance); + scene.directionalLight.direction = lightSphere.position + .clone() + .setY(0) .clone() .normalize() .multiply(-1); + lightSphere.position.y = height / 2; + render(document.getElementById("root")!, scene, viewport, camera); }); } From 87d7d1377e2f2d38c4740e81574119da8e151ac6 Mon Sep 17 00:00:00 2001 From: Francois Date: Tue, 26 Sep 2023 09:47:05 -0400 Subject: [PATCH 14/20] Looking better, but still having some subtle issues. In the SingleCylder demo I'm getting some sort of flip effect with the lighting --- src/math/Matrix4x4.ts | 20 +++++++++++++------- src/renderer/renderCylinder.ts | 25 +++++++++---------------- workbench/scenes/SingleCylinder.ts | 4 +--- 3 files changed, 23 insertions(+), 26 deletions(-) diff --git a/src/math/Matrix4x4.ts b/src/math/Matrix4x4.ts index 6962a64..c69cc91 100644 --- a/src/math/Matrix4x4.ts +++ b/src/math/Matrix4x4.ts @@ -26,7 +26,7 @@ export interface Matrix4x4 { copyPosition(matrix: Matrix4x4): Matrix4x4; getTranslation(): Vector3; getScale(): Vector3; - extractBasis(xAxis: Vector3, yAxis: Vector3, zAxis: Vector3): Matrix4x4; + extractBasis(xAxis?: Vector3, yAxis?: Vector3, zAxis?: Vector3): Matrix4x4; extractRotation(): Matrix4x4; lookAt(eye: Vector3, target: Vector3, up: Vector3): Matrix4x4; @@ -313,13 +313,19 @@ const Matrix4x4Proto = { extractBasis( this: Matrix4x4, - xAxis: Vector3, - yAxis: Vector3, - zAxis: Vector3 + xAxis?: Vector3, + yAxis?: Vector3, + zAxis?: Vector3 ) { - setVector3FromMatrixElements(xAxis, this.elements, 0); - setVector3FromMatrixElements(yAxis, this.elements, 4); - setVector3FromMatrixElements(zAxis, this.elements, 8); + if (xAxis !== undefined) { + setVector3FromMatrixElements(xAxis, this.elements, 0); + } + if (yAxis !== undefined) { + setVector3FromMatrixElements(yAxis, this.elements, 4); + } + if (zAxis !== undefined) { + setVector3FromMatrixElements(zAxis, this.elements, 8); + } return this; }, diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index 136dc6f..31730f4 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -47,11 +47,7 @@ export function renderCylinder( }); const yAxisWorldSpace = Vector3(0, 0, 0); - worldTransform.extractBasis( - Vector3(0, 0, 0), - yAxisWorldSpace, - Vector3(0, 0, 0) - ); + worldTransform.extractBasis(undefined, yAxisWorldSpace, undefined); const yAxisCameraSpace = yAxisWorldSpace.clone(); inverseCameraMatrix.extractRotation().applyToVector3(yAxisCameraSpace); @@ -75,9 +71,12 @@ export function renderCylinder( )}, ${yAxisCameraSpace.y.toFixed(2)}, ${yAxisCameraSpace.z.toFixed(2)}` ); - const yAxisScreenSpace = Vector3(yAxisCameraSpace.x, -yAxisCameraSpace.y, 0) - .normalize() - .multiply(isTopVisible ? 1 : -1); + const yAxisScreenSpace = Vector3( + yAxisCameraSpace.x, + -yAxisCameraSpace.y, + 0 + ).normalize(); + // .multiply(isTopVisible ? 1 : -1); const capCenter = isTopVisible ? points[CylinderEnds.Top] @@ -163,16 +162,10 @@ export function renderCylinder( svg.appendChild(tubePath); // Convert the light direction into camera space (not projected into screen space) - const directionalLightInCameraSpace = scene.directionalLight.direction - .clone() - .multiply(-1); + const directionalLightInCameraSpace = reversedLightDirection.clone(); inverseCameraMatrix .extractRotation() .applyToVector3(directionalLightInCameraSpace); - // Then convert it into screen aligned cylinder space - Matrix4x4() - .makeRotationZ((xAxisRotation / 180) * Math.PI) - .applyToVector3(directionalLightInCameraSpace); const uuid = crypto.randomUUID(); const fillUuid = uuid + "-fill"; @@ -221,7 +214,7 @@ export function renderCylinder( for (let i = 0; i < GradientSteps; i++) { const normalized = i / (GradientSteps - 1); const x = Math.cos(normalized * Math.PI + Math.PI / 2); - const z = Math.cos(normalized * Math.PI + Math.PI); + const z = -Math.cos(normalized * Math.PI + Math.PI); const normal = Vector3(x, 0, z); lightingSpace.applyToVector3(normal); diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index f88f445..3b6bda5 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -130,13 +130,11 @@ export default function () { // lightSphere.position = Vector3(1, 0, 1).multiply(lightDistance); scene.directionalLight.direction = lightSphere.position - .clone() - .setY(0) .clone() .normalize() .multiply(-1); - lightSphere.position.y = height / 2; + // lightSphere.position.y = height / 2; render(document.getElementById("root")!, scene, viewport, camera); }); From ce9d0f028d5c93c2ae0ff9916f7dbbbe1bdb850e Mon Sep 17 00:00:00 2001 From: Francois Date: Tue, 26 Sep 2023 09:53:59 -0400 Subject: [PATCH 15/20] ... --- workbench/scenes/SingleCylinder.ts | 13 ++++++++++++- 1 file changed, 12 insertions(+), 1 deletion(-) diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index 3b6bda5..07c004e 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -45,6 +45,17 @@ export default function () { strokeWidth: 0, }); + // const cylinder = Box({ + // id: "reference", + // position, + // width: referenceRadius * 2, + // height, + // // radius: referenceRadius, + // fill: Color(255, 255, 255), + // stroke: Color(0, 0, 0), + // strokeWidth: 0, + // }); + const scene: Scene = { ...getLighting("black and white"), shapes: [ @@ -114,7 +125,7 @@ export default function () { // updateCamera(45, 20); - // cylinder.rotation.x = now * 90; + // cylinder.rotation.x = (now * 90) % 360; // cylinder.rotation.x = 45; cylinder.rotation.x = 90; // cylinder.rotation.y = now * 90; From 9afbd3c680bd8da88fd812be3e0cce79eb45153e Mon Sep 17 00:00:00 2001 From: Francois Date: Sat, 14 Oct 2023 21:45:09 -0400 Subject: [PATCH 16/20] Added start of debugrender functions. Not finished/not hooked up --- src/renderer/DebugRenderer.ts | 19 +++++++++++++++++++ 1 file changed, 19 insertions(+) create mode 100644 src/renderer/DebugRenderer.ts diff --git a/src/renderer/DebugRenderer.ts b/src/renderer/DebugRenderer.ts new file mode 100644 index 0000000..9aa843a --- /dev/null +++ b/src/renderer/DebugRenderer.ts @@ -0,0 +1,19 @@ +import { Color, ColorToCSS } from "../colors/Color"; + +export function DebugLine2D( + svg: SVGElement, + x: number, + y: number, + x2: number, + y2: number, + stroke: Color = Color(0, 0, 0) +) { + const line = document.createElementNS("http://www.w3.org/2000/svg", "line"); + line.setAttribute("x1", x.toFixed(2)); + line.setAttribute("y1", y.toFixed(2)); + line.setAttribute("x2", x2.toFixed(2)); + line.setAttribute("y2", y2.toFixed(2)); + line.setAttribute("stroke", ColorToCSS(stroke)); + // line.setAttribute("stroke-width", "0.1"); + svg.appendChild(line); +} From d63bf474900578672df1e9a8a6db7b0cf1e18ca6 Mon Sep 17 00:00:00 2001 From: Francois Date: Mon, 16 Oct 2023 09:25:16 -0400 Subject: [PATCH 17/20] Cylinder rendering is looking correct --- src/index.ts | 1 + src/renderer/DebugRenderer.ts | 22 +++-- src/renderer/Renderer.ts | 5 ++ src/renderer/renderCylinder.ts | 130 +++++++++++++++++++---------- workbench/main.ts | 4 +- workbench/scenes/SingleCylinder.ts | 11 ++- 6 files changed, 119 insertions(+), 54 deletions(-) diff --git a/src/index.ts b/src/index.ts index 8807392..bf527a2 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,6 +1,7 @@ export * from "./renderer/Scene"; export * from "./renderer/Viewport"; export * from "./renderer/Renderer"; +export * from "./renderer/DebugRenderer"; export * from "./colors/Color"; export * from "./lighting/DirectionalLight"; export * from "./lighting/LightingModel"; diff --git a/src/renderer/DebugRenderer.ts b/src/renderer/DebugRenderer.ts index 9aa843a..d11575c 100644 --- a/src/renderer/DebugRenderer.ts +++ b/src/renderer/DebugRenderer.ts @@ -1,19 +1,31 @@ import { Color, ColorToCSS } from "../colors/Color"; +import { Viewport } from "./Viewport"; export function DebugLine2D( svg: SVGElement, + viewport: Viewport, x: number, y: number, x2: number, y2: number, stroke: Color = Color(0, 0, 0) ) { + const centerX = viewport.width / 2; + const centerY = viewport.height / 2; const line = document.createElementNS("http://www.w3.org/2000/svg", "line"); - line.setAttribute("x1", x.toFixed(2)); - line.setAttribute("y1", y.toFixed(2)); - line.setAttribute("x2", x2.toFixed(2)); - line.setAttribute("y2", y2.toFixed(2)); + line.style.zIndex = "1000"; + line.setAttribute("x1", (x + centerX).toFixed(2)); + line.setAttribute("y1", (y + centerY).toFixed(2)); + line.setAttribute("x2", (x2 + centerX).toFixed(2)); + line.setAttribute("y2", (y2 + centerY).toFixed(2)); line.setAttribute("stroke", ColorToCSS(stroke)); // line.setAttribute("stroke-width", "0.1"); - svg.appendChild(line); + + // @ts-ignore + if (!svg.debugQueue) { + // @ts-ignore + svg.debugQueue = []; + } + // @ts-ignore + svg.debugQueue.push(line); } diff --git a/src/renderer/Renderer.ts b/src/renderer/Renderer.ts index 6ad559f..03df6be 100644 --- a/src/renderer/Renderer.ts +++ b/src/renderer/Renderer.ts @@ -131,6 +131,11 @@ export function render( } } + // @ts-ignore + svg.debugQueue?.forEach((element) => { + svg.appendChild(element); + }); + container.appendChild(svg); } diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index 31730f4..0ed2b35 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -9,6 +9,7 @@ import { applyLighting } from "../lighting/LightingModel"; import { Matrix4x4 } from "../math/Matrix4x4"; import { Vector3 } from "../math/Vector3"; import { CylinderShape } from "../shapes/Shape"; +import { DebugLine2D } from "./DebugRenderer"; import { Scene } from "./Scene"; import { Viewport } from "./Viewport"; @@ -31,7 +32,8 @@ export function renderCylinder( inverseCameraMatrix: Matrix4x4, inverseAndProjectionMatrix: Matrix4x4 ) { - const points: Vector3[] = [ + // Get screen space coordinates for the top and bottom of the cylinder + const capsInScreenSpace: Vector3[] = [ // Top Vector3(0, cylinder.height / 2, 0), // Bottom @@ -46,17 +48,31 @@ export function renderCylinder( ); }); - const yAxisWorldSpace = Vector3(0, 0, 0); - worldTransform.extractBasis(undefined, yAxisWorldSpace, undefined); - const yAxisCameraSpace = yAxisWorldSpace.clone(); - inverseCameraMatrix.extractRotation().applyToVector3(yAxisCameraSpace); + const cylinderYAxisWorldSpace = Vector3(0, 0, 0); + worldTransform.extractBasis(undefined, cylinderYAxisWorldSpace, undefined); + + const cylinderYAxisCameraSpace = cylinderYAxisWorldSpace.clone(); + inverseCameraMatrix + .extractRotation() + .applyToVector3(cylinderYAxisCameraSpace); + + // DebugLine2D( + // svg, + // viewport, + // 0, + // 0, + // cylinderYAxisCameraSpace.x * 100, + // -cylinderYAxisCameraSpace.y * 100, + // Color(255, 0, 0) + // ); // Top === -1 // Front === 0 // Bottom === 1 - const dotProduct = yAxisCameraSpace.dotProduct(Vector3(0, 0, 1)); // This boils down to just taking the z component + const dotProduct = cylinderYAxisCameraSpace.dotProduct(Vector3(0, 0, 1)); // This boils down to just taking the z component const dotProductAbsolute = Math.abs(dotProduct); const isTopVisible = dotProduct > 0; + console.log(`isTopVisible: ${isTopVisible}`); const cylinderScale = worldTransform.getScale().x; const cylinderScaleFactor = cylinderScale * cameraZoom; @@ -66,25 +82,35 @@ export function renderCylinder( console.log( `scenario: ${isTopVisible ? "top" : "bottom"} dotProduct: ${dotProduct.toFixed(3)} - yAxisCameraSpace: ${yAxisCameraSpace.x.toFixed( + yAxisCameraSpace: ${cylinderYAxisCameraSpace.x.toFixed( 2 - )}, ${yAxisCameraSpace.y.toFixed(2)}, ${yAxisCameraSpace.z.toFixed(2)}` + )}, ${cylinderYAxisCameraSpace.y.toFixed( + 2 + )}, ${cylinderYAxisCameraSpace.z.toFixed(2)}` ); - const yAxisScreenSpace = Vector3( - yAxisCameraSpace.x, - -yAxisCameraSpace.y, + const visibleUpAxisWorldSpace = isTopVisible + ? cylinderYAxisWorldSpace.clone() + : cylinderYAxisWorldSpace.clone().multiply(-1); + + const visibleUpAxisCameraSpace = isTopVisible + ? cylinderYAxisCameraSpace.clone() + : cylinderYAxisCameraSpace.clone().multiply(-1); + + const visibleUpAxisScreenSpace = Vector3( + visibleUpAxisCameraSpace.x, + -visibleUpAxisCameraSpace.y, 0 ).normalize(); // .multiply(isTopVisible ? 1 : -1); - const capCenter = isTopVisible - ? points[CylinderEnds.Top] - : points[CylinderEnds.Bottom]; + const visibleCapCenter = isTopVisible + ? capsInScreenSpace[CylinderEnds.Top] + : capsInScreenSpace[CylinderEnds.Bottom]; - const tailCenter = isTopVisible - ? points[CylinderEnds.Bottom] - : points[CylinderEnds.Top]; + const hiddenCapCenter = isTopVisible + ? capsInScreenSpace[CylinderEnds.Bottom] + : capsInScreenSpace[CylinderEnds.Top]; // addCylinderEnd( // capCenter, @@ -94,12 +120,32 @@ export function renderCylinder( // svg // ); - const leftNormal = Vector3(-yAxisScreenSpace.y, yAxisScreenSpace.x, 0); - const rightNormal = Vector3(yAxisScreenSpace.y, -yAxisScreenSpace.x, 0); - const topLeftPoint = leftNormal.clone().multiply(Radius).add(capCenter); - const topRightPoint = rightNormal.clone().multiply(Radius).add(capCenter); - const bottomLeftPoint = leftNormal.clone().multiply(Radius).add(tailCenter); - const bottomRightPoint = rightNormal.clone().multiply(Radius).add(tailCenter); + const leftNormal = Vector3( + -visibleUpAxisScreenSpace.y, + visibleUpAxisScreenSpace.x, + 0 + ); + const rightNormal = Vector3( + visibleUpAxisScreenSpace.y, + -visibleUpAxisScreenSpace.x, + 0 + ); + const visibleLeftPoint = leftNormal + .clone() + .multiply(Radius) + .add(visibleCapCenter); + const visibleRightPoint = rightNormal + .clone() + .multiply(Radius) + .add(visibleCapCenter); + const hiddenLeftPoint = leftNormal + .clone() + .multiply(Radius) + .add(hiddenCapCenter); + const hiddenRightPoint = rightNormal + .clone() + .multiply(Radius) + .add(hiddenCapCenter); const xAxisRotation = normalToXAxisDegrees(rightNormal.x, rightNormal.y); const largeArcFlag = 0; @@ -113,15 +159,13 @@ export function renderCylinder( "http://www.w3.org/2000/svg", "path" ); - capPath.setAttribute("id", "cylinder-top"); + capPath.setAttribute("id", isTopVisible ? "cylinder-top" : "cylinder-bottom"); const capFill = applyLighting( scene.directionalLight.color, cylinder.fill, scene.ambientLightColor, - isTopVisible - ? reversedLightDirection.dotProduct(yAxisWorldSpace) - : reversedLightDirection.dotProduct(yAxisWorldSpace.clone().multiply(-1)) + reversedLightDirection.dotProduct(visibleUpAxisWorldSpace.clone()) ); capPath.setAttribute("fill", capFill); @@ -131,14 +175,14 @@ export function renderCylinder( capPath.setAttribute( "d", ` - M ${topLeftPoint.x} ${ - topLeftPoint.y + M ${visibleLeftPoint.x} ${ + visibleLeftPoint.y } A ${Radius} ${ShortRadius} ${xAxisRotation} ${largeArcFlag} ${sweepFlag} ${ - topRightPoint.x - } ${topRightPoint.y} + visibleRightPoint.x + } ${visibleRightPoint.y} A ${Radius} ${ShortRadius} ${xAxisRotation} ${1} ${sweepFlag} ${ - topLeftPoint.x - } ${topLeftPoint.y}` + visibleLeftPoint.x + } ${visibleLeftPoint.y}` ); const tubePath = document.createElementNS( @@ -150,10 +194,10 @@ export function renderCylinder( tubePath.setAttribute( "d", ` - M ${topLeftPoint.x} ${topLeftPoint.y} - A ${Radius} ${ShortRadius} ${xAxisRotation} 0 1 ${topRightPoint.x} ${topRightPoint.y} - L ${bottomRightPoint.x} ${bottomRightPoint.y} - A ${Radius} ${ShortRadius} ${xAxisRotation} 0 0 ${bottomLeftPoint.x} ${bottomLeftPoint.y} + M ${visibleLeftPoint.x} ${visibleLeftPoint.y} + A ${Radius} ${ShortRadius} ${xAxisRotation} 0 1 ${visibleRightPoint.x} ${visibleRightPoint.y} + L ${hiddenRightPoint.x} ${hiddenRightPoint.y} + A ${Radius} ${ShortRadius} ${xAxisRotation} 0 0 ${hiddenLeftPoint.x} ${hiddenLeftPoint.y} Z ` ); @@ -184,13 +228,13 @@ export function renderCylinder( // Make the control points of the gradient the center of the cylinder's // just to keep it nice and organized when editing in Figma and such later - const leftOfTubeCenter = topLeftPoint + const leftOfTubeCenter = visibleLeftPoint .clone() - .add(bottomLeftPoint) + .add(hiddenLeftPoint) .multiply(0.5); - const rightOfTubeCenter = topRightPoint + const rightOfTubeCenter = visibleRightPoint .clone() - .add(bottomRightPoint) + .add(hiddenRightPoint) .multiply(0.5); linearGradient.setAttribute("x1", leftOfTubeCenter.x.toString()); @@ -199,13 +243,13 @@ export function renderCylinder( linearGradient.setAttribute("y2", rightOfTubeCenter.y.toString()); const leftEdgeNormal = Vector3(0, 0, 1) - .crossProduct(yAxisCameraSpace) + .crossProduct(visibleUpAxisCameraSpace) .normalize(); const lightingSpace = Matrix4x4().lookAt( Vector3(0, 0, 0), leftEdgeNormal, - yAxisCameraSpace + visibleUpAxisCameraSpace ); // Add the gradient stops diff --git a/workbench/main.ts b/workbench/main.ts index 1ebb03f..407f086 100644 --- a/workbench/main.ts +++ b/workbench/main.ts @@ -13,9 +13,9 @@ import SingleCylinder from "./scenes/SingleCylinder"; // Transforms(); // Octopus(); // Spheres(); -// Cylinders(); +Cylinders(); // SingleSphere(); -SingleCylinder(); +// SingleCylinder(); // Worm(); document diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index 07c004e..190804f 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -33,7 +33,8 @@ export default function () { }); const height = referenceRadius * 4; - const position = Vector3(0, height / 2, 0); + // const position = Vector3(0, height / 2, 0); + const position = Vector3(0, 0, 0); const cylinder = Cylinder({ id: "reference", position, @@ -60,7 +61,7 @@ export default function () { ...getLighting("black and white"), shapes: [ getEnvironment("grid"), - // Axii(Vector3(-referenceRadius * 3, 0, 0)), + Axii(Vector3(-referenceRadius * 3, 0, 0)), // Group({ // position: Vector3(0, 0, 0), // rotation: Vector3(45, 0, 0), @@ -118,6 +119,8 @@ export default function () { document.addEventListener("pointermove", onPointerEvent); document.addEventListener("pointerup", onPointerEvent); + const rotationSpeed = 0.3; + onUpdate(({ now, deltaTime }) => { // const cameraSpeed = 0.1; const cameraSpeed = 0.0; @@ -125,9 +128,9 @@ export default function () { // updateCamera(45, 20); - // cylinder.rotation.x = (now * 90) % 360; + cylinder.rotation.x = (now * 90 * rotationSpeed) % 360; // cylinder.rotation.x = 45; - cylinder.rotation.x = 90; + // cylinder.rotation.x = 90; // cylinder.rotation.y = now * 90; // cylinder.rotation.x = now * 90; From 4c02506a5bc03e68aeb23fb3b5bf1d65551143fc Mon Sep 17 00:00:00 2001 From: Francois Date: Mon, 16 Oct 2023 09:43:09 -0400 Subject: [PATCH 18/20] ... --- src/renderer/renderCylinder.ts | 41 ++---------------------------- src/renderer/renderSphere.ts | 24 ++++++++--------- workbench/main.ts | 4 +-- workbench/scenes/KitchenSink.ts | 2 +- workbench/scenes/SingleCylinder.ts | 2 +- 5 files changed, 18 insertions(+), 55 deletions(-) diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index 0ed2b35..5fa7a7c 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -4,12 +4,11 @@ // https://observablehq.com/d/011f054fc7eaf966 import { projectToScreenCoordinate } from "../cameras/Camera"; -import { Color, ColorToCSS } from "../colors/Color"; +import { ColorToCSS } from "../colors/Color"; import { applyLighting } from "../lighting/LightingModel"; import { Matrix4x4 } from "../math/Matrix4x4"; import { Vector3 } from "../math/Vector3"; import { CylinderShape } from "../shapes/Shape"; -import { DebugLine2D } from "./DebugRenderer"; import { Scene } from "./Scene"; import { Viewport } from "./Viewport"; @@ -28,7 +27,7 @@ export function renderCylinder( viewport: Viewport, worldTransform: Matrix4x4, cameraZoom: number, - cameraDirection: Vector3, + _cameraDirection: Vector3, inverseCameraMatrix: Matrix4x4, inverseAndProjectionMatrix: Matrix4x4 ) { @@ -112,14 +111,6 @@ export function renderCylinder( ? capsInScreenSpace[CylinderEnds.Bottom] : capsInScreenSpace[CylinderEnds.Top]; - // addCylinderEnd( - // capCenter, - // cylinder.radius, - // dotProduct, - // isTopVisible ? Color(255, 0, 0) : Color(0, 0, 255), - // svg - // ); - const leftNormal = Vector3( -visibleUpAxisScreenSpace.y, visibleUpAxisScreenSpace.x, @@ -343,34 +334,6 @@ function addStrokeAttribute( } } -function addCylinderEnd( - { x, y }: Vector3, - radius: number, - dotProductAbsolute: number, - fill: Color, - svg: SVGElement -) { - // Create a 'circle' element - const circle = document.createElementNS( - "http://www.w3.org/2000/svg", - "ellipse" - ); - - circle.id = "sphere"; - circle.setAttribute("cx", x.toString()); - circle.setAttribute("cy", y.toString()); - - // TODO: Factor in camera projection matrix, this currectly - // ignores all zoom factors. Can we even handle skew with sphere?! - // I don't think we can. - circle.setAttribute("rx", radius.toString()); - circle.setAttribute("ry", (radius * dotProductAbsolute).toString()); - - circle.setAttribute("fill", ColorToCSS(fill)); - - svg.appendChild(circle); -} - function normalToXAxisDegrees(x: number, y: number) { return (Math.atan2(y, x) / Math.PI) * 180; } diff --git a/src/renderer/renderSphere.ts b/src/renderer/renderSphere.ts index 966dc69..9f9fb04 100644 --- a/src/renderer/renderSphere.ts +++ b/src/renderer/renderSphere.ts @@ -21,20 +21,20 @@ function dotProductToDegrees(dotProduct: number) { return angleInDegrees; } -// function normalToDegrees(x: number, z: number) { -// return (Math.atan2(x, z) / Math.PI) * 180; +// // function normalToDegrees(x: number, z: number) { +// // return (Math.atan2(x, z) / Math.PI) * 180; +// // } + +// function normalizeDegrees(degrees: number) { +// let adjustedDegrees = degrees; +// if (adjustedDegrees < 0) { +// adjustedDegrees = 360 + (adjustedDegrees % 360); +// } else { +// adjustedDegrees = adjustedDegrees % 360; +// } +// return adjustedDegrees; // } -function normalizeDegrees(degrees: number) { - let adjustedDegrees = degrees; - if (adjustedDegrees < 0) { - adjustedDegrees = 360 + (adjustedDegrees % 360); - } else { - adjustedDegrees = adjustedDegrees % 360; - } - return adjustedDegrees; -} - function calculateRotationAngle(x: number, y: number) { const angleInRadians = Math.atan2(y, x); // Convert radians to degrees diff --git a/workbench/main.ts b/workbench/main.ts index 407f086..d6fdb50 100644 --- a/workbench/main.ts +++ b/workbench/main.ts @@ -9,11 +9,11 @@ import { getPaused, setPaused } from "./Settings"; import Cylinders from "./scenes/Cylinders"; import SingleCylinder from "./scenes/SingleCylinder"; -// KitchenSink(); +KitchenSink(); // Transforms(); // Octopus(); // Spheres(); -Cylinders(); +// Cylinders(); // SingleSphere(); // SingleCylinder(); // Worm(); diff --git a/workbench/scenes/KitchenSink.ts b/workbench/scenes/KitchenSink.ts index 17027c5..6acbf18 100644 --- a/workbench/scenes/KitchenSink.ts +++ b/workbench/scenes/KitchenSink.ts @@ -344,7 +344,7 @@ export default function () { const cylinderScaleSpeed = 0.25; const cylinderTranslationSpeed = 1; // cylinder.rotation.x = 90; - // cylinder.rotation.z = now * 360 * cylinderRotationSpeed; + cylinder.rotation.z = now * 360 * cylinderRotationSpeed; const boxRotationSpeed = 0.25; // transparentBox.rotation.y = now * 360 * boxRotationSpeed; diff --git a/workbench/scenes/SingleCylinder.ts b/workbench/scenes/SingleCylinder.ts index 190804f..2d470da 100644 --- a/workbench/scenes/SingleCylinder.ts +++ b/workbench/scenes/SingleCylinder.ts @@ -61,7 +61,7 @@ export default function () { ...getLighting("black and white"), shapes: [ getEnvironment("grid"), - Axii(Vector3(-referenceRadius * 3, 0, 0)), + // Axii(Vector3(-referenceRadius * 3, 0, 0)), // Group({ // position: Vector3(0, 0, 0), // rotation: Vector3(45, 0, 0), From 2599f69b72f91ee1be46069b498abe44131e9c95 Mon Sep 17 00:00:00 2001 From: Francois Date: Mon, 16 Oct 2023 10:03:03 -0400 Subject: [PATCH 19/20] Stop cylinder spinning --- workbench/scenes/KitchenSink.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/workbench/scenes/KitchenSink.ts b/workbench/scenes/KitchenSink.ts index 6acbf18..17027c5 100644 --- a/workbench/scenes/KitchenSink.ts +++ b/workbench/scenes/KitchenSink.ts @@ -344,7 +344,7 @@ export default function () { const cylinderScaleSpeed = 0.25; const cylinderTranslationSpeed = 1; // cylinder.rotation.x = 90; - cylinder.rotation.z = now * 360 * cylinderRotationSpeed; + // cylinder.rotation.z = now * 360 * cylinderRotationSpeed; const boxRotationSpeed = 0.25; // transparentBox.rotation.y = now * 360 * boxRotationSpeed; From 0ae0b4d80aed18defc2f8ad379e7ea34c84a67eb Mon Sep 17 00:00:00 2001 From: Francois Date: Mon, 16 Oct 2023 13:54:00 -0400 Subject: [PATCH 20/20] Removed all logging --- src/renderer/renderCylinder.ts | 11 ----------- workbench/scenes/KitchenSink.ts | 6 +++--- 2 files changed, 3 insertions(+), 14 deletions(-) diff --git a/src/renderer/renderCylinder.ts b/src/renderer/renderCylinder.ts index 5fa7a7c..0a07458 100644 --- a/src/renderer/renderCylinder.ts +++ b/src/renderer/renderCylinder.ts @@ -71,23 +71,12 @@ export function renderCylinder( const dotProduct = cylinderYAxisCameraSpace.dotProduct(Vector3(0, 0, 1)); // This boils down to just taking the z component const dotProductAbsolute = Math.abs(dotProduct); const isTopVisible = dotProduct > 0; - console.log(`isTopVisible: ${isTopVisible}`); const cylinderScale = worldTransform.getScale().x; const cylinderScaleFactor = cylinderScale * cameraZoom; const Radius = cylinder.radius * cylinderScaleFactor; const ShortRadius = Radius * dotProductAbsolute; - console.log( - `scenario: ${isTopVisible ? "top" : "bottom"} - dotProduct: ${dotProduct.toFixed(3)} - yAxisCameraSpace: ${cylinderYAxisCameraSpace.x.toFixed( - 2 - )}, ${cylinderYAxisCameraSpace.y.toFixed( - 2 - )}, ${cylinderYAxisCameraSpace.z.toFixed(2)}` - ); - const visibleUpAxisWorldSpace = isTopVisible ? cylinderYAxisWorldSpace.clone() : cylinderYAxisWorldSpace.clone().multiply(-1); diff --git a/workbench/scenes/KitchenSink.ts b/workbench/scenes/KitchenSink.ts index 17027c5..7a125d3 100644 --- a/workbench/scenes/KitchenSink.ts +++ b/workbench/scenes/KitchenSink.ts @@ -33,12 +33,12 @@ export default function () { // https://www.figma.com/file/735rFnz0E5ib3rq4ha5MMF/Figma-Experiments?type=design&node-id=1312-16&mode=design&t=w03Fbw0ybh430M6y-4 const pathFromFigmaCircle = "M1 0.5C1 0.776142 0.776142 1 0.5 1C0.223858 1 0 0.776142 0 0.5C0 0.223858 0.223858 0 0.5 0C0.776142 0 1 0.223858 1 0.5Z"; - console.log(pathFromFigmaCircle); + // console.log(pathFromFigmaCircle); const pathSegments = svgPathParser(pathFromFigmaCircle, true); - console.log(pathSegments); + // console.log(pathSegments); const svg3DCommands = svgPathToSvg3DCommands(pathSegments); - console.log(svg3DCommands); + // console.log(svg3DCommands); const size = 35; const lightSphere = Sphere({