feat: add basic setup for storybook

This commit is contained in:
p-sw 2024-05-18 21:11:11 +09:00
parent 20476b8687
commit f7013ae546
12 changed files with 10590 additions and 0 deletions

View File

@ -0,0 +1,14 @@
module.exports = {
root: true,
env: { browser: true, es2020: true },
extends: ['eslint:recommended', 'plugin:@typescript-eslint/recommended', 'plugin:react-hooks/recommended', 'plugin:storybook/recommended'],
ignorePatterns: ['dist', '.eslintrc.cjs'],
parser: '@typescript-eslint/parser',
plugins: ['react-refresh'],
rules: {
'react-refresh/only-export-components': [
'warn',
{ allowConstantExport: true },
],
},
}

26
packages/react/.gitignore vendored Normal file
View File

@ -0,0 +1,26 @@
# Logs
logs
*.log
npm-debug.log*
yarn-debug.log*
yarn-error.log*
pnpm-debug.log*
lerna-debug.log*
node_modules
dist
dist-ssr
*.local
# Editor directories and files
.vscode/*
!.vscode/extensions.json
.idea
.DS_Store
*.suo
*.ntvs*
*.njsproj
*.sln
*.sw?
*storybook.log

View File

@ -0,0 +1,30 @@
import type { StorybookConfig } from "@storybook/react-vite";
import { join, dirname } from "path";
/**
* This function is used to resolve the absolute path of a package.
* It is needed in projects that use Yarn PnP or are set up within a monorepo.
*/
function getAbsolutePath(value: string): any {
return dirname(require.resolve(join(value, "package.json")));
}
const config: StorybookConfig = {
stories: [
"../stories/**/*.mdx",
"../stories/**/*.stories.@(js|jsx|mjs|ts|tsx)",
],
addons: [
getAbsolutePath("@storybook/addon-onboarding"),
getAbsolutePath("@storybook/addon-links"),
getAbsolutePath("@storybook/addon-essentials"),
getAbsolutePath("@chromatic-com/storybook"),
getAbsolutePath("@storybook/addon-interactions"),
"@storybook/addon-themes"
],
framework: {
name: getAbsolutePath("@storybook/react-vite"),
options: {},
},
};
export default config;

View File

@ -0,0 +1,105 @@
import React from "react";
import type { Preview } from "@storybook/react";
import type { DecoratorFunction, Renderer } from "@storybook/types";
import { useEffect, addons, useParameter } from "@storybook/preview-api";
import "../src/tailwind.css";
import {
Controls,
Description,
Primary,
Stories,
Subtitle,
Title,
} from "@storybook/blocks";
const classStringToArray = (classString: string) =>
classString.split(" ").filter(Boolean);
const withThemeByClassName = <TRenderer extends Renderer = any>(
themes: Record<string, string>,
defaultTheme: string,
parentSelector: string
): DecoratorFunction<TRenderer> => {
addons.getChannel().emit("storybook/themes/REGISTER_THEMES", {
defaultTheme,
themes: Object.keys(themes),
});
return (storyFn, context) => {
const { themeOverride } = useParameter<{ themeOverride?: string }>(
"themes",
{}
) as { themeOverride?: string };
const selected = context.globals["theme"] || "";
useEffect(() => {
const selectedThemeName = themeOverride || selected || defaultTheme;
const parentElement = document.querySelectorAll(parentSelector);
if (!parentElement) {
return;
}
Object.entries(themes)
.filter(([themeName]) => themeName !== selectedThemeName)
.forEach(([_, className]) => {
const classes = classStringToArray(className);
if (classes.length > 0) {
parentElement.forEach((element) =>
element.classList.remove(...classes)
);
}
});
const newThemeClasses = classStringToArray(themes[selectedThemeName]);
if (newThemeClasses.length > 0) {
parentElement.forEach((element) => {
console.log(element);
element.classList.add(...newThemeClasses);
});
}
}, [themeOverride, selected, parentSelector]);
return storyFn();
};
};
export const decorators = [
withThemeByClassName(
{
light: "light",
dark: "dark",
},
"dark",
"html,.sbdocs.sbdocs-preview"
),
];
const autoDocsTemplate = () => (
<>
<Title />
<Subtitle />
<Description />
<Primary />
<Controls />
<Stories />
</>
);
const preview: Preview = {
parameters: {
/*controls: {
matchers: {
color: /(background|color)$/i,
date: /Date$/i,
},
},*/
docs: {
page: autoDocsTemplate,
},
},
tags: ["autodocs"],
};
export default preview;

View File

@ -0,0 +1,46 @@
{
"name": "react",
"private": true,
"version": "0.0.0",
"type": "module",
"scripts": {
"dev": "vite",
"build": "tsc && vite build",
"lint": "eslint . --ext ts,tsx --report-unused-disable-directives --max-warnings 0",
"preview": "vite preview",
"storybook": "storybook dev -p 6006",
"build-storybook": "storybook build"
},
"dependencies": {
"react": "^18.2.0",
"react-dom": "^18.2.0",
"tailwind-merge": "^2.3.0"
},
"devDependencies": {
"@chromatic-com/storybook": "^1.3.5",
"@storybook/addon-essentials": "^8.1.0",
"@storybook/addon-interactions": "^8.1.0",
"@storybook/addon-links": "^8.1.0",
"@storybook/addon-onboarding": "^8.1.0",
"@storybook/addon-themes": "^8.1.0",
"@storybook/blocks": "^8.1.0",
"@storybook/react": "^8.1.0",
"@storybook/react-vite": "^8.1.0",
"@storybook/test": "^8.1.0",
"@types/react": "^18.2.66",
"@types/react-dom": "^18.2.22",
"@typescript-eslint/eslint-plugin": "^7.2.0",
"@typescript-eslint/parser": "^7.2.0",
"@vitejs/plugin-react": "^4.2.1",
"autoprefixer": "^10.4.19",
"eslint": "^8.57.0",
"eslint-plugin-react-hooks": "^4.6.0",
"eslint-plugin-react-refresh": "^0.4.6",
"eslint-plugin-storybook": "^0.8.0",
"postcss": "^8.4.38",
"storybook": "^8.1.0",
"tailwindcss": "^3.4.3",
"typescript": "^5.2.2",
"vite": "^5.2.0"
}
}

35
packages/react/shared.ts Normal file
View File

@ -0,0 +1,35 @@
import { twMerge } from "tailwind-merge";
export function vcn<V extends Record<string, Record<string, string>>>({
base,
variants,
defaults,
}: {
base: string;
variants: V;
defaults: { [VariantKey in keyof V]: keyof V[VariantKey] };
}): (
variantProps: Partial<typeof defaults> & { className?: string }
) => string {
return ({ className, ...variantProps }) => {
return twMerge(
base,
...(
Object.entries(defaults) as [keyof V, keyof V[keyof V]][]
).map<string>(
([variantKey, defaultValue]) =>
variants[variantKey][
(variantProps as unknown as Partial<typeof defaults>)[variantKey] ??
defaultValue
]
),
className
);
};
}
export type VariantProps<F extends ReturnType<typeof vcn>> = F extends (
props: infer P
) => string
? P
: never;

View File

@ -0,0 +1,37 @@
@tailwind base;
@tailwind components;
@tailwind utilities;
@layer base {
:root {
--dark-bg-color: #000;
--dark-text-color: #fff;
--dark-btn-bg-color: #2f2f2f;
--dark-btn-text-color: #fff;
--dark-code-bg-color: #1f1f1f;
}
body {
@apply bg-white dark:bg-[var(--dark-bg-color)] text-black dark:text-[var(--dark-text-color)] transition-colors duration-300;
}
.sbdocs.sbdocs-preview.dark {
@apply dark:bg-[var(--dark-bg-color)] dark:text-[var(--dark-text-color)] transition-colors duration-300;
& .sb-bar {
@apply dark:bg-[var(--dark-bg-color)] dark:text-[var(--dark-text-color)];
}
& .sb-bar button, .docs-story > div:nth-child(2) > button, .docs-story + div > div:nth-child(2) > button {
@apply dark:bg-[var(--dark-btn-bg-color)] dark:text-[var(--dark-btn-text-color)];
}
& .prismjs {
@apply pb-8;
& > div {
@apply dark:bg-[var(--dark-code-bg-color)] rounded-md p-4;
}
}
}
}

View File

@ -0,0 +1,9 @@
/** @type {import('tailwindcss').Config} */
module.exports = {
content: ["./{components,stories,src}/**/*.{js,jsx,ts,tsx,css}"],
darkMode: "selector",
theme: {
extend: {},
},
plugins: [],
};

View File

@ -0,0 +1,25 @@
{
"compilerOptions": {
"target": "ES2020",
"useDefineForClassFields": true,
"lib": ["ES2020", "DOM", "DOM.Iterable"],
"module": "ESNext",
"skipLibCheck": true,
/* Bundler mode */
"moduleResolution": "bundler",
"allowImportingTsExtensions": true,
"resolveJsonModule": true,
"isolatedModules": true,
"noEmit": true,
"jsx": "react-jsx",
/* Linting */
"strict": true,
"noUnusedLocals": true,
"noUnusedParameters": true,
"noFallthroughCasesInSwitch": true
},
"include": ["components", "stories", "./stories.ts", "src"],
"references": [{ "path": "./tsconfig.node.json" }]
}

View File

@ -0,0 +1,11 @@
{
"compilerOptions": {
"composite": true,
"skipLibCheck": true,
"module": "ESNext",
"moduleResolution": "bundler",
"allowSyntheticDefaultImports": true,
"strict": true
},
"include": ["vite.config.ts"]
}

View File

@ -0,0 +1,13 @@
import { defineConfig } from "vite";
import react from "@vitejs/plugin-react";
import tailwindcss from "tailwindcss";
// https://vitejs.dev/config/
export default defineConfig({
plugins: [react()],
css: {
postcss: {
plugins: [tailwindcss()],
},
},
});

10239
yarn.lock Normal file

File diff suppressed because it is too large Load Diff