Files
blockly/core/zoom_controls.ts
Christopher Allen 7ccdcc5cef refactor(events): introduce EventType enum in separate module (#8530)
* refactor(events): Use "export ... from" where applicable

* refactor(events): Introduce EventType enum

  Introduce an enum for the event .type values.  We can't actually
  use it as the type of the .type property on Abstract events,
  because we want to allow developers to add their own custom
  event types inheriting from this type, but at least this way we
  can be reasonably sure that all of our own event subclasses have
  distinct .type values—plus consistent use of enum syntax
  (EventType.TYPE_NAME) is probably good for readability overall.

  Put it in a separate module from the rest of events/utils.ts
  because it would be helpful if event utils could use

      event instanceof SomeEventType

  for type narrowing but but at the moment most events are in
  modules that depend on events/utils.ts for their .type
  constant, and although circular ESM dependencies should work
  in principle there are various restrictions and this
  particular circularity causes issues at the moment.

  A few of the event classes also depend on utils.ts for fire()
  or other functions, which will be harder to deal with, but at
  least this commit is win in terms of reducing the complexity
  of our dependencies, making most of the Abstract event subclass
  module dependent on type.ts, which has no imports, rather than
  on utils.ts which has multiple imports.
2024-08-20 08:36:33 +01:00

494 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 {EventType} from './events/type.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: ComponentManager.ComponentWeight.ZOOM_CONTROLS_WEIGHT,
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(EventType.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;
}
`);