@mpxjs/core
Version:
mpx runtime core
138 lines (131 loc) • 4.55 kB
JavaScript
import { createElement, useState, useMemo, memo } from 'react'
import { useSafeAreaInsets } from 'react-native-safe-area-context'
import { StatusBar, processColor, TouchableWithoutFeedback, Image, View, StyleSheet, Text } from 'react-native'
import Mpx from '../../index'
function convertToHex (color) {
try {
const intColor = processColor(color)
if (intColor === null || intColor === undefined) {
return null
}
// 将32位整数颜色值转换为RGBA
const r = (intColor >> 16) & 255
const g = (intColor >> 8) & 255
const b = intColor & 255
// 转换为十六进制
const hexR = r.toString(16).padStart(2, '0')
const hexG = g.toString(16).padStart(2, '0')
const hexB = b.toString(16).padStart(2, '0')
return `#${hexR}${hexG}${hexB}`
} catch (error) {
return null
}
}
const titleHeight = 44
export function useInnerHeaderHeight (pageconfig) {
if (pageconfig.navigationStyle === 'custom') {
return 0
} else {
const safeAreaTop = useSafeAreaInsets()?.top || 0
const headerHeight = safeAreaTop + titleHeight
return headerHeight
}
}
const styles = StyleSheet.create({
header: {
elevation: 3
},
headerContent: {
flexDirection: 'row',
alignItems: 'center',
justifyContent: 'center'
},
backButton: {
position: 'absolute',
height: '100%',
width: 40,
left: 0,
top: 0,
alignItems: 'center',
justifyContent: 'center'
},
backButtonImage: {
width: 22,
height: 22
},
title: {
fontSize: 17,
fontWeight: 600,
width: '60%',
textAlign: 'center'
}
})
const NavColor = {
White: '#ffffff',
Black: '#000000'
}
// navigationBarTextStyle只支持黑白'white'/'black
const validBarTextStyle = (textStyle) => {
const textStyleColor = convertToHex(textStyle)
if (textStyle && [NavColor.White, NavColor.Black].includes(textStyleColor)) {
return textStyleColor
} else {
return NavColor.White
}
}
export const innerNav = memo(({ pageConfig, navigation }) => {
const [innerPageConfig, setPageConfig] = useState(pageConfig || {})
navigation.setPageConfig = (config) => {
const newConfig = Object.assign({}, innerPageConfig, config)
setPageConfig(newConfig)
}
const isCustom = innerPageConfig.navigationStyle === 'custom'
const navigationBarTextStyle = useMemo(() => validBarTextStyle(innerPageConfig.navigationBarTextStyle), [innerPageConfig.navigationBarTextStyle])
// 状态栏的颜色
const statusBarElement = createElement(StatusBar, {
translucent: true,
backgroundColor: 'transparent',
barStyle: (navigationBarTextStyle === NavColor.White) ? 'light-content' : 'dark-content' // 'default'/'light-content'/'dark-content'
})
if (isCustom) return statusBarElement
const safeAreaTop = useSafeAreaInsets()?.top || 0
// 假设是栈导航,获取栈的长度
const stackLength = navigation.getState()?.routes?.length
const onStackTopBack = Mpx.config?.rnConfig?.onStackTopBack
const isHandleStackTopBack = typeof onStackTopBack === 'function'
// 回退按钮与图标
const backElement = stackLength > 1 || isHandleStackTopBack
? createElement(TouchableWithoutFeedback, {
onPress: () => {
if (stackLength <= 1 && isHandleStackTopBack) {
onStackTopBack()
return
}
navigation.goBack()
}
}, createElement(View, {
style: [styles.backButton]
}, createElement(Image, {
source: { uri: '' },
// 回退按钮的颜色与设置的title文案颜色一致
style: [styles.backButtonImage, { tintColor: navigationBarTextStyle }]
})
))
: null
return createElement(View, {
style: [styles.header, {
paddingTop: safeAreaTop,
backgroundColor: innerPageConfig.navigationBarBackgroundColor || '#000000'
}]
},
statusBarElement,
createElement(View, {
style: styles.headerContent,
height: titleHeight
}, backElement,
createElement(Text, {
style: [styles.title, { color: navigationBarTextStyle }],
numberOfLines: 1
}, innerPageConfig.navigationBarTitleText?.trim() || ''))
)
})