
<html>
<body>
<br/>
<br/>
<br/>
<center>
<canvas id="myCanvas" width="1180" height="400" style="border:2px solid #000000;">
</canvas>
<img src="axis.png">
</center>

This prototype was tested in the Chrome web browser on Microsoft Windows 10.
The left mouse button should allow you to raise, move, and resize windows.
Holding down Ctrl and then clicking with the left mouse button should trigger
teleportation of a window to the foreground layer (shown with the orange border),
or dismissing a window to the background.
When the foreground layer contains one or more windows,
holding down Ctrl and then holding down Alt should cause the foreground layer to become transparent,
allowing you to interact with windows behind the foreground layer.

<p>
Pressing and releasing the Alt key by itself, without Ctrl,
can cause keyboard focus to be transferred off the webpage,
preventing the prototype from working properly.
If this happens,
to return keyboard focus to the prototype, press and release the mouse button on the webpage,
and avoid holding down Alt without first holding down Ctrl.

<script src="geom2D.js"></script>
<script>

// BEGIN random number generator
// This random number generator allows us to set a seed and always get the same sequence of random numbers.
// From
// https://stackoverflow.com/questions/521295/seeding-the-random-number-generator-in-javascript
//

function cyrb128(str) {
    let h1 = 1779033703, h2 = 3144134277,
        h3 = 1013904242, h4 = 2773480762;
    for (let i = 0, k; i < str.length; i++) {
        k = str.charCodeAt(i);
        h1 = h2 ^ Math.imul(h1 ^ k, 597399067);
        h2 = h3 ^ Math.imul(h2 ^ k, 2869860233);
        h3 = h4 ^ Math.imul(h3 ^ k, 951274213);
        h4 = h1 ^ Math.imul(h4 ^ k, 2716044179);
    }
    h1 = Math.imul(h3 ^ (h1 >>> 18), 597399067);
    h2 = Math.imul(h4 ^ (h2 >>> 22), 2869860233);
    h3 = Math.imul(h1 ^ (h3 >>> 17), 951274213);
    h4 = Math.imul(h2 ^ (h4 >>> 19), 2716044179);
    return [(h1^h2^h3^h4)>>>0, (h2^h1)>>>0, (h3^h1)>>>0, (h4^h1)>>>0];
}

function mulberry32(a) {
    return function() {
      var t = a += 0x6D2B79F5;
      t = Math.imul(t ^ t >>> 15, t | 1);
      t ^= t + Math.imul(t ^ t >>> 7, t | 61);
      return ((t ^ t >>> 14) >>> 0) / 4294967296;
    }
}

let myRandSeed = cyrb128("apples");
let myRand = mulberry32(myRandSeed[0]);



//
// END random number generator

let NUM_WINDOWS = 28;

let frontLayerBorderThickness = 5;

let frontLayerBorderColor = "rgba(255, 128, 0, 0.9)"; // "#808080";
let frontLayerBackground = "rgba(255, 255, 255, 0.75)"; // "rgba(255, 255, 255, 0.75)";
let frontLayerBorderColorVeryTransparent = "rgba(255, 128, 0, 0.5)";
let connectionColor = "rgba(255, 128, 0, 0.5)";

let defaultBorderColor = '#000000';
let defaultBackgroundColor = '#ffffff';
let defaultBackgroundColorWithAlpha = "rgba(255, 255, 255, 0.75)";




const KEYCODE_SHIFT = 16;
const KEYCODE_CTRL = 17;
const KEYCODE_ALT = 18;
let isShiftDown = false;
let isCtrlDown = false;
let isAltDown = false;

// If the user hits Ctrl, we enter a modified state, and the front layer becomes almost invisible.
// If the user hits Ctrl+Alt, we are in a modified state *and* the front layer remains visible.
function isModifiedStateActive() {
    return isCtrlDown;
}
function isFrontLayerVeryTransparent() {
    return isCtrlDown && isAltDown;
}

function keyDownHandler(e) {
    //console.log("DOWN keyCode: " + e.keyCode );
    // e = e || window.event;

    if ( e.keyCode === KEYCODE_SHIFT ) {
        isShiftDown = true;
    }
    else if ( e.keyCode === KEYCODE_CTRL ) {
        isCtrlDown = true;
    }
    else if ( e.keyCode === KEYCODE_ALT ) {
        isAltDown = true;
    }
    //console.log(""+(isCtrlDown?"Ctrl ":"")+(isAltDown?"Alt ":"")+(isShiftDown?"Shift ":""));
    updateHighlight();
    draw();
}
function keyUpHandler(e) {
    //console.log("UP keyCode: " + e.keyCode );
    // e = e || window.event;
    if ( e.keyCode === KEYCODE_SHIFT ) {
        isShiftDown = false;
    }
    else if ( e.keyCode === KEYCODE_CTRL ) {
        isCtrlDown = false;
    }
    else if ( e.keyCode === KEYCODE_ALT ) {
        isAltDown = false;
    }
    //console.log(""+(isCtrlDown?"Ctrl ":"")+(isAltDown?"Alt ":"")+(isShiftDown?"Shift ":""));
    updateHighlight();
    draw();
}

document.onkeydown = keyDownHandler;
document.onkeyup = keyUpHandler;

let canvas = document.getElementById("myCanvas");
let canvas_context = canvas.getContext("2d");
let fontHeight = 20;
let fontName1 = '20pt Calibri';
let fontHeight2 = 10;
let fontName2 = '10pt Calibri';

// state of the mouse
let mouse_x = 0, mouse_y = 0;
let isMouseButtonDown = false;


// layers
let L_BACK = 0; // back layer
let L_FRONT = 1; // front layer

let MWIN_MOVE_HANDLE_SIZE = 15; // in pixels
let MWIN_RESIZE_HANDLE_SIZE = 15; // in pixels

// used to identify parts of the window
let MWIN_NONE = -1;
let MWIN_BODY = 0;
let MWIN_MOVE_HANDLE = 1;
let MWIN_RESIZE_HANDLE = 2;

class MWin { // a mini window
    constructor(X0,Y0,W,H,L) {
        this.label = L;

        this.currentLayer = L_BACK;

        this.isAnimating = false;
        this.animation_u = 0; // between 0.0 and 1.0
        this.animation_x = 0; // used only during animation
        this.animation_y = 0; // used only during animation

        // If the window is animating from, e.g., back to front layer, then currentLayer will be front.
        // In other words, the currentLayer is always the target of an animated transition.

        this.x = [];
        this.y = [];
        // The indices of 0 and 1 refer to back and front layer, respectively
        this.x[0] = X0;
        this.y[0] = Y0;
        this.hasPositionInFrontLayer = false;
        this.x[1] = 0;
        this.y[1] = 0;
        this.w = W; // width, in pixels
        this.h = H; // height, in pixels
        this.boundingRect = [];
        this.boundingRect[0] = new Box2();
        this.boundingRect[1] = new Box2();
        this.isBoundingRectDirty = [];
        this.isBoundingRectDirty[0] = true;
        this.isBoundingRectDirty[1] = true;
    }
    getBoundingRect( index ) {
        if ( this.isBoundingRectDirty[ index ] ) {
            this.boundingRect[ index ].clear();
            this.boundingRect[ index ].boundPoint( new Vec2( this.x[ index ], this.y[ index ] ) );
            this.boundingRect[ index ].boundPoint( new Vec2( this.x[ index ]+this.w, this.y[ index ]+this.h ) );
            this.isBoundingRectDirty[ index ] = false;
        }
        return this.boundingRect[ index ];
    }
    getPartContainingPoint( i /*layer index*/, p /*point*/ ) {
        if (
            this.x[i] <= p.x && p.x <= this.x[i]+this.w
            &&
            this.y[i] <= p.y && p.y <= this.y[i]+this.h
        ) {
            if ( p.y <= this.y[i]+MWIN_MOVE_HANDLE_SIZE ) {
                return MWIN_MOVE_HANDLE;
            }
            else if ( this.x[i]+this.w-MWIN_RESIZE_HANDLE_SIZE <= p.x && this.y[i]+this.h-MWIN_RESIZE_HANDLE_SIZE <= p.y ) {
                return MWIN_RESIZE_HANDLE;
            }
            return MWIN_BODY;
        }
        return MWIN_NONE;
    }
    animate(delta_u /*same value between 0.0 and 1.0*/) { // returns false if the animation is finished
        if ( this.isAnimating ) {
            this.animation_u += delta_u;
            if ( this.animation_u >= 1.0 ) {
                this.animation_u = 1.0;
                this.isAnimating = false;
            }
            let targetLayer = this.currentLayer;
            let sourceLayer = 1 - targetLayer;
            this.animation_x = (1-this.animation_u)*this.x[sourceLayer] + (this.animation_u)*this.x[targetLayer];
            this.animation_y = (1-this.animation_u)*this.y[sourceLayer] + (this.animation_u)*this.y[targetLayer];
            return true;
        }
        return false;
    }
    move(dx,dy) {
        this.x[this.currentLayer] += dx;
        this.y[this.currentLayer] += dy;
        this.isBoundingRectDirty[ this.currentLayer ] = true;
    }
    resize(dx,dy) {
        this.w += dx;
        this.h += dy;
        this.isBoundingRectDirty[ this.currentLayer ] = true;
    }
    draw( partHilited/*e.g. MWIN_MOVE_HANDLE*/, drawConnection ) {
        let xx = this.x[this.currentLayer];
        let yy = this.y[this.currentLayer];
        if ( this.isAnimating ) {
            xx = this.animation_x;
            yy = this.animation_y;
        }
        canvas_context.fillStyle = defaultBackgroundColorWithAlpha;
        canvas_context.fillRect(xx,yy,this.w,this.h);
        canvas_context.strokeStyle = (partHilited==MWIN_BODY?"#0080ff":defaultBorderColor);
        canvas_context.strokeRect(xx,yy,this.w,this.h);
        canvas_context.strokeStyle = (partHilited==MWIN_MOVE_HANDLE?"#0080ff":defaultBorderColor);
        canvas_context.strokeRect(xx,yy,this.w,MWIN_MOVE_HANDLE_SIZE);
        canvas_context.strokeStyle = (partHilited==MWIN_RESIZE_HANDLE?"#0080ff":defaultBorderColor);
        canvas_context.strokeRect(xx+this.w-MWIN_RESIZE_HANDLE_SIZE,yy+this.h-MWIN_RESIZE_HANDLE_SIZE,MWIN_RESIZE_HANDLE_SIZE,MWIN_RESIZE_HANDLE_SIZE);
        let labelWidth = canvas_context.measureText(this.label).width;
        if ( labelWidth < this.w && fontHeight < this.h-MWIN_MOVE_HANDLE_SIZE ) {
            canvas_context.fillStyle = defaultBorderColor;
            canvas_context.fillText(this.label, xx+(this.w-labelWidth)/2, yy+this.h-(this.h-MWIN_MOVE_HANDLE_SIZE-fontHeight)/2);
        }
        if ( drawConnection ) {
            if ( this.currentLayer==L_FRONT || this.hasPositionInFrontLayer ) {
                let xx2 = this.x[1-this.currentLayer];
                let yy2 = this.y[1-this.currentLayer];
                canvas_context.fillStyle = connectionColor;
                if ( yy2 < yy ) {
                    canvas_context.beginPath();
                    canvas_context.moveTo(xx,yy);
                    canvas_context.lineTo(xx2,yy2);
                    canvas_context.lineTo(xx2+this.w,yy2);
                    canvas_context.lineTo(xx+this.w,yy);
                    canvas_context.fill();
                }
                if ( yy < yy2 ) {
                    canvas_context.beginPath();
                    canvas_context.moveTo(xx,yy+this.h);
                    canvas_context.lineTo(xx2,yy2+this.h);
                    canvas_context.lineTo(xx2+this.w,yy2+this.h);
                    canvas_context.lineTo(xx+this.w,yy+this.h);
                    canvas_context.fill();
                }
                if ( xx2 < xx ) {
                    canvas_context.beginPath();
                    canvas_context.moveTo(xx,yy);
                    canvas_context.lineTo(xx2,yy2);
                    canvas_context.lineTo(xx2,yy2+this.h);
                    canvas_context.lineTo(xx,yy+this.h);
                    canvas_context.fill();
                }
                if ( xx < xx2 ) {
                    canvas_context.beginPath();
                    canvas_context.moveTo(xx+this.w,yy);
                    canvas_context.lineTo(xx2+this.w,yy2);
                    canvas_context.lineTo(xx2+this.w,yy2+this.h);
                    canvas_context.lineTo(xx+this.w,yy+this.h);
                    canvas_context.fill();
                }
            }
        }
    }

}

let wins = [];

let frontLayer = {
    x : 0,
    y : 0,
    w : 0,
    h : 0,
    isAnimating : false,
    animation_u : 0,
    anim_xi : 0, anim_xf : 0, anim_yi : 0, anim_yf : 0, anim_wi : 0, anim_wf : 0, anim_hi : 0, anim_hf : 0,
    boundingRect : new Box2(),
    isBoundingRectDirty : true,
    getBoundingRect : function() {
        if ( this.isBoundingRectDirty ) {
            this.boundingRect.clear();
            for ( let j = 0; j < wins.length; ++j ) {
                if ( wins[j].currentLayer===L_FRONT ) {
                    this.boundingRect.boundBox( wins[j].getBoundingRect(L_FRONT) );
                }
            }
            this.isBoundingRectDirty = false;
        }
        return this.boundingRect;
    },
    recomputeGeometry : function() {
        this.isBoundingRectDirty = true;
        let b = this.getBoundingRect();
        this.x = b.min.x;
        this.y = b.min.y;
        this.w = b.width();
        this.h = b.height();
    },
    animate : function(delta_u /*same value between 0.0 and 1.0*/) { // returns false if the animation is finished
        if ( this.isAnimating ) {
            this.animation_u += delta_u;
            if ( this.animation_u >= 1.0 ) {
                this.animation_u = 1.0;
                this.isAnimating = false;
            }
            let new_x = (1-this.animation_u)*this.anim_xi + (this.animation_u)*this.anim_xf;
            let new_y = (1-this.animation_u)*this.anim_yi + (this.animation_u)*this.anim_yf;
            let delta_x = new_x - this.x;
            let delta_y = new_y - this.y;
            this.x = new_x;
            this.y = new_y;
            this.w = (1-this.animation_u)*this.anim_wi + (this.animation_u)*this.anim_wf;
            this.h = (1-this.animation_u)*this.anim_hi + (this.animation_u)*this.anim_hf;

            // update positions of windows on the front layer
            for ( let j = 0; j < wins.length; ++j ) {
                // if ( wins[j].currentLayer===L_FRONT && ! wins[j].isAnimating ) {
                if ( wins[j].hasPositionInFrontLayer ) {
                    wins[j].x[L_FRONT] += delta_x;
                    wins[j].y[L_FRONT] += delta_y;
                    wins[j].isBoundingRectDirty[L_FRONT] = true;
                }
            }

            this.isBoundingRectDirty = true;

            return true;
        }
        return false;
    },
    startAnimationIfNecessary : function() {
        let b = this.getBoundingRect();
        if ( ! b.isEmpty ) {
            let c = b.center();
            if ( Math.abs(c.x - canvas.width/2) > 1 || Math.abs(c.y - canvas.height/2) ) {
                this.anim_xi = this.x;
                this.anim_xf = canvas.width/2 - b.width()/2;
                this.anim_yi = this.y;
                this.anim_yf = canvas.height/2 - b.height()/2;
                this.anim_wi = this.w;
                this.anim_wf = b.width();
                this.anim_hi = this.h;
                this.anim_hf = b.height();

                this.isAnimating = true;
                this.animation_u = 0;
            }
        }
    },
    draw : function( veryTransparent ) {
        let b = this.getBoundingRect();
        if ( b.isEmpty ) return;
        canvas_context.lineWidth = frontLayerBorderThickness;
        if ( veryTransparent ) {
            canvas_context.strokeStyle = frontLayerBorderColorVeryTransparent;
            canvas_context.strokeRect(this.x,this.y,this.w,this.h);
        }
        else {
            canvas_context.fillStyle = frontLayerBackground;
            canvas_context.fillRect(this.x,this.y,this.w,this.h);
            canvas_context.strokeStyle = frontLayerBorderColor;
            canvas_context.strokeRect(this.x,this.y,this.w,this.h);
        }
        canvas_context.lineWidth = 1;
    }
}

function init() {
    canvas_context.font = fontName1;
    for ( let j = 0; j < NUM_WINDOWS; ++j ) {
        let w = myRand()*70 + 50;
        let h = myRand()*60 + 40;
        let x = (0.05+myRand()*0.9)*(canvas.width-w);
        let y = (0.05+myRand()*0.9)*(canvas.height-h);
        let numColumns = 5;
        let numRows = 3;
        if ( j < numColumns * numRows  ) {
            let margin = 5; // margin between windows
            let marginAtBottom = 30; // along bottom of canvas
            w = canvas.width / numColumns;
            h = (canvas.height-marginAtBottom) / numRows;
            x = (j % numColumns) * w + margin;
            y = Math.floor(j/numColumns) * h + margin;
            w -= 2*margin;
            h -= 2*margin;
        }
        let characterCode = 65 + (j%25);
        if ( characterCode >= "I".charCodeAt(0) ) // we don't want letter I, because it can look like a letter 1 or l when drawn
            characterCode ++;
        let s = String.fromCharCode(characterCode) + ((j%9)+1);

        // let randomCharacterCode = Math.floor( 65 + Math.random()*24 );
        // if ( randomCharacterCode >= "I".charCodeAt(0) ) // we don't want letter I, because it can look like a letter 1 or l when drawn
        //     randomCharacterCode ++;
        // let s = String.fromCharCode(randomCharacterCode) + Math.floor(Math.random()*9+1);

        let mw = new MWin(x,y,w,h,s);
        wins.push( mw );
    }
}

let winUnderMouse = -1; // -1 for none
let partUnderMouse = -1; // -1 for none

function findPartOfWinUnderPoint( p ) { // returns { windowIndex:int, part:int }
    if ( ! isFrontLayerVeryTransparent() ) {
        for ( let j = wins.length-1; j >= 0 ; --j ) {
            if ( !wins[j].isAnimating && wins[j].currentLayer===L_FRONT ) {
                let part = wins[j].getPartContainingPoint(L_FRONT,p)
                if ( part != MWIN_NONE ) {
                    return {windowIndex:j, part:part};
                }
            }
        }
    }
    for ( let j = wins.length-1; j >= 0 ; --j ) {
        if ( !wins[j].isAnimating && wins[j].currentLayer===L_BACK ) {
            let part = wins[j].getPartContainingPoint(L_BACK,p)
            if ( part != MWIN_NONE ) {
                return {windowIndex:j, part:part};
            }
        }
    }
    return {windowIndex:-1,part:-1};
}

// returns true if a redraw is necessary
function updateHighlight() {
    let returnValue = findPartOfWinUnderPoint( new Vec2( mouse_x, mouse_y ) );
    if ( returnValue.windowIndex != winUnderMouse || returnValue.part != partUnderMouse ) {
        winUnderMouse = returnValue.windowIndex;
        partUnderMouse = returnValue.part;
        return true;
    }
    return false;
}

function drawStringInBox(x,y,w,h,s,inverted) {
    canvas_context.font = fontName2;
    if ( inverted ) {
        canvas_context.fillStyle = defaultBorderColor;
        canvas_context.fillRect(x,y,w,h);
    }
    else {
        canvas_context.strokeStyle = defaultBorderColor;
        canvas_context.strokeRect(x,y,w,h);
    }
    let labelWidth = canvas_context.measureText(s).width;
    canvas_context.fillStyle = inverted ? defaultBackgroundColor : defaultBorderColor;
    canvas_context.fillText(s, x+(w-labelWidth)/2, y+h-(h-fontHeight2)/2);
    canvas_context.font = fontName1;
}

function draw() {

    canvas_context.clearRect(0, 0, canvas.width, canvas.height);

    for ( let j = 0; j < wins.length; ++j ) {
        if ( !wins[j].isAnimating && wins[j].currentLayer===L_BACK ) {
            wins[j].draw( winUnderMouse==j?partUnderMouse:MWIN_NONE, winUnderMouse==j && isModifiedStateActive() );
        }
    }
    for ( let j = 0; j < wins.length; ++j ) {
        if ( wins[j].isAnimating ) {
            wins[j].draw(winUnderMouse==j?partUnderMouse:MWIN_NONE,false);
        }
    }
    let veryTransparent = isFrontLayerVeryTransparent();
    frontLayer.draw( veryTransparent );
    if ( ! veryTransparent ) {
        for ( let j = 0; j < wins.length; ++j ) {
            if ( !wins[j].isAnimating && wins[j].currentLayer===L_FRONT ) {
                wins[j].draw( winUnderMouse==j?partUnderMouse:MWIN_NONE, winUnderMouse==j && isModifiedStateActive() );
            }
        }
    }

    // draw states of keys and buttons
    let boxWidth = 40;
    let boxHeight = fontHeight2 * 2;
    let boxMargin = 5;
    drawStringInBox( boxMargin, canvas.height-boxHeight-boxMargin, boxWidth, boxHeight, "Ctrl", isCtrlDown );
    drawStringInBox( boxMargin*2+boxWidth, canvas.height-boxHeight-boxMargin, boxWidth, boxHeight, "Alt", isAltDown );
    // drawStringInBox( canvas.width - boxMargin-boxWidth, canvas.height-boxHeight-boxMargin, boxWidth, boxHeight, "Left", isMouseButtonDown );
    drawStringInBox( boxMargin*3+boxWidth*2, canvas.height-boxHeight-boxMargin, boxWidth, boxHeight, "Left", isMouseButtonDown );
}

let deltaT = 33; // in milliseconds
let durationOfAnimation = 400; // in milliseconds
let deltaU = deltaT / durationOfAnimation;

function mouseDownHandler(e) {
    //console.log("mouse down");
    let canvas_rectangle = canvas.getBoundingClientRect();
    mouse_x = e.clientX - canvas_rectangle.left;
    mouse_y = e.clientY - canvas_rectangle.top;
    isMouseButtonDown = true;

    if ( winUnderMouse != -1 ) {
        let w = wins.splice(winUnderMouse,1);
        wins.push( w[0] );
        winUnderMouse = wins.length-1;
    }
    draw();
}
function mouseUpHandler(e) {
    //console.log("mouse up");
    let canvas_rectangle = canvas.getBoundingClientRect();
    mouse_x = e.clientX - canvas_rectangle.left;
    mouse_y = e.clientY - canvas_rectangle.top;
    isMouseButtonDown = false;

    if ( isModifiedStateActive() ) {
        if ( winUnderMouse != -1 && partUnderMouse == MWIN_BODY ) {
            // initiate animated teleportation
            if ( wins[ winUnderMouse ].currentLayer === L_BACK ) {
                if ( ! wins[ winUnderMouse ].hasPositionInFrontLayer ) {
                    let b = frontLayer.getBoundingRect();
                    if ( b.isEmpty ) {
                        wins[ winUnderMouse ].x[L_FRONT] = canvas.width/2 - wins[ winUnderMouse ].w/2;
                    }
                    else if ( wins[ winUnderMouse ].x[L_BACK] + wins[ winUnderMouse ].w/2 < canvas.width/2 ) {
                        // add the window on the left side of the front layer
                        wins[ winUnderMouse ].x[L_FRONT] = b.min.x - wins[ winUnderMouse ].w - 10;
                    }
                    else {
                        // add the window on the right side of the front layer
                        wins[ winUnderMouse ].x[L_FRONT] = b.max.x + 10;
                    }
                    wins[ winUnderMouse ].y[L_FRONT] = canvas.height/2 - wins[ winUnderMouse ].h/2;
                    wins[ winUnderMouse ].isBoundingRectDirty[L_FRONT] = true;
                    wins[ winUnderMouse ].hasPositionInFrontLayer = true;
                }

                wins[ winUnderMouse ].currentLayer = L_FRONT;
                wins[ winUnderMouse ].isAnimating = true;
                wins[ winUnderMouse ].animation_u = 0;
                wins[ winUnderMouse ].animate(deltaU);
            }
            else {
                wins[ winUnderMouse ].currentLayer = L_BACK;
                wins[ winUnderMouse ].isAnimating = true;
                wins[ winUnderMouse ].animation_u = 0;
                wins[ winUnderMouse ].animate(deltaU);
            }
            frontLayer.recomputeGeometry();
        }
    }

    frontLayer.startAnimationIfNecessary();
    draw();
}
function mouseMoveHandler(e) {
    //console.log("mouse move");
    let canvas_rectangle = canvas.getBoundingClientRect();
    let event_x = e.clientX - canvas_rectangle.left;
    let event_y = e.clientY - canvas_rectangle.top;
    let dx = event_x - mouse_x;
    let dy = event_y - mouse_y;
    mouse_x = event_x;
    mouse_y = event_y;

    let flag = false;
    if ( ! isMouseButtonDown ) {
        flag = updateHighlight();
    }
    else if ( winUnderMouse != -1 ) {
        if ( partUnderMouse == MWIN_MOVE_HANDLE ) {
            wins[ winUnderMouse ].move( dx, dy );
        }
        else if ( partUnderMouse == MWIN_RESIZE_HANDLE ) {
            wins[ winUnderMouse ].resize( dx, dy );
        }
        if ( wins[winUnderMouse].currentLayer===L_FRONT ) {
            frontLayer.recomputeGeometry();
        }
        flag = true;
    }
    if ( flag ) draw();
}

canvas.addEventListener('mousedown',mouseDownHandler);
canvas.addEventListener('mouseup',mouseUpHandler);
canvas.addEventListener('mousemove',mouseMoveHandler);

init();
draw();

function advanceOneDelta() {
    let returnValue = false;
    for ( let j = 0; j < wins.length; ++j ) {
        if ( wins[j].isAnimating ) {
            returnValue = returnValue || wins[j].animate( deltaU );
        }
    }
    // returnValue = returnValue || frontLayer.animate( deltaU );
    // if ( returnValue ) draw();
    let returnValue2 = frontLayer.animate( deltaU );
    if ( returnValue || returnValue2 ) draw();
}

setInterval( function() { advanceOneDelta(); }, deltaT );

</script>
</body>
</html>

