• Jump To … +
    ./source/asset-management/image-asset.js ./source/asset-management/noise-asset.js ./source/asset-management/raw-asset.js ./source/asset-management/reaction-diffusion-asset.js ./source/asset-management/sprite-asset.js ./source/asset-management/video-asset.js ./source/core/animation-loop.js ./source/core/display-cycle.js ./source/core/document.js ./source/core/events.js ./source/core/init.js ./source/core/library.js ./source/core/snippets.js ./source/core/user-interaction.js ./source/factory/action.js ./source/factory/anchor.js ./source/factory/animation.js ./source/factory/bezier.js ./source/factory/block.js ./source/factory/button.js ./source/factory/canvas.js ./source/factory/cell.js ./source/factory/cog.js ./source/factory/color.js ./source/factory/conic-gradient.js ./source/factory/crescent.js ./source/factory/element.js ./source/factory/emitter.js ./source/factory/enhanced-label.js ./source/factory/filter.js ./source/factory/gradient.js ./source/factory/grid.js ./source/factory/group.js ./source/factory/label.js ./source/factory/line-spiral.js ./source/factory/line.js ./source/factory/loom.js ./source/factory/mesh.js ./source/factory/net.js ./source/factory/oval.js ./source/factory/particle-force.js ./source/factory/particle-spring.js ./source/factory/particle-world.js ./source/factory/particle.js ./source/factory/pattern.js ./source/factory/picture.js ./source/factory/polygon.js ./source/factory/polyline.js ./source/factory/quadratic.js ./source/factory/radial-gradient.js ./source/factory/rectangle.js ./source/factory/render-animation.js ./source/factory/shape.js ./source/factory/spiral.js ./source/factory/stack.js ./source/factory/star.js ./source/factory/tetragon.js ./source/factory/ticker.js ./source/factory/tracer.js ./source/factory/tween.js ./source/factory/unstacked-element.js ./source/factory/wheel.js ./source/helper/array-pool.js ./source/helper/color-engine.js ./source/helper/document-root-elements.js ./source/helper/filter-engine-bluenoise-data.js ./source/helper/filter-engine.js ./source/helper/random-seed.js ./source/helper/shape-path-calculation.js ./source/helper/shared-vars.js ./source/helper/system-flags.js ./source/helper/utilities.js ./source/helper/workstore.js ./source/mixin/anchor.js ./source/mixin/asset-advanced-functionality.js ./source/mixin/asset-consumer.js ./source/mixin/asset.js ./source/mixin/base.js ./source/mixin/button.js ./source/mixin/cascade.js ./source/mixin/cell-key-functions.js ./source/mixin/delta.js ./source/mixin/display-shape.js ./source/mixin/dom.js ./source/mixin/entity.js ./source/mixin/filter.js ./source/mixin/hidden-dom-elements.js ./source/mixin/mimic.js ./source/mixin/path.js ./source/mixin/pattern.js ./source/mixin/pivot.js ./source/mixin/position.js ./source/mixin/shape-basic.js ./source/mixin/shape-curve.js ./source/mixin/styles.js ./source/mixin/text.js ./source/mixin/tween.js ./source/scrawl.js ./source/untracked-factory/cell-fragment.js ./source/untracked-factory/coordinate.js ./source/untracked-factory/drag-zone.js ./source/untracked-factory/keyboard-zone.js ./source/untracked-factory/observe-update.js ./source/untracked-factory/palette.js ./source/untracked-factory/particle-history.js ./source/untracked-factory/quaternion.js ./source/untracked-factory/state.js ./source/untracked-factory/text-style.js ./source/untracked-factory/vector.js
  • §

    Filter mixin

    The filter mixin adds functionality to Cell, Group and all entity factories which allows those objects to use Scrawl-canvas Filter objects in their output.

  • §

    Imports

    import { asset, filter, styles } from '../core/library.js';
    
    import { generateUuid, mergeOver, removeItem, Ωempty } from '../helper/utilities.js';
    
    import { checkForWorkstoreItem, setWorkstoreItem } from '../helper/workstore.js';
    
    import { releaseCell, requestCell } from '../untracked-factory/cell-fragment.js';
  • §

    Shared constants

    import { _abs, _floor, _isArray, PROCESS_IMAGE, SOURCE_OVER, T_FILTER, T_IMAGE, ZERO_STR } from '../helper/shared-vars.js';
  • §

    Local constants (none defined)

  • §

    Export function

    export default function (P = Ωempty) {
  • §

    Filter-related attributes

    All factories using the filter mixin will add these attributes to their objects

        const defaultAttributes = {
  • §

    filters - An array of filter object String names. If only one filter is to be applied, then it is enough to use the String name of that filter object - Scrawl-canvas will make sure it gets added to the Array.

    • To add/remove new filters to the filters array, use the addFilters and removeFilters functions. Note that the set function will replace all the existing filters in the array with the new filters. To remove all existing filters from the array, use the clearFilters function
    • Multiple filters will be batch-applied to an entity, group of entitys, or an entire cell in one operation. Filters are applied in the order that they appear in in the filters array.
    • Be aware that the “filters” (plural) attribute is different to the CSS/SVG “filter” (singular) attribute - details about how Scrawl-canvas uses CSS/SVG filter Strings to produce filtering effects (at the entity and Cell levels only) are investigated in the Filter Demos 501 to 505. CSS/SVG filter Strings can be applied in addition to Scrawl-canvas filters Array objects, and will be applied after them.
            filters: null,
  • §

    isStencil - Use the entity as a stencil. When this flag is set filter effects will be applied to the background imagery covered by the entity (or Group of entitys, or Cell), the results of which will replace the entity/Group/Cell in the final display.

            isStencil: false,
  • §

    memoizeFilterOutput - SC uses memoization as a means to enhance the speed of filter application. When an entity has filters sety on it, and the memoizeFilterOutput flag is set to true, the filter engine will cache the generated output after its first run and, for subsequent Display cycles, serve up the cached result rather than perform the filter calculations again. Things to note:

    • Entitys will automatically request their filters to recalculate and re-memoize after any start, handle, offset, scale, rotation or flip change. They also request re-memoization when other attributes change, for instance: dimensions, fill or stroke styles, line parameters, font or text updates, etc.
    • Re-memoization is also triggered by any changes to the entity’s filters array, or when the attributes of a filter in the array update.
    • Memoization is limited to entitys (not Groups or Cells). If the isStencil flag is set to true the memoizeFilterOutput flag is ignored.
    • Memoization is also ignored for Picture entitys using a spritesheet or video asset for their source.
            memoizeFilterOutput: false,
        };
        P.defs = mergeOver(P.defs, defaultAttributes);
  • §

    Get, Set, deltaSet

        const S = P.setters;
  • §

    filters - _Dangerous action! - replaces the existing filters Array with a new filters Array. If a string name is supplied, will add that name to the existing filters array

        S.filters = function (item) {
    
            if (item) {
    
                this.filters.length = 0;
    
                if (!_isArray(item)) item = [item];
    
                item.forEach(f => {
    
                    if (f.substring) this.filters.push(f);
                    else if (f.type === T_FILTER) this.filters.push(f.name);
                }, this);
    
                this.dirtyFilters = true;
                this.dirtyImageSubscribers = true;
                this.dirtyFilterIdentifier = true;
            }
        };
  • §

    memoizeFilterOutput

        S.memoizeFilterOutput = function (item) {
    
            this.memoizeFilterOutput = item;
            this.updateFilterIdentifier(!!item);
        };
  • §

    updateFilterIdentifier - manually trigger re-memoization. The function’s (optional) argument is a boolean.

        P.updateFilterIdentifier = function (item) {
    
            this.dirtyFilterIdentifier = false;
            if (this.state) this.state.dirtyFilterIdentifier = false;
    
            if (this.memoizeFilterOutput && item) this.filterIdentifier = generateUuid();
            else this.filterIdentifier = ZERO_STR;
    
        };
  • §

    Packet management

    No additional packet functionality defined here

  • §

    Clone management

    No additional clone functionality defined here

  • §

    Kill management

    No additional kill functionality defined here

  • §

    Get, Set, deltaSet

    No additional functionality defined here

  • §

    Prototype functions

    cleanFilters - Internal housekeeping

    • Filter objects do not have an order attribute. They will be processed in the order in which they appear in the filters Array.
        P.cleanFilters = function () {
    
            this.dirtyFilters = false;
            this.dirtyFiltersCache = true;
    
            const {filters, currentFilters} = this;
    
            currentFilters.length = 0;
    
            filters.forEach(f => {
    
                if (f.substring) {
    
                    const obj = filter[f];
    
                    if (obj) currentFilters.push(obj);
                }
                else if (f && f.type && f.type === T_FILTER) currentFilters.push(f);
            });
        };
  • §

    addFilters, removeFilters - Add or remove one or more filter name strings to/from the filters array. Filter name strings, or the filter objects themselves, can be supplied as comma-separated arguments to the function.

    • Filters are added to the end of the filters array. If the filters need to be reordered, use the set functionality instead to replace the array with an array containing the desired filter order
        P.addFilters = function (...args) {
    
            args.forEach(f => {
    
                if (f && f.type === T_FILTER) f = f.name;
                this.filters.push(f);
    
            }, this);
    
            this.dirtyFilters = true;
            this.dirtyImageSubscribers = true;
            this.dirtyFilterIdentifier = true;
    
            return this;
        };
    
        P.removeFilters = function (...args) {
    
            args.forEach(f => {
    
                if (f && f.type === T_FILTER) f = f.name;
                removeItem(this.filters, f);
    
            }, this);
    
            this.dirtyFilters = true;
            this.dirtyImageSubscribers = true;
            this.dirtyFilterIdentifier = true;
    
            return this;
        };
  • §

    clearFilters - Clears the filters array

        P.clearFilters = function () {
    
            this.filters.length = 0;
    
            this.dirtyFilters = true;
            this.dirtyImageSubscribers = true;
            this.dirtyFilterIdentifier = true;
    
            return this;
        };
  • §

    hasFilters - A quick check to see if any filters have been assigned to this artefact

        P.hasFilters = function () {
    
            return !!this.filters.length;
        };
  • §

    preprocessFilters - internal function called as part of the Display cycle. The process-image filter action loads a Scrawl-canvas asset into the SC Workstore, where it can be used as a lineIn or lineMix argument for other filter actions.

        P.preprocessFilters = function (filters, hostWidth, hostHeight) {
    
            for (let i = 0, iz = filters.length, filter; i < iz; i++) {
    
                filter = filters[i];
    
                for (let j = 0, jz = filter.actions.length, obj; j < jz; j++) {
    
                    obj = filter.actions[j];
    
                    if (obj.action === PROCESS_IMAGE) {
    
                        const img = asset[obj.asset];
    
                        if (img) {
    
                            let width, height, snd, cnd,
                                copyX, copyY, copyWidth, copyHeight;
    
                            if (T_IMAGE !== img.type) {
    
                                img.checkSource();
                                obj.identifier = `user-image-${obj.asset}-${generateUuid()}`;
                                this.dirtyFilterIdentifier = true;
                            }
    
                            if (!checkForWorkstoreItem(obj.identifier)) {
    
                                width = img.sourceNaturalWidth;
                                height = img.sourceNaturalHeight;
                                snd = img.sourceNaturalDimensions;
                                cnd = img.currentDimensions;
    
                                if (!width || !height) {
    
                                    if (snd && snd[0] && snd[1]) {
    
                                        width = width || snd[0];
                                        height = height || snd[1];
                                    }
    
                                    else if (cnd && cnd[0] && cnd[1]) {
    
                                        width = width || cnd[0];
                                        height = height || cnd[1];
                                    }
                                }
    
                                if (width && height) {
    
                                    copyX = obj.copyX || 0;
                                    copyY = obj.copyY || 0;
                                    copyWidth = obj.copyWidth || 1;
                                    copyHeight = obj.copyHeight || 1;
    
                                    if (copyX.substring) copyX = (parseFloat(copyX) / 100) * width;
                                    if (copyY.substring) copyY = (parseFloat(copyY) / 100) * height;
                                    if (copyWidth.substring) copyWidth = (parseFloat(copyWidth) / 100) * width;
                                    if (copyHeight.substring) copyHeight = (parseFloat(copyHeight) / 100) * height;
    
    
                                    copyX = _floor(_abs(copyX));
                                    copyY = _floor(_abs(copyY));
                                    copyWidth = _floor(_abs(copyWidth));
                                    copyHeight = _floor(_abs(copyHeight));
    
                                    if (copyX >= width) {
    
                                        copyX = width - 1;
                                        copyWidth = 1;
                                    }
    
                                    if (copyWidth > width) {
    
                                        copyWidth = width;
                                        copyX = 0;
                                    }
    
                                    if (copyX + copyWidth > width) copyX = width - copyWidth;
    
                                    if (copyY >= height) {
    
                                        copyY = height - 1;
                                        copyHeight = 1;
                                    }
    
                                    if (copyHeight > height) {
    
                                        copyHeight = height;
                                        copyY = 0;
                                    }
    
                                    if (copyY + copyHeight > height) copyY = height - copyHeight;
    
                                    const mycell = requestCell(),
                                        engine = mycell.engine,
                                        canvas = mycell.element;
  • §

    Always render into a host-sized buffer; let canvas clip if the copy rect is larger

                                    canvas.width  = hostWidth;
                                    canvas.height = hostHeight;
  • §

    Placement: center if smaller; pin to top/left if larger

                                    let dx = 0, dy = 0;
    
                                    if (copyWidth < hostWidth)  dx = ((hostWidth  - copyWidth)  / 2) | 0;
    
                                    if (copyHeight < hostHeight) dy = ((hostHeight - copyHeight) / 2) | 0;
    
                                    engine.resetTransform();
                                    engine.globalCompositeOperation = SOURCE_OVER;
                                    engine.globalAlpha = 1;
                                    engine.imageSmoothingEnabled = false;
    
                                    const src = img.source || img.element;
  • §

    No scaling: dest size == copy rect size; canvas clips when larger than host

                                    engine.clearRect(0, 0, hostWidth, hostHeight);
    
                                    engine.drawImage(
                                      src,
                                      copyX, copyY, copyWidth, copyHeight,
                                      dx, dy, copyWidth, copyHeight,
                                    );
  • §

    Store a host-sized ImageData for PROCESS_IMAGE to consume as-is

                                    setWorkstoreItem(obj.identifier, engine.getImageData(0, 0, hostWidth, hostHeight));
    
                                    releaseCell(mycell);
                                }
                            }
                        }
                    }
                }
                if (filter.dirtyFilterIdentifier) this.dirtyFilterIdentifier = true;
            }
    
            const state = this.state;
    
            if (state) {
    
                if (state.dirtyFilterIdentifier) this.dirtyFilterIdentifier = true;
                else {
    
                    const {fillStyle, strokeStyle} = state;
    
                    if (styles[fillStyle] && styles[fillStyle].dirtyFilterIdentifier) this.dirtyFilterIdentifier = true;
    
                    else if (styles[strokeStyle] && styles[strokeStyle].dirtyFilterIdentifier) this.dirtyFilterIdentifier = true;
                }
            }
    
            if (this.dirtyFilterIdentifier || (this.state && this.state.dirtyFilterIdentifier)) this.updateFilterIdentifier(true);
        };
    }