mirror of
https://github.com/usememos/memos.git
synced 2025-06-05 22:09:59 +02:00
chore: update about page
This commit is contained in:
@ -1,53 +0,0 @@
|
|||||||
import { Divider, IconButton } from "@mui/joy";
|
|
||||||
import { useGlobalStore } from "@/store/module";
|
|
||||||
import { useTranslate } from "@/utils/i18n";
|
|
||||||
import { generateDialog } from "./Dialog";
|
|
||||||
import Icon from "./Icon";
|
|
||||||
|
|
||||||
type Props = DialogProps;
|
|
||||||
|
|
||||||
const AboutSiteDialog: React.FC<Props> = ({ destroy }: Props) => {
|
|
||||||
const t = useTranslate();
|
|
||||||
const globalStore = useGlobalStore();
|
|
||||||
const profile = globalStore.state.systemStatus.profile;
|
|
||||||
const customizedProfile = globalStore.state.systemStatus.customizedProfile;
|
|
||||||
|
|
||||||
const handleCloseBtnClick = () => {
|
|
||||||
destroy();
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<div className="dialog-header-container">
|
|
||||||
<p className="title-text flex items-center">
|
|
||||||
{t("common.about")} {customizedProfile.name}
|
|
||||||
</p>
|
|
||||||
<IconButton size="sm" onClick={handleCloseBtnClick}>
|
|
||||||
<Icon.X className="opacity-70" />
|
|
||||||
</IconButton>
|
|
||||||
</div>
|
|
||||||
<div className="flex flex-col justify-start items-start max-w-full w-96">
|
|
||||||
<p className="text-xs">{t("about.memos-description")}</p>
|
|
||||||
<p className="text-sm mt-2">{customizedProfile.description || t("about.no-server-description")}</p>
|
|
||||||
<Divider className="!my-3" />
|
|
||||||
<div className="w-full flex flex-row justify-start items-center text-sm italic">
|
|
||||||
{t("about.powered-by")}
|
|
||||||
<a className="shrink-0 flex flex-row justify-start items-center mx-1 hover:underline" href="https://usememos.com" target="_blank">
|
|
||||||
<img className="w-auto h-7" src="https://www.usememos.com/full-logo-landscape.png" alt="" />
|
|
||||||
</a>
|
|
||||||
<span>v{profile.version}</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export default function showAboutSiteDialog(): void {
|
|
||||||
generateDialog(
|
|
||||||
{
|
|
||||||
className: "about-site-dialog",
|
|
||||||
dialogName: "about-site-dialog",
|
|
||||||
},
|
|
||||||
AboutSiteDialog
|
|
||||||
);
|
|
||||||
}
|
|
@ -98,10 +98,16 @@ const Navigation = () => {
|
|||||||
title: t("common.sign-in"),
|
title: t("common.sign-in"),
|
||||||
icon: <Icon.LogIn className="mr-3 w-6 h-auto opacity-70" />,
|
icon: <Icon.LogIn className="mr-3 w-6 h-auto opacity-70" />,
|
||||||
};
|
};
|
||||||
|
const aboutNavLink: NavLinkItem = {
|
||||||
|
id: "header-about",
|
||||||
|
path: "/about",
|
||||||
|
title: t("common.about"),
|
||||||
|
icon: <Icon.Smile className="mr-3 w-6 h-auto opacity-70" />,
|
||||||
|
};
|
||||||
|
|
||||||
const navLinks: NavLinkItem[] = user
|
const navLinks: NavLinkItem[] = user
|
||||||
? [homeNavLink, timelineNavLink, resourcesNavLink, exploreNavLink, profileNavLink, inboxNavLink, archivedNavLink, settingNavLink]
|
? [homeNavLink, timelineNavLink, resourcesNavLink, exploreNavLink, profileNavLink, inboxNavLink, archivedNavLink, settingNavLink]
|
||||||
: [exploreNavLink, signInNavLink];
|
: [exploreNavLink, signInNavLink, aboutNavLink];
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<header className="w-full h-full overflow-auto flex flex-col justify-start items-start py-4 md:pt-6 z-30">
|
<header className="w-full h-full overflow-auto flex flex-col justify-start items-start py-4 md:pt-6 z-30">
|
||||||
|
@ -2,7 +2,6 @@ import * as api from "@/helpers/api";
|
|||||||
import useCurrentUser from "@/hooks/useCurrentUser";
|
import useCurrentUser from "@/hooks/useCurrentUser";
|
||||||
import { useGlobalStore } from "@/store/module";
|
import { useGlobalStore } from "@/store/module";
|
||||||
import { useTranslate } from "@/utils/i18n";
|
import { useTranslate } from "@/utils/i18n";
|
||||||
import showAboutSiteDialog from "./AboutSiteDialog";
|
|
||||||
import Icon from "./Icon";
|
import Icon from "./Icon";
|
||||||
import UserAvatar from "./UserAvatar";
|
import UserAvatar from "./UserAvatar";
|
||||||
import Dropdown from "./kit/Dropdown";
|
import Dropdown from "./kit/Dropdown";
|
||||||
@ -14,10 +13,6 @@ const UserBanner = () => {
|
|||||||
const user = useCurrentUser();
|
const user = useCurrentUser();
|
||||||
const title = user ? user.nickname || user.username : systemStatus.customizedProfile.name || "memos";
|
const title = user ? user.nickname || user.username : systemStatus.customizedProfile.name || "memos";
|
||||||
|
|
||||||
const handleAboutBtnClick = () => {
|
|
||||||
showAboutSiteDialog();
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleSignOutBtnClick = async () => {
|
const handleSignOutBtnClick = async () => {
|
||||||
await api.signout();
|
await api.signout();
|
||||||
window.location.href = "/auth";
|
window.location.href = "/auth";
|
||||||
@ -33,22 +28,17 @@ const UserBanner = () => {
|
|||||||
<span className="text-lg font-medium text-slate-800 dark:text-gray-200 shrink truncate">{title}</span>
|
<span className="text-lg font-medium text-slate-800 dark:text-gray-200 shrink truncate">{title}</span>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
|
disabled={user == undefined}
|
||||||
actionsClassName="min-w-[128px] max-w-full"
|
actionsClassName="min-w-[128px] max-w-full"
|
||||||
positionClassName="top-full mt-2"
|
positionClassName="top-full mt-2"
|
||||||
actions={
|
actions={
|
||||||
<>
|
<>
|
||||||
<button
|
|
||||||
className="w-full px-3 truncate text-left leading-10 cursor-pointer rounded flex flex-row justify-start items-center dark:text-gray-200 hover:bg-gray-100 dark:hover:bg-zinc-800"
|
|
||||||
onClick={handleAboutBtnClick}
|
|
||||||
>
|
|
||||||
<Icon.Info className="w-5 h-auto mr-2 opacity-80" /> {t("common.about")}
|
|
||||||
</button>
|
|
||||||
{user != undefined && (
|
{user != undefined && (
|
||||||
<button
|
<button
|
||||||
className="w-full px-3 truncate text-left leading-10 cursor-pointer rounded flex flex-row justify-start items-center dark:text-gray-200 hover:bg-gray-100 dark:hover:bg-zinc-800"
|
className="w-full px-3 truncate text-left leading-10 cursor-pointer rounded flex flex-row justify-start items-center dark:text-gray-200 hover:bg-gray-100 dark:hover:bg-zinc-800"
|
||||||
onClick={handleSignOutBtnClick}
|
onClick={handleSignOutBtnClick}
|
||||||
>
|
>
|
||||||
<Icon.LogOut className="w-5 h-auto mr-2 opacity-80" /> {t("common.sign-out")}
|
<Icon.LogOut className="w-5 h-auto mr-1 opacity-60" /> {t("common.sign-out")}
|
||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
|
@ -5,13 +5,14 @@ import Icon from "../Icon";
|
|||||||
interface Props {
|
interface Props {
|
||||||
trigger?: ReactNode;
|
trigger?: ReactNode;
|
||||||
actions?: ReactNode;
|
actions?: ReactNode;
|
||||||
|
disabled?: boolean;
|
||||||
className?: string;
|
className?: string;
|
||||||
actionsClassName?: string;
|
actionsClassName?: string;
|
||||||
positionClassName?: string;
|
positionClassName?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
const Dropdown: React.FC<Props> = (props: Props) => {
|
const Dropdown: React.FC<Props> = (props: Props) => {
|
||||||
const { trigger, actions, className, actionsClassName, positionClassName } = props;
|
const { trigger, actions, disabled, className, actionsClassName, positionClassName } = props;
|
||||||
const [dropdownStatus, toggleDropdownStatus] = useToggle(false);
|
const [dropdownStatus, toggleDropdownStatus] = useToggle(false);
|
||||||
const dropdownWrapperRef = useRef<HTMLDivElement>(null);
|
const dropdownWrapperRef = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
@ -29,11 +30,18 @@ const Dropdown: React.FC<Props> = (props: Props) => {
|
|||||||
}
|
}
|
||||||
}, [dropdownStatus]);
|
}, [dropdownStatus]);
|
||||||
|
|
||||||
|
const handleDropdownClick = () => {
|
||||||
|
if (disabled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
toggleDropdownStatus();
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
ref={dropdownWrapperRef}
|
ref={dropdownWrapperRef}
|
||||||
className={`relative flex flex-col justify-start items-start select-none ${className ?? ""}`}
|
className={`relative flex flex-col justify-start items-start select-none ${className ?? ""}`}
|
||||||
onClick={() => toggleDropdownStatus()}
|
onClick={handleDropdownClick}
|
||||||
>
|
>
|
||||||
{trigger ? (
|
{trigger ? (
|
||||||
trigger
|
trigger
|
||||||
|
36
web/src/pages/About.tsx
Normal file
36
web/src/pages/About.tsx
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
import { Link } from "@mui/joy";
|
||||||
|
import Icon from "@/components/Icon";
|
||||||
|
import MobileHeader from "@/components/MobileHeader";
|
||||||
|
|
||||||
|
const About = () => {
|
||||||
|
return (
|
||||||
|
<section className="@container w-full max-w-5xl min-h-full flex flex-col justify-start items-center sm:pt-3 md:pt-6 pb-8">
|
||||||
|
<MobileHeader />
|
||||||
|
<div className="w-full px-4 sm:px-6">
|
||||||
|
<div className="w-full shadow flex flex-col justify-start items-start px-4 py-3 rounded-xl bg-white dark:bg-zinc-800 text-black dark:text-gray-300">
|
||||||
|
<img className="w-auto h-12" src="https://www.usememos.com/full-logo-landscape.png" alt="" />
|
||||||
|
<p className="text-base">A privacy-first, lightweight note-taking service. Easily capture and share your great thoughts.</p>
|
||||||
|
<div className="mt-1 flex flex-row items-center flex-wrap">
|
||||||
|
<Link underline="always" href="https://github.com/usememos/memos">
|
||||||
|
GitHub Repo
|
||||||
|
</Link>
|
||||||
|
<Icon.Dot className="w-4 h-auto opacity-60" />
|
||||||
|
<Link underline="always" href="https://www.usememos.com/">
|
||||||
|
Offical Website
|
||||||
|
</Link>
|
||||||
|
<Icon.Dot className="w-4 h-auto opacity-60" />
|
||||||
|
<Link underline="always" href="https://www.usememos.com/blog">
|
||||||
|
Blogs
|
||||||
|
</Link>
|
||||||
|
<Icon.Dot className="w-4 h-auto opacity-60" />
|
||||||
|
<Link underline="always" href="https://www.usememos.com/blog">
|
||||||
|
Documents
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default About;
|
@ -10,7 +10,6 @@ import UserAvatar from "@/components/UserAvatar";
|
|||||||
import { DEFAULT_MEMO_LIMIT } from "@/helpers/consts";
|
import { DEFAULT_MEMO_LIMIT } from "@/helpers/consts";
|
||||||
import { getTimeStampByDate } from "@/helpers/datetime";
|
import { getTimeStampByDate } from "@/helpers/datetime";
|
||||||
import useLoading from "@/hooks/useLoading";
|
import useLoading from "@/hooks/useLoading";
|
||||||
import { useFilterStore } from "@/store/module";
|
|
||||||
import { useMemoList, useMemoStore, useUserStore } from "@/store/v1";
|
import { useMemoList, useMemoStore, useUserStore } from "@/store/v1";
|
||||||
import { User } from "@/types/proto/api/v2/user_service";
|
import { User } from "@/types/proto/api/v2/user_service";
|
||||||
import { useTranslate } from "@/utils/i18n";
|
import { useTranslate } from "@/utils/i18n";
|
||||||
@ -21,12 +20,10 @@ const UserProfile = () => {
|
|||||||
const userStore = useUserStore();
|
const userStore = useUserStore();
|
||||||
const loadingState = useLoading();
|
const loadingState = useLoading();
|
||||||
const [user, setUser] = useState<User>();
|
const [user, setUser] = useState<User>();
|
||||||
const filterStore = useFilterStore();
|
|
||||||
const memoStore = useMemoStore();
|
const memoStore = useMemoStore();
|
||||||
const memoList = useMemoList();
|
const memoList = useMemoList();
|
||||||
const [isRequesting, setIsRequesting] = useState(true);
|
const [isRequesting, setIsRequesting] = useState(true);
|
||||||
const [isComplete, setIsComplete] = useState(false);
|
const [isComplete, setIsComplete] = useState(false);
|
||||||
const { tag: tagQuery, text: textQuery } = filterStore.state;
|
|
||||||
const sortedMemos = memoList.value
|
const sortedMemos = memoList.value
|
||||||
.sort((a, b) => getTimeStampByDate(b.displayTime) - getTimeStampByDate(a.displayTime))
|
.sort((a, b) => getTimeStampByDate(b.displayTime) - getTimeStampByDate(a.displayTime))
|
||||||
.sort((a, b) => Number(b.pinned) - Number(a.pinned));
|
.sort((a, b) => Number(b.pinned) - Number(a.pinned));
|
||||||
@ -56,7 +53,7 @@ const UserProfile = () => {
|
|||||||
|
|
||||||
memoList.reset();
|
memoList.reset();
|
||||||
fetchMemos();
|
fetchMemos();
|
||||||
}, [user, tagQuery, textQuery]);
|
}, [user]);
|
||||||
|
|
||||||
const fetchMemos = async () => {
|
const fetchMemos = async () => {
|
||||||
if (!user) {
|
if (!user) {
|
||||||
@ -64,16 +61,6 @@ const UserProfile = () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const filters = [`creator == "${user.name}"`, `row_status == "NORMAL"`, `order_by_pinned == true`];
|
const filters = [`creator == "${user.name}"`, `row_status == "NORMAL"`, `order_by_pinned == true`];
|
||||||
const contentSearch: string[] = [];
|
|
||||||
if (tagQuery) {
|
|
||||||
contentSearch.push(`"#${tagQuery}"`);
|
|
||||||
}
|
|
||||||
if (textQuery) {
|
|
||||||
contentSearch.push(`"${textQuery}"`);
|
|
||||||
}
|
|
||||||
if (contentSearch.length > 0) {
|
|
||||||
filters.push(`content_search == [${contentSearch.join(", ")}]`);
|
|
||||||
}
|
|
||||||
setIsRequesting(true);
|
setIsRequesting(true);
|
||||||
const data = await memoStore.fetchMemos({
|
const data = await memoStore.fetchMemos({
|
||||||
filter: filters.join(" && "),
|
filter: filters.join(" && "),
|
||||||
@ -91,18 +78,17 @@ const UserProfile = () => {
|
|||||||
{!loadingState.isLoading &&
|
{!loadingState.isLoading &&
|
||||||
(user ? (
|
(user ? (
|
||||||
<>
|
<>
|
||||||
|
<div className="relative -mt-6 top-8 w-full flex justify-end items-center">
|
||||||
|
<a className="" href={`/u/${user?.id}/rss.xml`} target="_blank" rel="noopener noreferrer">
|
||||||
|
<Button color="neutral" variant="outlined" endDecorator={<Icon.Rss className="w-4 h-auto opacity-60" />}>
|
||||||
|
RSS
|
||||||
|
</Button>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
<div className="w-full flex flex-col justify-start items-center py-8">
|
<div className="w-full flex flex-col justify-start items-center py-8">
|
||||||
<UserAvatar className="!w-20 !h-20 mb-2 drop-shadow" avatarUrl={user?.avatarUrl} />
|
<UserAvatar className="!w-20 !h-20 mb-2 drop-shadow" avatarUrl={user?.avatarUrl} />
|
||||||
<div className="w-full flex flex-row justify-center items-center">
|
<div className="w-full flex flex-row justify-center items-center">
|
||||||
<p className="text-3xl text-black leading-none opacity-80 dark:text-gray-200">{user?.nickname}</p>
|
<p className="text-3xl text-black leading-none opacity-80 dark:text-gray-200">{user?.nickname}</p>
|
||||||
<a
|
|
||||||
className="ml-1 cursor-pointer text-gray-500"
|
|
||||||
href={`/u/${user?.id}/rss.xml`}
|
|
||||||
target="_blank"
|
|
||||||
rel="noopener noreferrer"
|
|
||||||
>
|
|
||||||
<Icon.Rss className="w-5 h-auto opacity-60 mt-0.5" />
|
|
||||||
</a>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{sortedMemos.map((memo) => (
|
{sortedMemos.map((memo) => (
|
||||||
|
@ -18,6 +18,7 @@ const Timeline = lazy(() => import("@/pages/Timeline"));
|
|||||||
const Resources = lazy(() => import("@/pages/Resources"));
|
const Resources = lazy(() => import("@/pages/Resources"));
|
||||||
const Inboxes = lazy(() => import("@/pages/Inboxes"));
|
const Inboxes = lazy(() => import("@/pages/Inboxes"));
|
||||||
const Setting = lazy(() => import("@/pages/Setting"));
|
const Setting = lazy(() => import("@/pages/Setting"));
|
||||||
|
const About = lazy(() => import("@/pages/About"));
|
||||||
const NotFound = lazy(() => import("@/pages/NotFound"));
|
const NotFound = lazy(() => import("@/pages/NotFound"));
|
||||||
const PermissionDenied = lazy(() => import("@/pages/PermissionDenied"));
|
const PermissionDenied = lazy(() => import("@/pages/PermissionDenied"));
|
||||||
|
|
||||||
@ -118,6 +119,10 @@ const router = createBrowserRouter([
|
|||||||
path: "u/:username",
|
path: "u/:username",
|
||||||
element: <UserProfile />,
|
element: <UserProfile />,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
path: "about",
|
||||||
|
element: <About />,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: "403",
|
path: "403",
|
||||||
element: <PermissionDenied />,
|
element: <PermissionDenied />,
|
||||||
|
Reference in New Issue
Block a user