mirror of
https://github.com/GoogleChromeLabs/squoosh.git
synced 2025-11-16 18:49:50 +00:00
wip
# Conflicts: # codecs/cpp.Dockerfile # codecs/imagequant/example.html # codecs/webp/dec/webp_dec.d.ts # codecs/webp/dec/webp_dec.js # codecs/webp/dec/webp_dec.wasm # codecs/webp/enc/webp_enc.d.ts # codecs/webp/enc/webp_enc.js # codecs/webp/enc/webp_enc.wasm # package-lock.json # package.json # src/codecs/tiny.webp # src_old/codecs/encoders.ts # src_old/codecs/processor-worker/tiny.avif # src_old/codecs/processor-worker/tiny.webp # src_old/codecs/tiny.webp # src_old/components/compress/index.tsx # src_old/lib/util.ts # src_old/sw/util.ts
This commit is contained in:
171
src_old/components/Output/custom-els/TwoUp/index.ts
Normal file
171
src_old/components/Output/custom-els/TwoUp/index.ts
Normal file
@@ -0,0 +1,171 @@
|
||||
import PointerTracker, { Pointer } from 'pointer-tracker';
|
||||
import * as styles from './styles.css';
|
||||
|
||||
const legacyClipCompatAttr = 'legacy-clip-compat';
|
||||
const orientationAttr = 'orientation';
|
||||
|
||||
type TwoUpOrientation = 'horizontal' | 'vertical';
|
||||
|
||||
/**
|
||||
* A split view that the user can adjust. The first child becomes
|
||||
* the left-hand side, and the second child becomes the right-hand side.
|
||||
*/
|
||||
export default class TwoUp extends HTMLElement {
|
||||
static get observedAttributes() {
|
||||
return [orientationAttr];
|
||||
}
|
||||
|
||||
private readonly _handle = document.createElement('div');
|
||||
/**
|
||||
* The position of the split in pixels.
|
||||
*/
|
||||
private _position = 0;
|
||||
/**
|
||||
* The position of the split in %.
|
||||
*/
|
||||
private _relativePosition = 0.5;
|
||||
/**
|
||||
* The value of _position when the pointer went down.
|
||||
*/
|
||||
private _positionOnPointerStart = 0;
|
||||
/**
|
||||
* Has connectedCallback been called yet?
|
||||
*/
|
||||
private _everConnected = false;
|
||||
|
||||
constructor() {
|
||||
super();
|
||||
this._handle.className = styles.twoUpHandle;
|
||||
|
||||
// Watch for children changes.
|
||||
// Note this won't fire for initial contents,
|
||||
// so _childrenChange is also called in connectedCallback.
|
||||
new MutationObserver(() => this._childrenChange()).observe(this, {
|
||||
childList: true,
|
||||
});
|
||||
|
||||
// Watch for element size changes.
|
||||
if ('ResizeObserver' in window) {
|
||||
new ResizeObserver(() => this._resetPosition()).observe(this);
|
||||
} else {
|
||||
window.addEventListener('resize', () => this._resetPosition());
|
||||
}
|
||||
|
||||
// Watch for pointers on the handle.
|
||||
const pointerTracker: PointerTracker = new PointerTracker(this._handle, {
|
||||
start: (_, event) => {
|
||||
// We only want to track 1 pointer.
|
||||
if (pointerTracker.currentPointers.length === 1) return false;
|
||||
event.preventDefault();
|
||||
this._positionOnPointerStart = this._position;
|
||||
return true;
|
||||
},
|
||||
move: () => {
|
||||
this._pointerChange(
|
||||
pointerTracker.startPointers[0],
|
||||
pointerTracker.currentPointers[0],
|
||||
);
|
||||
},
|
||||
});
|
||||
}
|
||||
|
||||
connectedCallback() {
|
||||
this._childrenChange();
|
||||
|
||||
this._handle.innerHTML = `<div class="${
|
||||
styles.scrubber
|
||||
}">${`<svg viewBox="0 0 27 20" fill="currentColor">${'<path d="M17 19.2l9.5-9.6L16.9 0zM9.6 0L0 9.6l9.6 9.6z"/>'}</svg>`}</div>`;
|
||||
|
||||
if (!this._everConnected) {
|
||||
this._resetPosition();
|
||||
this._everConnected = true;
|
||||
}
|
||||
}
|
||||
|
||||
attributeChangedCallback(name: string) {
|
||||
if (name === orientationAttr) {
|
||||
this._resetPosition();
|
||||
}
|
||||
}
|
||||
|
||||
private _resetPosition() {
|
||||
// Set the initial position of the handle.
|
||||
requestAnimationFrame(() => {
|
||||
const bounds = this.getBoundingClientRect();
|
||||
const dimensionAxis =
|
||||
this.orientation === 'vertical' ? 'height' : 'width';
|
||||
this._position = bounds[dimensionAxis] * this._relativePosition;
|
||||
this._setPosition();
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* If true, this element works in browsers that don't support clip-path (Edge).
|
||||
* However, this means you'll have to set the height of this element manually.
|
||||
*/
|
||||
get legacyClipCompat() {
|
||||
return this.hasAttribute(legacyClipCompatAttr);
|
||||
}
|
||||
|
||||
set legacyClipCompat(val: boolean) {
|
||||
if (val) {
|
||||
this.setAttribute(legacyClipCompatAttr, '');
|
||||
} else {
|
||||
this.removeAttribute(legacyClipCompatAttr);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Split vertically rather than horizontally.
|
||||
*/
|
||||
get orientation(): TwoUpOrientation {
|
||||
const value = this.getAttribute(orientationAttr);
|
||||
|
||||
// This mirrors the behaviour of input.type, where setting just sets the attribute, but getting
|
||||
// returns the value only if it's valid.
|
||||
if (value && value.toLowerCase() === 'vertical') return 'vertical';
|
||||
return 'horizontal';
|
||||
}
|
||||
|
||||
set orientation(val: TwoUpOrientation) {
|
||||
this.setAttribute(orientationAttr, val);
|
||||
}
|
||||
|
||||
/**
|
||||
* Called when element's child list changes
|
||||
*/
|
||||
private _childrenChange() {
|
||||
// Ensure the handle is the last child.
|
||||
// The CSS depends on this.
|
||||
if (this.lastElementChild !== this._handle) {
|
||||
this.appendChild(this._handle);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Called when a pointer moves.
|
||||
*/
|
||||
private _pointerChange(startPoint: Pointer, currentPoint: Pointer) {
|
||||
const pointAxis = this.orientation === 'vertical' ? 'clientY' : 'clientX';
|
||||
const dimensionAxis = this.orientation === 'vertical' ? 'height' : 'width';
|
||||
const bounds = this.getBoundingClientRect();
|
||||
|
||||
this._position =
|
||||
this._positionOnPointerStart +
|
||||
(currentPoint[pointAxis] - startPoint[pointAxis]);
|
||||
|
||||
// Clamp position to element bounds.
|
||||
this._position = Math.max(
|
||||
0,
|
||||
Math.min(this._position, bounds[dimensionAxis]),
|
||||
);
|
||||
this._relativePosition = this._position / bounds[dimensionAxis];
|
||||
this._setPosition();
|
||||
}
|
||||
|
||||
private _setPosition() {
|
||||
this.style.setProperty('--split-point', `${this._position}px`);
|
||||
}
|
||||
}
|
||||
|
||||
customElements.define('two-up', TwoUp);
|
||||
52
src_old/components/Output/custom-els/TwoUp/missing-types.d.ts
vendored
Normal file
52
src_old/components/Output/custom-els/TwoUp/missing-types.d.ts
vendored
Normal file
@@ -0,0 +1,52 @@
|
||||
declare interface CSSStyleDeclaration {
|
||||
willChange: string | null;
|
||||
}
|
||||
|
||||
// TypeScript, you make me sad.
|
||||
// https://github.com/Microsoft/TypeScript/issues/18756
|
||||
interface Window {
|
||||
PointerEvent: typeof PointerEvent;
|
||||
Touch: typeof Touch;
|
||||
}
|
||||
|
||||
interface TwoUpAttributes extends JSX.HTMLAttributes {
|
||||
orientation?: string;
|
||||
'legacy-clip-compat'?: boolean;
|
||||
}
|
||||
|
||||
declare namespace JSX {
|
||||
interface IntrinsicElements {
|
||||
'two-up': TwoUpAttributes;
|
||||
}
|
||||
}
|
||||
|
||||
interface DOMRectReadOnly {
|
||||
readonly x: number;
|
||||
readonly y: number;
|
||||
readonly width: number;
|
||||
readonly height: number;
|
||||
readonly top: number;
|
||||
readonly right: number;
|
||||
readonly bottom: number;
|
||||
readonly left: number;
|
||||
}
|
||||
|
||||
interface ResizeObserverCallback {
|
||||
(entries: ResizeObserverEntry[], observer: ResizeObserver): void;
|
||||
}
|
||||
|
||||
interface ResizeObserverEntry {
|
||||
readonly target: Element;
|
||||
readonly contentRect: DOMRectReadOnly;
|
||||
}
|
||||
|
||||
interface ResizeObserver {
|
||||
observe(target: Element): void;
|
||||
unobserve(target: Element): void;
|
||||
disconnect(): void;
|
||||
}
|
||||
|
||||
declare var ResizeObserver: {
|
||||
prototype: ResizeObserver;
|
||||
new (callback: ResizeObserverCallback): ResizeObserver;
|
||||
};
|
||||
131
src_old/components/Output/custom-els/TwoUp/styles.css
Normal file
131
src_old/components/Output/custom-els/TwoUp/styles.css
Normal file
@@ -0,0 +1,131 @@
|
||||
two-up {
|
||||
display: grid;
|
||||
position: relative;
|
||||
--split-point: 0;
|
||||
--accent-color: #777;
|
||||
--track-color: var(--accent-color);
|
||||
--thumb-background: #fff;
|
||||
--thumb-color: var(--accent-color);
|
||||
--thumb-size: 62px;
|
||||
--bar-size: 6px;
|
||||
--bar-touch-size: 30px;
|
||||
}
|
||||
|
||||
two-up > * {
|
||||
/* Overlay all children on top of each other, and let two-up's layout contain all of them. */
|
||||
grid-area: 1/1;
|
||||
}
|
||||
|
||||
two-up[legacy-clip-compat] > :not(.two-up-handle) {
|
||||
/* Legacy mode uses clip rather than clip-path (Edge doesn't support clip-path), but clip requires
|
||||
elements to be positioned absolutely */
|
||||
position: absolute;
|
||||
}
|
||||
|
||||
.two-up-handle {
|
||||
touch-action: none;
|
||||
position: relative;
|
||||
width: var(--bar-touch-size);
|
||||
transform: translateX(var(--split-point)) translateX(-50%);
|
||||
will-change: transform;
|
||||
cursor: ew-resize;
|
||||
}
|
||||
|
||||
.two-up-handle::before {
|
||||
content: '';
|
||||
display: block;
|
||||
height: 100%;
|
||||
width: var(--bar-size);
|
||||
margin: 0 auto;
|
||||
box-shadow: inset calc(var(--bar-size) / 2) 0 0 rgba(0, 0, 0, 0.1),
|
||||
0 1px 4px rgba(0, 0, 0, 0.4);
|
||||
background: var(--track-color);
|
||||
}
|
||||
|
||||
.scrubber {
|
||||
display: flex;
|
||||
position: absolute;
|
||||
top: 50%;
|
||||
left: 50%;
|
||||
transform-origin: 50% 50%;
|
||||
transform: translate(-50%, -50%);
|
||||
width: var(--thumb-size);
|
||||
height: calc(var(--thumb-size) * 0.9);
|
||||
background: var(--thumb-background);
|
||||
border: 1px solid rgba(0, 0, 0, 0.2);
|
||||
border-radius: var(--thumb-size);
|
||||
box-shadow: 0 1px 4px rgba(0, 0, 0, 0.1);
|
||||
color: var(--thumb-color);
|
||||
box-sizing: border-box;
|
||||
padding: 0 calc(var(--thumb-size) * 0.24);
|
||||
}
|
||||
|
||||
.scrubber svg {
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
two-up[orientation='vertical'] .two-up-handle {
|
||||
width: auto;
|
||||
height: var(--bar-touch-size);
|
||||
transform: translateY(var(--split-point)) translateY(-50%);
|
||||
cursor: ns-resize;
|
||||
}
|
||||
|
||||
two-up[orientation='vertical'] .two-up-handle::before {
|
||||
width: auto;
|
||||
height: var(--bar-size);
|
||||
box-shadow: inset 0 calc(var(--bar-size) / 2) 0 rgba(0, 0, 0, 0.1),
|
||||
0 1px 4px rgba(0, 0, 0, 0.4);
|
||||
margin: calc((var(--bar-touch-size) - var(--bar-size)) / 2) 0 0 0;
|
||||
}
|
||||
|
||||
two-up[orientation='vertical'] .scrubber {
|
||||
box-shadow: 1px 0 4px rgba(0, 0, 0, 0.1);
|
||||
transform: translate(-50%, -50%) rotate(-90deg);
|
||||
}
|
||||
|
||||
two-up > :nth-child(1):not(.two-up-handle) {
|
||||
-webkit-clip-path: inset(0 calc(100% - var(--split-point)) 0 0);
|
||||
clip-path: inset(0 calc(100% - var(--split-point)) 0 0);
|
||||
}
|
||||
|
||||
two-up > :nth-child(2):not(.two-up-handle) {
|
||||
-webkit-clip-path: inset(0 0 0 var(--split-point));
|
||||
clip-path: inset(0 0 0 var(--split-point));
|
||||
}
|
||||
|
||||
two-up[orientation='vertical'] > :nth-child(1):not(.two-up-handle) {
|
||||
-webkit-clip-path: inset(0 0 calc(100% - var(--split-point)) 0);
|
||||
clip-path: inset(0 0 calc(100% - var(--split-point)) 0);
|
||||
}
|
||||
|
||||
two-up[orientation='vertical'] > :nth-child(2):not(.two-up-handle) {
|
||||
-webkit-clip-path: inset(var(--split-point) 0 0 0);
|
||||
clip-path: inset(var(--split-point) 0 0 0);
|
||||
}
|
||||
|
||||
/*
|
||||
Even in legacy-clip-compat, prefer clip-path if it's supported.
|
||||
It performs way better in Safari.
|
||||
*/
|
||||
@supports not (
|
||||
(clip-path: inset(0 0 0 0)) or (-webkit-clip-path: inset(0 0 0 0))
|
||||
) {
|
||||
two-up[legacy-clip-compat] > :nth-child(1):not(.two-up-handle) {
|
||||
clip: rect(auto var(--split-point) auto auto);
|
||||
}
|
||||
|
||||
two-up[legacy-clip-compat] > :nth-child(2):not(.two-up-handle) {
|
||||
clip: rect(auto auto auto var(--split-point));
|
||||
}
|
||||
|
||||
two-up[orientation='vertical'][legacy-clip-compat]
|
||||
> :nth-child(1):not(.two-up-handle) {
|
||||
clip: rect(auto auto var(--split-point) auto);
|
||||
}
|
||||
|
||||
two-up[orientation='vertical'][legacy-clip-compat]
|
||||
> :nth-child(2):not(.two-up-handle) {
|
||||
clip: rect(var(--split-point) auto auto auto);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user