mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-05 16:11:33 +08:00
Add Vue3 Webpack5 Storybook template components for JavaScript and TypeScript
Expand the Vue3 Webpack5 Storybook template with comprehensive component support: - Add Button, Header, and Page components for JavaScript and TypeScript (4.9) - Include stories, components, and CSS files - Implement Storybook best practices with autodocs and component testing
This commit is contained in:
parent
7418c0edba
commit
543fbbacb6
@ -0,0 +1,44 @@
|
|||||||
|
import { fn } from '@storybook/test';
|
||||||
|
|
||||||
|
import MyButton from './Button.vue';
|
||||||
|
|
||||||
|
// More on how to set up stories at: https://storybook.js.org/docs/writing-stories
|
||||||
|
export default {
|
||||||
|
title: 'Example/Button',
|
||||||
|
component: MyButton,
|
||||||
|
tags: ['autodocs'],
|
||||||
|
argTypes: {
|
||||||
|
size: { control: { type: 'select' }, options: ['small', 'medium', 'large'] },
|
||||||
|
backgroundColor: { control: 'color' },
|
||||||
|
},
|
||||||
|
// Use `fn` to spy on the onClick arg, which will appear in the actions panel once invoked: https://storybook.js.org/docs/essentials/actions#action-args
|
||||||
|
args: { onClick: fn() },
|
||||||
|
};
|
||||||
|
|
||||||
|
// More on writing stories with args: https://storybook.js.org/docs/writing-stories/args
|
||||||
|
export const Primary = {
|
||||||
|
args: {
|
||||||
|
primary: true,
|
||||||
|
label: 'Button',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Secondary = {
|
||||||
|
args: {
|
||||||
|
label: 'Button',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Large = {
|
||||||
|
args: {
|
||||||
|
size: 'large',
|
||||||
|
label: 'Button',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Small = {
|
||||||
|
args: {
|
||||||
|
size: 'small',
|
||||||
|
label: 'Button',
|
||||||
|
},
|
||||||
|
};
|
52
code/frameworks/vue3-webpack5/template/cli/js/Button.vue
Normal file
52
code/frameworks/vue3-webpack5/template/cli/js/Button.vue
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
<template>
|
||||||
|
<button type="button" :class="classes" @click="onClick" :style="style">{{ label }}</button>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import './button.css';
|
||||||
|
import { reactive, computed } from 'vue';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'my-button',
|
||||||
|
|
||||||
|
props: {
|
||||||
|
label: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
primary: {
|
||||||
|
type: Boolean,
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
size: {
|
||||||
|
type: String,
|
||||||
|
validator: function (value) {
|
||||||
|
return ['small', 'medium', 'large'].indexOf(value) !== -1;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
backgroundColor: {
|
||||||
|
type: String,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
emits: ['click'],
|
||||||
|
|
||||||
|
setup(props, { emit }) {
|
||||||
|
props = reactive(props);
|
||||||
|
return {
|
||||||
|
classes: computed(() => ({
|
||||||
|
'storybook-button': true,
|
||||||
|
'storybook-button--primary': props.primary,
|
||||||
|
'storybook-button--secondary': !props.primary,
|
||||||
|
[`storybook-button--${props.size || 'medium'}`]: true,
|
||||||
|
})),
|
||||||
|
style: computed(() => ({
|
||||||
|
backgroundColor: props.backgroundColor,
|
||||||
|
})),
|
||||||
|
onClick() {
|
||||||
|
emit('click');
|
||||||
|
},
|
||||||
|
};
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
@ -0,0 +1,48 @@
|
|||||||
|
import { fn } from '@storybook/test';
|
||||||
|
|
||||||
|
import MyHeader from './Header.vue';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
title: 'Example/Header',
|
||||||
|
component: MyHeader,
|
||||||
|
// This component will have an automatically generated Autodocs entry: https://storybook.js.org/docs/writing-docs/autodocs
|
||||||
|
tags: ['autodocs'],
|
||||||
|
render: (args) => ({
|
||||||
|
// Components used in your story `template` are defined in the `components` object
|
||||||
|
components: {
|
||||||
|
MyHeader,
|
||||||
|
},
|
||||||
|
// The story's `args` need to be mapped into the template through the `setup()` method
|
||||||
|
setup() {
|
||||||
|
// Story args can be spread into the returned object
|
||||||
|
return {
|
||||||
|
...args,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
// Then, the spread values can be accessed directly in the template
|
||||||
|
template: '<my-header :user="user" />',
|
||||||
|
}),
|
||||||
|
parameters: {
|
||||||
|
// More on how to position stories at: https://storybook.js.org/docs/configure/story-layout
|
||||||
|
layout: 'fullscreen',
|
||||||
|
},
|
||||||
|
args: {
|
||||||
|
onLogin: fn(),
|
||||||
|
onLogout: fn(),
|
||||||
|
onCreateAccount: fn(),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const LoggedIn = {
|
||||||
|
args: {
|
||||||
|
user: {
|
||||||
|
name: 'Jane Doe',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const LoggedOut = {
|
||||||
|
args: {
|
||||||
|
user: null,
|
||||||
|
},
|
||||||
|
};
|
59
code/frameworks/vue3-webpack5/template/cli/js/Header.vue
Normal file
59
code/frameworks/vue3-webpack5/template/cli/js/Header.vue
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
<template>
|
||||||
|
<header>
|
||||||
|
<div class="storybook-header">
|
||||||
|
<div>
|
||||||
|
<svg width="32" height="32" viewBox="0 0 32 32" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<g fill="none" fill-rule="evenodd">
|
||||||
|
<path
|
||||||
|
d="M10 0h12a10 10 0 0110 10v12a10 10 0 01-10 10H10A10 10 0 010 22V10A10 10 0 0110 0z"
|
||||||
|
fill="#FFF"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M5.3 10.6l10.4 6v11.1l-10.4-6v-11zm11.4-6.2l9.7 5.5-9.7 5.6V4.4z"
|
||||||
|
fill="#555AB9"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M27.2 10.6v11.2l-10.5 6V16.5l10.5-6zM15.7 4.4v11L6 10l9.7-5.5z"
|
||||||
|
fill="#91BAF8"
|
||||||
|
/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
<h1>Acme</h1>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<span class="welcome" v-if="user"
|
||||||
|
>Welcome, <b>{{ user.name }}</b
|
||||||
|
>!</span
|
||||||
|
>
|
||||||
|
<my-button size="small" @click="$emit('logout')" label="Log out" v-if="user" />
|
||||||
|
<my-button size="small" @click="$emit('login')" label="Log in" v-if="!user" />
|
||||||
|
<my-button
|
||||||
|
primary
|
||||||
|
size="small"
|
||||||
|
@click="$emit('createAccount')"
|
||||||
|
label="Sign up"
|
||||||
|
v-if="!user"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import './header.css';
|
||||||
|
import MyButton from './Button.vue';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'my-header',
|
||||||
|
|
||||||
|
components: { MyButton },
|
||||||
|
|
||||||
|
props: {
|
||||||
|
user: {
|
||||||
|
type: Object,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
emits: ['login', 'logout', 'createAccount'],
|
||||||
|
};
|
||||||
|
</script>
|
@ -0,0 +1,34 @@
|
|||||||
|
import { expect, userEvent, within } from '@storybook/test';
|
||||||
|
|
||||||
|
import MyPage from './Page.vue';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
title: 'Example/Page',
|
||||||
|
component: MyPage,
|
||||||
|
parameters: {
|
||||||
|
// More on how to position stories at: https://storybook.js.org/docs/configure/story-layout
|
||||||
|
layout: 'fullscreen',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const LoggedOut = {};
|
||||||
|
|
||||||
|
// More on component testing: https://storybook.js.org/docs/writing-tests/component-testing
|
||||||
|
export const LoggedIn = {
|
||||||
|
render: () => ({
|
||||||
|
components: {
|
||||||
|
MyPage,
|
||||||
|
},
|
||||||
|
template: '<my-page />',
|
||||||
|
}),
|
||||||
|
play: async ({ canvasElement }) => {
|
||||||
|
const canvas = within(canvasElement);
|
||||||
|
const loginButton = canvas.getByRole('button', { name: /Log in/i });
|
||||||
|
await expect(loginButton).toBeInTheDocument();
|
||||||
|
await userEvent.click(loginButton);
|
||||||
|
await expect(loginButton).not.toBeInTheDocument();
|
||||||
|
|
||||||
|
const logoutButton = canvas.getByRole('button', { name: /Log out/i });
|
||||||
|
await expect(logoutButton).toBeInTheDocument();
|
||||||
|
},
|
||||||
|
};
|
83
code/frameworks/vue3-webpack5/template/cli/js/Page.vue
Normal file
83
code/frameworks/vue3-webpack5/template/cli/js/Page.vue
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
<template>
|
||||||
|
<article>
|
||||||
|
<my-header :user="user" @login="onLogin" @logout="onLogout" @createAccount="onCreateAccount" />
|
||||||
|
|
||||||
|
<section class="storybook-page">
|
||||||
|
<h2>Pages in Storybook</h2>
|
||||||
|
<p>
|
||||||
|
We recommend building UIs with a
|
||||||
|
<a href="https://componentdriven.org" target="_blank" rel="noopener noreferrer">
|
||||||
|
<strong>component-driven</strong>
|
||||||
|
</a>
|
||||||
|
process starting with atomic components and ending with pages.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
Render pages with mock data. This makes it easy to build and review page states without
|
||||||
|
needing to navigate to them in your app. Here are some handy patterns for managing page data
|
||||||
|
in Storybook:
|
||||||
|
</p>
|
||||||
|
<ul>
|
||||||
|
<li>
|
||||||
|
Use a higher-level connected component. Storybook helps you compose such data from the
|
||||||
|
"args" of child component stories
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
Assemble data in the page component from your services. You can mock these services out
|
||||||
|
using Storybook.
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
<p>
|
||||||
|
Get a guided tutorial on component-driven development at
|
||||||
|
<a href="https://storybook.js.org/tutorials/" target="_blank" rel="noopener noreferrer"
|
||||||
|
>Storybook tutorials</a
|
||||||
|
>
|
||||||
|
. Read more in the
|
||||||
|
<a href="https://storybook.js.org/docs" target="_blank" rel="noopener noreferrer">docs</a>
|
||||||
|
.
|
||||||
|
</p>
|
||||||
|
<div class="tip-wrapper">
|
||||||
|
<span class="tip">Tip</span>
|
||||||
|
Adjust the width of the canvas with the
|
||||||
|
<svg width="10" height="10" viewBox="0 0 12 12" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<g fill="none" fill-rule="evenodd">
|
||||||
|
<path
|
||||||
|
d="M1.5 5.2h4.8c.3 0 .5.2.5.4v5.1c-.1.2-.3.3-.4.3H1.4a.5.5 0 01-.5-.4V5.7c0-.3.2-.5.5-.5zm0-2.1h6.9c.3 0 .5.2.5.4v7a.5.5 0 01-1 0V4H1.5a.5.5 0 010-1zm0-2.1h9c.3 0 .5.2.5.4v9.1a.5.5 0 01-1 0V2H1.5a.5.5 0 010-1zm4.3 5.2H2V10h3.8V6.2z"
|
||||||
|
id="a"
|
||||||
|
fill="#999"
|
||||||
|
/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
Viewports addon in the toolbar
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</article>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import './page.css';
|
||||||
|
import MyHeader from './Header.vue';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'my-page',
|
||||||
|
|
||||||
|
components: { MyHeader },
|
||||||
|
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
user: null,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
onLogin() {
|
||||||
|
this.user = { name: 'Jane Doe' };
|
||||||
|
},
|
||||||
|
onLogout() {
|
||||||
|
this.user = null;
|
||||||
|
},
|
||||||
|
onCreateAccount() {
|
||||||
|
this.user = { name: 'Jane Doe' };
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
@ -0,0 +1,57 @@
|
|||||||
|
import { fn } from '@storybook/test';
|
||||||
|
|
||||||
|
import type { Meta, StoryObj } from '@storybook/vue3-webpack5';
|
||||||
|
|
||||||
|
import Button from './Button.vue';
|
||||||
|
|
||||||
|
// More on how to set up stories at: https://storybook.js.org/docs/writing-stories
|
||||||
|
const meta = {
|
||||||
|
title: 'Example/Button',
|
||||||
|
component: Button,
|
||||||
|
// This component will have an automatically generated docsPage entry: https://storybook.js.org/docs/writing-docs/autodocs
|
||||||
|
tags: ['autodocs'],
|
||||||
|
argTypes: {
|
||||||
|
size: { control: 'select', options: ['small', 'medium', 'large'] },
|
||||||
|
backgroundColor: { control: 'color' },
|
||||||
|
},
|
||||||
|
args: {
|
||||||
|
primary: false,
|
||||||
|
// Use `fn` to spy on the onClick arg, which will appear in the actions panel once invoked: https://storybook.js.org/docs/essentials/actions#action-args
|
||||||
|
onClick: fn(),
|
||||||
|
},
|
||||||
|
} satisfies Meta<typeof Button>;
|
||||||
|
|
||||||
|
export default meta;
|
||||||
|
type Story = StoryObj<typeof meta>;
|
||||||
|
/*
|
||||||
|
*👇 Render functions are a framework specific feature to allow you control on how the component renders.
|
||||||
|
* See https://storybook.js.org/docs/api/csf
|
||||||
|
* to learn how to use render functions.
|
||||||
|
*/
|
||||||
|
export const Primary: Story = {
|
||||||
|
args: {
|
||||||
|
primary: true,
|
||||||
|
label: 'Button',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Secondary: Story = {
|
||||||
|
args: {
|
||||||
|
primary: false,
|
||||||
|
label: 'Button',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Large: Story = {
|
||||||
|
args: {
|
||||||
|
label: 'Button',
|
||||||
|
size: 'large',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const Small: Story = {
|
||||||
|
args: {
|
||||||
|
label: 'Button',
|
||||||
|
size: 'small',
|
||||||
|
},
|
||||||
|
};
|
48
code/frameworks/vue3-webpack5/template/cli/ts-4-9/Button.vue
Normal file
48
code/frameworks/vue3-webpack5/template/cli/ts-4-9/Button.vue
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
<template>
|
||||||
|
<button type="button" :class="classes" @click="onClick" :style="style">{{ label }} </button>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import './button.css';
|
||||||
|
import { computed } from 'vue';
|
||||||
|
|
||||||
|
const props = withDefaults(defineProps<{
|
||||||
|
/**
|
||||||
|
* The label of the button
|
||||||
|
*/
|
||||||
|
label: string,
|
||||||
|
/**
|
||||||
|
* primary or secondary button
|
||||||
|
*/
|
||||||
|
primary?: boolean,
|
||||||
|
/**
|
||||||
|
* size of the button
|
||||||
|
*/
|
||||||
|
size?: 'small' | 'medium' | 'large',
|
||||||
|
/**
|
||||||
|
* background color of the button
|
||||||
|
*/
|
||||||
|
backgroundColor?: string,
|
||||||
|
|
||||||
|
}>(), { primary: false });
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(e: 'click', id: number): void;
|
||||||
|
}>();
|
||||||
|
|
||||||
|
const classes = computed(() => ({
|
||||||
|
'storybook-button': true,
|
||||||
|
'storybook-button--primary': props.primary,
|
||||||
|
'storybook-button--secondary': !props.primary,
|
||||||
|
[`storybook-button--${props.size || 'medium'}`]: true,
|
||||||
|
}));
|
||||||
|
|
||||||
|
const style = computed(() => ({
|
||||||
|
backgroundColor: props.backgroundColor
|
||||||
|
}));
|
||||||
|
|
||||||
|
const onClick = () => {
|
||||||
|
emit("click", 1)
|
||||||
|
};
|
||||||
|
|
||||||
|
</script>
|
@ -0,0 +1,49 @@
|
|||||||
|
import { fn } from '@storybook/test';
|
||||||
|
|
||||||
|
import type { Meta, StoryObj } from '@storybook/vue3-webpack5';
|
||||||
|
|
||||||
|
import MyHeader from './Header.vue';
|
||||||
|
|
||||||
|
const meta = {
|
||||||
|
/* 👇 The title prop is optional.
|
||||||
|
* See https://storybook.js.org/docs/configure/#configure-story-loading
|
||||||
|
* to learn how to generate automatic titles
|
||||||
|
*/
|
||||||
|
title: 'Example/Header',
|
||||||
|
component: MyHeader,
|
||||||
|
render: (args: any) => ({
|
||||||
|
components: { MyHeader },
|
||||||
|
setup() {
|
||||||
|
return { args };
|
||||||
|
},
|
||||||
|
template: '<my-header :user="args.user" />',
|
||||||
|
}),
|
||||||
|
parameters: {
|
||||||
|
// More on how to position stories at: https://storybook.js.org/docs/configure/story-layout
|
||||||
|
layout: 'fullscreen',
|
||||||
|
},
|
||||||
|
args: {
|
||||||
|
onLogin: fn(),
|
||||||
|
onLogout: fn(),
|
||||||
|
onCreateAccount: fn(),
|
||||||
|
},
|
||||||
|
// This component will have an automatically generated docsPage entry: https://storybook.js.org/docs/writing-docs/autodocs
|
||||||
|
tags: ['autodocs'],
|
||||||
|
} satisfies Meta<typeof MyHeader>;
|
||||||
|
|
||||||
|
export default meta;
|
||||||
|
type Story = StoryObj<typeof meta>;
|
||||||
|
|
||||||
|
export const LoggedIn: Story = {
|
||||||
|
args: {
|
||||||
|
user: {
|
||||||
|
name: 'Jane Doe',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const LoggedOut: Story = {
|
||||||
|
args: {
|
||||||
|
user: null,
|
||||||
|
},
|
||||||
|
};
|
37
code/frameworks/vue3-webpack5/template/cli/ts-4-9/Header.vue
Normal file
37
code/frameworks/vue3-webpack5/template/cli/ts-4-9/Header.vue
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
<template>
|
||||||
|
<header>
|
||||||
|
<div class="storybook-header">
|
||||||
|
<div>
|
||||||
|
<svg width="32" height="32" viewBox="0 0 32 32" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<g fill="none" fill-rule="evenodd">
|
||||||
|
<path d="M10 0h12a10 10 0 0110 10v12a10 10 0 01-10 10H10A10 10 0 010 22V10A10 10 0 0110 0z" fill="#FFF" />
|
||||||
|
<path d="M5.3 10.6l10.4 6v11.1l-10.4-6v-11zm11.4-6.2l9.7 5.5-9.7 5.6V4.4z" fill="#555AB9" />
|
||||||
|
<path d="M27.2 10.6v11.2l-10.5 6V16.5l10.5-6zM15.7 4.4v11L6 10l9.7-5.5z" fill="#91BAF8" />
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
<h1>Acme</h1>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<span class="welcome" v-if="user">Welcome, <b>{{ user.name }}</b>!</span>
|
||||||
|
<my-button size="small" @click="$emit('logout')" label="Log out" v-if="user" />
|
||||||
|
<my-button size="small" @click="$emit('login')" label="Log in" v-if="!user" />
|
||||||
|
<my-button primary size="small" @click="$emit('createAccount')" label="Sign up" v-if="!user" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import './header.css';
|
||||||
|
import MyButton from './Button.vue';
|
||||||
|
|
||||||
|
defineProps<{ user: { name: string } | null }>();
|
||||||
|
|
||||||
|
defineEmits<{
|
||||||
|
(event: 'createAccount'): void;
|
||||||
|
(event: 'login'): void;
|
||||||
|
(event: 'logout'): void;
|
||||||
|
}>();
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
@ -0,0 +1,39 @@
|
|||||||
|
import { expect, userEvent, within } from '@storybook/test';
|
||||||
|
|
||||||
|
import type { Meta, StoryObj } from '@storybook/vue3-webpack5';
|
||||||
|
|
||||||
|
import MyPage from './Page.vue';
|
||||||
|
|
||||||
|
const meta = {
|
||||||
|
title: 'Example/Page',
|
||||||
|
component: MyPage,
|
||||||
|
render: () => ({
|
||||||
|
components: { MyPage },
|
||||||
|
template: '<my-page />',
|
||||||
|
}),
|
||||||
|
parameters: {
|
||||||
|
// More on how to position stories at: https://storybook.js.org/docs/configure/story-layout
|
||||||
|
layout: 'fullscreen',
|
||||||
|
},
|
||||||
|
// This component will have an automatically generated docsPage entry: https://storybook.js.org/docs/writing-docs/autodocs
|
||||||
|
tags: ['autodocs'],
|
||||||
|
} satisfies Meta<typeof MyPage>;
|
||||||
|
|
||||||
|
export default meta;
|
||||||
|
type Story = StoryObj<typeof meta>;
|
||||||
|
|
||||||
|
// More on component testing: https://storybook.js.org/docs/writing-tests/component-testing
|
||||||
|
export const LoggedIn: Story = {
|
||||||
|
play: async ({ canvasElement }: any) => {
|
||||||
|
const canvas = within(canvasElement);
|
||||||
|
const loginButton = canvas.getByRole('button', { name: /Log in/i });
|
||||||
|
await expect(loginButton).toBeInTheDocument();
|
||||||
|
await userEvent.click(loginButton);
|
||||||
|
await expect(loginButton).not.toBeInTheDocument();
|
||||||
|
|
||||||
|
const logoutButton = canvas.getByRole('button', { name: /Log out/i });
|
||||||
|
await expect(logoutButton).toBeInTheDocument();
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const LoggedOut: Story = {};
|
73
code/frameworks/vue3-webpack5/template/cli/ts-4-9/Page.vue
Normal file
73
code/frameworks/vue3-webpack5/template/cli/ts-4-9/Page.vue
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
<template>
|
||||||
|
<article>
|
||||||
|
<my-header :user="user" @login="onLogin" @logout="onLogout" @create-account="onCreateAccount" />
|
||||||
|
|
||||||
|
<section class="storybook-page">
|
||||||
|
<h2>Pages in Storybook</h2>
|
||||||
|
<p>
|
||||||
|
We recommend building UIs with a
|
||||||
|
<a href="https://componentdriven.org" target="_blank" rel="noopener noreferrer">
|
||||||
|
<strong>component-driven</strong>
|
||||||
|
</a>
|
||||||
|
process starting with atomic components and ending with pages.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
Render pages with mock data. This makes it easy to build and review page states without
|
||||||
|
needing to navigate to them in your app. Here are some handy patterns for managing page data
|
||||||
|
in Storybook:
|
||||||
|
</p>
|
||||||
|
<ul>
|
||||||
|
<li>
|
||||||
|
Use a higher-level connected component. Storybook helps you compose such data from the
|
||||||
|
"args" of child component stories
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
Assemble data in the page component from your services. You can mock these services out
|
||||||
|
using Storybook.
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
<p>
|
||||||
|
Get a guided tutorial on component-driven development at
|
||||||
|
<a href="https://storybook.js.org/tutorials/" target="_blank" rel="noopener noreferrer"
|
||||||
|
>Storybook tutorials</a
|
||||||
|
>
|
||||||
|
. Read more in the
|
||||||
|
<a href="https://storybook.js.org/docs" target="_blank" rel="noopener noreferrer">docs</a>
|
||||||
|
.
|
||||||
|
</p>
|
||||||
|
<div class="tip-wrapper">
|
||||||
|
<span class="tip">Tip</span>
|
||||||
|
Adjust the width of the canvas with the
|
||||||
|
<svg width="10" height="10" viewBox="0 0 12 12" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<g fill="none" fill-rule="evenodd">
|
||||||
|
<path
|
||||||
|
d="M1.5 5.2h4.8c.3 0 .5.2.5.4v5.1c-.1.2-.3.3-.4.3H1.4a.5.5 0 01-.5-.4V5.7c0-.3.2-.5.5-.5zm0-2.1h6.9c.3 0 .5.2.5.4v7a.5.5 0 01-1 0V4H1.5a.5.5 0 010-1zm0-2.1h9c.3 0 .5.2.5.4v9.1a.5.5 0 01-1 0V2H1.5a.5.5 0 010-1zm4.3 5.2H2V10h3.8V6.2z"
|
||||||
|
id="a"
|
||||||
|
fill="#999"
|
||||||
|
/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
Viewports addon in the toolbar
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</article>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import './page.css';
|
||||||
|
import MyHeader from './Header.vue';
|
||||||
|
|
||||||
|
import { ref } from 'vue';
|
||||||
|
|
||||||
|
const user = ref<{ name: string } | null>(null);
|
||||||
|
|
||||||
|
const onLogin = () => {
|
||||||
|
user.value = { name: 'Jane Doe' };
|
||||||
|
};
|
||||||
|
const onLogout = () => {
|
||||||
|
user.value = null;
|
||||||
|
};
|
||||||
|
const onCreateAccount = () => {
|
||||||
|
user.value = { name: 'Jane Doe' };
|
||||||
|
};
|
||||||
|
</script>
|
Loading…
x
Reference in New Issue
Block a user