| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184 |
- /*
- Copyright (C) 2025 QuantumNous
- This program is free software: you can redistribute it and/or modify
- it under the terms of the GNU Affero General Public License as
- published by the Free Software Foundation, either version 3 of the
- License, or (at your option) any later version.
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU Affero General Public License for more details.
- You should have received a copy of the GNU Affero General Public License
- along with this program. If not, see <https://www.gnu.org/licenses/>.
- For commercial licensing, please contact support@quantumnous.com
- */
- import React from 'react';
- import { Link } from 'react-router-dom';
- import {
- Avatar,
- Button,
- Dropdown,
- Typography,
- } from '@douyinfe/semi-ui';
- import {
- IconChevronDown,
- IconExit,
- IconUserSetting,
- IconCreditCard,
- IconKey,
- } from '@douyinfe/semi-icons';
- import { stringToColor } from '../../../helpers/index.js';
- import SkeletonWrapper from './SkeletonWrapper.js';
- const UserArea = ({
- userState,
- isLoading,
- isMobile,
- isSelfUseMode,
- logout,
- navigate,
- t,
- }) => {
- if (isLoading) {
- return (
- <SkeletonWrapper
- loading={true}
- type="userArea"
- width={50}
- isMobile={isMobile}
- />
- );
- }
- if (userState.user) {
- return (
- <Dropdown
- position="bottomRight"
- render={
- <Dropdown.Menu className="!bg-semi-color-bg-overlay !border-semi-color-border !shadow-lg !rounded-lg dark:!bg-gray-700 dark:!border-gray-600">
- <Dropdown.Item
- onClick={() => {
- navigate('/console/personal');
- }}
- className="!px-3 !py-1.5 !text-sm !text-semi-color-text-0 hover:!bg-semi-color-fill-1 dark:!text-gray-200 dark:hover:!bg-blue-500 dark:hover:!text-white"
- >
- <div className="flex items-center gap-2">
- <IconUserSetting size="small" className="text-gray-500 dark:text-gray-400" />
- <span>{t('个人设置')}</span>
- </div>
- </Dropdown.Item>
- <Dropdown.Item
- onClick={() => {
- navigate('/console/token');
- }}
- className="!px-3 !py-1.5 !text-sm !text-semi-color-text-0 hover:!bg-semi-color-fill-1 dark:!text-gray-200 dark:hover:!bg-blue-500 dark:hover:!text-white"
- >
- <div className="flex items-center gap-2">
- <IconKey size="small" className="text-gray-500 dark:text-gray-400" />
- <span>{t('令牌管理')}</span>
- </div>
- </Dropdown.Item>
- <Dropdown.Item
- onClick={() => {
- navigate('/console/topup');
- }}
- className="!px-3 !py-1.5 !text-sm !text-semi-color-text-0 hover:!bg-semi-color-fill-1 dark:!text-gray-200 dark:hover:!bg-blue-500 dark:hover:!text-white"
- >
- <div className="flex items-center gap-2">
- <IconCreditCard size="small" className="text-gray-500 dark:text-gray-400" />
- <span>{t('钱包管理')}</span>
- </div>
- </Dropdown.Item>
- <Dropdown.Item onClick={logout} className="!px-3 !py-1.5 !text-sm !text-semi-color-text-0 hover:!bg-semi-color-fill-1 dark:!text-gray-200 dark:hover:!bg-red-500 dark:hover:!text-white">
- <div className="flex items-center gap-2">
- <IconExit size="small" className="text-gray-500 dark:text-gray-400" />
- <span>{t('退出')}</span>
- </div>
- </Dropdown.Item>
- </Dropdown.Menu>
- }
- >
- <Button
- theme="borderless"
- type="tertiary"
- className="flex items-center gap-1.5 !p-1 !rounded-full hover:!bg-semi-color-fill-1 dark:hover:!bg-gray-700 !bg-semi-color-fill-0 dark:!bg-semi-color-fill-1 dark:hover:!bg-semi-color-fill-2"
- >
- <Avatar
- size="extra-small"
- color={stringToColor(userState.user.username)}
- className="mr-1"
- >
- {userState.user.username[0].toUpperCase()}
- </Avatar>
- <span className="hidden md:inline">
- <Typography.Text className="!text-xs !font-medium !text-semi-color-text-1 dark:!text-gray-300 mr-1">
- {userState.user.username}
- </Typography.Text>
- </span>
- <IconChevronDown className="text-xs text-semi-color-text-2 dark:text-gray-400" />
- </Button>
- </Dropdown>
- );
- } else {
- const showRegisterButton = !isSelfUseMode;
- const commonSizingAndLayoutClass = "flex items-center justify-center !py-[10px] !px-1.5";
- const loginButtonSpecificStyling = "!bg-semi-color-fill-0 dark:!bg-semi-color-fill-1 hover:!bg-semi-color-fill-1 dark:hover:!bg-gray-700 transition-colors";
- let loginButtonClasses = `${commonSizingAndLayoutClass} ${loginButtonSpecificStyling}`;
- let registerButtonClasses = `${commonSizingAndLayoutClass}`;
- const loginButtonTextSpanClass = "!text-xs !text-semi-color-text-1 dark:!text-gray-300 !p-1.5";
- const registerButtonTextSpanClass = "!text-xs !text-white !p-1.5";
- if (showRegisterButton) {
- if (isMobile) {
- loginButtonClasses += " !rounded-full";
- } else {
- loginButtonClasses += " !rounded-l-full !rounded-r-none";
- }
- registerButtonClasses += " !rounded-r-full !rounded-l-none";
- } else {
- loginButtonClasses += " !rounded-full";
- }
- return (
- <div className="flex items-center">
- <Link to="/login" className="flex">
- <Button
- theme="borderless"
- type="tertiary"
- className={loginButtonClasses}
- >
- <span className={loginButtonTextSpanClass}>
- {t('登录')}
- </span>
- </Button>
- </Link>
- {showRegisterButton && (
- <div className="hidden md:block">
- <Link to="/register" className="flex -ml-px">
- <Button
- theme="solid"
- type="primary"
- className={registerButtonClasses}
- >
- <span className={registerButtonTextSpanClass}>
- {t('注册')}
- </span>
- </Button>
- </Link>
- </div>
- )}
- </div>
- );
- }
- };
- export default UserArea;
|