Skip to content

Commit

Permalink
改造portal
Browse files Browse the repository at this point in the history
  • Loading branch information
wangcuijuan committed Jan 21, 2025
1 parent 638bc95 commit 34f1e0a
Show file tree
Hide file tree
Showing 9 changed files with 118 additions and 35 deletions.
12 changes: 11 additions & 1 deletion packages/api-proxy/src/common/js/utils.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { hasOwn, noop, getEnvObj, getFocusedNavigation } from '@mpxjs/utils'
import { getCurrentInstance } from '@mpxjs/core'

/**
*
Expand Down Expand Up @@ -87,6 +88,14 @@ function failHandle (result, fail, complete) {
typeof complete === 'function' && complete(result)
}

function getPageId () {
const navigation = getFocusedNavigation()
const currentInstance = getCurrentInstance()
console.log(currentInstance, currentInstance?.getPageId, navigation, navigation?.pageId)
const id = currentInstance?.getPageId || navigation?.pageId || null
return id
}

const ENV_OBJ = getEnvObj()

export {
Expand All @@ -101,5 +110,6 @@ export {
defineUnsupportedProps,
successHandle,
failHandle,
getFocusedNavigation
getFocusedNavigation,
getPageId
}
14 changes: 11 additions & 3 deletions packages/api-proxy/src/platform/api/action-sheet/rnActionSheet.jsx
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { View, Text, StyleSheet } from 'react-native'
import { successHandle, failHandle } from '../../../common/js'
import { successHandle, failHandle, getPageId, error } from '../../../common/js'
import Portal from '@mpxjs/webpack-plugin/lib/runtime/components/react/dist/mpx-portal/index'
import { getWindowInfo } from '../system/rnSystem'
import Animated, {
Expand All @@ -9,7 +9,16 @@ import Animated, {
} from 'react-native-reanimated'

function showActionSheet (options = {}) {
const id = getPageId()
const { alertText, itemList = [], itemColor = '#000000', success, fail, complete } = options
if (id === null) {
error('showActionSheet cannot be invoked outside the mpx life cycle in React Native environments')
const result = {
errMsg: 'showActionSheet:fail cannot be invoked outside the mpx life cycle in React Native environments'
}
failHandle(result, fail, complete)
return
}
if (itemList.length > 6) {
const result = {
errMsg: 'showActionSheet:fail parameter error: itemList should not be large than 6'
Expand Down Expand Up @@ -116,8 +125,7 @@ function showActionSheet (options = {}) {
</View>
)
}

actionSheetKey = Portal.add(<ActionSheet/>)
actionSheetKey = Portal.add(<ActionSheet />, id)
}

export {
Expand Down
13 changes: 11 additions & 2 deletions packages/api-proxy/src/platform/api/modal/rnModal.jsx
Original file line number Diff line number Diff line change
@@ -1,8 +1,9 @@
import { View, Dimensions, Text, StyleSheet, TouchableOpacity, ScrollView, TextInput } from 'react-native'
import { successHandle, failHandle } from '../../../common/js'
import { successHandle, failHandle, getPageId, error } from '../../../common/js'
import Portal from '@mpxjs/webpack-plugin/lib/runtime/components/react/dist/mpx-portal/index'
const { width, height } = Dimensions.get('window')
const showModal = function (options = {}) {
const id = getPageId()
const {
title,
content,
Expand All @@ -17,6 +18,14 @@ const showModal = function (options = {}) {
fail,
complete
} = options
if (id === null) {
error('showModal cannot be invoked outside the mpx life cycle in React Native environments')
const result = {
errMsg: 'showModal:fail cannot be invoked outside the mpx life cycle in React Native environments'
}
failHandle(result, fail, complete)
return
}
const modalWidth = width * 0.8
const styles = StyleSheet.create({
modalTask: {
Expand Down Expand Up @@ -156,7 +165,7 @@ const showModal = function (options = {}) {
</View>
</View>
try {
modalKey = Portal.add(ModalView)
modalKey = Portal.add(ModalView, id)
} catch (e) {
const result = {
errMsg: `showModal:fail invalid ${e}`
Expand Down
4 changes: 2 additions & 2 deletions packages/api-proxy/src/platform/api/route/index.ios.js
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@ function resolvePath (relative, base) {
}
return stack.join('/')
}
function isLock(navigationHelper, type, options) {
function isLock (navigationHelper, type, options) {
if (navigationHelper.lastSuccessCallback && navigationHelper.lastFailCallback) {
const res = { errMsg: `${type}:fail the previous routing event didn't complete` }
failHandle(res, options.fail, options.complete)
Expand All @@ -48,7 +48,7 @@ function isLock(navigationHelper, type, options) {
return false
}

function navigateTo (options = {}) {
function navigateTo (options = {}) {
const navigationHelper = global.__navigationHelper
if (isLock(navigationHelper, 'navigateTo', options)) {
return
Expand Down
13 changes: 11 additions & 2 deletions packages/api-proxy/src/platform/api/toast/rnToast.jsx
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { View, Text, Image, StyleSheet, ActivityIndicator, Dimensions } from 'react-native'
import { successHandle, failHandle } from '../../../common/js'
import { successHandle, failHandle, getPageId, error } from '../../../common/js'
import Portal from '@mpxjs/webpack-plugin/lib/runtime/components/react/dist/mpx-portal/index'

let toastKey
Expand Down Expand Up @@ -55,7 +55,16 @@ const styles = StyleSheet.create({
})

function showToast (options = {}) {
const id = getPageId()
const { title, icon = 'success', image, duration = 1500, mask = false, success, fail, complete, isLoading } = options
if (id === null) {
error('showToast cannot be invoked outside the mpx life cycle in React Native environments')
const result = {
errMsg: 'showToast:fail cannot be invoked outside the mpx life cycle in React Native environments'
}
failHandle(result, fail, complete)
return
}
let ToastView
const successPng = ''
const errorPng = ''
Expand Down Expand Up @@ -101,7 +110,7 @@ function showToast (options = {}) {
if (toastKey) {
Portal.remove(toastKey)
}
toastKey = Portal.add(ToastView)
toastKey = Portal.add(ToastView, id)
if (!isLoading) {
tId = setTimeout(() => {
Portal.remove(toastKey)
Expand Down
5 changes: 3 additions & 2 deletions packages/core/src/platform/patch/getDefaultOptions.ios.js
Original file line number Diff line number Diff line change
Expand Up @@ -489,7 +489,6 @@ export function getDefaultOptions ({ type, rawOptions = {}, currentInject }) {
const currentPageId = useMemo(() => ++pageId, [])
const intersectionObservers = useRef({})
usePageStatus(navigation, currentPageId)

useLayoutEffect(() => {
const isCustom = pageConfig.navigationStyle === 'custom'
navigation.setOptions({
Expand Down Expand Up @@ -566,7 +565,9 @@ export function getDefaultOptions ({ type, rawOptions = {}, currentInject }) {
value: intersectionObservers.current
},
createElement(Provider,
null,
{
pageId: currentPageId
},
createElement(defaultOptions,
{
navigation,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,7 @@ export interface PortalManagerContextValue {
}

export interface PortalContextValue {
mount: (children: React.ReactNode, key?: number) => number| undefined
mount: (children: React.ReactNode, key?: number, id?: number) => number| undefined
update: (key: number, children: React.ReactNode) => void
unmount: (key: number) => void
manager?: PortalManagerContextValue
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import { ReactNode } from 'react'
import { PortalContext, PortalContextValue } from '../context'
import PortalConsumer from './portal-consumer'
import { useNavigation } from '@react-navigation/native'
import PortalHost, { portal } from './portal-host'

export type PortalProps = {
Expand All @@ -9,10 +10,13 @@ export type PortalProps = {
*/
children?: ReactNode
key?: string
manager?: PortalContextValue
manager?: PortalContextValue,
pageId?: number
}

const Portal = ({ children }:PortalProps): JSX.Element => {
const navigation = useNavigation()
const pageId = navigation?.pageId
return (
<PortalContext.Consumer>
{(manager) => (
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { useEffect, useRef, ReactNode } from 'react'
import { useEffect, useRef, ReactNode, useMemo } from 'react'
import {
View,
DeviceEventEmitter,
Expand All @@ -11,7 +11,8 @@ import { useNavigation } from '@react-navigation/native'
import { PortalManagerContextValue, PortalContext } from '../context'

export type PortalHostProps = {
children: ReactNode
children: ReactNode,
pageId: number
}

type addIdsMapsType = {
Expand All @@ -38,9 +39,9 @@ const styles = StyleSheet.create({

class PortalGuard {
private nextKey = 10000
add = (e: ReactNode) => {
add = (e: ReactNode, id: number) => {
const key = this.nextKey++
TopViewEventEmitter.emit(addType, e, key)
TopViewEventEmitter.emit(addType, e, key, id)
return key
}

Expand All @@ -57,61 +58,102 @@ class PortalGuard {
*/
export const portal = new PortalGuard()

const PortalHost = ({ children } :PortalHostProps): JSX.Element => {
const PortalHost = ({ children, pageId } :PortalHostProps): JSX.Element => {
const isMounted = useRef<boolean>(false)
const _nextKey = useRef(0)
const _addType = useRef<EventSubscription | null>(null)
const _removeType = useRef<EventSubscription | null>(null)
const _updateType = useRef<EventSubscription | null>(null)
const manager = useRef<PortalManagerContextValue | null>(null)
const focusState = useRef<Boolean>(false)
const _mount = (children: ReactNode, _key?: number) => {
if (!focusState.current) {
return
const queue = useRef<Array<{ type: string, key: number; children: ReactNode }>>([])
const _mount = (children: ReactNode, _key?: number, id?: number) => {
if (id !== pageId) return
const key = _key || _nextKey.current++
if (!isMounted.current) {
queue.current.push({ type: 'mount', key, children })
} else if (manager.current) {
manager.current.mount(key, children)
}
return key
}

const _unmount = (key: number) => {
if (!isMounted.current) {
queue.current.push({ type: 'unmount', key, children })
} else if (manager.current) {
manager.current.unmount(key)
}
}

const _update = (key: number, children?: ReactNode) => {
if (!isMounted.current) {
const operation = { type: 'mount', key, children }
const index = queue.current.findIndex((q) => q.key === key)
if (index > -1) {
queue.current[index] = operation
} else {
queue.current.push(operation)
}
} else if (manager.current) {
manager.current.update(key, children)
}
}

const mount = (children: ReactNode, _key?: number) => {
const key = _key || _nextKey.current++
if (manager.current) {
manager.current.mount(key, children)
}
return key
}

const _unmount = (key: number) => {
const unmount = (key: number) => {
if (manager.current) {
manager.current.unmount(key)
}
}

const _update = (key: number, children?: ReactNode, curPageId?: number) => {
const update = (key: number, children?: ReactNode) => {
if (manager.current) {
manager.current.update(key, children)
}
}
const navigation = useNavigation()
useEffect(() => {

useMemo(() => {
_addType.current = TopViewEventEmitter.addListener(addType, _mount)
_removeType.current = TopViewEventEmitter.addListener(removeType, _unmount)
_updateType.current = TopViewEventEmitter.addListener(updateType, _update)
}, [])
const navigation = useNavigation()
useEffect(() => {
while (queue.current.length && manager.current) {
const operation = queue.current.shift()
if (!operation) return
switch (operation.type) {
case 'mount':
manager.current.mount(operation.key, operation.children)
break
case 'unmount':
manager.current.unmount(operation.key)
}
}
const focusSubscription = navigation.addListener('focus', () => {
focusState.current = true
})
const blurSubscription = navigation.addListener('blur', () => {
focusState.current = false
isMounted.current = true
})

return () => {
_addType.current?.remove()
_removeType.current?.remove()
_updateType.current?.remove()
focusSubscription()
blurSubscription()
}
}, [])
return (
<PortalContext.Provider
value={{
mount: _mount,
update: _update,
unmount: _unmount
mount,
update,
unmount
}}
>
<View style={styles.container} collapsable={false}>
Expand Down

0 comments on commit 34f1e0a

Please sign in to comment.