forked from akveo/react-native-ui-kitten
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathstyled.tsx
181 lines (157 loc) · 5.95 KB
/
styled.tsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
/**
* @license
* Copyright Akveo. All Rights Reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*/
import React from 'react';
import hoistNonReactStatics from 'hoist-non-react-statics';
import { ThemeStyleType } from '@eva-design/dss';
import { StyleConsumerService } from './styleConsumer.service';
import {
Interaction,
StyleType,
} from './style.service';
import { MappingContext } from '../mapping/mappingContext';
import { ThemeContext } from '../theme/themeContext';
import { ThemeType } from '../theme/theme.service';
interface PrivateRefProps<T = React.ReactInstance> {
forwardedRef?: React.Ref<T>;
}
export interface EvaProp {
theme?: ThemeType;
style?: StyleType;
dispatch?: (interaction: Interaction[]) => void;
}
export interface StyledComponentProps {
appearance?: string;
eva?: EvaProp;
}
interface State {
interaction: Interaction[];
}
type WrappedComponentProps = any;
// TODO: Better types. React.ComponentType<WrappedComponentProps>?
type WrappedComponent = any;
type StyledComponent = any;
/**
* `@styled` is a High Order Function to apply style mapping on a component.
* Used by all UI Kitten components and can be used when building custom components with Eva.
*
* @property {string} appearance - Appearance of component. Default is provided by mapping.
*
* @property {EvaProp} eva - Additional property injected to all `styled` components. Includes following properties:
* `theme` - current theme,
* `style` - style provided by Eva,
* `dispatch` - Function for requesting styles from Eva for current component state.
*
* @param name - Type: {string}. Name of the component in mapping.json.
*
* @overview-example StyledComponentSimpleUsage
* The below examples demonstrate how components can be styled with Eva.
* See [the guide](design-system/custom-component-mapping) for more details.
*
* @overview-example StyledComponentEvaProp
* A styled function injects `eva` property into props of decorated component, where
* theme - a current theme,
* styles - a styles object provided by Eva
* dispatch - a function to request styles for a particular state.
*
* ```
* interface EvaProp {
* theme: ThemeType;
* style: StyleType;
* dispatch: (interaction: Interaction[]) => void;
* }
* ```
*
* @overview-example StyledComponentStates
* Styled components may describe its style for a particular state.
*
* Let's say we don't like the standard behavior of TouchableOpacity when it's pressed and
* we want the component to change it's color rather being highlighted.
*
* We define an active state in `meta` key and in mapping, so that component will change `backgroundColor`,
* when `active` is requested. To do this, we call `dispatch` function when Touchable is pressed.
* Then, when touch is released, we request nothing, which stands for `default`.
*
* @overview-example StyledComponentVariants
* Now let's say we want to have a `status` property to make component change its color, for example,
* within the forms validation. Furthermore, we want to control its color for both statuses when it is pressed.
*
* The `variantGroups` key in `meta` defines all properties that can be applied by component.
* Each key in variant group is a string value that can be passed to component props.
*/
export const styled = (name: string): StyledComponent => {
return (component: WrappedComponent): StyledComponent => {
return styleInjector(component, name);
};
};
const styleInjector = (Component: WrappedComponent, name: string): StyledComponent => {
if (!name) {
console.warn('Components annotated with @styled function should also have its in mapping.json.');
return null;
}
class Wrapper extends React.Component<PrivateRefProps, State> {
public state: State = {
interaction: [],
};
private init: boolean = false;
private defaultProps: WrappedComponentProps;
private service: StyleConsumerService;
private onInit = (style: ThemeStyleType, theme: ThemeType): void => {
this.service = new StyleConsumerService(name, style);
this.defaultProps = this.service.createDefaultProps();
this.init = true;
};
private onDispatch = (interaction: Interaction[]): void => {
this.setState({ interaction });
};
private withEvaProp = (sourceProps: WrappedComponentProps,
sourceStyle: ThemeStyleType,
theme: ThemeType): StyledComponentProps => {
const props: WrappedComponentProps = { ...this.defaultProps, ...sourceProps };
const style: StyleType = this.service.createStyleProp(props, sourceStyle, theme, this.state.interaction);
return {
...props,
eva: {
theme,
style,
dispatch: this.onDispatch,
},
};
};
private renderWrappedElement = (style: ThemeStyleType, theme: ThemeType): React.ReactElement => {
if (!this.init) {
this.onInit(style, theme);
}
const { forwardedRef, ...restProps } = this.props;
return (
<Component
{...this.withEvaProp(restProps as any, style, theme)}
ref={forwardedRef}
/>
);
};
public render(): React.ReactElement {
return (
<MappingContext.Consumer>{(style: ThemeStyleType): React.ReactElement => (
<ThemeContext.Consumer>{(theme: ThemeType): React.ReactElement => {
return this.renderWrappedElement(style, theme);
}}</ThemeContext.Consumer>
)}</MappingContext.Consumer>
);
}
}
const WrappingElement = (props: WrappedComponentProps, ref: React.Ref<React.ReactInstance>): React.ReactElement => {
return (
<Wrapper
{...props}
forwardedRef={ref}
/>
);
};
const ResultComponent = React.forwardRef<React.ReactInstance, PrivateRefProps>(WrappingElement);
ResultComponent.displayName = Component.displayName || Component.name;
hoistNonReactStatics(ResultComponent, Component);
return ResultComponent;
};