mirror of
https://github.com/google/blockly.git
synced 2025-12-16 14:20:10 +01:00
* fix(build): Restore erroneously-deleted filter function This was deleted in PR #7406 as it was mainly being used to filter core/ vs. test/mocha/ deps into separate deps files - but it turns out also to be used for filtering error messages too. Oops. * refactor(tests): Migrate advanced compilation test to ES Modules * refactor(build): Migrate main.js to TypeScript This turns out to be pretty straight forward, even if it would cause crashing if one actually tried to import this module instead of just feeding it to Closure Compiler. * chore(build): Remove goog.declareModuleId calls Replace goog.declareModuleId calls with a comment recording the former module ID for posterity (or at least until we decide how to reformat the renamings file. * chore(tests): Delete closure/goog/* For the moment we still need something to serve as base.js for the benefit of closure-make-deps, so we keep a vestigial base.js around, containing only the @provideGoog declaration. * refactor(build): Remove vestigial base.js By changing slightly the command line arguments to closure-make-deps and closure-calculate-chunks the need to have any base.js is eliminated. * chore: Typo fix for PR #7415
493 lines
14 KiB
TypeScript
493 lines
14 KiB
TypeScript
/**
|
|
* @license
|
|
* Copyright 2015 Google LLC
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
/**
|
|
* Object representing a zoom icons.
|
|
*
|
|
* @class
|
|
*/
|
|
// Former goog.module ID: Blockly.ZoomControls
|
|
|
|
// Unused import preserved for side-effects. Remove if unneeded.
|
|
import './events/events_click.js';
|
|
|
|
import * as browserEvents from './browser_events.js';
|
|
import {ComponentManager} from './component_manager.js';
|
|
import * as Css from './css.js';
|
|
import * as eventUtils from './events/utils.js';
|
|
import type {IPositionable} from './interfaces/i_positionable.js';
|
|
import type {UiMetrics} from './metrics_manager.js';
|
|
import * as uiPosition from './positionable_helpers.js';
|
|
import {SPRITE} from './sprites.js';
|
|
import * as Touch from './touch.js';
|
|
import * as dom from './utils/dom.js';
|
|
import {Rect} from './utils/rect.js';
|
|
import {Size} from './utils/size.js';
|
|
import {Svg} from './utils/svg.js';
|
|
import type {WorkspaceSvg} from './workspace_svg.js';
|
|
|
|
/**
|
|
* Class for a zoom controls.
|
|
*/
|
|
export class ZoomControls implements IPositionable {
|
|
/**
|
|
* The unique ID for this component that is used to register with the
|
|
* ComponentManager.
|
|
*/
|
|
id = 'zoomControls';
|
|
|
|
/**
|
|
* Array holding info needed to unbind events.
|
|
* Used for disposing.
|
|
* Ex: [[node, name, func], [node, name, func]].
|
|
*/
|
|
private boundEvents: browserEvents.Data[] = [];
|
|
|
|
/** The zoom in svg <g> element. */
|
|
private zoomInGroup: SVGGElement | null = null;
|
|
|
|
/** The zoom out svg <g> element. */
|
|
private zoomOutGroup: SVGGElement | null = null;
|
|
|
|
/** The zoom reset svg <g> element. */
|
|
private zoomResetGroup: SVGGElement | null = null;
|
|
|
|
/** Width of the zoom controls. */
|
|
private readonly WIDTH = 32;
|
|
|
|
/** Height of each zoom control. */
|
|
private readonly HEIGHT = 32;
|
|
|
|
/** Small spacing used between the zoom in and out control, in pixels. */
|
|
private readonly SMALL_SPACING = 2;
|
|
|
|
/**
|
|
* Large spacing used between the zoom in and reset control, in pixels.
|
|
*/
|
|
private readonly LARGE_SPACING = 11;
|
|
|
|
/** Distance between zoom controls and bottom or top edge of workspace. */
|
|
private readonly MARGIN_VERTICAL = 20;
|
|
|
|
/** Distance between zoom controls and right or left edge of workspace. */
|
|
private readonly MARGIN_HORIZONTAL = 20;
|
|
|
|
/** The SVG group containing the zoom controls. */
|
|
private svgGroup: SVGElement | null = null;
|
|
|
|
/** Left coordinate of the zoom controls. */
|
|
private left = 0;
|
|
|
|
/** Top coordinate of the zoom controls. */
|
|
private top = 0;
|
|
|
|
/** Whether this has been initialized. */
|
|
private initialized = false;
|
|
|
|
/** @param workspace The workspace to sit in. */
|
|
constructor(private readonly workspace: WorkspaceSvg) {}
|
|
|
|
/**
|
|
* Create the zoom controls.
|
|
*
|
|
* @returns The zoom controls SVG group.
|
|
*/
|
|
createDom(): SVGElement {
|
|
this.svgGroup = dom.createSvgElement(Svg.G, {});
|
|
|
|
// Each filter/pattern needs a unique ID for the case of multiple Blockly
|
|
// instances on a page. Browser behaviour becomes undefined otherwise.
|
|
// https://neil.fraser.name/news/2015/11/01/
|
|
const rnd = String(Math.random()).substring(2);
|
|
this.createZoomOutSvg(rnd);
|
|
this.createZoomInSvg(rnd);
|
|
if (this.workspace.isMovable()) {
|
|
// If we zoom to the center and the workspace isn't movable we could
|
|
// loose blocks at the edges of the workspace.
|
|
this.createZoomResetSvg(rnd);
|
|
}
|
|
return this.svgGroup;
|
|
}
|
|
|
|
/** Initializes the zoom controls. */
|
|
init() {
|
|
this.workspace.getComponentManager().addComponent({
|
|
component: this,
|
|
weight: 2,
|
|
capabilities: [ComponentManager.Capability.POSITIONABLE],
|
|
});
|
|
this.initialized = true;
|
|
}
|
|
|
|
/**
|
|
* Disposes of this zoom controls.
|
|
* Unlink from all DOM elements to prevent memory leaks.
|
|
*/
|
|
dispose() {
|
|
this.workspace.getComponentManager().removeComponent('zoomControls');
|
|
if (this.svgGroup) {
|
|
dom.removeNode(this.svgGroup);
|
|
}
|
|
for (const event of this.boundEvents) {
|
|
browserEvents.unbind(event);
|
|
}
|
|
this.boundEvents.length = 0;
|
|
}
|
|
|
|
/**
|
|
* Returns the bounding rectangle of the UI element in pixel units relative to
|
|
* the Blockly injection div.
|
|
*
|
|
* @returns The UI elements's bounding box. Null if bounding box should be
|
|
* ignored by other UI elements.
|
|
*/
|
|
getBoundingRectangle(): Rect | null {
|
|
let height = this.SMALL_SPACING + 2 * this.HEIGHT;
|
|
if (this.zoomResetGroup) {
|
|
height += this.LARGE_SPACING + this.HEIGHT;
|
|
}
|
|
const bottom = this.top + height;
|
|
const right = this.left + this.WIDTH;
|
|
return new Rect(this.top, bottom, this.left, right);
|
|
}
|
|
|
|
/**
|
|
* Positions the zoom controls.
|
|
* It is positioned in the opposite corner to the corner the
|
|
* categories/toolbox starts at.
|
|
*
|
|
* @param metrics The workspace metrics.
|
|
* @param savedPositions List of rectangles that are already on the workspace.
|
|
*/
|
|
position(metrics: UiMetrics, savedPositions: Rect[]) {
|
|
// Not yet initialized.
|
|
if (!this.initialized) {
|
|
return;
|
|
}
|
|
|
|
const cornerPosition = uiPosition.getCornerOppositeToolbox(
|
|
this.workspace,
|
|
metrics,
|
|
);
|
|
let height = this.SMALL_SPACING + 2 * this.HEIGHT;
|
|
if (this.zoomResetGroup) {
|
|
height += this.LARGE_SPACING + this.HEIGHT;
|
|
}
|
|
const startRect = uiPosition.getStartPositionRect(
|
|
cornerPosition,
|
|
new Size(this.WIDTH, height),
|
|
this.MARGIN_HORIZONTAL,
|
|
this.MARGIN_VERTICAL,
|
|
metrics,
|
|
this.workspace,
|
|
);
|
|
|
|
const verticalPosition = cornerPosition.vertical;
|
|
const bumpDirection =
|
|
verticalPosition === uiPosition.verticalPosition.TOP
|
|
? uiPosition.bumpDirection.DOWN
|
|
: uiPosition.bumpDirection.UP;
|
|
const positionRect = uiPosition.bumpPositionRect(
|
|
startRect,
|
|
this.MARGIN_VERTICAL,
|
|
bumpDirection,
|
|
savedPositions,
|
|
);
|
|
|
|
if (verticalPosition === uiPosition.verticalPosition.TOP) {
|
|
const zoomInTranslateY = this.SMALL_SPACING + this.HEIGHT;
|
|
this.zoomInGroup?.setAttribute(
|
|
'transform',
|
|
'translate(0, ' + zoomInTranslateY + ')',
|
|
);
|
|
if (this.zoomResetGroup) {
|
|
const zoomResetTranslateY =
|
|
zoomInTranslateY + this.LARGE_SPACING + this.HEIGHT;
|
|
this.zoomResetGroup.setAttribute(
|
|
'transform',
|
|
'translate(0, ' + zoomResetTranslateY + ')',
|
|
);
|
|
}
|
|
} else {
|
|
const zoomInTranslateY = this.zoomResetGroup
|
|
? this.LARGE_SPACING + this.HEIGHT
|
|
: 0;
|
|
this.zoomInGroup?.setAttribute(
|
|
'transform',
|
|
'translate(0, ' + zoomInTranslateY + ')',
|
|
);
|
|
const zoomOutTranslateY =
|
|
zoomInTranslateY + this.SMALL_SPACING + this.HEIGHT;
|
|
this.zoomOutGroup?.setAttribute(
|
|
'transform',
|
|
'translate(0, ' + zoomOutTranslateY + ')',
|
|
);
|
|
}
|
|
|
|
this.top = positionRect.top;
|
|
this.left = positionRect.left;
|
|
this.svgGroup?.setAttribute(
|
|
'transform',
|
|
'translate(' + this.left + ',' + this.top + ')',
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Create the zoom in icon and its event handler.
|
|
*
|
|
* @param rnd The random string to use as a suffix in the clip path's ID.
|
|
* These IDs must be unique in case there are multiple Blockly instances
|
|
* on the same page.
|
|
*/
|
|
private createZoomOutSvg(rnd: string) {
|
|
/* This markup will be generated and added to the .svgGroup:
|
|
<g class="blocklyZoom">
|
|
<clipPath id="blocklyZoomoutClipPath837493">
|
|
<rect width="32" height="32></rect>
|
|
</clipPath>
|
|
<image width="96" height="124" x="-64" y="-92"
|
|
xlink:href="media/sprites.png"
|
|
clip-path="url(#blocklyZoomoutClipPath837493)"></image>
|
|
</g>
|
|
*/
|
|
this.zoomOutGroup = dom.createSvgElement(
|
|
Svg.G,
|
|
{'class': 'blocklyZoom blocklyZoomOut'},
|
|
this.svgGroup,
|
|
);
|
|
const clip = dom.createSvgElement(
|
|
Svg.CLIPPATH,
|
|
{'id': 'blocklyZoomoutClipPath' + rnd},
|
|
this.zoomOutGroup,
|
|
);
|
|
dom.createSvgElement(
|
|
Svg.RECT,
|
|
{
|
|
'width': 32,
|
|
'height': 32,
|
|
},
|
|
clip,
|
|
);
|
|
const zoomoutSvg = dom.createSvgElement(
|
|
Svg.IMAGE,
|
|
{
|
|
'width': SPRITE.width,
|
|
'height': SPRITE.height,
|
|
'x': -64,
|
|
'y': -92,
|
|
'clip-path': 'url(#blocklyZoomoutClipPath' + rnd + ')',
|
|
},
|
|
this.zoomOutGroup,
|
|
);
|
|
zoomoutSvg.setAttributeNS(
|
|
dom.XLINK_NS,
|
|
'xlink:href',
|
|
this.workspace.options.pathToMedia + SPRITE.url,
|
|
);
|
|
|
|
// Attach listener.
|
|
this.boundEvents.push(
|
|
browserEvents.conditionalBind(
|
|
this.zoomOutGroup,
|
|
'pointerdown',
|
|
null,
|
|
this.zoom.bind(this, -1),
|
|
),
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Create the zoom out icon and its event handler.
|
|
*
|
|
* @param rnd The random string to use as a suffix in the clip path's ID.
|
|
* These IDs must be unique in case there are multiple Blockly instances
|
|
* on the same page.
|
|
*/
|
|
private createZoomInSvg(rnd: string) {
|
|
/* This markup will be generated and added to the .svgGroup:
|
|
<g class="blocklyZoom">
|
|
<clipPath id="blocklyZoominClipPath837493">
|
|
<rect width="32" height="32"></rect>
|
|
</clipPath>
|
|
<image width="96" height="124" x="-32" y="-92"
|
|
xlink:href="media/sprites.png"
|
|
clip-path="url(#blocklyZoominClipPath837493)"></image>
|
|
</g>
|
|
*/
|
|
this.zoomInGroup = dom.createSvgElement(
|
|
Svg.G,
|
|
{'class': 'blocklyZoom blocklyZoomIn'},
|
|
this.svgGroup,
|
|
);
|
|
const clip = dom.createSvgElement(
|
|
Svg.CLIPPATH,
|
|
{'id': 'blocklyZoominClipPath' + rnd},
|
|
this.zoomInGroup,
|
|
);
|
|
dom.createSvgElement(
|
|
Svg.RECT,
|
|
{
|
|
'width': 32,
|
|
'height': 32,
|
|
},
|
|
clip,
|
|
);
|
|
const zoominSvg = dom.createSvgElement(
|
|
Svg.IMAGE,
|
|
{
|
|
'width': SPRITE.width,
|
|
'height': SPRITE.height,
|
|
'x': -32,
|
|
'y': -92,
|
|
'clip-path': 'url(#blocklyZoominClipPath' + rnd + ')',
|
|
},
|
|
this.zoomInGroup,
|
|
);
|
|
zoominSvg.setAttributeNS(
|
|
dom.XLINK_NS,
|
|
'xlink:href',
|
|
this.workspace.options.pathToMedia + SPRITE.url,
|
|
);
|
|
|
|
// Attach listener.
|
|
this.boundEvents.push(
|
|
browserEvents.conditionalBind(
|
|
this.zoomInGroup,
|
|
'pointerdown',
|
|
null,
|
|
this.zoom.bind(this, 1),
|
|
),
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Handles a mouse down event on the zoom in or zoom out buttons on the
|
|
* workspace.
|
|
*
|
|
* @param amount Amount of zooming. Negative amount values zoom out, and
|
|
* positive amount values zoom in.
|
|
* @param e A mouse down event.
|
|
*/
|
|
private zoom(amount: number, e: PointerEvent) {
|
|
this.workspace.markFocused();
|
|
this.workspace.zoomCenter(amount);
|
|
this.fireZoomEvent();
|
|
Touch.clearTouchIdentifier(); // Don't block future drags.
|
|
e.stopPropagation(); // Don't start a workspace scroll.
|
|
e.preventDefault(); // Stop double-clicking from selecting text.
|
|
}
|
|
|
|
/**
|
|
* Create the zoom reset icon and its event handler.
|
|
*
|
|
* @param rnd The random string to use as a suffix in the clip path's ID.
|
|
* These IDs must be unique in case there are multiple Blockly instances
|
|
* on the same page.
|
|
*/
|
|
private createZoomResetSvg(rnd: string) {
|
|
/* This markup will be generated and added to the .svgGroup:
|
|
<g class="blocklyZoom">
|
|
<clipPath id="blocklyZoomresetClipPath837493">
|
|
<rect width="32" height="32"></rect>
|
|
</clipPath>
|
|
<image width="96" height="124" x="-32" y="-92"
|
|
xlink:href="media/sprites.png"
|
|
clip-path="url(#blocklyZoomresetClipPath837493)"></image>
|
|
</g>
|
|
*/
|
|
this.zoomResetGroup = dom.createSvgElement(
|
|
Svg.G,
|
|
{'class': 'blocklyZoom blocklyZoomReset'},
|
|
this.svgGroup,
|
|
);
|
|
const clip = dom.createSvgElement(
|
|
Svg.CLIPPATH,
|
|
{'id': 'blocklyZoomresetClipPath' + rnd},
|
|
this.zoomResetGroup,
|
|
);
|
|
dom.createSvgElement(Svg.RECT, {'width': 32, 'height': 32}, clip);
|
|
const zoomresetSvg = dom.createSvgElement(
|
|
Svg.IMAGE,
|
|
{
|
|
'width': SPRITE.width,
|
|
'height': SPRITE.height,
|
|
'y': -92,
|
|
'clip-path': 'url(#blocklyZoomresetClipPath' + rnd + ')',
|
|
},
|
|
this.zoomResetGroup,
|
|
);
|
|
zoomresetSvg.setAttributeNS(
|
|
dom.XLINK_NS,
|
|
'xlink:href',
|
|
this.workspace.options.pathToMedia + SPRITE.url,
|
|
);
|
|
|
|
// Attach event listeners.
|
|
this.boundEvents.push(
|
|
browserEvents.conditionalBind(
|
|
this.zoomResetGroup,
|
|
'pointerdown',
|
|
null,
|
|
this.resetZoom.bind(this),
|
|
),
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Handles a mouse down event on the reset zoom button on the workspace.
|
|
*
|
|
* @param e A mouse down event.
|
|
*/
|
|
private resetZoom(e: PointerEvent) {
|
|
this.workspace.markFocused();
|
|
|
|
// zoom is passed amount and computes the new scale using the formula:
|
|
// targetScale = currentScale * Math.pow(speed, amount)
|
|
const targetScale = this.workspace.options.zoomOptions.startScale;
|
|
const currentScale = this.workspace.scale;
|
|
const speed = this.workspace.options.zoomOptions.scaleSpeed;
|
|
// To compute amount:
|
|
// amount = log(speed, (targetScale / currentScale))
|
|
// Math.log computes natural logarithm (ln), to change the base, use
|
|
// formula: log(base, value) = ln(value) / ln(base)
|
|
const amount = Math.log(targetScale / currentScale) / Math.log(speed);
|
|
this.workspace.beginCanvasTransition();
|
|
this.workspace.zoomCenter(amount);
|
|
this.workspace.scrollCenter();
|
|
|
|
setTimeout(this.workspace.endCanvasTransition.bind(this.workspace), 500);
|
|
this.fireZoomEvent();
|
|
Touch.clearTouchIdentifier(); // Don't block future drags.
|
|
e.stopPropagation(); // Don't start a workspace scroll.
|
|
e.preventDefault(); // Stop double-clicking from selecting text.
|
|
}
|
|
|
|
/** Fires a zoom control UI event. */
|
|
private fireZoomEvent() {
|
|
const uiEvent = new (eventUtils.get(eventUtils.CLICK))(
|
|
null,
|
|
this.workspace.id,
|
|
'zoom_controls',
|
|
);
|
|
eventUtils.fire(uiEvent);
|
|
}
|
|
}
|
|
|
|
/** CSS for zoom controls. See css.js for use. */
|
|
Css.register(`
|
|
.blocklyZoom>image, .blocklyZoom>svg>image {
|
|
opacity: .4;
|
|
}
|
|
|
|
.blocklyZoom>image:hover, .blocklyZoom>svg>image:hover {
|
|
opacity: .6;
|
|
}
|
|
|
|
.blocklyZoom>image:active, .blocklyZoom>svg>image:active {
|
|
opacity: .8;
|
|
}
|
|
`);
|