@mpxjs/webpack-plugin 2.9.69 → 2.9.70

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.
Files changed (111) hide show
  1. package/lib/parser.js +1 -1
  2. package/lib/platform/json/wx/index.js +21 -8
  3. package/lib/platform/style/wx/index.js +51 -54
  4. package/lib/platform/template/wx/component-config/fix-component-name.js +15 -12
  5. package/lib/platform/template/wx/component-config/index.js +1 -1
  6. package/lib/platform/template/wx/component-config/input.js +1 -1
  7. package/lib/platform/template/wx/component-config/rich-text.js +8 -0
  8. package/lib/platform/template/wx/component-config/swiper.js +1 -1
  9. package/lib/platform/template/wx/component-config/textarea.js +1 -1
  10. package/lib/platform/template/wx/component-config/unsupported.js +1 -1
  11. package/lib/react/processTemplate.js +3 -0
  12. package/lib/runtime/components/react/context.ts +4 -0
  13. package/lib/runtime/components/react/dist/context.js +1 -0
  14. package/lib/runtime/components/react/dist/event.config.js +24 -24
  15. package/lib/runtime/components/react/dist/getInnerListeners.js +183 -166
  16. package/lib/runtime/components/react/dist/mpx-button.jsx +35 -42
  17. package/lib/runtime/components/react/dist/mpx-canvas/html.js +2 -4
  18. package/lib/runtime/components/react/dist/mpx-canvas/index.jsx +35 -13
  19. package/lib/runtime/components/react/dist/mpx-checkbox-group.jsx +13 -19
  20. package/lib/runtime/components/react/dist/mpx-checkbox.jsx +29 -38
  21. package/lib/runtime/components/react/dist/mpx-form.jsx +16 -19
  22. package/lib/runtime/components/react/dist/mpx-icon.jsx +8 -16
  23. package/lib/runtime/components/react/dist/mpx-image.jsx +291 -0
  24. package/lib/runtime/components/react/dist/mpx-input.jsx +54 -27
  25. package/lib/runtime/components/react/dist/mpx-label.jsx +15 -22
  26. package/lib/runtime/components/react/dist/mpx-movable-area.jsx +13 -16
  27. package/lib/runtime/components/react/dist/mpx-movable-view.jsx +13 -13
  28. package/lib/runtime/components/react/dist/mpx-navigator.jsx +2 -4
  29. package/lib/runtime/components/react/dist/mpx-picker/date.jsx +6 -2
  30. package/lib/runtime/components/react/dist/mpx-picker/index.jsx +5 -3
  31. package/lib/runtime/components/react/dist/mpx-picker/multiSelector.jsx +6 -2
  32. package/lib/runtime/components/react/dist/mpx-picker/region.jsx +6 -2
  33. package/lib/runtime/components/react/dist/mpx-picker/selector.jsx +6 -2
  34. package/lib/runtime/components/react/dist/mpx-picker/time.jsx +10 -15
  35. package/lib/runtime/components/react/dist/mpx-picker-view-column.jsx +3 -1
  36. package/lib/runtime/components/react/dist/mpx-picker-view.jsx +5 -3
  37. package/lib/runtime/components/react/dist/mpx-radio-group.jsx +11 -19
  38. package/lib/runtime/components/react/dist/mpx-radio.jsx +27 -42
  39. package/lib/runtime/components/react/dist/mpx-rich-text/html.js +39 -0
  40. package/lib/runtime/components/react/dist/mpx-rich-text/index.jsx +63 -0
  41. package/lib/runtime/components/react/dist/mpx-root-portal.jsx +6 -4
  42. package/lib/runtime/components/react/dist/mpx-scroll-view.jsx +47 -41
  43. package/lib/runtime/components/react/dist/mpx-simple-text.jsx +11 -0
  44. package/lib/runtime/components/react/dist/mpx-swiper/carouse.jsx +4 -2
  45. package/lib/runtime/components/react/dist/mpx-swiper-item.jsx +4 -2
  46. package/lib/runtime/components/react/dist/mpx-switch.jsx +20 -10
  47. package/lib/runtime/components/react/dist/mpx-text.jsx +11 -10
  48. package/lib/runtime/components/react/dist/mpx-textarea.jsx +8 -3
  49. package/lib/runtime/components/react/dist/mpx-view.jsx +34 -46
  50. package/lib/runtime/components/react/dist/mpx-web-view.jsx +105 -42
  51. package/lib/runtime/components/react/dist/useAnimationHooks.js +35 -10
  52. package/lib/runtime/components/react/dist/utils.jsx +107 -82
  53. package/lib/runtime/components/react/event.config.ts +25 -26
  54. package/lib/runtime/components/react/getInnerListeners.ts +238 -188
  55. package/lib/runtime/components/react/mpx-button.tsx +64 -50
  56. package/lib/runtime/components/react/mpx-canvas/html.ts +2 -4
  57. package/lib/runtime/components/react/mpx-canvas/index.tsx +46 -48
  58. package/lib/runtime/components/react/mpx-checkbox-group.tsx +28 -25
  59. package/lib/runtime/components/react/mpx-checkbox.tsx +48 -49
  60. package/lib/runtime/components/react/mpx-form.tsx +25 -28
  61. package/lib/runtime/components/react/mpx-icon.tsx +12 -17
  62. package/lib/runtime/components/react/mpx-image.tsx +436 -0
  63. package/lib/runtime/components/react/mpx-input.tsx +77 -57
  64. package/lib/runtime/components/react/mpx-label.tsx +26 -27
  65. package/lib/runtime/components/react/mpx-movable-area.tsx +18 -23
  66. package/lib/runtime/components/react/mpx-movable-view.tsx +21 -25
  67. package/lib/runtime/components/react/mpx-navigator.tsx +2 -8
  68. package/lib/runtime/components/react/mpx-picker/date.tsx +5 -2
  69. package/lib/runtime/components/react/mpx-picker/index.tsx +3 -2
  70. package/lib/runtime/components/react/mpx-picker/multiSelector.tsx +5 -2
  71. package/lib/runtime/components/react/mpx-picker/region.tsx +5 -2
  72. package/lib/runtime/components/react/mpx-picker/selector.tsx +5 -2
  73. package/lib/runtime/components/react/mpx-picker/time.tsx +10 -15
  74. package/lib/runtime/components/react/mpx-picker/type.ts +48 -43
  75. package/lib/runtime/components/react/mpx-picker-view-column.tsx +4 -1
  76. package/lib/runtime/components/react/mpx-picker-view.tsx +7 -1
  77. package/lib/runtime/components/react/mpx-radio-group.tsx +24 -27
  78. package/lib/runtime/components/react/mpx-radio.tsx +45 -54
  79. package/lib/runtime/components/react/mpx-rich-text/html.ts +40 -0
  80. package/lib/runtime/components/react/mpx-rich-text/index.tsx +121 -0
  81. package/lib/runtime/components/react/mpx-root-portal.tsx +3 -5
  82. package/lib/runtime/components/react/mpx-scroll-view.tsx +72 -71
  83. package/lib/runtime/components/react/mpx-simple-text.tsx +18 -0
  84. package/lib/runtime/components/react/mpx-swiper/carouse.tsx +4 -2
  85. package/lib/runtime/components/react/mpx-swiper-item.tsx +3 -2
  86. package/lib/runtime/components/react/mpx-switch.tsx +29 -23
  87. package/lib/runtime/components/react/mpx-text.tsx +14 -18
  88. package/lib/runtime/components/react/mpx-textarea.tsx +11 -10
  89. package/lib/runtime/components/react/mpx-view.tsx +55 -65
  90. package/lib/runtime/components/react/mpx-web-view.tsx +108 -63
  91. package/lib/runtime/components/react/types/global.d.ts +3 -17
  92. package/lib/runtime/components/react/useAnimationHooks.ts +36 -12
  93. package/lib/runtime/components/react/utils.tsx +113 -82
  94. package/lib/runtime/components/web/getInnerListeners.js +6 -6
  95. package/lib/runtime/components/web/mpx-movable-view.vue +334 -344
  96. package/lib/runtime/components/web/mpx-picker-view-column.vue +75 -75
  97. package/lib/runtime/components/web/mpx-picker.vue +382 -385
  98. package/lib/runtime/components/web/mpx-web-view.vue +175 -161
  99. package/lib/runtime/optionProcessor.js +7 -38
  100. package/lib/runtime/utils.js +2 -0
  101. package/lib/style-compiler/plugins/scope-id.js +30 -2
  102. package/lib/template-compiler/bind-this.js +7 -2
  103. package/lib/template-compiler/compiler.js +77 -46
  104. package/lib/template-compiler/gen-node-react.js +2 -2
  105. package/lib/utils/pre-process-json.js +9 -5
  106. package/package.json +5 -4
  107. package/lib/runtime/components/react/dist/mpx-image/index.jsx +0 -226
  108. package/lib/runtime/components/react/dist/mpx-image/svg.jsx +0 -7
  109. package/lib/runtime/components/react/mpx-image/index.tsx +0 -345
  110. package/lib/runtime/components/react/mpx-image/svg.tsx +0 -22
  111. package/lib/runtime/components/web/event.js +0 -105
@@ -1,13 +1,13 @@
1
1
  /**
2
2
  * ✔ bindchange
3
3
  */
4
- import { useRef, forwardRef, useContext, useMemo, useEffect } from 'react';
4
+ import { useRef, forwardRef, useContext, useMemo, useEffect, createElement } from 'react';
5
5
  import { View } from 'react-native';
6
6
  import { warn } from '@mpxjs/utils';
7
7
  import { FormContext, CheckboxGroupContext } from './context';
8
8
  import useInnerProps, { getCustomEvent } from './getInnerListeners';
9
9
  import useNodesRef from './useNodesRef';
10
- import { useLayout, useTransformStyle, wrapChildren } from './utils';
10
+ import { useLayout, useTransformStyle, wrapChildren, extendObject } from './utils';
11
11
  const CheckboxGroup = forwardRef((props, ref) => {
12
12
  const propsRef = useRef({});
13
13
  propsRef.current = props;
@@ -22,13 +22,10 @@ const CheckboxGroup = forwardRef((props, ref) => {
22
22
  flexDirection: 'row',
23
23
  flexWrap: 'wrap'
24
24
  };
25
- const styleObj = {
26
- ...defaultStyle,
27
- ...style
28
- };
25
+ const styleObj = extendObject({}, defaultStyle, style);
29
26
  const { hasSelfPercent, normalStyle, hasVarDec, varContextRef, setWidth, setHeight } = useTransformStyle(styleObj, { enableVar, externalVarContext, parentFontSize, parentWidth, parentHeight });
30
27
  const nodeRef = useRef(null);
31
- useNodesRef(props, ref, nodeRef, { defaultStyle });
28
+ useNodesRef(props, ref, nodeRef, { style: normalStyle });
32
29
  const { layoutRef, layoutStyle, layoutProps } = useLayout({ props, hasSelfPercent, setWidth, setHeight, nodeRef });
33
30
  const getValue = () => {
34
31
  const arr = [];
@@ -60,11 +57,12 @@ const CheckboxGroup = forwardRef((props, ref) => {
60
57
  }
61
58
  };
62
59
  }, []);
63
- const innerProps = useInnerProps(props, {
60
+ const innerProps = useInnerProps(props, extendObject({
64
61
  ref: nodeRef,
65
- style: { ...normalStyle, ...layoutStyle },
66
- ...layoutProps
67
- }, [], {
62
+ style: extendObject({}, normalStyle, layoutStyle)
63
+ }, layoutProps), [
64
+ 'name'
65
+ ], {
68
66
  layoutRef
69
67
  });
70
68
  const contextValue = useMemo(() => {
@@ -83,14 +81,10 @@ const CheckboxGroup = forwardRef((props, ref) => {
83
81
  notifyChange
84
82
  };
85
83
  }, []);
86
- return (<View {...innerProps}>
87
- <CheckboxGroupContext.Provider value={contextValue}>
88
- {wrapChildren(props, {
89
- hasVarDec,
90
- varContext: varContextRef.current
91
- })}
92
- </CheckboxGroupContext.Provider>
93
- </View>);
84
+ return createElement(View, innerProps, createElement(CheckboxGroupContext.Provider, { value: contextValue }, wrapChildren(props, {
85
+ hasVarDec,
86
+ varContext: varContextRef.current
87
+ })));
94
88
  });
95
89
  CheckboxGroup.displayName = 'MpxCheckboxGroup';
96
90
  export default CheckboxGroup;
@@ -4,13 +4,13 @@
4
4
  * ✔ checked
5
5
  * ✔ color
6
6
  */
7
- import { useState, useRef, forwardRef, useEffect, useContext } from 'react';
7
+ import { useState, useRef, forwardRef, useEffect, useContext, createElement } from 'react';
8
8
  import { View, StyleSheet } from 'react-native';
9
9
  import { warn } from '@mpxjs/utils';
10
10
  import useInnerProps, { getCustomEvent } from './getInnerListeners';
11
11
  import useNodesRef from './useNodesRef';
12
12
  import Icon from './mpx-icon';
13
- import { splitProps, splitStyle, useLayout, useTransformStyle, wrapChildren } from './utils';
13
+ import { splitProps, splitStyle, useLayout, useTransformStyle, wrapChildren, extendObject } from './utils';
14
14
  import { CheckboxGroupContext, LabelContext } from './context';
15
15
  const styles = StyleSheet.create({
16
16
  container: {
@@ -40,19 +40,13 @@ const styles = StyleSheet.create({
40
40
  });
41
41
  const Checkbox = forwardRef((checkboxProps, ref) => {
42
42
  const { textProps, innerProps: props = {} } = splitProps(checkboxProps);
43
- const { value = '', disabled = false, checked = false, color = '#09BB07', style = {}, 'enable-var': enableVar, 'external-var-context': externalVarContext, 'parent-font-size': parentFontSize, 'parent-width': parentWidth, 'parent-height': parentHeight, bindtap, catchtap } = props;
43
+ const { value = '', disabled = false, checked = false, color = '#09BB07', style = {}, 'enable-var': enableVar, 'external-var-context': externalVarContext, 'parent-font-size': parentFontSize, 'parent-width': parentWidth, 'parent-height': parentHeight, bindtap, _onChange } = props;
44
44
  const [isChecked, setIsChecked] = useState(!!checked);
45
45
  const groupContext = useContext(CheckboxGroupContext);
46
46
  let groupValue;
47
47
  let notifyChange;
48
- const defaultStyle = {
49
- ...styles.wrapper,
50
- ...(disabled && styles.wrapperDisabled)
51
- };
52
- const styleObj = {
53
- ...styles.container,
54
- ...style
55
- };
48
+ const defaultStyle = extendObject({}, styles.wrapper, disabled ? styles.wrapperDisabled : null);
49
+ const styleObj = extendObject({}, styles.container, style);
56
50
  const onChange = (evt) => {
57
51
  if (disabled)
58
52
  return;
@@ -62,27 +56,21 @@ const Checkbox = forwardRef((checkboxProps, ref) => {
62
56
  groupValue[value].checked = checked;
63
57
  }
64
58
  notifyChange && notifyChange(evt);
59
+ // Called when the switch type attribute is checkbox
60
+ _onChange && _onChange(evt, { checked });
65
61
  };
66
62
  const onTap = (evt) => {
67
- if (disabled)
68
- return;
69
63
  bindtap && bindtap(getCustomEvent('tap', evt, { layoutRef }, props));
70
64
  onChange(evt);
71
65
  };
72
- const catchTap = (evt) => {
73
- if (disabled)
74
- return;
75
- catchtap && catchtap(getCustomEvent('tap', evt, { layoutRef }, props));
76
- onChange(evt);
77
- };
78
66
  const { hasSelfPercent, normalStyle, hasVarDec, varContextRef, setWidth, setHeight } = useTransformStyle(styleObj, { enableVar, externalVarContext, parentFontSize, parentWidth, parentHeight });
79
67
  const nodeRef = useRef(null);
80
68
  useNodesRef(props, ref, nodeRef, {
81
- defaultStyle,
69
+ style: extendObject({}, defaultStyle, normalStyle),
82
70
  change: onChange
83
71
  });
84
72
  const { layoutRef, layoutStyle, layoutProps } = useLayout({ props, hasSelfPercent, setWidth, setHeight, nodeRef });
85
- const { textStyle, backgroundStyle, innerStyle } = splitStyle(normalStyle);
73
+ const { textStyle, backgroundStyle, innerStyle = {} } = splitStyle(normalStyle);
86
74
  if (backgroundStyle) {
87
75
  warn('Checkbox does not support background image-related styles!');
88
76
  }
@@ -94,13 +82,16 @@ const Checkbox = forwardRef((checkboxProps, ref) => {
94
82
  if (labelContext) {
95
83
  labelContext.current.triggerChange = onChange;
96
84
  }
97
- const innerProps = useInnerProps(props, {
85
+ const innerProps = useInnerProps(props, extendObject({
98
86
  ref: nodeRef,
99
- style: { ...innerStyle, ...layoutStyle },
100
- ...layoutProps,
101
- bindtap: onTap,
102
- catchtap: catchTap
103
- }, [], {
87
+ style: extendObject({}, innerStyle, layoutStyle)
88
+ }, layoutProps, {
89
+ bindtap: !disabled && onTap
90
+ }), [
91
+ 'value',
92
+ 'disabled',
93
+ 'checked'
94
+ ], {
104
95
  layoutRef
105
96
  });
106
97
  useEffect(() => {
@@ -124,17 +115,17 @@ const Checkbox = forwardRef((checkboxProps, ref) => {
124
115
  }
125
116
  }
126
117
  }, [checked]);
127
- return (<View {...innerProps}>
128
- <View style={defaultStyle}>
129
- <Icon type='success_no_circle' size={18} color={disabled ? '#ADADAD' : color} style={isChecked ? styles.iconChecked : styles.icon}/>
130
- </View>
131
- {wrapChildren(props, {
132
- hasVarDec,
133
- varContext: varContextRef.current,
134
- textStyle,
135
- textProps
136
- })}
137
- </View>);
118
+ return createElement(View, innerProps, createElement(View, { style: defaultStyle }, createElement(Icon, {
119
+ type: 'success_no_circle',
120
+ size: 18,
121
+ color: disabled ? '#ADADAD' : color,
122
+ style: isChecked ? styles.iconChecked : styles.icon
123
+ })), wrapChildren(props, {
124
+ hasVarDec,
125
+ varContext: varContextRef.current,
126
+ textStyle,
127
+ textProps
128
+ }));
138
129
  });
139
130
  Checkbox.displayName = 'MpxCheckbox';
140
131
  export default Checkbox;
@@ -5,26 +5,27 @@
5
5
  * ✔ bindreset
6
6
  */
7
7
  import { View } from 'react-native';
8
- import { useRef, forwardRef, useMemo } from 'react';
8
+ import { useRef, forwardRef, useMemo, createElement } from 'react';
9
9
  import useNodesRef from './useNodesRef';
10
10
  import useInnerProps, { getCustomEvent } from './getInnerListeners';
11
11
  import { FormContext } from './context';
12
- import { useTransformStyle, splitProps, splitStyle, useLayout, wrapChildren } from './utils';
12
+ import { useTransformStyle, splitProps, splitStyle, useLayout, wrapChildren, extendObject } from './utils';
13
13
  const _Form = forwardRef((fromProps, ref) => {
14
14
  const { textProps, innerProps: props = {} } = splitProps(fromProps);
15
15
  const { style, 'enable-var': enableVar, 'external-var-context': externalVarContext, 'parent-font-size': parentFontSize, 'parent-width': parentWidth, 'parent-height': parentHeight } = props;
16
16
  const { hasSelfPercent, normalStyle, hasVarDec, varContextRef, setWidth, setHeight } = useTransformStyle(style, { enableVar, externalVarContext, parentFontSize, parentWidth, parentHeight });
17
- const { textStyle, innerStyle } = splitStyle(normalStyle);
17
+ const { textStyle, innerStyle = {} } = splitStyle(normalStyle);
18
18
  const formRef = useRef(null);
19
- useNodesRef(props, ref, formRef);
19
+ useNodesRef(props, ref, formRef, {
20
+ style: normalStyle
21
+ });
20
22
  const propsRef = useRef({});
21
23
  propsRef.current = props;
22
24
  const { layoutRef, layoutStyle, layoutProps } = useLayout({ props, hasSelfPercent, setWidth, setHeight, nodeRef: formRef });
23
- const innerProps = useInnerProps(props, {
24
- style: { ...innerStyle, ...layoutStyle },
25
- ref: formRef,
26
- ...layoutProps
27
- }, [
25
+ const innerProps = useInnerProps(props, extendObject({
26
+ style: extendObject({}, innerStyle, layoutStyle),
27
+ ref: formRef
28
+ }, layoutProps), [
28
29
  'bindsubmit',
29
30
  'bindreset'
30
31
  ], { layoutRef });
@@ -56,16 +57,12 @@ const _Form = forwardRef((fromProps, ref) => {
56
57
  reset
57
58
  };
58
59
  }, []);
59
- return (<View {...innerProps}>
60
- <FormContext.Provider value={contextValue}>
61
- {wrapChildren(props, {
62
- hasVarDec,
63
- varContext: varContextRef.current,
64
- textStyle,
65
- textProps
66
- })}
67
- </FormContext.Provider>
68
- </View>);
60
+ return createElement(View, innerProps, createElement(FormContext.Provider, { value: contextValue }, wrapChildren(props, {
61
+ hasVarDec,
62
+ varContext: varContextRef.current,
63
+ textStyle,
64
+ textProps
65
+ })));
69
66
  });
70
67
  _Form.displayName = 'MpxForm';
71
68
  export default _Form;
@@ -3,11 +3,11 @@
3
3
  * ✔ size
4
4
  * ✔ color
5
5
  */
6
- import { forwardRef, useRef } from 'react';
6
+ import { forwardRef, useRef, createElement } from 'react';
7
7
  import { Image } from 'react-native';
8
8
  import useInnerProps from './getInnerListeners';
9
9
  import useNodesRef from './useNodesRef';
10
- import { useLayout, useTransformStyle } from './utils';
10
+ import { useLayout, useTransformStyle, extendObject } from './utils';
11
11
  const IconTypeMap = new Map([
12
12
  ['success', ''],
13
13
  ['success_no_circle', ''],
@@ -23,27 +23,19 @@ const Icon = forwardRef((props, ref) => {
23
23
  const { type, size = 23, color, style = {}, 'enable-var': enableVar, 'external-var-context': externalVarContext, 'parent-font-size': parentFontSize, 'parent-width': parentWidth, 'parent-height': parentHeight } = props;
24
24
  const uri = IconTypeMap.get(type);
25
25
  const defaultStyle = { width: ~~size, height: ~~size };
26
- const styleObj = {
27
- ...defaultStyle,
28
- ...style
29
- };
26
+ const styleObj = extendObject({}, defaultStyle, style);
30
27
  const { hasSelfPercent, normalStyle, setWidth, setHeight } = useTransformStyle(styleObj, { enableVar, externalVarContext, parentFontSize, parentWidth, parentHeight });
31
28
  const nodeRef = useRef(null);
32
- useNodesRef(props, ref, nodeRef, { defaultStyle });
29
+ useNodesRef(props, ref, nodeRef, { style: normalStyle });
33
30
  const { layoutRef, layoutStyle, layoutProps } = useLayout({ props, hasSelfPercent, setWidth, setHeight, nodeRef });
34
- const innerProps = useInnerProps(props, {
31
+ const innerProps = useInnerProps(props, extendObject({
35
32
  ref: nodeRef,
36
- style: {
37
- ...normalStyle,
38
- ...layoutStyle,
39
- tintColor: color
40
- },
41
33
  source: { uri },
42
- ...layoutProps
43
- }, [], {
34
+ style: extendObject({}, normalStyle, layoutStyle, { tintColor: color })
35
+ }, layoutProps), [], {
44
36
  layoutRef
45
37
  });
46
- return <Image {...innerProps}/>;
38
+ return createElement(Image, innerProps);
47
39
  });
48
40
  Icon.displayName = 'MpxIcon';
49
41
  export default Icon;
@@ -0,0 +1,291 @@
1
+ /**
2
+ * ✔ src
3
+ * ✔ mode
4
+ * ✘ show-menu-by-longpress
5
+ * ✔ binderror
6
+ * ✔ bindload
7
+ * ✘ fade-in
8
+ * ✔ webp
9
+ * ✘ lazy-load
10
+ * ✔ bindtap
11
+ * ✔ DEFAULT_SIZE
12
+ */
13
+ import { useEffect, useMemo, useState, useRef, forwardRef, createElement } from 'react';
14
+ import { Image as RNImage, View } from 'react-native';
15
+ import { noop } from '@mpxjs/utils';
16
+ import { SvgCssUri } from 'react-native-svg/css';
17
+ import useInnerProps, { getCustomEvent } from './getInnerListeners';
18
+ import useNodesRef from './useNodesRef';
19
+ import { SVG_REGEXP, useLayout, useTransformStyle, renderImage, extendObject } from './utils';
20
+ const DEFAULT_IMAGE_WIDTH = 320;
21
+ const DEFAULT_IMAGE_HEIGHT = 240;
22
+ const cropMode = [
23
+ 'top',
24
+ 'bottom',
25
+ 'center',
26
+ 'right',
27
+ 'left',
28
+ 'top left',
29
+ 'top right',
30
+ 'bottom left',
31
+ 'bottom right'
32
+ ];
33
+ const ModeMap = new Map([
34
+ ['scaleToFill', 'stretch'],
35
+ ['aspectFit', 'contain'],
36
+ ['aspectFill', 'cover'],
37
+ ['widthFix', 'stretch'],
38
+ ['heightFix', 'stretch'],
39
+ ...cropMode.map(mode => [mode, 'stretch'])
40
+ ]);
41
+ const isNumber = (value) => typeof value === 'number';
42
+ const relativeCenteredSize = (viewSize, imageSize) => (viewSize - imageSize) / 2;
43
+ function noMeetCalcRule(isSvg, mode, viewWidth, viewHeight, ratio) {
44
+ const isMeetSize = viewWidth && viewHeight && ratio;
45
+ if (isSvg && !isMeetSize)
46
+ return true;
47
+ if (!isSvg && !['scaleToFill', 'aspectFit', 'aspectFill'].includes(mode) && !isMeetSize)
48
+ return true;
49
+ return false;
50
+ }
51
+ const Image = forwardRef((props, ref) => {
52
+ const { src = '', mode = 'scaleToFill', style = {}, 'enable-var': enableVar, 'external-var-context': externalVarContext, 'parent-font-size': parentFontSize, 'enable-fast-image': enableFastImage, 'parent-width': parentWidth, 'parent-height': parentHeight, bindload, binderror } = props;
53
+ const defaultStyle = {
54
+ width: DEFAULT_IMAGE_WIDTH,
55
+ height: DEFAULT_IMAGE_HEIGHT
56
+ };
57
+ const styleObj = extendObject({}, defaultStyle, style, { overflow: 'hidden' });
58
+ const state = useRef({});
59
+ const nodeRef = useRef(null);
60
+ useNodesRef(props, ref, nodeRef, {
61
+ defaultStyle
62
+ });
63
+ const isSvg = SVG_REGEXP.test(src);
64
+ const isWidthFixMode = mode === 'widthFix';
65
+ const isHeightFixMode = mode === 'heightFix';
66
+ const isCropMode = cropMode.includes(mode);
67
+ const isLayoutMode = isWidthFixMode || isHeightFixMode || isCropMode;
68
+ const resizeMode = ModeMap.get(mode) || 'stretch';
69
+ const onLayout = ({ nativeEvent: { layout: { width, height } } }) => {
70
+ state.current.viewWidth = width;
71
+ state.current.viewHeight = height;
72
+ if (state.current.imageWidth && state.current.imageHeight && state.current.ratio) {
73
+ setViewWidth(width);
74
+ setViewHeight(height);
75
+ setRatio(state.current.ratio);
76
+ setImageWidth(state.current.imageWidth);
77
+ setImageHeight(state.current.imageHeight);
78
+ state.current = {};
79
+ setLoaded(true);
80
+ }
81
+ };
82
+ const { normalStyle, hasSelfPercent, setWidth, setHeight } = useTransformStyle(styleObj, { enableVar, externalVarContext, parentFontSize, parentWidth, parentHeight });
83
+ const { layoutRef, layoutStyle, layoutProps } = useLayout({
84
+ props,
85
+ hasSelfPercent,
86
+ setWidth,
87
+ setHeight,
88
+ nodeRef,
89
+ onLayout: isLayoutMode ? onLayout : noop
90
+ });
91
+ const { width, height } = normalStyle;
92
+ const [viewWidth, setViewWidth] = useState(isNumber(width) ? width : 0);
93
+ const [viewHeight, setViewHeight] = useState(isNumber(height) ? height : 0);
94
+ const [imageWidth, setImageWidth] = useState(0);
95
+ const [imageHeight, setImageHeight] = useState(0);
96
+ const [ratio, setRatio] = useState(0);
97
+ const [loaded, setLoaded] = useState(!isLayoutMode);
98
+ const fixedHeight = useMemo(() => {
99
+ const fixed = viewWidth * ratio;
100
+ return !fixed ? viewHeight : fixed;
101
+ }, [ratio, viewWidth, viewHeight]);
102
+ const fixedWidth = useMemo(() => {
103
+ if (!ratio)
104
+ return viewWidth;
105
+ const fixed = viewHeight / ratio;
106
+ return !fixed ? viewWidth : fixed;
107
+ }, [ratio, viewWidth, viewHeight]);
108
+ const modeStyle = useMemo(() => {
109
+ if (noMeetCalcRule(isSvg, mode, viewWidth, viewHeight, ratio))
110
+ return {};
111
+ switch (mode) {
112
+ case 'scaleToFill':
113
+ case 'aspectFit':
114
+ if (isSvg) {
115
+ const scale = ratio <= 1
116
+ ? imageWidth >= viewWidth ? viewWidth / imageWidth : imageWidth / viewWidth
117
+ : imageHeight >= viewHeight ? viewHeight / imageHeight : imageHeight / viewHeight;
118
+ return {
119
+ transform: [
120
+ { scale },
121
+ ratio <= 1 ? { translateY: -(imageHeight * scale - viewHeight) / 2 / scale } : { translateX: -(imageWidth * scale - viewWidth) / 2 / scale }
122
+ ]
123
+ };
124
+ }
125
+ return {};
126
+ case 'aspectFill':
127
+ if (isSvg) {
128
+ const scale = ratio >= 1
129
+ ? imageWidth >= viewWidth ? viewWidth / imageWidth : imageWidth / viewWidth
130
+ : imageHeight >= viewHeight ? viewHeight / imageHeight : imageHeight / viewHeight;
131
+ return {
132
+ transform: [
133
+ { scale },
134
+ ratio >= 1 ? { translateY: -(imageHeight * scale - viewHeight) / 2 / scale } : { translateX: -(imageWidth * scale - viewWidth) / 2 / scale }
135
+ ]
136
+ };
137
+ }
138
+ return {};
139
+ case 'widthFix':
140
+ case 'heightFix':
141
+ if (isSvg) {
142
+ const scale = ratio >= 1
143
+ ? imageWidth >= fixedWidth ? fixedWidth / imageWidth : imageWidth / fixedWidth
144
+ : imageHeight >= fixedHeight ? fixedHeight / imageHeight : imageHeight / fixedHeight;
145
+ return {
146
+ transform: [{ scale }]
147
+ };
148
+ }
149
+ return {};
150
+ case 'top':
151
+ return {
152
+ transform: [
153
+ { translateX: relativeCenteredSize(viewWidth, imageWidth) }
154
+ ]
155
+ };
156
+ case 'bottom':
157
+ return {
158
+ transform: [
159
+ { translateY: viewHeight - imageHeight },
160
+ { translateX: relativeCenteredSize(viewWidth, imageWidth) }
161
+ ]
162
+ };
163
+ case 'center':
164
+ return {
165
+ transform: [
166
+ { translateY: relativeCenteredSize(viewHeight, imageHeight) },
167
+ { translateX: relativeCenteredSize(viewWidth, imageWidth) }
168
+ ]
169
+ };
170
+ case 'left':
171
+ return {
172
+ transform: [
173
+ { translateY: relativeCenteredSize(viewHeight, imageHeight) }
174
+ ]
175
+ };
176
+ case 'right':
177
+ return {
178
+ transform: [
179
+ { translateY: relativeCenteredSize(viewHeight, imageHeight) },
180
+ { translateX: viewWidth - imageWidth }
181
+ ]
182
+ };
183
+ case 'top left':
184
+ return {};
185
+ case 'top right':
186
+ return {
187
+ transform: [
188
+ { translateX: viewWidth - imageWidth }
189
+ ]
190
+ };
191
+ case 'bottom left':
192
+ return {
193
+ transform: [
194
+ { translateY: viewHeight - imageHeight }
195
+ ]
196
+ };
197
+ case 'bottom right':
198
+ return {
199
+ transform: [
200
+ { translateY: viewHeight - imageHeight },
201
+ { translateX: viewWidth - imageWidth }
202
+ ]
203
+ };
204
+ default:
205
+ return {};
206
+ }
207
+ }, [isSvg, mode, viewWidth, viewHeight, imageWidth, imageHeight, ratio, fixedWidth, fixedHeight]);
208
+ const onSvgLoad = (evt) => {
209
+ const { width, height } = evt.nativeEvent.layout;
210
+ setRatio(!width ? 0 : height / width);
211
+ setImageWidth(width);
212
+ setImageHeight(height);
213
+ bindload && bindload(getCustomEvent('load', evt, {
214
+ detail: { width, height },
215
+ layoutRef
216
+ }, props));
217
+ };
218
+ const onSvgError = (evt) => {
219
+ binderror(getCustomEvent('error', evt, {
220
+ detail: { errMsg: evt?.message },
221
+ layoutRef
222
+ }, props));
223
+ };
224
+ const onImageLoad = (evt) => {
225
+ evt.persist();
226
+ RNImage.getSize(src, (width, height) => {
227
+ bindload(getCustomEvent('load', evt, {
228
+ detail: { width, height },
229
+ layoutRef
230
+ }, props));
231
+ });
232
+ };
233
+ const onImageError = (evt) => {
234
+ binderror(getCustomEvent('error', evt, {
235
+ detail: { errMsg: evt.nativeEvent.error },
236
+ layoutRef
237
+ }, props));
238
+ };
239
+ useEffect(() => {
240
+ if (!isSvg && isLayoutMode) {
241
+ RNImage.getSize(src, (width, height) => {
242
+ state.current.imageWidth = width;
243
+ state.current.imageHeight = height;
244
+ state.current.ratio = !width ? 0 : height / width;
245
+ if (isWidthFixMode
246
+ ? state.current.viewWidth
247
+ : isHeightFixMode
248
+ ? state.current.viewHeight
249
+ : state.current.viewWidth && state.current.viewHeight) {
250
+ state.current.viewWidth && setViewWidth(state.current.viewWidth);
251
+ state.current.viewHeight && setViewHeight(state.current.viewHeight);
252
+ setRatio(!width ? 0 : height / width);
253
+ setImageWidth(width);
254
+ setImageHeight(height);
255
+ state.current = {};
256
+ setLoaded(true);
257
+ }
258
+ });
259
+ }
260
+ }, [src, isSvg, isLayoutMode]);
261
+ const innerProps = useInnerProps(props, extendObject({
262
+ ref: nodeRef,
263
+ style: extendObject({}, normalStyle, layoutStyle, isHeightFixMode ? { width: fixedWidth } : {}, isWidthFixMode ? { height: fixedHeight } : {})
264
+ }, layoutProps), [
265
+ 'src',
266
+ 'mode',
267
+ 'svg'
268
+ ], {
269
+ layoutRef
270
+ });
271
+ return createElement(View, innerProps, isSvg
272
+ ? createElement(SvgCssUri, {
273
+ uri: src,
274
+ onLayout: onSvgLoad,
275
+ onError: binderror && onSvgError,
276
+ style: extendObject({ transformOrigin: 'top left' }, modeStyle)
277
+ })
278
+ : loaded && renderImage({
279
+ source: { uri: src },
280
+ resizeMode: resizeMode,
281
+ onLoad: bindload && onImageLoad,
282
+ onError: binderror && onImageError,
283
+ style: extendObject({
284
+ transformOrigin: 'top left',
285
+ width: isCropMode ? imageWidth : '100%',
286
+ height: isCropMode ? imageHeight : '100%'
287
+ }, isCropMode ? modeStyle : {})
288
+ }, enableFastImage));
289
+ });
290
+ Image.displayName = 'mpx-image';
291
+ export default Image;