Compare commits

...

15 Commits

Author SHA1 Message Date
Jake Archibald
b139119551 Prevent options overflow at larger widths 2018-11-06 13:37:13 +00:00
Jake Archibald
85323dff87 No longer need this. 2018-11-06 13:37:13 +00:00
Jake Archibald
849441f23a Range bubble now behaves properly on mobile 2018-11-06 13:37:12 +00:00
Jake Archibald
c125af564a Allow two-up and pinch-zoom to work beneath controls 2018-11-06 13:37:12 +00:00
Jake Archibald
ce67f6c538 Expand/collapse icon 2018-11-06 13:37:11 +00:00
Jake Archibald
c8e0c56687 Fixing animation bugs 2018-11-06 13:37:11 +00:00
Jake Archibald
ac4f845d8e Adding height animation to multi-panel 2018-11-06 13:37:11 +00:00
Jake Archibald
52f61dfccc Adding labels to collapsed view 2018-11-06 13:37:10 +00:00
Jake Archibald
068dfe1b19 Ordering of items in mobile view. Changing scrolling element. 2018-11-06 13:37:10 +00:00
Jake Archibald
637e859a1e Abstracting results so it can be used as a heading. 2018-11-06 13:37:09 +00:00
Jake Archibald
da072a015b Edge cases for one-open 2018-11-06 13:37:08 +00:00
Jake Archibald
04492f8f5e Allow multi-panel to keep one open only 2018-11-06 13:37:08 +00:00
Jake Archibald
b34dca744d Adding margin so you can still access the two-up 2018-11-06 13:37:08 +00:00
Jake Archibald
c3edde280a Fixing thumb on two-up 2018-11-06 13:37:07 +00:00
Jake Archibald
8db8892529 Basic grid setup 2018-11-06 13:37:07 +00:00
20 changed files with 555 additions and 322 deletions

View File

@@ -1,7 +1,7 @@
import { h, Component } from 'preact'; import { h, Component } from 'preact';
import * as style from './style.scss'; import * as style from './style.scss';
import { bind, Fileish } from '../../lib/initial-util'; import { bind } from '../../lib/initial-util';
import { cleanSet, cleanMerge } from '../../lib/clean-modify'; import { cleanSet, cleanMerge } from '../../lib/clean-modify';
import OptiPNGEncoderOptions from '../../codecs/optipng/options'; import OptiPNGEncoderOptions from '../../codecs/optipng/options';
import MozJpegEncoderOptions from '../../codecs/mozjpeg/options'; import MozJpegEncoderOptions from '../../codecs/mozjpeg/options';
@@ -35,13 +35,10 @@ import {
import { QuantizeOptions } from '../../codecs/imagequant/processor-meta'; import { QuantizeOptions } from '../../codecs/imagequant/processor-meta';
import { ResizeOptions } from '../../codecs/resize/processor-meta'; import { ResizeOptions } from '../../codecs/resize/processor-meta';
import { PreprocessorState } from '../../codecs/preprocessors'; import { PreprocessorState } from '../../codecs/preprocessors';
import FileSize from './FileSize';
import { DownloadIcon } from '../../lib/icons';
import { SourceImage } from '../App'; import { SourceImage } from '../App';
import Checkbox from '../checkbox'; import Checkbox from '../checkbox';
import Expander from '../expander'; import Expander from '../expander';
import Select from '../select'; import Select from '../select';
import '../custom-els/LoadingSpinner';
const encoderOptionsComponentMap = { const encoderOptionsComponentMap = {
[identity.type]: undefined, [identity.type]: undefined,
@@ -60,12 +57,9 @@ const encoderOptionsComponentMap = {
}; };
interface Props { interface Props {
orientation: 'horizontal' | 'vertical'; mobileView: boolean;
loading: boolean;
source?: SourceImage; source?: SourceImage;
imageIndex: number; imageIndex: number;
imageFile?: Fileish;
downloadUrl?: string;
encoderState: EncoderState; encoderState: EncoderState;
preprocessorState: PreprocessorState; preprocessorState: PreprocessorState;
onEncoderTypeChange(newType: EncoderType): void; onEncoderTypeChange(newType: EncoderType): void;
@@ -76,39 +70,18 @@ interface Props {
interface State { interface State {
encoderSupportMap?: EncoderSupportMap; encoderSupportMap?: EncoderSupportMap;
showLoadingState: boolean;
} }
const loadingReactionDelay = 500;
export default class Options extends Component<Props, State> { export default class Options extends Component<Props, State> {
state: State = { state: State = {
encoderSupportMap: undefined, encoderSupportMap: undefined,
showLoadingState: false,
}; };
/** The timeout ID between entering the loading state, and changing UI */
private loadingTimeoutId: number = 0;
constructor() { constructor() {
super(); super();
encodersSupported.then(encoderSupportMap => this.setState({ encoderSupportMap })); encodersSupported.then(encoderSupportMap => this.setState({ encoderSupportMap }));
} }
componentDidUpdate(prevProps: Props, prevState: State) {
if (prevProps.loading && !this.props.loading) {
// Just stopped loading
clearTimeout(this.loadingTimeoutId);
this.setState({ showLoadingState: false });
} else if (!prevProps.loading && this.props.loading) {
// Just started loading
this.loadingTimeoutId = self.setTimeout(
() => this.setState({ showLoadingState: true }),
loadingReactionDelay,
);
}
}
@bind @bind
onEncoderTypeChange(event: Event) { onEncoderTypeChange(event: Event) {
const el = event.currentTarget as HTMLSelectElement; const el = event.currentTarget as HTMLSelectElement;
@@ -153,20 +126,17 @@ export default class Options extends Component<Props, State> {
{ {
source, source,
imageIndex, imageIndex,
imageFile,
downloadUrl,
orientation,
encoderState, encoderState,
preprocessorState, preprocessorState,
onEncoderOptionsChange, onEncoderOptionsChange,
}: Props, }: Props,
{ encoderSupportMap, showLoadingState }: State, { encoderSupportMap }: State,
) { ) {
// tslint:disable variable-name // tslint:disable variable-name
const EncoderOptionComponent = encoderOptionsComponentMap[encoderState.type]; const EncoderOptionComponent = encoderOptionsComponentMap[encoderState.type];
return ( return (
<div class={style.options}> <div class={style.optionsScroller}>
<Expander> <Expander>
{encoderState.type === identity.type ? null : {encoderState.type === identity.type ? null :
<div> <div>
@@ -211,32 +181,30 @@ export default class Options extends Component<Props, State> {
<h3 class={style.optionsTitle}>Compress</h3> <h3 class={style.optionsTitle}>Compress</h3>
<div class={style.optionsScroller}> <section class={`${style.optionOneCell} ${style.optionsSection}`}>
<section class={`${style.optionOneCell} ${style.optionsSection}`}> {encoderSupportMap ?
{encoderSupportMap ? <Select value={encoderState.type} onChange={this.onEncoderTypeChange} large>
<Select value={encoderState.type} onChange={this.onEncoderTypeChange} large> {encoders.filter(encoder => encoderSupportMap[encoder.type]).map(encoder => (
{encoders.filter(encoder => encoderSupportMap[encoder.type]).map(encoder => ( <option value={encoder.type}>{encoder.label}</option>
<option value={encoder.type}>{encoder.label}</option> ))}
))} </Select>
</Select> :
: <Select large><option>Loading</option></Select>
<Select large><option>Loading</option></Select> }
} </section>
</section>
<Expander> <Expander>
{EncoderOptionComponent ? {EncoderOptionComponent ?
<EncoderOptionComponent <EncoderOptionComponent
options={ options={
// Casting options, as encoderOptionsComponentMap[encodeData.type] ensures // Casting options, as encoderOptionsComponentMap[encodeData.type] ensures
// the correct type, but typescript isn't smart enough. // the correct type, but typescript isn't smart enough.
encoderState.options as any encoderState.options as any
} }
onChange={onEncoderOptionsChange} onChange={onEncoderOptionsChange}
/> />
: null} : null}
</Expander> </Expander>
</div>
<div class={style.optionsCopy}> <div class={style.optionsCopy}>
<button onClick={this.onCopyToOtherClick} class={style.copyButton}> <button onClick={this.onCopyToOtherClick} class={style.copyButton}>
@@ -245,33 +213,6 @@ export default class Options extends Component<Props, State> {
{imageIndex === 0 && ' →'} {imageIndex === 0 && ' →'}
</button> </button>
</div> </div>
<div class={style.results}>
<div class={style.resultData}>
{!imageFile || showLoadingState ? 'Working…' :
<FileSize
blob={imageFile}
compareTo={(source && imageFile !== source.file) ? source.file : undefined}
/>
}
</div>
<div class={style.download}>
{(downloadUrl && imageFile) && (
<a
class={`${style.downloadLink} ${showLoadingState ? style.downloadLinkDisable : ''}`}
href={downloadUrl}
download={imageFile.name}
title="Download"
>
<DownloadIcon class={style.downloadIcon} />
</a>
)}
{showLoadingState && <loading-spinner class={style.spinner} />}
</div>
</div>
</div> </div>
); );
} }

View File

@@ -1,15 +1,5 @@
$horizontalPadding: 15px; $horizontalPadding: 15px;
.options {
color: #fff;
width: 300px;
opacity: 0.9;
font-size: 1.2rem;
max-height: 100%;
display: flex;
flex-flow: column;
}
.options-title { .options-title {
background: rgba(0, 0, 0, 0.9); background: rgba(0, 0, 0, 0.9);
margin: 0; margin: 0;
@@ -66,35 +56,6 @@ $horizontalPadding: 15px;
overflow-y: auto; overflow-y: auto;
} }
.results {
display: grid;
grid-template-columns: 1fr auto;
background: rgba(0, 0, 0, 0.9);
font-size: 1.4rem;
}
.result-data {
display: flex;
align-items: center;
padding: 0 $horizontalPadding;
}
.size-delta {
font-size: 1.1rem;
font-style: italic;
position: relative;
top: -1px;
margin-left: 0.3em;
}
.size-increase {
color: #e35050;
}
.size-decrease {
color: #50e3c2;
}
.options-copy { .options-copy {
display: grid; display: grid;
background: rgba(0, 0, 0, 0.9); background: rgba(0, 0, 0, 0.9);
@@ -109,58 +70,3 @@ $horizontalPadding: 15px;
text-align: left; text-align: left;
padding: 5px 10px; padding: 5px 10px;
} }
@keyframes action-enter {
from {
transform: rotate(-90deg);
opacity: 0;
animation-timing-function: ease-out;
}
}
@keyframes action-leave {
from {
transform: rotate(0deg);
opacity: 1;
animation-timing-function: ease-out;
}
}
.download {
background: #34B9EB;
--size: 38px;
width: var(--size);
height: var(--size);
display: grid;
align-items: center;
justify-items: center;
}
.download-link {
animation: action-enter 0.2s;
grid-area: 1/1;
}
.download-link-disable {
pointer-events: none;
opacity: 0;
transform: rotate(90deg);
animation: action-leave 0.2s;
}
.download-icon {
color: #fff;
display: block;
--size: 24px;
width: var(--size);
height: var(--size);
padding: 7px;
filter: drop-shadow(0 1px 0 rgba(0, 0, 0, 0.7));
}
.spinner {
--color: #fff;
--delay: 0;
--size: 22px;
grid-area: 1/1;
}

View File

@@ -56,7 +56,7 @@ two-up[legacy-clip-compat] > :not(.two-up-handle) {
box-shadow: 0 1px 4px rgba(0,0,0,0.1); box-shadow: 0 1px 4px rgba(0,0,0,0.1);
color: var(--thumb-color); color: var(--thumb-color);
box-sizing: border-box; box-sizing: border-box;
padding: 0 48%; padding: 0 calc(var(--thumb-size) * 0.24);
} }
.scrubber svg { .scrubber svg {

View File

@@ -10,7 +10,7 @@ import { twoUpHandle } from './custom-els/TwoUp/styles.css';
interface Props { interface Props {
originalImage?: ImageData; originalImage?: ImageData;
orientation: 'horizontal' | 'vertical'; mobileView: boolean;
leftCompressed?: ImageData; leftCompressed?: ImageData;
rightCompressed?: ImageData; rightCompressed?: ImageData;
leftImgContain: boolean; leftImgContain: boolean;
@@ -180,10 +180,7 @@ export default class Output extends Component<Props, State> {
} }
render( render(
{ { mobileView, leftImgContain, rightImgContain, originalImage }: Props,
orientation, leftCompressed, rightCompressed, leftImgContain, rightImgContain,
originalImage,
}: Props,
{ scale, editingScale, altBackground }: State, { scale, editingScale, altBackground }: State,
) { ) {
const leftDraw = this.leftDrawable(); const leftDraw = this.leftDrawable();
@@ -194,7 +191,7 @@ export default class Output extends Component<Props, State> {
<two-up <two-up
legacy-clip-compat legacy-clip-compat
class={style.twoUp} class={style.twoUp}
orientation={orientation} orientation={mobileView ? 'vertical' : 'horizontal'}
// Event redirecting. See onRetargetableEvent. // Event redirecting. See onRetargetableEvent.
onTouchStartCapture={this.onRetargetableEvent} onTouchStartCapture={this.onRetargetableEvent}
onTouchEndCapture={this.onRetargetableEvent} onTouchEndCapture={this.onRetargetableEvent}

View File

@@ -1,7 +1,3 @@
/*
Note: These styles are temporary. They will be replaced before going live.
*/
.output { .output {
composes: abs-fill from '../../lib/util.scss'; composes: abs-fill from '../../lib/util.scss';
@@ -47,6 +43,12 @@ Note: These styles are temporary. They will be replaced before going live.
flex-wrap: wrap; flex-wrap: wrap;
contain: content; contain: content;
// Allow clicks to fall through to the pinch zoom area
pointer-events: none;
& > * {
pointer-events: auto;
}
@media (min-width: 860px) { @media (min-width: 860px) {
top: auto; top: auto;
left: 320px; left: 320px;

View File

@@ -1,19 +1,75 @@
import './styles.css'; import * as style from './styles.css';
import { transitionHeight } from '../../../../lib/util';
function getClosestHeading(el: Element) { interface CloseAllOptions {
const closestEl = el.closest('multi-panel > *'); exceptFirst?: boolean;
if (closestEl && closestEl.classList.contains('panel-heading')) { }
return closestEl;
const openOneOnlyAttr = 'open-one-only';
function getClosestHeading(el: Element): HTMLElement | undefined {
// Look for the child of multi-panel, but stop at interactive elements like links & buttons
const closestEl = el.closest('multi-panel > *, a, button');
if (closestEl && closestEl.classList.contains(style.panelHeading)) {
return closestEl as HTMLElement;
} }
return undefined; return undefined;
} }
async function close(heading: HTMLElement) {
const content = heading.nextElementSibling as HTMLElement;
// if there is no content, nothing to expand
if (!content) return;
const from = content.getBoundingClientRect().height;
heading.removeAttribute('content-expanded');
content.setAttribute('aria-expanded', 'false');
// Wait a microtask so other calls to open/close can get the final sizes.
await null;
await transitionHeight(content, {
from,
to: 0,
duration: 300,
});
content.style.height = '';
}
async function open(heading: HTMLElement) {
const content = heading.nextElementSibling as HTMLElement;
// if there is no content, nothing to expand
if (!content) return;
const from = content.getBoundingClientRect().height;
heading.setAttribute('content-expanded', '');
content.setAttribute('aria-expanded', 'true');
const to = content.getBoundingClientRect().height;
// Wait a microtask so other calls to open/close can get the final sizes.
await null;
await transitionHeight(content, {
from, to,
duration: 300,
});
content.style.height = '';
}
/** /**
* A multi-panel view that the user can add any number of 'panels'. * A multi-panel view that the user can add any number of 'panels'.
* 'a panel' consists of two elements. Even index element becomes heading, * 'a panel' consists of two elements. Even index element becomes heading,
* and odd index element becomes the expandable content. * and odd index element becomes the expandable content.
*/ */
export default class MultiPanel extends HTMLElement { export default class MultiPanel extends HTMLElement {
static get observedAttributes() { return [openOneOnlyAttr]; }
constructor() { constructor() {
super(); super();
@@ -31,12 +87,18 @@ export default class MultiPanel extends HTMLElement {
this._childrenChange(); this._childrenChange();
} }
attributeChangedCallback(name: string, oldValue: string | null, newValue: string | null) {
if (name === openOneOnlyAttr && newValue === null) {
this._closeAll({ exceptFirst: true });
}
}
// Click event handler // Click event handler
private _onClick(event: MouseEvent) { private _onClick(event: MouseEvent) {
const el = event.target as Element; const el = event.target as HTMLElement;
const heading = getClosestHeading(el); const heading = getClosestHeading(el);
if (!heading) return; if (!heading) return;
this._expand(heading); this._toggle(heading);
} }
// KeyDown event handler // KeyDown event handler
@@ -53,7 +115,8 @@ export default class MultiPanel extends HTMLElement {
// dont handle modifier shortcuts used by assistive technology. // dont handle modifier shortcuts used by assistive technology.
if (event.altKey) return; if (event.altKey) return;
let newHeading:HTMLElement | undefined; let newHeading: HTMLElement | undefined;
switch (event.key) { switch (event.key) {
case 'ArrowLeft': case 'ArrowLeft':
case 'ArrowUp': case 'ArrowUp':
@@ -77,7 +140,7 @@ export default class MultiPanel extends HTMLElement {
case 'Enter': case 'Enter':
case ' ': case ' ':
case 'Spacebar': case 'Spacebar':
this._expand(heading); this._toggle(heading);
break; break;
// Any other key press is ignored and passed back to the browser. // Any other key press is ignored and passed back to the browser.
@@ -93,26 +156,32 @@ export default class MultiPanel extends HTMLElement {
} }
} }
private _expand(heading: Element) { private _toggle(heading: HTMLElement) {
if (!heading) return; if (!heading) return;
const content = heading.nextElementSibling;
// if there is no content, nothing to expand
if (!content) return;
// toggle expanded and aria-expanded attributes // toggle expanded and aria-expanded attributes
if (content.hasAttribute('expanded')) { if (heading.hasAttribute('content-expanded')) {
content.removeAttribute('expanded'); close(heading);
content.setAttribute('aria-expanded', 'false');
} else { } else {
content.setAttribute('expanded', ''); if (this.openOneOnly) this._closeAll();
content.setAttribute('aria-expanded', 'true'); open(heading);
} }
} }
private _closeAll(options: CloseAllOptions = {}): void {
const { exceptFirst = false } = options;
let els = [...this.children].filter(el => el.matches('[content-expanded]')) as HTMLElement[];
if (exceptFirst) {
els = els.slice(1);
}
for (const el of els) close(el);
}
// children of multi-panel should always be even number (heading/content pair) // children of multi-panel should always be even number (heading/content pair)
private _childrenChange() { private _childrenChange() {
let preserveTabIndex : boolean = false; let preserveTabIndex = false;
let heading = this.firstElementChild; let heading = this.firstElementChild;
while (heading) { while (heading) {
@@ -123,31 +192,23 @@ export default class MultiPanel extends HTMLElement {
// it means it has odd number of elements. log error and set heading to end the loop. // it means it has odd number of elements. log error and set heading to end the loop.
if (!content) { if (!content) {
console.error('<multi-panel> requires an even number of element children.'); console.error('<multi-panel> requires an even number of element children.');
heading = null; break;
continue;
} }
// When odd number of elements were inserted in the middle, // When odd number of elements were inserted in the middle,
// what was heading before may become content after the insertion. // what was heading before may become content after the insertion.
// Remove classes and attributes to prepare for this change. // Remove classes and attributes to prepare for this change.
heading.classList.remove('panel-content'); heading.classList.remove(style.panelContent);
content.classList.remove(style.panelHeading);
if (content.classList.contains('panel-heading')) { heading.removeAttribute('aria-expanded');
content.classList.remove('panel-heading'); heading.removeAttribute('content-expanded');
}
if (heading.hasAttribute('expanded') && heading.hasAttribute('aria-expanded')) {
heading.removeAttribute('expanded');
heading.removeAttribute('aria-expanded');
}
// If appreciable, remove tabindex from content which used to be header. // If appreciable, remove tabindex from content which used to be header.
if (content.hasAttribute('tabindex')) { content.removeAttribute('tabindex');
content.removeAttribute('tabindex');
}
// Assign heading and content classes // Assign heading and content classes
heading.classList.add('panel-heading'); heading.classList.add(style.panelHeading);
content.classList.add('panel-content'); content.classList.add(style.panelContent);
// Assign ids and aria-X for heading/content pair. // Assign ids and aria-X for heading/content pair.
heading.id = `panel-heading-${randomId}`; heading.id = `panel-heading-${randomId}`;
@@ -163,6 +224,13 @@ export default class MultiPanel extends HTMLElement {
heading.setAttribute('tabindex', '-1'); heading.setAttribute('tabindex', '-1');
} }
// It's possible that the heading & content expanded attributes are now out of sync. Resync
// them using the heading as the source of truth.
content.setAttribute(
'aria-expanded',
heading.hasAttribute('content-expanded') ? 'true' : 'false',
);
// next sibling of content = next heading // next sibling of content = next heading
heading = content.nextElementSibling; heading = content.nextElementSibling;
} }
@@ -171,6 +239,9 @@ export default class MultiPanel extends HTMLElement {
if (!preserveTabIndex && this.firstElementChild) { if (!preserveTabIndex && this.firstElementChild) {
this.firstElementChild.setAttribute('tabindex', '0'); this.firstElementChild.setAttribute('tabindex', '0');
} }
// In case we're openOneOnly, and an additional open item has been added:
if (this.openOneOnly) this._closeAll({ exceptFirst: true });
} }
// returns heading that is before currently selected one. // returns heading that is before currently selected one.
@@ -208,7 +279,7 @@ export default class MultiPanel extends HTMLElement {
private _lastHeading() { private _lastHeading() {
// if the last element is heading, return last element // if the last element is heading, return last element
const lastEl = this.lastElementChild as HTMLElement; const lastEl = this.lastElementChild as HTMLElement;
if (lastEl && lastEl.classList.contains('panel-heading')) { if (lastEl && lastEl.classList.contains(style.panelHeading)) {
return lastEl; return lastEl;
} }
// otherwise return 2nd from the last // otherwise return 2nd from the last
@@ -217,6 +288,21 @@ export default class MultiPanel extends HTMLElement {
return lastContent.previousElementSibling as HTMLElement; return lastContent.previousElementSibling as HTMLElement;
} }
} }
/**
* If true, only one panel can be open at once. When one opens, others close.
*/
get openOneOnly() {
return this.hasAttribute(openOneOnlyAttr);
}
set openOneOnly(val: boolean) {
if (val) {
this.setAttribute(openOneOnlyAttr, '');
} else {
this.removeAttribute(openOneOnlyAttr);
}
}
} }
customElements.define('multi-panel', MultiPanel); customElements.define('multi-panel', MultiPanel);

View File

@@ -0,0 +1,9 @@
interface MultiPanelAttributes extends JSX.HTMLAttributes {
'open-one-only'?: boolean;
}
declare namespace JSX {
interface IntrinsicElements {
'multi-panel': MultiPanelAttributes;
}
}

View File

@@ -0,0 +1,10 @@
.panel-heading {
background: gray;
}
.panel-content {
height: 0px;
overflow: auto;
}
.panel-content[aria-expanded=true] {
height: auto;
}

View File

@@ -24,18 +24,17 @@ import {
EncoderOptions, EncoderOptions,
encoderMap, encoderMap,
} from '../../codecs/encoders'; } from '../../codecs/encoders';
import { import {
PreprocessorState, PreprocessorState,
defaultPreprocessorState, defaultPreprocessorState,
} from '../../codecs/preprocessors'; } from '../../codecs/preprocessors';
import { decodeImage } from '../../codecs/decoders'; import { decodeImage } from '../../codecs/decoders';
import { cleanMerge, cleanSet } from '../../lib/clean-modify'; import { cleanMerge, cleanSet } from '../../lib/clean-modify';
import Processor from '../../codecs/processor'; import Processor from '../../codecs/processor';
import { VectorResizeOptions, BitmapResizeOptions } from '../../codecs/resize/processor-meta'; import { VectorResizeOptions, BitmapResizeOptions } from '../../codecs/resize/processor-meta';
import './custom-els/MultiPanel';
type Orientation = 'horizontal' | 'vertical'; import Results from '../results';
import { ExpandIcon } from '../../lib/icons';
export interface SourceImage { export interface SourceImage {
file: File | Fileish; file: File | Fileish;
@@ -69,7 +68,7 @@ interface State {
loading: boolean; loading: boolean;
loadingCounter: number; loadingCounter: number;
error?: string; error?: string;
orientation: Orientation; mobileView: boolean;
} }
interface UpdateImageOptions { interface UpdateImageOptions {
@@ -155,8 +154,11 @@ async function processSvg(blob: Blob): Promise<HTMLImageElement> {
return blobToImg(new Blob([newSource], { type: 'image/svg+xml' })); return blobToImg(new Blob([newSource], { type: 'image/svg+xml' }));
} }
// These are only used in the mobile view
const resultTitles = ['Top', 'Bottom'];
export default class Compress extends Component<Props, State> { export default class Compress extends Component<Props, State> {
widthQuery = window.matchMedia('(min-width: 500px)'); widthQuery = window.matchMedia('(max-width: 599px)');
state: State = { state: State = {
source: undefined, source: undefined,
@@ -178,7 +180,7 @@ export default class Compress extends Component<Props, State> {
loading: false, loading: false,
}, },
], ],
orientation: this.widthQuery.matches ? 'horizontal' : 'vertical', mobileView: this.widthQuery.matches,
}; };
private readonly encodeCache = new ResultCache(); private readonly encodeCache = new ResultCache();
@@ -193,7 +195,7 @@ export default class Compress extends Component<Props, State> {
@bind @bind
private onMobileWidthChange() { private onMobileWidthChange() {
this.setState({ orientation: this.widthQuery.matches ? 'horizontal' : 'vertical' }); this.setState({ mobileView: this.widthQuery.matches });
} }
private onEncoderTypeChange(index: 0 | 1, newType: EncoderType): void { private onEncoderTypeChange(index: 0 | 1, newType: EncoderType): void {
@@ -395,36 +397,69 @@ export default class Compress extends Component<Props, State> {
this.setState({ images }); this.setState({ images });
} }
render({ }: Props, { loading, images, source, orientation }: State) { render({ }: Props, { loading, images, source, mobileView }: State) {
const [leftImage, rightImage] = images; const [leftImage, rightImage] = images;
const [leftImageData, rightImageData] = images.map(i => i.data); const [leftImageData, rightImageData] = images.map(i => i.data);
const options = images.map((image, index) => (
<Options
source={source}
mobileView={mobileView}
imageIndex={index}
preprocessorState={image.preprocessorState}
encoderState={image.encoderState}
onEncoderTypeChange={this.onEncoderTypeChange.bind(this, index)}
onEncoderOptionsChange={this.onEncoderOptionsChange.bind(this, index)}
onPreprocessorOptionsChange={this.onPreprocessorOptionsChange.bind(this, index)}
onCopyToOtherClick={this.onCopyToOtherClick.bind(this, index)}
/>
));
const results = images.map((image, i) => (
<Results
downloadUrl={image.downloadUrl}
imageFile={image.file}
source={source}
loading={loading || image.loading}
>
{!mobileView ? null : [
<ExpandIcon class={style.expandIcon} key="expand-icon"/>,
`${resultTitles[i]} (${encoderMap[image.encoderState.type].label})`,
]}
</Results>
));
return ( return (
<div class={`${style.compress} ${style[orientation]}`}> <div class={style.compress}>
<Output <Output
originalImage={source && source.data} originalImage={source && source.data}
orientation={orientation} mobileView={mobileView}
leftCompressed={leftImageData} leftCompressed={leftImageData}
rightCompressed={rightImageData} rightCompressed={rightImageData}
leftImgContain={leftImage.preprocessorState.resize.fitMethod === 'cover'} leftImgContain={leftImage.preprocessorState.resize.fitMethod === 'cover'}
rightImgContain={rightImage.preprocessorState.resize.fitMethod === 'cover'} rightImgContain={rightImage.preprocessorState.resize.fitMethod === 'cover'}
/> />
{images.map((image, index) => ( {mobileView
<Options ? (
loading={loading || image.loading} <div class={style.options}>
source={source} <multi-panel class={style.multiPanel} open-one-only>
orientation={orientation} {results[0]}
imageIndex={index} {options[0]}
imageFile={image.file} {results[1]}
downloadUrl={image.downloadUrl} {options[1]}
preprocessorState={image.preprocessorState} </multi-panel>
encoderState={image.encoderState} </div>
onEncoderTypeChange={this.onEncoderTypeChange.bind(this, index)} ) : ([
onEncoderOptionsChange={this.onEncoderOptionsChange.bind(this, index)} <div class={style.options} key="options0">
onPreprocessorOptionsChange={this.onPreprocessorOptionsChange.bind(this, index)} {options[0]}
onCopyToOtherClick={this.onCopyToOtherClick.bind(this, index)} {results[0]}
/> </div>,
))} <div class={style.options} key="options1">
{options[1]}
{results[1]}
</div>,
])
}
</div> </div>
); );
} }

View File

@@ -3,20 +3,72 @@
height: 100%; height: 100%;
contain: strict; contain: strict;
display: grid; display: grid;
align-items: end;
align-content: end;
grid-template-rows: 1fr auto;
&.horizontal { @media (min-width: 600px) {
grid-template-columns: 1fr auto; grid-template-columns: 1fr auto;
grid-template-rows: calc(100% - 75px); grid-template-rows: 100%;
@media (min-width: 860px) {
grid-template-rows: 100%;
}
align-items: end;
align-content: end;
}
&.vertical {
// TODO: make the mobile view work
background: red;
} }
} }
.options {
color: #fff;
opacity: 0.9;
font-size: 1.2rem;
display: flex;
flex-flow: column;
max-width: 400px;
margin: 0 auto;
width: calc(100% - 60px);
max-height: calc(100% - 143px);
overflow: hidden;
@media (min-width: 600px) {
max-height: calc(100% - 75px);
width: 300px;
margin: 0;
}
@media (min-width: 860px) {
max-height: 100%;
}
}
.multi-panel {
position: relative;
display: flex;
flex-flow: column;
// Reorder so headings appear after content:
& > :nth-child(1) {
order: 2;
margin-bottom: 10px;
}
& > :nth-child(2) {
order: 1;
}
& > :nth-child(3) {
order: 4;
}
& > :nth-child(4) {
order: 3;
}
}
.expand-icon {
transform: rotate(180deg);
margin-left: -12px;
}
[content-expanded] .expand-icon {
transform: none;
}
:focus .expand-icon {
fill: #34B9EB;
}

View File

@@ -1,6 +1,6 @@
import { h, Component, ComponentChild, ComponentChildren } from 'preact'; import { h, Component, ComponentChild, ComponentChildren } from 'preact';
import * as style from './style.scss'; import * as style from './style.scss';
import { linkRef } from '../../lib/initial-util'; import { transitionHeight } from '../../lib/util';
interface Props { interface Props {
children: ComponentChildren; children: ComponentChildren;
@@ -13,7 +13,6 @@ export default class Expander extends Component<Props, State> {
state: State = { state: State = {
outgoingChildren: [], outgoingChildren: [],
}; };
private el?: HTMLDivElement;
private lastElHeight: number = 0; private lastElHeight: number = 0;
componentWillReceiveProps(nextProps: Props) { componentWillReceiveProps(nextProps: Props) {
@@ -32,10 +31,10 @@ export default class Expander extends Component<Props, State> {
// Only interested if going from empty to not-empty, or not-empty to empty. // Only interested if going from empty to not-empty, or not-empty to empty.
if ((children[0] && nextChildren[0]) || (!children[0] && !nextChildren[0])) return; if ((children[0] && nextChildren[0]) || (!children[0] && !nextChildren[0])) return;
this.lastElHeight = this.el!.getBoundingClientRect().height; this.lastElHeight = this.base!.getBoundingClientRect().height;
} }
componentDidUpdate(previousProps: Props) { async componentDidUpdate(previousProps: Props) {
const children = this.props.children as ComponentChild[]; const children = this.props.children as ComponentChild[];
const previousChildren = previousProps.children as ComponentChild[]; const previousChildren = previousProps.children as ComponentChild[];
@@ -43,37 +42,20 @@ export default class Expander extends Component<Props, State> {
if ((children[0] && previousChildren[0]) || (!children[0] && !previousChildren[0])) return; if ((children[0] && previousChildren[0]) || (!children[0] && !previousChildren[0])) return;
// What height do we need to transition to? // What height do we need to transition to?
this.el!.style.transition = 'none'; this.base!.style.height = '';
this.el!.style.height = ''; this.base!.style.overflow = 'hidden';
const newHeight = children[0] ? this.el!.getBoundingClientRect().height : 0; const newHeight = children[0] ? this.base!.getBoundingClientRect().height : 0;
if (this.lastElHeight === newHeight) { await transitionHeight(this.base!, {
this.el!.style.transition = ''; duration: 300,
return; from: this.lastElHeight,
} to: newHeight,
});
// Set the currently rendered height absolutely. // Unset the height & overflow, so element changes do the right thing.
this.el!.style.height = this.lastElHeight + 'px'; this.base!.style.height = '';
this.el!.style.transition = ''; this.base!.style.overflow = '';
this.el!.style.overflow = 'hidden'; if (this.state.outgoingChildren[0]) this.setState({ outgoingChildren: [] });
// Force a style calc so the browser picks up the start value.
getComputedStyle(this.el!).height;
// Animate to the new height.
this.el!.style.height = newHeight + 'px';
const listener = () => {
// Unset the height & overflow, so element changes do the right thing.
this.el!.style.height = '';
this.el!.style.overflow = '';
this.el!.removeEventListener('transitionend', listener);
this.el!.removeEventListener('transitioncancel', listener);
if (this.state.outgoingChildren[0]) {
this.setState({ outgoingChildren: [] });
}
};
this.el!.addEventListener('transitionend', listener);
this.el!.addEventListener('transitioncancel', listener);
} }
render(props: Props, { outgoingChildren }: State) { render(props: Props, { outgoingChildren }: State) {
@@ -81,10 +63,7 @@ export default class Expander extends Component<Props, State> {
const childrenExiting = !children[0] && outgoingChildren[0]; const childrenExiting = !children[0] && outgoingChildren[0];
return ( return (
<div <div class={childrenExiting ? style.childrenExiting : ''}>
ref={linkRef(this, 'el')}
class={`${style.expander} ${childrenExiting ? style.childrenExiting : ''}`}
>
{children[0] ? children : outgoingChildren} {children[0] ? children : outgoingChildren}
</div> </div>
); );

View File

@@ -1,7 +1,3 @@
.expander {
transition: height 200ms ease-in-out;
}
.children-exiting { .children-exiting {
& > * { & > * {
pointer-events: none; pointer-events: none;

View File

@@ -1,11 +0,0 @@
multi-panel > .panel-heading {
background:gray;
}
multi-panel > .panel-content {
height:0px;
overflow:scroll;
transition: height 1s;
}
multi-panel > .panel-content[expanded] {
height:auto;
}

View File

@@ -0,0 +1,78 @@
import { h, Component, ComponentChildren, ComponentChild } from 'preact';
import * as style from './style.scss';
import FileSize from './FileSize';
import { DownloadIcon } from '../../lib/icons';
import '../custom-els/LoadingSpinner';
import { SourceImage } from '../compress';
import { Fileish } from '../../lib/initial-util';
interface Props {
loading: boolean;
source?: SourceImage;
imageFile?: Fileish;
downloadUrl?: string;
children: ComponentChildren;
}
interface State {
showLoadingState: boolean;
}
const loadingReactionDelay = 500;
export default class Results extends Component<Props, State> {
state: State = {
showLoadingState: false,
};
/** The timeout ID between entering the loading state, and changing UI */
private loadingTimeoutId: number = 0;
componentDidUpdate(prevProps: Props, prevState: State) {
if (prevProps.loading && !this.props.loading) {
// Just stopped loading
clearTimeout(this.loadingTimeoutId);
this.setState({ showLoadingState: false });
} else if (!prevProps.loading && this.props.loading) {
// Just started loading
this.loadingTimeoutId = self.setTimeout(
() => this.setState({ showLoadingState: true }),
loadingReactionDelay,
);
}
}
render({ source, imageFile, downloadUrl, children }: Props, { showLoadingState }: State) {
return (
<div class={style.results}>
<div class={style.resultData}>
{(children as ComponentChild[])[0]
? <div class={style.resultTitle}>{children}</div>
: null
}
{!imageFile || showLoadingState ? 'Working…' :
<FileSize
blob={imageFile}
compareTo={(source && imageFile !== source.file) ? source.file : undefined}
/>
}
</div>
<div class={style.download}>
{(downloadUrl && imageFile) && (
<a
class={`${style.downloadLink} ${showLoadingState ? style.downloadLinkDisable : ''}`}
href={downloadUrl}
download={imageFile.name}
title="Download"
>
<DownloadIcon class={style.downloadIcon} />
</a>
)}
{showLoadingState && <loading-spinner class={style.spinner} />}
</div>
</div>
);
}
}

View File

@@ -0,0 +1,95 @@
@keyframes action-enter {
from {
transform: rotate(-90deg);
opacity: 0;
animation-timing-function: ease-out;
}
}
@keyframes action-leave {
from {
transform: rotate(0deg);
opacity: 1;
animation-timing-function: ease-out;
}
}
.results {
display: grid;
grid-template-columns: 1fr auto;
background: rgba(0, 0, 0, 0.9);
font-size: 1.4rem;
&:focus {
outline: none;
}
}
.result-data {
display: flex;
align-items: center;
padding: 0 15px;
white-space: nowrap;
overflow: hidden;
}
.result-title {
display: flex;
align-items: center;
margin-right: 0.4em;
}
.size-delta {
font-size: 1.1rem;
font-style: italic;
position: relative;
top: -1px;
margin-left: 0.3em;
}
.size-increase {
color: #e35050;
}
.size-decrease {
color: #50e3c2;
}
.download {
background: #34B9EB;
--size: 38px;
width: var(--size);
height: var(--size);
display: grid;
align-items: center;
justify-items: center;
}
.download-link {
animation: action-enter 0.2s;
grid-area: 1/1;
}
.download-link-disable {
pointer-events: none;
opacity: 0;
transform: rotate(90deg);
animation: action-leave 0.2s;
}
.download-icon {
color: #fff;
display: block;
--size: 24px;
width: var(--size);
height: var(--size);
padding: 7px;
filter: drop-shadow(0 1px 0 rgba(0, 0, 0, 0.7));
}
.spinner {
--color: #fff;
--delay: 0;
--size: 22px;
grid-area: 1/1;
}

View File

@@ -1,5 +1,6 @@
import { bind } from '../../lib/initial-util'; import { bind } from '../../lib/initial-util';
import * as style from './styles.css'; import * as style from './styles.css';
import { PointerTracker } from '../../lib/PointerTracker';
const RETARGETED_EVENTS = ['focus', 'blur']; const RETARGETED_EVENTS = ['focus', 'blur'];
const UPDATE_EVENTS = ['input', 'change']; const UPDATE_EVENTS = ['input', 'change'];
@@ -26,6 +27,17 @@ class RangeInputElement extends HTMLElement {
this._input.type = 'range'; this._input.type = 'range';
this._input.className = style.input; this._input.className = style.input;
const tracker = new PointerTracker(this._input, {
start: (): boolean => {
if (tracker.currentPointers.length !== 0) return false;
this._input.classList.add(style.touchActive);
return true;
},
end: () => {
this._input.classList.remove(style.touchActive);
},
});
for (const event of RETARGETED_EVENTS) { for (const event of RETARGETED_EVENTS) {
this._input.addEventListener(event, this._retargetEvent, true); this._input.addEventListener(event, this._retargetEvent, true);
} }

View File

@@ -84,11 +84,11 @@ range-input::before {
overflow: hidden; overflow: hidden;
} }
.input:active + .thumb-wrapper .value-display { .touch-active + .thumb-wrapper .value-display {
opacity: 1; opacity: 1;
transform: scale(1); transform: scale(1);
} }
.input:active + .thumb-wrapper .thumb { .touch-active + .thumb-wrapper .thumb {
box-shadow: 0 1px 3px rgba(0,0,0,0.5); box-shadow: 0 1px 3px rgba(0,0,0,0.5);
} }

View File

@@ -41,3 +41,9 @@ export const CheckedIcon = (props: JSX.HTMLAttributes) => (
<path d="M21.3 0H2.7A2.7 2.7 0 0 0 0 2.7v18.6A2.7 2.7 0 0 0 2.7 24h18.6a2.7 2.7 0 0 0 2.7-2.7V2.7A2.7 2.7 0 0 0 21.3 0zm-12 18.7L2.7 12l1.8-1.9L9.3 15 19.5 4.8l1.8 1.9z"/> <path d="M21.3 0H2.7A2.7 2.7 0 0 0 0 2.7v18.6A2.7 2.7 0 0 0 2.7 24h18.6a2.7 2.7 0 0 0 2.7-2.7V2.7A2.7 2.7 0 0 0 21.3 0zm-12 18.7L2.7 12l1.8-1.9L9.3 15 19.5 4.8l1.8 1.9z"/>
</Icon> </Icon>
); );
export const ExpandIcon = (props: JSX.HTMLAttributes) => (
<Icon {...props}>
<path d="M16.6 8.6L12 13.2 7.4 8.6 6 10l6 6 6-6z"/>
</Icon>
);

View File

@@ -258,3 +258,43 @@ export function konami(): Promise<void> {
window.addEventListener('keydown', listener); window.addEventListener('keydown', listener);
}); });
} }
interface TransitionOptions {
from?: number;
to?: number;
duration?: number;
easing?: string;
}
export async function transitionHeight(el: HTMLElement, opts: TransitionOptions): Promise<void> {
const {
from = el.getBoundingClientRect().height,
to = el.getBoundingClientRect().height,
duration = 1000,
easing = 'ease-in-out',
} = opts;
if (from === to || duration === 0) {
el.style.height = to + 'px';
return;
}
el.style.height = from + 'px';
// Force a style calc so the browser picks up the start value.
getComputedStyle(el).transform;
el.style.transition = `height ${duration}ms ${easing}`;
el.style.height = to + 'px';
return new Promise<void>((resolve) => {
const listener = (event: Event) => {
if (event.target !== el) return;
el.style.transition = '';
el.removeEventListener('transitionend', listener);
el.removeEventListener('transitioncancel', listener);
resolve();
};
el.addEventListener('transitionend', listener);
el.addEventListener('transitioncancel', listener);
});
}