@draftbit/core 47.0.0-alpha.0 → 47.0.0-df8093.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/commonjs/components/Banner.js +168 -0
- package/lib/commonjs/components/Banner.js.map +1 -0
- package/lib/commonjs/components/Stepper.js +1 -0
- package/lib/commonjs/index.js +14 -0
- package/lib/commonjs/index.js.map +1 -1
- package/lib/commonjs/mappings/View.js.map +1 -1
- package/lib/commonjs/mappings/WebView.js.map +1 -1
- package/lib/module/components/Banner.js +158 -0
- package/lib/module/components/Banner.js.map +1 -0
- package/lib/module/components/Stepper.js +1 -0
- package/lib/module/index.js +2 -0
- package/lib/module/index.js.map +1 -1
- package/lib/typescript/src/components/Banner.d.ts +23 -0
- package/lib/typescript/src/components/Banner.d.ts.map +1 -0
- package/lib/typescript/src/index.d.ts +2 -0
- package/lib/typescript/src/index.d.ts.map +1 -1
- package/package.json +3 -3
- package/src/Provider.js +9 -0
- package/src/components/Accordion/AccordionGroup.js +44 -0
- package/src/components/Accordion/AccordionItem.js +32 -0
- package/src/components/Accordion/index.js +2 -0
- package/src/components/ActionSheet/ActionSheet.js +45 -0
- package/src/components/ActionSheet/ActionSheetCancel.js +6 -0
- package/src/components/ActionSheet/ActionSheetItem.js +30 -0
- package/src/components/ActionSheet/index.js +3 -0
- package/src/components/Banner.js +109 -0
- package/src/components/Banner.tsx +204 -0
- package/src/components/Button.js +114 -0
- package/src/components/Checkbox/Checkbox.js +63 -0
- package/src/components/Checkbox/CheckboxGroup.js +21 -0
- package/src/components/Checkbox/CheckboxGroupRow.js +77 -0
- package/src/components/Checkbox/CheckboxRow.js +78 -0
- package/src/components/Checkbox/context.js +14 -0
- package/src/components/Checkbox/index.js +3 -0
- package/src/components/Config.js +64 -0
- package/src/components/DatePicker/DatePicker.js +368 -0
- package/src/components/DatePicker/DatePickerComponent.js +13 -0
- package/src/components/DatePicker/DatePickerComponent.web.js +30 -0
- package/src/components/DatePicker/DatePickerComponentType.js +1 -0
- package/src/components/DeprecatedButton.js +95 -0
- package/src/components/Elevation.js +20 -0
- package/src/components/FormRow.js +19 -0
- package/src/components/IconButton.js +35 -0
- package/src/components/Layout.js +50 -0
- package/src/components/NumberInput.js +49 -0
- package/src/components/Picker/Picker.js +267 -0
- package/src/components/Picker/PickerComponent.android.js +69 -0
- package/src/components/Picker/PickerComponent.ios.js +79 -0
- package/src/components/Picker/PickerComponent.web.js +70 -0
- package/src/components/Picker/PickerTypes.js +1 -0
- package/src/components/Portal/Portal.js +35 -0
- package/src/components/Portal/PortalConsumer.js +27 -0
- package/src/components/Portal/PortalHost.js +107 -0
- package/src/components/Portal/PortalManager.js +32 -0
- package/src/components/RadioButton/RadioButton.js +17 -0
- package/src/components/RadioButton/RadioButtonFieldGroup.js +17 -0
- package/src/components/RadioButton/RadioButtonGroup.js +43 -0
- package/src/components/RadioButton/RadioButtonRow.js +76 -0
- package/src/components/RadioButton/context.js +14 -0
- package/src/components/RadioButton/index.js +4 -0
- package/src/components/SVG.js +13 -0
- package/src/components/ScreenContainer.js +34 -0
- package/src/components/Slider.js +63 -0
- package/src/components/StarRating.js +50 -0
- package/src/components/Stepper.js +39 -0
- package/src/components/Surface.js +32 -0
- package/src/components/Swiper/Swiper.js +29 -0
- package/src/components/Swiper/SwiperItem.js +9 -0
- package/src/components/Swiper/index.js +2 -0
- package/src/components/Switch.js +56 -0
- package/src/components/Text.js +33 -0
- package/src/components/TextField.js +428 -0
- package/src/components/Touchable.js +12 -0
- package/src/components/Touchable.web.js +2 -0
- package/src/constants.js +10 -0
- package/src/hooks.js +12 -0
- package/src/index.js +35 -0
- package/src/index.tsx +2 -0
- package/src/interfaces/Icon.js +8 -0
- package/src/mappings/Accordion.js +41 -0
- package/src/mappings/AccordionItem.js +16 -0
- package/src/mappings/ActionSheet.js +13 -0
- package/src/mappings/ActionSheetCancel.js +19 -0
- package/src/mappings/ActionSheetItem.js +23 -0
- package/src/mappings/ActivityIndicator.js +58 -0
- package/src/mappings/AudioPlayer.js +20 -0
- package/src/mappings/BlurView.js +42 -0
- package/src/mappings/Button.js +87 -0
- package/src/mappings/Checkbox.js +46 -0
- package/src/mappings/CheckboxGroup.js +26 -0
- package/src/mappings/CheckboxRow.js +61 -0
- package/src/mappings/CustomCode.js +8 -0
- package/src/mappings/DatePicker.js +157 -0
- package/src/mappings/Fetch.js +13 -0
- package/src/mappings/FlashList.js +33 -0
- package/src/mappings/FlatList.js +24 -0
- package/src/mappings/Icon.js +32 -0
- package/src/mappings/IconButton.js +35 -0
- package/src/mappings/Image.js +35 -0
- package/src/mappings/ImageBackground.js +29 -0
- package/src/mappings/KeyboardAvoidingView.js +41 -0
- package/src/mappings/KeyboardAwareScrollView.js +50 -0
- package/src/mappings/Layout.js +200 -0
- package/src/mappings/LinearGradient.js +77 -0
- package/src/mappings/MapCallout.js +21 -0
- package/src/mappings/MapMarker.js +47 -0
- package/src/mappings/MapView.js +139 -0
- package/src/mappings/Modal.js +42 -0
- package/src/mappings/NumberInput.js +254 -0
- package/src/mappings/Picker.js +148 -0
- package/src/mappings/RadioButton.js +51 -0
- package/src/mappings/RadioButtonGroup.js +17 -0
- package/src/mappings/RadioButtonRow.js +42 -0
- package/src/mappings/SVG.js +20 -0
- package/src/mappings/SafeAreaView.js +33 -0
- package/src/mappings/ScrollView.js +31 -0
- package/src/mappings/Slider.js +60 -0
- package/src/mappings/StarRating.js +43 -0
- package/src/mappings/Stepper.js +29 -0
- package/src/mappings/Surface.js +14 -0
- package/src/mappings/Swiper.js +60 -0
- package/src/mappings/SwiperItem.js +8 -0
- package/src/mappings/Switch.js +81 -0
- package/src/mappings/Text.js +251 -0
- package/src/mappings/TextArea.js +263 -0
- package/src/mappings/TextField.js +381 -0
- package/src/mappings/TextInput.js +391 -0
- package/src/mappings/Touchable.js +17 -0
- package/src/mappings/Video.js +81 -0
- package/src/mappings/View.js +207 -0
- package/src/mappings/WebView.js +88 -0
- package/src/styles/DarkTheme.js +26 -0
- package/src/styles/DefaultTheme.js +111 -0
- package/src/styles/fonts.js +62 -0
- package/src/styles/overlay.js +60 -0
- package/src/styles/shadow.js +51 -0
- package/src/theming.js +3 -0
- package/src/utilities.js +102 -0
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
import { View, StyleSheet } from "react-native";
|
|
3
|
+
import { SafeAreaView } from "react-native-safe-area-context";
|
|
4
|
+
import omit from "lodash.omit";
|
|
5
|
+
import { Picker as NativePicker } from "@react-native-picker/picker";
|
|
6
|
+
import { withTheme } from "../../theming";
|
|
7
|
+
import Portal from "../Portal/Portal";
|
|
8
|
+
import Button from "../DeprecatedButton";
|
|
9
|
+
import TextField from "../TextField";
|
|
10
|
+
import Touchable from "../Touchable";
|
|
11
|
+
import { extractStyles } from "../../utilities";
|
|
12
|
+
const Picker = ({ Icon, style, options, placeholder, selectedValue, disabled = false, onValueChange = () => { }, theme: { colors }, ...props }) => {
|
|
13
|
+
var _a, _b;
|
|
14
|
+
const { viewStyles: { borderRadius, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
15
|
+
borderWidth, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
16
|
+
borderTopWidth, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
17
|
+
borderRightWidth, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
18
|
+
borderBottomWidth, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
19
|
+
borderLeftWidth, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
20
|
+
borderColor, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
21
|
+
backgroundColor, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
22
|
+
padding, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
23
|
+
paddingTop, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
24
|
+
paddingRight, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
25
|
+
paddingBottom, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
26
|
+
paddingLeft, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
27
|
+
...viewStyles }, } = extractStyles(style);
|
|
28
|
+
const textField = React.useRef(undefined);
|
|
29
|
+
const [pickerVisible, setIsPickerVisible] = React.useState(false);
|
|
30
|
+
const toggleVisibility = () => {
|
|
31
|
+
setIsPickerVisible(!pickerVisible);
|
|
32
|
+
// @ts-ignore
|
|
33
|
+
textField.current.toggleFocus(); // cannot determine if method exists due to component being wrapped in a withTheme()
|
|
34
|
+
};
|
|
35
|
+
const stylesWithoutMargin = style &&
|
|
36
|
+
omit(StyleSheet.flatten(style), [
|
|
37
|
+
"margin",
|
|
38
|
+
"marginTop",
|
|
39
|
+
"marginRight",
|
|
40
|
+
"marginBottom",
|
|
41
|
+
"marginLeft",
|
|
42
|
+
]);
|
|
43
|
+
const selectedLabel = selectedValue &&
|
|
44
|
+
((_b = (_a = options.find((o) => o.value === selectedValue)) === null || _a === void 0 ? void 0 : _a.label) !== null && _b !== void 0 ? _b : selectedValue);
|
|
45
|
+
return (React.createElement(View, { style: [styles.container, viewStyles] },
|
|
46
|
+
React.createElement(Touchable, { disabled: disabled, onPress: toggleVisibility },
|
|
47
|
+
React.createElement(TextField, { ...props, value: String(selectedLabel), placeholder: placeholder,
|
|
48
|
+
// @ts-ignore
|
|
49
|
+
ref: textField, disabled: disabled, pointerEvents: "none",
|
|
50
|
+
// @ts-expect-error
|
|
51
|
+
style: stylesWithoutMargin, Icon: Icon })),
|
|
52
|
+
pickerVisible && (React.createElement(Portal, null,
|
|
53
|
+
React.createElement(View, { style: [styles.picker, { backgroundColor: colors.divider }] },
|
|
54
|
+
React.createElement(SafeAreaView, { style: styles.pickerContainer },
|
|
55
|
+
React.createElement(Button, { Icon: Icon, type: "text", onPress: toggleVisibility, style: styles.closeButton }, "Close"),
|
|
56
|
+
React.createElement(NativePicker, { style: { backgroundColor: "white" }, selectedValue: selectedValue, onValueChange: onValueChange }, options.map((o) => (React.createElement(NativePicker.Item, { label: o.label, value: o.value, key: o.value }))))))))));
|
|
57
|
+
};
|
|
58
|
+
const styles = StyleSheet.create({
|
|
59
|
+
container: {
|
|
60
|
+
alignSelf: "stretch",
|
|
61
|
+
},
|
|
62
|
+
picker: {
|
|
63
|
+
position: "absolute",
|
|
64
|
+
bottom: 0,
|
|
65
|
+
left: 0,
|
|
66
|
+
right: 0,
|
|
67
|
+
flexDirection: "row",
|
|
68
|
+
justifyContent: "center",
|
|
69
|
+
},
|
|
70
|
+
pickerContainer: {
|
|
71
|
+
backgroundColor: "white",
|
|
72
|
+
flexDirection: "column",
|
|
73
|
+
width: "100%",
|
|
74
|
+
},
|
|
75
|
+
closeButton: {
|
|
76
|
+
alignSelf: "flex-end",
|
|
77
|
+
},
|
|
78
|
+
});
|
|
79
|
+
export default withTheme(Picker);
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
import { View, StyleSheet } from "react-native";
|
|
3
|
+
import { Picker as NativePicker } from "@react-native-picker/picker";
|
|
4
|
+
import omit from "lodash.omit";
|
|
5
|
+
import { withTheme } from "../../theming";
|
|
6
|
+
import { extractStyles } from "../../utilities";
|
|
7
|
+
import TextField from "../TextField";
|
|
8
|
+
import Touchable from "../Touchable";
|
|
9
|
+
const Picker = ({ style, options, placeholder, selectedValue, disabled = false, onValueChange: onValueChangeOverride = () => { }, ...props }) => {
|
|
10
|
+
var _a, _b;
|
|
11
|
+
const { viewStyles: { borderRadius, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
12
|
+
borderWidth, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
13
|
+
borderTopWidth, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
14
|
+
borderRightWidth, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
15
|
+
borderBottomWidth, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
16
|
+
borderLeftWidth, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
17
|
+
borderColor, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
18
|
+
backgroundColor, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
19
|
+
padding, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
20
|
+
paddingTop, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
21
|
+
paddingRight, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
22
|
+
paddingBottom, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
23
|
+
paddingLeft, // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
24
|
+
...viewStyles }, } = extractStyles(style);
|
|
25
|
+
const textField = React.useRef(undefined);
|
|
26
|
+
const onValueChange = (itemValue, itemIndex) => {
|
|
27
|
+
toggleFocus();
|
|
28
|
+
onValueChangeOverride(itemValue, itemIndex);
|
|
29
|
+
};
|
|
30
|
+
const toggleFocus = () => {
|
|
31
|
+
if (!disabled) {
|
|
32
|
+
// @ts-ignore
|
|
33
|
+
textField.current.toggleFocus(); // cannot determine if method exists due to component being wrapped in a withTheme()
|
|
34
|
+
}
|
|
35
|
+
};
|
|
36
|
+
const stylesWithoutMargin = style &&
|
|
37
|
+
omit(StyleSheet.flatten(style), [
|
|
38
|
+
"margin",
|
|
39
|
+
"marginTop",
|
|
40
|
+
"marginRight",
|
|
41
|
+
"marginBottom",
|
|
42
|
+
"marginLeft",
|
|
43
|
+
]);
|
|
44
|
+
const selectedLabel = selectedValue &&
|
|
45
|
+
((_b = (_a = options.find((o) => o.value === selectedValue)) === null || _a === void 0 ? void 0 : _a.label) !== null && _b !== void 0 ? _b : selectedValue);
|
|
46
|
+
return (React.createElement(Touchable, { disabled: disabled, onPress: toggleFocus, style: [styles.container, viewStyles] },
|
|
47
|
+
React.createElement(View, null,
|
|
48
|
+
React.createElement(NativePicker, { enabled: !disabled, selectedValue: selectedValue, onValueChange: onValueChange, style: {
|
|
49
|
+
flex: 1,
|
|
50
|
+
opacity: 0,
|
|
51
|
+
position: "absolute",
|
|
52
|
+
top: 0,
|
|
53
|
+
left: 0,
|
|
54
|
+
right: 0,
|
|
55
|
+
bottom: 0,
|
|
56
|
+
width: "100%",
|
|
57
|
+
} }, options.map((o) => (React.createElement(NativePicker.Item, { label: o.label, value: o.value, key: o.value })))),
|
|
58
|
+
React.createElement(View, { pointerEvents: "none" },
|
|
59
|
+
React.createElement(TextField, { ...props, value: selectedLabel, placeholder: placeholder,
|
|
60
|
+
// @ts-ignore
|
|
61
|
+
ref: textField, disabled: disabled,
|
|
62
|
+
// @ts-expect-error
|
|
63
|
+
style: stylesWithoutMargin })))));
|
|
64
|
+
};
|
|
65
|
+
const styles = StyleSheet.create({
|
|
66
|
+
container: {
|
|
67
|
+
alignSelf: "stretch",
|
|
68
|
+
},
|
|
69
|
+
});
|
|
70
|
+
export default withTheme(Picker);
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
import PortalConsumer from "./PortalConsumer";
|
|
3
|
+
import PortalHost, { PortalContext } from "./PortalHost";
|
|
4
|
+
import { ThemeProvider, withTheme } from "../../theming";
|
|
5
|
+
/**
|
|
6
|
+
* Portal allows to render a component at a different place in the parent tree.
|
|
7
|
+
* You can use it to render content which should appear above other elements, similar to `Modal`.
|
|
8
|
+
* It requires a [`Portal.Host`](portal-host.html) component to be rendered somewhere in the parent tree.
|
|
9
|
+
*
|
|
10
|
+
* ## Usage
|
|
11
|
+
* ```js
|
|
12
|
+
* import * as React from 'react';
|
|
13
|
+
* import { Portal, Text } from 'react-native-paper';
|
|
14
|
+
*
|
|
15
|
+
* export default class MyComponent extends React.Component {
|
|
16
|
+
* render() {
|
|
17
|
+
* return (
|
|
18
|
+
* <Portal>
|
|
19
|
+
* <Text>This is rendered at a different place</Text>
|
|
20
|
+
* </Portal>
|
|
21
|
+
* );
|
|
22
|
+
* }
|
|
23
|
+
* }
|
|
24
|
+
* ```
|
|
25
|
+
*/
|
|
26
|
+
class Portal extends React.Component {
|
|
27
|
+
render() {
|
|
28
|
+
const { children, theme } = this.props;
|
|
29
|
+
return (React.createElement(PortalContext.Consumer, null, (manager) => (React.createElement(PortalConsumer, { manager: manager },
|
|
30
|
+
React.createElement(ThemeProvider, { theme: theme }, children)))));
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
// @component ./PortalHost.tsx
|
|
34
|
+
Portal.Host = PortalHost;
|
|
35
|
+
export default withTheme(Portal);
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
export default class PortalConsumer extends React.Component {
|
|
3
|
+
async componentDidMount() {
|
|
4
|
+
this.checkManager();
|
|
5
|
+
// Delay updating to prevent React from going to infinite loop
|
|
6
|
+
await Promise.resolve();
|
|
7
|
+
this.key = this.props.manager.mount(this.props.children);
|
|
8
|
+
}
|
|
9
|
+
componentDidUpdate() {
|
|
10
|
+
this.checkManager();
|
|
11
|
+
this.props.manager.update(this.key, this.props.children);
|
|
12
|
+
}
|
|
13
|
+
componentWillUnmount() {
|
|
14
|
+
this.checkManager();
|
|
15
|
+
this.props.manager.unmount(this.key);
|
|
16
|
+
}
|
|
17
|
+
checkManager() {
|
|
18
|
+
if (!this.props.manager) {
|
|
19
|
+
throw new Error("Looks like you forgot to wrap your root component with `Provider` component from `react-native-paper`.\n\n" +
|
|
20
|
+
"Please read our getting-started guide and make sure you've followed all the required steps.\n\n" +
|
|
21
|
+
"https://callstack.github.io/react-native-paper/getting-started.html");
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
render() {
|
|
25
|
+
return null;
|
|
26
|
+
}
|
|
27
|
+
}
|
|
@@ -0,0 +1,107 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
import { View, StyleSheet } from "react-native";
|
|
3
|
+
import PortalManager from "./PortalManager";
|
|
4
|
+
export const PortalContext = React.createContext(null);
|
|
5
|
+
/**
|
|
6
|
+
* Portal host renders all of its children `Portal` elements.
|
|
7
|
+
* For example, you can wrap a screen in `Portal.Host` to render items above the screen.
|
|
8
|
+
* If you're using the `Provider` component, it already includes `Portal.Host`.
|
|
9
|
+
*
|
|
10
|
+
* ## Usage
|
|
11
|
+
* ```js
|
|
12
|
+
* import * as React from 'react';
|
|
13
|
+
* import { Text } from 'react-native';
|
|
14
|
+
* import { Portal } from 'react-native-paper';
|
|
15
|
+
*
|
|
16
|
+
* export default class MyComponent extends React.Component {
|
|
17
|
+
* render() {
|
|
18
|
+
* return (
|
|
19
|
+
* <Portal.Host>
|
|
20
|
+
* <Text>Content of the app</Text>
|
|
21
|
+
* </Portal.Host>
|
|
22
|
+
* );
|
|
23
|
+
* }
|
|
24
|
+
* }
|
|
25
|
+
* ```
|
|
26
|
+
*
|
|
27
|
+
* Here any `Portal` elements under `<App />` are rendered alongside `<App />` and will appear above `<App />` like a `Modal`.
|
|
28
|
+
*/
|
|
29
|
+
export default class PortalHost extends React.Component {
|
|
30
|
+
constructor() {
|
|
31
|
+
super(...arguments);
|
|
32
|
+
this.setManager = (manager) => {
|
|
33
|
+
this.manager = manager;
|
|
34
|
+
};
|
|
35
|
+
this.mount = (children) => {
|
|
36
|
+
const key = this.nextKey++;
|
|
37
|
+
if (this.manager) {
|
|
38
|
+
this.manager.mount(key, children);
|
|
39
|
+
}
|
|
40
|
+
else {
|
|
41
|
+
this.queue.push({ type: "mount", key, children });
|
|
42
|
+
}
|
|
43
|
+
return key;
|
|
44
|
+
};
|
|
45
|
+
this.update = (key, children) => {
|
|
46
|
+
if (this.manager) {
|
|
47
|
+
this.manager.update(key, children);
|
|
48
|
+
}
|
|
49
|
+
else {
|
|
50
|
+
const op = { type: "mount", key, children };
|
|
51
|
+
const index = this.queue.findIndex((o) => o.type === "mount" || (o.type === "update" && o.key === key));
|
|
52
|
+
if (index > -1) {
|
|
53
|
+
// @ts-ignore
|
|
54
|
+
this.queue[index] = op;
|
|
55
|
+
}
|
|
56
|
+
else {
|
|
57
|
+
this.queue.push(op);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
};
|
|
61
|
+
this.unmount = (key) => {
|
|
62
|
+
if (this.manager) {
|
|
63
|
+
this.manager.unmount(key);
|
|
64
|
+
}
|
|
65
|
+
else {
|
|
66
|
+
this.queue.push({ type: "unmount", key });
|
|
67
|
+
}
|
|
68
|
+
};
|
|
69
|
+
this.nextKey = 0;
|
|
70
|
+
this.queue = [];
|
|
71
|
+
}
|
|
72
|
+
componentDidMount() {
|
|
73
|
+
const manager = this.manager;
|
|
74
|
+
const queue = this.queue;
|
|
75
|
+
while (queue.length && manager) {
|
|
76
|
+
const action = queue.pop();
|
|
77
|
+
if (action) {
|
|
78
|
+
switch (action.type) {
|
|
79
|
+
case "mount":
|
|
80
|
+
manager.mount(action.key, action.children);
|
|
81
|
+
break;
|
|
82
|
+
case "update":
|
|
83
|
+
manager.update(action.key, action.children);
|
|
84
|
+
break;
|
|
85
|
+
case "unmount":
|
|
86
|
+
manager.unmount(action.key);
|
|
87
|
+
break;
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
render() {
|
|
93
|
+
return (React.createElement(PortalContext.Provider, { value: {
|
|
94
|
+
mount: this.mount,
|
|
95
|
+
update: this.update,
|
|
96
|
+
unmount: this.unmount,
|
|
97
|
+
} },
|
|
98
|
+
React.createElement(View, { style: styles.container, collapsable: false, pointerEvents: "box-none" }, this.props.children),
|
|
99
|
+
React.createElement(PortalManager, { ref: this.setManager })));
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
PortalHost.displayName = "Portal.Host";
|
|
103
|
+
const styles = StyleSheet.create({
|
|
104
|
+
container: {
|
|
105
|
+
flex: 1,
|
|
106
|
+
},
|
|
107
|
+
});
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
import { View, StyleSheet } from "react-native";
|
|
3
|
+
/**
|
|
4
|
+
* Portal host is the component which actually renders all Portals.
|
|
5
|
+
*/
|
|
6
|
+
export default class PortalManager extends React.PureComponent {
|
|
7
|
+
constructor() {
|
|
8
|
+
super(...arguments);
|
|
9
|
+
this.state = {
|
|
10
|
+
portals: [],
|
|
11
|
+
};
|
|
12
|
+
this.mount = (key, children) => {
|
|
13
|
+
this.setState((state) => ({
|
|
14
|
+
portals: [...state.portals, { key, children }],
|
|
15
|
+
}));
|
|
16
|
+
};
|
|
17
|
+
this.update = (key, children) => this.setState((state) => ({
|
|
18
|
+
portals: state.portals.map((item) => {
|
|
19
|
+
if (item.key === key) {
|
|
20
|
+
return { ...item, children };
|
|
21
|
+
}
|
|
22
|
+
return item;
|
|
23
|
+
}),
|
|
24
|
+
}));
|
|
25
|
+
this.unmount = (key) => this.setState((state) => ({
|
|
26
|
+
portals: state.portals.filter((item) => item.key !== key),
|
|
27
|
+
}));
|
|
28
|
+
}
|
|
29
|
+
render() {
|
|
30
|
+
return this.state.portals.map(({ key, children }) => (React.createElement(View, { key: key, collapsable: false /* Need collapsable=false here to clip the elevations, otherwise they appear above sibling components */, pointerEvents: "box-none", style: StyleSheet.absoluteFill }, children)));
|
|
31
|
+
}
|
|
32
|
+
}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
import Config from "../Config";
|
|
3
|
+
import IconButton from "../IconButton";
|
|
4
|
+
import { getValueForRadioButton } from "../../utilities";
|
|
5
|
+
import { useRadioButtonGroupContext } from "./context";
|
|
6
|
+
const RadioButton = ({ Icon, disabled = false, color, value = "", selected, unselectedColor, onPress, size = Config.radioButtonSize, selectedIcon = "MaterialIcons/radio-button-checked", unselectedIcon = "MaterialIcons/radio-button-unchecked", style, ...rest }) => {
|
|
7
|
+
const { value: contextValue, onValueChange } = useRadioButtonGroupContext();
|
|
8
|
+
const realValue = getValueForRadioButton(value);
|
|
9
|
+
const realContextValue = getValueForRadioButton(contextValue);
|
|
10
|
+
const isSelected = selected !== null && selected !== void 0 ? selected : realContextValue === realValue;
|
|
11
|
+
const handlePress = () => {
|
|
12
|
+
onPress === null || onPress === void 0 ? void 0 : onPress(realValue);
|
|
13
|
+
onValueChange === null || onValueChange === void 0 ? void 0 : onValueChange(realValue);
|
|
14
|
+
};
|
|
15
|
+
return (React.createElement(IconButton, { Icon: Icon, icon: isSelected ? selectedIcon : unselectedIcon, color: isSelected ? color : unselectedColor, disabled: disabled, onPress: handlePress, size: size, style: style, ...rest }));
|
|
16
|
+
};
|
|
17
|
+
export default RadioButton;
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import React from "react";
|
|
2
|
+
import { View } from "react-native";
|
|
3
|
+
import Text from "../Text";
|
|
4
|
+
import { withTheme } from "../../theming";
|
|
5
|
+
import RadioButtonGroup from "./RadioButtonGroup";
|
|
6
|
+
const RadioButtonFieldGroup = ({ label, children, theme, labelStyle, style, ...rest }) => {
|
|
7
|
+
return (React.createElement(View, { style: style },
|
|
8
|
+
React.createElement(Text, { style: [
|
|
9
|
+
{
|
|
10
|
+
fontSize: theme.typography.headline4.fontSize,
|
|
11
|
+
color: theme.typography.headline4.color,
|
|
12
|
+
},
|
|
13
|
+
labelStyle,
|
|
14
|
+
] }, label),
|
|
15
|
+
React.createElement(RadioButtonGroup, { theme: theme, ...rest }, children)));
|
|
16
|
+
};
|
|
17
|
+
export default withTheme(RadioButtonFieldGroup);
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
import { View } from "react-native";
|
|
3
|
+
import { getValueForRadioButton } from "../../utilities";
|
|
4
|
+
import { radioButtonGroupContext, Direction } from "./context";
|
|
5
|
+
const { Provider } = radioButtonGroupContext;
|
|
6
|
+
const RadioButtonGroup = ({ direction = Direction.Vertical, value = "", onValueChange, defaultValue, style, children, ...rest }) => {
|
|
7
|
+
const [internalValue, setInternalValue] = React.useState("");
|
|
8
|
+
React.useEffect(() => {
|
|
9
|
+
if (value != null) {
|
|
10
|
+
const realValue = getValueForRadioButton(value);
|
|
11
|
+
setInternalValue(realValue);
|
|
12
|
+
}
|
|
13
|
+
}, [value]);
|
|
14
|
+
React.useEffect(() => {
|
|
15
|
+
if (defaultValue != null) {
|
|
16
|
+
const realDefaultValue = getValueForRadioButton(defaultValue);
|
|
17
|
+
setInternalValue(realDefaultValue);
|
|
18
|
+
}
|
|
19
|
+
}, [defaultValue]);
|
|
20
|
+
const handleValueChange = (newValue) => {
|
|
21
|
+
const realNewValue = getValueForRadioButton(newValue);
|
|
22
|
+
setInternalValue(realNewValue);
|
|
23
|
+
onValueChange === null || onValueChange === void 0 ? void 0 : onValueChange(realNewValue);
|
|
24
|
+
};
|
|
25
|
+
const _containerStyle = [
|
|
26
|
+
{
|
|
27
|
+
flexDirection: direction === Direction.Horizontal ? "row" : "column",
|
|
28
|
+
},
|
|
29
|
+
];
|
|
30
|
+
if (direction !== Direction.Vertical) {
|
|
31
|
+
_containerStyle.push({
|
|
32
|
+
alignItems: "center",
|
|
33
|
+
});
|
|
34
|
+
}
|
|
35
|
+
return (React.createElement(View, { style: [{ minHeight: 40 }, style], ...rest },
|
|
36
|
+
React.createElement(Provider, { value: {
|
|
37
|
+
value: internalValue,
|
|
38
|
+
onValueChange: handleValueChange,
|
|
39
|
+
direction,
|
|
40
|
+
} },
|
|
41
|
+
React.createElement(View, { style: _containerStyle }, children))));
|
|
42
|
+
};
|
|
43
|
+
export default RadioButtonGroup;
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
import { StyleSheet, View, Platform, } from "react-native";
|
|
3
|
+
import RadioButton from "./RadioButton";
|
|
4
|
+
import Text from "../Text";
|
|
5
|
+
import { useRadioButtonGroupContext } from "./context";
|
|
6
|
+
import { Direction as GroupDirection } from "./context";
|
|
7
|
+
import Touchable from "../Touchable";
|
|
8
|
+
import { extractStyles, getValueForRadioButton } from "../../utilities";
|
|
9
|
+
export var Direction;
|
|
10
|
+
(function (Direction) {
|
|
11
|
+
Direction["Row"] = "row";
|
|
12
|
+
Direction["RowReverse"] = "row-reverse";
|
|
13
|
+
})(Direction || (Direction = {}));
|
|
14
|
+
const getRadioButtonAlignment = (parentDirection, direction) => {
|
|
15
|
+
if (parentDirection === GroupDirection.Horizontal) {
|
|
16
|
+
return direction === Direction.Row ? "flex-start" : "flex-end";
|
|
17
|
+
}
|
|
18
|
+
else if (direction === Direction.RowReverse) {
|
|
19
|
+
return "flex-start";
|
|
20
|
+
}
|
|
21
|
+
else {
|
|
22
|
+
return "flex-end";
|
|
23
|
+
}
|
|
24
|
+
};
|
|
25
|
+
const renderLabel = (value, labelStyle, textStyle) => {
|
|
26
|
+
if (typeof value === "string") {
|
|
27
|
+
return React.createElement(Text, { style: [labelStyle, textStyle] }, value);
|
|
28
|
+
}
|
|
29
|
+
else {
|
|
30
|
+
return React.createElement(React.Fragment, null, value);
|
|
31
|
+
}
|
|
32
|
+
};
|
|
33
|
+
const RadioButtonRow = ({ Icon, label, value = "", color, unselectedColor, onPress, labelContainerStyle, labelStyle, radioButtonStyle, direction = Direction.Row, selected, disabled, style, ...rest }) => {
|
|
34
|
+
const { value: contextValue, onValueChange, direction: parentDirection, } = useRadioButtonGroupContext();
|
|
35
|
+
const realValue = getValueForRadioButton(value);
|
|
36
|
+
const realContextValue = getValueForRadioButton(contextValue);
|
|
37
|
+
const isSelected = selected !== null && selected !== void 0 ? selected : realContextValue === realValue;
|
|
38
|
+
const handlePress = () => {
|
|
39
|
+
onPress === null || onPress === void 0 ? void 0 : onPress(realValue);
|
|
40
|
+
onValueChange === null || onValueChange === void 0 ? void 0 : onValueChange(realValue);
|
|
41
|
+
};
|
|
42
|
+
const { textStyles, viewStyles } = extractStyles(style);
|
|
43
|
+
return (React.createElement(Touchable, { onPress: handlePress, style: [styles.mainParent, { flexDirection: direction }, viewStyles], disabled: disabled, ...rest },
|
|
44
|
+
React.createElement(View, { style: [
|
|
45
|
+
styles.label,
|
|
46
|
+
{
|
|
47
|
+
alignItems: direction === Direction.Row ? "flex-start" : "flex-end",
|
|
48
|
+
},
|
|
49
|
+
labelContainerStyle,
|
|
50
|
+
] }, renderLabel(label, labelStyle, textStyles)),
|
|
51
|
+
React.createElement(View, { style: {
|
|
52
|
+
flex: 1,
|
|
53
|
+
alignItems: getRadioButtonAlignment(parentDirection, direction),
|
|
54
|
+
} },
|
|
55
|
+
React.createElement(RadioButton, { Icon: Icon, selected: isSelected, value: realValue, color: color, unselectedColor: unselectedColor, style: radioButtonStyle }))));
|
|
56
|
+
};
|
|
57
|
+
const styles = StyleSheet.create({
|
|
58
|
+
mainParent: {
|
|
59
|
+
alignItems: "center",
|
|
60
|
+
justifyContent: "space-around",
|
|
61
|
+
paddingStart: 20,
|
|
62
|
+
minHeight: 50,
|
|
63
|
+
paddingEnd: 20,
|
|
64
|
+
display: "flex",
|
|
65
|
+
...Platform.select({
|
|
66
|
+
web: {
|
|
67
|
+
cursor: "pointer",
|
|
68
|
+
userSelect: "none",
|
|
69
|
+
},
|
|
70
|
+
}),
|
|
71
|
+
},
|
|
72
|
+
label: {
|
|
73
|
+
flex: 3,
|
|
74
|
+
},
|
|
75
|
+
});
|
|
76
|
+
export default RadioButtonRow;
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
import { createContext, useContext } from "react";
|
|
2
|
+
export var Direction;
|
|
3
|
+
(function (Direction) {
|
|
4
|
+
Direction["Horizontal"] = "horizontal";
|
|
5
|
+
Direction["Vertical"] = "vertical";
|
|
6
|
+
})(Direction || (Direction = {}));
|
|
7
|
+
export const radioButtonGroupContext = createContext({
|
|
8
|
+
onValueChange: () => { },
|
|
9
|
+
value: "",
|
|
10
|
+
direction: undefined,
|
|
11
|
+
});
|
|
12
|
+
export function useRadioButtonGroupContext() {
|
|
13
|
+
return useContext(radioButtonGroupContext);
|
|
14
|
+
}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
import { View, Platform, Image } from "react-native";
|
|
3
|
+
import { SvgUri } from "react-native-svg";
|
|
4
|
+
import Config from "./Config";
|
|
5
|
+
const SVG = ({ source = Config.placeholderSvgURL, style, }) => {
|
|
6
|
+
return (React.createElement(React.Fragment, null,
|
|
7
|
+
Platform.OS === "ios" && (React.createElement(View, { style: style },
|
|
8
|
+
React.createElement(SvgUri, { width: "100%", height: "100%", uri: source }))),
|
|
9
|
+
Platform.OS === "android" && (React.createElement(View, { style: style },
|
|
10
|
+
React.createElement(SvgUri, { width: "100%", height: "100%", uri: source }))),
|
|
11
|
+
Platform.OS === "web" && (React.createElement(Image, { style: style, source: { uri: source } }))));
|
|
12
|
+
};
|
|
13
|
+
export default SVG;
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
import { StyleSheet, ScrollView, View, } from "react-native";
|
|
3
|
+
import { SafeAreaView } from "react-native-safe-area-context";
|
|
4
|
+
import { withTheme } from "../theming";
|
|
5
|
+
function ScreenContainer({ scrollable = false, hasSafeArea = false, hasBottomSafeArea = false, hasTopSafeArea = false, theme, style, children, ...rest }) {
|
|
6
|
+
const backgroundColor = theme.colors.background;
|
|
7
|
+
const edges = ["left", "right"];
|
|
8
|
+
if (hasSafeArea || hasTopSafeArea) {
|
|
9
|
+
edges.push("top");
|
|
10
|
+
}
|
|
11
|
+
if (hasSafeArea || hasBottomSafeArea) {
|
|
12
|
+
edges.push("bottom");
|
|
13
|
+
}
|
|
14
|
+
return (React.createElement(SafeAreaView, { edges: edges, style: [
|
|
15
|
+
styles.container,
|
|
16
|
+
{
|
|
17
|
+
backgroundColor,
|
|
18
|
+
},
|
|
19
|
+
], ...rest }, scrollable ? (React.createElement(ScrollView, { contentContainerStyle: [
|
|
20
|
+
styles.scrollViewContainer,
|
|
21
|
+
{ backgroundColor },
|
|
22
|
+
style,
|
|
23
|
+
] }, children)) : (React.createElement(View, { style: [styles.container, { backgroundColor }, style] }, children))));
|
|
24
|
+
}
|
|
25
|
+
const styles = StyleSheet.create({
|
|
26
|
+
container: {
|
|
27
|
+
flex: 1,
|
|
28
|
+
},
|
|
29
|
+
scrollViewContainer: {
|
|
30
|
+
flexGrow: 1,
|
|
31
|
+
flex: undefined,
|
|
32
|
+
},
|
|
33
|
+
});
|
|
34
|
+
export default withTheme(ScreenContainer);
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
import { View, StyleSheet } from "react-native";
|
|
3
|
+
import NativeSlider from "@react-native-community/slider";
|
|
4
|
+
import isNumber from "lodash.isnumber";
|
|
5
|
+
import toNumber from "lodash.tonumber";
|
|
6
|
+
import { withTheme } from "../theming";
|
|
7
|
+
function maybeParseValue(value) {
|
|
8
|
+
if (value === undefined) {
|
|
9
|
+
return undefined;
|
|
10
|
+
}
|
|
11
|
+
if (isNumber(value)) {
|
|
12
|
+
return value;
|
|
13
|
+
}
|
|
14
|
+
try {
|
|
15
|
+
const maybe = toNumber(value);
|
|
16
|
+
if (isNumber(maybe)) {
|
|
17
|
+
return maybe;
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
catch {
|
|
21
|
+
return undefined;
|
|
22
|
+
}
|
|
23
|
+
return undefined;
|
|
24
|
+
}
|
|
25
|
+
function Slider({ Icon, leftIcon, rightIcon, leftIconColor, rightIconColor, value, defaultValue, minimumTrackTintColor, maximumTrackTintColor, thumbTintColor, minimumValue = 0, maximumValue = 100, tapToSeek, step = 1, onValueChange = () => { }, style, theme, ...rest }) {
|
|
26
|
+
const [internalValue, setInternalValue] = React.useState(value || defaultValue);
|
|
27
|
+
React.useEffect(() => {
|
|
28
|
+
if (value != null) {
|
|
29
|
+
setInternalValue(value);
|
|
30
|
+
}
|
|
31
|
+
}, [value]);
|
|
32
|
+
React.useEffect(() => {
|
|
33
|
+
if (defaultValue != null) {
|
|
34
|
+
setInternalValue(defaultValue);
|
|
35
|
+
}
|
|
36
|
+
}, [defaultValue]);
|
|
37
|
+
const minTrackColor = minimumTrackTintColor || theme.colors.primary;
|
|
38
|
+
const maxTrackColor = maximumTrackTintColor || theme.colors.light;
|
|
39
|
+
const thumbColor = thumbTintColor || theme.colors.primary;
|
|
40
|
+
const leftIconThemeColor = leftIconColor || theme.colors.light;
|
|
41
|
+
const rightIconThemeColor = rightIconColor || theme.colors.light;
|
|
42
|
+
const parsedValue = maybeParseValue(internalValue);
|
|
43
|
+
const handleSlidingComplete = (newValue) => {
|
|
44
|
+
setInternalValue(newValue);
|
|
45
|
+
onValueChange(newValue);
|
|
46
|
+
};
|
|
47
|
+
return (React.createElement(View, { style: [styles.container, style], ...rest },
|
|
48
|
+
leftIcon ? (React.createElement(Icon, { color: leftIconThemeColor, name: leftIcon, size: 24 })) : null,
|
|
49
|
+
React.createElement(NativeSlider, { value: parsedValue, step: step, minimumValue: minimumValue, maximumValue: maximumValue, tapToSeek: tapToSeek, minimumTrackTintColor: minTrackColor, maximumTrackTintColor: maxTrackColor, thumbTintColor: thumbColor, onSlidingComplete: handleSlidingComplete, style: styles.slider }),
|
|
50
|
+
rightIcon ? (React.createElement(Icon, { color: rightIconThemeColor, name: rightIcon, size: 24 })) : null));
|
|
51
|
+
}
|
|
52
|
+
const styles = StyleSheet.create({
|
|
53
|
+
container: {
|
|
54
|
+
height: 40,
|
|
55
|
+
flexDirection: "row",
|
|
56
|
+
alignItems: "center",
|
|
57
|
+
},
|
|
58
|
+
slider: {
|
|
59
|
+
flex: 1,
|
|
60
|
+
marginHorizontal: 12,
|
|
61
|
+
},
|
|
62
|
+
});
|
|
63
|
+
export default withTheme(Slider);
|