如何在初始渲染后重新渲染自定义钩子

IT技术 javascript reactjs react-hooks react-functional-component
2021-05-14 04:08:32

我有useIsUserSubscribed一个名为的自定义钩子,用于检查是否订阅了特定用户。如果用户已订阅,则返回 true,如果用户未订阅,则返回 false...

import { useState, useEffect } from "react";
import { useSelector } from "react-redux";
import { checkSubscription } from "../services";

// this hook checks if the current user is subscribed to a particular user(publisherId)
function useIsUserSubscribed(publisherId) {
  const [userIsSubscribed, setUserIsSubscribed] = useState(null);
  const currentUserId = useSelector((state) => state.auth.user?.id);

  useEffect(() => {
    if (!currentUserId || !publisherId) return;

    async function fetchCheckSubscriptionData() {
      try {
        const res = await checkSubscription(publisherId);
        setUserIsSubscribed(true);
      } catch (err) {
        setUserIsSubscribed(false);
      }
    }

    fetchCheckSubscriptionData();
  }, [publisherId, currentUserId]);

  return userIsSubscribed;
}

export default useIsUserSubscribed;

...我有一个使用这个钩子的按钮,它根据从useIsUserSubscribed...返回的布尔值有条件地呈现文本

import React, { useEffect, useState } from "react";
import { add, remove } from "../../services";
import useIsUserSubscribed from "../../hooks/useIsUserSubscribed";

const SubscribeUnsubscribeBtn = ({profilePageUserId}) => {

  const userIsSubscribed = useIsUserSubscribed(profilePageUserId);
  
  const onClick = async () => {
    if (userIsSubscribed) {
       // this is an API Call to the backend
      await removeSubscription(profilePageUserId);

    } else {
      // this is an API Call to the backend
      await addSubscription(profilePageUserId);
    }
    // HOW CAN I RERENDER THE HOOK HERE!!!!?
  }

  return (
    <button type="button" className="sub-edit-unsub-btn bsc-button" onClick={onClick}>
          {userIsSubscribed ? 'Subscribed' : 'Unsubscribed'}
    </button>
  );
} 

onClick我想重新渲染我的useIsUserSubscribed钩子以便我的按钮文本切换之后。这能做到吗?

3个回答

你不能为此目的在你的钩子中使用 useEffect 试试这个:

钩 :

function useIsUserSubscribed() {
  const currentUserId = useSelector((state) => state.auth.user?.id);


  const checkUser = useCallback(async (publisherId, setUserIsSubscribed) => {
    if (!currentUserId || !publisherId) return;
      try {
        const res = await checkSubscription(publisherId);
        setUserIsSubscribed(true);
      } catch (err) {
        setUserIsSubscribed(false);
      }
    
  }, [currentUserId]);

  return {checkUser};
}

export default useIsUserSubscribed;

零件 :

const SubscribeUnsubscribeBtn = ({profilePageUserId}) => {
    const [userIsSubscribed,setUserIsSubscribed]=useState(false);
    const { checkUser } = useIsUserSubscribed();

     useEffect(()=>{
        checkUser(profilePageUserId,setUserIsSubscribed)
     },[checkUser,profilePageUserId]);
  
  const onClick = async () => {
    if (userIsSubscribed) {
       // this is an API Call to the backend
      await removeSubscription(profilePageUserId);

    } else {
      // this is an API Call to the backend
      await addSubscription(profilePageUserId);
    }
    // HOW CAN I RERENDER THE HOOK HERE!!!!?
    checkUser(profilePageUserId,setUserIsSubscribed)
  }

  return (
    <button type="button" className="sub-edit-unsub-btn bsc-button" onClick={onClick}>
          {userIsSubscribed ? 'Subscribed' : 'Unsubscribed'}
    </button>
  );
} 

您还可以在钩子中添加一些加载状态并返回它们,以便您可以检查进程是否已经完成

添加对 useIsUserSubscribed 的 useEffect 的依赖。

钩 :

function useIsUserSubscribed(publisherId) {
    const [userIsSubscribed, setUserIsSubscribed] = useState(null);
    const currentUserId = useSelector((state) => state.auth.user?.id);
    // add refresh dependece
    const refresh = useSelector((state) => state.auth.refresh);

    useEffect(() => {
        ...
    }, [publisherId, currentUserId, refresh]);
    ...
}

零件 :

const onClick = async () => {
    ...
    // HOW CAN I RERENDER THE HOOK HERE!!!!?
    // when click, you can dispatch a refresh flag.
    dispatch(refreshSubState([]))
}

暴露 forceUpdate 方法。

钩 :

function useIsUserSubscribed(publisherId) {
    const [update, setUpdate] = useState({});
    const forceUpdate = () => {
        setUpdate({});
    }  

    return {userIsSubscribed, forceUpdate};
}

零件 :

const {userIsSubscribed, forceUpdate} = useIsUserSubscribed(profilePageUserId);

const onClick = async () => {
    ...
    forceUpdate();
}

这是用户@bitspook 的另一个解决方案

SubscribeUnsubscribeBtn依赖于useIsUserSubscribed,但useIsUserSubscribed不依赖于SubscribeUnsubscribeBtn. 相反,useIsUserSubscribed是保持本地状态。您在这里有几个选择:

  1. 移动有关用户是否订阅或未订阅的状态,因为您正在使用 Redux,也许在 Redux 中。
  2. 沟通,useIsUserSubscribed你需要改变它的内部状态。

为了 1)

  const [userIsSubscribed, setUserIsSubscribed] = useState(null);

将此状态移动到 Redux 存储并与 useSelector 一起使用。

对于2),从钩子返回值和回调的数组,而不仅仅是值。它将允许您从组件通信回钩子。

useIsUserSubscribed,

  return [userIsSubscribed, setUserIsSubscribed];

然后在 中onClick,您可以调用setUserIsSubscribed(false),更改钩子的内部状态,并重新呈现您的组件。