Addon-docs: Fix link CORS errors using channel navigate event (#9381)

Addon-docs: Fix link CORS errors using channel navigate event
This commit is contained in:
Michael Shilman 2020-01-11 10:09:47 +08:00 committed by GitHub
commit c5d9783356
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 56 additions and 30 deletions

View File

@ -49,6 +49,7 @@
"@storybook/addons": "5.3.0-rc.12", "@storybook/addons": "5.3.0-rc.12",
"@storybook/api": "5.3.0-rc.12", "@storybook/api": "5.3.0-rc.12",
"@storybook/components": "5.3.0-rc.12", "@storybook/components": "5.3.0-rc.12",
"@storybook/core-events": "5.3.0-rc.12",
"@storybook/csf": "0.0.1", "@storybook/csf": "0.0.1",
"@storybook/postinstall": "5.3.0-rc.12", "@storybook/postinstall": "5.3.0-rc.12",
"@storybook/source-loader": "5.3.0-rc.12", "@storybook/source-loader": "5.3.0-rc.12",

View File

@ -29,7 +29,12 @@ export const DocsContainer: FunctionComponent<DocsContainerProps> = ({ context,
const allComponents = { ...defaultComponents, ...userComponents }; const allComponents = { ...defaultComponents, ...userComponents };
useEffect(() => { useEffect(() => {
const url = new URL(window.parent.location); let url;
try {
url = new URL(window.parent.location);
} catch (err) {
return;
}
if (url.hash) { if (url.hash) {
const element = document.getElementById(url.hash.substring(1)); const element = document.getElementById(url.hash.substring(1));
if (element) { if (element) {

View File

@ -1,11 +1,11 @@
import React, { FC, SyntheticEvent } from 'react'; import React, { FC, SyntheticEvent } from 'react';
import addons from '@storybook/addons';
import { Source } from '@storybook/components'; import { Source } from '@storybook/components';
import { NAVIGATE_URL } from '@storybook/core-events';
import { Code, components } from '@storybook/components/html'; import { Code, components } from '@storybook/components/html';
import { document, window } from 'global'; import { document } from 'global';
import { isNil } from 'lodash';
import { styled } from '@storybook/theming'; import { styled } from '@storybook/theming';
import { DocsContext, DocsContextProps } from './DocsContext'; import { DocsContext, DocsContextProps } from './DocsContext';
import { scrollToElement } from './utils';
// Hacky utility for asserting identifiers in MDX Story elements // Hacky utility for asserting identifiers in MDX Story elements
export const assertIsFn = (val: any) => { export const assertIsFn = (val: any) => {
@ -48,11 +48,8 @@ export const CodeOrSourceMdx: FC<CodeOrSourceMdxProps> = ({ className, children,
); );
}; };
function generateHrefWithHash(hash: string): string { function navigate(url: string) {
const url = new URL(window.parent.location); addons.getChannel().emit(NAVIGATE_URL, url);
const href = `${url.origin}/${url.search}#${hash}`;
return href;
} }
// @ts-ignore // @ts-ignore
@ -65,14 +62,12 @@ interface AnchorInPageProps {
const AnchorInPage: FC<AnchorInPageProps> = ({ hash, children }) => ( const AnchorInPage: FC<AnchorInPageProps> = ({ hash, children }) => (
<A <A
href={hash} href={hash}
target="_self"
onClick={(event: SyntheticEvent) => { onClick={(event: SyntheticEvent) => {
event.preventDefault(); const id = hash.substring(1);
const element = document.getElementById(id);
const hashValue = hash.substring(1); if (element) {
const element = document.getElementById(hashValue); navigate(hash);
if (!isNil(element)) {
window.parent.history.replaceState(null, '', generateHrefWithHash(hashValue));
scrollToElement(element);
} }
}} }}
> >
@ -88,7 +83,7 @@ interface AnchorMdxProps {
export const AnchorMdx: FC<AnchorMdxProps> = props => { export const AnchorMdx: FC<AnchorMdxProps> = props => {
const { href, target, children, ...rest } = props; const { href, target, children, ...rest } = props;
if (!isNil(href)) { if (href) {
// Enable scrolling for in-page anchors. // Enable scrolling for in-page anchors.
if (href.startsWith('#')) { if (href.startsWith('#')) {
return <AnchorInPage hash={href}>{children}</AnchorInPage>; return <AnchorInPage hash={href}>{children}</AnchorInPage>;
@ -96,11 +91,16 @@ export const AnchorMdx: FC<AnchorMdxProps> = props => {
// Links to other pages of SB should use the base URL of the top level iframe instead of the base URL of the preview iframe. // Links to other pages of SB should use the base URL of the top level iframe instead of the base URL of the preview iframe.
if (target !== '_blank') { if (target !== '_blank') {
const parentUrl = new URL(window.parent.location.href);
const newHref = `${parentUrl.origin}${href}`;
return ( return (
<A href={newHref} target={target} {...rest}> <A
href={href}
onClick={(event: SyntheticEvent) => {
event.preventDefault();
navigate(href);
}}
target={target}
{...rest}
>
{children} {children}
</A> </A>
); );
@ -149,17 +149,19 @@ const HeaderWithOcticonAnchor: FC<HeaderWithOcticonAnchorProps> = ({
}) => { }) => {
// @ts-ignore // @ts-ignore
const OcticonHeader = OcticonHeaders[as]; const OcticonHeader = OcticonHeaders[as];
const hash = `#${id}`;
return ( return (
<OcticonHeader id={id} {...rest}> <OcticonHeader id={id} {...rest}>
<OcticonAnchor <OcticonAnchor
aria-hidden="true" aria-hidden="true"
href={generateHrefWithHash(id)} href={hash}
tabIndex={-1} tabIndex={-1}
onClick={() => { target="_self"
onClick={(event: SyntheticEvent) => {
const element = document.getElementById(id); const element = document.getElementById(id);
if (!isNil(element)) { if (element) {
scrollToElement(element); navigate(hash);
} }
}} }}
> >
@ -184,7 +186,7 @@ export const HeaderMdx: FC<HeaderMdxProps> = props => {
const { as, id, children, ...rest } = props; const { as, id, children, ...rest } = props;
// An id should have been added on every header by the "remark-slug" plugin. // An id should have been added on every header by the "remark-slug" plugin.
if (!isNil(id)) { if (id) {
return ( return (
<HeaderWithOcticonAnchor as={as} id={id} {...rest}> <HeaderWithOcticonAnchor as={as} id={id} {...rest}>
{children} {children}

View File

@ -129,13 +129,19 @@ Right aligned columns
## Links ## Links
[link text](https://hichroma.com) [external link](https://hichroma.com)
[link with title](https://hichroma.com 'Insert title!') [external link with title](https://hichroma.com 'Insert title!')
[link to in page anchor](#h1-heading) [link to in page anchor](#h1-heading)
[link to another story](/?path=/docs/addons-docs-docs-only--page#bottom) [link to another story (docs)](?path=/docs/addons-docs-docs-only--page)
[link to another story (canvas)](?path=/story/addons-docs-buttongroup--basic)
[link to about page](?path=/settings/about)
[link to absolute local url](/absolute)
## Images ## Images

View File

@ -26,12 +26,14 @@
"prepare": "node ./scripts/generateVersion.js && node ../../scripts/prepare.js" "prepare": "node ./scripts/generateVersion.js && node ../../scripts/prepare.js"
}, },
"dependencies": { "dependencies": {
"@reach/router": "^1.2.1",
"@storybook/channels": "5.3.0-rc.12", "@storybook/channels": "5.3.0-rc.12",
"@storybook/client-logger": "5.3.0-rc.12", "@storybook/client-logger": "5.3.0-rc.12",
"@storybook/core-events": "5.3.0-rc.12", "@storybook/core-events": "5.3.0-rc.12",
"@storybook/csf": "0.0.1", "@storybook/csf": "0.0.1",
"@storybook/router": "5.3.0-rc.12", "@storybook/router": "5.3.0-rc.12",
"@storybook/theming": "5.3.0-rc.12", "@storybook/theming": "5.3.0-rc.12",
"@types/reach__router": "^1.2.3",
"core-js": "^3.0.1", "core-js": "^3.0.1",
"fast-deep-equal": "^2.0.1", "fast-deep-equal": "^2.0.1",
"global": "^4.3.2", "global": "^4.3.2",

View File

@ -38,7 +38,7 @@ export { Options as StoreOptions, Listener as ChannelListener };
const ManagerContext = createContext({ api: undefined, state: getInitialState({}) }); const ManagerContext = createContext({ api: undefined, state: getInitialState({}) });
const { STORY_CHANGED, SET_STORIES, SELECT_STORY } = Events; const { STORY_CHANGED, SET_STORIES, SELECT_STORY, NAVIGATE_URL } = Events;
export type Module = StoreData & export type Module = StoreData &
RouterData & RouterData &
@ -186,6 +186,9 @@ class ManagerProvider extends Component<Props, State> {
api.selectStory(kind, story, rest); api.selectStory(kind, story, rest);
} }
); );
api.on(NAVIGATE_URL, (url: string, options: { [k: string]: any }) => {
api.navigateUrl(url, options);
});
this.state = state; this.state = state;
this.api = api; this.api = api;

View File

@ -1,3 +1,4 @@
import { navigate as navigateRouter, NavigateOptions } from '@reach/router';
import { queryFromLocation } from '@storybook/router'; import { queryFromLocation } from '@storybook/router';
import { toId } from '@storybook/csf'; import { toId } from '@storybook/csf';
@ -92,6 +93,7 @@ export interface QueryParams {
} }
export interface SubAPI { export interface SubAPI {
navigateUrl: (url: string, options: NavigateOptions<{}>) => void;
getQueryParam: (key: string) => string | undefined; getQueryParam: (key: string) => string | undefined;
getUrlState: () => { getUrlState: () => {
queryParams: QueryParams; queryParams: QueryParams;
@ -139,6 +141,9 @@ export default function({ store, navigate, state, provider, ...rest }: Module) {
}, },
}); });
}, },
navigateUrl(url: string, options: NavigateOptions<{}>) {
navigateRouter(url, options);
},
}; };
return { return {

View File

@ -21,6 +21,7 @@ enum events {
STORIES_COLLAPSE_ALL = 'storiesCollapseAll', STORIES_COLLAPSE_ALL = 'storiesCollapseAll',
STORIES_EXPAND_ALL = 'storiesExpandAll', STORIES_EXPAND_ALL = 'storiesExpandAll',
DOCS_RENDERED = 'docsRendered', DOCS_RENDERED = 'docsRendered',
NAVIGATE_URL = 'navigateUrl',
} }
// Enables: `import Events from ...` // Enables: `import Events from ...`
@ -50,4 +51,5 @@ export const {
STORIES_EXPAND_ALL, STORIES_EXPAND_ALL,
STORY_THREW_EXCEPTION, STORY_THREW_EXCEPTION,
DOCS_RENDERED, DOCS_RENDERED,
NAVIGATE_URL,
} = events; } = events;