Skip to content

Commit 474c5ab

Browse files
committed
chore: initial commit
0 parents  commit 474c5ab

12 files changed

+6264
-0
lines changed

.eslintrc.json

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
{
2+
"root": true,
3+
"parser": "@typescript-eslint/parser",
4+
"plugins": ["@typescript-eslint"],
5+
"extends": ["eslint:recommended", "plugin:@typescript-eslint/recommended"]
6+
}

.gitignore

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
.DS_Store
2+
Thumbs.db
3+
4+
node_modules
5+
dist

.prettierrc.yaml

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
singleQuote: true

.travis.yml

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
language: node_js
2+
node_js:
3+
- '12'
4+
os: linux
5+
dist: xenial
6+
install:
7+
- npm i

README.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
# UniformPlugin for `shader-art`
2+
3+
This plugin lets you configure uniform variables. Uses dat.GUI.
4+
5+
Example: https://codepen.io/terabaud/pen/jOymzJN?editors=1000

jest.config.cjs

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
module.exports = {
2+
preset: 'ts-jest',
3+
testEnvironment: 'jsdom',
4+
setupFiles: ['jest-webgl-canvas-mock'],
5+
};

package-lock.json

+5,849
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+46
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
{
2+
"name": "@shader-art/plugin-uniform",
3+
"version": "0.0.1",
4+
"description": "uniform variables with dat.gui for shader-art",
5+
"repository": "shader-art/plugin-uniform",
6+
"type": "module",
7+
"main": "./dist/index.js",
8+
"types": "./dist/index.d.ts",
9+
"source": "./src/index.ts",
10+
"unpkg": "./dist/index.esm.js",
11+
"files": [
12+
"dist"
13+
],
14+
"exports": {
15+
"import": "./dist/index.esm.js",
16+
"require": "./dist/index.cjs"
17+
},
18+
"scripts": {
19+
"build": "npm run build:types -s && npm run build:js -s && npm run build:cjs -s",
20+
"build:types": "tsc -d --emitDeclarationOnly",
21+
"build:js": "esbuild --format=esm --bundle --minify src/index.ts > dist/index.esm.js",
22+
"build:cjs": "esbuild --format=cjs --bundle --minify src/index.ts > dist/index.cjs",
23+
"lint": "npm run lint:js -s && npm run lint:package -s",
24+
"lint:js": "eslint src/**/*.ts",
25+
"lint:package": "npx @skypack/package-check",
26+
"test": "jest -c jest.config.cjs"
27+
},
28+
"keywords": ["webgl"],
29+
"author": "Lea Rosema",
30+
"license": "MIT",
31+
"devDependencies": {
32+
"@shader-art/plugin-base": "^0.1.1",
33+
"@types/dat.gui": "^0.7.6",
34+
"@types/jest": "^26.0.22",
35+
"@typescript-eslint/eslint-plugin": "^4.20.0",
36+
"@typescript-eslint/parser": "^4.20.0",
37+
"eslint": "^7.23.0",
38+
"jest": "^26.6.3",
39+
"jest-webgl-canvas-mock": "^0.2.3",
40+
"ts-jest": "^26.5.4",
41+
"typescript": "^4.2.3"
42+
},
43+
"dependencies": {
44+
"dat.gui": "^0.7.7"
45+
}
46+
}

src/index.test.ts

+113
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,113 @@
1+
import { ShaderArtPlugin } from '@shader-art/plugin-base';
2+
import { ShaderArtShim } from './test-utils/shader-art-shim';
3+
4+
jest.mock('dat.gui', () => ({
5+
GUI: function () {
6+
const that: any = {
7+
add: jest.fn().mockImplementation(() => that),
8+
addColor: jest.fn().mockImplementation(() => that),
9+
min: jest.fn().mockImplementation(() => that),
10+
max: jest.fn().mockImplementation(() => that),
11+
step: jest.fn().mockImplementation(() => that),
12+
destroy: jest.fn().mockImplementation(() => that),
13+
onChange: jest.fn(),
14+
};
15+
return that;
16+
},
17+
}));
18+
19+
import { UniformPlugin } from './index';
20+
21+
const html = (x: any) => x;
22+
23+
function asynced(fn: (...args: any[]) => void, timeout = 0): Promise<void> {
24+
return new Promise((resolve) => {
25+
setTimeout(() => {
26+
fn();
27+
resolve();
28+
}, timeout);
29+
});
30+
}
31+
32+
const vertexShader = html`
33+
<script type="vert">
34+
precision highp float;
35+
attribute vec4 position;
36+
void main() {
37+
gl_Position = position;
38+
}
39+
</script>
40+
`;
41+
42+
const fragmentShader = html`
43+
<script type="frag">
44+
precision highp float;
45+
uniform vec2 resolution;
46+
unifrom texture2D texture;
47+
void main() {
48+
vec2 p = gl_FragCoord.xy / resolution;
49+
gl_FragColor = texture2D(texture, p);
50+
}
51+
</script>
52+
`;
53+
54+
const uniformFloat = html`
55+
<uniform type="float" name="testFloat" value="123." />
56+
`;
57+
58+
const uniformColor = html`
59+
<uniform type="float" name="testColor" value="#ff00ff" />
60+
`;
61+
62+
const createShaderArt = (html: string): ShaderArtShim => {
63+
const element = document.createElement('shader-art');
64+
element.setAttribute('autoplay', '');
65+
element.innerHTML = html;
66+
document.body.appendChild(element);
67+
return element as ShaderArtShim;
68+
};
69+
70+
describe('UniformPlugin tests', () => {
71+
beforeAll(() => {
72+
ShaderArtShim.register([() => new UniformPlugin()]);
73+
});
74+
75+
test('shader-art creation', () => {
76+
const element = createShaderArt(vertexShader + fragmentShader);
77+
expect(element).toBeDefined();
78+
expect(element.canvas).toBeInstanceOf(HTMLCanvasElement);
79+
expect(element.activePlugins.map((p: ShaderArtPlugin) => p.name)).toContain(
80+
'UniformPlugin'
81+
);
82+
});
83+
84+
test('shader-art creates a uniform float', () => {
85+
const element = createShaderArt(
86+
uniformFloat + vertexShader + fragmentShader
87+
);
88+
expect(element).toBeDefined();
89+
expect(element.canvas).toBeInstanceOf(HTMLCanvasElement);
90+
expect(
91+
(element.activePlugins[0] as UniformPlugin)?.gui?.add
92+
).toHaveBeenCalled();
93+
});
94+
95+
test('shader-art creates a uniform color', () => {
96+
const element = createShaderArt(
97+
uniformColor + vertexShader + fragmentShader
98+
);
99+
expect(element).toBeDefined();
100+
expect(element.canvas).toBeInstanceOf(HTMLCanvasElement);
101+
expect(
102+
(element.activePlugins[0] as UniformPlugin)?.gui?.add
103+
).toHaveBeenCalled();
104+
});
105+
106+
afterEach(() => {
107+
const sc = document.querySelector('shader-art');
108+
if (sc) {
109+
sc.remove();
110+
}
111+
jest.resetAllMocks();
112+
});
113+
});

src/index.ts

+69
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,69 @@
1+
import { ShaderArtPlugin } from '@shader-art/plugin-base';
2+
import dat from 'dat.gui';
3+
4+
function color(str: string): number[] {
5+
if (/^#[0-9a-f]{6}$/g.test(str)) {
6+
const r = parseInt(str.slice(1, 3), 16) / 255;
7+
const g = parseInt(str.slice(3, 5), 16) / 255;
8+
const b = parseInt(str.slice(5, 7), 16) / 255;
9+
return [r, g, b];
10+
}
11+
return [0, 0, 0];
12+
}
13+
14+
export class UniformPlugin implements ShaderArtPlugin {
15+
name = 'UniformPlugin';
16+
gui: dat.GUI | null = null;
17+
18+
setup(
19+
hostElement: HTMLElement,
20+
gl: WebGLRenderingContext | WebGL2RenderingContext,
21+
program: WebGLProgram,
22+
canvas: HTMLCanvasElement
23+
): void | Promise<void> {
24+
const gui = new dat.GUI();
25+
this.gui = gui;
26+
const elements = [...hostElement.querySelectorAll('uniform')];
27+
for (const el of elements) {
28+
const params: Record<string, number | string> = {};
29+
const name = el.getAttribute('name');
30+
const type = el.getAttribute('type');
31+
if (!name || !type) {
32+
continue;
33+
}
34+
if (type === 'float') {
35+
const value = parseFloat(el.getAttribute('value') || '0.');
36+
const min = parseFloat(el.getAttribute('min') || '0.');
37+
const max = parseFloat(el.getAttribute('max') || '0.');
38+
const step = parseFloat(el.getAttribute('step') || '0.');
39+
const useGui = !el.hasAttribute('no-gui');
40+
params[name] = value;
41+
const uName = gl.getUniformLocation(program, name);
42+
gl.uniform1f(uName, value);
43+
if (useGui) {
44+
gui
45+
.add(params, name)
46+
.min(min)
47+
.max(max)
48+
.step(step)
49+
.onChange(() => {
50+
gl.uniform1f(uName, params[name] as number);
51+
});
52+
}
53+
}
54+
if (type === 'color') {
55+
const value = el.getAttribute('value') || '#000000';
56+
params[name] = value;
57+
const uName = gl.getUniformLocation(program, name);
58+
gl.uniform3fv(uName, color(value));
59+
gui.addColor(params, name).onChange(() => {
60+
gl.uniform3fv(uName, color(params[name] as string));
61+
});
62+
}
63+
}
64+
}
65+
66+
dispose() {
67+
this.gui?.destroy();
68+
}
69+
}

src/test-utils/shader-art-shim.ts

+82
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,82 @@
1+
import { ShaderArtPlugin } from '@shader-art/plugin-base';
2+
3+
export class ShaderArtShim extends HTMLElement {
4+
canvas: HTMLCanvasElement | null = null;
5+
gl: WebGLRenderingContext | WebGL2RenderingContext | null = null;
6+
program: WebGLProgram | null = null;
7+
activePlugins: ShaderArtPlugin[] = [];
8+
initialized = false;
9+
10+
static plugins: (() => ShaderArtPlugin)[] = [];
11+
static register(plugins: (() => ShaderArtPlugin)[]): void {
12+
ShaderArtShim.plugins = plugins;
13+
if (typeof customElements.get('shader-art') === 'undefined') {
14+
customElements.define('shader-art', ShaderArtShim);
15+
}
16+
}
17+
18+
constructor() {
19+
super();
20+
}
21+
22+
setup(): void {
23+
const canvas = document.createElement('canvas');
24+
this.appendChild(canvas);
25+
26+
const gl = canvas.getContext('webgl');
27+
if (!gl) {
28+
throw new Error('create WebGLRenderingContext failed');
29+
}
30+
31+
const program = gl.createProgram();
32+
if (!program) {
33+
throw new Error('create WebGLProgram failed');
34+
}
35+
36+
this.gl = gl;
37+
this.canvas = canvas;
38+
this.program = program;
39+
40+
const pluginQueue: Promise<void>[] = [];
41+
for (const plugin of ShaderArtShim.plugins) {
42+
const instance = plugin();
43+
this.activePlugins.push(instance);
44+
const result = instance.setup(this, gl, program, canvas);
45+
if (result instanceof Promise) {
46+
pluginQueue.push(result);
47+
}
48+
}
49+
Promise.all(pluginQueue).then(() => {
50+
this.initialized = true;
51+
});
52+
}
53+
54+
connectedCallback(): void {
55+
if (!this.gl) {
56+
this.setup();
57+
}
58+
}
59+
60+
disconnectedCallback(): void {
61+
this.dispose();
62+
}
63+
64+
dispose(): void {
65+
if (!this.gl || !this.canvas) {
66+
return;
67+
}
68+
for (const plugin of this.activePlugins) {
69+
plugin.dispose();
70+
}
71+
this.activePlugins = [];
72+
this.gl = null;
73+
this.program = null;
74+
this.removeChild(this.canvas);
75+
}
76+
77+
render(): void {
78+
if (this.gl) {
79+
this.gl.drawArrays(this.gl.TRIANGLES, 0, 0);
80+
}
81+
}
82+
}

0 commit comments

Comments
 (0)