Allow use of useCurrentUser/useCurrentTeam hooks in unauthenticated components

This commit is contained in:
Tom Moor
2023-10-28 11:31:42 -04:00
parent 56f9755cd9
commit 964d2b6bb3
17 changed files with 109 additions and 71 deletions

View File

@@ -2,6 +2,7 @@ import { observer } from "mobx-react";
import * as React from "react"; import * as React from "react";
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
import { Redirect } from "react-router-dom"; import { Redirect } from "react-router-dom";
import useCurrentUser from "~/hooks/useCurrentUser";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
import { changeLanguage } from "~/utils/language"; import { changeLanguage } from "~/utils/language";
import LoadingIndicator from "./LoadingIndicator"; import LoadingIndicator from "./LoadingIndicator";
@@ -13,10 +14,11 @@ type Props = {
const Authenticated = ({ children }: Props) => { const Authenticated = ({ children }: Props) => {
const { auth } = useStores(); const { auth } = useStores();
const { i18n } = useTranslation(); const { i18n } = useTranslation();
const language = auth.user?.language; const user = useCurrentUser({ rejectOnEmpty: false });
const language = user?.language;
// Watching for language changes here as this is the earliest point we have // Watching for language changes here as this is the earliest point we might have the user
// the user available and means we can start loading translations faster // available and means we can start loading translations faster
React.useEffect(() => { React.useEffect(() => {
void changeLanguage(language, i18n); void changeLanguage(language, i18n);
}, [i18n, language]); }, [i18n, language]);

View File

@@ -12,6 +12,7 @@ import Sidebar from "~/components/Sidebar";
import SidebarRight from "~/components/Sidebar/Right"; import SidebarRight from "~/components/Sidebar/Right";
import SettingsSidebar from "~/components/Sidebar/Settings"; import SettingsSidebar from "~/components/Sidebar/Settings";
import type { Editor as TEditor } from "~/editor"; import type { Editor as TEditor } from "~/editor";
import useCurrentTeam from "~/hooks/useCurrentTeam";
import usePolicy from "~/hooks/usePolicy"; import usePolicy from "~/hooks/usePolicy";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
import history from "~/utils/history"; import history from "~/utils/history";
@@ -45,7 +46,7 @@ const AuthenticatedLayout: React.FC = ({ children }: Props) => {
const { ui, auth } = useStores(); const { ui, auth } = useStores();
const location = useLocation(); const location = useLocation();
const can = usePolicy(ui.activeCollectionId); const can = usePolicy(ui.activeCollectionId);
const { user, team } = auth; const team = useCurrentTeam();
const documentContext = useLocalStore<DocumentContextValue>(() => ({ const documentContext = useLocalStore<DocumentContextValue>(() => ({
editor: null, editor: null,
setEditor: (editor: TEditor) => { setEditor: (editor: TEditor) => {
@@ -76,16 +77,14 @@ const AuthenticatedLayout: React.FC = ({ children }: Props) => {
return <ErrorSuspended />; return <ErrorSuspended />;
} }
const showSidebar = auth.authenticated && user && team; const sidebar = (
const sidebar = showSidebar ? (
<Fade> <Fade>
<Switch> <Switch>
<Route path={settingsPath()} component={SettingsSidebar} /> <Route path={settingsPath()} component={SettingsSidebar} />
<Route component={Sidebar} /> <Route component={Sidebar} />
</Switch> </Switch>
</Fade> </Fade>
) : undefined; );
const showHistory = !!matchPath(location.pathname, { const showHistory = !!matchPath(location.pathname, {
path: matchDocumentHistory, path: matchDocumentHistory,
@@ -98,7 +97,7 @@ const AuthenticatedLayout: React.FC = ({ children }: Props) => {
!showHistory && !showHistory &&
ui.activeDocumentId && ui.activeDocumentId &&
ui.commentsExpanded.includes(ui.activeDocumentId) && ui.commentsExpanded.includes(ui.activeDocumentId) &&
team?.getPreference(TeamPreference.Commenting); team.getPreference(TeamPreference.Commenting);
const sidebarRight = ( const sidebarRight = (
<AnimatePresence <AnimatePresence
@@ -121,7 +120,7 @@ const AuthenticatedLayout: React.FC = ({ children }: Props) => {
return ( return (
<DocumentContext.Provider value={documentContext}> <DocumentContext.Provider value={documentContext}>
<Layout title={team?.name} sidebar={sidebar} sidebarRight={sidebarRight}> <Layout title={team.name} sidebar={sidebar} sidebarRight={sidebarRight}>
<RegisterKeyDown trigger="n" handler={goToNewDocument} /> <RegisterKeyDown trigger="n" handler={goToNewDocument} />
<RegisterKeyDown trigger="t" handler={goToSearch} /> <RegisterKeyDown trigger="t" handler={goToSearch} />
<RegisterKeyDown trigger="/" handler={goToSearch} /> <RegisterKeyDown trigger="/" handler={goToSearch} />

View File

@@ -21,6 +21,7 @@ import ClickablePadding from "~/components/ClickablePadding";
import ErrorBoundary from "~/components/ErrorBoundary"; import ErrorBoundary from "~/components/ErrorBoundary";
import HoverPreview from "~/components/HoverPreview"; import HoverPreview from "~/components/HoverPreview";
import type { Props as EditorProps, Editor as SharedEditor } from "~/editor"; import type { Props as EditorProps, Editor as SharedEditor } from "~/editor";
import useCurrentUser from "~/hooks/useCurrentUser";
import useDictionary from "~/hooks/useDictionary"; import useDictionary from "~/hooks/useDictionary";
import useEmbeds from "~/hooks/useEmbeds"; import useEmbeds from "~/hooks/useEmbeds";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
@@ -65,12 +66,12 @@ function Editor(props: Props, ref: React.RefObject<SharedEditor> | null) {
} = props; } = props;
const userLocale = useUserLocale(); const userLocale = useUserLocale();
const locale = dateLocale(userLocale); const locale = dateLocale(userLocale);
const { auth, comments, documents } = useStores(); const { comments, documents } = useStores();
const dictionary = useDictionary(); const dictionary = useDictionary();
const embeds = useEmbeds(!shareId); const embeds = useEmbeds(!shareId);
const history = useHistory(); const history = useHistory();
const localRef = React.useRef<SharedEditor>(); const localRef = React.useRef<SharedEditor>();
const preferences = auth.user?.preferences; const preferences = useCurrentUser({ rejectOnEmpty: false })?.preferences;
const previousHeadings = React.useRef<Heading[] | null>(null); const previousHeadings = React.useRef<Heading[] | null>(null);
const [activeLinkElement, setActiveLink] = const [activeLinkElement, setActiveLink] =
React.useState<HTMLAnchorElement | null>(null); React.useState<HTMLAnchorElement | null>(null);

View File

@@ -5,6 +5,7 @@ import styled from "styled-components";
import { NavigationNode } from "@shared/types"; import { NavigationNode } from "@shared/types";
import Scrollable from "~/components/Scrollable"; import Scrollable from "~/components/Scrollable";
import SearchPopover from "~/components/SearchPopover"; import SearchPopover from "~/components/SearchPopover";
import useCurrentUser from "~/hooks/useCurrentUser";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
import history from "~/utils/history"; import history from "~/utils/history";
import { homePath, sharedDocumentPath } from "~/utils/routeHelpers"; import { homePath, sharedDocumentPath } from "~/utils/routeHelpers";
@@ -22,7 +23,8 @@ type Props = {
function SharedSidebar({ rootNode, shareId }: Props) { function SharedSidebar({ rootNode, shareId }: Props) {
const team = useTeamContext(); const team = useTeamContext();
const { ui, documents, auth } = useStores(); const user = useCurrentUser({ rejectOnEmpty: false });
const { ui, documents } = useStores();
const { t } = useTranslation(); const { t } = useTranslation();
return ( return (
@@ -33,7 +35,7 @@ function SharedSidebar({ rootNode, shareId }: Props) {
image={<TeamLogo model={team} size={32} alt={t("Logo")} />} image={<TeamLogo model={team} size={32} alt={t("Logo")} />}
onClick={() => onClick={() =>
history.push( history.push(
auth.user ? homePath() : sharedDocumentPath(shareId, rootNode.url) user ? homePath() : sharedDocumentPath(shareId, rootNode.url)
) )
} }
/> />

View File

@@ -6,6 +6,7 @@ import styled, { css, useTheme } from "styled-components";
import breakpoint from "styled-components-breakpoint"; import breakpoint from "styled-components-breakpoint";
import { depths, s } from "@shared/styles"; import { depths, s } from "@shared/styles";
import Flex from "~/components/Flex"; import Flex from "~/components/Flex";
import useCurrentUser from "~/hooks/useCurrentUser";
import useMenuContext from "~/hooks/useMenuContext"; import useMenuContext from "~/hooks/useMenuContext";
import usePrevious from "~/hooks/usePrevious"; import usePrevious from "~/hooks/usePrevious";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
@@ -33,11 +34,11 @@ const Sidebar = React.forwardRef<HTMLDivElement, Props>(function _Sidebar(
) { ) {
const [isCollapsing, setCollapsing] = React.useState(false); const [isCollapsing, setCollapsing] = React.useState(false);
const theme = useTheme(); const theme = useTheme();
const { ui, auth } = useStores(); const { ui } = useStores();
const location = useLocation(); const location = useLocation();
const previousLocation = usePrevious(location); const previousLocation = usePrevious(location);
const { isMenuOpen } = useMenuContext(); const { isMenuOpen } = useMenuContext();
const { user } = auth; const user = useCurrentUser({ rejectOnEmpty: false });
const width = ui.sidebarWidth; const width = ui.sidebarWidth;
const collapsed = ui.sidebarIsClosed && !isMenuOpen; const collapsed = ui.sidebarIsClosed && !isMenuOpen;
const maxWidth = theme.sidebarMaxWidth; const maxWidth = theme.sidebarMaxWidth;

View File

@@ -10,6 +10,7 @@ import User from "~/models/User";
import Avatar from "~/components/Avatar"; import Avatar from "~/components/Avatar";
import { AvatarSize } from "~/components/Avatar/Avatar"; import { AvatarSize } from "~/components/Avatar/Avatar";
import Flex from "~/components/Flex"; import Flex from "~/components/Flex";
import useCurrentUser from "~/hooks/useCurrentUser";
import useRequest from "~/hooks/useRequest"; import useRequest from "~/hooks/useRequest";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
import MentionMenuItem from "./MentionMenuItem"; import MentionMenuItem from "./MentionMenuItem";
@@ -39,8 +40,9 @@ function MentionMenu({ search, isActive, ...rest }: Props) {
const [loaded, setLoaded] = React.useState(false); const [loaded, setLoaded] = React.useState(false);
const [items, setItems] = React.useState<MentionItem[]>([]); const [items, setItems] = React.useState<MentionItem[]>([]);
const { t } = useTranslation(); const { t } = useTranslation();
const { users, auth } = useStores(); const { users } = useStores();
const location = useLocation(); const location = useLocation();
const user = useCurrentUser({ rejectOnEmpty: false });
const documentId = parseDocumentSlug(location.pathname); const documentId = parseDocumentSlug(location.pathname);
const { data, loading, request } = useRequest( const { data, loading, request } = useRequest(
React.useCallback( React.useCallback(
@@ -69,7 +71,7 @@ function MentionMenu({ search, isActive, ...rest }: Props) {
id: v4(), id: v4(),
type: MentionType.User, type: MentionType.User,
modelId: user.id, modelId: user.id,
actorId: auth.user?.id, actorId: user?.id,
label: user.name, label: user.name,
}, },
})); }));
@@ -77,7 +79,7 @@ function MentionMenu({ search, isActive, ...rest }: Props) {
setItems(items); setItems(items);
setLoaded(true); setLoaded(true);
} }
}, [auth.user?.id, loading, data]); }, [user?.id, loading, data]);
// Prevent showing the menu until we have data otherwise it will be positioned // Prevent showing the menu until we have data otherwise it will be positioned
// incorrectly due to the height being unknown. // incorrectly due to the height being unknown.

View File

@@ -1,8 +1,23 @@
import invariant from "invariant"; import invariant from "invariant";
import Team from "~/models/Team";
import useStores from "./useStores"; import useStores from "./useStores";
export default function useCurrentTeam() { /**
* Returns the current team, or undefined if there is no current team and `rejectOnEmpty` is set to
* false.
*
* @param options.rejectOnEmpty - If true, throws an error if there is no current team. Defaults to true.
*/
function useCurrentTeam(options: { rejectOnEmpty: false }): Team | undefined;
function useCurrentTeam(options?: { rejectOnEmpty: true }): Team;
function useCurrentTeam({
rejectOnEmpty = true,
}: { rejectOnEmpty?: boolean } = {}) {
const { auth } = useStores(); const { auth } = useStores();
invariant(auth.team, "team required"); if (rejectOnEmpty) {
return auth.team; invariant(auth.team, "team required");
}
return auth.team || undefined;
} }
export default useCurrentTeam;

View File

@@ -1,8 +1,23 @@
import invariant from "invariant"; import invariant from "invariant";
import User from "~/models/User";
import useStores from "./useStores"; import useStores from "./useStores";
export default function useCurrentUser() { /**
* Returns the current user, or undefined if there is no current user and `rejectOnEmpty` is set to
* false.
*
* @param options.rejectOnEmpty - If true, throws an error if there is no current user. Defaults to true.
*/
function useCurrentUser(options: { rejectOnEmpty: false }): User | undefined;
function useCurrentUser(options?: { rejectOnEmpty: true }): User;
function useCurrentUser({
rejectOnEmpty = true,
}: { rejectOnEmpty?: boolean } = {}) {
const { auth } = useStores(); const { auth } = useStores();
invariant(auth.user, "user required"); if (rejectOnEmpty) {
return auth.user; invariant(auth.user, "user required");
}
return auth.user || undefined;
} }
export default useCurrentUser;

View File

@@ -1,4 +1,4 @@
import useStores from "./useStores"; import useCurrentUser from "./useCurrentUser";
/** /**
* Returns the user's locale, or undefined if the user is not logged in. * Returns the user's locale, or undefined if the user is not logged in.
@@ -7,12 +7,12 @@ import useStores from "./useStores";
* @returns The user's locale, or undefined if the user is not logged in * @returns The user's locale, or undefined if the user is not logged in
*/ */
export default function useUserLocale(languageCode?: boolean) { export default function useUserLocale(languageCode?: boolean) {
const { auth } = useStores(); const user = useCurrentUser({ rejectOnEmpty: false });
if (!auth.user?.language) { if (!user?.language) {
return undefined; return undefined;
} }
const { language } = auth.user; const { language } = user;
return languageCode ? language.split("_")[0] : language; return languageCode ? language.split("_")[0] : language;
} }

View File

@@ -17,6 +17,7 @@ import Modal from "~/components/Modal";
import PaginatedList from "~/components/PaginatedList"; import PaginatedList from "~/components/PaginatedList";
import Text from "~/components/Text"; import Text from "~/components/Text";
import useBoolean from "~/hooks/useBoolean"; import useBoolean from "~/hooks/useBoolean";
import useCurrentTeam from "~/hooks/useCurrentTeam";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
type Props = { type Props = {
@@ -29,11 +30,11 @@ function AddGroupsToCollection(props: Props) {
const [newGroupModalOpen, handleNewGroupModalOpen, handleNewGroupModalClose] = const [newGroupModalOpen, handleNewGroupModalOpen, handleNewGroupModalClose] =
useBoolean(false); useBoolean(false);
const [query, setQuery] = React.useState(""); const [query, setQuery] = React.useState("");
const team = useCurrentTeam();
const { auth, collectionGroupMemberships, groups, policies } = useStores(); const { collectionGroupMemberships, groups, policies } = useStores();
const { fetchPage: fetchGroups } = groups;
const { t } = useTranslation(); const { t } = useTranslation();
const { fetchPage: fetchGroups } = groups;
const can = policies.abilities(team.id);
const debouncedFetch = React.useMemo( const debouncedFetch = React.useMemo(
() => debounce((query) => fetchGroups({ query }), 250), () => debounce((query) => fetchGroups({ query }), 250),
@@ -65,13 +66,6 @@ function AddGroupsToCollection(props: Props) {
} }
}; };
const { user, team } = auth;
if (!user || !team) {
return null;
}
const can = policies.abilities(team.id);
return ( return (
<Flex column> <Flex column>
{can.createGroup ? ( {can.createGroup ? (

View File

@@ -18,6 +18,7 @@ import { TeamContext } from "~/components/TeamContext";
import Text from "~/components/Text"; import Text from "~/components/Text";
import env from "~/env"; import env from "~/env";
import useBuildTheme from "~/hooks/useBuildTheme"; import useBuildTheme from "~/hooks/useBuildTheme";
import useCurrentUser from "~/hooks/useCurrentUser";
import usePolicy from "~/hooks/usePolicy"; import usePolicy from "~/hooks/usePolicy";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
import { AuthorizationError, OfflineError } from "~/utils/errors"; import { AuthorizationError, OfflineError } from "~/utils/errors";
@@ -83,8 +84,9 @@ function useDocumentId(documentSlug: string, response?: Response) {
} }
function SharedDocumentScene(props: Props) { function SharedDocumentScene(props: Props) {
const { ui, auth } = useStores(); const { ui } = useStores();
const location = useLocation(); const location = useLocation();
const user = useCurrentUser({ rejectOnEmpty: false });
const searchParams = React.useMemo( const searchParams = React.useMemo(
() => new URLSearchParams(location.search), () => new URLSearchParams(location.search),
[location.search] [location.search]
@@ -104,10 +106,10 @@ function SharedDocumentScene(props: Props) {
const theme = useBuildTheme(response?.team?.customTheme, themeOverride); const theme = useBuildTheme(response?.team?.customTheme, themeOverride);
React.useEffect(() => { React.useEffect(() => {
if (!auth.user) { if (!user) {
void changeLanguage(detectLanguage(), i18n); void changeLanguage(detectLanguage(), i18n);
} }
}, [auth, i18n]); }, [user, i18n]);
// ensure the wider page color always matches the theme // ensure the wider page color always matches the theme
React.useEffect(() => { React.useEffect(() => {

View File

@@ -7,6 +7,8 @@ import Document from "~/models/Document";
import Revision from "~/models/Revision"; import Revision from "~/models/Revision";
import Error404 from "~/scenes/Error404"; import Error404 from "~/scenes/Error404";
import ErrorOffline from "~/scenes/ErrorOffline"; import ErrorOffline from "~/scenes/ErrorOffline";
import useCurrentTeam from "~/hooks/useCurrentTeam";
import useCurrentUser from "~/hooks/useCurrentUser";
import usePolicy from "~/hooks/usePolicy"; import usePolicy from "~/hooks/usePolicy";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
import Logger from "~/utils/Logger"; import Logger from "~/utils/Logger";
@@ -16,12 +18,16 @@ import { matchDocumentEdit, settingsPath } from "~/utils/routeHelpers";
import Loading from "./Loading"; import Loading from "./Loading";
type Params = { type Params = {
/** The document urlId + slugified title */
documentSlug: string; documentSlug: string;
/** A specific revision id to load. */
revisionId?: string; revisionId?: string;
/** The share ID to use to load data. */
shareId?: string; shareId?: string;
}; };
type LocationState = { type LocationState = {
/** The document title, if preloaded */
title?: string; title?: string;
restore?: boolean; restore?: boolean;
revisionId?: string; revisionId?: string;
@@ -41,17 +47,10 @@ type Props = RouteComponentProps<Params, StaticContext, LocationState> & {
}; };
function DataLoader({ match, children }: Props) { function DataLoader({ match, children }: Props) {
const { const { ui, views, shares, comments, documents, revisions, subscriptions } =
ui, useStores();
views, const team = useCurrentTeam();
shares, const user = useCurrentUser();
comments,
documents,
auth,
revisions,
subscriptions,
} = useStores();
const { team } = auth;
const [error, setError] = React.useState<Error | null>(null); const [error, setError] = React.useState<Error | null>(null);
const { revisionId, shareId, documentSlug } = match.params; const { revisionId, shareId, documentSlug } = match.params;
@@ -73,7 +72,7 @@ function DataLoader({ match, children }: Props) {
: undefined; : undefined;
const isEditRoute = const isEditRoute =
match.path === matchDocumentEdit || match.path.startsWith(settingsPath()); match.path === matchDocumentEdit || match.path.startsWith(settingsPath());
const isEditing = isEditRoute || !auth.user?.separateEditMode; const isEditing = isEditRoute || !user?.separateEditMode;
const can = usePolicy(document?.id); const can = usePolicy(document?.id);
const location = useLocation<LocationState>(); const location = useLocation<LocationState>();
@@ -180,7 +179,7 @@ function DataLoader({ match, children }: Props) {
// Prevents unauthorized request to load share information for the document // Prevents unauthorized request to load share information for the document
// when viewing a public share link // when viewing a public share link
if (can.read) { if (can.read) {
if (team?.getPreference(TeamPreference.Commenting)) { if (team.getPreference(TeamPreference.Commenting)) {
void comments.fetchDocumentComments(document.id, { void comments.fetchDocumentComments(document.id, {
limit: 100, limit: 100,
}); });
@@ -199,7 +198,7 @@ function DataLoader({ match, children }: Props) {
return error instanceof OfflineError ? <ErrorOffline /> : <Error404 />; return error instanceof OfflineError ? <ErrorOffline /> : <Error404 />;
} }
if (!document || !team || (revisionId && !revision)) { if (!document || (revisionId && !revision)) {
return ( return (
<> <>
<Loading location={location} /> <Loading location={location} />

View File

@@ -10,6 +10,7 @@ import Document from "~/models/Document";
import Revision from "~/models/Revision"; import Revision from "~/models/Revision";
import DocumentMeta from "~/components/DocumentMeta"; import DocumentMeta from "~/components/DocumentMeta";
import Fade from "~/components/Fade"; import Fade from "~/components/Fade";
import useCurrentTeam from "~/hooks/useCurrentTeam";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
import { documentPath, documentInsightsPath } from "~/utils/routeHelpers"; import { documentPath, documentInsightsPath } from "~/utils/routeHelpers";
@@ -29,10 +30,10 @@ function TitleDocumentMeta({
revision, revision,
...rest ...rest
}: Props) { }: Props) {
const { auth, views, comments, ui } = useStores(); const { views, comments, ui } = useStores();
const { t } = useTranslation(); const { t } = useTranslation();
const { team } = auth;
const match = useRouteMatch(); const match = useRouteMatch();
const team = useCurrentTeam();
const documentViews = useObserver(() => views.inDocument(document.id)); const documentViews = useObserver(() => views.inDocument(document.id));
const totalViewers = documentViews.length; const totalViewers = documentViews.length;
const onlyYou = totalViewers === 1 && documentViews[0].userId; const onlyYou = totalViewers === 1 && documentViews[0].userId;
@@ -45,7 +46,7 @@ function TitleDocumentMeta({
return ( return (
<Meta document={document} revision={revision} to={to} replace {...rest}> <Meta document={document} revision={revision} to={to} replace {...rest}>
{team?.getPreference(TeamPreference.Commenting) && ( {team.getPreference(TeamPreference.Commenting) && (
<> <>
&nbsp;&nbsp; &nbsp;&nbsp;
<CommentLink <CommentLink

View File

@@ -10,6 +10,8 @@ import Document from "~/models/Document";
import { RefHandle } from "~/components/ContentEditable"; import { RefHandle } from "~/components/ContentEditable";
import Editor, { Props as EditorProps } from "~/components/Editor"; import Editor, { Props as EditorProps } from "~/components/Editor";
import Flex from "~/components/Flex"; import Flex from "~/components/Flex";
import useCurrentTeam from "~/hooks/useCurrentTeam";
import useCurrentUser from "~/hooks/useCurrentUser";
import useFocusedComment from "~/hooks/useFocusedComment"; import useFocusedComment from "~/hooks/useFocusedComment";
import usePolicy from "~/hooks/usePolicy"; import usePolicy from "~/hooks/usePolicy";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
@@ -49,8 +51,9 @@ function DocumentEditor(props: Props, ref: React.RefObject<any>) {
const { t } = useTranslation(); const { t } = useTranslation();
const match = useRouteMatch(); const match = useRouteMatch();
const focusedComment = useFocusedComment(); const focusedComment = useFocusedComment();
const { ui, comments, auth } = useStores(); const { ui, comments } = useStores();
const { user, team } = auth; const user = useCurrentUser({ rejectOnEmpty: false });
const team = useCurrentTeam({ rejectOnEmpty: false });
const history = useHistory(); const history = useHistory();
const { const {
document, document,

View File

@@ -29,6 +29,8 @@ import { publishDocument } from "~/actions/definitions/documents";
import { navigateToTemplateSettings } from "~/actions/definitions/navigation"; import { navigateToTemplateSettings } from "~/actions/definitions/navigation";
import { restoreRevision } from "~/actions/definitions/revisions"; import { restoreRevision } from "~/actions/definitions/revisions";
import useActionContext from "~/hooks/useActionContext"; import useActionContext from "~/hooks/useActionContext";
import useCurrentTeam from "~/hooks/useCurrentTeam";
import useCurrentUser from "~/hooks/useCurrentUser";
import useMobile from "~/hooks/useMobile"; import useMobile from "~/hooks/useMobile";
import usePolicy from "~/hooks/usePolicy"; import usePolicy from "~/hooks/usePolicy";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
@@ -84,10 +86,11 @@ function DocumentHeader({
headings, headings,
}: Props) { }: Props) {
const { t } = useTranslation(); const { t } = useTranslation();
const { ui, auth } = useStores(); const { ui } = useStores();
const theme = useTheme(); const theme = useTheme();
const team = useCurrentTeam({ rejectOnEmpty: false });
const user = useCurrentUser({ rejectOnEmpty: false });
const { resolvedTheme } = ui; const { resolvedTheme } = ui;
const { team, user } = auth;
const isMobile = useMobile(); const isMobile = useMobile();
const isRevision = !!revision; const isRevision = !!revision;
const isEditingFocus = useEditingFocus(); const isEditingFocus = useEditingFocus();

View File

@@ -16,6 +16,7 @@ import Modal from "~/components/Modal";
import PaginatedList from "~/components/PaginatedList"; import PaginatedList from "~/components/PaginatedList";
import Text from "~/components/Text"; import Text from "~/components/Text";
import useBoolean from "~/hooks/useBoolean"; import useBoolean from "~/hooks/useBoolean";
import useCurrentTeam from "~/hooks/useCurrentTeam";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
import GroupMemberListItem from "./components/GroupMemberListItem"; import GroupMemberListItem from "./components/GroupMemberListItem";
@@ -27,7 +28,8 @@ type Props = {
function AddPeopleToGroup(props: Props) { function AddPeopleToGroup(props: Props) {
const { group } = props; const { group } = props;
const { users, auth, groupMemberships } = useStores(); const { users, groupMemberships } = useStores();
const team = useCurrentTeam();
const { t } = useTranslation(); const { t } = useTranslation();
const [query, setQuery] = React.useState(""); const [query, setQuery] = React.useState("");
@@ -69,11 +71,6 @@ function AddPeopleToGroup(props: Props) {
} }
}; };
const { user, team } = auth;
if (!user || !team) {
return null;
}
return ( return (
<Flex column> <Flex column>
<Text type="secondary"> <Text type="secondary">

View File

@@ -22,6 +22,7 @@ import PageTitle from "~/components/PageTitle";
import TeamLogo from "~/components/TeamLogo"; import TeamLogo from "~/components/TeamLogo";
import Text from "~/components/Text"; import Text from "~/components/Text";
import env from "~/env"; import env from "~/env";
import useCurrentUser from "~/hooks/useCurrentUser";
import useLastVisitedPath from "~/hooks/useLastVisitedPath"; import useLastVisitedPath from "~/hooks/useLastVisitedPath";
import useQuery from "~/hooks/useQuery"; import useQuery from "~/hooks/useQuery";
import useStores from "~/hooks/useStores"; import useStores from "~/hooks/useStores";
@@ -43,12 +44,13 @@ function Login({ children }: Props) {
const notice = query.get("notice"); const notice = query.get("notice");
const { t } = useTranslation(); const { t } = useTranslation();
const user = useCurrentUser({ rejectOnEmpty: false });
const { auth } = useStores(); const { auth } = useStores();
const { config } = auth; const { config } = auth;
const [error, setError] = React.useState(null); const [error, setError] = React.useState(null);
const [emailLinkSentTo, setEmailLinkSentTo] = React.useState(""); const [emailLinkSentTo, setEmailLinkSentTo] = React.useState("");
const isCreate = location.pathname === "/create"; const isCreate = location.pathname === "/create";
const rememberLastPath = !!auth.user?.getPreference( const rememberLastPath = !!user?.getPreference(
UserPreference.RememberLastPath UserPreference.RememberLastPath
); );
const [lastVisitedPath] = useLastVisitedPath(); const [lastVisitedPath] = useLastVisitedPath();