UNPKG

25.3 kBJavaScriptView Raw
1'use strict';
2
3Object.defineProperty(exports, "__esModule", {
4 value: true
5});
6exports.Shader = undefined;
7
8var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; };
9
10var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; };
11
12var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
13
14var _core = require('../core');
15
16var _utils = require('../utils');
17
18var _scope = require('../scope');
19
20function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
21
22function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
23
24function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
25
26var Shader = exports.Shader = function (_Component) {
27 _inherits(Shader, _Component);
28
29 _createClass(Shader, null, [{
30 key: 'defaults',
31 value: function defaults() {
32 return _extends({}, _core.ShaderLib.defaults(), { defines: {} });
33 }
34 }]);
35
36 function Shader(ctx) {
37 var initialState = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
38
39 _classCallCheck(this, Shader);
40
41 (0, _utils.assignDefaults)(initialState, Shader.defaults());
42 var defines = initialState.defines,
43 shaderName = initialState.shaderName;
44
45 var contextCache = {};
46 var shaderCache = {};
47 var shaderLib = new _core.ShaderLib(_extends({}, initialState, { defines: defines }));
48
49 var injectParentContext = new _scope.ScopedContext(ctx, {
50 defines: function (_defines) {
51 function defines(_x2) {
52 return _defines.apply(this, arguments);
53 }
54
55 defines.toString = function () {
56 return _defines.toString();
57 };
58
59 return defines;
60 }(function (_ref) {
61 var contextDefines = _ref.defines;
62
63 return _extends({}, contextDefines, defines);
64 })
65 });
66
67 var injectContext = null;
68
69 var fragmentShaderUncompiled = null;
70 var vertexShaderUncompiled = null;
71 var fragmentShader = null;
72 var vertexShader = null;
73
74 var _this = _possibleConstructorReturn(this, (Shader.__proto__ || Object.getPrototypeOf(Shader)).call(this, ctx, initialState, update));
75
76 function update(state, block, previousState) {
77 injectParentContext(function (reglContext) {
78 var _state$forceCompile = state.forceCompile,
79 forceCompile = _state$forceCompile === undefined ? false : _state$forceCompile;
80
81 Object.assign(defines, _extends({}, reglContext.defines, state.defines));
82
83 if (Object.keys(defines).length) {
84 if (shaderLib.preprocessor.define(defines)) {
85 forceCompile = true;
86 }
87 }
88
89 if (forceCompile || shouldCompile(reglContext, state)) {
90 compile(reglContext, state);
91 }
92
93 setInjectContext(reglContext, state);
94
95 if ('function' == typeof injectContext) {
96 injectContext(state, block);
97 } else {
98 block(state);
99 }
100 });
101 }
102
103 function getShaderFromCache(reglContext, currentState, shader) {
104 shader = getViableShader(reglContext, currentState, shader);
105 return shaderCache[shaderLib.hash(shader)];
106 }
107
108 function setInjectContext(reglContext, currentState) {
109 var opts = {
110 context: {
111 fragmentShader: function (_fragmentShader) {
112 function fragmentShader(_x3) {
113 return _fragmentShader.apply(this, arguments);
114 }
115
116 fragmentShader.toString = function () {
117 return _fragmentShader.toString();
118 };
119
120 return fragmentShader;
121 }(function (_ref2) {
122 var fs = _ref2.fragmentShader;
123 return fragmentShader || fs;
124 }),
125 vertexShader: function (_vertexShader) {
126 function vertexShader(_x4) {
127 return _vertexShader.apply(this, arguments);
128 }
129
130 vertexShader.toString = function () {
131 return _vertexShader.toString();
132 };
133
134 return vertexShader;
135 }(function (_ref3) {
136 var vs = _ref3.vertexShader;
137 return vertexShader || vs;
138 })
139 }
140 };
141
142 var requestedFragmentShader = getShaderFromCache(reglContext, currentState, currentState.fragmentShader);
143
144 var requestedVertexShader = getShaderFromCache(reglContext, currentState, currentState.vertexShader);
145
146 if (requestedFragmentShader && requestedFragmentShader != fragmentShader) {
147 fragmentShader = requestedFragmentShader;
148 }
149
150 if (requestedVertexShader && requestedVertexShader != vertexShader) {
151 vertexShader = requestedVertexShader;
152 }
153
154 if ('string' == typeof vertexShader) {
155 opts.vert = vertexShader;
156 }
157 if ('string' == typeof fragmentShader) {
158 opts.frag = fragmentShader;
159 }
160
161 if ('string' == typeof opts.vert || 'string' == typeof opts.frag) {
162 var hash = [shaderLib.hash(opts.vert), shaderLib.hash(opts.frag)].filter(Boolean).join('');
163 if (null == contextCache[hash]) {
164 injectContext = ctx.regl(opts);
165 contextCache[hash] = injectContext;
166 injectContext.opts = opts;
167 } else {
168 injectContext = contextCache[hash];
169 }
170 }
171 }
172
173 function compile(reglContext, currentState) {
174 if (!isShaderCached(currentState.vertexShader)) {
175 compileVertexShader();
176 }
177
178 if (!isShaderCached(currentState.fragmentShader)) {
179 compileFragmentShader();
180 }
181
182 function compileShader(type, shader) {
183 var compiled = null;
184 var uncompiled = null;
185 if (isViableShader(shader)) {
186 uncompiled = getViableShader(reglContext, currentState, shader);
187 compiled = shaderLib.compile(shaderName + ' (' + type + ')', uncompiled);
188 compiled = shaderLib.preprocess(compiled);
189 return { compiled: compiled, uncompiled: uncompiled };
190 }
191 return null;
192 }
193
194 function isShaderCached(shader) {
195 return Boolean(getShaderFromCache(reglContext, currentState, shader));
196 }
197
198 function compileVertexShader() {
199 var result = compileShader('vertex', currentState.vertexShader);
200 if (result) {
201 vertexShader = result.compiled;
202 vertexShaderUncompiled = result.uncompiled;
203 shaderCache[shaderLib.hash(vertexShaderUncompiled)] = vertexShader;
204 }
205 }
206
207 function compileFragmentShader() {
208 var result = compileShader('fragment', currentState.fragmentShader);
209 if (result) {
210 fragmentShader = result.compiled;
211 fragmentShaderUncompiled = result.uncompiled;
212 shaderCache[shaderLib.hash(fragmentShaderUncompiled)] = fragmentShader;
213 }
214 }
215 }
216
217 function getViableShader(reglContext, currentState, shader) {
218 if ('string' == typeof shader) {
219 return shader;
220 } else if ('function' == typeof shader) {
221 return shader(reglContext, currentState);
222 }
223 }
224
225 function isViableShader(shader) {
226 return ['string', 'function'].indexOf(typeof shader === 'undefined' ? 'undefined' : _typeof(shader)) > -1;
227 }
228
229 function shouldCompile(reglContext, currentState) {
230 var needsCompile = false;
231 check('function' != typeof injectContext);
232 checkShader(vertexShaderUncompiled, currentState.vertexShader);
233 checkShader(fragmentShaderUncompiled, currentState.fragmentShader);
234 return needsCompile;
235
236 function check(cond) {
237 if (cond) {
238 needsCompile = true;
239 }
240 }
241
242 function checkShader(current, next) {
243 next = getViableShader(reglContext, currentState, next);
244 if (shaderCache[shaderLib.hash(next)]) {
245 return check(true);
246 } else if ('string' != typeof current && next) {
247 return check(true);
248 } else if ('string' == typeof next && current != next) {
249 return check(true);
250 }
251 }
252 }
253 return _this;
254 }
255
256 return Shader;
257}(_core.Component);
258//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/shader/shader.js"],"names":["Shader","defaults","defines","ctx","initialState","shaderName","contextCache","shaderCache","shaderLib","injectParentContext","contextDefines","injectContext","fragmentShaderUncompiled","vertexShaderUncompiled","fragmentShader","vertexShader","update","state","block","previousState","reglContext","forceCompile","Object","assign","keys","length","preprocessor","define","shouldCompile","compile","setInjectContext","getShaderFromCache","currentState","shader","getViableShader","hash","opts","context","fs","vs","requestedFragmentShader","requestedVertexShader","vert","frag","filter","Boolean","join","regl","isShaderCached","compileVertexShader","compileFragmentShader","compileShader","type","compiled","uncompiled","isViableShader","preprocess","result","indexOf","needsCompile","check","checkShader","cond","current","next"],"mappings":"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","file":"shader.js","sourcesContent":["import { Component, ShaderLib } from '../core'\nimport { assignDefaults } from '../utils'\nimport { ScopedContext } from '../scope'\n\nexport class Shader extends Component {\n  static defaults() { return { ...ShaderLib.defaults(), defines: {} } }\n  constructor(ctx, initialState = {}) {\n    assignDefaults(initialState, Shader.defaults())\n    const { defines, shaderName } = initialState\n    const contextCache = {}\n    const shaderCache = {}\n    const shaderLib = new ShaderLib({ ...initialState, defines })\n\n    const injectParentContext = new ScopedContext(ctx, {\n      defines({defines: contextDefines}) {\n        return { ...contextDefines, ...defines }\n      }\n    })\n\n    let injectContext = null\n\n    let fragmentShaderUncompiled = null\n    let vertexShaderUncompiled = null\n    let fragmentShader = null\n    let vertexShader = null\n\n    super(ctx, initialState, update)\n    function update(state, block, previousState) {\n      injectParentContext((reglContext) => {\n        let {forceCompile = false} = state\n        Object.assign(defines, { ...reglContext.defines, ...state.defines })\n\n        if (Object.keys(defines).length) {\n          if (shaderLib.preprocessor.define(defines)) {\n            forceCompile = true\n          }\n        }\n\n        if (forceCompile || shouldCompile(reglContext, state)) {\n          compile(reglContext, state)\n        }\n\n        setInjectContext(reglContext, state)\n\n        if ('function' == typeof injectContext) {\n          injectContext(state, block)\n        } else {\n          block(state)\n        }\n      })\n    }\n\n    function getShaderFromCache(reglContext, currentState, shader) {\n      shader = getViableShader(reglContext, currentState, shader)\n      return shaderCache[shaderLib.hash(shader)]\n    }\n\n    function setInjectContext(reglContext, currentState) {\n      const opts = {\n        context: {\n          fragmentShader: ({fragmentShader: fs}) => fragmentShader || fs,\n          vertexShader: ({vertexShader: vs }) => vertexShader || vs,\n        }\n      }\n\n      const requestedFragmentShader = getShaderFromCache(\n        reglContext,\n        currentState,\n        currentState.fragmentShader)\n\n      const requestedVertexShader = getShaderFromCache(\n        reglContext,\n        currentState,\n        currentState.vertexShader)\n\n      if (requestedFragmentShader && requestedFragmentShader != fragmentShader) {\n        fragmentShader = requestedFragmentShader\n      }\n\n      if (requestedVertexShader && requestedVertexShader != vertexShader) {\n        vertexShader = requestedVertexShader\n      }\n\n      if ('string' == typeof vertexShader) { opts.vert = vertexShader }\n      if ('string' == typeof fragmentShader) { opts.frag = fragmentShader }\n\n      if ('string' == typeof opts.vert || 'string' == typeof opts.frag) {\n        const hash = [shaderLib.hash(opts.vert), shaderLib.hash(opts.frag)]\n        .filter(Boolean).join('')\n        if (null == contextCache[hash]) {\n          injectContext = ctx.regl(opts)\n          contextCache[hash] = injectContext\n          injectContext.opts = opts\n        } else {\n          injectContext = contextCache[hash]\n        }\n      }\n    }\n\n    function compile(reglContext, currentState) {\n      if (!isShaderCached(currentState.vertexShader)) {\n        compileVertexShader()\n      }\n\n      if (!isShaderCached(currentState.fragmentShader)) {\n        compileFragmentShader()\n      }\n\n      function compileShader(type, shader) {\n        let compiled = null\n        let uncompiled = null\n        if (isViableShader(shader)) {\n          uncompiled = getViableShader(reglContext, currentState, shader)\n          compiled = shaderLib.compile(`${shaderName} (${type})`, uncompiled)\n          compiled = shaderLib.preprocess(compiled)\n          return {compiled, uncompiled}\n        }\n        return null\n      }\n\n      function isShaderCached(shader) {\n        return Boolean(getShaderFromCache(reglContext, currentState, shader))\n      }\n\n      function compileVertexShader() {\n        const result = compileShader('vertex', currentState.vertexShader)\n        if (result) {\n          vertexShader = result.compiled\n          vertexShaderUncompiled = result.uncompiled\n          shaderCache[shaderLib.hash(vertexShaderUncompiled)] = vertexShader\n        }\n      }\n\n      function compileFragmentShader() {\n        const result = compileShader('fragment', currentState.fragmentShader)\n        if (result) {\n          fragmentShader = result.compiled\n          fragmentShaderUncompiled = result.uncompiled\n          shaderCache[shaderLib.hash(fragmentShaderUncompiled)] = fragmentShader\n        }\n      }\n    }\n\n    function getViableShader(reglContext, currentState, shader) {\n      if ('string' == typeof shader) { return shader }\n      else if ('function' == typeof shader) {\n        return shader(reglContext, currentState)\n      }\n    }\n\n    function isViableShader(shader) {\n      return ['string', 'function'].indexOf(typeof shader) > -1\n    }\n\n    function shouldCompile(reglContext, currentState) {\n      let needsCompile = false\n      check('function' != typeof injectContext)\n      checkShader(vertexShaderUncompiled, currentState.vertexShader)\n      checkShader(fragmentShaderUncompiled, currentState.fragmentShader)\n      return needsCompile\n\n      function check(cond) {\n        if (cond) { needsCompile = true }\n      }\n\n      function checkShader(current, next) {\n        let cond = false\n        next = getViableShader(reglContext, currentState, next)\n        if (shaderCache[shaderLib.hash(next)]) {\n          return check(true)\n        } else if ('string' != typeof current && next) {\n          return check(true)\n        } else if ('string' == typeof next && current != next) {\n          return check(true)\n        }\n      }\n    }\n  }\n}\n"]}
\No newline at end of file