diff --git a/apps/client/package.json b/apps/client/package.json
index b6310cdd..288181e6 100644
--- a/apps/client/package.json
+++ b/apps/client/package.json
@@ -29,6 +29,8 @@
"date-fns": "^4.1.0",
"emoji-mart": "^5.6.0",
"file-saver": "^2.0.5",
+ "i18next": "^23.14.0",
+ "i18next-http-backend": "^2.6.1",
"jotai": "^2.10.3",
"jotai-optics": "^0.4.0",
"js-cookie": "^3.0.5",
@@ -43,6 +45,7 @@
"react-drawio": "^1.0.1",
"react-error-boundary": "^4.1.2",
"react-helmet-async": "^2.0.5",
+ "react-i18next": "^15.0.1",
"react-router-dom": "^7.0.1",
"socket.io-client": "^4.8.1",
"tippy.js": "^6.3.7",
diff --git a/apps/client/public/locales/en-US/translation.json b/apps/client/public/locales/en-US/translation.json
new file mode 100644
index 00000000..1ff10a40
--- /dev/null
+++ b/apps/client/public/locales/en-US/translation.json
@@ -0,0 +1,335 @@
+{
+ "Account": "Account",
+ "Active": "Active",
+ "Add": "Add ",
+ "Add group members": "Add group members",
+ "Add groups": "Add groups",
+ "Add members": "Add members",
+ "Add to groups": "Add to groups",
+ "Add space members": "Add space members",
+ "Admin": "Admin",
+ "Are you sure you want to delete this group? Members will lose access to resources this group has access to.": "Are you sure you want to delete this group? Members will lose access to resources this group has access to.",
+ "Are you sure you want to delete this page?": "Are you sure you want to delete this page?",
+ "Are you sure you want to remove this user from the group? The user will lose access to resources this group has access to.": "Are you sure you want to remove this user from the group? The user will lose access to resources this group has access to.",
+ "Are you sure you want to remove this user from the space? The user will lose all access to this space.": "Are you sure you want to remove this user from the space? The user will lose all access to this space.",
+ "Are you sure you want to restore this version? Any changes not versioned will be lost.": "Are you sure you want to restore this version? Any changes not versioned will be lost.",
+ "Can become members of groups and spaces in workspace": "Can become members of groups and spaces in workspace",
+ "Can create and edit pages in space.": "Can create and edit pages in space.",
+ "Can edit": "Can edit",
+ "Can manage workspace": "Can manage workspace",
+ "Can manage workspace but cannot delete it": "Can manage workspace but cannot delete it",
+ "Can view": "Can view",
+ "Can view pages in space but not edit.": "Can view pages in space but not edit.",
+ "Cancel": "Cancel",
+ "Change email": "Change email",
+ "Change password": "Change password",
+ "Change photo": "Change photo",
+ "Choose a role": "Choose a role",
+ "Choose your preferred color scheme.": "Choose your preferred color scheme.",
+ "Choose your preferred interface language.": "Choose your preferred interface language.",
+ "Choose your preferred page width.": "Choose your preferred page width.",
+ "Confirm": "Confirm",
+ "Copy link": "Copy link",
+ "Create": "Create",
+ "Create group": "Create group",
+ "Create page": "Create page",
+ "Create space": "Create space",
+ "Create workspace": "Create workspace",
+ "Current password": "Current password",
+ "Dark": "Dark",
+ "Date": "Date",
+ "Delete": "Delete",
+ "Delete group": "Delete group",
+ "Are you sure you want to delete this page? This will delete its children and page history. This action is irreversible.": "Are you sure you want to delete this page? This will delete its children and page history. This action is irreversible.",
+ "Description": "Description",
+ "Details": "Details",
+ "e.g ACME": "e.g ACME",
+ "e.g ACME Inc": "e.g ACME Inc",
+ "e.g Developers": "e.g Developers",
+ "e.g Group for developers": "e.g Group for developers",
+ "e.g product": "e.g product",
+ "e.g Product Team": "e.g Product Team",
+ "e.g Sales": "e.g Sales",
+ "e.g Space for product team": "e.g Space for product team",
+ "e.g Space for sales team to collaborate": "e.g Space for sales team to collaborate",
+ "Edit": "Edit",
+ "Edit group": "Edit group",
+ "Email": "Email",
+ "Enter a strong password": "Enter a strong password",
+ "Enter valid email addresses separated by comma or space max_50": "Enter valid email addresses separated by comma or space [max: 50]",
+ "enter valid emails addresses": "enter valid emails addresses",
+ "Enter your current password": "Enter your current password",
+ "enter your full name": "enter your full name",
+ "Enter your new password": "Enter your new password",
+ "Enter your new preferred email": "Enter your new preferred email",
+ "Enter your password": "Enter your password",
+ "Error fetching page data.": "Error fetching page data.",
+ "Error loading page history.": "Error loading page history.",
+ "Export": "Export",
+ "Failed to create page": "Failed to create page",
+ "Failed to delete page": "Failed to delete page",
+ "Failed to fetch recent pages": "Failed to fetch recent pages",
+ "Failed to import pages": "Failed to import pages",
+ "Failed to load page. An error occurred.": "Failed to load page. An error occurred.",
+ "Failed to update data": "Failed to update data",
+ "Full access": "Full access",
+ "Full page width": "Full page width",
+ "Full width": "Full width",
+ "General": "General",
+ "Group": "Group",
+ "Group description": "Group description",
+ "Group name": "Group name",
+ "Groups": "Groups",
+ "Has full access to space settings and pages.": "Has full access to space settings and pages.",
+ "Home": "Home",
+ "Import pages": "Import pages",
+ "Import pages & space settings": "Import pages & space settings",
+ "Importing pages": "Importing pages",
+ "invalid invitation link": "invalid invitation link",
+ "Invitation signup": "Invitation signup",
+ "Invite by email": "Invite by email",
+ "Invite members": "Invite members",
+ "Invite new members": "Invite new members",
+ "Invited members who are yet to accept their invitation will appear here.": "Invited members who are yet to accept their invitation will appear here.",
+ "Invited members will be granted access to spaces the groups can access": "Invited members will be granted access to spaces the groups can access",
+ "Join the workspace": "Join the workspace",
+ "Language": "Language",
+ "Light": "Light",
+ "Link copied": "Link copied",
+ "Login": "Login",
+ "Logout": "Logout",
+ "Manage Group": "Manage Group",
+ "Manage members": "Manage members",
+ "member": "member",
+ "Member": "Member",
+ "members": "members",
+ "Members": "Members",
+ "My preferences": "My preferences",
+ "My Profile": "My Profile",
+ "My profile": "My profile",
+ "Name": "Name",
+ "New email": "New email",
+ "New page": "New page",
+ "New password": "New password",
+ "No group found": "No group found",
+ "No page history saved yet.": "No page history saved yet.",
+ "No pages yet": "No pages yet",
+ "No results found...": "No results found...",
+ "No user found": "No user found",
+ "Overview": "Overview",
+ "Owner": "Owner",
+ "page": "page",
+ "Page deleted successfully": "Page deleted successfully",
+ "Page history": "Page history",
+ "Page import is in progress. Please do not close this tab.": "Page import is in progress. Please do not close this tab.",
+ "Pages": "Pages",
+ "pages": "pages",
+ "Password": "Password",
+ "Password changed successfully": "Password changed successfully",
+ "Pending": "Pending",
+ "Please confirm your action": "Please confirm your action",
+ "Preferences": "Preferences",
+ "Print PDF": "Print PDF",
+ "Profile": "Profile",
+ "Recently updated": "Recently updated",
+ "Remove": "Remove",
+ "Remove group member": "Remove group member",
+ "Remove space member": "Remove space member",
+ "Restore": "Restore",
+ "Role": "Role",
+ "Save": "Save",
+ "Search": "Search",
+ "Search for groups": "Search for groups",
+ "Search for users": "Search for users",
+ "Search for users and groups": "Search for users and groups",
+ "Search...": "Search...",
+ "Select language": "Select language",
+ "Select role": "Select role",
+ "Select role to assign to all invited members": "Select role to assign to all invited members",
+ "Select theme": "Select theme",
+ "Send invitation": "Send invitation",
+ "Settings": "Settings",
+ "Setup workspace": "Setup workspace",
+ "Sign In": "Sign In",
+ "Sign Up": "Sign Up",
+ "Slug": "Slug",
+ "Space": "Space",
+ "Space description": "Space description",
+ "Space menu": "Space menu",
+ "Space name": "Space name",
+ "Space settings": "Space settings",
+ "Space slug": "Space slug",
+ "Spaces": "Spaces",
+ "Spaces you belong to": "Spaces you belong to",
+ "No space found": "No space found",
+ "Search for spaces": "Search for spaces",
+ "Start typing to search...": "Start typing to search...",
+ "Status": "Status",
+ "Successfully imported": "Successfully imported",
+ "Successfully restored": "Successfully restored",
+ "System settings": "System settings",
+ "Theme": "Theme",
+ "To change your email, you have to enter your password and new email.": "To change your email, you have to enter your password and new email.",
+ "Toggle full page width": "Toggle full page width",
+ "Unable to import pages. Please try again.": "Unable to import pages. Please try again.",
+ "untitled": "untitled",
+ "Untitled": "Untitled",
+ "Updated successfully": "Updated successfully",
+ "User": "User",
+ "Workspace": "Workspace",
+ "Workspace Name": "Workspace Name",
+ "Workspace settings": "Workspace settings",
+ "You can change your password here.": "You can change your password here.",
+ "Your Email": "Your Email",
+ "Your import is complete.": "Your import is complete.",
+ "Your name": "Your name",
+ "Your Name": "Your Name",
+ "Your password": "Your password",
+ "Your password must be a minimum of 8 characters.": "Your password must be a minimum of 8 characters.",
+ "Sidebar toggle": "Sidebar toggle",
+ "Comments": "Comments",
+ "404 page not found": "404 page not found",
+ "Sorry, we can't find the page you are looking for.": "Sorry, we can't find the page you are looking for.",
+ "Take me back to homepage": "Take me back to homepage",
+ "Forgot password": "Forgot password",
+ "Forgot your password?": "Forgot your password?",
+ "A password reset link has been sent to your email. Please check your inbox.": "A password reset link has been sent to your email. Please check your inbox.",
+ "Send reset link": "Send reset link",
+ "Password reset": "Password reset",
+ "Your new password": "Your new password",
+ "Set password": "Set password",
+ "Write a comment": "Write a comment",
+ "Reply...": "Reply...",
+ "Error loading comments.": "Error loading comments.",
+ "No comments yet.": "No comments yet.",
+ "Edit comment": "Edit comment",
+ "Delete comment": "Delete comment",
+ "Are you sure you want to delete this comment?": "Are you sure you want to delete this comment?",
+ "Comment created successfully": "Comment created successfully",
+ "Error creating comment": "Error creating comment",
+ "Comment updated successfully": "Comment updated successfully",
+ "Failed to update comment": "Failed to update comment",
+ "Comment deleted successfully": "Comment deleted successfully",
+ "Failed to delete comment": "Failed to delete comment",
+ "Comment resolved successfully": "Comment resolved successfully",
+ "Failed to resolve comment": "Failed to resolve comment",
+ "Revoke invitation": "Revoke invitation",
+ "Revoke": "Revoke",
+ "Don't": "Don't",
+ "Are you sure you want to revoke this invitation? The user will not be able to join the workspace.": "Are you sure you want to revoke this invitation? The user will not be able to join the workspace.",
+ "Resend invitation": "Resend invitation",
+ "Anyone with this link can join this workspace.": "Anyone with this link can join this workspace.",
+ "Invite link": "Invite link",
+ "Copy": "Copy",
+ "Copied": "Copied",
+ "Select a user": "Select a user",
+ "Select a group": "Select a group",
+ "Export all pages and attachments in this space.": "Export all pages and attachments in this space.",
+ "Delete space": "Delete space",
+ "Delete this space with all its pages and data.": "Delete this space with all its pages and data.",
+ "Format": "Format",
+ "Include subpages": "Include subpages",
+ "Include attachments": "Include attachments",
+ "Select export format": "Select export format",
+ "Export failed:": "Export failed:",
+ "export error": "export error",
+ "Export page": "Export page",
+ "Export space": "Export space",
+ "Export {{type}}": "Export {{type}}",
+ "File exceeds the {{limit}} attachment limit": "File exceeds the {{limit}} attachment limit",
+ "Align left": "Align left",
+ "Align right": "Align right",
+ "Align center": "Align center",
+ "Merge cells": "Merge cells",
+ "Split cell": "Split cell",
+ "Delete column": "Delete column",
+ "Delete row": "Delete row",
+ "Add left column": "Add left column",
+ "Add right column": "Add right column",
+ "Add row above": "Add row above",
+ "Add row below": "Add row below",
+ "Delete table": "Delete table",
+ "Info": "Info",
+ "Success": "Success",
+ "Warning": "Warning",
+ "Danger": "Danger",
+ "Mermaid diagram error:": "Mermaid diagram error:",
+ "Invalid Mermaid diagram": "Invalid Mermaid diagram",
+ "Double-click to edit Draw.io diagram": "Double-click to edit Draw.io diagram",
+ "Exit": "Exit",
+ "Save & Exit": "Save & Exit",
+ "Double-click to edit Excalidraw diagram": "Double-click to edit Excalidraw diagram",
+ "Paste link": "Paste link",
+ "Edit link": "Edit link",
+ "Remove link": "Remove link",
+ "Add link": "Add link",
+ "Please enter a valid url": "Please enter a valid url",
+ "Empty equation": "Empty equation",
+ "Invalid equation": "Invalid equation",
+ "Color": "Color",
+ "Text color": "Text color",
+ "Default": "Default",
+ "Blue": "Blue",
+ "Green": "Green",
+ "Purple": "Purple",
+ "Red": "Red",
+ "Yellow": "Yellow",
+ "Orange": "Orange",
+ "Pink": "Pink",
+ "Gray": "Gray",
+ "Embed link": "Embed link",
+ "Invalid {{provider}} embed link": "",
+ "Embed {{provider}}": "Embed {{provider}}",
+ "Enter {{provider}} link to embed": "Enter {{provider}} link to embed",
+ "Bold": "Bold",
+ "Italic": "Italic",
+ "Underline": "Underline",
+ "Strike": "Strike",
+ "Code": "Code",
+ "Comment": "Comment",
+ "Text": "Text",
+ "Heading 1": "Heading 1",
+ "Heading 2": "Heading 2",
+ "Heading 3": "Heading 3",
+ "To-do List": "To-do List",
+ "Bullet List": "Bullet List",
+ "Numbered List": "Numbered List",
+ "Blockquote": "Blockquote",
+ "Just start typing with plain text.": "Just start typing with plain text.",
+ "Track tasks with a to-do list.": "Track tasks with a to-do list.",
+ "Big section heading.": "Big section heading.",
+ "Medium section heading.": "Medium section heading.",
+ "Small section heading.": "Small section heading.",
+ "Create a simple bullet list.": "Create a simple bullet list.",
+ "Create a list with numbering.": "Create a list with numbering.",
+ "Create block quote.": "Create block quote.",
+ "Insert code snippet.": "Insert code snippet.",
+ "Insert horizontal rule divider": "Insert horizontal rule divider",
+ "Upload any image from your device.": "Upload any image from your device.",
+ "Upload any video from your device.": "Upload any video from your device.",
+ "Upload any file from your device.": "Upload any file from your device.",
+ "Table": "Table",
+ "Insert a table.": "Insert a table.",
+ "Insert collapsible block.": "Insert collapsible block.",
+ "Video": "Video",
+ "Divider": "Divider",
+ "Quote": "Quote",
+ "Image": "Image",
+ "File attachment": "File attachment",
+ "Toggle block": "Toggle block",
+ "Callout": "Callout",
+ "Insert callout notice.": "Insert callout notice.",
+ "Math inline": "Math inline",
+ "Insert inline math equation.": "Insert inline math equation.",
+ "Math block": "Math block",
+ "Insert math equation": "Insert math equation",
+ "Mermaid diagram": "Mermaid diagram",
+ "Insert mermaid diagram": "Insert mermaid diagram",
+ "Insert and design Drawio diagrams": "Insert and design Drawio diagrams",
+ "Insert current date": "Insert current date",
+ "Draw and sketch excalidraw diagrams": "Draw and sketch excalidraw diagrams",
+ "Multiple": "Multiple",
+ "Heading {{level}}": "Heading {{level}}",
+ "Toggle title": "Toggle title",
+ "Write anything. Enter \"/\" for commands": "Write anything. Enter \"/\" for commands"
+}
diff --git a/apps/client/public/locales/zh-CN/translation.json b/apps/client/public/locales/zh-CN/translation.json
new file mode 100644
index 00000000..900ae8b0
--- /dev/null
+++ b/apps/client/public/locales/zh-CN/translation.json
@@ -0,0 +1,187 @@
+{
+ "Account": "账户",
+ "Active": "活跃",
+ "add": "添加",
+ "Add group members": "添加群组成员",
+ "Add groups": "添加群组",
+ "Add members": "添加成员",
+ "Add to groups": "添加到群组",
+ "add space members": "添加空间成员",
+ "Admin": "管理员",
+ "Are you sure you want to delete this group? Members will lose access to resources this group has access to.": "您确定要删除这个群组吗?成员将失去对该群组可访问资源的访问权限。",
+ "Are you sure you want to delete this page?": "您确定要删除这个页面吗?",
+ "Are you sure you want to remove this user from the group? The user will lose access to resources this group has access to.": "您确定要从群组中移除这个用户吗?该用户将失去对该群组可访问资源的访问权限。",
+ "Are you sure you want to remove this user from the space? The user will lose all access to this space.": "您确定要从空间中移除这个用户吗?该用户将失去对这个空间的所有访问权限。",
+ "Are you sure you want to restore this version? Any changes not versioned will be lost.": "您确定要恢复此版本吗?任何未版本化的更改将会丢失。",
+ "Can become members of groups and spaces in workspace": "可以成为工作区中群组和空间的成员",
+ "Can create and edit pages in space": "可以在空间中创建和编辑页面",
+ "Can edit": "可以编辑",
+ "Can manage workspace": "可以管理工作区",
+ "Can manage workspace but cannot delete it": "可以管理工作区但不能删除它",
+ "Can view": "可以查看",
+ "Can view pages in space but not edit": "可以查看空间中的页面但不能编辑",
+ "Cancel": "取消",
+ "Change email": "更改电子邮箱",
+ "Change password": "更改密码",
+ "Change photo": "更改照片",
+ "Choose a role": "选择一个角色",
+ "Choose your preferred color scheme.": "选择您喜欢的配色方案。",
+ "Choose your preferred interface language.": "选择您喜欢的界面语言。",
+ "Choose your preferred page width.": "选择您喜欢的页面宽度。",
+ "Confirm": "确认",
+ "Copy link": "复制链接",
+ "Create": "创建",
+ "Create group": "创建群组",
+ "Create page": "创建页面",
+ "Create space": "创建空间",
+ "Create workspace": "创建工作空间",
+ "Current password": "当前密码",
+ "Dark": "深色",
+ "Date": "日期",
+ "Delete": "删除",
+ "Delete group": "删除群组",
+ "Are you sure you want to delete this page? This will delete its children and page history. This action is irreversible.": "您确定要删除这个页面吗?这将删除其子页面和页面历史记录。此操作不可逆。",
+ "Description": "描述",
+ "Details": "详情",
+ "e.g ACME": "例如:ACME",
+ "e.g ACME Inc": "例如:ACME Inc",
+ "e.g Developers": "例如:开发人员",
+ "e.g Group for developers": "例如:开发人员群组",
+ "e.g product": "例如:product",
+ "e.g Product Team": "例如:产品团队",
+ "e.g Sales": "例如:销售",
+ "e.g Space for product team": "例如:产品团队的空间",
+ "e.g Space for sales team to collaborate": "例如:销售团队协作的空间",
+ "Edit": "编辑",
+ "Edit group": "编辑群组",
+ "Email": "电子邮箱",
+ "Enter a strong password": "输入一个强密码",
+ "Enter valid email addresses separated by comma or space max_50": "输入有效的电子邮箱地址,用逗号或空格分隔 [最多:50个]",
+ "enter valid emails addresses": "输入有效的电子邮箱地址",
+ "Enter your current password": "输入您的当前密码",
+ "enter your full name": "输入您的全名",
+ "Enter your new password": "输入您的新密码",
+ "Enter your new preferred email": "输入您新的首选电子邮箱",
+ "Enter your password": "输入您的密码",
+ "Error fetching page data.": "获取页面数据时出错。",
+ "Error loading page history.": "加载页面历史时出错。",
+ "Export": "导出",
+ "Failed to create page": "创建页面失败",
+ "Failed to delete page": "删除页面失败",
+ "Failed to fetch recent pages": "获取最近页面失败",
+ "Failed to import pages": "导入页面失败",
+ "Failed to load page. An error occurred.": "页面加载失败。发生了一个错误。",
+ "Failed to update data": "数据更新失败",
+ "Full access": "完全访问",
+ "Full page width": "全页宽度",
+ "Full width": "全宽",
+ "General": "常规",
+ "Group": "群组",
+ "Group description": "群组描述",
+ "Group name": "群组名称",
+ "Groups": "群组",
+ "Has full access to space settings and pages": "具有对空间设置和页面的完全访问权限",
+ "Home": "首页",
+ "Import pages": "导入页面",
+ "Import pages & space settings": "导入页面和空间设置",
+ "Importing pages": "正在导入页面",
+ "invalid invitation link": "无效的邀请链接",
+ "Invitation signup": "邀请注册",
+ "Invite by email": "通过电子邮箱邀请",
+ "Invite members": "邀请成员",
+ "Invite new members": "邀请新成员",
+ "Invited members who are yet to accept their invitation will appear here.": "尚未接受邀请的成员将显示在这里。",
+ "Invited members will be granted access to spaces the groups can access": "被邀请的成员将被授予访问群组可以访问的空间的权限",
+ "Join the workspace": "加入工作空间",
+ "Language": "语言",
+ "Light": "浅色",
+ "Link copied": "链接已复制",
+ "Login": "登录",
+ "Logout": "退出登录",
+ "Manage Group": "管理群组",
+ "Manage members": "管理成员",
+ "member": "成员",
+ "Member": "成员",
+ "Members": "成员",
+ "members": "成员",
+ "My preferences": "我的偏好设置",
+ "My Profile": "我的个人资料",
+ "My profile": "我的个人资料",
+ "Name": "名称",
+ "New email": "新电子邮箱",
+ "New page": "新建页面",
+ "New password": "新密码",
+ "No group found": "未找到群组",
+ "No page history saved yet.": "尚未保存页面历史。",
+ "No pages yet": "暂无页面",
+ "No results found...": "未找到结果...",
+ "No user found": "未找到用户",
+ "Overview": "概览",
+ "Owner": "所有者",
+ "page": "个页面",
+ "Page deleted successfully": "页面已成功删除",
+ "Page history": "页面历史",
+ "Page import is in progress. Please do not close this tab.": "页面导入正在进行中。请不要关闭此标签页。",
+ "Pages": "页面",
+ "pages": "个页面",
+ "Password": "密码",
+ "Password changed successfully": "密码更改成功",
+ "Pending": "待定",
+ "Please confirm your action": "请确认您的操作",
+ "Preferences": "偏好设置",
+ "Print PDF": "打印 PDF",
+ "Profile": "个人资料",
+ "Recently updated": "最近更新",
+ "Remove": "移除",
+ "Remove group member": "移除群组成员",
+ "Remove space member": "移除空间成员",
+ "Restore": "恢复",
+ "Role": "角色",
+ "Save": "保存",
+ "Search": "搜索",
+ "Search for groups": "搜索群组",
+ "Search for users": "搜索用户",
+ "Search for users and groups": "搜索用户和群组",
+ "Search...": "搜索...",
+ "Select language": "选择语言",
+ "Select role": "选择角色",
+ "Select role to assign to all invited members": "选择要分配给所有被邀请成员的角色",
+ "Select theme": "选择主题",
+ "Send invitation": "发送邀请",
+ "Settings": "设置",
+ "Setup workspace": "设置工作空间",
+ "Sign In": "登录",
+ "Sign Up": "注册",
+ "Slug": "短链接",
+ "Space": "空间",
+ "Space description": "空间描述",
+ "Space menu": "空间菜单",
+ "Space name": "空间名称",
+ "Space settings": "空间设置",
+ "Space slug": "空间短链接",
+ "Spaces": "空间",
+ "Spaces you belong to": "您所属的空间",
+ "Start typing to search...": "开始输入以搜索...",
+ "Status": "状态",
+ "Successfully imported": "成功导入",
+ "Successfully restored": "恢复成功",
+ "System settings": "系统设置",
+ "Theme": "主题",
+ "To change your email, you have to enter your password and new email.": "要更改您的电子邮箱,您需要输入密码和新的电子邮箱地址。",
+ "Toggle full page width": "切换全页宽度",
+ "Unable to import pages. Please try again.": "无法导入页面。请重试。",
+ "untitled": "无标题",
+ "Untitled": "无标题",
+ "Updated successfully": "更新成功",
+ "User": "用户",
+ "Workspace": "工作区",
+ "Workspace Name": "工作空间名称",
+ "Workspace settings": "工作区设置",
+ "You can change your password here.": "您可以在这里更改密码。",
+ "Your Email": "您的电子邮箱",
+ "Your import is complete.": "导入已完成。",
+ "Your name": "您的姓名",
+ "Your Name": "您的姓名",
+ "Your password": "您的密码",
+ "Your password must be a minimum of 8 characters.": "您的密码必须至少包含8个字符。"
+}
diff --git a/apps/client/src/App.tsx b/apps/client/src/App.tsx
index f01d4490..f0ef858f 100644
--- a/apps/client/src/App.tsx
+++ b/apps/client/src/App.tsx
@@ -26,8 +26,10 @@ import { ErrorBoundary } from "react-error-boundary";
import InviteSignup from "@/pages/auth/invite-signup.tsx";
import ForgotPassword from "@/pages/auth/forgot-password.tsx";
import PasswordReset from "./pages/auth/password-reset";
+import { useTranslation } from "react-i18next";
export default function App() {
+ const { t } = useTranslation();
const [, setSocket] = useAtom(socketAtom);
const authToken = useAtomValue(authTokensAtom);
@@ -78,7 +80,7 @@ export default function App() {
path={"/s/:spaceSlug/p/:pageSlug"}
element={
Failed to load page. An error occurred.>}
+ fallback={<>{t("Failed to load page. An error occurred.")}>}
>
diff --git a/apps/client/src/components/common/export-modal.tsx b/apps/client/src/components/common/export-modal.tsx
index 1891849b..5d15a4ce 100644
--- a/apps/client/src/components/common/export-modal.tsx
+++ b/apps/client/src/components/common/export-modal.tsx
@@ -12,6 +12,7 @@ import { useState } from "react";
import { ExportFormat } from "@/features/page/types/page.types.ts";
import { notifications } from "@mantine/notifications";
import { exportSpace } from "@/features/space/services/space-service";
+import { useTranslation } from "react-i18next";
interface ExportModalProps {
id: string;
@@ -29,6 +30,7 @@ export default function ExportModal({
const [format, setFormat] = useState(ExportFormat.Markdown);
const [includeChildren, setIncludeChildren] = useState(false);
const [includeAttachments, setIncludeAttachments] = useState(true);
+ const { t } = useTranslation();
const handleExport = async () => {
try {
@@ -73,7 +75,7 @@ export default function ExportModal({
- Format
+ {t("Format")}
@@ -84,7 +86,7 @@ export default function ExportModal({
- Include subpages
+ {t("Include subpages")}
@@ -102,7 +104,7 @@ export default function ExportModal({
- Include attachments
+ {t("Include attachments")}
@@ -116,9 +118,9 @@ export default function ExportModal({
-
+
@@ -131,6 +133,8 @@ interface ExportFormatSelection {
onChange: (value: string) => void;
}
function ExportFormatSelection({ format, onChange }: ExportFormatSelection) {
+ const { t } = useTranslation();
+
return (
);
}
diff --git a/apps/client/src/components/common/recent-changes.tsx b/apps/client/src/components/common/recent-changes.tsx
index 0f89c81e..75935fa5 100644
--- a/apps/client/src/components/common/recent-changes.tsx
+++ b/apps/client/src/components/common/recent-changes.tsx
@@ -8,17 +8,19 @@ import {
} from '@mantine/core';
import {Link} from 'react-router-dom';
import PageListSkeleton from '@/components/ui/page-list-skeleton.tsx';
-import {buildPageUrl} from '@/features/page/page.utils.ts';
-import {formattedDate} from '@/lib/time.ts';
-import {useRecentChangesQuery} from '@/features/page/queries/page-query.ts';
-import {IconFileDescription} from '@tabler/icons-react';
-import {getSpaceUrl} from '@/lib/config.ts';
+import { buildPageUrl } from '@/features/page/page.utils.ts';
+import { formattedDate } from '@/lib/time.ts';
+import { useRecentChangesQuery } from '@/features/page/queries/page-query.ts';
+import { IconFileDescription } from '@tabler/icons-react';
+import { getSpaceUrl } from '@/lib/config.ts';
+import { useTranslation } from "react-i18next";
interface Props {
spaceId?: string;
}
export default function RecentChanges({spaceId}: Props) {
+ const { t } = useTranslation();
const {data: pages, isLoading, isError} = useRecentChangesQuery(spaceId);
if (isLoading) {
@@ -26,7 +28,7 @@ export default function RecentChanges({spaceId}: Props) {
}
if (isError) {
- return Failed to fetch recent pages;
+ return {t("Failed to fetch recent pages")};
}
return pages && pages.items.length > 0 ? (
@@ -48,7 +50,7 @@ export default function RecentChanges({spaceId}: Props) {
)}
- {page.title || 'Untitled'}
+ {page.title || t("Untitled")}
@@ -78,7 +80,7 @@ export default function RecentChanges({spaceId}: Props) {
) : (
- No pages yet
+ {t("No pages yet")}
);
}
diff --git a/apps/client/src/components/layouts/global/app-header.tsx b/apps/client/src/components/layouts/global/app-header.tsx
index 1eeb735d..04b2b09d 100644
--- a/apps/client/src/components/layouts/global/app-header.tsx
+++ b/apps/client/src/components/layouts/global/app-header.tsx
@@ -11,10 +11,12 @@ import {
} from "@/components/layouts/global/hooks/atoms/sidebar-atom.ts";
import {useToggleSidebar} from "@/components/layouts/global/hooks/hooks/use-toggle-sidebar.ts";
import SidebarToggle from "@/components/ui/sidebar-toggle-button.tsx";
+import { useTranslation } from "react-i18next";
const links = [{link: APP_ROUTE.HOME, label: "Home"}];
export function AppHeader() {
+ const { t } = useTranslation();
const [mobileOpened] = useAtom(mobileSidebarAtom);
const toggleMobile = useToggleSidebar(mobileSidebarAtom);
@@ -25,7 +27,7 @@ export function AppHeader() {
const items = links.map((link) => (
- {link.label}
+ {t(link.label)}
));
@@ -35,10 +37,10 @@ export function AppHeader() {
{!isHomeRoute && (
<>
-
+
-
+
- {title}
+ {t(title)}
- Workspace
+ {t("Workspace")}}
>
- Workspace settings
+ {t("Workspace settings")}
}
>
- Manage members
+ {t("Manage members")}
- Account
+ {t("Account")}}
>
- My profile
+ {t("My profile")}
}
>
- My preferences
+ {t("My preferences")}
}>
- Logout
+ {t("Logout")}
diff --git a/apps/client/src/components/settings/settings-sidebar.tsx b/apps/client/src/components/settings/settings-sidebar.tsx
index e4ee2799..8ffb7c4a 100644
--- a/apps/client/src/components/settings/settings-sidebar.tsx
+++ b/apps/client/src/components/settings/settings-sidebar.tsx
@@ -11,6 +11,7 @@ import {
} from "@tabler/icons-react";
import { Link, useLocation, useNavigate } from "react-router-dom";
import classes from "./settings.module.css";
+import { useTranslation } from "react-i18next";
interface DataItem {
label: string;
@@ -51,6 +52,7 @@ const groupedData: DataGroup[] = [
];
export default function SettingsSidebar() {
+ const { t } = useTranslation();
const location = useLocation();
const [active, setActive] = useState(location.pathname);
const navigate = useNavigate();
@@ -62,7 +64,7 @@ export default function SettingsSidebar() {
const menuItems = groupedData.map((group) => (