Skip to content

Commit

Permalink
Merge pull request #12845 from Tomastomaslol/12324_zoom_buttons_in_do…
Browse files Browse the repository at this point in the history
…cs_do_not_work

Addon-docs: Fix Preview scaling with transform instead of zoom
  • Loading branch information
shilman committed Nov 19, 2020
2 parents 428b6e0 + 790b371 commit 4747fea
Show file tree
Hide file tree
Showing 7 changed files with 242 additions and 79 deletions.
97 changes: 97 additions & 0 deletions lib/components/src/Zoom/Zoom.stories.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,97 @@
import React, { CSSProperties, useEffect, useState } from 'react';
import { Zoom } from './Zoom';

export default {
component: Zoom,
title: 'Basics/Zoom',
argTypes: {
scale: {
control: { type: 'range', min: 0.2, max: 30, step: 0.02 },
},
},
};
const EXAMPLE_ELEMENT = (
<div
style={{
width: 2000,
height: 2000,
border: '10px solid orangered',
background: `url('')`,
}}
/>
);

const TemplateElement = (args) => <Zoom.Element {...args} />;

export const elementActualSize = TemplateElement.bind({});

elementActualSize.args = {
scale: 1,
children: EXAMPLE_ELEMENT,
};

export const elementZoomedIn = TemplateElement.bind({});

elementZoomedIn.args = {
scale: 0.7,
children: EXAMPLE_ELEMENT,
};

export const elementZoomedOut = TemplateElement.bind({});

elementZoomedOut.args = {
scale: 3,
children: EXAMPLE_ELEMENT,
};

const style: CSSProperties = {
width: '500px',
height: '500px',
border: '2px solid hotpink',
position: 'relative',
};

const TemplateIFrame = (args) => {
const iFrameRef = React.useRef<HTMLIFrameElement>(null);
const [scale, setScale] = useState(1);
const [loaded, hasLoaded] = useState(false);

useEffect(() => {
if (loaded) {
setScale(args.scale);
}
}, [args.scale, loaded]);
return (
<Zoom.IFrame iFrameRef={iFrameRef} scale={scale} active={args.active}>
<iframe
id="iframe"
title="UI Panel"
onLoad={() => hasLoaded(true)}
src="/iframe.html?id=ui-panel--default&viewMode=story"
style={style}
ref={iFrameRef}
allowFullScreen
/>
</Zoom.IFrame>
);
};
export const iFrameActualSize = TemplateIFrame.bind({});

iFrameActualSize.args = {
scale: 1,
active: true,
};

export const iFrameZoomedIn = TemplateIFrame.bind({});

iFrameZoomedIn.args = {
scale: 0.7,
active: true,
};

export const iFrameZoomedOut = TemplateIFrame.bind({});

iFrameZoomedOut.args = {
scale: 3,
active: true,
};
11 changes: 11 additions & 0 deletions lib/components/src/Zoom/Zoom.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
import window from 'global';
import { ZoomElement as Element } from './ZoomElement';
import { ZoomIFrame as IFrame } from './ZoomIFrame';

export const browserSupportsCssZoom = (): boolean =>
window.document.implementation.createHTMLDocument().body.style.zoom !== undefined;

export const Zoom = {
Element,
IFrame,
};
40 changes: 40 additions & 0 deletions lib/components/src/Zoom/ZoomElement.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
import React, { ReactElement, useEffect, useState } from 'react';
import { styled } from '@storybook/theming';
import { browserSupportsCssZoom } from './Zoom';

const ZoomElementWrapper = styled.div<{ scale: number; height: number }>(({ scale = 1, height }) =>
browserSupportsCssZoom()
? {
'> *': {
zoom: 1 / scale,
},
}
: {
height: height + 50,
transformOrigin: 'top left',
transform: `scale(${1 / scale})`,
}
);
type ZoomProps = {
scale: number;
children: ReactElement | ReactElement[];
};

export function ZoomElement({ scale, children }: ZoomProps) {
const componentWrapperRef = React.useRef<HTMLDivElement>(null);
const [height, setHeight] = useState(0);

useEffect(() => {
if (componentWrapperRef.current) {
setHeight(componentWrapperRef.current.getBoundingClientRect().height);
}
}, [scale, componentWrapperRef.current]);

return (
<ZoomElementWrapper scale={scale} height={height}>
<div ref={componentWrapperRef} className="innerZoomElementWrapper">
{children}
</div>
</ZoomElementWrapper>
);
}
67 changes: 67 additions & 0 deletions lib/components/src/Zoom/ZoomIFrame.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
import { Component, ReactElement } from 'react';
import { browserSupportsCssZoom } from './Zoom';

export type IZoomIFrameProps = {
scale: number;
children: ReactElement<HTMLIFrameElement>;
iFrameRef: React.MutableRefObject<HTMLIFrameElement>;
active?: boolean;
};

export class ZoomIFrame extends Component<IZoomIFrameProps> {
iframe: HTMLIFrameElement = null;

componentDidMount() {
const { iFrameRef } = this.props;
this.iframe = iFrameRef.current;
}

shouldComponentUpdate(nextProps: IZoomIFrameProps) {
const { scale, active } = this.props;

if (scale !== nextProps.scale) {
this.setIframeInnerZoom(nextProps.scale);
}

if (active !== nextProps.active) {
this.iframe.setAttribute('data-is-storybook', nextProps.active ? 'true' : 'false');
}

// this component renders an iframe, which gets updates via post-messages
// never update this component, it will cause the iframe to refresh
return false;
}

setIframeInnerZoom(scale: number) {
try {
if (browserSupportsCssZoom()) {
Object.assign(this.iframe.contentDocument.body.style, {
zoom: 1 / scale,
});
} else {
Object.assign(this.iframe.contentDocument.body.style, {
width: `${scale * 100}%`,
height: `${scale * 100}%`,
transform: `scale(${1 / scale})`,
transformOrigin: 'top left',
});
}
} catch (e) {
this.setIframeZoom(scale);
}
}

setIframeZoom(scale: number) {
Object.assign(this.iframe.style, {
width: `${scale * 100}%`,
height: `${scale * 100}%`,
transform: `scale(${1 / scale})`,
transformOrigin: 'top left',
});
}

render() {
const { children } = this.props;
return children;
}
}
31 changes: 15 additions & 16 deletions lib/components/src/blocks/Preview.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import { Source, SourceProps } from './Source';
import { ActionBar, ActionItem } from '../ActionBar/ActionBar';
import { Toolbar } from './Toolbar';
import { ZoomContext } from './ZoomContext';
import { Zoom } from '../Zoom/Zoom';

export interface PreviewProps {
isColumn?: boolean;
Expand All @@ -20,15 +21,15 @@ export interface PreviewProps {

type layout = 'padded' | 'fullscreen' | 'centered';

const ChildrenContainer = styled.div<PreviewProps & { zoom: number; layout: layout }>(
const ChildrenContainer = styled.div<PreviewProps & { layout: layout }>(
({ isColumn, columns, layout }) => ({
display: isColumn || !columns ? 'block' : 'flex',
position: 'relative',
flexWrap: 'wrap',
overflow: 'auto',
flexDirection: isColumn ? 'column' : 'row',

'& > *': isColumn
'& .innerZoomElementWrapper > *': isColumn
? {
width: layout !== 'fullscreen' ? 'calc(100% - 20px)' : '100%',
display: 'block',
Expand All @@ -43,7 +44,7 @@ const ChildrenContainer = styled.div<PreviewProps & { zoom: number; layout: layo
? {
padding: '30px 20px',
margin: -10,
'& > *': {
'& .innerZoomElementWrapper > *': {
width: 'auto',
border: '10px solid transparent!important',
},
Expand All @@ -59,13 +60,10 @@ const ChildrenContainer = styled.div<PreviewProps & { zoom: number; layout: layo
alignItems: 'center',
}
: {},
({ zoom = 1 }) => ({
'> *': {
zoom: 1 / zoom,
},
}),
({ columns }) =>
columns && columns > 1 ? { '> *': { minWidth: `calc(100% / ${columns} - 20px)` } } : {}
columns && columns > 1
? { '.innerZoomElementWrapper > *': { minWidth: `calc(100% / ${columns} - 20px)` } }
: {}
);

const StyledSource = styled(Source)<{}>(({ theme }) => ({
Expand Down Expand Up @@ -217,15 +215,16 @@ const Preview: FunctionComponent<PreviewProps> = ({
<ChildrenContainer
isColumn={isColumn || !Array.isArray(children)}
columns={columns}
zoom={scale}
layout={layout}
>
{Array.isArray(children) ? (
// eslint-disable-next-line react/no-array-index-key
children.map((child, i) => <div key={i}>{child}</div>)
) : (
<div>{children}</div>
)}
<Zoom.Element scale={scale}>
{Array.isArray(children) ? (
// eslint-disable-next-line react/no-array-index-key
children.map((child, i) => <div key={i}>{child}</div>)
) : (
<div>{children}</div>
)}
</Zoom.Element>
</ChildrenContainer>
<ActionBar actionItems={actionItems} />
</Relative>
Expand Down
1 change: 1 addition & 0 deletions lib/components/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ export { ActionBar } from './ActionBar/ActionBar';
export { Spaced } from './spaced/Spaced';
export { Placeholder } from './placeholder/placeholder';
export { ScrollArea } from './ScrollArea/ScrollArea';
export { Zoom } from './Zoom/Zoom';

// Forms
export { Button } from './Button/Button';
Expand Down
74 changes: 11 additions & 63 deletions lib/ui/src/components/preview/iframe.tsx
Original file line number Diff line number Diff line change
@@ -1,9 +1,6 @@
import window from 'global';
import React, { Component, IframeHTMLAttributes } from 'react';

import React, { IframeHTMLAttributes } from 'react';
import { styled } from '@storybook/theming';

const FIREFOX_BROWSER = 'Firefox';
import { Zoom } from '@storybook/components';

const StyledIframe = styled.iframe({
position: 'absolute',
Expand All @@ -25,70 +22,21 @@ export interface IFrameProps {
active: boolean;
}

export class IFrame extends Component<IFrameProps & IframeHTMLAttributes<HTMLIFrameElement>> {
iframe: HTMLIFrameElement = null;

componentDidMount() {
const { id } = this.props;
this.iframe = window.document.getElementById(id);
}

shouldComponentUpdate(nextProps: IFrameProps) {
const { scale, active } = this.props;

if (scale !== nextProps.scale) {
this.setIframeInnerZoom(nextProps.scale);
}

if (active !== nextProps.active) {
this.iframe.setAttribute('data-is-storybook', nextProps.active ? 'true' : 'false');
}

// this component renders an iframe, which gets updates via post-messages
// never update this component, it will cause the iframe to refresh
return false;
}

setIframeInnerZoom(scale: number) {
try {
if (window.navigator.userAgent.indexOf(FIREFOX_BROWSER) !== -1) {
Object.assign(this.iframe.contentDocument.body.style, {
width: `${scale * 100}%`,
height: `${scale * 100}%`,
transform: `scale(${1 / scale})`,
transformOrigin: 'top left',
});
} else {
Object.assign(this.iframe.contentDocument.body.style, {
zoom: 1 / scale,
});
}
} catch (e) {
this.setIframeZoom(scale);
}
}

setIframeZoom(scale: number) {
Object.assign(this.iframe.style, {
width: `${scale * 100}%`,
height: `${scale * 100}%`,
transform: `scale(${1 / scale})`,
transformOrigin: 'top left',
});
}

render() {
const { id, title, src, allowFullScreen, scale, active, ...rest } = this.props;
return (
export function IFrame(props: IFrameProps & IframeHTMLAttributes<HTMLIFrameElement>) {
const { active, id, title, src, allowFullScreen, scale, ...rest } = props;
const iFrameRef = React.useRef<HTMLIFrameElement>(null);
return (
<Zoom.IFrame scale={scale} active={active} iFrameRef={iFrameRef}>
<StyledIframe
onLoad={() => this.iframe.setAttribute('data-is-loaded', 'true')}
data-is-storybook={active ? 'true' : 'false'}
onLoad={(e) => e.currentTarget.setAttribute('data-is-loaded', 'true')}
id={id}
title={title}
src={src}
allowFullScreen={allowFullScreen}
ref={iFrameRef}
{...rest}
/>
);
}
</Zoom.IFrame>
);
}

0 comments on commit 4747fea

Please sign in to comment.