diff --git a/package-lock.json b/package-lock.json index dfe4481dd5..3850b77134 100644 --- a/package-lock.json +++ b/package-lock.json @@ -13,6 +13,7 @@ "acorn": "^8.12.1", "acorn-walk": "^8.3.4", "colorjs.io": "^0.5.2", + "escodegen": "^2.1.0", "file-saver": "^1.3.8", "gifenc": "^1.0.3", "i18next": "^19.0.2", @@ -1425,6 +1426,12 @@ } } }, + "node_modules/@rollup/plugin-commonjs/node_modules/estree-walker": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", + "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==", + "dev": true + }, "node_modules/@rollup/plugin-json": { "version": "6.1.0", "resolved": "https://registry.npmjs.org/@rollup/plugin-json/-/plugin-json-6.1.0.tgz", @@ -1539,6 +1546,12 @@ } } }, + "node_modules/@rollup/pluginutils/node_modules/estree-walker": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", + "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==", + "dev": true + }, "node_modules/@rollup/rollup-android-arm-eabi": { "version": "4.34.0", "resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.34.0.tgz", @@ -2099,16 +2112,6 @@ } } }, - "node_modules/@vitest/mocker/node_modules/estree-walker": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-3.0.3.tgz", - "integrity": "sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==", - "dev": true, - "license": "MIT", - "dependencies": { - "@types/estree": "^1.0.0" - } - }, "node_modules/@vitest/pretty-format": { "version": "2.1.8", "resolved": "https://registry.npmjs.org/@vitest/pretty-format/-/pretty-format-2.1.8.tgz", @@ -2194,6 +2197,13 @@ "source-map-js": "^1.2.0" } }, + "node_modules/@vue/compiler-core/node_modules/estree-walker": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", + "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==", + "dev": true, + "optional": true + }, "node_modules/@vue/compiler-dom": { "version": "3.5.13", "resolved": "https://registry.npmjs.org/@vue/compiler-dom/-/compiler-dom-3.5.13.tgz", @@ -2225,6 +2235,13 @@ "source-map-js": "^1.2.0" } }, + "node_modules/@vue/compiler-sfc/node_modules/estree-walker": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", + "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==", + "dev": true, + "optional": true + }, "node_modules/@vue/compiler-ssr": { "version": "3.5.13", "resolved": "https://registry.npmjs.org/@vue/compiler-ssr/-/compiler-ssr-3.5.13.tgz", @@ -4387,8 +4404,6 @@ "version": "2.1.0", "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-2.1.0.tgz", "integrity": "sha512-2NlIDTwUWJN0mRPQOdtQBzbUHvdGY2P1VXSyU83Q3xKxM7WHX2Ql8dKq782Q9TgQUNOLEzEYu9bzLNj1q88I5w==", - "dev": true, - "license": "BSD-2-Clause", "dependencies": { "esprima": "^4.0.1", "estraverse": "^5.2.0", @@ -4409,7 +4424,6 @@ "version": "0.6.1", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true, "license": "BSD-3-Clause", "optional": true, "engines": { @@ -4567,7 +4581,6 @@ "version": "4.0.1", "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", - "dev": true, "license": "BSD-2-Clause", "bin": { "esparse": "bin/esparse.js", @@ -4607,24 +4620,24 @@ "version": "5.3.0", "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", - "dev": true, "license": "BSD-2-Clause", "engines": { "node": ">=4.0" } }, "node_modules/estree-walker": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", - "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==", + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-3.0.3.tgz", + "integrity": "sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g==", "dev": true, - "license": "MIT" + "dependencies": { + "@types/estree": "^1.0.0" + } }, "node_modules/esutils": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", - "dev": true, "license": "BSD-2-Clause", "engines": { "node": ">=0.10.0" diff --git a/package.json b/package.json index 30c737d432..74b920737b 100644 --- a/package.json +++ b/package.json @@ -28,6 +28,7 @@ "acorn": "^8.12.1", "acorn-walk": "^8.3.4", "colorjs.io": "^0.5.2", + "escodegen": "^2.1.0", "file-saver": "^1.3.8", "gifenc": "^1.0.3", "i18next": "^19.0.2", diff --git a/preview/global/index.html b/preview/global/index.html index 7c8fa7d0d4..3235c1e5bd 100644 --- a/preview/global/index.html +++ b/preview/global/index.html @@ -13,9 +13,9 @@ } - + - + \ No newline at end of file diff --git a/preview/global/sketch.js b/preview/global/sketch.js index 4789f83f36..f6d045ff6b 100644 --- a/preview/global/sketch.js +++ b/preview/global/sketch.js @@ -1,55 +1,50 @@ -const vertSrc = `#version 300 es - precision mediump float; - uniform mat4 uModelViewMatrix; - uniform mat4 uProjectionMatrix; - - in vec3 aPosition; - in vec2 aOffset; - - void main(){ - vec4 positionVec4 = vec4(aPosition.xyz, 1.0); - positionVec4.xy += aOffset; - gl_Position = uProjectionMatrix * uModelViewMatrix * positionVec4; - } -`; - -const fragSrc = `#version 300 es - precision mediump float; - out vec4 outColor; - void main(){ - outColor = vec4(0.0, 1.0, 1.0, 1.0); - } -`; - -let myShader; -function setup(){ - createCanvas(100, 100, WEBGL); +p5.disableFriendlyErrors = true; +function windowResized() { + resizeCanvas(windowWidth, windowHeight); +} - // Create and use the custom shader. - myShader = createShader(vertSrc, fragSrc); +let myModel; +let starShader; +let starStrokeShader; +let stars; + +function starShaderCallback() { + const time = uniformFloat(() => millis()); + getWorldInputs((inputs) => { + inputs.position.y += instanceID() * 20 - 1000; + inputs.position.x += 40*sin(time * 0.001 + instanceID()); + return inputs; + }); + getObjectInputs((inputs) => { + inputs.position *= sin(time*0.001 + instanceID()); + return inputs; + }) +} - describe('A wobbly, cyan circle on a gray background.'); +async function setup(){ + createCanvas(windowWidth, windowHeight, WEBGL); + stars = buildGeometry(() => sphere(20, 7, 4)) + starShader = baseMaterialShader().modify(starShaderCallback); + starStrokeShader = baseStrokeShader().modify(starShaderCallback) } function draw(){ - // Set the styles - background(125); + background(0,200,240); + orbitControl(); + // noStroke(); + + push(); + stroke(255,0,255) + fill(255,200,255) + strokeShader(starStrokeShader) + shader(starShader); + model(stars, 100); + pop(); + push(); + shader(baseMaterialShader()); noStroke(); - shader(myShader); - - // Draw the circle. - beginShape(); - for (let i = 0; i < 30; i++){ - const x = 40 * cos(i/30 * TWO_PI); - const y = 40 * sin(i/30 * TWO_PI); - - // Apply some noise to the coordinates. - const xOff = 10 * noise(x + millis()/1000) - 5; - const yOff = 10 * noise(y + millis()/1000) - 5; - - // Apply these noise values to the following vertex. - vertexProperty('aOffset', [xOff, yOff]); - vertex(x, y); - } - endShape(CLOSE); + rotateX(HALF_PI); + translate(0, 0, -250); + plane(10000) + pop(); } diff --git a/preview/global/vite.config.mjs b/preview/global/vite.config.mjs index e322f16aa4..ed95741666 100644 --- a/preview/global/vite.config.mjs +++ b/preview/global/vite.config.mjs @@ -17,7 +17,7 @@ export default defineConfig({ name: 'reload', configureServer(server) { const { ws, watcher } = server; - const buildLibPath = path.resolve(libPath, './p5.rollup.js'); + const buildLibPath = path.resolve(libPath, './p5.js'); watcher.add(buildLibPath); watcher.on('change', file => { if(file === buildLibPath){ diff --git a/src/image/filterRenderer2D.js b/src/image/filterRenderer2D.js index 243652d5f8..5e2dcc3107 100644 --- a/src/image/filterRenderer2D.js +++ b/src/image/filterRenderer2D.js @@ -234,7 +234,8 @@ class FilterRenderer2D { this._shader.setUniform('canvasSize', [this.pInst.width, this.pInst.height]); this._shader.setUniform('radius', Math.max(1, this.filterParameter)); this._shader.setUniform('filterParameter', this.filterParameter); - + this._shader.setDefaultUniforms(); + this.pInst.states.setValue('rectMode', constants.CORNER); this.pInst.states.setValue('imageMode', constants.CORNER); this.pInst.blendMode(constants.BLEND); diff --git a/src/webgl/ShaderGenerator.js b/src/webgl/ShaderGenerator.js new file mode 100644 index 0000000000..7dfd76be32 --- /dev/null +++ b/src/webgl/ShaderGenerator.js @@ -0,0 +1,891 @@ +/** +* @module 3D +* @submodule ShaderGenerator +* @for p5 +* @requires core +*/ + +import { parse } from 'acorn'; +import { ancestor } from 'acorn-walk'; +import escodegen from 'escodegen'; + +function shadergenerator(p5, fn) { + let GLOBAL_SHADER; + + const oldModify = p5.Shader.prototype.modify + + p5.Shader.prototype.modify = function(shaderModifier, options = { parser: true, srcLocations: false }) { + if (shaderModifier instanceof Function) { + let generatorFunction; + if (options.parser) { + const sourceString = shaderModifier.toString() + const ast = parse(sourceString, { + ecmaVersion: 2021, + locations: options.srcLocations + }); + ancestor(ast, ASTCallbacks); + const transpiledSource = escodegen.generate(ast); + generatorFunction = new Function( + transpiledSource.slice( + transpiledSource.indexOf("{") + 1, + transpiledSource.lastIndexOf("}") + ) + ); + } else { + generatorFunction = shaderModifier; + } + const generator = new ShaderGenerator(generatorFunction, this, options.srcLocations) + const generatedModifyArgument = generator.generate(); + console.log("SRC STRING: ", generatorFunction); + console.log("NEW OPTIONS:", generatedModifyArgument) + return oldModify.call(this, generatedModifyArgument); + } + else { + return oldModify.call(this, shaderModifier) + } + } + + // AST Transpiler Callbacks and their helpers + function replaceBinaryOperator(codeSource) { + switch (codeSource) { + case '+': return 'add'; + case '-': return 'sub'; + case '*': return 'mult'; + case '/': return 'div'; + case '%': return 'mod'; + } + } + + const ASTCallbacks = { + VariableDeclarator(node, _state, _ancestors) { + if (node.init.callee && node.init.callee.name?.startsWith('uniform')) { + const uniformNameLiteral = { + type: 'Literal', + value: node.id.name + } + node.init.arguments.unshift(uniformNameLiteral); + } + }, + // The callbacks for AssignmentExpression and BinaryExpression handle + // operator overloading including +=, *= assignment expressions + AssignmentExpression(node, _state, _ancestors) { + if (node.operator !== '=') { + const methodName = replaceBinaryOperator(node.operator.replace('=','')); + const rightReplacementNode = { + type: 'CallExpression', + callee: { + type: "MemberExpression", + object: node.left, + property: { + type: "Identifier", + name: methodName, + }, + computed: false, + }, + arguments: [node.right] + } + node.operator = '='; + node.right = rightReplacementNode; + } + }, + BinaryExpression(node, _state, ancestors) { + // Don't convert uniform default values to node methods, as + // they should be evaluated at runtime, not compiled. + const isUniform = (ancestor) => { + return ancestor.type === 'CallExpression' + && ancestor.callee?.type === 'Identifier' + && ancestor.callee?.name.startsWith('uniform'); + } + if (ancestors.some(isUniform)) { + return; + } + // If the left hand side of an expression is one of these types, + // we should construct a node from it. + const unsafeTypes = ["Literal", "ArrayExpression"] + if (unsafeTypes.includes(node.left.type)) { + const leftReplacementNode = { + type: "CallExpression", + callee: { + type: "Identifier", + name: "makeNode", + }, + arguments: [node.left, node.right] + } + node.left = leftReplacementNode; + } + + // Replace the binary operator with a call expression + // in other words a call to BaseNode.mult(), .div() etc. + node.type = 'CallExpression'; + node.callee = { + type: "MemberExpression", + object: node.left, + property: { + type: "Identifier", + name: replaceBinaryOperator(node.operator), + }, + }; + node.arguments = [node.right]; + }, + } + + // This unfinished function lets you do 1 * 10 + // and turns it into float.mult(10) + fn.makeNode = function(leftValue, rightValue) { + if (typeof leftValue === 'number') { + return new FloatNode(leftValue); + } + } + + // Javascript Node API. + // These classes are for expressing GLSL functions in Javascript without + // needing to transpile the user's code. + + + class BaseNode { + constructor(isInternal, type) { + if (new.target === BaseNode) { + throw new TypeError("Cannot construct BaseNode instances directly. This is an abstract class."); + } + this.type = type; + this.componentNames = []; + this.swizzleChanged = false; + // For tracking recursion depth and creating temporary variables + this.isInternal = isInternal; + this.usedIn = []; + this.dependsOn = []; + this.srcLine = null; + // Stack Capture is used to get the original line of user code for Debug purposes + if (GLOBAL_SHADER.srcLocations === true && isInternal === false) { + try { + throw new Error("StackCapture"); + } catch (e) { + const lines = e.stack.split("\n"); + let userSketchLineIndex = 5; + if (isBinaryOperatorNode(this)) { userSketchLineIndex--; }; + this.srcLine = lines[userSketchLineIndex].trim(); + } + } + } + // get type() { + // return this._type; + // } + + // set type(value) { + // this._type = value; + // } + + addVectorComponents() { + if (this.type.startsWith('vec')) { + const vectorDimensions = +this.type.slice(3); + this.componentNames = ['x', 'y', 'z', 'w'].slice(0, vectorDimensions); + + for (let componentName of this.componentNames) { + // let value = new FloatNode() + let value = new ComponentNode(this, componentName, 'float', true); + Object.defineProperty(this, componentName, { + get() { + return value; + }, + set(newValue) { + this.swizzleChanged = true; + value = newValue; + } + }) + } + } + } + + // The base node implements a version of toGLSL which determines whether the generated code should be stored in a temporary variable. + toGLSLBase(context){ + if (this.shouldUseTemporaryVariable()) { + return this.getTemporaryVariable(context); + } + else { + return this.toGLSL(context); + } + } + + shouldUseTemporaryVariable() { + if (this.swizzleChanged) { return true; } + if (this.isInternal || isVariableNode(this)) { return false; } + let score = 0; + score += isBinaryOperatorNode(this); + score += isVectorNode(this) * 2; + score += this.usedIn.length; + return score > 3; + } + + getTemporaryVariable(context) { + if (!this.temporaryVariable) { + this.temporaryVariable = `temp_${context.getNextID()}`; + let line = ""; + if (this.srcLine) { + line += `\n// From ${this.srcLine}\n`; + } + if (this.swizzleChanged) { + const valueArgs = []; + for (let componentName of this.componentNames) { + valueArgs.push(this[componentName]) + } + const replacement = nodeConstructors[this.type](valueArgs) + line += this.type + " " + this.temporaryVariable + " = " + this.toGLSL(context) + ";"; + line += `\n` + this.temporaryVariable + " = " + replacement.toGLSL(context) + ";"; + } else { + line += this.type + " " + this.temporaryVariable + " = " + this.toGLSL(context) + ";"; + } + context.declarations.push(line); + } + return this.temporaryVariable; + }; + + // Binary Operators + add(other) { return new BinaryOperatorNode(this, this.enforceType(other), '+'); } + sub(other) { return new BinaryOperatorNode(this, this.enforceType(other), '-'); } + mult(other) { return new BinaryOperatorNode(this, this.enforceType(other), '*'); } + div(other) { return new BinaryOperatorNode(this, this.enforceType(other), '/'); } + mod(other) { return new ModulusNode(this, this.enforceType(other)); } + + // Check that the types of the operands are compatible. + enforceType(other){ + if (isShaderNode(other)){ + if (!isGLSLNativeType(other.type)) { + throw new TypeError (`You've tried to perform an operation on a struct of type: ${other.type}. Try accessing a member on that struct with '.'`) + } + if (!isGLSLNativeType(other.type)) { + throw new TypeError (`You've tried to perform an operation on a struct of type: ${other.type}. Try accessing a member on that struct with '.'`) + } + if ((isFloatNode(this) || isVectorNode(this)) && isIntNode(other)) { + return new FloatNode(other) + } + return other; + } + else if(typeof other === 'number') { + if (isIntNode(this)) { + return new IntNode(other); + } + return new FloatNode(other); + } + else if(Array.isArray(other)) { + return new VectorNode(other, `vec${other.length}`) + } + else { + return new this.constructor(other); + } + } + + toFloat() { + if (isFloatNode(this)) { + return this; + } else if (isIntNode(this)) { + return new FloatNode(this); + } else { + throw new TypeError(`Can't convert from type '${this.type}' to 'float'.`) + } + } + + toGLSL(context){ + throw new TypeError("Not supposed to call this function on BaseNode, which is an abstract class."); + } + } + + // Primitive Types + class IntNode extends BaseNode { + constructor(x = 0, isInternal = false) { + super(isInternal, 'int'); + this.x = x; + } + + toGLSL(context) { + if (isShaderNode(this.x)) { + let code = this.x.toGLSLBase(context); + return isIntNode(this.x.type) ? code : `int(${code})`; + } + else if (typeof this.x === "number") { + return `${Math.floor(this.x)}`; + } + else { + return `int(${this.x})`; + } + } + } + + class FloatNode extends BaseNode { + constructor(x = 0, isInternal = false){ + super(isInternal, 'float'); + this.x = x; + } + + toGLSL(context) { + if (isShaderNode(this.x)) { + let code = this.x.toGLSLBase(context); + return isFloatNode(this.x) ? code : `float(${code})`; + } + else if (typeof this.x === "number") { + return `${this.x.toFixed(4)}`; + } + else { + return `float(${this.x})`; + } + } + } + + class VectorNode extends BaseNode { + constructor(values, type, isInternal = false) { + super(isInternal, type); + this.componentNames = ['x', 'y', 'z', 'w'].slice(0, values.length); + this.componentNames.forEach((component, i) => { + this[component] = new FloatNode(values[i], true); + }); + } + + toGLSL(context) { + let glslArgs = ``; + + this.componentNames.forEach((component, i) => { + const comma = i === this.componentNames.length - 1 ? `` : `, `; + glslArgs += `${this[component].toGLSLBase(context)}${comma}`; + }) + + return `${this.type}(${glslArgs})`; + } + } + + // Function Call Nodes + class FunctionCallNode extends BaseNode { + constructor(name, args, properties, isInternal = false) { + let inferredType = args.find((arg, i) => { + properties.args[i] === 'genType' + && isShaderNode(arg) + })?.type; + if (!inferredType) { + let arrayArg = args.find(arg => Array.isArray(arg)); + inferredType = arrayArg ? `vec${arrayArg.length}` : undefined; + } + if (!inferredType) { + inferredType = 'float'; + } + args = args.map((arg, i) => { + if (!isShaderNode(arg)) { + const typeName = properties.args[i] === 'genType' ? inferredType : properties.args[i]; + arg = nodeConstructors[typeName](arg); + } + return arg; + }) + if (properties.returnType === 'genType') { + properties.returnType = inferredType; + } + super(isInternal, properties.returnType); + this.name = name; + this.args = args; + this.argumentTypes = properties.args; + this.addVectorComponents(); + } + + deconstructArgs(context) { + let argsString = this.args.map((argNode, i) => { + if (isIntNode(argNode) && this.argumentTypes[i] != 'float') { + argNode = argNode.toFloat(); + } + return argNode.toGLSLBase(context); + }).join(', '); + return argsString; + } + + toGLSL(context) { + return `${this.name}(${this.deconstructArgs(context)})`; + } + } + + // Variables and member variable nodes + class VariableNode extends BaseNode { + constructor(name, type, isInternal = false) { + super(isInternal, type); + this.name = name; + this.addVectorComponents(); + } + + + toGLSL(context) { + return `${this.name}`; + } + } + + class ComponentNode extends BaseNode { + constructor(parent, componentName, type, isInternal = false) { + super(isInternal, type); + this.parent = parent; + this.componentName = componentName; + this.type = type; + } + toGLSL(context) { + const parentName = this.parent.toGLSLBase(context); + // const parentName = this.parent.temporaryVariable ? this.parent.temporaryVariable : this.parent.name; + return `${parentName}.${this.componentName}`; + } + } + + // Binary Operator Nodes + class BinaryOperatorNode extends BaseNode { + constructor(a, b, operator, isInternal = false) { + super(isInternal, null); + this.op = operator; + this.a = a; + this.b = b; + for (const operand of [a, b]) { + operand.usedIn.push(this); + } + this.type = this.determineType(); + this.addVectorComponents(); + } + + // We know that both this.a and this.b are nodes because of BaseNode.enforceType + determineType() { + if (this.a.type === this.b.type) { + return this.a.type; + } + else if (isVectorNode(this.a) && isFloatNode(this.b)) { + return this.a.type; + } + else if (isVectorNode(this.b) && isFloatNode(this.a)) { + return this.b.type; + } + else if (isFloatNode(this.a) && isIntNode(this.b) + || isIntNode(this.a) && isFloatNode(this.b) + ) { + return 'float'; + } + else { + throw new Error("Incompatible types for binary operator"); + } + } + + processOperand(operand, context) { + if (operand.temporaryVariable) { return operand.temporaryVariable; } + let code = operand.toGLSLBase(context); + if (isBinaryOperatorNode(operand) && !operand.temporaryVariable) { + code = `(${code})`; + } + if (this.type === 'float' && isIntNode(operand)) { + code = `float(${code})`; + } + return code; + } + + toGLSL(context) { + const a = this.processOperand(this.a, context); + const b = this.processOperand(this.b, context); + return `${a} ${this.op} ${b}`; + } + } + + // TODO: Correct the implementation for floats/ genType etc + class ModulusNode extends BinaryOperatorNode { + constructor(a, b) { + super(a, b); + } + toGLSL(context) { + // Switch on type between % or mod() + if (isVectorNode(this) || isFloatNode(this)) { + return `mod(${this.a.toGLSLBase(context)}, ${this.b.toGLSLBase(context)})`; + } + return `${this.processOperand(context, this.a)} % ${this.processOperand(context, this.b)}`; + } + } + + // TODO: finish If Node + class ConditionalNode extends BaseNode { + constructor(value) { + super(value); + this.value = value; + this.condition = null; + this.thenBranch = null; + this.elseBranch = null; + } + // conditions + equalTo(value){} + greaterThan(value) {} + greaterThanEqualTo(value) {} + lessThan(value) {} + lessThanEqualTo(value) {} + // modifiers + not() {} + or() {} + and() {} + // returns + thenReturn(value) {} + elseReturn(value) {} + // + thenDiscard() { + new ConditionalDiscard(this.condition); + }; + }; + + class ConditionalDiscard extends BaseNode { + constructor(condition){ + this.condition = condition; + } + toGLSL(context) { + context.discardConditions.push(`if(${this.condition}{discard;})`); + } + } + + fn.if = function (value) { + return new ConditionalNode(value); + } + + // Node Helper functions + function isShaderNode(node) { + return (node instanceof BaseNode); + } + + function isIntNode(node) { + return (isShaderNode(node) && (node.type === 'int')); + } + + function isFloatNode(node) { + return (isShaderNode(node) && (node.type === 'float')); + } + + function isVectorNode(node) { + return (isShaderNode(node) && (node.type === 'vec2'|| node.type === 'vec3' || node.type === 'vec4')); + } + + function isBinaryOperatorNode(node) { + return (node instanceof BinaryOperatorNode); + } + + function isVariableNode(node) { + return (node instanceof VariableNode || node instanceof ComponentNode || typeof(node.temporaryVariable) != 'undefined'); + } + + // Helper function to check if a type is a user defined struct or native type + function isGLSLNativeType(typeName) { + // Supported types for now + const glslNativeTypes = ['int', 'float', 'vec2', 'vec3', 'vec4', 'sampler2D']; + return glslNativeTypes.includes(typeName); + } + + // Helper function to check if a type is a user defined struct or native type + function isGLSLNativeType(typeName) { + // Supported types for now + const glslNativeTypes = ['int', 'float', 'vec2', 'vec3', 'vec4', 'sampler2D']; + return glslNativeTypes.includes(typeName); + } + + // Shader Generator + // This class is responsible for converting the nodes into an object containing GLSL code, to be used by p5.Shader.modify + + class ShaderGenerator { + constructor(userCallback, originalShader, srcLocations) { + GLOBAL_SHADER = this; + this.userCallback = userCallback; + this.userCallback = userCallback; + this.srcLocations = srcLocations; + this.cleanup = () => {}; + this.generateHookOverrides(originalShader); + this.output = { + uniforms: {}, + } + this.resetGLSLContext(); + this.isGenerating = false; + } + + generate() { + const prevFESDisabled = p5.disableFriendlyErrors; + // We need a custom error handling system within shader generation + p5.disableFriendlyErrors = true; + + this.isGenerating = true; + this.userCallback(); + this.isGenerating = false; + + this.cleanup(); + p5.disableFriendlyErrors = prevFESDisabled; + return this.output; + } + + // This method generates the hook overrides which the user calls in their modify function. + generateHookOverrides(originalShader) { + const availableHooks = { + ...originalShader.hooks.vertex, + ...originalShader.hooks.fragment, + } + + const windowOverrides = {}; + + Object.keys(availableHooks).forEach((hookName) => { + const hookTypes = originalShader.hookTypes(hookName); + console.log(hookTypes); + this[hookTypes.name] = function(userCallback) { + // Create the initial nodes which are passed to the user callback + // Also generate a string of the arguments for the code generation + const argNodes = [] + const argsArray = []; + + hookTypes.parameters.forEach((parameter) => { + // For hooks with structs as input we should pass an object populated with variable nodes + if (!isGLSLNativeType(parameter.type.typeName)) { + const structArg = {}; + parameter.type.properties.forEach((property) => { + structArg[property.name] = new VariableNode(`${parameter.name}.${property.name}`, property.type.typeName, true); + }); + argNodes.push(structArg); + } else { + argNodes.push( + new VariableNode(parameter.name, parameter.type.typeName, true) + ); + } + const qualifiers = parameter.type.qualifiers.length > 0 ? parameter.type.qualifiers.join(' ') : ''; + argsArray.push(`${qualifiers} ${parameter.type.typeName} ${parameter.name}`.trim()) + }) + + let returnedValue = userCallback(...argNodes); + const expectedReturnType = hookTypes.returnType; + const toGLSLResults = {}; + + // If the expected return type is a struct we need to evaluate each of its properties + if (!isGLSLNativeType(expectedReturnType.typeName)) { + Object.entries(returnedValue).forEach(([propertyName, propertyNode]) => { + toGLSLResults[propertyName] = propertyNode.toGLSLBase(this.context); + }) + } else { + // We can accept raw numbers or arrays otherwise + if (!isShaderNode(returnedValue)) { + returnedValue = nodeConstructors[expectedReturnType.typeName](returnedValue) + } + toGLSLResults['notAProperty'] = returnedValue.toGLSLBase(this.context); + } + + // Build the final GLSL string. + // The order of this code is a bit confusing, we need to call toGLSLBase + let codeLines = [ + `(${argsArray.join(', ')}) {`, + ...this.context.declarations, + `${hookTypes.returnType.typeName} finalReturnValue;` + ]; + + Object.entries(toGLSLResults).forEach(([propertyName, result]) => { + const propString = expectedReturnType.properties ? `.${propertyName}` : ''; + codeLines.push(`finalReturnValue${propString} = ${result};`) + }) + + codeLines.push('return finalReturnValue;', '}'); + this.output[hookName] = codeLines.join('\n'); + this.resetGLSLContext(); + } + + windowOverrides[hookTypes.name] = window[hookTypes.name]; + + // Expose the Functions to global scope for users to use + window[hookTypes.name] = function(userOverride) { + GLOBAL_SHADER[hookTypes.name](userOverride); + }; + }); + + this.cleanup = () => { + for (const key in windowOverrides) { + window[key] = windowOverrides[key]; + } + }; + } + + resetGLSLContext() { + this.context = { + id: 0, + getNextID: function() { return this.id++ }, + declarations: [], + } + } + } + + // User function helpers + function conformVectorParameters(value, vectorDimensions) { + // Allow arguments as arrays ([0,0,0,0]) or not (0,0,0,0) + value = value.flat(); + // Populate arguments so uniformVector3(0) becomes [0,0,0] + if (value.length === 1) { + value = Array(vectorDimensions).fill(value[0]); + } + return value; + } + + // User functions + fn.instanceID = function() { + return new VariableNode('gl_InstanceID', 'int'); + } + + fn.discard = function() { + return new VariableNode('discard', 'keyword'); + } + + // Generating uniformFloat, uniformVec, createFloat, etc functions + // Maps a GLSL type to the name suffix for method names + const GLSLTypesToIdentifiers = { + int: 'Int', + float: 'Float', + vec2: 'Vector2', + vec3: 'Vector3', + vec4: 'Vector4', + sampler2D: 'Texture', + }; + + const nodeConstructors = { + int: (value) => new IntNode(value), + float: (value) => new FloatNode(value), + vec2: (value) => new VectorNode(value, 'vec2'), + vec3: (value) => new VectorNode(value, 'vec3'), + vec4: (value) => new VectorNode(value, 'vec4'), + }; + + for (const glslType in GLSLTypesToIdentifiers) { + // Generate uniform*() Methods for creating uniforms + const typeIdentifier = GLSLTypesToIdentifiers[glslType]; + const uniformMethodName = `uniform${typeIdentifier}`; + + ShaderGenerator.prototype[uniformMethodName] = function(...args) { + let [name, ...defaultValue] = args; + + if(glslType.startsWith('vec')) { + defaultValue = conformVectorParameters(defaultValue, +glslType.slice(3)); + this.output.uniforms[`${glslType} ${name}`] = defaultValue; + } + else { + this.output.uniforms[`${glslType} ${name}`] = defaultValue[0]; + } + const uniform = new VariableNode(name, glslType, false); + return uniform; + }; + + fn[uniformMethodName] = function (...args) { + return GLOBAL_SHADER[uniformMethodName](...args); + }; + + // We don't need a createTexture method. + if (glslType === 'sampler2D') { continue; } + + // Generate the create*() Methods for creating variables in shaders + const createMethodName = `create${typeIdentifier}`; + fn[createMethodName] = function (...value) { + if(glslType.startsWith('vec')) { + value = conformVectorParameters(value, +glslType.slice(3)); + } else { + value = value[0]; + } + return nodeConstructors[glslType](value); + } + } + + // GLSL Built in functions + // Add a whole lot of these functions. + // https://docs.gl/el3/abs + // In reality many of these have multiple overrides which will need to address later. + // Also, their return types depend on the genType which will need to address urgently + // genType clamp(genType x, + // genType minVal, + // genType maxVal); + // genType clamp(genType x, + // float minVal, + // float maxVal); + const builtInGLSLFunctions = { + //////////// Trigonometry ////////// + 'acos': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'acosh': { args: ['genType'], returnType: 'genType', isp5Function: false}, + 'asin': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'asinh': { args: ['genType'], returnType: 'genType', isp5Function: false}, + 'atan': { args: ['genType', 'genType'], returnType: 'genType', isp5Function: false}, + 'atanh': { args: ['genType'], returnType: 'genType', isp5Function: false}, + 'cos': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'cosh': { args: ['genType'], returnType: 'genType', isp5Function: false}, + 'degrees': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'radians': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'sin': { args: ['genType'], returnType: 'genType' , isp5Function: true}, + 'sinh': { args: ['genType'], returnType: 'genType', isp5Function: false}, + 'tan': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'tanh': { args: ['genType'], returnType: 'genType', isp5Function: false}, + + ////////// Mathematics ////////// + 'abs': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'ceil': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'clamp': { args: ['genType', 'genType', 'genType'], returnType: 'genType', isp5Function: false}, + 'dFdx': { args: ['genType'], returnType: 'genType', isp5Function: false}, + 'dFdy': { args: ['genType'], returnType: 'genType', isp5Function: false}, + 'exp': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'exp2': { args: ['genType'], returnType: 'genType', isp5Function: false}, + 'floor': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'fma': { args: ['genType', 'genType', 'genType'], returnType: 'genType', isp5Function: false}, + 'fract': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'fwidth': { args: ['genType'], returnType: 'genType', isp5Function: false}, + 'inversesqrt': { args: ['genType'], returnType: 'genType', isp5Function: true}, + // 'isinf': {}, + // 'isnan': {}, + 'log': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'log2': { args: ['genType'], returnType: 'genType', isp5Function: false}, + 'max': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'min': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'mix': { args: ['genType'], returnType: 'genType', isp5Function: false}, + // 'mod': {}, + // 'modf': {}, + 'pow': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'round': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'roundEven': { args: ['genType'], returnType: 'genType', isp5Function: false}, + // 'sign': {}, + 'smoothstep': { args: ['genType', 'genType', 'genType'], returnType: 'genType', isp5Function: false}, + 'sqrt': { args: ['genType'], returnType: 'genType', isp5Function: true}, + 'step': { args: ['genType', 'genType'], returnType: 'genType', isp5Function: false}, + 'trunc': { args: ['genType'], returnType: 'genType', isp5Function: false}, + + ////////// Vector ////////// + 'cross': { args: ['vec3', 'vec3'], returnType: 'vec3', isp5Function: true}, + 'distance': { args: ['genType', 'genType'], returnType: 'float', isp5Function: true}, + 'dot': { args: ['genType', 'genType'], returnType: 'float', isp5Function: true}, + // 'equal': {}, + 'faceforward': { args: ['genType', 'genType', 'genType'], returnType: 'genType', isp5Function: false}, + 'length': { args: ['genType'], returnType: 'float', isp5Function: false}, + 'normalize': { args: ['genType'], returnType: 'genType', isp5Function: true}, + // 'notEqual': {}, + 'reflect': { args: ['genType', 'genType'], returnType: 'genType', isp5Function: false}, + 'refract': { args: ['genType', 'genType', 'float'], returnType: 'genType', isp5Function: false}, + // Texture sampling + 'texture': {args: ['sampler2D', 'vec2'], returnType: 'vec4', isp5Function: true}, + } + + Object.entries(builtInGLSLFunctions).forEach(([functionName, properties]) => { + if (properties.isp5Function) { + const originalFn = fn[functionName]; + + fn[functionName] = function (...args) { + if (GLOBAL_SHADER?.isGenerating) { + return new FunctionCallNode(functionName, args, properties) + } else { + return originalFn.apply(this, args); + } + } + } else { + fn[functionName] = function (...args) { + if (GLOBAL_SHADER?.isGenerating) { + return new FunctionCallNode(functionName, args, properties); + } else { + p5._friendlyError( + `It looks like you've called ${functionName} outside of a shader's modify() function.` + ); + } + } + } + }) + + // const oldTexture = p5.prototype.texture; + // p5.prototype.texture = function(...args) { + // if (isShaderNode(args[0])) { + // return new FunctionCallNode('texture', args, 'vec4'); + // } else { + // return oldTexture.apply(this, args); + // } + // } +} + +export default shadergenerator; + +if (typeof p5 !== 'undefined') { + p5.registerAddon(shadergenerator) +} diff --git a/src/webgl/index.js b/src/webgl/index.js index c2515fce5a..7ba587b132 100644 --- a/src/webgl/index.js +++ b/src/webgl/index.js @@ -14,6 +14,7 @@ import shader from './p5.Shader'; import camera from './p5.Camera'; import texture from './p5.Texture'; import rendererGL from './p5.RendererGL'; +import shadergenerator from './ShaderGenerator'; export default function(p5){ rendererGL(p5, p5.prototype); @@ -32,4 +33,5 @@ export default function(p5){ dataArray(p5, p5.prototype); shader(p5, p5.prototype); texture(p5, p5.prototype); + shadergenerator(p5, p5.prototype); } diff --git a/src/webgl/p5.RendererGL.js b/src/webgl/p5.RendererGL.js index 4f988c425c..3e1b440cc4 100644 --- a/src/webgl/p5.RendererGL.js +++ b/src/webgl/p5.RendererGL.js @@ -636,7 +636,7 @@ class RendererGL extends Renderer { this._useVertexColor = geometry.vertexColors.length > 0; const shader = - this._drawingFilter && this.states.userFillShader + !this._drawingFilter && this.states.userFillShader ? this.states.userFillShader : this._getFillShader(); shader.bindShader();