UNPKG

@mpxjs/webpack-plugin

Version:

mpx compile core

276 lines (275 loc) 13 kB
/** * ✔ size * ✔ type * ✔ plain * ✔ disabled * ✔ loading * ✔ form-type * - open-type: Partially. Only support `share`、`getUserInfo` * ✔ hover-class: Convert hoverClass to hoverStyle. * ✔ hover-style * ✘ hover-stop-propagation * ✔ hover-start-time * ✔ hover-stay-time * ✘ lang * ✘ session-from * ✘ send-message-title * ✘ send-message-path * ✘ send-message-img * ✘ app-parameter * ✘ show-message-card * ✘ phone-number-no-quota-toast * ✘ bindgetuserinfo * ✘ bindcontact * ✘ createliveactivity * ✘ bindgetphonenumber * ✘ bindgetphonenumber * ✘ bindgetrealtimephonenumber * ✘ binderror * ✘ bindopensetting * ✘ bindlaunchapp * ✘ bindlaunchapp * ✘ bindchooseavatar * ✘ bindchooseavatar * ✘ bindagreeprivacyauthorization * ✔ bindtap */ import { createElement, useEffect, useRef, forwardRef, useContext } from 'react'; import { View, StyleSheet, Animated, Easing, useAnimatedValue } from 'react-native'; import { warn } from '@mpxjs/utils'; import { GestureDetector } from 'react-native-gesture-handler'; import { getCurrentPage, splitProps, splitStyle, useLayout, useTransformStyle, wrapChildren, extendObject, useHover } from './utils'; import useInnerProps, { getCustomEvent } from './getInnerListeners'; import useNodesRef from './useNodesRef'; import { RouteContext, FormContext } from './context'; import Portal from './mpx-portal'; const LOADING_IMAGE_URI = ''; const TypeColorMap = { default: ['#F8F8F8', '#DEDEDE', '35,35,35', '#F7F7F7'], primary: ['#1AAD19', '#179B16', '26,173,25', '#9ED99D'], warn: ['#E64340', '#CE3C39', '230,67,64', '#EC8B89'] }; const OpenTypeEventsMap = new Map([ ['share', 'onShareAppMessage'], ['getUserInfo', 'onUserInfo'] ]); const styles = StyleSheet.create({ button: { width: '100%', flexDirection: 'row', justifyContent: 'center', alignItems: 'center', height: 46, borderRadius: 5, backgroundColor: '#F8F8F8', marginHorizontal: 'auto' // 按钮默认居中 }, buttonMini: { height: 30 }, text: { fontSize: 18, color: '#000000' }, textMini: { fontSize: 13 }, loading: { width: 20, height: 20 } }); const getOpenTypeEvent = (openType) => { if (!openType) return; if (!global.__mpx?.config?.rnConfig) { warn('Environment not supported'); return; } const eventName = OpenTypeEventsMap.get(openType); if (!eventName) { warn(`open-type not support ${openType}`); return; } const event = global.__mpx.config.rnConfig.openTypeHandler?.[eventName]; if (!event) { warn(`Unregistered ${eventName} event`); return; } return event; }; const timer = (data, time = 3000) => new Promise((resolve) => { setTimeout(() => { resolve(data); }, time); }); const Loading = ({ alone = false }) => { const image = useAnimatedValue(0); const rotate = image.interpolate({ inputRange: [0, 1], outputRange: ['0deg', '360deg'] }); useEffect(() => { const animation = Animated.loop(Animated.timing(image, { toValue: 1, duration: 1000, easing: Easing.linear, useNativeDriver: true, isInteraction: false })); animation.start(); return () => { animation.stop(); }; }, []); const loadingStyle = extendObject({}, styles.loading, { transform: [{ rotate }], marginRight: alone ? 0 : 5 }); return <Animated.Image testID="loading" style={loadingStyle} source={{ uri: LOADING_IMAGE_URI }}/>; }; const Button = forwardRef((buttonProps, ref) => { const { textProps, innerProps: props = {} } = splitProps(buttonProps); const { size = 'default', type = 'default', plain = false, disabled = false, loading = false, 'hover-class': hoverClass, 'hover-style': hoverStyle = {}, 'hover-start-time': hoverStartTime = 20, 'hover-stay-time': hoverStayTime = 70, 'open-type': openType, 'form-type': formType, 'enable-var': enableVar, 'external-var-context': externalVarContext, 'parent-font-size': parentFontSize, 'parent-width': parentWidth, 'parent-height': parentHeight, style = {}, children, bindgetuserinfo, bindtap } = props; const { pageId } = useContext(RouteContext) || {}; const formContext = useContext(FormContext); const enableHover = hoverClass !== 'none'; const { isHover, gesture } = useHover({ enableHover, hoverStartTime, hoverStayTime, disabled }); let submitFn; let resetFn; if (formContext) { submitFn = formContext.submit; resetFn = formContext.reset; } const isMiniSize = size === 'mini'; const [color, hoverColor, plainColor, disabledColor] = TypeColorMap[type]; const normalBackgroundColor = disabled ? disabledColor : isHover || loading ? hoverColor : color; const plainBorderColor = disabled ? 'rgba(0, 0, 0, .2)' : isHover ? `rgba(${plainColor},.6)` : `rgb(${plainColor})`; const normalBorderColor = type === 'default' ? 'rgba(0, 0, 0, .2)' : normalBackgroundColor; const plainTextColor = disabled ? 'rgba(0, 0, 0, .2)' : isHover ? `rgba(${plainColor}, .6)` : `rgb(${plainColor})`; const normalTextColor = type === 'default' ? `rgba(0, 0, 0, ${disabled ? 0.3 : isHover || loading ? 0.6 : 1})` : `rgba(255 ,255 ,255 , ${disabled || isHover || loading ? 0.6 : 1})`; const viewStyle = { borderWidth: 1, borderStyle: 'solid', borderColor: plain ? plainBorderColor : normalBorderColor, backgroundColor: plain ? 'transparent' : normalBackgroundColor }; const defaultViewStyle = extendObject({}, styles.button, isMiniSize ? styles.buttonMini : null, viewStyle); const defaultTextStyle = extendObject({}, styles.text, isMiniSize ? styles.textMini : {}, { color: plain ? plainTextColor : normalTextColor }); const defaultStyle = extendObject({}, defaultViewStyle, defaultTextStyle); const styleObj = extendObject({}, defaultStyle, style, isHover ? hoverStyle : {}); const { hasPositionFixed, hasSelfPercent, normalStyle, hasVarDec, varContextRef, setWidth, setHeight } = useTransformStyle(styleObj, { enableVar, externalVarContext, parentFontSize, parentWidth, parentHeight }); const nodeRef = useRef(null); useNodesRef(props, ref, nodeRef, { style: normalStyle }); const { layoutRef, layoutStyle, layoutProps } = useLayout({ props, hasSelfPercent, setWidth, setHeight, nodeRef }); const { textStyle, backgroundStyle, innerStyle = {} } = splitStyle(normalStyle); if (backgroundStyle) { warn('Button does not support background image-related styles!'); } const handleOpenTypeEvent = (evt) => { const handleEvent = getOpenTypeEvent(openType); if (!handleEvent) return; if (openType === 'share') { const currentPage = getCurrentPage(pageId); const event = { from: 'button', target: getCustomEvent('tap', evt, { layoutRef }, props).target, webViewUrl: currentPage?.__webViewUrl }; if (currentPage) { const defaultMessage = { title: global.__mpx.config.rnConfig.projectName || 'AwesomeProject', path: currentPage.route || '' }; if (currentPage.onShareAppMessage) { const { promise, ...message } = currentPage.onShareAppMessage(event) || {}; if (promise) { Promise.race([Promise.resolve(promise), timer(message)]) .then((msg) => { handleEvent(Object.assign({}, defaultMessage, msg)); }); } else { handleEvent(Object.assign({}, defaultMessage, message)); } } else { handleEvent(defaultMessage); } } else { warn('Current page not found'); // Todo handleEvent(event) } } if (openType === 'getUserInfo' && bindgetuserinfo) { Promise.resolve(handleEvent) .then((userInfo) => { if (typeof userInfo === 'object') { bindgetuserinfo(userInfo); } }); } }; const handleFormTypeFn = () => { if (formType === 'submit') { submitFn && submitFn(); } else if (formType === 'reset') { resetFn && resetFn(); } }; const onTap = (evt) => { if (disabled) return; bindtap && bindtap(getCustomEvent('tap', evt, { layoutRef }, props)); handleOpenTypeEvent(evt); handleFormTypeFn(); }; const innerProps = useInnerProps(extendObject({}, props, layoutProps, { ref: nodeRef, style: extendObject({}, innerStyle, layoutStyle), bindtap: !disabled && onTap }), [ 'disabled', 'size', 'type', 'plain', 'loading', 'hover-class', 'hover-style', 'hover-start-time', 'hover-stay-time', 'open-type', 'form-type' ], { layoutRef, disableTap: disabled }); const baseButton = createElement(View, innerProps, loading && createElement(Loading, { alone: !children }), wrapChildren(props, { hasVarDec, varContext: varContextRef.current, textStyle, textProps })); const finalComponent = enableHover ? createElement(GestureDetector, { gesture: gesture }, baseButton) : baseButton; if (hasPositionFixed) { return createElement(Portal, null, finalComponent); } return finalComponent; }); Button.displayName = 'MpxButton'; export default Button;