Source: /srcshader-configurator.js

/**
 * Script for shader configuration
 * - requires webgl module, ui_components.js, loader.js
 *
 * - build* methods render static pages
 * - run* methods accept callback and run an interactive selection
 * @namespace ShaderConfigurator
 */
var ShaderConfigurator = {

    /**
     * Prints info about shaders and controls available
     * @param nodeId DOM ID or node to render the content into
     */
    buildShadersAndControlsDocs: function(nodeId) {
        let node = typeof nodeId === "string" ? document.getElementById(nodeId) : nodeId;
        node.innerHTML = this.staticShadersDocs() + this.staticControlsDocs();
    },

    /**
     * Run shader sĨelector
     * @param nodeId DOM ID or node to render the content into
     * @param onFinish, callback with shader ID as argument
     */
    runShaderSelector: function(nodeId, onFinish) {
        this.picker.init(this, nodeId, {
            onFinish: onFinish
        });
    },

    /**
     * Run shader and controls selector
     * @param nodeId DOM ID or node to render the content into
     * @param onFinish callback, argument is the visualization config with given shader and controls
     */
    runShaderAndControlSelector: function(nodeId, onFinish) {
        const _this = this;
        this.runShaderSelector(nodeId, (shaderId) => {
            const src = _this.picker.granularity("image")
                || _this.picker.selectionRules.granularity._config.image.granular;
            if (src) {
                const image = document.createElement('img');
                image.onload = () => {
                    ShaderConfigurator.setData(image);
                    _this.runControlSelector(nodeId, shaderId, onFinish);
                }
                image.src = src;
            } else {
                _this.runControlSelector(nodeId, shaderId, onFinish);
            }
        });
    },

    /**
     * Run controls selector for given shader
     * @param nodeId DOM ID or node to render the content into
     * @param shaderId shader ID to configure controls for
     * @param onFinish callback, argument is the visualization config with given shader and controls
     */
    runControlSelector: function(nodeId, shaderId, onFinish=undefined) {
        let node = typeof nodeId === "string" ? document.getElementById(nodeId) : nodeId;
        this._onControlSelectFinish = onFinish;
        node.innerHTML = this.getInteractiveControlsHtmlFor(shaderId);
        this.setModuleActiveRender();
    },


    /**
     * Set data for realtime data postprocessing - interactive selector can render 'how it looks'
     * @param {(Image|Canvas)} data to process
     */
    setData: function(data) {
        const module = this._getModule(this._uniqueId + 'live-setup-interactive-controls');
        if (module) {
            module.setDimensions(data.width, data.height);
        }
        this._renderData = data;
    },

    setUniqueId: function(id) {
        this._uniqueId = id;
    },

    /**********************/
    /*** STATIC RENDER ****/
    /**********************/
    _uniqueId: "live-setup-",

    staticShadersDocs: function() {
        let html = ["<div><h3>Available shaders and their parameters</h3><br>"];
        const uiControls = this._buildControls();
        for (let shader of WebGLModule.ShaderMediator.availableShaders()) {
            let id = shader.type();

            html.push( "<div class='d-flex'><div style='min-width: 150px'><p class='f3-light mb-0'>",
                shader.name(), "</p><p style='max-width: 150px;'>", shader.description(),
                "</p></div><div class='d-inline-block mx-1 px-1 py-1 pointer v-align-top rounded-2' style='border: 3px solid transparent'>",
                "<img alt='' style='max-width: 150px; max-height: 150px;' class='rounded-2' src='modules/webgl/shaders/",
                shader.type(),".png'></div><div><code class='f4'>", id, "</code>");

            const supports = this.getAvailableControlsForShader(shader);
            for (let control in supports) {
                let supported = supports[control];
                html.push("<div><span style='width: 20%;direction:rtl;transform: translate(0px, -4px);'",
                    "class='position-relative'><span class='flex-1'>Control <code>",
                    control, "</code> | Supports: ", supported.join(", ") ,"</span></span></div>");
            }

            let didParams = false;
            for (let param in shader.customParams) {
                if (!didParams) {
                    didParams = true;
                    html.push("<hr>");
                }
                html.push("<div><span style='width: 20%;direction:rtl;transform: translate(0px, -4px);'",
                    "class='position-relative'><span class='flex-1'>Parameter <code>",
                    param, "</code> <br><span class='text-small'>", shader.customParams[param].usage ,"</span></span></span></div>");
            }
            html.push("</div></div><br>");
        }
        html.push("</div><br>");
        return html.join("");
    },

    staticControlsDocs: function() {
        let html = ["<div><h3>Available controls and their parameters</h3><br>"];
        const uiControls = this._buildControls();

        for (let type in uiControls) {
            html.push("<div><h4>Type <code>", type, "</code></h4>");
            for (let ctrl of uiControls[type]) {
                html.push( "<div class='d-flex'><div style='min-width: 150px'><p class='f3-light mb-0'>",
                    ctrl.name,
                    "</p></div><div class='d-inline-block mx-1 px-1 py-1 pointer v-align-top rounded-2' style='border: 3px solid transparent'>",
                    "</div><div>");

                html.push("<div><pre>", JSON.stringify({type: ctrl.name, ...ctrl.supports}, null, 4) ,"</pre></div>");
                html.push("</div></div><br>");
            }
            html.push("</div>");
        }
        html.push("</div><br>");
        return html.join("");
    },

    /**********************/
    /*** DYNAMIC RENDER ***/
    /**********************/

    REF: 'ShaderConfigurator',

    //todo why this.REF not working
    __chngtml: (paramName, key, valueGetter) =>
        `ShaderConfigurator.refreshUserUpdated(this, '${paramName}', '${key}', ${valueGetter})`,

    _onControlSelectFinish: undefined,

    /**
     * Render number by its value, a map of [uiType] => function values
     * @param name name of the control as defined by its shader
     * @param params for supported parameters see the control 'c.supports' value
     */
    uiRenderers: {},

    refreshUserSwitched(controlId) {
        if (this.renderStyle.advanced(controlId)) {
            this.renderStyle.setUi(controlId);
        } else {
            this.renderStyle.setAdvanced(controlId);
        }
        this.refresh();
    },

    refreshUserSelected(controlId, type) {
        if (!this.setup.params[controlId]) {
            this.setup.params[controlId] = {};
        }
        this.setup.params[controlId].type = type;
        this.refresh();
    },

    refreshUserScripted(node, controlId) {
        try {
            this.parseJSONConfig($(node).val(), controlId);
            this.refresh();
        } catch (e) {
            node.style.background = 'var(--color-bg-danger-inverse)';
        }
    },

    refreshUserUpdated(node, controlId, keyChain, value) {
        try {
            const ensure = (o, key) => {
                if (!o[key]) o[key] = {};
                return o[key];
            }

            let ref = ensure(this.setup.params, controlId);
            const keys = keyChain.split('.');
            const key = keys.pop();
            keys.forEach(x => ref = ensure(ref, x));
            ref[key] = value;
            this.refresh();
        } catch (e) {
            node.style.background = 'var(--color-bg-danger-inverse)';
        }
    },

    parseJSONConfig(value, controlId) {
        const config = JSON.parse(value);
        const control = this.active.layer[controlId];
        const t = WebGLModule.UIControls.IControl.getVarType;

        function extendValuesBy(to, nameMap, suffix="") {
            Object.keys(nameMap).forEach(key => {
                const tVal = to[key],
                    fVal = nameMap[key],
                    tType = t(tVal),
                    fType = t(fVal);

                if (!tVal) return;
                if (fVal && tType === "object" && fType === "object") {
                    extendValuesBy(tVal, fVal, key + ".");
                    return;
                }

                if (tVal == fVal) {
                    //override config with cached values, only if cached did not change
                    to[key] = control.load(tVal, suffix + key);
                }
            });
            return to;
        }
        extendValuesBy(config, control.supports);

        config.type = this.active.layer[controlId].uiControlType;
        this.setup.params[controlId] = config;
        return config;
    },

    refresh() {
        this.setup.shader.cache = {};
        $("#"+this._uniqueId+"interactive-container").replaceWith(this.getInteractiveControlsHtmlFor(this.setup.shader.type));
        this.setModuleActiveRender();
    },

    setModuleActiveRender() {
        if (this._renderData) {
            const module = this._getModule(this._uniqueId + 'interactive-controls');
            if (module) {
                 //timeout so that DOM gets loaded
                const _this = this;
                setTimeout(()=>{
                    document.getElementById(_this._uniqueId + "realtime-rendering-example").appendChild(module.gl.canvas);
                    document.getElementById(_this._uniqueId + "realtime-rendering-example").appendChild(this._renderData);
                    module.processImage(this._renderData,
                        {width: this._renderData.width, height: this._renderData.height},
                        0,
                        1);
                }, 150);
            }
        }
    },

    _buildControlJSONHtml(controlId) {
        let control = this.active.layer[controlId];
        const params = {...control.params};
        delete params.type;

        return `<div id='${this._uniqueId}interactive-control-${controlId}'>
<textarea rows='5' class='form-control m-2 layer-params' style='resize: vertical; width: 90%;box-sizing: border-box;' 
onchange="${this.REF}.refreshUserScripted(this, '${controlId}');">
${JSON.stringify(params, null, '\t')}
</textarea></div>`;
    },

    setup: {
        _visualization: {
            name: "Shader controls and configuration",
            shaders: {
                "1": {
                    name: undefined,
                    dataReferences: undefined,
                    params: {}
                }
            }
        },

        get vis () { return this._visualization },
        get shader() { return this._visualization.shaders["1"] },
        get params() { return this._visualization.shaders["1"].params },
    },

    active: {
        mod: function(id) {
            let _this = window.ShaderConfigurator,
                module = _this._getModule(_this._uniqueId + id);
            if (module) return module;
            throw "Module not instantiated!";
        },
        get vis () { return this.mod('interactive-controls').visualization(0)},
        get shader() { return this.mod('interactive-controls').visualization(0).shaders["1"] },
        get layer() { return this.mod('interactive-controls').visualization(0).shaders["1"]._renderContext }
    },

    renderStyle: {
        _styles: {},
        advanced: function (key) {
            return this._styles[key] == true;
        },
        setAdvanced: function (key) {
            this._styles[key] = true;
        },
        ui: function (key) {
            return !this.advanced(key)
        },
        setUi: function (key) {
            delete this._styles[key];
        }
    },

    getInteractiveControlsHtmlFor: function(shaderId) {
        let shader;
        for (let s of WebGLModule.ShaderMediator.availableShaders()) {
            if (shaderId === s.type()) {
                shader = s;
                break;
            }
        }
        if (!shader) throw "Invalid shader: " + shaderId + ". Not present.";

        const supports = this.getAvailableControlsForShader(shader);
        const _this = this;

        function onLoaded() {}
        const module = this._buildModule(this._uniqueId + 'interactive-controls', function (title, html, dataId, isVisible, layer, isControllable = true) {
            const renders = [];
            for (let control in layer._renderContext) {
                let supported = supports[control];
                if (!supported) continue; //skip other props, supports keep only control

                //todo onchange

                //render type and renderer switching
                renders.push("<div><div class='rounded-2 m-1 px-2 py-1' style='background: var(--color-bg-tertiary)'><span style='width: 20%;direction:rtl;transform: translate(0px, -4px);'",
                    "class='position-relative'><span class='flex-1'>Control <code>",
                    control, "</code> | One of supported: &nbsp;", `<select class='form-control' 
onchange="${_this.REF}.refreshUserSelected('${control}', this.value);">`);

                const activeControl = layer._renderContext[control],
                    activeType = activeControl.uiControlType;
                for (let supType of supported) {
                    let active = activeType === supType ? "selected" : "";
                    renders.push("<option value='", supType ,"' ", active, ">", supType, "</option>");
                }
                const params = {...activeControl.params};
                delete params.type;
                renders.push("</select></span></span>");

                const uiRenderer = _this.uiRenderers[activeType],
                    willRenderUi = _this.renderStyle.ui(control) && uiRenderer;

                if (uiRenderer) {
                    renders.push(`&emsp;<span class="float-right">Simple configuration &nbsp;<input type='checkbox' class='form-control' 
onchange="${_this.REF}.refreshUserSwitched('${control}')" ${willRenderUi ? "checked " : ""}></span>`);
                }
                renders.push("</div>");

                //render control config
                if (willRenderUi) {
                    let controlObject = _this.active.layer[control];
                    const params = {...controlObject.params};
                    delete params.type;
                    renders.push("<div class='m-2 layer-params'>", uiRenderer(control, controlObject.params), "</div>");
                } else {
                    renders.push(_this._buildControlJSONHtml(control));
                }
                renders.push("</div>");
            }

            return `<div id="${_this._uniqueId}interactive-container">
${renders.join("")}
<style>.configurable-border.shader-input span{display: inline-block; margin: 5px 15px;}</style>
<div class="m-2 p-2 border rounded-2">
    <div id="${_this._uniqueId}interactive-shader-head" style="max-width: 500px; min-width: 400px;" class="d-inline-block configurable-border shader-input">
        <div class="shader-part-name px-2 f3-light">${title}.</div>
        <div class=" px-2 py-1">note: resets on config change, permanent changes perform above</div>
        ${html}
    </div>
<div id="${_this._uniqueId}realtime-rendering-example" class="d-inline-block"></div>
</div>
</div>
</div>`;
        }, onLoaded);
        module.reset();

        const data = shader.sources(); //read static sources declaration
        this.setup.shader.type = shaderId;
        this.setup.shader.dataReferences = data.map((x, i) => i);
        this.setup.shader.name = "Configuration: " + shaderId;

        const finish = this._onControlSelectFinish ?
            `<button class="btn" onclick="${this.REF}._onControlSelectFinish(${this.REF}.getCurrentShaderConfig());">Done</button>` : '';

        module.addVisualization(this.setup.vis);
        module.prepareAndInit(data.map(x => ""), this._renderData?.width, this._renderData?.height);
        return `<div><h3>Available controls and their parameters</h3><br><div id='${this._uniqueId}interactive-controls'></div></div>${finish}<br>`;
    },

    getCurrentShaderConfig() {
        return JSON.parse(JSON.stringify(this.setup.shader, WebGLModule.jsonReplacer))
    },

    /**********************/
    /***** UTILITIES ******/
    /**********************/

    getAvailableControlsForShader: function(shader) {
        const uiControls = this._buildControls();
        let controls = shader.defaultControls;

        //this is done with visualization layer as hard-coded control option, include here as well
        if (controls.opacity === undefined || (typeof controls.opacity === "object" && !controls.opacity.accepts("float"))) {
            controls.opacity = {
                default: {type: "range", default: 1, min: 0, max: 1, step: 0.1, title: "Opacity: "},
                accepts: (type, instance) => type === "float"
            };
        }

        const result = {};
        for (let control in controls) {
            if (control.startsWith("use_")) continue;

            let supported = [];
            if (controls[control] === false) continue;
            if (controls[control].required?.type) {
                supported.push(controls[control].required.type);
            } else {
                for (let glType in uiControls) {
                    for (let existing of uiControls[glType]) {
                        if (controls[control] === false) continue;
                        if (!controls[control].accepts(glType, existing)) continue;
                        supported.push(existing.name);
                    }
                }
            }
            result[control] = supported;
        }
        return result;
    },

    _buildModule: function(id, htmlRenderer, onReady) {
        if (this["__module_"+id]) return this["__module_"+id];
        const _this = this;
        const module = new WebGLModule({
            htmlControlsId: id,
            webGlPreferredVersion: "2.0",
            htmlShaderPartHeader: htmlRenderer,
            ready: onReady,
            resetCallback: () => {
                if (_this._renderData) {
                    module.processImage(_this._renderData,
                        {width: _this._renderData.width, height: _this._renderData.height},
                        0,
                        1);
                }

            }
        });
        this["__module_"+id] = module;
        return module;
    },

    _getModule: function(id) {
        return this["__module_"+id];
    },

    _buildControls: function () {
        if (this.__uicontrols) return this.__uicontrols;
        this.__uicontrols = {};
        let types = WebGLModule.UIControls.types();
        let fallbackLayer = new WebGLModule.IdentityLayer("id", {layer: {}});
        fallbackLayer.construct({}, [0]);
        for (let type of types) {
            let ctrl = WebGLModule.UIControls.build(fallbackLayer, type, {type: type});
            let glType = ctrl.type;
            ctrl.name = type;
            if (!this.__uicontrols.hasOwnProperty(glType)) this.__uicontrols[glType] = [];
            this.__uicontrols[glType].push(ctrl);
        }
        return this.__uicontrols;
    },

    /**
     * Shader picking
     */
    picker: {
        selectionRules: {
            granularity: {
                _config: {
                    type: "radio",
                    name: "The data granularity",
                    description: "Data granularity tells us what detail we aim for - the data might be large regions of same values and we are interested in bounaries only, or we might be interested in value difference at close proximity.",
                    info: {
                        plain: "Large regions of same or similar values",
                        other: "Not very granular, but still diverse values",
                        granular: "Dense data with mild or high variety",
                    },
                    image: {
                        plain: "",
                        other: "",
                        granular: "",
                    }
                },
                plain: {
                    "bipolar-heatmap": 0.1,
                    "edge": 0.8,
                    "heatmap": 0.1,
                },
                other: {
                    "bipolar-heatmap": 0.4,
                    "heatmap": 0.4,
                    "colormap": 0.1,
                    "identity": 0.1,
                },
                granular: {
                    "bipolar-heatmap": 0.3,
                    "heatmap": 0.3,
                    "colormap": 0.3,
                    "identity": 0.1,
                }
            },
            ordering: {
                _config: {
                    type: "radio",
                    name: "The data ordering",
                    description: "Describes the data order / direction.",
                    names: {
                        sequential: "Sequential",
                        bidirectional: "Diverging",
                        cyclic: "Cyclic",
                        categorical: "Categorical"
                    },
                    info: {
                        sequential: "Both minimum and maximum in the data is equal to min and max in pixels, or vice versa. The data is linear.",
                        bidirectional: "Your data min or max point is in the middle, and the data ranges in two directions linearly.",
                        cyclic: "Your data have the same value semantics at both high and low pixel values.",
                        categorical: "The data has no ordering and each value represents a discrete category."
                    },
                    image: {
                        sequential: "",
                        bidirectional: "",
                        cyclic: "",
                        categorical: ""
                    }
                },
                sequential: {
                    "edge": 0.1,
                    "heatmap": 0.4,
                    "colormap": 0.4,
                    "identity": 0.1,
                },
                bidirectional: {
                    "bipolar-heatmap": 0.4,
                    "colormap": 0.4,
                    "identity": 0.2,
                },
                cyclic: {
                    "bipolar-heatmap": 0.15,
                    "colormap": 0.8,
                    "identity": 0.05,
                },
                categorical: {
                    "bipolar-heatmap": 0.9,
                    "identity": 0.1,
                },
            },
            significance: {
                _config: {
                    type: "checkbox",
                    name: "Data significance",
                    description: "Where are interesting values?",
                    names: {
                        low: "Significant low values in the data",
                        mid: "Significant middle values in the data",
                        high: "Significant high values in the data",
                        other: "It's complicated..."
                    }
                },
                low: {
                    "bipolar-heatmap": 0.2,
                    "edge": 0.1,
                    "heatmap": 0.3,
                    "colormap": 0.2,
                    "identity": 0.2,
                },
                mid: {
                    "colormap": 0.8,
                    "identity": 0.2,
                },
                high: {
                    "bipolar-heatmap": 0.2,
                    "edge": 0.1,
                    "heatmap": 0.3,
                    "colormap": 0.2,
                    "identity": 0.2,
                },
                other: {
                    "colormap": 1.0,
                }
            }
        },
        selection: {
            shaders: {

            },
            stages: {

            }
        },

        /**
         * Get current granularity data (context -> config KEY)
         * @param {string} context if not set, get selected map {type => true}, else config value of the selected value(s)
         */
        granularity(context="") {
            return this.getSelected("granularity", context);
        },

        ordering(context="") {
            return this.getSelected("granularity", context);
        },

        significance(context="") {
            return this.getSelected("significance", context);
        },

        getSelected(type, context, single=true) {
            if (context === "") return this.selection.stages[type];
            const selected = this.selection.stages[type];

            const result = [];
            for (let selection in selected) {
                if (selected[selection]) {
                    if (single) return this.selectionRules[type]?._config[context][selection];
                    result.push(this.selectionRules[type]?._config[context][selection]);
                }
            }
            return result.length > 0 ? result : undefined;
        },


        /**
         * Init the shader selector routine
         * @param context parent context reference
         * @param nodeId DOM ID or node to render into
         * @param params
         * @param params.idPrefix ID prefix to add to all IDs
         * @param params.onFinish callback with shader ID to call on selector finish
         */
        init(context, nodeId, params) {
            let index = 0;
            const _this = this,
                keys = Object.keys(this.selectionRules),
                REF = context.REF + ".picker",
                allShaderTypeList = WebGLModule.ShaderMediator.availableShaders();

            const idPrefix = params.idPrefix || "shader-picker-";
            this.onFinish = params.onFinish || (() => {})

            function computeSelection() {
                _this.selection.shaders = {};
                for (let stageKey in _this.selection.stages) {
                    const selection = _this.selection.stages[stageKey];
                    for (let selectedKey in selection) {

                        //todo multiple could be chosen - what to do in that case?
                        if (selection[selectedKey]) { //flags t/f

                            const rules = _this.selectionRules[stageKey][selectedKey];
                            for (let shader of allShaderTypeList) {
                                const shaderId = shader.type(),
                                    value = _this.selection.shaders[shaderId],
                                    multiplier = rules[shaderId] || 0;
                                if (value === undefined) _this.selection.shaders[shaderId] = multiplier;
                                else _this.selection.shaders[shaderId] = value * multiplier;
                            }
                        }
                    }
                }
                return _this.selection.shaders;


                // for (let shader in _this.selection._stage) {
                //
                //     //todo cache old, make removable
                //     const value = _this.selection.shaders[shader.type()];
                //     const multiplier = rules[shader.type()] || 0;
                //     if (!value) {
                //         _this.selection.shaders[shader.type()] = multiplier;
                //     } else {
                //         //todo based on type... this is radio
                //         console.log("INCrease", shader.type(), value, multiplier, "TO", value * multiplier);
                //         _this.selection.shaders[shader.type()] = value * multiplier;
                //     }
                // }
                // if (!_this.selection.stages[key]) _this.selection.stages[key] = {};
                // _this.selection.stages[key][type] = true;
            }


            this.renderSelection = () => {
                const key = keys[index];
                if (!key) {
                    computeSelection();
                    let best = -1, selected = "";
                    for (let s in _this.selection.shaders) {
                        if (_this.selection.shaders[s] > best) {
                            best = _this.selection.shaders[s];
                            selected = s;
                        }
                    }
                    console.log("Computed probabilities:", _this.selection.shaders);
                    if (selected) {
                        _this.onFinish(selected);
                    } else {
                        //todo
                        render();
                    }
                    return;
                }
                index++;
                if (key.startsWith("_")) {
                    return _this.renderSelection();
                }

                (typeof nodeId === "string" ? document.getElementById(nodeId) : nodeId).innerHTML = renderPage(key);
            }

            this.recordSelection = (key, selected, add = true) => {
                if (!_this.selection.stages[key]) _this.selection.stages[key] = {};
                _this.selection.stages[key][selected] = true;
            }

            function renderPage(key) {
                const page = _this.selectionRules[key];
                const conf = page._config;

                const output = [];
                output.push('<h3 class="f3-light">', conf.name || key, '</h3>');
                output.push('<p>', conf.description, '</p>');

                const inputType = conf.type;
                output.push(...Object.entries(page).map(([type, value]) => {
                    if (type.startsWith("_")) return "";

                    let description = "";
                    if (conf.names?. [type]) {
                        description = `<h4 class="f5-light" style='max-width: 150px;'>${conf.names[type]}</h4>`;
                    }
                    if (conf.info?.[type]) description = `${description}<p style='max-width: 150px;'>${conf.info[type]}</p>`;
                    if (conf.image?.[type]) {
                        description = `<div style='min-width: 150px'>${description}
<span class='d-inline-block mx-1 px-1 py-1 pointer v-align-top rounded-2' style='border: 3px solid transparent'>
<img alt='' style='width: 150px; height: 150px;' class='rounded-2' src='${conf.image[type]}'></span></div>`;
                    } else {
                        description = `<div style='min-width: 150px'>${description}</div>`;
                    }

                    return `
<div class="d-inline-block"><input type="${inputType}" class="d-block" id="${idPrefix}selector-shader-${key}-${type}" name="shader_selector" onchange="${REF}.recordSelection('${key}', '${type}', this.checked);" value="${type}">      
<label class="d-inline-block" for="${idPrefix}selector-shader-${key}-${type}">${description}</label>  </div>
      `;
                }));
                output.push('<button class="btn" onclick="', REF, '.renderSelection();">Next</button>');
                return output.join("");
            }

            function render() {
                const html = [`<div style='cursor:pointer;' class="border rounded px-2 py-1 d-inline-block" onclick="${REF}.renderSelection();">
      <p class='f2-light mb-0'>
      Help me to select a correct shader.</p>
      <p style='max-width: 150px;'>Run interactive selection that helps you to select the shader.</p></div><br><br><br><p class='f4-light mb-0'>&emsp;OR SELECT ONE BELOW:</p><br><br>`];
                html.push()

                for (let shader of allShaderTypeList) {
                    let id = shader.type();

                    html.push(`<div class="d-flex"><div style="min-width: 150px; cursor:pointer;" onclick="${REF}.onFinish('${id}');"><p class="f3-light mb-0">`,
                        shader.name(), "</p><p style='max-width: 150px;'>", shader.description(),
                        "</p></div><div class='d-inline-block mx-1 px-1 py-1 pointer v-align-top rounded-2' style='border: 3px solid transparent'>",
                        "<img alt='' style='max-width: 150px; max-height: 150px;' class='rounded-2' src='", shader.preview(),
                        "'></div><div><code class='f4'>", id, "</code></div></div><br>");
                }
                (typeof nodeId === "string" ? document.getElementById(nodeId) : nodeId).innerHTML = html.join("");
            }

            render();
        }
    }
};

/**
 * Definition of tailored setters for shader controls
 */
ShaderConfigurator.uiRenderers.number = (name, params, onChange) => `
Title: &emsp; ${UIComponents.Elements.textInput({...params, default: params.title,
        onchange: ShaderConfigurator.__chngtml(name, 'title', 'this.value')})}<br>
Interactive: &emsp; ${UIComponents.Elements.checkBox({...params, default: params.interactive,
        onchange: ShaderConfigurator.__chngtml(name, 'interactive', 'this.checked')})}<br>
Value between min and max<br>
Default value: &emsp; ${UIComponents.Elements.numberInput({...params,
        onchange: ShaderConfigurator.__chngtml(name, 'default', 'Number.parseFloat(this.value)')})}<br>
Minimum value: &emsp; ${UIComponents.Elements.numberInput({...params, min: -1e+5, max: +1e+5, step: 1e-5, default: params.min,
        onchange: ShaderConfigurator.__chngtml(name, 'min', 'Number.parseFloat(this.value)')})}<br>
Maximum value: &emsp; ${UIComponents.Elements.numberInput({...params, min: -1e+5, max: +1e+5, step: 1e-5, default: params.max,
        onchange: ShaderConfigurator.__chngtml(name, 'max', 'Number.parseFloat(this.value)')})}<br>
Step: &emsp; ${UIComponents.Elements.numberInput({...params, min: -1e+5, max: +1e+5, step: 1e-5, default: params.step,
        onchange: ShaderConfigurator.__chngtml(name, 'step', 'Number.parseFloat(this.value)')})}<br>
`;
ShaderConfigurator.uiRenderers.range = ShaderConfigurator.uiRenderers.number;
ShaderConfigurator.uiRenderers.range_input = ShaderConfigurator.uiRenderers.number;
ShaderConfigurator.uiRenderers.color = (name, params, onChange) => `
Title: &emsp; ${UIComponents.Elements.textInput({...params, default: params.title,
        onchange: ShaderConfigurator.__chngtml(name, 'title', 'this.value')})}<br>
Interactive: &emsp; ${UIComponents.Elements.checkBox({...params, default: params.interactive,
        onchange: ShaderConfigurator.__chngtml(name, 'interactive', 'this.checked')})}<br>
Default value: &emsp; ${UIComponents.Elements.colorInput({...params,
        onchange: ShaderConfigurator.__chngtml(name, 'default', 'this.value')})}<br>
`;
ShaderConfigurator.uiRenderers.colormap = (name, params, onChange) => `
Title: &emsp; ${UIComponents.Elements.textInput({...params, default: params.title,
        onchange: ShaderConfigurator.__chngtml(name, 'title', 'this.value')})}<br>
Interactive: &emsp;  ${UIComponents.Elements.checkBox({...params, default: params.interactive,
        onchange: ShaderConfigurator.__chngtml(name, 'interactive', 'this.checked')})}<br>
Default value: &emsp; ${UIComponents.Elements.select({...params, options: ColorMaps.schemeGroups[params.mode],
        onchange: ShaderConfigurator.__chngtml(name, 'default', 'this.value')})}<br>
Continuous: &emsp; ${UIComponents.Elements.checkBox({...params, default: params.continuous,
        onchange: ShaderConfigurator.__chngtml(name, 'continuous', 'this.checked')})}<br>
Mode: &emsp; ${UIComponents.Elements.select({...params, default: params.mode, options: Object.keys(ColorMaps.schemeGroups),
        onchange: ShaderConfigurator.__chngtml(name, 'mode', 'this.value')})}<br>
Steps: &emsp; ${UIComponents.Elements.numberInput({...params, min: 0, max: 8, step: 1, default: params.steps,
        onchange: ShaderConfigurator.__chngtml(name, 'steps', 'Number.parseInt(this.value)')})}<br>
`;
ShaderConfigurator.uiRenderers.advanced_slider = (name, params, onChange) => `
Title: &emsp; ${UIComponents.Elements.textInput({...params, default: params.title,
        onchange: ShaderConfigurator.__chngtml(name, 'title', 'this.value')})}<br>
Interactive: &emsp;  ${UIComponents.Elements.checkBox({...params, default: params.interactive,
        onchange: ShaderConfigurator.__chngtml(name, 'interactive', 'this.checked')})}<br>
Sample Mask or Uniformly (mask=false) <br>
Read Mask: &emsp; ${UIComponents.Elements.checkBox({...params, default: params.maskOnly,
    onchange: ShaderConfigurator.__chngtml(name, 'maskOnly', 'this.checked')})}<br>
Select starting positions (between min and max) <br>
Breaks: &emsp; ${UIComponents.Elements.numberArray({...params, default: params.breaks,
        onchange: ShaderConfigurator.__chngtml(name, 'breaks', 'this.values')})}<br>
Minimum value: &emsp; ${UIComponents.Elements.numberInput({...params, min: -1e+5, max: +1e+5, step: 1e-5, default: params.min,
        onchange: ShaderConfigurator.__chngtml(name, 'min', 'Number.parseFloat(this.value)')})}<br>
Maximum value: &emsp; ${UIComponents.Elements.numberInput({...params, min: -1e+5, max: +1e+5, step: 1e-5, default: params.max,
        onchange: ShaderConfigurator.__chngtml(name, 'max', 'Number.parseFloat(this.value)')})}<br>
Step: &emsp; ${UIComponents.Elements.numberInput({...params, min: -1e+5, max: +1e+5, step: 1e-5, default: params.minGap,
        onchange: ShaderConfigurator.__chngtml(name, 'minGap', 'Number.parseFloat(this.value)')})}<br>
Mask should have #Breaks+1 elements, either 0 or 1 depending on whether given range is enabled or disabled<br>
Mask: &emsp; ${UIComponents.Elements.numberArray({...params, default: params.mask,
        onchange: ShaderConfigurator.__chngtml(name, 'mask', 'this.values')})}<br>
Labels on scale, in %<br>
Pips: &emsp; ${UIComponents.Elements.numberArray({...params, default: params.pips.values,
        onchange: ShaderConfigurator.__chngtml(name, 'pips.values', 'this.values')})}<br>
`;
ShaderConfigurator.uiRenderers.bool = (name, params, onChange) => `
Title: &emsp; ${UIComponents.Elements.textInput({...params, default: params.title,
    onchange: ShaderConfigurator.__chngtml(name, 'title', 'this.value')})}<br>
Interactive: &emsp;  ${UIComponents.Elements.checkBox({...params, default: params.interactive,
    onchange: ShaderConfigurator.__chngtml(name, 'interactive', 'this.checked')})}<br>
Default value: &emsp; ${UIComponents.Elements.checkBox({...params,
    onchange: ShaderConfigurator.__chngtml(name, 'default', 'this.checked')})}<br>
`;