mirror of
https://github.com/logto-io/logto.git
synced 2025-03-31 22:51:25 -05:00
refactor(phrases): extract ui phrases (#1247)
* refactor(phrases): extract ui-phrases extract ui phrases * fix(phrase-ui): cr update cr update
This commit is contained in:
parent
86aec6cdf3
commit
2a34684587
49 changed files with 323 additions and 193 deletions
11
packages/phrases-ui/README.md
Normal file
11
packages/phrases-ui/README.md
Normal file
|
@ -0,0 +1,11 @@
|
|||
# `@logto/phrases-ui`
|
||||
|
||||
> TODO: description
|
||||
|
||||
## Usage
|
||||
|
||||
```
|
||||
const uiPhrases = require('@logto/ui-phrases');
|
||||
|
||||
// TODO: DEMONSTRATE API
|
||||
```
|
45
packages/phrases-ui/package.json
Normal file
45
packages/phrases-ui/package.json
Normal file
|
@ -0,0 +1,45 @@
|
|||
{
|
||||
"name": "@logto/phrases-ui",
|
||||
"version": "0.1.0",
|
||||
"description": "i18n phrases for main-flow",
|
||||
"author": "Silverhand Inc. <contact@silverhand.io>",
|
||||
"homepage": "https://github.com/logto-io/logto#readme",
|
||||
"license": "MPL-2.0",
|
||||
"main": "lib/index.js",
|
||||
"private": true,
|
||||
"directories": {
|
||||
"lib": "lib"
|
||||
},
|
||||
"files": [
|
||||
"lib"
|
||||
],
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git+https://github.com/logto-io/logto.git"
|
||||
},
|
||||
"scripts": {
|
||||
"precommit": "lint-staged",
|
||||
"build": "rm -rf lib/ && tsc",
|
||||
"lint": "eslint --ext .ts src",
|
||||
"lint:report": "pnpm lint --format json --output-file report.json",
|
||||
"prepack": "pnpm build"
|
||||
},
|
||||
"bugs": {
|
||||
"url": "https://github.com/logto-io/logto/issues"
|
||||
},
|
||||
"dependencies": {
|
||||
"@silverhand/essentials": "^1.1.4"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@silverhand/eslint-config": "^0.14.0",
|
||||
"@silverhand/ts-config": "^0.14.0",
|
||||
"eslint": "^8.10.0",
|
||||
"lint-staged": "^13.0.0",
|
||||
"prettier": "^2.3.2",
|
||||
"typescript": "^4.6.2"
|
||||
},
|
||||
"eslintConfig": {
|
||||
"extends": "@silverhand"
|
||||
},
|
||||
"prettier": "@silverhand/eslint-config/.prettierrc"
|
||||
}
|
16
packages/phrases-ui/src/index.ts
Normal file
16
packages/phrases-ui/src/index.ts
Normal file
|
@ -0,0 +1,16 @@
|
|||
import { NormalizeKeyPaths } from '@silverhand/essentials';
|
||||
|
||||
import en from './locales/en';
|
||||
import zhCN from './locales/zh-cn';
|
||||
import { Resource, Language } from './types';
|
||||
|
||||
export { Language } from './types';
|
||||
|
||||
export type I18nKey = NormalizeKeyPaths<typeof en.translation>;
|
||||
|
||||
const resource: Resource = {
|
||||
[Language.English]: en,
|
||||
[Language.Chinese]: zhCN,
|
||||
};
|
||||
|
||||
export default resource;
|
79
packages/phrases-ui/src/locales/en.ts
Normal file
79
packages/phrases-ui/src/locales/en.ts
Normal file
|
@ -0,0 +1,79 @@
|
|||
const translation = {
|
||||
input: {
|
||||
username: 'Username',
|
||||
password: 'Password',
|
||||
email: 'Email',
|
||||
phone_number: 'Phone number',
|
||||
confirm_password: 'Confirm password',
|
||||
},
|
||||
secondary: {
|
||||
sign_in_with: 'Sign in with {{methods, list(type: disjunction;)}}',
|
||||
social_bind_with:
|
||||
'Already have an account? Sign in to bind {{methods, list(type: disjunction;)}} with your social identity.',
|
||||
},
|
||||
action: {
|
||||
sign_in: 'Sign In',
|
||||
continue: 'Continue',
|
||||
create_account: 'Create Account',
|
||||
create: 'Create',
|
||||
enter_passcode: 'Enter Passcode',
|
||||
confirm: 'Confirm',
|
||||
cancel: 'Cancel',
|
||||
bind: 'Binding with {{address}}',
|
||||
back: 'Go Back',
|
||||
nav_back: 'Back',
|
||||
agree: 'Agree',
|
||||
got_it: 'Got it',
|
||||
sign_in_with: 'Sign in with {{name}}',
|
||||
},
|
||||
description: {
|
||||
email: 'email',
|
||||
phone: 'phone',
|
||||
phone_number: 'phone number',
|
||||
reminder: 'Reminder',
|
||||
not_found: '404 Not Found',
|
||||
agree_with_terms: 'I have read and agree to the ',
|
||||
agree_with_terms_modal: 'Please read the {{terms}} and then agree the box first.',
|
||||
terms_of_use: 'Terms of Use',
|
||||
create_account: 'Create Account',
|
||||
forgot_password: 'Forgot Password?',
|
||||
or: 'or',
|
||||
enter_passcode: 'The passcode has been sent to your {{address}}',
|
||||
passcode_sent: 'The passcode has been resent',
|
||||
resend_after_seconds: 'Resend after {{seconds}} seconds',
|
||||
resend_passcode: 'Resend Passcode',
|
||||
continue_with: 'Continue with',
|
||||
create_account_id_exists:
|
||||
'The account with {{type}} {{value}} already exists, would you like to sign in?',
|
||||
sign_in_id_does_not_exists:
|
||||
'The account with {{type}} {{value}} does not exist, would you like to create a new account?',
|
||||
bind_account_title: 'Binding Logto account',
|
||||
social_create_account: 'No account? You can create a new account and bind.',
|
||||
social_bind_account: 'Already have an account? Sign in to bind it with your social identity.',
|
||||
social_bind_with_existing: 'We find a related account, you can bind it directly.',
|
||||
},
|
||||
error: {
|
||||
username_password_mismatch: 'Username and password do not match.',
|
||||
username_required: 'Username is required.',
|
||||
password_required: 'Password is required.',
|
||||
username_exists: 'Username already exists.',
|
||||
username_should_not_start_with_number: 'Username should not start with a number.',
|
||||
username_valid_charset: 'Username should only contain letters, numbers, or underscore.',
|
||||
invalid_email: 'The email is invalid',
|
||||
invalid_phone: 'The phone number is invalid',
|
||||
password_min_length: 'Password requires a minimum of {{min}} characters.',
|
||||
passwords_do_not_match: 'Passwords do not match.',
|
||||
agree_terms_required: 'You must agree to the Terms of Use before continuing.',
|
||||
invalid_passcode: 'The passcode is invalid.',
|
||||
invalid_connector_auth: 'The authorization is invalid.',
|
||||
invalid_connector_request: 'The connector data is invalid.',
|
||||
unknown: 'Unknown error, please try again later.',
|
||||
invalid_session: 'Session not found. Please go back and sign in again.',
|
||||
},
|
||||
};
|
||||
|
||||
const en = Object.freeze({
|
||||
translation,
|
||||
});
|
||||
|
||||
export default en;
|
79
packages/phrases-ui/src/locales/zh-cn.ts
Normal file
79
packages/phrases-ui/src/locales/zh-cn.ts
Normal file
|
@ -0,0 +1,79 @@
|
|||
import en from './en';
|
||||
|
||||
const translation = {
|
||||
input: {
|
||||
username: '用户名',
|
||||
password: '密码',
|
||||
email: '邮箱',
|
||||
phone_number: '手机号',
|
||||
confirm_password: '确认密码',
|
||||
},
|
||||
secondary: {
|
||||
sign_in_with: '通过 {{methods, list(type: disjunction;), zhOrSpaces}} 登录',
|
||||
social_bind_with:
|
||||
'绑定到已有账户? 使用 {{methods, list(type: disjunction;), zhOrSpaces}} 登录并绑定。',
|
||||
},
|
||||
action: {
|
||||
sign_in: '登录',
|
||||
continue: '继续',
|
||||
create_account: '创建账号',
|
||||
create: '创建',
|
||||
enter_passcode: '输入验证码',
|
||||
cancel: '取消',
|
||||
confirm: '确认',
|
||||
bind: '绑定到 {{address}}',
|
||||
back: '返回',
|
||||
nav_back: '返回',
|
||||
agree: '同意',
|
||||
got_it: '知道了',
|
||||
sign_in_with: '通过 {{name}} 登录',
|
||||
},
|
||||
description: {
|
||||
email: '邮箱',
|
||||
phone: '手机',
|
||||
phone_number: '手机',
|
||||
reminder: '提示',
|
||||
not_found: '404 页面不存在',
|
||||
agree_with_terms: '我已阅读并同意 ',
|
||||
agree_with_terms_modal: 'Please read the {{terms}} and then agree the box first.',
|
||||
terms_of_use: '使用条款',
|
||||
create_account: '创建账号',
|
||||
forgot_password: '忘记密码?',
|
||||
or: '或',
|
||||
enter_passcode: '验证码已经发送至您的{{ address }}',
|
||||
passcode_sent: '验证码已经发送',
|
||||
resend_after_seconds: '在 {{ seconds }} 秒后重发',
|
||||
resend_passcode: '重发验证码',
|
||||
continue_with: '通过以下方式继续',
|
||||
create_account_id_exists: '{{type}}为 {{ value }} 的账号已存在,您要登录吗?',
|
||||
sign_in_id_does_not_exists: '{{type}}为 {{ value }} 的账号不存在,您要创建一个新账号吗?',
|
||||
bind_account_title: '绑定 Logto 账号',
|
||||
social_create_account: 'No account? You can create a new account and bind.',
|
||||
social_bind_account: 'Already have an account? Sign in to bind it with your social identity.',
|
||||
social_bind_with_existing: 'We find a related account, you can bind it directly.',
|
||||
},
|
||||
error: {
|
||||
username_password_mismatch: '用户名和密码不匹配。',
|
||||
username_required: '用户名必填',
|
||||
password_required: '密码必填',
|
||||
username_exists: '用户名已存在。',
|
||||
username_should_not_start_with_number: '用户名不能以数字开头。',
|
||||
username_valid_charset: '用户名只能包含英文字母、数字或下划线。',
|
||||
invalid_email: '无效的邮箱。',
|
||||
invalid_phone: '无效的手机号。',
|
||||
password_min_length: '密码最少需要{{min}}个字符。',
|
||||
passwords_do_not_match: '密码不匹配。',
|
||||
agree_terms_required: '你需要同意使用条款以继续。',
|
||||
invalid_passcode: '无效的验证码。',
|
||||
invalid_connector_auth: '登录失败。',
|
||||
invalid_connector_request: '无效的登录请求。',
|
||||
unknown: '未知错误,请稍后重试。',
|
||||
invalid_session: '未找到有效的会话,请重新登录。',
|
||||
},
|
||||
};
|
||||
|
||||
const zhCN: typeof en = Object.freeze({
|
||||
translation,
|
||||
});
|
||||
|
||||
export default zhCN;
|
11
packages/phrases-ui/src/types.ts
Normal file
11
packages/phrases-ui/src/types.ts
Normal file
|
@ -0,0 +1,11 @@
|
|||
/* Copied from i18next/index.d.ts */
|
||||
export type Resource = Record<Language, ResourceLanguage>;
|
||||
|
||||
export type ResourceLanguage = Record<string, ResourceKey>;
|
||||
|
||||
export type ResourceKey = string | Record<string, any>;
|
||||
|
||||
export enum Language {
|
||||
English = 'en',
|
||||
Chinese = 'zh-CN',
|
||||
}
|
8
packages/phrases-ui/tsconfig.json
Normal file
8
packages/phrases-ui/tsconfig.json
Normal file
|
@ -0,0 +1,8 @@
|
|||
{
|
||||
"extends": "@silverhand/ts-config/tsconfig.base",
|
||||
"compilerOptions": {
|
||||
"outDir": "lib",
|
||||
"declaration": true
|
||||
},
|
||||
"include": ["src"]
|
||||
}
|
|
@ -20,79 +20,6 @@ const translation = {
|
|||
set_up: 'Set up',
|
||||
customize: 'Customize',
|
||||
},
|
||||
main_flow: {
|
||||
input: {
|
||||
username: 'Username',
|
||||
password: 'Password',
|
||||
email: 'Email',
|
||||
phone_number: 'Phone number',
|
||||
confirm_password: 'Confirm password',
|
||||
},
|
||||
secondary: {
|
||||
sign_in_with: 'Sign in with {{methods, list(type: disjunction;)}}',
|
||||
social_bind_with:
|
||||
'Already have an account? Sign in to bind {{methods, list(type: disjunction;)}} with your social identity.',
|
||||
},
|
||||
action: {
|
||||
sign_in: 'Sign In',
|
||||
continue: 'Continue',
|
||||
create_account: 'Create Account',
|
||||
create: 'Create',
|
||||
enter_passcode: 'Enter Passcode',
|
||||
confirm: 'Confirm',
|
||||
cancel: 'Cancel',
|
||||
bind: 'Binding with {{address}}',
|
||||
back: 'Go Back',
|
||||
nav_back: 'Back',
|
||||
agree: 'Agree',
|
||||
got_it: 'Got it',
|
||||
sign_in_with: 'Sign in with {{name}}',
|
||||
},
|
||||
description: {
|
||||
email: 'email',
|
||||
phone: 'phone',
|
||||
phone_number: 'phone number',
|
||||
reminder: 'Reminder',
|
||||
not_found: '404 Not Found',
|
||||
agree_with_terms: 'I have read and agree to the ',
|
||||
agree_with_terms_modal: 'Please read the {{terms}} and then agree the box first.',
|
||||
terms_of_use: 'Terms of Use',
|
||||
create_account: 'Create Account',
|
||||
forgot_password: 'Forgot Password?',
|
||||
or: 'or',
|
||||
enter_passcode: 'The passcode has been sent to your {{address}}',
|
||||
passcode_sent: 'The passcode has been resent',
|
||||
resend_after_seconds: 'Resend after {{seconds}} seconds',
|
||||
resend_passcode: 'Resend Passcode',
|
||||
continue_with: 'Continue with',
|
||||
create_account_id_exists:
|
||||
'The account with {{type}} {{value}} already exists, would you like to sign in?',
|
||||
sign_in_id_does_not_exists:
|
||||
'The account with {{type}} {{value}} does not exist, would you like to create a new account?',
|
||||
bind_account_title: 'Binding Logto account',
|
||||
social_create_account: 'No account? You can create a new account and bind.',
|
||||
social_bind_account: 'Already have an account? Sign in to bind it with your social identity.',
|
||||
social_bind_with_existing: 'We find a related account, you can bind it directly.',
|
||||
},
|
||||
error: {
|
||||
username_password_mismatch: 'Username and password do not match.',
|
||||
username_required: 'Username is required.',
|
||||
password_required: 'Password is required.',
|
||||
username_exists: 'Username already exists.',
|
||||
username_should_not_start_with_number: 'Username should not start with a number.',
|
||||
username_valid_charset: 'Username should only contain letters, numbers, or underscore.',
|
||||
invalid_email: 'The email is invalid',
|
||||
invalid_phone: 'The phone number is invalid',
|
||||
password_min_length: 'Password requires a minimum of {{min}} characters.',
|
||||
passwords_do_not_match: 'Passwords do not match.',
|
||||
agree_terms_required: 'You must agree to the Terms of Use before continuing.',
|
||||
invalid_passcode: 'The passcode is invalid.',
|
||||
invalid_connector_auth: 'The authorization is invalid.',
|
||||
invalid_connector_request: 'The connector data is invalid.',
|
||||
unknown: 'Unknown error, please try again later.',
|
||||
invalid_session: 'Session not found. Please go back and sign in again.',
|
||||
},
|
||||
},
|
||||
admin_console: {
|
||||
title: 'Admin Console',
|
||||
sign_out: 'Sign out',
|
||||
|
|
|
@ -22,77 +22,6 @@ const translation = {
|
|||
set_up: '配置',
|
||||
customize: '自定义',
|
||||
},
|
||||
main_flow: {
|
||||
input: {
|
||||
username: '用户名',
|
||||
password: '密码',
|
||||
email: '邮箱',
|
||||
phone_number: '手机号',
|
||||
confirm_password: '确认密码',
|
||||
},
|
||||
secondary: {
|
||||
sign_in_with: '通过 {{methods, list(type: disjunction;), zhOrSpaces}} 登录',
|
||||
social_bind_with:
|
||||
'绑定到已有账户? 使用 {{methods, list(type: disjunction;), zhOrSpaces}} 登录并绑定。',
|
||||
},
|
||||
action: {
|
||||
sign_in: '登录',
|
||||
continue: '继续',
|
||||
create_account: '创建账号',
|
||||
create: '创建',
|
||||
enter_passcode: '输入验证码',
|
||||
cancel: '取消',
|
||||
confirm: '确认',
|
||||
bind: '绑定到 {{address}}',
|
||||
back: '返回',
|
||||
nav_back: '返回',
|
||||
agree: '同意',
|
||||
got_it: '知道了',
|
||||
sign_in_with: '通过 {{name}} 登录',
|
||||
},
|
||||
description: {
|
||||
email: '邮箱',
|
||||
phone: '手机',
|
||||
phone_number: '手机',
|
||||
reminder: '提示',
|
||||
not_found: '404 页面不存在',
|
||||
agree_with_terms: '我已阅读并同意 ',
|
||||
agree_with_terms_modal: 'Please read the {{terms}} and then agree the box first.',
|
||||
terms_of_use: '使用条款',
|
||||
create_account: '创建账号',
|
||||
forgot_password: '忘记密码?',
|
||||
or: '或',
|
||||
enter_passcode: '验证码已经发送至您的{{ address }}',
|
||||
passcode_sent: '验证码已经发送',
|
||||
resend_after_seconds: '在 {{ seconds }} 秒后重发',
|
||||
resend_passcode: '重发验证码',
|
||||
continue_with: '通过以下方式继续',
|
||||
create_account_id_exists: '{{type}}为 {{ value }} 的账号已存在,您要登录吗?',
|
||||
sign_in_id_does_not_exists: '{{type}}为 {{ value }} 的账号不存在,您要创建一个新账号吗?',
|
||||
bind_account_title: '绑定 Logto 账号',
|
||||
social_create_account: 'No account? You can create a new account and bind.',
|
||||
social_bind_account: 'Already have an account? Sign in to bind it with your social identity.',
|
||||
social_bind_with_existing: 'We find a related account, you can bind it directly.',
|
||||
},
|
||||
error: {
|
||||
username_password_mismatch: '用户名和密码不匹配。',
|
||||
username_required: '用户名必填',
|
||||
password_required: '密码必填',
|
||||
username_exists: '用户名已存在。',
|
||||
username_should_not_start_with_number: '用户名不能以数字开头。',
|
||||
username_valid_charset: '用户名只能包含英文字母、数字或下划线。',
|
||||
invalid_email: '无效的邮箱。',
|
||||
invalid_phone: '无效的手机号。',
|
||||
password_min_length: '密码最少需要{{min}}个字符。',
|
||||
passwords_do_not_match: '密码不匹配。',
|
||||
agree_terms_required: '你需要同意使用条款以继续。',
|
||||
invalid_passcode: '无效的验证码。',
|
||||
invalid_connector_auth: '登录失败。',
|
||||
invalid_connector_request: '无效的登录请求。',
|
||||
unknown: '未知错误,请稍后重试。',
|
||||
invalid_session: '未找到有效的会话,请重新登录。',
|
||||
},
|
||||
},
|
||||
admin_console: {
|
||||
title: '管理面板',
|
||||
sign_out: '登出',
|
||||
|
|
|
@ -47,6 +47,7 @@
|
|||
"@logto/connector-types": "^0.1.0",
|
||||
"@logto/phrases": "^0.1.0",
|
||||
"@logto/shared": "^0.1.0",
|
||||
"@logto/phrases-ui": "^0.1.0",
|
||||
"zod": "^3.14.3"
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import { LogtoErrorCode } from '@logto/phrases';
|
||||
import type { LogtoErrorCode } from '@logto/phrases';
|
||||
|
||||
export type RequestErrorMetadata = Record<string, unknown> & {
|
||||
code: LogtoErrorCode;
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import { Language } from '@logto/phrases';
|
||||
import { Language } from '@logto/phrases-ui';
|
||||
import { hexColorRegEx } from '@logto/shared';
|
||||
import { z } from 'zod';
|
||||
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import { Language } from '@logto/phrases';
|
||||
import { Language } from '@logto/phrases-ui';
|
||||
|
||||
import { CreateSignInExperience, SignInMode } from '../db-entries';
|
||||
import { BrandingStyle, SignInMethodState } from '../foundations';
|
||||
|
|
|
@ -8,7 +8,7 @@
|
|||
"start": "parcel src/index.html",
|
||||
"dev": "cross-env PORT=5001 parcel src/index.html --no-cache --hmr-port 6001",
|
||||
"check": "tsc --noEmit",
|
||||
"build": "pnpm check && rm -rf dist && parcel build src/index.html --no-autoinstall --no-cache",
|
||||
"build": "pnpm check && rm -rf dist && parcel build src/index.html --no-autoinstall --no-cache --detailed-report",
|
||||
"lint": "eslint --ext .ts --ext .tsx src",
|
||||
"lint:report": "pnpm lint --format json --output-file report.json",
|
||||
"stylelint": "stylelint \"src/**/*.scss\"",
|
||||
|
@ -18,6 +18,7 @@
|
|||
"devDependencies": {
|
||||
"@logto/phrases": "^0.1.0",
|
||||
"@logto/schemas": "^0.1.0",
|
||||
"@logto/phrases-ui": "^0.1.0",
|
||||
"@parcel/core": "2.6.2",
|
||||
"@parcel/transformer-sass": "2.6.2",
|
||||
"@parcel/transformer-svg-react": "2.6.2",
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import { Language } from '@logto/phrases';
|
||||
import { Language } from '@logto/phrases-ui';
|
||||
import {
|
||||
BrandingStyle,
|
||||
ConnectorPlatform,
|
||||
|
|
|
@ -20,7 +20,7 @@ const SocialLinkButton = ({ isDisabled, className, connector, onClick }: Props)
|
|||
const {
|
||||
t,
|
||||
i18n: { language },
|
||||
} = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
} = useTranslation();
|
||||
// TODO: LOG-2393 should fix name[locale] syntax error
|
||||
const foundName = Object.entries(name).find(([lang]) => lang === language);
|
||||
const localName = foundName ? foundName[1] : name.en;
|
||||
|
|
|
@ -20,7 +20,7 @@ const AcModal = ({
|
|||
onConfirm,
|
||||
onClose,
|
||||
}: ModalProps) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
|
||||
return (
|
||||
<ReactModal
|
||||
|
|
|
@ -21,7 +21,7 @@ const IframeConfirmModal = ({
|
|||
onConfirm,
|
||||
onClose,
|
||||
}: Props) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const [isLoading, setIsLoading] = useState(true);
|
||||
|
||||
return (
|
||||
|
|
|
@ -18,7 +18,7 @@ const MobileModal = ({
|
|||
onConfirm,
|
||||
onClose,
|
||||
}: ModalProps) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
|
||||
return (
|
||||
<ReactModal
|
||||
|
|
|
@ -5,8 +5,8 @@ export type ModalProps = {
|
|||
className?: string;
|
||||
isOpen?: boolean;
|
||||
children: ReactNode;
|
||||
cancelText?: TFuncKey<'translation', 'main_flow'>;
|
||||
confirmText?: TFuncKey<'translation', 'main_flow'>;
|
||||
cancelText?: TFuncKey;
|
||||
confirmText?: TFuncKey;
|
||||
onConfirm?: () => void;
|
||||
onClose: () => void;
|
||||
};
|
||||
|
|
|
@ -6,11 +6,11 @@ import * as styles from './index.module.scss';
|
|||
|
||||
type Props = {
|
||||
className?: string;
|
||||
label?: TFuncKey<'translation', 'main_flow'>;
|
||||
label?: TFuncKey;
|
||||
};
|
||||
|
||||
const Divider = ({ className, label }: Props) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
|
||||
return (
|
||||
<div className={classNames(styles.divider, className)}>
|
||||
|
|
|
@ -4,7 +4,7 @@ import { TFuncKey, useTranslation } from 'react-i18next';
|
|||
|
||||
import * as styles from './index.module.scss';
|
||||
|
||||
type ErrorCode = TFuncKey<'translation', 'main_flow.error'>;
|
||||
type ErrorCode = TFuncKey<'translation', 'error'>;
|
||||
|
||||
export type ErrorType = ErrorCode | { code: ErrorCode; data?: Record<string, unknown> };
|
||||
|
||||
|
@ -15,7 +15,7 @@ export type Props = {
|
|||
};
|
||||
|
||||
const ErrorMessage = ({ error, className, children }: Props) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow.error' });
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'error' });
|
||||
|
||||
const getMessage = () => {
|
||||
if (!error) {
|
||||
|
|
|
@ -12,7 +12,7 @@ type Props = {
|
|||
|
||||
const NavBar = ({ title }: Props) => {
|
||||
const navigate = useNavigate();
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
|
||||
return (
|
||||
<div className={styles.navBar}>
|
||||
|
|
|
@ -13,7 +13,7 @@ type Props = {
|
|||
};
|
||||
|
||||
const Notification = ({ className, message, onClose }: Props) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
|
||||
return (
|
||||
<div className={classNames(styles.notification, className)}>
|
||||
|
|
|
@ -7,7 +7,7 @@ import TermsOfUse from '.';
|
|||
describe('Terms of Use', () => {
|
||||
const onChange = jest.fn();
|
||||
const contentUrl = 'http://logto.dev/';
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const prefix = t('description.agree_with_terms');
|
||||
|
||||
beforeEach(() => {
|
||||
|
|
|
@ -17,7 +17,7 @@ type Props = {
|
|||
};
|
||||
|
||||
const TermsOfUse = ({ name, className, termsUrl, isChecked, onChange, onTermsClick }: Props) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
|
||||
const prefix = t('description.agree_with_terms');
|
||||
|
||||
|
|
|
@ -8,13 +8,13 @@ import * as styles from './index.module.scss';
|
|||
export type Props = AnchorHTMLAttributes<HTMLAnchorElement> & {
|
||||
className?: string;
|
||||
children?: ReactNode;
|
||||
text?: TFuncKey<'translation', 'main_flow'>;
|
||||
text?: TFuncKey;
|
||||
type?: 'primary' | 'secondary';
|
||||
to?: string;
|
||||
};
|
||||
|
||||
const TextLink = ({ className, children, text, type = 'primary', to, ...rest }: Props) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
|
||||
if (to) {
|
||||
return (
|
||||
|
|
|
@ -36,7 +36,7 @@ const defaultState: FieldState = {
|
|||
};
|
||||
|
||||
const CreateAccount = ({ className, autoFocus }: Props) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const { termsValidation } = useTerms();
|
||||
const {
|
||||
fieldValue,
|
||||
|
|
|
@ -34,7 +34,7 @@ const PasscodeValidation = ({ type, method, className, target }: Props) => {
|
|||
const [code, setCode] = useState<string[]>([]);
|
||||
const [error, setError] = useState<string>();
|
||||
const { setToast } = useContext(PageContext);
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
|
||||
const { seconds, isRunning, restart } = useTimer({
|
||||
autoStart: true,
|
||||
|
|
|
@ -34,7 +34,7 @@ const defaultState: FieldState = { email: '' };
|
|||
const EmailPasswordless = ({ type, autoFocus, className }: Props) => {
|
||||
const { setToast } = useContext(PageContext);
|
||||
const [showPasswordlessConfirmModal, setShowPasswordlessConfirmModal] = useState(false);
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const navigate = useNavigate();
|
||||
const { termsValidation } = useTerms();
|
||||
const { fieldValue, setFieldValue, setFieldErrors, register, validateForm } =
|
||||
|
|
|
@ -18,7 +18,7 @@ type Props = {
|
|||
};
|
||||
|
||||
const PasswordlessConfirmModal = ({ className, isOpen, type, method, value, onClose }: Props) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const sendPasscode = getSendPasscodeApi(type, method);
|
||||
const navigate = useNavigate();
|
||||
const { isMobile } = usePlatform();
|
||||
|
|
|
@ -34,7 +34,7 @@ const defaultState: FieldState = { phone: '' };
|
|||
const PhonePasswordless = ({ type, autoFocus, className }: Props) => {
|
||||
const { setToast } = useContext(PageContext);
|
||||
const [showPasswordlessConfirmModal, setShowPasswordlessConfirmModal] = useState(false);
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const { phoneNumber, setPhoneNumber, isValidPhoneNumber } = usePhoneNumber();
|
||||
const navigate = useNavigate();
|
||||
const { termsValidation } = useTerms();
|
||||
|
|
|
@ -13,11 +13,11 @@ type Props = {
|
|||
signInMethods: LocalSignInMethod[];
|
||||
search?: string;
|
||||
className?: string;
|
||||
template?: TFuncKey<'translation', 'main_flow.secondary'>;
|
||||
template?: TFuncKey<'translation', 'secondary'>;
|
||||
};
|
||||
|
||||
const SignInMethodsKeyMap: {
|
||||
[key in LocalSignInMethod]: TFuncKey<'translation', 'main_flow.input'>;
|
||||
[key in LocalSignInMethod]: TFuncKey<'translation', 'input'>;
|
||||
} = {
|
||||
username: 'username',
|
||||
email: 'email',
|
||||
|
@ -26,7 +26,7 @@ const SignInMethodsKeyMap: {
|
|||
|
||||
const SignInMethodsLink = ({ signInMethods, template, search, className }: Props) => {
|
||||
const navigate = useNavigate();
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
|
||||
const signInMethodsLink = useMemo(
|
||||
() =>
|
||||
|
|
|
@ -16,7 +16,7 @@ type Props = {
|
|||
};
|
||||
|
||||
const SocialCreateAccount = ({ connectorId, className }: Props) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const { relatedUser, localSignInMethods, registerWithSocial, bindSocialRelatedUser } =
|
||||
useBindSocial();
|
||||
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import { Language } from '@logto/phrases';
|
||||
import { Language } from '@logto/phrases-ui';
|
||||
import React, { useState, useCallback } from 'react';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
|
||||
|
|
|
@ -17,7 +17,7 @@ type Props = {
|
|||
};
|
||||
|
||||
const TermsOfUseConfirmModal = ({ isOpen = false, termsUrl, onConfirm, onClose }: Props) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
|
||||
const terms = t('description.terms_of_use');
|
||||
const content = t('description.agree_with_terms_modal', { terms });
|
||||
|
|
|
@ -33,7 +33,7 @@ const defaultState: FieldState = {
|
|||
};
|
||||
|
||||
const UsernameSignin = ({ className, autoFocus }: Props) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const { termsValidation } = useTerms();
|
||||
const {
|
||||
fieldValue,
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import { LogtoErrorCode } from '@logto/phrases';
|
||||
import type { LogtoErrorCode } from '@logto/phrases';
|
||||
import { RequestErrorBody } from '@logto/schemas';
|
||||
import { HTTPError } from 'ky';
|
||||
import { useState, useCallback, useContext, useEffect } from 'react';
|
||||
|
@ -23,7 +23,7 @@ function useApi<Args extends any[], Response>(
|
|||
api: (...args: Args) => Promise<Response>,
|
||||
errorHandlers?: ErrorHandlers
|
||||
): UseApi<Args, Response> {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const [error, setError] = useState<RequestErrorBody>();
|
||||
const [result, setResult] = useState<Response>();
|
||||
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import { Language } from '@logto/phrases';
|
||||
import { Language } from '@logto/phrases-ui';
|
||||
import { AppearanceMode, ConnectorPlatform } from '@logto/schemas';
|
||||
import { conditionalString } from '@silverhand/essentials';
|
||||
import i18next from 'i18next';
|
||||
|
|
|
@ -10,7 +10,7 @@ import { PageContext } from './use-page-context';
|
|||
const useSocialLandingHandler = () => {
|
||||
const [loading, setLoading] = useState(true);
|
||||
const { setToast } = useContext(PageContext);
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const { search } = window.location;
|
||||
|
||||
const socialLandingHandler = useCallback(
|
||||
|
|
|
@ -11,7 +11,7 @@ import { PageContext } from './use-page-context';
|
|||
|
||||
const useSocialSignInListener = () => {
|
||||
const { setToast } = useContext(PageContext);
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const parameters = useParams();
|
||||
const navigate = useNavigate();
|
||||
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import resources from '@logto/phrases';
|
||||
import resources from '@logto/phrases-ui';
|
||||
import { LanguageInfo } from '@logto/schemas';
|
||||
import i18next, { InitOptions } from 'i18next';
|
||||
import LanguageDetector from 'i18next-browser-languagedetector';
|
||||
|
|
2
packages/ui/src/include.d/react-i18next.d.ts
vendored
2
packages/ui/src/include.d/react-i18next.d.ts
vendored
|
@ -2,7 +2,7 @@
|
|||
|
||||
// eslint-disable-next-line import/no-unassigned-import
|
||||
import 'react-i18next';
|
||||
import en from '@logto/phrases/lib/locales/en.js';
|
||||
import en from '@logto/phrases-ui/lib/locales/en.js';
|
||||
|
||||
declare module 'react-i18next' {
|
||||
interface CustomTypeOptions {
|
||||
|
|
|
@ -9,13 +9,13 @@ import NavBar from '@/components/NavBar';
|
|||
import * as styles from './index.module.scss';
|
||||
|
||||
type Props = {
|
||||
title?: TFuncKey<'translation', 'main_flow'>;
|
||||
message?: TFuncKey<'translation', 'main_flow'>;
|
||||
title?: TFuncKey;
|
||||
message?: TFuncKey;
|
||||
rawMessage?: string;
|
||||
};
|
||||
|
||||
const ErrorPage = ({ title = 'description.not_found', message, rawMessage }: Props) => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const navigate = useNavigate();
|
||||
|
||||
const errorMessage = rawMessage || (message && t(message));
|
||||
|
|
|
@ -19,7 +19,7 @@ type Parameters = {
|
|||
type StateType = Nullable<Record<string, string>>;
|
||||
|
||||
const Passcode = () => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const { method, type } = useParams<Parameters>();
|
||||
const state = useLocation().state as StateType;
|
||||
const invalidType = type !== 'sign-in' && type !== 'register';
|
||||
|
|
|
@ -14,7 +14,7 @@ type Parameters = {
|
|||
};
|
||||
|
||||
const Register = () => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const { method = 'username' } = useParams<Parameters>();
|
||||
|
||||
const registerForm = useMemo(() => {
|
||||
|
|
|
@ -14,7 +14,7 @@ type Props = {
|
|||
};
|
||||
|
||||
const SecondarySignIn = () => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const { method = 'username' } = useParams<Props>();
|
||||
|
||||
const signInForm = useMemo(() => {
|
||||
|
|
|
@ -13,7 +13,7 @@ type Parameters = {
|
|||
};
|
||||
|
||||
const SocialRegister = () => {
|
||||
const { t } = useTranslation(undefined, { keyPrefix: 'main_flow' });
|
||||
const { t } = useTranslation();
|
||||
const { connector } = useParams<Parameters>();
|
||||
const { isMobile } = usePlatform();
|
||||
|
||||
|
|
27
pnpm-lock.yaml
generated
27
pnpm-lock.yaml
generated
|
@ -1066,10 +1066,30 @@ importers:
|
|||
prettier: 2.5.1
|
||||
typescript: 4.6.2
|
||||
|
||||
packages/phrases-ui:
|
||||
specifiers:
|
||||
'@silverhand/eslint-config': ^0.14.0
|
||||
'@silverhand/essentials': ^1.1.4
|
||||
'@silverhand/ts-config': ^0.14.0
|
||||
eslint: ^8.10.0
|
||||
lint-staged: ^13.0.0
|
||||
prettier: ^2.3.2
|
||||
typescript: ^4.6.2
|
||||
dependencies:
|
||||
'@silverhand/essentials': 1.1.7
|
||||
devDependencies:
|
||||
'@silverhand/eslint-config': 0.14.0_j52666lpucfyl6yew5a5mxpfce
|
||||
'@silverhand/ts-config': 0.14.0_typescript@4.7.2
|
||||
eslint: 8.10.0
|
||||
lint-staged: 13.0.0
|
||||
prettier: 2.5.1
|
||||
typescript: 4.7.2
|
||||
|
||||
packages/schemas:
|
||||
specifiers:
|
||||
'@logto/connector-types': ^0.1.0
|
||||
'@logto/phrases': ^0.1.0
|
||||
'@logto/phrases-ui': ^0.1.0
|
||||
'@logto/shared': ^0.1.0
|
||||
'@silverhand/eslint-config': ^0.14.0
|
||||
'@silverhand/essentials': ^1.1.6
|
||||
|
@ -1089,6 +1109,7 @@ importers:
|
|||
dependencies:
|
||||
'@logto/connector-types': link:../connector-types
|
||||
'@logto/phrases': link:../phrases
|
||||
'@logto/phrases-ui': link:../phrases-ui
|
||||
'@logto/shared': link:../shared
|
||||
zod: 3.14.3
|
||||
devDependencies:
|
||||
|
@ -1138,6 +1159,7 @@ importers:
|
|||
packages/ui:
|
||||
specifiers:
|
||||
'@logto/phrases': ^0.1.0
|
||||
'@logto/phrases-ui': ^0.1.0
|
||||
'@logto/schemas': ^0.1.0
|
||||
'@parcel/core': 2.6.2
|
||||
'@parcel/transformer-sass': 2.6.2
|
||||
|
@ -1188,6 +1210,7 @@ importers:
|
|||
use-debounced-loader: ^0.1.1
|
||||
devDependencies:
|
||||
'@logto/phrases': link:../phrases
|
||||
'@logto/phrases-ui': link:../phrases-ui
|
||||
'@logto/schemas': link:../schemas
|
||||
'@parcel/core': 2.6.2
|
||||
'@parcel/transformer-sass': 2.6.2_@parcel+core@2.6.2
|
||||
|
@ -10637,10 +10660,10 @@ packages:
|
|||
dev: true
|
||||
|
||||
/lodash.orderby/4.6.0:
|
||||
resolution: {integrity: sha1-5pfwTOXXhSL1TZM4syuBozk+TrM=}
|
||||
resolution: {integrity: sha512-T0rZxKmghOOf5YPnn8EY5iLYeWCpZq8G41FfqoVHH5QDTAFaghJRmAdLiadEDq+ztgM2q5PjA+Z1fOwGrLgmtg==}
|
||||
|
||||
/lodash.pick/4.4.0:
|
||||
resolution: {integrity: sha1-UvBWEP/53tQiYRRB7R/BI6AwAbM=}
|
||||
resolution: {integrity: sha512-hXt6Ul/5yWjfklSGvLQl8vM//l3FtyHZeuelpzK6mm99pNvN9yTDruNZPEJZD1oWrqo+izBmB7oUfWgcCX7s4Q==}
|
||||
|
||||
/lodash.set/4.3.2:
|
||||
resolution: {integrity: sha1-2HV7HagH3eJIFrDWqEvqGnYjCyM=}
|
||||
|
|
Loading…
Add table
Reference in a new issue