diff --git a/examples/example-configurator/public/Monkey w_ freckles.json b/examples/example-configurator/public/Monkey w_ freckles.json new file mode 100644 index 0000000..dde45b9 --- /dev/null +++ b/examples/example-configurator/public/Monkey w_ freckles.json @@ -0,0 +1,150 @@ +{ + "metadata": { "version": 1, "type": "mat" }, + "base": { + "constructor": "LayerMaterial", + "currents": { + "metadata": { "version": 4.5, "type": "Material", "generator": "Material.toJSON" }, + "uuid": "b5b5d6dc57b444d92fc9ccb623668bb71de8a11a", + "type": "MeshBasicMaterial", + "name": "Monkey w/ freckles", + "color": 16711422, + "reflectivity": 1, + "refractionRatio": 0.98, + "transparent": true, + "depthFunc": 3, + "depthTest": true, + "depthWrite": true, + "colorWrite": true, + "stencilWrite": false, + "stencilWriteMask": 255, + "stencilFunc": 519, + "stencilRef": 0, + "stencilFuncMask": 255, + "stencilFail": 7680, + "stencilZFail": 7680, + "stencilZPass": 7680, + "lighting": "basic" + } + }, + "layers": [ + { + "constructor": "Abstract", + "fragment": " \n uniform float u_alpha;\n uniform float u_near;\n uniform float u_far;\n uniform float u_isVector;\n uniform vec3 u_origin;\n uniform vec3 u_colorA;\n uniform vec3 u_colorB;\n\n varying vec3 v_worldPosition;\n varying vec3 v_position;\n\n void main() {\n float f_dist = lamina_mapping_template;\n float f_depth = (f_dist - u_near) / (u_far - u_near);\n\t\t\tvec3 f_depthColor = mix(u_colorB, u_colorA, 1.0 - clamp(f_depth, 0., 1.));\n \n \n return vec4(f_depthColor, u_alpha);\n }\n ", + "vertex": "\n varying vec3 v_worldPosition;\n varying vec3 v_position;\n\n void main() {\n v_worldPosition = (vec4(position, 1.0) * modelMatrix).xyz;\n v_position = position;\n }\n ", + "uniforms": { "near": 2, "far": 10, "origin": [0, 0, 3], "colorA": "white", "colorB": "black", "alpha": 1 }, + "nonUniforms": { "mode": "normal", "visible": true, "name": "Depth$1", "mapping": "vector" }, + "currents": { + "near": -0.07, + "far": 5, + "origin": [0, 0, 3], + "colorA": "#fe4eb8", + "colorB": "#24dbf8", + "alpha": 1, + "mode": "normal", + "visible": true, + "name": "Depth", + "mapping": "vector" + }, + "functions": { + "onShaderParse": "self => {\n function getMapping(uuid, type) {\n switch (type) {\n default:\n case 'vector':\n return `length(v_${uuid}_worldPosition - u_${uuid}_origin)`;\n\n case 'world':\n return `length(v_${uuid}_position - vec3(0.))`;\n\n case 'camera':\n return `length(v_${uuid}_worldPosition - cameraPosition)`;\n }\n }\n\n self.schema.push({\n value: self.mapping,\n label: 'mapping',\n options: ['vector', 'world', 'camera']\n });\n const mapping = getMapping(self.uuid, self.mapping);\n self.fragmentShader = self.fragmentShader.replace('lamina_mapping_template', mapping);\n }" + } + }, + { + "constructor": "Abstract", + "fragment": " \n uniform float u_alpha;\n uniform float u_near;\n uniform float u_far;\n uniform float u_isVector;\n uniform vec3 u_origin;\n uniform vec3 u_colorA;\n uniform vec3 u_colorB;\n\n varying vec3 v_worldPosition;\n varying vec3 v_position;\n\n void main() {\n float f_dist = lamina_mapping_template;\n float f_depth = (f_dist - u_near) / (u_far - u_near);\n\t\t\tvec3 f_depthColor = mix(u_colorB, u_colorA, 1.0 - clamp(f_depth, 0., 1.));\n \n \n return vec4(f_depthColor, u_alpha);\n }\n ", + "vertex": "\n varying vec3 v_worldPosition;\n varying vec3 v_position;\n\n void main() {\n v_worldPosition = (vec4(position, 1.0) * modelMatrix).xyz;\n v_position = position;\n }\n ", + "uniforms": { "near": 2, "far": 10, "origin": [0, 0, -1.37], "colorA": "white", "colorB": "black", "alpha": 1 }, + "nonUniforms": { "mode": "normal", "visible": true, "name": "Depth$1", "mapping": "vector" }, + "currents": { + "near": 1, + "far": 3, + "origin": [0, 0, -1.37], + "colorA": "#fe7800", + "colorB": "#000000", + "alpha": 1, + "mode": "screen", + "visible": true, + "name": "Depth", + "mapping": "vector" + }, + "functions": { + "onShaderParse": "self => {\n function getMapping(uuid, type) {\n switch (type) {\n default:\n case 'vector':\n return `length(v_${uuid}_worldPosition - u_${uuid}_origin)`;\n\n case 'world':\n return `length(v_${uuid}_position - vec3(0.))`;\n\n case 'camera':\n return `length(v_${uuid}_worldPosition - cameraPosition)`;\n }\n }\n\n self.schema.push({\n value: self.mapping,\n label: 'mapping',\n options: ['vector', 'world', 'camera']\n });\n const mapping = getMapping(self.uuid, self.mapping);\n self.fragmentShader = self.fragmentShader.replace('lamina_mapping_template', mapping);\n }" + } + }, + { + "constructor": "Abstract", + "fragment": " \n uniform vec3 u_color;\n uniform float u_alpha;\n uniform float u_bias;\n uniform float u_intensity;\n uniform float u_power;\n uniform float u_factor;\n\n varying vec3 v_worldPosition;\n varying vec3 v_worldNormal;\n\n void main() {\n float f_a = (u_factor + dot(v_worldPosition, v_worldNormal));\n float f_fresnel = u_bias + u_intensity * pow(abs(f_a), u_power);\n\n f_fresnel = clamp(f_fresnel, 0.0, 1.0);\n return vec4(f_fresnel * u_color, u_alpha);\n }\n ", + "vertex": "\n varying vec3 v_worldPosition;\n varying vec3 v_worldNormal;\n\n void main() {\n v_worldPosition = vec3(-viewMatrix[0][2], -viewMatrix[1][2], -viewMatrix[2][2]);\n v_worldNormal = normalize( mat3( modelMatrix[0].xyz, modelMatrix[1].xyz, modelMatrix[2].xyz ) * normal );\n \n }\n ", + "uniforms": { "color": "white", "alpha": 1, "bias": 0, "intensity": 1, "power": 2, "factor": 1 }, + "nonUniforms": { "mode": "normal", "visible": true, "name": "Fresnel$1" }, + "currents": { + "color": "#fefefe", + "alpha": 1, + "bias": 0, + "intensity": 1, + "power": 1.91, + "factor": 1, + "mode": "softlight", + "visible": true, + "name": "Fresnel" + }, + "functions": {} + }, + { + "constructor": "Abstract", + "fragment": " \n uniform vec3 u_colorA;\n uniform vec3 u_colorB;\n uniform vec3 u_colorC;\n uniform vec3 u_colorD;\n uniform vec3 u_offset;\n\n uniform float u_alpha;\n uniform float u_scale;\n\n varying vec3 v_position;\n\n\n void main() {\n float f_n = lamina_noise_template((v_position + u_offset) * u_scale);\n\n float f_step1 = 0.;\n float f_step2 = 0.2;\n float f_step3 = 0.6;\n float f_step4 = 1.;\n\n vec3 f_color = mix(u_colorA, u_colorB, smoothstep(f_step1, f_step2, f_n));\n f_color = mix(f_color, u_colorC, smoothstep(f_step2, f_step3, f_n));\n f_color = mix(f_color, u_colorD, smoothstep(f_step3, f_step4, f_n));\n\n return vec4(f_color, u_alpha);\n }\n ", + "vertex": "\n varying vec3 v_position;\n\n void main() {\n v_position = lamina_mapping_template;\n }\n ", + "uniforms": { + "colorA": "#666666", + "colorB": "#666666", + "colorC": "#FFFFFF", + "colorD": "#FFFFFF", + "alpha": 1, + "scale": 1, + "offset": [0, 0, 0] + }, + "nonUniforms": { "mode": "normal", "visible": true, "name": "Noise$1", "type": "perlin", "mapping": "local" }, + "currents": { + "colorA": "#3bfed0", + "colorB": "#4e4e4e", + "colorC": "#000000", + "colorD": "#000000", + "alpha": 1, + "scale": 50, + "offset": [0, 0, 0], + "mode": "lighten", + "visible": true, + "name": "noise", + "type": "perlin", + "mapping": "local" + }, + "functions": { + "onShaderParse": "self => {\n function getNoiseFunction(type) {\n switch (type) {\n default:\n case 'perlin':\n return `lamina_noise_perlin`;\n\n case 'simplex':\n return `lamina_noise_simplex`;\n\n case 'cell':\n return `lamina_noise_worley`;\n\n case 'white':\n return `lamina_noise_white`;\n\n case 'curl':\n return `lamina_noise_swirl`;\n }\n }\n\n function getMapping(type) {\n switch (type) {\n default:\n case 'local':\n return `position`;\n\n case 'world':\n return `(modelMatrix * vec4(position,1.0)).xyz`;\n\n case 'uv':\n return `vec3(uv, 0.)`;\n }\n }\n\n self.schema.push({\n value: self.type,\n label: 'type',\n options: ['perlin', 'simplex', 'cell', 'curl', 'white']\n });\n self.schema.push({\n value: self.mapping,\n label: 'mapping',\n options: ['local', 'world', 'uv']\n });\n const noiseFunc = getNoiseFunction(self.type);\n const mapping = getMapping(self.mapping);\n self.vertexShader = self.vertexShader.replace('lamina_mapping_template', mapping);\n self.fragmentShader = self.fragmentShader.replace('lamina_noise_template', noiseFunc);\n }" + } + }, + { + "constructor": "Abstract", + "fragment": " \n\t\tuniform sampler2D u_map; \n\t\tuniform float u_alpha; \n\t\tvarying vec2 v_uv;\n\n void main() {\n\t\t\tvec3 f_color = texture2D(u_map, v_uv).rgb;\n return vec4(f_color, u_alpha);\n }\n ", + "vertex": "\n varying vec2 v_uv;\n \n void main() {\n v_uv = uv;\n }\n ", + "uniforms": { "alpha": 1 }, + "nonUniforms": { "mode": "normal", "visible": true, "name": "Texture$1" }, + "currents": { "alpha": 1, "mode": "normal", "visible": true, "name": "Texture$1" }, + "functions": {} + }, + { + "constructor": "Texture", + "fragment": " \n\t\tuniform sampler2D u_map; \n\t\tuniform float u_alpha; \n\t\tvarying vec2 v_uv;\n\n void main() {\n\t\t\tvec3 f_color = texture2D(u_map, v_uv).rgb;\n return vec4(f_color, u_alpha);\n }\n ", + "vertex": "\n varying vec2 v_uv;\n \n void main() {\n v_uv = uv;\n }\n ", + "uniforms": { "alpha": 1, "map": {} }, + "nonUniforms": { "mode": "normal", "visible": true, "name": "Texture$1" }, + "currents": { + "alpha": 1, + "map": "", + "mode": "normal", + "visible": true, + "name": "Texture$1" + }, + "functions": {} + } + ] +} diff --git a/examples/example-configurator/public/UV.jpg b/examples/example-configurator/public/UV.jpg new file mode 100644 index 0000000..e05927d Binary files /dev/null and b/examples/example-configurator/public/UV.jpg differ diff --git a/examples/example-configurator/src/Monkey.tsx b/examples/example-configurator/src/Monkey.tsx index 5ff465c..8cff0f0 100644 --- a/examples/example-configurator/src/Monkey.tsx +++ b/examples/example-configurator/src/Monkey.tsx @@ -2,15 +2,15 @@ Auto-generated by: https://github.com/pmndrs/gltfjsx */ -import React, { useEffect, useMemo, useState } from 'react' -import { useGLTF, TransformControls, OrbitControls } from '@react-three/drei' -import { Color, DebugLayerMaterial, Depth, LayerMaterial } from 'lamina' +import React, { useState } from 'react' +import { useGLTF, TransformControls, OrbitControls, useTexture } from '@react-three/drei' +import { DebugLayerMaterial } from 'lamina/debug' import * as LAYERS from 'lamina' import { button, useControls } from 'leva' -import { SerializedLayer } from 'lamina/types' +import { Texture } from 'lamina' export default function Monkey() { - const { nodes, scene } = useGLTF('/monkey.glb') as any + const { nodes } = useGLTF('/monkey.glb') as any const orbitControls = React.useRef(null!) const transformControls = React.useRef(null!) @@ -24,21 +24,7 @@ export default function Monkey() { }) const [layers, setLayers] = useState([]) - const [materialProps, setMaterialProps] = useState<{ [key: string]: any }>({}) - - useEffect(() => { - ;(async () => { - const json = await (await fetch('/initialMaterial.json')).json() - const l = json.layers.map((layer: SerializedLayer) => { - // @ts-ignore - const Component = LAYERS[layer.constructor] - return - }) - - setMaterialProps(json.properties) - setLayers(l) - })() - }, []) + const InitialMaterial = LAYERS.useLamina('/Monkey w_ freckles.json') useControls('Layers', { Type: { @@ -64,13 +50,15 @@ export default function Monkey() { }), }) + const tex = useTexture('/UV.jpg') + return ( <> - - {...layers} + + {layers} diff --git a/examples/example-configurator/yarn.lock b/examples/example-configurator/yarn.lock index 4dfa110..fbbfe99 100644 --- a/examples/example-configurator/yarn.lock +++ b/examples/example-configurator/yarn.lock @@ -1519,9 +1519,9 @@ destroy@1.2.0: integrity sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg== detect-gpu@^4.0.19: - version "4.0.19" - resolved "https://registry.yarnpkg.com/detect-gpu/-/detect-gpu-4.0.19.tgz#e89abefb7de298405186e8a0b6f7fb6a156ad66e" - integrity sha512-ZOPhyEkOBc1WqCfrdB6ymLW5MZqxge2QPvqPt/MhWev+vBrAwxH452dkghRfdNYvCikmjUnsZsHQtBpmXH4uJA== + version "4.0.20" + resolved "https://registry.yarnpkg.com/detect-gpu/-/detect-gpu-4.0.20.tgz#e0eb95bfc6f6e96303c8140dc7b206be43dd5a0a" + integrity sha512-pHQnqxYn4TC9Xw6b4DZCJxn+X+YGcPp4c+Eq6u6OhjbrniZeqiQiBFN3Sz2JuNanx4IzyDvm2FTdj3yEmrAjYQ== dependencies: webgl-constants "^1.1.1" @@ -1603,9 +1603,9 @@ ee-first@1.1.1: integrity sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0= electron-to-chromium@^1.4.118: - version "1.4.127" - resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.127.tgz#4ef19d5d920abe2676d938f4170729b44f7f423a" - integrity sha512-nhD6S8nKI0O2MueC6blNOEZio+/PWppE/pevnf3LOlQA/fKPCrDp2Ao4wx4LFwmIkJpVdFdn2763YWLy9ENIZg== + version "1.4.129" + resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.129.tgz#c675793885721beefff99da50f57c6525c2cd238" + integrity sha512-GgtN6bsDtHdtXJtlMYZWGB/uOyjZWjmRDumXTas7dGBaB9zUyCjzHet1DY2KhyHN8R0GLbzZWqm4efeddqqyRQ== encodeurl@~1.0.2: version "1.0.2" @@ -1713,9 +1713,9 @@ execa@^5.0.0: strip-final-newline "^2.0.0" express@^4.17.3: - version "4.18.0" - resolved "https://registry.yarnpkg.com/express/-/express-4.18.0.tgz#7a426773325d0dd5406395220614c0db10b6e8e2" - integrity sha512-EJEXxiTQJS3lIPrU1AE2vRuT7X7E+0KBbpm5GSoK524yl0K8X+er8zS2P14E64eqsVNoWbMCT7MpmQ+ErAhgRg== + version "4.18.1" + resolved "https://registry.yarnpkg.com/express/-/express-4.18.1.tgz#7797de8b9c72c857b9cd0e14a5eea80666267caf" + integrity sha512-zZBcOX9TfehHQhtupq57OF8lFZ3UZi08Y97dwFCkD8p9d/d2Y3M+ykKcwaMDEL+4qyUolgBDX6AblpR3fL212Q== dependencies: accepts "~1.3.8" array-flatten "1.1.1" @@ -1837,9 +1837,9 @@ for-in@^1.0.2: integrity sha1-gQaNKVqBQuwKxybG4iAMMPttXoA= fork-ts-checker-webpack-plugin@^7.2.1: - version "7.2.8" - resolved "https://registry.yarnpkg.com/fork-ts-checker-webpack-plugin/-/fork-ts-checker-webpack-plugin-7.2.8.tgz#afe403ed92bb49a3456549ee518e28b35d213452" - integrity sha512-kvOM0w3Hi66o3qNLWNN9mWcMXwZNjuQ0+LMloxAM+QL5ZRGV6sLZnFyM86qKLNWklEG5XofcYWSAVosjtNg+vg== + version "7.2.9" + resolved "https://registry.yarnpkg.com/fork-ts-checker-webpack-plugin/-/fork-ts-checker-webpack-plugin-7.2.9.tgz#4677c1bde10b492df6dc56ebaa538c14f0370782" + integrity sha512-tiK7gAyND3pK7CFXgKXjzBQ4w2SuxPCHDsIUSN7qJD6vUNKt2LHAb+4nxm5GhxI5ADK8aFYIWQv3knC24k4sRg== dependencies: "@babel/code-frame" "^7.16.7" chalk "^4.1.2" @@ -2389,20 +2389,7 @@ ktx-parse@^0.2.1: resolved "https://registry.yarnpkg.com/ktx-parse/-/ktx-parse-0.2.2.tgz#b037b66044855215b332cb73104590af49e47791" integrity sha512-cFBc1jnGG2WlUf52NbDUXK2obJ+Mo9WUkBRvr6tP6CKxRMvZwDDFNV3JAS4cewETp5KyexByfWm9sm+O8AffiQ== -lamina@^1.1.16, lamina@^1.1.17: - version "1.1.17" - resolved "https://registry.yarnpkg.com/lamina/-/lamina-1.1.17.tgz#9f2141debac6b1d9e4e2b23bb08f205a8166ccd1" - integrity sha512-VwsS791VZ/IPH+DQ7nIqJZz9Orcwbb6KmMZeKSbKeQleKYDMVAJLvAf7yXb8D0N3hYo67anc+HkoeacGgh3e6w== - dependencies: - glsl-token-descope "^1.0.2" - glsl-token-functions "^1.0.1" - glsl-token-string "^1.0.1" - glsl-tokenizer "^2.1.5" - lamina "^1.1.16" - leva "^0.9.20" - three-custom-shader-material "^3.3.3" - -lamina@^1.1.18: +lamina@^1.1.17, lamina@^1.1.18: version "1.1.18" resolved "https://registry.yarnpkg.com/lamina/-/lamina-1.1.18.tgz#0de06c2698eba89708426f0f22e9a77a066f1bfa" integrity sha512-v9xWmIqTfRSpp2VFg+Y6IpXvRtRsVlLxvdq1iLpl2yxukMvumFdp8M79QNciVgHh7UbaK3C9sOF6bUTsPwNipA== @@ -2628,7 +2615,7 @@ multicast-dns@^7.2.4: dns-packet "^5.2.2" thunky "^1.0.2" -nanoid@*, nanoid@^3.3.1, nanoid@^3.3.3: +nanoid@*, nanoid@^3.3.3: version "3.3.3" resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.3.tgz#fd8e8b7aa761fe807dba2d1b98fb7241bb724a25" integrity sha512-p1sjXuopFs0xg+fPASzQ28agW1oHD7xDsd9Xkf3T15H3c/cifrFHVwrh74PdoklAPi+i7MdRsE47vm2r6JoB+w== @@ -2916,11 +2903,11 @@ postcss-value-parser@^4.0.2, postcss-value-parser@^4.1.0, postcss-value-parser@^ integrity sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ== postcss@^8.4.7: - version "8.4.12" - resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.4.12.tgz#1e7de78733b28970fa4743f7da6f3763648b1905" - integrity sha512-lg6eITwYe9v6Hr5CncVbK70SoioNQIq81nsaG86ev5hAidQvmOeETBqs7jm43K2F5/Ley3ytDtriImV6TpNiSg== + version "8.4.13" + resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.4.13.tgz#7c87bc268e79f7f86524235821dfdf9f73e5d575" + integrity sha512-jtL6eTBrza5MPzy8oJLFuUscHDXTV5KcLlqAWHl5q5WYRfnNRGSmOZmOZ1T6Gy7A99mOZfqungmZMpMmCVJ8ZA== dependencies: - nanoid "^3.3.1" + nanoid "^3.3.3" picocolors "^1.0.0" source-map-js "^1.0.2" @@ -3554,7 +3541,7 @@ terser@^5.10.0, terser@^5.7.2: source-map "~0.8.0-beta.0" source-map-support "~0.5.20" -three-custom-shader-material@^3.3.3, three-custom-shader-material@^3.3.4: +three-custom-shader-material@^3.3.4: version "3.3.4" resolved "https://registry.yarnpkg.com/three-custom-shader-material/-/three-custom-shader-material-3.3.4.tgz#e96e50d6d29b9647bcaa089deee65e4942b46521" integrity sha512-WjdFw/eobZfr5ezzzpFWF7Jn/YHYOKshDDboF1rplfCePSKUegwlZyfkC3/Et3m7RCNb7QHmYqb542ad2+m2hw== @@ -3651,9 +3638,9 @@ troika-worker-utils@^0.46.0: integrity sha512-bzOx5f2ZBxkFhXtIvDJlLn2AI3bzCkGVbCndl/2dL5QZrwHEKl45OEIilCxYQQWJG1rEbOD9O80tMjoYjw19OA== ts-loader@^9.2.8: - version "9.2.9" - resolved "https://registry.yarnpkg.com/ts-loader/-/ts-loader-9.2.9.tgz#0653e07fa1b4f225d0ca57a84fddbfd43d930f9e" - integrity sha512-b0+vUY2/enb0qYtDQuNlDnJ9900NTiPiJcDJ6sY7ax1CCCwXfYIqPOMm/BwW7jsF1km+Oz8W9s31HLuD+FLIMg== + version "9.3.0" + resolved "https://registry.yarnpkg.com/ts-loader/-/ts-loader-9.3.0.tgz#980f4dbfb60e517179e15e10ed98e454b132159f" + integrity sha512-2kLLAdAD+FCKijvGKi9sS0OzoqxLCF3CxHpok7rVgCZ5UldRzH0TkbwG9XECKjBzHsAewntC5oDaI/FwKzEUog== dependencies: chalk "^4.1.0" enhanced-resolve "^5.0.0" diff --git a/examples/mesh-gradients/yarn.lock b/examples/mesh-gradients/yarn.lock index 4c6c119..d2a492f 100644 --- a/examples/mesh-gradients/yarn.lock +++ b/examples/mesh-gradients/yarn.lock @@ -3790,9 +3790,9 @@ destroy@1.2.0: integrity sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg== detect-gpu@^4.0.19: - version "4.0.19" - resolved "https://registry.yarnpkg.com/detect-gpu/-/detect-gpu-4.0.19.tgz#e89abefb7de298405186e8a0b6f7fb6a156ad66e" - integrity sha512-ZOPhyEkOBc1WqCfrdB6ymLW5MZqxge2QPvqPt/MhWev+vBrAwxH452dkghRfdNYvCikmjUnsZsHQtBpmXH4uJA== + version "4.0.20" + resolved "https://registry.yarnpkg.com/detect-gpu/-/detect-gpu-4.0.20.tgz#e0eb95bfc6f6e96303c8140dc7b206be43dd5a0a" + integrity sha512-pHQnqxYn4TC9Xw6b4DZCJxn+X+YGcPp4c+Eq6u6OhjbrniZeqiQiBFN3Sz2JuNanx4IzyDvm2FTdj3yEmrAjYQ== dependencies: webgl-constants "^1.1.1" @@ -3977,9 +3977,9 @@ ejs@^3.1.6: jake "^10.8.5" electron-to-chromium@^1.4.118: - version "1.4.127" - resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.127.tgz#4ef19d5d920abe2676d938f4170729b44f7f423a" - integrity sha512-nhD6S8nKI0O2MueC6blNOEZio+/PWppE/pevnf3LOlQA/fKPCrDp2Ao4wx4LFwmIkJpVdFdn2763YWLy9ENIZg== + version "1.4.129" + resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.129.tgz#c675793885721beefff99da50f57c6525c2cd238" + integrity sha512-GgtN6bsDtHdtXJtlMYZWGB/uOyjZWjmRDumXTas7dGBaB9zUyCjzHet1DY2KhyHN8R0GLbzZWqm4efeddqqyRQ== emittery@^0.10.2: version "0.10.2" @@ -4421,9 +4421,9 @@ expect@^27.5.1: jest-message-util "^27.5.1" express@^4.17.3: - version "4.18.0" - resolved "https://registry.yarnpkg.com/express/-/express-4.18.0.tgz#7a426773325d0dd5406395220614c0db10b6e8e2" - integrity sha512-EJEXxiTQJS3lIPrU1AE2vRuT7X7E+0KBbpm5GSoK524yl0K8X+er8zS2P14E64eqsVNoWbMCT7MpmQ+ErAhgRg== + version "4.18.1" + resolved "https://registry.yarnpkg.com/express/-/express-4.18.1.tgz#7797de8b9c72c857b9cd0e14a5eea80666267caf" + integrity sha512-zZBcOX9TfehHQhtupq57OF8lFZ3UZi08Y97dwFCkD8p9d/d2Y3M+ykKcwaMDEL+4qyUolgBDX6AblpR3fL212Q== dependencies: accepts "~1.3.8" array-flatten "1.1.1" @@ -6070,9 +6070,9 @@ jsonpointer@^5.0.0: integrity sha512-PNYZIdMjVIvVgDSYKTT63Y+KZ6IZvGRNNWcxwD+GNnUz1MKPfv30J8ueCjdwcN0nDx2SlshgyB7Oy0epAzVRRg== "jsx-ast-utils@^2.4.1 || ^3.0.0", jsx-ast-utils@^3.2.1: - version "3.2.2" - resolved "https://registry.yarnpkg.com/jsx-ast-utils/-/jsx-ast-utils-3.2.2.tgz#6ab1e52c71dfc0c0707008a91729a9491fe9f76c" - integrity sha512-HDAyJ4MNQBboGpUnHAVUNJs6X0lh058s6FuixsFGP7MgJYpD6Vasd6nzSG5iIfXu1zAYlHJ/zsOKNlrenTUBnw== + version "3.3.0" + resolved "https://registry.yarnpkg.com/jsx-ast-utils/-/jsx-ast-utils-3.3.0.tgz#e624f259143b9062c92b6413ff92a164c80d3ccb" + integrity sha512-XzO9luP6L0xkxwhIJMTJQpZo/eeN60K08jHdexfD569AGxeNug6UketeHXEhROoM8aR7EcUoOQmIhcJQjcuq8Q== dependencies: array-includes "^3.1.4" object.assign "^4.1.2" @@ -6470,7 +6470,7 @@ multicast-dns@^7.2.4: dns-packet "^5.2.2" thunky "^1.0.2" -nanoid@*, nanoid@^3.3.1, nanoid@^3.3.3: +nanoid@*, nanoid@^3.3.3: version "3.3.3" resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.3.tgz#fd8e8b7aa761fe807dba2d1b98fb7241bb724a25" integrity sha512-p1sjXuopFs0xg+fPASzQ28agW1oHD7xDsd9Xkf3T15H3c/cifrFHVwrh74PdoklAPi+i7MdRsE47vm2r6JoB+w== @@ -7416,11 +7416,11 @@ postcss@^7.0.35: source-map "^0.6.1" postcss@^8.3.5, postcss@^8.4.12, postcss@^8.4.4, postcss@^8.4.7: - version "8.4.12" - resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.4.12.tgz#1e7de78733b28970fa4743f7da6f3763648b1905" - integrity sha512-lg6eITwYe9v6Hr5CncVbK70SoioNQIq81nsaG86ev5hAidQvmOeETBqs7jm43K2F5/Ley3ytDtriImV6TpNiSg== + version "8.4.13" + resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.4.13.tgz#7c87bc268e79f7f86524235821dfdf9f73e5d575" + integrity sha512-jtL6eTBrza5MPzy8oJLFuUscHDXTV5KcLlqAWHl5q5WYRfnNRGSmOZmOZ1T6Gy7A99mOZfqungmZMpMmCVJ8ZA== dependencies: - nanoid "^3.3.1" + nanoid "^3.3.3" picocolors "^1.0.0" source-map-js "^1.0.2" @@ -7974,9 +7974,9 @@ rollup-plugin-terser@^7.0.0: terser "^5.0.0" rollup@^2.43.1: - version "2.70.2" - resolved "https://registry.yarnpkg.com/rollup/-/rollup-2.70.2.tgz#808d206a8851628a065097b7ba2053bd83ba0c0d" - integrity sha512-EitogNZnfku65I1DD5Mxe8JYRUCy0hkK5X84IlDtUs+O6JRMpRciXTzyCUuX11b5L5pvjH+OmFXiQ3XjabcXgg== + version "2.71.1" + resolved "https://registry.yarnpkg.com/rollup/-/rollup-2.71.1.tgz#82b259af7733dfd1224a8171013aaaad02971a22" + integrity sha512-lMZk3XfUBGjrrZQpvPSoXcZSfKcJ2Bgn+Z0L1MoW2V8Wh7BVM+LOBJTPo16yul2MwL59cXedzW1ruq3rCjSRgw== optionalDependencies: fsevents "~2.3.2" diff --git a/package.json b/package.json index d15c588..4a574f9 100644 --- a/package.json +++ b/package.json @@ -20,6 +20,11 @@ "types": "./vanilla.d.ts", "require": "./vanilla.cjs.js", "import": "./vanilla.js" + }, + "./debug": { + "types": "./debug.d.ts", + "require": "./debug.cjs.js", + "import": "./debug.js" } }, "keywords": [ diff --git a/rollup.config.js b/rollup.config.js index e2a07a9..b6b6ddc 100644 --- a/rollup.config.js +++ b/rollup.config.js @@ -35,13 +35,13 @@ const getBabelOptions = ({ useESModules }) => ({ export default [ { - input: `./src/index.tsx`, + input: `./src/index.ts`, output: { file: `dist/index.js`, format: 'esm' }, external, plugins: [babel(getBabelOptions({ useESModules: true })), resolve({ extensions })], }, { - input: `./src/index.tsx`, + input: `./src/index.ts`, output: { file: `dist/index.cjs.js`, format: 'cjs' }, external, plugins: [babel(getBabelOptions({ useESModules: false })), resolve({ extensions })], @@ -58,4 +58,16 @@ export default [ external, plugins: [babel(getBabelOptions({ useESModules: false })), resolve({ extensions })], }, + { + input: `./src/debug.ts`, + output: { file: `dist/debug.js`, format: 'esm' }, + external, + plugins: [babel(getBabelOptions({ useESModules: true })), resolve({ extensions })], + }, + { + input: `./src/debug.ts`, + output: { file: `dist/debug.cjs.js`, format: 'cjs' }, + external, + plugins: [babel(getBabelOptions({ useESModules: false })), resolve({ extensions })], + }, ] diff --git a/scripts/link.sh b/scripts/link.sh index 3d63303..e7c5f9f 100755 --- a/scripts/link.sh +++ b/scripts/link.sh @@ -36,3 +36,13 @@ cd ../../../ yarn link three yarn link @react-three/fiber cd ../../ + + cd ./examples/example-configurator + rm -rf ./node_modeules ./yarn.lock + yarn + + yarn link lamina + yarn link react + yarn link three + yarn link @react-three/fiber + cd ../../ diff --git a/src/CSM/index.tsx b/src/CSM/index.tsx deleted file mode 100644 index c7755ff..0000000 --- a/src/CSM/index.tsx +++ /dev/null @@ -1,13 +0,0 @@ -import React, { forwardRef, useMemo } from 'react' -import CustomShaderMaterial from './vanilla' -import { iCSMProps } from './types' - -export default forwardRef( - ({ baseMaterial, fragmentShader, vertexShader, uniforms, ...rest }, ref) => { - const material = useMemo( - () => new CustomShaderMaterial(baseMaterial, fragmentShader, vertexShader, uniforms), - [baseMaterial, fragmentShader, vertexShader, uniforms] - ) - return - } -) diff --git a/src/CSM/keywords.ts b/src/CSM/keywords.ts deleted file mode 100644 index 9366e40..0000000 --- a/src/CSM/keywords.ts +++ /dev/null @@ -1,8 +0,0 @@ -export default { - positon: 'csm_Position', - emissive: 'csm_Emissive', - normal: 'csm_Normal', - pointSize: 'csm_PointSize', - diffuseColor: 'csm_DiffuseColor', - fragColor: 'csm_FragColor', -} diff --git a/src/CSM/patchMaps.ts b/src/CSM/patchMaps.ts deleted file mode 100644 index ffc20bf..0000000 --- a/src/CSM/patchMaps.ts +++ /dev/null @@ -1,42 +0,0 @@ -import keywords from './keywords' - -export const VERT = { - [`${keywords.normal}`]: { - '#include ': ` - vec3 objectNormal = ${keywords.normal}; - #ifdef USE_TANGENT - vec3 objectTangent = vec3( tangent.xyz ); - #endif - `, - }, - [`${keywords.positon}`]: { - '#include ': ` - vec3 transformed = ${keywords.positon}; - `, - }, - [`${keywords.pointSize}`]: { - 'gl_PointSize = size;': ` - gl_PointSize = ${keywords.pointSize}; - `, - }, -} - -export const FRAG = { - [`${keywords.diffuseColor}`]: { - '#include ': ` - #include - diffuseColor = ${keywords.diffuseColor}; - `, - }, - [`${keywords.fragColor}`]: { - '#include ': ` - #include - gl_FragColor = ${keywords.fragColor}; - `, - }, - [`${keywords.emissive}`]: { - 'vec3 totalEmissiveRadiance = emissive;': ` - vec3 totalEmissiveRadiance = ${keywords.emissive}; - `, - }, -} diff --git a/src/CSM/types.ts b/src/CSM/types.ts deleted file mode 100644 index a35644a..0000000 --- a/src/CSM/types.ts +++ /dev/null @@ -1,31 +0,0 @@ -import * as THREE from 'three' -import * as FIBER from '@react-three/fiber' - -export type AllMaterialParams = THREE.MeshPhongMaterialParameters & - THREE.MeshPhysicalMaterialParameters & - THREE.MeshToonMaterialParameters & - THREE.MeshBasicMaterialParameters & - THREE.MeshLambertMaterialParameters & - THREE.MeshStandardMaterialParameters & - THREE.PointsMaterialParameters - -export type AllMaterialProps = FIBER.MeshPhongMaterialProps & // - FIBER.MeshPhysicalMaterialProps & - FIBER.MeshToonMaterialProps & - FIBER.MeshBasicMaterialProps & - FIBER.MeshLambertMaterialProps & - FIBER.MeshStandardMaterialProps & - FIBER.PointsMaterialProps - -export interface iCSMShader { - defines: string - header: string - main: string -} - -export type iCSMProps = { - baseMaterial: new () => THREE.Material - vertexShader?: string - fragmentShader?: string - uniforms?: { [key: string]: THREE.IUniform } -} & AllMaterialProps diff --git a/src/CSM/vanilla.ts b/src/CSM/vanilla.ts deleted file mode 100644 index 7df091d..0000000 --- a/src/CSM/vanilla.ts +++ /dev/null @@ -1,160 +0,0 @@ -import { IUniform, Material, MathUtils } from 'three' -import { AllMaterialParams, iCSMProps, iCSMShader } from './types' - -import * as PATCH_MAP from './patchMaps' - -export default class CustomShaderMaterial extends Material { - uniforms: { [key: string]: IUniform } - private base: Material - - constructor( - baseMaterial: new () => Material, - fragmentShader?: string, - vertexShader?: string, - uniforms?: { [key: string]: THREE.IUniform }, - opts?: AllMaterialParams - ) { - // @ts-ignore - const base = new baseMaterial(opts) - super() - this.base = base - - this.uniforms = uniforms || {} - - for (const key in this.base) { - let k = key - if (key.startsWith('_')) { - k = key.split('_')[1] - } - - // @ts-ignore - if (this[k] === undefined) this[k] = 0 - // @ts-ignore - this[k] = this.base[k] - } - - this.update(fragmentShader, vertexShader, uniforms) - } - - update( - fragmentShader: iCSMProps['fragmentShader'], - vertexShader: iCSMProps['vertexShader'], - uniforms: iCSMProps['uniforms'] - ) { - this.generateMaterial(fragmentShader, vertexShader, uniforms) - } - - private generateMaterial( - fragmentShader: iCSMProps['fragmentShader'], - vertexShader: iCSMProps['vertexShader'], - uniforms: iCSMProps['uniforms'] - ) { - const parsedFragmentShdaer = this.parseShader(fragmentShader) - const parsedVertexShdaer = this.parseShader(vertexShader) - - this.uniforms = uniforms || {} - this.customProgramCacheKey = () => { - return this.uuid - } - - this.onBeforeCompile = (shader) => { - if (parsedFragmentShdaer) { - const patchedFragmentShdaer = this.patchShader(parsedFragmentShdaer, shader.fragmentShader, PATCH_MAP.FRAG) - shader.fragmentShader = patchedFragmentShdaer - } - if (parsedVertexShdaer) { - const patchedVertexShdaer = this.patchShader(parsedVertexShdaer, shader.vertexShader, PATCH_MAP.VERT) - - shader.vertexShader = '#define IS_VERTEX;\n' + patchedVertexShdaer - } - - shader.uniforms = { ...shader.uniforms, ...this.uniforms } - this.uniforms = shader.uniforms - this.needsUpdate = true - } - } - - private patchShader( - customShader: iCSMShader, - shader: string, - patchMap: { - [key: string]: { - [key: string]: any - } - } - ): string { - let patchedShader: string = shader - - Object.keys(patchMap).forEach((name: string) => { - Object.keys(patchMap[name]).forEach((key) => { - if (customShader.main.includes(name)) { - patchedShader = replaceAll(patchedShader, key, patchMap[name][key]) - } - }) - }) - - patchedShader = patchedShader.replace( - 'void main() {', - ` - ${customShader.header} - void main() { - vec3 csm_Position; - vec3 csm_Normal; - vec3 csm_Emissive; - - #ifdef IS_VERTEX - csm_Position = position; - #endif - - #ifdef IS_VERTEX - csm_Normal = normal; - #endif - - #ifndef IS_VERTEX - #ifdef STANDARD - csm_Emissive = emissive; - #endif - #endif - - vec4 csm_DiffuseColor = vec4(1., 0., 0., 1.); - vec4 csm_FragColor = vec4(1., 0., 0., 1.); - float csm_PointSize = 1.; - - ${customShader.main} - ` - ) - - patchedShader = customShader.defines + patchedShader - return patchedShader - } - - private parseShader(shader?: string): iCSMShader | undefined { - if (!shader) return - const parsedShader: iCSMShader = { - defines: '', - header: '', - main: '', - } - - const main = shader.match(/^(\s*)(void\s*main\s*\(.*\)\s*).*?{[\s\S]*?^\1}\s*$/gm) - - if (main?.length) { - const mainBody = main[0].match(/{[\w\W\s\S]*}/gm) - - if (mainBody?.length) { - parsedShader.main = mainBody[0] - } - - const rest = shader.replace(main[0], '') - const defines = rest.match(/#(.*?;)/g) || [] - const header = defines.reduce((prev, curr) => prev.replace(curr, ''), rest) - - parsedShader.header = header - parsedShader.defines = defines.join('\n') - } - - return parsedShader - } -} - -const replaceAll = (str: string, find: string, rep: string) => str.split(find).join(rep) diff --git a/src/core/LayerMaterial.ts b/src/core/LayerMaterial.ts new file mode 100644 index 0000000..051088b --- /dev/null +++ b/src/core/LayerMaterial.ts @@ -0,0 +1,172 @@ +import * as THREE from 'three' +import Abstract from './layers/Abstract' +import Depth from './layers/Depth' +import Color from './layers/Color' +import Noise from './layers/Noise' +import Fresnel from './layers/Fresnel' +import Gradient from './layers/Gradient' +import Matcap from './layers/Matcap' +import Texture from './layers/Texture' +import Displace from './layers/Displace' +import Normal from './layers/Normal' +import Shader from './layers/Shader' + +import BlendModesChunk from '../chunks/BlendModes' +import NoiseChunk from '../chunks/Noise' +import HelpersChunk from '../chunks/Helpers' +import { LayerMaterialParameters, SerializedBase, SerializedLayer, ShadingType, ShadingTypes } from '../types' +import { + ColorRepresentation, + MeshBasicMaterialParameters, + MeshLambertMaterialParameters, + MeshPhongMaterialParameters, + MeshPhysicalMaterialParameters, + MeshStandardMaterialParameters, + MeshToonMaterialParameters, +} from 'three' +import CustomShaderMaterial from 'three-custom-shader-material/vanilla' + +type AllMaterialParams = + | MeshPhongMaterialParameters + | MeshPhysicalMaterialParameters + | MeshToonMaterialParameters + | MeshBasicMaterialParameters + | MeshLambertMaterialParameters + | MeshStandardMaterialParameters + +class LayerMaterial extends CustomShaderMaterial { + layers: Abstract[] = [] + lighting: ShadingType = 'basic' + + constructor({ color, alpha, lighting, layers, ...props }: LayerMaterialParameters & AllMaterialParams = {}) { + super({ + baseMaterial: ShadingTypes[lighting || 'basic'], + ...props, + }) + + const _baseColor = color || 'white' + const _alpha = alpha ?? 1 + + this.uniforms = { + u_lamina_color: { + value: typeof _baseColor === 'string' ? new THREE.Color(_baseColor).convertSRGBToLinear() : _baseColor, + }, + u_lamina_alpha: { + value: _alpha, + }, + } + + this.layers = layers || this.layers + this.lighting = lighting || this.lighting + + this.refresh() + } + + genShaders() { + let vertexVariables = '' + let fragmentVariables = '' + let vertexShader = '' + let fragmentShader = '' + let uniforms: any = {} + + this.layers + .filter((l) => l.visible) + .forEach((l) => { + // l.buildShaders(l.constructor) + + vertexVariables += l.vertexVariables + '\n' + fragmentVariables += l.fragmentVariables + '\n' + vertexShader += l.vertexShader + '\n' + fragmentShader += l.fragmentShader + '\n' + + uniforms = { + ...uniforms, + ...l.uniforms, + } + }) + + uniforms = { + ...uniforms, + ...this.uniforms, + } + + return { + uniforms, + vertexShader: ` + ${HelpersChunk} + ${NoiseChunk} + ${vertexVariables} + + void main() { + vec3 lamina_finalPosition = position; + vec3 lamina_finalNormal = normal; + + ${vertexShader} + + csm_Position = lamina_finalPosition; + csm_Normal = lamina_finalNormal; + } + `, + fragmentShader: ` + ${HelpersChunk} + ${NoiseChunk} + ${BlendModesChunk} + ${fragmentVariables} + + uniform vec3 u_lamina_color; + uniform float u_lamina_alpha; + + void main() { + vec4 lamina_finalColor = vec4(u_lamina_color, u_lamina_alpha); + + ${fragmentShader} + + csm_DiffuseColor = lamina_finalColor; + + } + `, + } + } + + refresh() { + const hashes = this.layers.map((layer) => { + if (!layer.__updateMaterial) { + layer.__updateMaterial = this.refresh.bind(this) + } + return layer.getHash() + }) + const { uniforms, fragmentShader, vertexShader } = this.genShaders() + super.update({ fragmentShader, vertexShader, uniforms }) + } + + serialize(): SerializedBase { + return { + constructor: 'LayerMaterial', + currents: this.toJSON(), + } + } + + set color(v: ColorRepresentation) { + if (this.uniforms?.u_lamina_color?.value) + this.uniforms.u_lamina_color.value = typeof v === 'string' ? new THREE.Color(v).convertSRGBToLinear() : v + } + get color() { + return this.uniforms?.u_lamina_color?.value + } + set alpha(v: number) { + this.uniforms.u_lamina_alpha.value = v + } + get alpha() { + return this.uniforms.u_lamina_alpha.value + } + + toJSON(meta?: any) { + return { + ...super.toJSON(), + lighting: this.lighting, + name: this.name, + } + } +} + +export { LayerMaterial, Abstract, Depth, Color, Noise, Fresnel, Gradient, Matcap, Texture, Displace, Normal, Shader } diff --git a/src/core/Loader.ts b/src/core/Loader.ts new file mode 100644 index 0000000..fde0cdf --- /dev/null +++ b/src/core/Loader.ts @@ -0,0 +1,97 @@ +import * as THREE from 'three' +import { LayerMaterial, Abstract, Shader } from './LayerMaterial' +import { LaminaLayerFile, LaminaMaterialFile } from '../types' + +class ImportedLayer extends Abstract { + constructor(props?: any) { + super(ImportedLayer, props) + } +} + +function isBase64UrlImage(s: string) { + return s.trim().startsWith('data:image') +} + +export class LaminaLoader extends THREE.Loader { + texLoader: THREE.TextureLoader + + constructor(manager?: THREE.LoadingManager) { + super(manager) + this.texLoader = new THREE.TextureLoader() + } + + load( + url: string, + onLoad?: (event: T) => void, + onError?: (event: Error) => void + ): void { + fetch(url) + .then((resp) => + resp + .json() + .then(async (json: T extends LayerMaterial ? LaminaMaterialFile : LaminaLayerFile) => { + if (json.metadata.type === 'mat') { + const data = json as LaminaMaterialFile + + const layers = await Promise.all( + data.layers.map(async (layer) => { + const l = new Abstract(ImportedLayer) + l.raw.fragment = layer.fragment + l.raw.vertex = layer.vertex + l.raw.uniforms = layer.uniforms + l.raw.nonUniforms = layer.nonUniforms + + l.onShaderParse = new Function(`return (${layer.functions.onShaderParse})`)() + l.onNonUniformsParse = new Function(`return (${layer.functions.onNonUniformsParse})`)() + l.onUniformsParse = new Function(`return (${layer.functions.onUniformsParse})`)() + + l.buildUniforms() + l.buildNonUniforms() + l.buildShaders() + + await Promise.all( + Object.entries(layer.currents).map(async ([key, val]) => { + if (typeof val === 'string' && isBase64UrlImage(val)) { + const t = await this.texLoader.loadAsync(val) + t.encoding = THREE.sRGBEncoding + //@ts-ignore + l[key] = t + } else { + //@ts-ignore + l[key] = val + } + }) + ) + + return l + }) + ) + + delete data.base.currents.metadata + const mat = new LayerMaterial({ + ...data.base.currents, + layers, + }) + + onLoad?.(mat as T) + } + }) + .catch((e) => onError?.(e)) + ) + .catch((e) => onError?.(e)) + } + + loadAsync(url: string): Promise { + return new Promise((resolve, reject) => { + this.load( + url, + (e) => { + resolve(e) + }, + (err) => { + reject(err) + } + ) + }) + } +} diff --git a/src/core/debugger/index.tsx b/src/core/debugger/index.tsx new file mode 100644 index 0000000..ee0940f --- /dev/null +++ b/src/core/debugger/index.tsx @@ -0,0 +1,32 @@ +import * as React from 'react' +import * as THREE from 'three' + +import { createRoot } from 'react-dom/client' + +import { AllMaterialProps } from 'three-custom-shader-material/types' +import { LaminaMaterialFile, LayerMaterialProps } from '../../types' + +import { LayerMaterial as LayerMaterialType } from '../../vanilla' +import { button, LevaPanel, useControls, useCreateStore } from 'leva' +import { useRef } from 'react' +import mergeRefs from 'react-merge-refs' +import { downloadObjectAsJson, serializedLayersToJSX } from '../../utils/ExportUtils' +import useExports from './useExports' +import useAttach from './useAttach' + +export const DebugLayerMaterial = React.forwardRef< + LayerMaterialType, + React.PropsWithChildren> +>(({ children, ...props }, forwardRef) => { + const ref = useRef(null!) + const store = useCreateStore() + + useExports(ref, store) + useAttach(ref, store) + + if (!Array.isArray(children)) { + return <>{React.cloneElement(children as React.ReactElement, { ref: mergeRefs([forwardRef, ref]) })} + } else { + throw new Error('Lamina Debugger must contain only one as child.') + } +}) diff --git a/src/core/debugger/useAttach.tsx b/src/core/debugger/useAttach.tsx new file mode 100644 index 0000000..fa77af6 --- /dev/null +++ b/src/core/debugger/useAttach.tsx @@ -0,0 +1,34 @@ +import * as React from 'react' +import { createRoot } from 'react-dom/client' +import { LevaPanel } from 'leva' +import { LayerMaterial } from '../../vanilla' +import { StoreType } from 'leva/dist/declarations/src/types' + +export default function useAttach(ref: React.MutableRefObject, store: StoreType) { + React.useLayoutEffect(() => { + let root = document.body.querySelector('#root') + if (!root) { + root = document.createElement('div') + root.id = 'root' + document.body.appendChild(root) + } + const div = document.createElement('div') + + if (root) { + root.appendChild(div) + const levaRoot = createRoot(div) + levaRoot.render( + + ) + } + + return () => { + div.remove() + } + }, []) +} diff --git a/src/core/debugger/useExports.ts b/src/core/debugger/useExports.ts new file mode 100644 index 0000000..232de9e --- /dev/null +++ b/src/core/debugger/useExports.ts @@ -0,0 +1,32 @@ +import * as React from 'react' +import { button, useControls } from 'leva' +import { downloadObjectAsJson, serializedLayersToJSX } from '../../utils/ExportUtils' +import { LayerMaterial } from '../../vanilla' +import { LaminaMaterialFile } from '../../types' +import { StoreType } from 'leva/dist/declarations/src/types' + +export default function useExports(ref: React.MutableRefObject, store: StoreType) { + useControls( + { + 'Copy JSX': button(() => { + const serialized = ref.current.layers.map((l) => l.serialize()) + const jsx = serializedLayersToJSX(serialized, ref.current.serialize()) + navigator.clipboard.writeText(jsx) + }), + Export: button(() => { + const serialized = ref.current.layers.map((l) => l.serialize()) + const serializedBase = ref.current.serialize() + const file: LaminaMaterialFile = { + metadata: { + version: 1, + type: 'mat', + }, + base: serializedBase, + layers: serialized, + } + downloadObjectAsJson(file, serializedBase.currents?.name) + }), + }, + { store } + ) +} diff --git a/src/index.tsx b/src/core/index.tsx similarity index 90% rename from src/index.tsx rename to src/core/index.tsx index c0c55e2..68ac31c 100644 --- a/src/index.tsx +++ b/src/core/index.tsx @@ -23,20 +23,14 @@ import { NormalProps, LayerMaterialParameters, ShaderProps, -} from './types' -import * as LAYERS from './vanilla' -import DebugLayerMaterial from './debug' -import { getLayerMaterialArgs } from './utils/Functions' -import { ColorRepresentation } from 'three' +} from '../types' +import * as LAYERS from '../vanilla' import { useEffect } from 'react' -import { useRef } from 'react' -import { useLayoutEffect } from 'react' declare global { namespace JSX { interface IntrinsicElements { layerMaterial: Node - debuglayerMaterial: Node depth_: Node color_: Node noise_: Node @@ -138,17 +132,4 @@ const Shader = React.forwardRef((props, ref) => { return }) as React.ForwardRefExoticComponent> -export { - DebugLayerMaterial, - LayerMaterial, - Depth, - Color, - Noise, - Fresnel, - Gradient, - Matcap, - Texture, - Displace, - Normal, - Shader, -} +export { LayerMaterial, Depth, Color, Noise, Fresnel, Gradient, Matcap, Texture, Displace, Normal, Shader } diff --git a/src/core/Abstract.ts b/src/core/layers/Abstract.ts similarity index 90% rename from src/core/Abstract.ts rename to src/core/layers/Abstract.ts index b240d9c..b9aa58b 100644 --- a/src/core/Abstract.ts +++ b/src/core/layers/Abstract.ts @@ -1,6 +1,6 @@ -import { getSpecialParameters, getUniform, isSerializableType, serializeProp } from '../utils/Functions' +import { getSpecialParameters, getUniform, isSerializableType, serializeProp } from '../../utils/Functions' import { Color, IUniform, MathUtils, Texture, Vector3 } from 'three' -import { BlendMode, BlendModes, LayerProps, SerializedLayer } from '../types' +import { BlendMode, BlendModes, LayerProps, SerializedLayer } from '../../types' import hash from 'object-hash' // @ts-ignore @@ -105,7 +105,6 @@ export default class Abstract { } init() { - console.log('init') const defaults = Object.getOwnPropertyNames(this.raw.constructor) defaults.forEach((v) => { @@ -139,7 +138,6 @@ export default class Abstract { } buildUniforms() { - console.log('buildUniforms') const properties: PropertyDescriptorMap & ThisType = {} Object.keys(this.raw.uniforms).map((propName) => { // @ts-ignore @@ -347,44 +345,38 @@ export default class Abstract { serialize(): SerializedLayer { const name = this.constructor.name.split('$')[0] - let nonUniformPropKeys = Object.keys(this) - nonUniformPropKeys = nonUniformPropKeys.filter( - (e) => - ![ - 'uuid', - 'uniforms', - 'schema', - 'fragmentShader', - 'vertexShader', - 'fragmentVariables', - 'vertexVariables', - 'attribs', - 'events', - '__r3f', - 'onParse', - ].includes(e) - ) - const nonUniformProps = {} - nonUniformPropKeys.forEach((k) => { - // @ts-ignore - nonUniformProps[k] = this[k] + + const uniforms: { [key: string]: any } = {} + Object.entries(this.raw.uniforms).forEach(([key, value]) => { + uniforms[key] = serializeProp(value) }) - const props: { [key: string]: any } = {} - for (const key in this.uniforms) { - const name = key.replace(`u_${this.uuid}_`, '') - props[name] = serializeProp(this.uniforms[key].value) - } + const nonUniforms: { [key: string]: any } = {} + Object.entries(this.raw.nonUniforms).forEach(([key, value]) => { + nonUniforms[key] = serializeProp(value) + }) + + const currents: { [key: string]: any } = {} + const allValueKeys = [...Object.keys(uniforms), ...Object.keys(nonUniforms)] + allValueKeys + // @ts-ignore + .map((key) => this[key]) + .forEach((value, i) => { + const key = allValueKeys[i] + currents[key] = serializeProp(value) + }) return { constructor: name, - properties: { - ...props, - ...nonUniformProps, - }, - shaders: { - fragment: this.raw.fragment, - vertex: this.raw.vertex, + fragment: this.raw.fragment, + vertex: this.raw.vertex, + uniforms: uniforms, + nonUniforms: nonUniforms, + currents: currents, + functions: { + onShaderParse: this.onShaderParse?.toString(), + onNonUniformsParse: this.onNonUniformsParse?.toString(), + onUniformsParse: this.onUniformsParse?.toString(), }, } } diff --git a/src/core/Color.ts b/src/core/layers/Color.ts similarity index 90% rename from src/core/Color.ts rename to src/core/layers/Color.ts index 2bd7eaf..c291afc 100644 --- a/src/core/Color.ts +++ b/src/core/layers/Color.ts @@ -1,4 +1,4 @@ -import { ColorProps } from '../types' +import { ColorProps } from '../../types' import Abstract from './Abstract' export default class Color extends Abstract { diff --git a/src/core/Depth.ts b/src/core/layers/Depth.ts similarity index 74% rename from src/core/Depth.ts rename to src/core/layers/Depth.ts index 92c4843..d7270ca 100644 --- a/src/core/Depth.ts +++ b/src/core/layers/Depth.ts @@ -1,5 +1,5 @@ import { Vector3 } from 'three' -import { DepthProps } from '../types' +import { DepthProps } from '../../types' import Abstract from './Abstract' type AbstractExtended = Abstract & { @@ -53,28 +53,28 @@ export default class Depth extends Abstract { name: 'Depth', ...props, onShaderParse: (self) => { + function getMapping(uuid: string, type?: string) { + switch (type) { + default: + case 'vector': + return `length(v_${uuid}_worldPosition - u_${uuid}_origin)` + case 'world': + return `length(v_${uuid}_position - vec3(0.))` + case 'camera': + return `length(v_${uuid}_worldPosition - cameraPosition)` + } + } + self.schema.push({ value: self.mapping, label: 'mapping', options: ['vector', 'world', 'camera'], }) - const mapping = Depth.getMapping(self.uuid, self.mapping) + const mapping = getMapping(self.uuid, self.mapping) self.fragmentShader = self.fragmentShader.replace('lamina_mapping_template', mapping) }, }) } - - private static getMapping(uuid: string, type?: string) { - switch (type) { - default: - case 'vector': - return `length(v_${uuid}_worldPosition - u_${uuid}_origin)` - case 'world': - return `length(v_${uuid}_position - vec3(0.))` - case 'camera': - return `length(v_${uuid}_worldPosition - cameraPosition)` - } - } } diff --git a/src/core/Displace.ts b/src/core/layers/Displace.ts similarity index 98% rename from src/core/Displace.ts rename to src/core/layers/Displace.ts index 3858a05..eb02834 100644 --- a/src/core/Displace.ts +++ b/src/core/layers/Displace.ts @@ -1,5 +1,5 @@ import { Vector3 } from 'three' -import { ColorProps, DisplaceProps, MappingType, MappingTypes, NoiseProps, NoiseType, NoiseTypes } from '../types' +import { ColorProps, DisplaceProps, MappingType, MappingTypes, NoiseProps, NoiseType, NoiseTypes } from '../../types' import Abstract from './Abstract' type AbstractExtended = Abstract & { diff --git a/src/core/Fresnel.ts b/src/core/layers/Fresnel.ts similarity index 96% rename from src/core/Fresnel.ts rename to src/core/layers/Fresnel.ts index 632fa57..43b632e 100644 --- a/src/core/Fresnel.ts +++ b/src/core/layers/Fresnel.ts @@ -1,4 +1,4 @@ -import { FresnelProps } from '../types' +import { FresnelProps } from '../../types' import Abstract from './Abstract' export default class Fresnel extends Abstract { diff --git a/src/core/Gradient.ts b/src/core/layers/Gradient.ts similarity index 96% rename from src/core/Gradient.ts rename to src/core/layers/Gradient.ts index 5149378..bd6bcfa 100644 --- a/src/core/Gradient.ts +++ b/src/core/layers/Gradient.ts @@ -1,5 +1,5 @@ import { Vector3 } from 'three' -import { GradientProps, MappingType, MappingTypes } from '../types' +import { GradientProps, MappingType, MappingTypes } from '../../types' import Abstract from './Abstract' export default class Gradient extends Abstract { diff --git a/src/core/Matcap.ts b/src/core/layers/Matcap.ts similarity index 84% rename from src/core/Matcap.ts rename to src/core/layers/Matcap.ts index 7124e67..1bb821f 100644 --- a/src/core/Matcap.ts +++ b/src/core/layers/Matcap.ts @@ -1,11 +1,11 @@ -import { MatcapProps } from "../types"; -import Abstract from "./Abstract"; +import { MatcapProps } from '../../types' +import Abstract from './Abstract' // Credits: https://www.clicktorelease.com/blog/creating-spherical-environment-mapping-shader/ export default class Matcap extends Abstract { - static u_alpha = 1; - static u_map = undefined; + static u_alpha = 1 + static u_map = undefined static vertexShader = ` varying vec3 v_position; @@ -15,7 +15,7 @@ export default class Matcap extends Abstract { v_position = normalize( vec3( modelViewMatrix * vec4( position, 1.0 ) ) ); v_normal = normalize( normalMatrix * normal ); } - `; + ` static fragmentShader = ` uniform sampler2D u_map; @@ -33,12 +33,12 @@ export default class Matcap extends Abstract { return vec4(f_base, u_alpha); } - `; + ` constructor(props?: MatcapProps) { super(Matcap, { - name: "Matcap", + name: 'Matcap', ...props, - }); + }) } } diff --git a/src/core/Noise.ts b/src/core/layers/Noise.ts similarity index 62% rename from src/core/Noise.ts rename to src/core/layers/Noise.ts index bcb6a39..50be529 100644 --- a/src/core/Noise.ts +++ b/src/core/layers/Noise.ts @@ -1,12 +1,7 @@ import { Vector3 } from 'three' -import { ColorProps, MappingType, MappingTypes, NoiseProps, NoiseType, NoiseTypes } from '../types' +import { MappingType, MappingTypes, NoiseProps, NoiseType, NoiseTypes } from '../../types' import Abstract from './Abstract' -type AbstractExtended = Abstract & { - type: NoiseType - mapping: MappingType -} - export default class Noise extends Abstract { static u_colorA = '#666666' static u_colorB = '#666666' @@ -62,52 +57,52 @@ export default class Noise extends Abstract { name: 'noise', ...props, onShaderParse: (self) => { + function getNoiseFunction(type?: string) { + switch (type) { + default: + case 'perlin': + return `lamina_noise_perlin` + case 'simplex': + return `lamina_noise_simplex` + case 'cell': + return `lamina_noise_worley` + case 'white': + return `lamina_noise_white` + case 'curl': + return `lamina_noise_swirl` + } + } + + function getMapping(type?: string) { + switch (type) { + default: + case 'local': + return `position` + case 'world': + return `(modelMatrix * vec4(position,1.0)).xyz` + case 'uv': + return `vec3(uv, 0.)` + } + } + self.schema.push({ value: self.type, label: 'type', - options: Object.values(NoiseTypes), + options: ['perlin', 'simplex', 'cell', 'curl', 'white'], }) self.schema.push({ value: self.mapping, label: 'mapping', - options: Object.values(MappingTypes), + options: ['local', 'world', 'uv'], }) - const noiseFunc = Noise.getNoiseFunction(self.type) - const mapping = Noise.getMapping(self.mapping) + const noiseFunc = getNoiseFunction(self.type) + const mapping = getMapping(self.mapping) self.vertexShader = self.vertexShader.replace('lamina_mapping_template', mapping) self.fragmentShader = self.fragmentShader.replace('lamina_noise_template', noiseFunc) }, }) } - - private static getNoiseFunction(type?: string) { - switch (type) { - default: - case 'perlin': - return `lamina_noise_perlin` - case 'simplex': - return `lamina_noise_simplex` - case 'cell': - return `lamina_noise_worley` - case 'white': - return `lamina_noise_white` - case 'curl': - return `lamina_noise_swirl` - } - } - - private static getMapping(type?: string) { - switch (type) { - default: - case 'local': - return `position` - case 'world': - return `(modelMatrix * vec4(position,1.0)).xyz` - case 'uv': - return `vec3(uv, 0.)` - } - } } diff --git a/src/core/Normal.ts b/src/core/layers/Normal.ts similarity index 94% rename from src/core/Normal.ts rename to src/core/layers/Normal.ts index dd2a8f7..b3cbd88 100644 --- a/src/core/Normal.ts +++ b/src/core/layers/Normal.ts @@ -1,5 +1,5 @@ import { Vector3 } from 'three' -import { NormalProps } from '../types' +import { NormalProps } from '../../types' import Abstract from './Abstract' export default class Normal extends Abstract { diff --git a/src/core/Shader.ts b/src/core/layers/Shader.ts similarity index 93% rename from src/core/Shader.ts rename to src/core/layers/Shader.ts index 2fd2a3e..648c87f 100644 --- a/src/core/Shader.ts +++ b/src/core/layers/Shader.ts @@ -1,5 +1,4 @@ -import { ShaderProps } from '../types' -import { getUniform } from '../utils/Functions' +import { ShaderProps } from '../../types' import Abstract from './Abstract' export default class Shader extends Abstract { diff --git a/src/core/Texture.ts b/src/core/layers/Texture.ts similarity index 88% rename from src/core/Texture.ts rename to src/core/layers/Texture.ts index 7e1f859..b687390 100644 --- a/src/core/Texture.ts +++ b/src/core/layers/Texture.ts @@ -1,9 +1,9 @@ -import { TextureProps } from '../types' +import { TextureProps } from '../../types' import Abstract from './Abstract' export default class Texture extends Abstract { static u_alpha = 1 - static u_map = undefined + static u_map = null static vertexShader = ` varying vec2 v_uv; diff --git a/src/core/layers/index.ts b/src/core/layers/index.ts new file mode 100644 index 0000000..7ac169e --- /dev/null +++ b/src/core/layers/index.ts @@ -0,0 +1,23 @@ +import _Abstract from './Abstract' +import _Depth from './Depth' +import _Color from './Color' +import _Noise from './Noise' +import _Fresnel from './Fresnel' +import _Gradient from './Gradient' +import _Matcap from './Matcap' +import _Texture from './Texture' +import _Displace from './Displace' +import _Normal from './Normal' +import _Shader from './Shader' + +export const Abstract = _Abstract +export const Depth = _Depth +export const Color = _Color +export const Noise = _Noise +export const Fresnel = _Fresnel +export const Gradient = _Gradient +export const Matcap = _Matcap +export const Texture = _Texture +export const Displace = _Displace +export const Normal = _Normal +export const Shader = _Shader diff --git a/src/core/useLoader.tsx b/src/core/useLoader.tsx new file mode 100644 index 0000000..fd79685 --- /dev/null +++ b/src/core/useLoader.tsx @@ -0,0 +1,33 @@ +import * as React from 'react' +import * as LAYERS from '../vanilla' + +import mergeRefs from 'react-merge-refs' +import { useLoader } from '@react-three/fiber' + +import { LayerMaterial } from '.' +import { LaminaLoader } from './Loader' +import { LayerMaterialProps, LayerProps } from '../types' + +export function useLamina(url: string) { + // @ts-ignore + const material = useLoader(LaminaLoader, url) + + return material instanceof LAYERS.LayerMaterial + ? React.forwardRef>>( + (props, forwardRef) => { + const ref = React.useRef(null!) + + return ( + + {material.layers.map((e) => ( + + ))} + {props.children} + + ) + } + ) + : React.forwardRef((props, ref) => { + return + }) +} diff --git a/src/debug.ts b/src/debug.ts new file mode 100644 index 0000000..d0cae4b --- /dev/null +++ b/src/debug.ts @@ -0,0 +1 @@ +export * from './core/debugger' diff --git a/src/debug.tsx b/src/debug.tsx deleted file mode 100644 index 99c2823..0000000 --- a/src/debug.tsx +++ /dev/null @@ -1,206 +0,0 @@ -import { - extend, - MeshPhongMaterialProps, - MeshPhysicalMaterialProps, - MeshToonMaterialProps, - MeshBasicMaterialProps, - MeshLambertMaterialProps, - MeshStandardMaterialProps, -} from '@react-three/fiber' -import { createRoot } from 'react-dom/client' - -import { button, LevaPanel, useControls, useCreateStore } from 'leva' -import { DataItem, StoreType } from 'leva/dist/declarations/src/types' -import React, { useEffect, useMemo, useState } from 'react' -import mergeRefs from 'react-merge-refs' -import { getLayerMaterialArgs, getUniform } from './utils/Functions' -import { serializedLayersToJSX, serializedLayersToJS } from './utils/ExportUtils' -import * as LAYERS from './vanilla' -import { Color, ColorRepresentation, TextureLoader } from 'three' -import { LayerMaterialProps, ShadingType, ShadingTypes } from './types' - -extend({ - LayerMaterial: LAYERS.LayerMaterial, -}) - -function DynamicLeva({ - name, - layers, - store, - setUpdate, -}: { - setUpdate: any - name: string - layers: any[] - store: StoreType -}) { - useControls( - name, - () => { - const o: any = {} - layers.forEach((layer, i: number) => { - const n = `${layer.label} ~${i}` - o[n] = layer - o[n].onChange = () => setUpdate([`${name}.${n}`, layer.label]) - }) - return o - }, - { store }, - [layers, name] - ) - - return null -} - -type AllMaterialProps = MeshPhongMaterialProps & // - MeshPhysicalMaterialProps & - MeshToonMaterialProps & - MeshBasicMaterialProps & - MeshLambertMaterialProps & - MeshStandardMaterialProps - -const DebugLayerMaterial = React.forwardRef< - LAYERS.LayerMaterial, - React.PropsWithChildren> ->(({ children, ...props }, forwardRef) => { - const ref = React.useRef< - LAYERS.LayerMaterial & { - [key: string]: any - } - >(null!) - const store = useCreateStore() - const [layers, setLayers] = React.useState<{ [name: string]: any[] }>({}) - const [path, setPath] = React.useState(['', '']) - const textureLoader = useMemo(() => new TextureLoader(), []) - - // useControls( - // { - // 'Copy JSX': button(() => { - // const serialized = ref.current.layers.map((l) => l.serialize()) - // const jsx = serializedLayersToJSX(serialized, ref.current.serialize()) - // navigator.clipboard.writeText(jsx) - // }), - // }, - // { store } - // ) - - const { Lighting } = useControls( - 'Base', - { - Color: { - value: '#' + new Color(ref.current?.color || props?.color || 'white').convertLinearToSRGB().getHexString(), - onChange: (v) => { - ref.current.color = v - }, - }, - Alpha: { - value: ref.current?.alpha || props?.alpha || 1, - min: 0, - max: 1, - onChange: (v) => { - ref.current.alpha = v - }, - }, - Lighting: { - value: ref.current?.lighting || props?.lighting || 'basic', - options: Object.keys(ShadingTypes), - }, - }, - { store } - ) - const [args, otherProps] = useMemo(() => getLayerMaterialArgs({ ...props, lighting: Lighting }), [props, Lighting]) - - React.useEffect(() => { - const layers = ref.current.layers - - const schema: { [name: string]: any[] } = {} - layers.forEach((layer: any, i: number) => { - if (layer.getSchema) schema[`${layer.name} ~${i}`] = layer.getSchema() - }) - - setLayers(schema) - }, [children]) - - React.useEffect(() => { - const data = store.getData() - const updatedData = data[path[0]] as DataItem & { - value: any - } - if (updatedData) { - const split = path[0].split('.') - const index = parseInt(split[0].split(' ~')[1]) - const property = path[1] - const id = ref.current.layers[index].uuid - const uniform = ref.current.uniforms[`u_${id}_${property}`] - const layer = ref.current.layers[index] as LAYERS.Abstract & { - [key: string]: any - } - console.log(property) - - // if (property !== 'map') { - // layer[property] = updatedData.value - // if (uniform) { - // uniform.value = getUniform(updatedData.value) - // } else { - // layer.buildShaders(layer.raw.constructor) - // ref.current.refresh() - // } - // } else { - // ;(async () => { - // try { - // if (updatedData.value) { - // const t = await textureLoader.loadAsync(updatedData.value) - // layer[property] = t - // uniform.value = t - // } else { - // layer[property] = undefined - // uniform.value = undefined - // } - // } catch (error) { - // console.error(error) - // } - // })() - // } - } - }, [path]) - - React.useLayoutEffect(() => { - ref.current.layers = (ref.current as any).__r3f.objects - ref.current.refresh() - }, [children, args]) - - React.useLayoutEffect(() => { - const root = document.body.querySelector('#root') - const div = document.createElement('div') - - if (root) { - root.appendChild(div) - const levaRoot = createRoot(div) - levaRoot.render( - - ) - } - - return () => { - div.remove() - } - }, [props.name]) - - return ( - <> - {Object.entries(layers).map(([name, layers], i) => ( - - ))} - - {children} - - - ) -}) - -export default DebugLayerMaterial diff --git a/src/index.ts b/src/index.ts new file mode 100644 index 0000000..49eb534 --- /dev/null +++ b/src/index.ts @@ -0,0 +1,2 @@ +export * from './core' +export * from './core/useLoader' diff --git a/src/types.ts b/src/types.ts index 0e0ea48..dd779fb 100644 --- a/src/types.ts +++ b/src/types.ts @@ -173,11 +173,44 @@ export interface ShaderProps extends LayerProps { export interface SerializedLayer { constructor: string - properties: { - [name: string]: any + fragment: string + vertex: string + uniforms: { + [key: string]: any } - shaders: { - fragment: string - vertex: string + nonUniforms: { + [key: string]: any } + currents: { + [key: string]: any + } + functions: { + onShaderParse?: string + onNonUniformsParse?: string + onUniformsParse?: string + } +} + +export interface SerializedBase { + constructor: string + currents: { + [key: string]: any + } +} + +export interface LaminaMaterialFile { + metadata: { + version: number + type: 'mat' + } + base: SerializedBase + layers: SerializedLayer[] +} + +export interface LaminaLayerFile { + metadata: { + version: number + type: 'layer' + } + base: SerializedLayer } diff --git a/src/utils/ExportUtils.ts b/src/utils/ExportUtils.ts index 6eb003a..44e84ab 100644 --- a/src/utils/ExportUtils.ts +++ b/src/utils/ExportUtils.ts @@ -1,12 +1,13 @@ -import { SerializedLayer } from 'src/types' +import { LaminaLayerFile, LaminaMaterialFile, SerializedLayer } from '../types' import * as LAYERS from '../vanilla' -function getPropsFromLayer(layer: SerializedLayer) { +function getPropsFromLayer(layer: Partial) { // @ts-ignore const constructor = LAYERS[layer.constructor] const instance = new constructor() let props = '' - Object.entries(layer.properties).forEach(([key, val]) => { + // @ts-ignore + Object.entries(layer.properties || {}).forEach(([key, val]) => { const defaultVal = constructor['u_' + key] ?? instance[key] switch (key) { @@ -27,7 +28,7 @@ function getPropsFromLayer(layer: SerializedLayer) { return props } -export function serializedLayersToJSX(layers: SerializedLayer[], material: SerializedLayer) { +export function serializedLayersToJSX(layers: SerializedLayer[], material: Partial) { const materialProps = getPropsFromLayer(material) const jsx = ` @@ -46,43 +47,47 @@ export function serializedLayersToJSX(layers: SerializedLayer[], material: Seria function getJSPropsFromLayer(layer: SerializedLayer) { // @ts-ignore - const constructor = LAYERS[layer.constructor]; - const instance = new constructor(); - let props = '\t'; - let entries = Object.entries(layer.properties); + const constructor = LAYERS[layer.constructor] + const instance = new constructor() + let props = '\t' + // @ts-ignore + let entries = Object.entries(layer.properties) entries.forEach(([key, val], idx) => { - var _constructor; - const eol = '\n\t\t'; + var _constructor + const eol = '\n\t\t' if (key.includes('color')) { - const v = typeof val === "string" ? val : '#' + val.getHexString(); - props += `${key}: ${JSON.stringify(v)},${eol}`; + // @ts-ignore + const v = typeof val === 'string' ? val : '#' + val.getHexString() + props += `${key}: ${JSON.stringify(v)},${eol}` } else { - const defaultVal = (_constructor = constructor['u_' + key]) != null ? _constructor : instance[key]; + const defaultVal = (_constructor = constructor['u_' + key]) != null ? _constructor : instance[key] switch (key) { case 'name': - if (val !== layer.constructor) props += `${key}: ${JSON.stringify(val)},${eol}`; - break; - - case 'visible': - if (!val) props += `${key}:${JSON.stringify(val)},${eol}`; - break; - - default: - if (val !== defaultVal) props += `${key}: ${JSON.stringify(val)},${eol}`; - break; - } + if (val !== layer.constructor) props += `${key}: ${JSON.stringify(val)},${eol}` + break + + case 'visible': + if (!val) props += `${key}:${JSON.stringify(val)},${eol}` + break + + default: + if (val !== defaultVal) props += `${key}: ${JSON.stringify(val)},${eol}` + break } - }); - return props; + } + }) + return props } -export function serializedLayersToJS(layers: SerializedLayer[], material: SerializedLayer){ - const materialProps = getJSPropsFromLayer(material); - const jsLayers = `${layers.map(l => { - return `new ${l.constructor}({ +export function serializedLayersToJS(layers: SerializedLayer[], material: SerializedLayer) { + const materialProps = getJSPropsFromLayer(material) + const jsLayers = `${layers + .map((l) => { + return `new ${l.constructor}({ ${getJSPropsFromLayer(l)} })` - }).join(',\n\t\t')}` + }) + .join(',\n\t\t')}` const js = ` new LayerMaterial({ @@ -92,5 +97,72 @@ export function serializedLayersToJS(layers: SerializedLayer[], material: Serial ] })` - return js; + return js +} + +function isValidHttpUrl(string: string) { + let url + + try { + url = new URL(string) + } catch (_) { + return false + } + + return url.protocol === 'http:' || url.protocol === 'https:' +} + +async function toDataURL(url: string) { + var xhr = new XMLHttpRequest() + + xhr.open('GET', url) + xhr.responseType = 'blob' + xhr.send() + + return new Promise((res, rej) => { + xhr.onload = function () { + var reader = new FileReader() + reader.onloadend = function () { + res(reader.result) + } + reader.readAsDataURL(xhr.response) + } + }) +} + +export async function downloadObjectAsJson(exportObj: LaminaMaterialFile | LaminaLayerFile, exportName: string) { + const obj = structuredClone(exportObj) + + if (obj.metadata.type === 'mat') { + const o = obj as LaminaMaterialFile + + await Promise.all( + o.layers.map(async (layer) => { + await Promise.all( + Object.entries(layer.currents).map(async ([key, val]) => { + if (isValidHttpUrl(val)) { + layer.currents[key] = await toDataURL(val) + } + }) + ) + }) + ) + } else { + const o = obj as LaminaLayerFile + await Promise.all( + Object.entries(o.base.currents).map(async ([key, val]) => { + if (isValidHttpUrl(val)) { + o.base.currents[key] = await toDataURL(val) + } + }) + ) + } + + var dataStr = 'data:text/json;charset=utf-8,' + encodeURIComponent(JSON.stringify(obj)) + var downloadAnchorNode = document.createElement('a') + downloadAnchorNode.setAttribute('href', dataStr) + downloadAnchorNode.setAttribute('download', exportName + '.lamina') + document.body.appendChild(downloadAnchorNode) // required for firefox + downloadAnchorNode.click() + downloadAnchorNode.remove() } diff --git a/src/utils/Functions.ts b/src/utils/Functions.ts index 2596a6d..5fab474 100644 --- a/src/utils/Functions.ts +++ b/src/utils/Functions.ts @@ -3,7 +3,7 @@ import { LayerMaterialProps } from '../types' export function getUniform(value: any) { if (typeof value === 'string') { - return new Color(value).convertLinearToSRGB() + return new Color(value) } return value @@ -43,6 +43,10 @@ export function getLayerMaterialArgs({ color, alpha, lighting, name, ...rest }: ] as any } +function roundToTwo(num: number) { + return +Math.round((num + Number.EPSILON) * 100) / 100 +} + export function isSerializableType(prop: any) { return ( prop instanceof Vector3 || @@ -55,12 +59,12 @@ export function isSerializableType(prop: any) { export function serializeProp(prop: any) { if (isSerializableType(prop)) { - return prop.toArray() + return (prop.toArray() as number[]).map((e) => roundToTwo(e)) } else if (prop instanceof Color) { - return '#' + prop.clone().convertLinearToSRGB().getHexString() + return '#' + prop.clone().getHexString() } else if (prop instanceof Texture) { return prop.image.src } - return prop + return typeof prop === 'number' ? roundToTwo(prop) : prop } diff --git a/src/vanilla.ts b/src/vanilla.ts index c92ca25..1d3fd0c 100644 --- a/src/vanilla.ts +++ b/src/vanilla.ts @@ -1,174 +1,2 @@ -import * as THREE from 'three' -import hash from 'object-hash' - -import Abstract from './core/Abstract' -import Depth from './core/Depth' -import Color from './core/Color' -import Noise from './core/Noise' -import Fresnel from './core/Fresnel' -import Gradient from './core/Gradient' -import Matcap from './core/Matcap' -import Texture from './core/Texture' -import Displace from './core/Displace' -import Normal from './core/Normal' -import Shader from './core/Shader' - -import BlendModesChunk from './chunks/BlendModes' -import NoiseChunk from './chunks/Noise' -import HelpersChunk from './chunks/Helpers' -import { LayerMaterialParameters, SerializedLayer, ShadingType, ShadingTypes } from './types' -import { - ColorRepresentation, - MeshBasicMaterialParameters, - MeshLambertMaterialParameters, - MeshPhongMaterialParameters, - MeshPhysicalMaterialParameters, - MeshStandardMaterialParameters, - MeshToonMaterialParameters, -} from 'three' -import CustomShaderMaterial from 'three-custom-shader-material/vanilla' - -type AllMaterialParams = - | MeshPhongMaterialParameters - | MeshPhysicalMaterialParameters - | MeshToonMaterialParameters - | MeshBasicMaterialParameters - | MeshLambertMaterialParameters - | MeshStandardMaterialParameters - -class LayerMaterial extends CustomShaderMaterial { - name: string = 'LayerMaterial' - layers: Abstract[] = [] - lighting: ShadingType = 'basic' - - constructor({ color, alpha, lighting, layers, name, ...props }: LayerMaterialParameters & AllMaterialParams = {}) { - super({ - baseMaterial: ShadingTypes[lighting || 'basic'], - ...props, - }) - - const _baseColor = color || 'white' - const _alpha = alpha ?? 1 - - this.uniforms = { - u_lamina_color: { - value: typeof _baseColor === 'string' ? new THREE.Color(_baseColor).convertSRGBToLinear() : _baseColor, - }, - u_lamina_alpha: { - value: _alpha, - }, - } - - this.layers = layers || this.layers - this.lighting = lighting || this.lighting - - this.refresh() - } - - genShaders() { - console.log('genShaders') - let vertexVariables = '' - let fragmentVariables = '' - let vertexShader = '' - let fragmentShader = '' - let uniforms: any = {} - - this.layers - .filter((l) => l.visible) - .forEach((l) => { - // l.buildShaders(l.constructor) - - vertexVariables += l.vertexVariables + '\n' - fragmentVariables += l.fragmentVariables + '\n' - vertexShader += l.vertexShader + '\n' - fragmentShader += l.fragmentShader + '\n' - - uniforms = { - ...uniforms, - ...l.uniforms, - } - }) - - uniforms = { - ...uniforms, - ...this.uniforms, - } - - return { - uniforms, - vertexShader: ` - ${HelpersChunk} - ${NoiseChunk} - ${vertexVariables} - - void main() { - vec3 lamina_finalPosition = position; - vec3 lamina_finalNormal = normal; - - ${vertexShader} - - csm_Position = lamina_finalPosition; - csm_Normal = lamina_finalNormal; - } - `, - fragmentShader: ` - ${HelpersChunk} - ${NoiseChunk} - ${BlendModesChunk} - ${fragmentVariables} - - uniform vec3 u_lamina_color; - uniform float u_lamina_alpha; - - void main() { - vec4 lamina_finalColor = vec4(u_lamina_color, u_lamina_alpha); - - ${fragmentShader} - - csm_DiffuseColor = lamina_finalColor; - - } - `, - } - } - - refresh() { - console.log('refresh') - const hashes = this.layers.map((layer) => { - if (!layer.__updateMaterial) { - layer.__updateMaterial = this.refresh.bind(this) - } - return layer.getHash() - }) - const { uniforms, fragmentShader, vertexShader } = this.genShaders() - super.update({ fragmentShader, vertexShader, uniforms }) - } - - serialize(): Partial { - return { - constructor: 'LayerMaterial', - properties: { - color: this.color, - alpha: this.alpha, - name: this.name, - lighting: this.lighting, - }, - } - } - - set color(v: ColorRepresentation) { - if (this.uniforms?.u_lamina_color?.value) - this.uniforms.u_lamina_color.value = typeof v === 'string' ? new THREE.Color(v).convertSRGBToLinear() : v - } - get color() { - return this.uniforms?.u_lamina_color?.value - } - set alpha(v: number) { - this.uniforms.u_lamina_alpha.value = v - } - get alpha() { - return this.uniforms.u_lamina_alpha.value - } -} - -export { LayerMaterial, Abstract, Depth, Color, Noise, Fresnel, Gradient, Matcap, Texture, Displace, Normal, Shader } +export * from './core/LayerMaterial' +export * from './core/Loader' diff --git a/tsconfig.json b/tsconfig.json index 4f7c3bc..b1b302a 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -12,8 +12,7 @@ "declaration": true, "declarationDir": "dist", "skipLibCheck": true, - "removeComments": false, - "baseUrl": "." + "removeComments": false }, "include": ["src/**/*"] } diff --git a/yarn.lock b/yarn.lock index 556610e..89931f4 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2394,9 +2394,9 @@ eastasianwidth@^0.2.0: integrity sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA== electron-to-chromium@^1.4.118: - version "1.4.127" - resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.127.tgz#4ef19d5d920abe2676d938f4170729b44f7f423a" - integrity sha512-nhD6S8nKI0O2MueC6blNOEZio+/PWppE/pevnf3LOlQA/fKPCrDp2Ao4wx4LFwmIkJpVdFdn2763YWLy9ENIZg== + version "1.4.129" + resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.129.tgz#c675793885721beefff99da50f57c6525c2cd238" + integrity sha512-GgtN6bsDtHdtXJtlMYZWGB/uOyjZWjmRDumXTas7dGBaB9zUyCjzHet1DY2KhyHN8R0GLbzZWqm4efeddqqyRQ== elliptic@^6.5.3: version "6.5.4" @@ -4180,9 +4180,9 @@ rollup-plugin-terser@^7.0.2: terser "^5.0.0" rollup@^2.70.2: - version "2.70.2" - resolved "https://registry.yarnpkg.com/rollup/-/rollup-2.70.2.tgz#808d206a8851628a065097b7ba2053bd83ba0c0d" - integrity sha512-EitogNZnfku65I1DD5Mxe8JYRUCy0hkK5X84IlDtUs+O6JRMpRciXTzyCUuX11b5L5pvjH+OmFXiQ3XjabcXgg== + version "2.71.1" + resolved "https://registry.yarnpkg.com/rollup/-/rollup-2.71.1.tgz#82b259af7733dfd1224a8171013aaaad02971a22" + integrity sha512-lMZk3XfUBGjrrZQpvPSoXcZSfKcJ2Bgn+Z0L1MoW2V8Wh7BVM+LOBJTPo16yul2MwL59cXedzW1ruq3rCjSRgw== optionalDependencies: fsevents "~2.3.2"