gpt4 book ai didi

react-native - RN OneSignal _open 事件

转载 作者:行者123 更新时间:2023-12-04 17:17:25 27 4
gpt4 key购买 nike

主屏幕启动后,通知打开事件上的 OneSignal 会触发,然后导航到所需的屏幕。我想检测应用程序是否在主屏幕呈现之前按下通知启动,以便我可以直接导航到第二个屏幕并避免不必要地调用 api。

  • "react-native-onesignal": "^3.9.3"
  • "react-navigation": "^4.0.0"

代码
   const _opened = openResult => {
const { additionalData, body } = openResult.notification.payload;
// how to navigate or set the initial screen depending on the payload
}

useEffect(() => {

onesignal.init();
onesignal.addEventListener('received', _received);
onesignal.addEventListener('opened', _opened);
SplashScreen.hide();

return () => {
// unsubscriber
onesignal.removeEventListener('received', _received);
onesignal.removeEventListener('opened', _opened);
}
}, []);
调试
enter image description here

最佳答案

您的问题是如何根据打开的通知负载导航或设置初始屏幕?
1) - 设置 initial screen取决于打开的通知负载。
根据 class Lifecycle useEffect在组件输出渲染后运行,所以监听器在 useEffect直到组件数量增加才监听,这就是登录前显示的登录主屏幕的原因useEffect ,请参阅此说明。

//this the problem (NavigationContainer called before useEffect).
function App() {
useEffect(() => {}); //called second.
return <NavigationContainer>; //called first.
}

//this the solution (useEffect called Before NavigationContainer).
function App() {
const [ready, setReady] = useState(false);

//called second.
useEffect(() => {
//listen here
setReady(true);
SplashScreen.hide();
});

//called first
//no function or apis run before useEffect here it just view.
if(!ready) return <></>;// or <LoadingView/>

//called third.
return <NavigationContainer>;
}
你的代码可能是这样的。
function App() {
const [ready, setReady] = useState(false);

const openedNotificationRef = useRef(null);

const _opened = openResult => {
openedNotificationRef.current = openResult.notification.payload;
}

const getInitialRouteName = () => {
if (openedNotificationRef.current) {
return "second"; //or what you want depending on the notification.
}
return "home";
}


useEffect(() => {
onesignal.addEventListener('opened', _opened);
//setTimeout(fn, 0) mean function cannot run until the stack on the main thread is empty.
//this ensure _opened is executed if app is opened from notification
setTimeout(() => {
setReady(true);
}, 0)
});


if(!ready) return <LoadingView/>


return (
<NavigationContainer initialRouteName={getInitialRouteName()}>
</NavigationContainer>
);

}
2) - navigate取决于打开的通知负载。
首先你需要知道

A navigator needs to be rendered to be able to handle actions If youtry to navigate without rendering a navigator or before the navigatorfinishes mounting, it will throw and crash your app if not handled. Soyou'll need to add an additional check to decide what to do until yourapp mounts.


阅读 docs

function App() {

const navigationRef = React.useRef(null);

const openedNotificationRef = useRef(null);

const _opened = openResult => {
openedNotificationRef.current = openResult.notification.payload;
//remove loading screen and start with what you want.
const routes = [
{name : 'home'}, //recommended add this to handle navigation go back
{name : 'orders'}, //recommended add this to handle navigation go back
{name : 'order', params : {id : payload.id}},
]
navigationRef.current.dispatch(
CommonActions.reset({
routes : routes,
index: routes.length - 1,
})
)
}

useEffect(() => {
//don't subscribe to `opened` here

//unsubscribe
return () => {
onesignal.removeEventListener('opened', _opened);
}
}, []);

//subscribe to `opened` after navigation is ready to can use navigate
const onReady = () => {
onesignal.addEventListener('opened', _opened);
//setTimeout(fn, 0) mean function cannot run until the stack on the main thread is empty.
//this ensure _opened is executed if app is opened from notification
setTimeout(() => {
if (!openedNotificationRef.current) {
//remove loading screen and start with home
navigationRef.current.dispatch(
CommonActions.reset({
routes : [{name : 'home'}],
index: 0,
})
)
}
}, 0)
};


return (
<NavigationContainer
ref={navigationRef}
onReady={onReady}
initialRouteName={"justLoadingScreen"}>
</NavigationContainer>
);

}

引用 setTimeout , CommonActions .

关于react-native - RN OneSignal _open 事件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/68379185/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com