• Jump To … +
    ./demo/canvas-001.js ./demo/canvas-002.js ./demo/canvas-003.js ./demo/canvas-004.js ./demo/canvas-005.js ./demo/canvas-006.js ./demo/canvas-007.js ./demo/canvas-008.js ./demo/canvas-009.js ./demo/canvas-010.js ./demo/canvas-011.js ./demo/canvas-012.js ./demo/canvas-013.js ./demo/canvas-014.js ./demo/canvas-015.js ./demo/canvas-016.js ./demo/canvas-017.js ./demo/canvas-018.js ./demo/canvas-019.js ./demo/canvas-020.js ./demo/canvas-021.js ./demo/canvas-022.js ./demo/canvas-023.js ./demo/canvas-024.js ./demo/canvas-025.js ./demo/canvas-026.js ./demo/canvas-027.js ./demo/canvas-028.js ./demo/canvas-029.js ./demo/canvas-030.js ./demo/canvas-031.js ./demo/canvas-032.js ./demo/canvas-033.js ./demo/canvas-034.js ./demo/canvas-035.js ./demo/canvas-036.js ./demo/canvas-037.js ./demo/canvas-038.js ./demo/canvas-039.js ./demo/canvas-040.js ./demo/canvas-041.js ./demo/canvas-042.js ./demo/canvas-043.js ./demo/canvas-044.js ./demo/canvas-045.js ./demo/canvas-046.js ./demo/canvas-047.js ./demo/canvas-048.js ./demo/canvas-049.js ./demo/canvas-050.js ./demo/canvas-051.js ./demo/canvas-052.js ./demo/canvas-053.js ./demo/canvas-054.js ./demo/canvas-055.js ./demo/canvas-056.js ./demo/canvas-057.js ./demo/canvas-058.js ./demo/canvas-059.js ./demo/canvas-060.js ./demo/canvas-061.js ./demo/canvas-062.js ./demo/canvas-063.js ./demo/canvas-064.js ./demo/canvas-065.js ./demo/canvas-066.js ./demo/canvas-067.js ./demo/canvas-068.js ./demo/canvas-069.js ./demo/canvas-070.js ./demo/canvas-071.js ./demo/canvas-072.js ./demo/canvas-073.js ./demo/canvas-201.js ./demo/canvas-202.js ./demo/canvas-203.js ./demo/canvas-204.js ./demo/canvas-205.js ./demo/canvas-206.js ./demo/canvas-207.js ./demo/canvas-208.js ./demo/canvas-209.js ./demo/canvas-210.js ./demo/canvas-211.js ./demo/canvas-212.js ./demo/delaunator-001.js ./demo/delaunator-002.js ./demo/dom-001.js ./demo/dom-002.js ./demo/dom-003.js ./demo/dom-004.js ./demo/dom-005.js ./demo/dom-006.js ./demo/dom-007.js ./demo/dom-008.js ./demo/dom-009.js ./demo/dom-010.js ./demo/dom-011.js ./demo/dom-012.js ./demo/dom-013.js ./demo/dom-015.js ./demo/dom-016.js ./demo/dom-017.js ./demo/dom-018.js ./demo/dom-019.js ./demo/dom-020.js ./demo/dom-021.js ./demo/filters-001.js ./demo/filters-002.js ./demo/filters-003.js ./demo/filters-004.js ./demo/filters-005.js ./demo/filters-006.js ./demo/filters-007.js ./demo/filters-008.js ./demo/filters-009.js ./demo/filters-010.js ./demo/filters-011.js ./demo/filters-012.js ./demo/filters-013.js ./demo/filters-014.js ./demo/filters-015.js ./demo/filters-016.js ./demo/filters-017.js ./demo/filters-018.js ./demo/filters-019.js ./demo/filters-020.js ./demo/filters-021.js ./demo/filters-022.js ./demo/filters-023.js ./demo/filters-024.js ./demo/filters-025.js ./demo/filters-026.js ./demo/filters-027.js ./demo/filters-028.js ./demo/filters-029.js ./demo/filters-030.js ./demo/filters-031.js ./demo/filters-032.js ./demo/filters-033.js ./demo/filters-034.js ./demo/filters-035.js ./demo/filters-036.js ./demo/filters-037.js ./demo/filters-101.js ./demo/filters-102.js ./demo/filters-103.js ./demo/filters-104.js ./demo/filters-105.js ./demo/filters-501.js ./demo/filters-502.js ./demo/filters-503.js ./demo/filters-504.js ./demo/filters-505.js ./demo/js/mediapipe/tasks-vision/vision-bundle.js ./demo/js/mediapipe/tasks-vision/wasm/vision_wasm_internal.js ./demo/js/mediapipe/tasks-vision/wasm/vision_wasm_nosimd_internal.js ./demo/mediapipe-001.js ./demo/mediapipe-002.js ./demo/mediapipe-003.js ./demo/modules-001.js ./demo/modules-002.js ./demo/modules-003.js ./demo/modules-004.js ./demo/modules-005.js ./demo/modules-006.js ./demo/modules/canvas-minimap.js ./demo/modules/canvas-scene-editor.js ./demo/modules/demo-m006-c1.js ./demo/modules/demo-m006-c2.js ./demo/modules/demo-m006-c3.js ./demo/modules/demo-m006-c4.js ./demo/modules/demo-m006-utils.js ./demo/modules/dom-entity-editor.js ./demo/modules/entity-copy-paste.js ./demo/modules/entity-manipulation-gui.js ./demo/modules/entity-navigation.js ./demo/modules/entity-ring-builder.js ./demo/modules/london-crime-lines.js ./demo/modules/london-crime-stacked-bars.js ./demo/modules/lottie-loader.js ./demo/modules/simple-chart-frame.js ./demo/modules/simple-graph-lines.js ./demo/modules/simple-graph-stacked-bars.js ./demo/modules/wikipedia-views-spiral-chart.js ./demo/packets-001.js ./demo/packets-002.js ./demo/particles-001.js ./demo/particles-002.js ./demo/particles-003.js ./demo/particles-004.js ./demo/particles-005.js ./demo/particles-006.js ./demo/particles-007.js ./demo/particles-008.js ./demo/particles-009.js ./demo/particles-010.js ./demo/particles-011.js ./demo/particles-012.js ./demo/particles-013.js ./demo/particles-014.js ./demo/particles-015.js ./demo/particles-016.js ./demo/particles-017.js ./demo/snippets-001.js ./demo/snippets-002.js ./demo/snippets-003.js ./demo/snippets-004.js ./demo/snippets-005.js ./demo/snippets-006.js ./demo/snippets/animated-highlight-gradient-text-snippet.js ./demo/snippets/animated-hover-gradient-snippet.js ./demo/snippets/animated-word-gradient-snippet.js ./demo/snippets/before-after-slider-infographic.js ./demo/snippets/bubbles-text-snippet.js ./demo/snippets/green-box-snippet.js ./demo/snippets/jazzy-button-snippet.js ./demo/snippets/page-performance-snippet-test.js ./demo/snippets/page-performance-snippet.js ./demo/snippets/pan-image-snippet.js ./demo/snippets/placeholder-effect-snippet.js ./demo/snippets/ripple-effect-snippet.js ./demo/snippets/risograph-text-gradient-snippet.js ./demo/snippets/spotlight-text-snippet-test.js ./demo/snippets/spotlight-text-snippet.js ./demo/snippets/swirling-stripes-text-snippet.js ./demo/snippets/word-highlighter-snippet.js ./demo/snippets/worley-text-gradient-snippet.js ./demo/temp-001.js ./demo/temp-shape-scale-investigation.js ./demo/tensorflow-001.js ./demo/tensorflow-002.js ./demo/utilities.js
  • §

    Demo Snippets 004

    Snippets included in the Scrawl-canvas demo/snippets folder

    Related files:

    • Snippets included in the Scrawl-canvas demo/snippets folder
    • Animated hover gradient snippet
    • Animated word gradient snippet
    • Green box snippet
    • Jazzy button snippet
    • Page performance snippet
    • Pan image snippet
    • Placeholder effect snippet
    • Ripple effect snippet
    • Spotlight text snippet
    • Word highlighter snippet
  • §

    ‘Pan image’ snippet

    Purpose: Displays a pan effect on an image, where users can click and drag the image to explore it

    Function input:

    • the DOM element - a block element - usually a <div> element - containing one or more <img> elements

    Function output: a Javascript object will be returned, containing the following attributes

    {
        element     // the Scrawl-canvas wrapper for the DOM element supplied to the function
        canvas      // the Scrawl-canvas wrapper for the snippet's canvas
        animation   // the Scrawl-canvas animation object
        demolish    // remove the snippet from the Scrawl-canvas library
    }
    
    Usage example:
    import panImage from './relative/or/absolute/path/to/this/file.js';
    
    let myElements = document.querySelectorAll('.some-class');
    
    myElements.forEach(el => panImage(scrawl, el));
    
  • §

    Effects on the element:

    • Sets the element background to transparent
    • Imports any <img> elements it finds in the element, switching their display styling to ‘none’
    export default function (scrawl, el) {
  • §

    Apply the snippet to the DOM element

        const snippet = scrawl.makeSnippet({
            domElement: el,
        });
    
        if (snippet) {
  • §

    Set some convenience variables

            const canvas = snippet.canvas,
                animation = snippet.animation,
                group = canvas.base.name,
                name = snippet.element.name;
  • §

    Set the DOM element’s background color to transparent so we can see the new canvas

            el.style.backgroundColor = 'transparent';
  • §

    Grab all the child <img> elements

            const images = el.querySelectorAll('img');
  • §

    If the coder gave the DOM element a sane id attribute value then all our names will be sensible names. If not, then Scrawl-canvas will have computer-generated an id for the element. These generated names often include characters that do not sit well with the CSS/JS querySelector string requirements. So we have to make the name safe to be used as a search string

            let imgName = `${name}-image`;
            imgName = imgName.replace('.', '');
  • §

    Update all the <img> elements we found. In particular:

    • add the name to the element’s class
    • set the indexed name as the element’s id value
    • set the element’s display styling attrinute to ‘none’ - we don’t weant it displaying in the DOM, we need it to display in the new canvas
            images.forEach((img, index) => {
    
                let classes = img.getAttribute('class')
    
                if (!classes) classes = '';
                classes += ` ${imgName}`;
    
                img.setAttribute('class', classes);
                img.id = `${imgName}-${index}`;
    
                img.style.display = 'none';
            });
  • §

    Import all the images we found into Scrawl-canvas

            scrawl.importDomImage(`.${imgName}`);
  • §

    Configuration - coders can set ‘data-x’ and ‘data-y’ attributes on the DOM element to indicate which area of the image should be displayed in the new canvas when the page first loads

            const copyStartX = el.dataset.x || 0,
                copyStartY = el.dataset.y || 0;
  • §

    We will display the first image we found in a Picture entity

            const [w, h] = canvas.get('dimensions');
    
            const asset = scrawl.library.asset[`${imgName}-0`];
    
            const panImage = scrawl.makePicture({
    
                name: `${imgName}-panimage`,
                group,
                asset,
                width: w,
                height: h,
    
                copyWidth: w,
                copyHeight: h,
                copyStartX,
                copyStartY,
            });
    
            let cursor = 'grab',
                lastX = 0,
                lastY = 0,
                aspectWidth = 1,
                aspectHeight = 1;
  • §

    The pan effect, where the user can click and drag the image within its container to explore large images in small areas. To achieve the effect we split the functionality across three separate event listeners, which capture the three key actions associated with the effect: starting a drag; continuing a drag; and ending a drag.

            scrawl.addListener('down', () => {
    
                const here = canvas.here;
    
                if (here.active && 'grab' === cursor) {
    
                    cursor = 'grabbing';
                    el.style.cursor = cursor;
    
                    const {x, y} = here;
    
                    lastX = x;
                    lastY = y;
                }
            }, el);
    
            scrawl.addListener('move', () => {
    
                if ('grabbing' === cursor) {
    
                    const {x, y} = canvas.here;
    
                    const dx = lastX - x,
                        dy = lastY - y;
    
                    panImage.setDelta({
                        copyStartX: dx * aspectWidth,
                        copyStartY: dy * aspectHeight,
                    });
    
                    lastX = x;
                    lastY = y;
                 }
            }, el);
    
            scrawl.addListener(['up', 'leave'], () => {
    
                if ('grabbing' === cursor) {
    
                    cursor = 'grab';
                    el.style.cursor = cursor;
    
                    const {x, y} = canvas.here;
    
                    const dx = lastX - x,
                        dy = lastY - y;
    
                    panImage.setDelta({
                        copyStartX: dx * aspectWidth,
                        copyStartY: dy * aspectHeight,
                    });
    
                    lastX = 0;
                    lastY = 0;
                }
            }, el);
  • §

    We’re checking dimensions here partly because images load asynchronously and during the early part of that process their <img> elements will supply incorrect information to the script which is trying to set up the canvas element which will (eventually) host their image data. But we also need to check because the script has no way of knowing what sort of image it is dealing with - for instance an <img> element with a “srcset” attribute will load new images if the user decides to expand their browser’s window - which would have an impact on the assumptions we make in our scaling and dragging calculations.

            const checkDimensions = function () {
    
                const [canvasWidth, canvasHeight] = canvas.get('dimensions');
                const [imageWidth, imageHeight] = panImage.get('dimensions');
  • §

    check to see if there’s been any resize browser resize or CSS layout change that affects our DOM element

                if (canvasWidth !== imageWidth || canvasHeight !== imageHeight) {
    
                    const [sourceWidth, sourceHeight] = panImage.get('sourceDimensions');
  • §

    We want the image we paint in our canvas to keep its asset’s original aspect ratio

                    aspectWidth = canvasWidth / sourceWidth;
                    aspectHeight = canvasHeight / sourceHeight;
  • §

    We want the zoom level to be x3 (along the width) with respect to the DOM element’s dimensions

                    const setToHalfWidth = 50 / (aspectWidth * 100);
    
                    const ratioWidth = aspectWidth * setToHalfWidth * 100,
                        ratioHeight = aspectHeight * setToHalfWidth * 100;
    
                    panImage.set({
                        width: canvasWidth,
                        height: canvasHeight,
                        copyWidth: ratioWidth + '%',
                        copyHeight: ratioHeight + '%',
                    });
                }
            };
  • §

    Invoke our ‘checkDimensions’ function at the start of every Display cycle

            animation.set({
                commence: checkDimensions,
            });
        }
  • §

    Return the snippet, so coders can access the snippet’s parts - in case they need to tweak the output to meet the web page’s specific requirements

        return snippet;
    }