mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-05 06:41:17 +08:00
feat: addon-ondevice-actions
This adds an ondevice actions addon that shows a logger for the actions on device.
This commit is contained in:
parent
165a5f6955
commit
8292cea322
@ -1,4 +1,5 @@
|
||||
export * from './constants';
|
||||
export * from './models';
|
||||
export * from './preview';
|
||||
|
||||
if (module && module.hot && module.hot.decline) {
|
||||
|
32
addons/ondevice-actions/README.md
Normal file
32
addons/ondevice-actions/README.md
Normal file
@ -0,0 +1,32 @@
|
||||
# Storybook Actions Addon for react-native
|
||||
|
||||
Storybook Actions Addon allows you to log events/actions inside stories in [Storybook](https://storybook.js.org).
|
||||
|
||||
[Framework Support](https://github.com/storybooks/storybook/blob/master/ADDONS_SUPPORT.md)
|
||||
|
||||
**This addon is a wrapper for addon [@storybook/addon-actions](https://github.com/storybooks/storybook/blob/master/addons/actions).
|
||||
Refer to its documentation to understand how to use actions**
|
||||
|
||||
## Installation
|
||||
|
||||
```sh
|
||||
yarn add -D @storybook/addon-ondevice-actions @storybook/addon-actions
|
||||
```
|
||||
|
||||
## Configuration
|
||||
|
||||
Create a file called `rn-addons.js` in your storybook config.
|
||||
|
||||
Add following content to it:
|
||||
|
||||
```js
|
||||
import '@storybook/addon-ondevice-actions/register';
|
||||
```
|
||||
|
||||
Then import `rn-addons.js` next to your `getStorybookUI` call.
|
||||
|
||||
```js
|
||||
import './rn-addons';
|
||||
```
|
||||
|
||||
See [@storybook/addon-actions](https://github.com/storybooks/storybook/blob/master/addons/actions) to learn how to write stories with actions and the [crna-kitchen-sink app](../../examples-native/crna-kitchen-sink) for more examples.
|
47
addons/ondevice-actions/package.json
Normal file
47
addons/ondevice-actions/package.json
Normal file
@ -0,0 +1,47 @@
|
||||
{
|
||||
"name": "@storybook/addon-ondevice-actions",
|
||||
"version": "5.1.0-alpha.29",
|
||||
"description": "Action Logger addon for storybook",
|
||||
"keywords": [
|
||||
"storybook"
|
||||
],
|
||||
"homepage": "https://github.com/storybooks/storybook/tree/master/addons/actions",
|
||||
"bugs": {
|
||||
"url": "https://github.com/storybooks/storybook/issues"
|
||||
},
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/storybooks/storybook.git"
|
||||
},
|
||||
"license": "MIT",
|
||||
"main": "dist/index.js",
|
||||
"jsnext:main": "src/index.js",
|
||||
"scripts": {
|
||||
"prepare": "node ../../scripts/prepare.js"
|
||||
},
|
||||
"dependencies": {
|
||||
"@storybook/addons": "5.1.0-alpha.29",
|
||||
"@storybook/api": "5.1.0-alpha.29",
|
||||
"@storybook/core-events": "5.1.0-alpha.29",
|
||||
"core-js": "^2.5.7",
|
||||
"fast-deep-equal": "^2.0.1",
|
||||
"global": "^4.3.2",
|
||||
"lodash": "^4.17.11",
|
||||
"make-error": "^1.3.5",
|
||||
"prop-types": "^15.6.2",
|
||||
"react-inspector": "^2.3.0",
|
||||
"uuid": "^3.3.2"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@storybook/addon-actions": "5.1.0-alpha.29"
|
||||
},
|
||||
"peerDependencies": {
|
||||
"@storybook/addon-actions": "*",
|
||||
"react": "*",
|
||||
"react-native": "*"
|
||||
},
|
||||
"publishConfig": {
|
||||
"access": "public"
|
||||
},
|
||||
"gitHead": "fbd7ee4c80df437fed4bdc6e11140733fd450080"
|
||||
}
|
1
addons/ondevice-actions/register.js
Normal file
1
addons/ondevice-actions/register.js
Normal file
@ -0,0 +1 @@
|
||||
require('./dist').register();
|
173
addons/ondevice-actions/src/components/ActionLogger/Inspect.tsx
Normal file
173
addons/ondevice-actions/src/components/ActionLogger/Inspect.tsx
Normal file
@ -0,0 +1,173 @@
|
||||
import React from 'react';
|
||||
import { Button, View, Text } from 'react-native';
|
||||
|
||||
const theme = {
|
||||
OBJECT_PREVIEW_ARRAY_MAX_PROPERTIES: 10,
|
||||
OBJECT_PREVIEW_OBJECT_MAX_PROPERTIES: 5,
|
||||
OBJECT_NAME_COLOR: 'rgb(136, 19, 145)',
|
||||
OBJECT_VALUE_NULL_COLOR: 'rgb(128, 128, 128)',
|
||||
OBJECT_VALUE_UNDEFINED_COLOR: 'rgb(128, 128, 128)',
|
||||
OBJECT_VALUE_REGEXP_COLOR: 'rgb(196, 26, 22)',
|
||||
OBJECT_VALUE_STRING_COLOR: 'rgb(196, 26, 22)',
|
||||
OBJECT_VALUE_SYMBOL_COLOR: 'rgb(196, 26, 22)',
|
||||
OBJECT_VALUE_NUMBER_COLOR: 'rgb(28, 0, 207)',
|
||||
OBJECT_VALUE_BOOLEAN_COLOR: 'rgb(28, 0, 207)',
|
||||
OBJECT_VALUE_FUNCTION_PREFIX_COLOR: 'rgb(13, 34, 170)',
|
||||
|
||||
ARROW_COLOR: '#6e6e6e',
|
||||
ARROW_MARGIN_RIGHT: 3,
|
||||
ARROW_FONT_SIZE: 12,
|
||||
ARROW_ANIMATION_DURATION: '0',
|
||||
};
|
||||
|
||||
class Inspect extends React.Component<{ name?: string; value: any }, { expanded: boolean }> {
|
||||
state = { expanded: false };
|
||||
render() {
|
||||
const { name, value } = this.props;
|
||||
const { expanded } = this.state;
|
||||
const toggle = (
|
||||
<View style={{ width: 40, height: 40 }}>
|
||||
{name &&
|
||||
((Array.isArray(value) && value.length) ||
|
||||
(value &&
|
||||
typeof value === 'object' &&
|
||||
!Array.isArray(value) &&
|
||||
Object.keys(value).length)) ? (
|
||||
<Button
|
||||
onPress={() => this.setState(s => ({ expanded: !s.expanded }))}
|
||||
title={!expanded ? '▶' : '▼'}
|
||||
/>
|
||||
) : null}
|
||||
</View>
|
||||
);
|
||||
|
||||
const nameComponent = name ? (
|
||||
<Text style={{ color: theme.OBJECT_NAME_COLOR }}>{name}</Text>
|
||||
) : null;
|
||||
|
||||
if (Array.isArray(value)) {
|
||||
if (name) {
|
||||
return (
|
||||
<>
|
||||
<View style={{ flexDirection: 'row', alignItems: 'center' }}>
|
||||
{toggle}
|
||||
{nameComponent}
|
||||
<Text>{': ' + (value.length === 0 ? '[]' : expanded ? '[' : '[...]')}</Text>
|
||||
</View>
|
||||
{expanded ? (
|
||||
<View style={{ marginLeft: 40 }}>
|
||||
{value.map((v, i) => (
|
||||
<View key={i} style={{ marginLeft: 40 }}>
|
||||
<Inspect value={v} />
|
||||
</View>
|
||||
))}
|
||||
<View style={{ marginLeft: 20 }}>
|
||||
<Text>{']'}</Text>
|
||||
</View>
|
||||
</View>
|
||||
) : null}
|
||||
</>
|
||||
);
|
||||
}
|
||||
return (
|
||||
<View>
|
||||
<Text>{'['}</Text>
|
||||
{value.map((v, i) => (
|
||||
<View key={i} style={{ marginLeft: 20 }}>
|
||||
<Inspect value={v} />
|
||||
</View>
|
||||
))}
|
||||
<Text>{']'}</Text>
|
||||
</View>
|
||||
);
|
||||
}
|
||||
if (typeof value === 'object' && !(value instanceof RegExp)) {
|
||||
if (name) {
|
||||
return (
|
||||
<>
|
||||
<View style={{ flexDirection: 'row', alignItems: 'center' }}>
|
||||
{toggle}
|
||||
{nameComponent}
|
||||
<Text>
|
||||
{': ' + (Object.keys(value).length === 0 ? '{}' : expanded ? '{' : '{...}')}
|
||||
</Text>
|
||||
</View>
|
||||
{expanded ? (
|
||||
<View style={{ marginLeft: 40 }}>
|
||||
{Object.entries(value).map(([key, v]) => (
|
||||
<View key={key}>
|
||||
<Inspect name={key} value={v} />
|
||||
</View>
|
||||
))}
|
||||
<View style={{ marginLeft: 20 }}>
|
||||
<Text>{'}'}</Text>
|
||||
</View>
|
||||
</View>
|
||||
) : null}
|
||||
</>
|
||||
);
|
||||
}
|
||||
return (
|
||||
<View>
|
||||
<Text>{'{'}</Text>
|
||||
{Object.entries(value).map(([key, v]) => (
|
||||
<View key={key}>
|
||||
<Inspect name={key} value={v} />
|
||||
</View>
|
||||
))}
|
||||
<Text>{'}'}</Text>
|
||||
</View>
|
||||
);
|
||||
}
|
||||
if (name) {
|
||||
return (
|
||||
<View style={{ flexDirection: 'row', alignItems: 'center' }}>
|
||||
{toggle}
|
||||
{nameComponent}
|
||||
<Text>{': '}</Text>
|
||||
<Value value={value} />
|
||||
</View>
|
||||
);
|
||||
}
|
||||
return (
|
||||
<View>
|
||||
<Value value={value} />
|
||||
</View>
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
function Value({ value }: { value: any }) {
|
||||
if (value === null) {
|
||||
return <Text style={{ color: theme.OBJECT_VALUE_NULL_COLOR }}>null</Text>;
|
||||
}
|
||||
if (value === undefined) {
|
||||
return <Text style={{ color: theme.OBJECT_VALUE_UNDEFINED_COLOR }}>undefined</Text>;
|
||||
}
|
||||
if (value instanceof RegExp) {
|
||||
return (
|
||||
<Text style={{ color: theme.OBJECT_VALUE_REGEXP_COLOR }}>
|
||||
{'/' + value.source + '/' + value.flags}
|
||||
</Text>
|
||||
);
|
||||
}
|
||||
switch (typeof value) {
|
||||
case 'string':
|
||||
return (
|
||||
<Text style={{ color: theme.OBJECT_VALUE_STRING_COLOR }}>{JSON.stringify(value)}</Text>
|
||||
);
|
||||
case 'number':
|
||||
return (
|
||||
<Text style={{ color: theme.OBJECT_VALUE_NUMBER_COLOR }}>{JSON.stringify(value)}</Text>
|
||||
);
|
||||
case 'boolean':
|
||||
return (
|
||||
<Text style={{ color: theme.OBJECT_VALUE_BOOLEAN_COLOR }}>{JSON.stringify(value)}</Text>
|
||||
);
|
||||
case 'function':
|
||||
return <Text style={{ color: theme.OBJECT_VALUE_FUNCTION_PREFIX_COLOR }}>[Function]</Text>;
|
||||
}
|
||||
return <Text>{JSON.stringify(value)}</Text>;
|
||||
}
|
||||
|
||||
export default Inspect;
|
@ -0,0 +1,31 @@
|
||||
import React from 'react';
|
||||
import { Button, View, Text, ScrollView } from 'react-native';
|
||||
import { ActionDisplay } from '@storybook/addon-actions';
|
||||
import Inspect from './Inspect';
|
||||
|
||||
interface ActionLoggerProps {
|
||||
actions: ActionDisplay[];
|
||||
onClear: () => void;
|
||||
}
|
||||
|
||||
export const ActionLogger = ({ actions, onClear }: ActionLoggerProps) => (
|
||||
<ScrollView>
|
||||
<ScrollView horizontal>
|
||||
<View>
|
||||
{actions.map((action: ActionDisplay) => (
|
||||
<View key={action.id} style={{ flexDirection: 'row' }}>
|
||||
<View>{action.count > 1 ? <Text>{action.count}</Text> : null}</View>
|
||||
<View style={{ flexGrow: 1 }}>
|
||||
<Inspect name={action.data.name} value={action.data.args || action.data} />
|
||||
</View>
|
||||
</View>
|
||||
))}
|
||||
</View>
|
||||
</ScrollView>
|
||||
<View>
|
||||
<Button onPress={onClear} title="CLEAR" />
|
||||
</View>
|
||||
</ScrollView>
|
||||
);
|
||||
|
||||
export default ActionLogger;
|
@ -0,0 +1,86 @@
|
||||
import React, { Component } from 'react';
|
||||
import deepEqual from 'fast-deep-equal';
|
||||
|
||||
import { API } from '@storybook/api';
|
||||
import { STORY_RENDERED } from '@storybook/core-events';
|
||||
import { EVENT_ID, ActionDisplay } from '@storybook/addon-actions';
|
||||
|
||||
import { ActionLogger as ActionLoggerComponent } from '../../components/ActionLogger';
|
||||
|
||||
interface ActionLoggerProps {
|
||||
active: boolean;
|
||||
api: API;
|
||||
}
|
||||
|
||||
interface ActionLoggerState {
|
||||
actions: ActionDisplay[];
|
||||
}
|
||||
|
||||
const safeDeepEqual = (a: any, b: any): boolean => {
|
||||
try {
|
||||
return deepEqual(a, b);
|
||||
} catch (e) {
|
||||
return false;
|
||||
}
|
||||
};
|
||||
|
||||
export default class ActionLogger extends Component<ActionLoggerProps, ActionLoggerState> {
|
||||
private mounted: boolean;
|
||||
|
||||
constructor(props: ActionLoggerProps) {
|
||||
super(props);
|
||||
|
||||
this.state = { actions: [] };
|
||||
}
|
||||
|
||||
componentDidMount() {
|
||||
this.mounted = true;
|
||||
const { api } = this.props;
|
||||
|
||||
api.on(EVENT_ID, this.addAction);
|
||||
api.on(STORY_RENDERED, this.handleStoryChange);
|
||||
}
|
||||
|
||||
componentWillUnmount() {
|
||||
this.mounted = false;
|
||||
const { api } = this.props;
|
||||
|
||||
api.off(STORY_RENDERED, this.handleStoryChange);
|
||||
api.off(EVENT_ID, this.addAction);
|
||||
}
|
||||
|
||||
handleStoryChange = () => {
|
||||
const { actions } = this.state;
|
||||
if (actions.length > 0 && actions[0].options.clearOnStoryChange) {
|
||||
this.clearActions();
|
||||
}
|
||||
};
|
||||
|
||||
addAction = (action: ActionDisplay) => {
|
||||
this.setState((prevState: ActionLoggerState) => {
|
||||
const actions = [...prevState.actions];
|
||||
const previous = actions.length && actions[0];
|
||||
if (previous && safeDeepEqual(previous.data, action.data)) {
|
||||
previous.count++; // eslint-disable-line
|
||||
} else {
|
||||
action.count = 1; // eslint-disable-line
|
||||
actions.unshift(action);
|
||||
}
|
||||
return { actions: actions.slice(0, action.options.limit) };
|
||||
});
|
||||
};
|
||||
|
||||
clearActions = () => {
|
||||
this.setState({ actions: [] });
|
||||
};
|
||||
|
||||
render() {
|
||||
const { actions = [] } = this.state;
|
||||
const { active } = this.props;
|
||||
const props = {
|
||||
actions,
|
||||
onClear: this.clearActions,
|
||||
};
|
||||
return active ? <ActionLoggerComponent {...props} /> : null;
|
||||
}
|
||||
}
|
13
addons/ondevice-actions/src/index.tsx
Normal file
13
addons/ondevice-actions/src/index.tsx
Normal file
@ -0,0 +1,13 @@
|
||||
import React from 'react';
|
||||
import addons from '@storybook/addons';
|
||||
import { ADDON_ID, PANEL_ID } from '@storybook/addon-actions';
|
||||
import ActionLogger from './containers/ActionLogger';
|
||||
|
||||
export function register() {
|
||||
addons.register(ADDON_ID, api => {
|
||||
addons.addPanel(PANEL_ID, {
|
||||
title: 'Actions',
|
||||
render: ({ active, key }) => <ActionLogger key={key} api={api} active={active} />,
|
||||
});
|
||||
});
|
||||
}
|
13
addons/ondevice-actions/tsconfig.json
Normal file
13
addons/ondevice-actions/tsconfig.json
Normal file
@ -0,0 +1,13 @@
|
||||
{
|
||||
"extends": "../../tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"rootDir": "./src",
|
||||
"types": ["webpack-env"]
|
||||
},
|
||||
"include": [
|
||||
"src/**/*"
|
||||
],
|
||||
"exclude": [
|
||||
"src/__tests__/**/*"
|
||||
]
|
||||
}
|
@ -16817,7 +16817,7 @@ make-dir@^2.0.0:
|
||||
pify "^4.0.1"
|
||||
semver "^5.6.0"
|
||||
|
||||
make-error@1.x, make-error@^1.1.1:
|
||||
make-error@1.x, make-error@^1.1.1, make-error@^1.3.5:
|
||||
version "1.3.5"
|
||||
resolved "https://registry.yarnpkg.com/make-error/-/make-error-1.3.5.tgz#efe4e81f6db28cadd605c70f29c831b58ef776c8"
|
||||
integrity sha512-c3sIjNUow0+8swNwVpqoH4YCShKNFkMaw6oH1mNS2haDZQqkeZFlHS3dhoeEbKKmJB4vXpJucU6oH75aDYeE9g==
|
||||
@ -21337,7 +21337,7 @@ react-input-autosize@^2.2.1:
|
||||
dependencies:
|
||||
prop-types "^15.5.8"
|
||||
|
||||
react-inspector@^2.3.1:
|
||||
react-inspector@^2.3.0, react-inspector@^2.3.1:
|
||||
version "2.3.1"
|
||||
resolved "https://registry.yarnpkg.com/react-inspector/-/react-inspector-2.3.1.tgz#f0eb7f520669b545b441af9d38ec6d706e5f649c"
|
||||
integrity sha512-tUUK7t3KWgZEIUktOYko5Ic/oYwvjEvQUFAGC1UeMeDaQ5za2yZFtItJa2RTwBJB//NxPr000WQK6sEbqC6y0Q==
|
||||
|
Loading…
x
Reference in New Issue
Block a user