123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325 |
- /**
- * Copyright (c) 2020 mol* contributors, licensed under MIT, See LICENSE file for more info.
- *
- * @author David Sehnal <david.sehnal@gmail.com>
- */
- import * as React from 'react';
- import { useEffect, useLayoutEffect, useRef, useState } from 'react';
- import { Observable, Subscription } from 'rxjs';
- import { Viewport } from '../../mol-canvas3d/camera/util';
- import { PluginContext } from '../../mol-plugin/context';
- import { ViewportScreenshotHelper } from '../../mol-plugin/util/viewport-screenshot';
- import { shallowEqual } from '../../mol-util/object';
- import { useBehavior } from '../hooks/use-behavior';
- export interface ScreenshotPreviewProps {
- plugin: PluginContext,
- suspend?: boolean,
- cropFrameColor?: string,
- borderColor?: string,
- borderWidth?: number,
- customBackground?: string
- }
- const _ScreenshotPreview = (props: ScreenshotPreviewProps) => {
- const { plugin, cropFrameColor } = props;
- const helper = plugin.helpers.viewportScreenshot;
- if (!helper) return null;
- const [currentCanvas, setCurrentCanvas] = useState<HTMLCanvasElement | null>(null);
- const canvasRef = useRef<HTMLCanvasElement | null>(null);
- const propsRef = useRef(props);
- useEffect(() => {
- propsRef.current = props;
- }, Object.values(props));
- useEffect(() => {
- if (currentCanvas !== canvasRef.current) {
- setCurrentCanvas(canvasRef.current);
- }
- });
- useEffect(() => {
- let isDirty = false;
- const subs: Subscription[] = [];
- function subscribe<T>(xs: Observable<T> | undefined, f: (v: T) => any) {
- if (!xs) return;
- subs.push(xs.subscribe(f));
- }
- function preview() {
- const p = propsRef.current;
- if (!p.suspend && canvasRef.current) {
- drawPreview(helper!, canvasRef.current, p.customBackground, p.borderColor, p.borderWidth);
- }
- if (!canvasRef.current) isDirty = true;
- }
- const interval = setInterval(() => {
- if (isDirty) {
- isDirty = false;
- preview();
- }
- }, 1000 / 8);
- subscribe(plugin.events.canvas3d.settingsUpdated, () => isDirty = true);
- subscribe(plugin.canvas3d?.didDraw, () => isDirty = true);
- subscribe(plugin.state.data.behaviors.isUpdating, v => {
- if (!v) isDirty = true;
- });
- subscribe(helper.behaviors.values, () => isDirty = true);
- subscribe(helper.behaviors.cropParams, () => isDirty = true);
- let resizeObserver: any = void 0;
- if (typeof ResizeObserver !== 'undefined') {
- resizeObserver = new ResizeObserver(() => isDirty = true);
- }
- const canvas = canvasRef.current;
- resizeObserver?.observe(canvas);
- preview();
- return () => {
- clearInterval(interval);
- subs.forEach(s => s.unsubscribe());
- resizeObserver?.unobserve(canvas);
- };
- }, [helper]);
- useLayoutEffect(() => {
- if (canvasRef.current) {
- drawPreview(helper, canvasRef.current, props.customBackground, props.borderColor, props.borderWidth);
- }
- }, [...Object.values(props)]);
- return <>
- <div style={{ position: 'relative', width: '100%', height: '100%' }}>
- <canvas ref={canvasRef} onContextMenu={e => { e.preventDefault(); e.stopPropagation(); }} style={{ display: 'block', width: '100%', height: '100%' }}></canvas>
- <ViewportFrame plugin={plugin} canvas={currentCanvas} color={cropFrameColor} />
- </div>
- </>;
- };
- export const ScreenshotPreview = React.memo(_ScreenshotPreview, (prev, next) => shallowEqual(prev, next));
- declare const ResizeObserver: any;
- function drawPreview(helper: ViewportScreenshotHelper, target: HTMLCanvasElement, customBackground?: string, borderColor?: string, borderWidth?: number) {
- const { canvas, width, height } = helper.getPreview()!;
- const ctx = target.getContext('2d');
- if (!ctx) return;
- const w = target.clientWidth;
- const h = target.clientHeight;
- target.width = w;
- target.height = h;
- ctx.clearRect(0, 0, w, h);
- const frame = getViewportFrame(width, height, w, h);
- if (customBackground) {
- ctx.fillStyle = customBackground;
- ctx.fillRect(frame.x, frame.y, frame.width, frame.height);
- } else if (helper.values.transparent) {
- // must be an odd number
- const s = 13;
- for (let i = 0; i < frame.width; i += s) {
- for (let j = 0; j < frame.height; j += s) {
- ctx.fillStyle = (i + j) % 2 ? '#ffffff' : '#bfbfbf';
- const x = frame.x + i, y = frame.y + j;
- const w = i + s > frame.width ? frame.width - i : s;
- const h = j + s > frame.height ? frame.height - j : s;
- ctx.fillRect(x, y, w, h);
- }
- }
- }
- ctx.drawImage(canvas, frame.x, frame.y, frame.width, frame.height);
- if (borderColor && borderWidth) {
- const w = borderWidth;
- ctx.rect(frame.x, frame.y, frame.width, frame.height);
- ctx.rect(frame.x + w, frame.y + w, frame.width - 2 * w, frame.height - 2 * w);
- ctx.fillStyle = borderColor;
- ctx.fill('evenodd');
- }
- }
- function ViewportFrame({ plugin, canvas, color = 'rgba(255, 87, 45, 0.75)' }: { plugin: PluginContext, canvas: HTMLCanvasElement | null, color?: string }) {
- const helper = plugin.helpers.viewportScreenshot;
- if (!helper || !canvas) return null;
- const params = useBehavior(helper.behaviors.values);
- const cropParams = useBehavior(helper.behaviors.cropParams);
- const crop = useBehavior(helper.behaviors.relativeCrop!);
- const cropFrameRef = useRef<Viewport>({ x: 0, y: 0, width: 0, height: 0 });
- useBehavior(params?.resolution.name === 'viewport' ? plugin.canvas3d?.resized : void 0);
- const [drag, setDrag] = React.useState<string>('');
- const [start, setStart] = useState([0, 0]);
- const [current, setCurrent] = useState([0, 0]);
- const { width, height } = helper.getSizeAndViewport();
- const frame = getViewportFrame(width, height, canvas.clientWidth, canvas.clientHeight);
- const cropFrame: Viewport = {
- x: frame.x + Math.floor(frame.width * crop.x),
- y: frame.y + Math.floor(frame.height * crop.y),
- width: Math.ceil(frame.width * crop.width),
- height: Math.ceil(frame.height * crop.height)
- };
- const rectCrop = toRect(cropFrame);
- const rectFrame = toRect(frame);
- if (drag === 'move') {
- rectCrop.l += current[0] - start[0];
- rectCrop.r += current[0] - start[0];
- rectCrop.t += current[1] - start[1];
- rectCrop.b += current[1] - start[1];
- } else if (drag) {
- if (drag.indexOf('left') >= 0) {
- rectCrop.l += current[0] - start[0];
- } else if (drag.indexOf('right') >= 0) {
- rectCrop.r += current[0] - start[0];
- }
- if (drag.indexOf('top') >= 0) {
- rectCrop.t += current[1] - start[1];
- } else if (drag.indexOf('bottom') >= 0) {
- rectCrop.b += current[1] - start[1];
- }
- }
- if (rectCrop.l > rectCrop.r) {
- const t = rectCrop.l;
- rectCrop.l = rectCrop.r;
- rectCrop.r = t;
- }
- if (rectCrop.t > rectCrop.b) {
- const t = rectCrop.t;
- rectCrop.t = rectCrop.b;
- rectCrop.b = t;
- }
- const pad = 40;
- rectCrop.l = Math.min(rectFrame.r - pad, Math.max(rectFrame.l, rectCrop.l));
- rectCrop.r = Math.max(rectFrame.l + pad, Math.min(rectFrame.r, rectCrop.r));
- rectCrop.t = Math.min(rectFrame.b - pad, Math.max(rectFrame.t, rectCrop.t));
- rectCrop.b = Math.max(rectFrame.t + pad, Math.min(rectFrame.b, rectCrop.b));
- cropFrame.x = rectCrop.l;
- cropFrame.y = rectCrop.t;
- cropFrame.width = rectCrop.r - rectCrop.l + 1;
- cropFrame.height = rectCrop.b - rectCrop.t + 1;
- cropFrameRef.current = cropFrame;
- const onMove = (e: MouseEvent) => {
- e.preventDefault();
- setCurrent([e.pageX, e.pageY]);
- };
- const onTouchMove = (e: TouchEvent) => {
- e.preventDefault();
- const t = e.touches[0];
- setCurrent([t.pageX, t.pageY]);
- };
- const onTouchStart = (e: React.TouchEvent) => {
- e.preventDefault();
- setDrag(e.currentTarget.getAttribute('data-drag')! as any);
- const t = e.touches[0];
- const p = [t.pageX, t.pageY];
- setStart(p);
- setCurrent(p);
- window.addEventListener('touchend', onTouchEnd);
- window.addEventListener('touchmove', onTouchMove);
- };
- const onStart = (e: React.MouseEvent<HTMLElement>) => {
- e.preventDefault();
- setDrag(e.currentTarget.getAttribute('data-drag')! as any);
- const p = [e.pageX, e.pageY];
- setStart(p);
- setCurrent(p);
- window.addEventListener('mouseup', onEnd);
- window.addEventListener('mousemove', onMove);
- };
- const onEnd = () => {
- window.removeEventListener('mouseup', onEnd);
- window.removeEventListener('mousemove', onMove);
- finish();
- };
- const onTouchEnd = () => {
- window.removeEventListener('touchend', onTouchEnd);
- window.removeEventListener('touchmove', onTouchMove);
- finish();
- };
- function finish() {
- const cropFrame = cropFrameRef.current;
- if (cropParams.auto) {
- helper?.behaviors.cropParams.next({ ...cropParams, auto: false });
- }
- helper?.behaviors.relativeCrop.next({
- x: (cropFrame.x - frame.x) / frame.width,
- y: (cropFrame.y - frame.y) / frame.height,
- width: cropFrame.width / frame.width,
- height: cropFrame.height / frame.height
- });
- setDrag('');
- const p = [0, 0];
- setStart(p);
- setCurrent(p);
- }
- const contextMenu = (e: React.MouseEvent) => {
- e.preventDefault();
- e.stopPropagation();
- };
- const d = 4;
- const border = `3px solid ${color}`;
- const transparent = 'transparent';
- return <>
- <div data-drag='move' style={{ position: 'absolute', left: cropFrame.x, top: cropFrame.y, width: cropFrame.width, height: cropFrame.height, border, cursor: 'move' }} onMouseDown={onStart} onTouchStart={onTouchStart} draggable={false} onContextMenu={contextMenu} />
- <div data-drag='left' style={{ position: 'absolute', left: cropFrame.x - d, top: cropFrame.y + d, width: 4 * d, height: cropFrame.height - d, background: transparent, cursor: 'w-resize' }} onMouseDown={onStart} onTouchStart={onTouchStart} draggable={false} onContextMenu={contextMenu} />
- <div data-drag='right' style={{ position: 'absolute', left: rectCrop.r - 2 * d, top: cropFrame.y, width: 4 * d, height: cropFrame.height - d, background: transparent, cursor: 'w-resize' }} onMouseDown={onStart} onTouchStart={onTouchStart} draggable={false} onContextMenu={contextMenu} />
- <div data-drag='top' style={{ position: 'absolute', left: cropFrame.x - d, top: cropFrame.y - d, width: cropFrame.width + 2 * d, height: 4 * d, background: transparent, cursor: 'n-resize' }} onMouseDown={onStart} onTouchStart={onTouchStart} draggable={false} onContextMenu={contextMenu} />
- <div data-drag='bottom' style={{ position: 'absolute', left: cropFrame.x - d, top: rectCrop.b - 2 * d, width: cropFrame.width + 2 * d, height: 4 * d, background: transparent, cursor: 'n-resize' }} onMouseDown={onStart} onTouchStart={onTouchStart} draggable={false} onContextMenu={contextMenu} />
- <div data-drag='top, left' style={{ position: 'absolute', left: rectCrop.l - d, top: rectCrop.t - d, width: 4 * d, height: 4 * d, background: transparent, cursor: 'nw-resize' }} onMouseDown={onStart} onTouchStart={onTouchStart} draggable={false} onContextMenu={contextMenu} />
- <div data-drag='bottom, right' style={{ position: 'absolute', left: rectCrop.r - 2 * d, top: rectCrop.b - 2 * d, width: 4 * d, height: 4 * d, background: transparent, cursor: 'nw-resize' }} onMouseDown={onStart} onTouchStart={onTouchStart} draggable={false} onContextMenu={contextMenu} />
- <div data-drag='top, right' style={{ position: 'absolute', left: rectCrop.r - 2 * d, top: rectCrop.t - d, width: 4 * d, height: 4 * d, background: transparent, cursor: 'ne-resize' }} onMouseDown={onStart} onTouchStart={onTouchStart} draggable={false} onContextMenu={contextMenu} />
- <div data-drag='bottom, left' style={{ position: 'absolute', left: rectCrop.l - d, top: rectCrop.b - 2 * d, width: 4 * d, height: 4 * d, background: transparent, cursor: 'ne-resize' }} onMouseDown={onStart} onTouchStart={onTouchStart} draggable={false} onContextMenu={contextMenu} />
- </>;
- }
- function toRect(viewport: Viewport) {
- return { l: viewport.x, t: viewport.y, r: viewport.x + viewport.width - 1, b: viewport.y + viewport.height - 1 };
- }
- function getViewportFrame(srcWidth: number, srcHeight: number, w: number, h: number): Viewport {
- const a0 = srcWidth / srcHeight;
- const a1 = w / h;
- if (a0 <= a1) {
- const t = h * a0;
- return { x: Math.round((w - t) / 2), y: 0, width: Math.round(t), height: h };
- } else {
- const t = w / a0;
- return { x: 0, y: Math.round((h - t) / 2), width: w, height: Math.round(t) };
- }
- }
|