mirror of
https://github.com/docmost/docmost.git
synced 2026-05-08 07:13:06 +08:00
Compare commits
32 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 3207222f32 | |||
| fb4df18035 | |||
| 306a4fdd04 | |||
| 213c1537f8 | |||
| c37e4a9125 | |||
| 2277f5d129 | |||
| b38658077f | |||
| d774611db0 | |||
| 642024ba9d | |||
| 147d028036 | |||
| 992691e6e0 | |||
| 9aaa6c731c | |||
| fd91b11c6c | |||
| af8b0ddf3a | |||
| 879aa2c3d8 | |||
| c180d0e487 | |||
| a062f7a165 | |||
| cbd0dd4a0b | |||
| 2d6d829581 | |||
| 5cea30cc5c | |||
| bca85a49d6 | |||
| c9cdfa0f17 | |||
| 412962204c | |||
| a42ac3d450 | |||
| 642c92f779 | |||
| ccb35517bb | |||
| cbdb37ed0a | |||
| aa27d57624 | |||
| 3829b6cbef | |||
| 17da762984 | |||
| 859f16740b | |||
| 7981ef462e |
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "client",
|
"name": "client",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "0.70.3",
|
"version": "0.71.1",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
"build": "tsc && vite build",
|
"build": "tsc && vite build",
|
||||||
@@ -25,7 +25,7 @@
|
|||||||
"@tabler/icons-react": "^3.40.0",
|
"@tabler/icons-react": "^3.40.0",
|
||||||
"@tanstack/react-query": "5.90.17",
|
"@tanstack/react-query": "5.90.17",
|
||||||
"alfaaz": "^1.1.0",
|
"alfaaz": "^1.1.0",
|
||||||
"axios": "^1.13.6",
|
"axios": "1.13.6",
|
||||||
"blueimp-load-image": "^5.16.0",
|
"blueimp-load-image": "^5.16.0",
|
||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
"emoji-mart": "^5.6.0",
|
"emoji-mart": "^5.6.0",
|
||||||
|
|||||||
@@ -341,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Horizontale Trennlinie einfügen",
|
"Insert horizontal rule divider": "Horizontale Trennlinie einfügen",
|
||||||
"Upload any image from your device.": "Laden Sie ein beliebiges Bild von Ihrem Gerät hoch.",
|
"Upload any image from your device.": "Laden Sie ein beliebiges Bild von Ihrem Gerät hoch.",
|
||||||
"Upload any video from your device.": "Laden Sie ein beliebiges Video von Ihrem Gerät hoch.",
|
"Upload any video from your device.": "Laden Sie ein beliebiges Video von Ihrem Gerät hoch.",
|
||||||
|
"Upload any audio from your device.": "Laden Sie beliebige Audiodateien von Ihrem Gerät hoch.",
|
||||||
"Upload any file from your device.": "Laden Sie eine beliebige Datei von Ihrem Gerät hoch.",
|
"Upload any file from your device.": "Laden Sie eine beliebige Datei von Ihrem Gerät hoch.",
|
||||||
"Uploading {{name}}": "Lade {{name}} hoch",
|
"Uploading {{name}}": "Lade {{name}} hoch",
|
||||||
"Uploading file": "Datei wird hochgeladen",
|
"Uploading file": "Datei wird hochgeladen",
|
||||||
@@ -351,6 +352,12 @@
|
|||||||
"Divider": "Trennlinie",
|
"Divider": "Trennlinie",
|
||||||
"Quote": "Zitat",
|
"Quote": "Zitat",
|
||||||
"Image": "Bild",
|
"Image": "Bild",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "PDF einbetten",
|
||||||
|
"Upload and embed a PDF file.": "Laden Sie eine PDF-Datei hoch und betten Sie sie ein.",
|
||||||
|
"Embed as PDF": "Als PDF einbetten",
|
||||||
|
"Failed to load PDF": "Fehler beim Laden der PDF",
|
||||||
|
"Convert to attachment": "In Anhang umwandeln",
|
||||||
"File attachment": "Dateianhang",
|
"File attachment": "Dateianhang",
|
||||||
"Toggle block": "Block umschalten",
|
"Toggle block": "Block umschalten",
|
||||||
"Callout": "Hinweisbox",
|
"Callout": "Hinweisbox",
|
||||||
@@ -442,6 +449,9 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Verhindern Sie, dass Mitglieder Seiten öffentlich teilen.",
|
"Prevent members from sharing pages publicly.": "Verhindern Sie, dass Mitglieder Seiten öffentlich teilen.",
|
||||||
"Toggle public sharing": "Öffentliches Teilen umschalten",
|
"Toggle public sharing": "Öffentliches Teilen umschalten",
|
||||||
"Toggle space public sharing": "Öffentliches Teilen im Bereich umschalten",
|
"Toggle space public sharing": "Öffentliches Teilen im Bereich umschalten",
|
||||||
|
"Allow viewers to comment": "Zuschauern erlauben, Kommentare zu hinterlassen",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Erlauben Sie Zuschauern, Kommentare auf Seiten in diesem Bereich hinzuzufügen.",
|
||||||
|
"Toggle viewer comments": "Zuschauerkommentare umschalten",
|
||||||
"Public sharing is disabled at the workspace level": "Öffentliches Teilen ist auf der Arbeitsbereichsebene deaktiviert",
|
"Public sharing is disabled at the workspace level": "Öffentliches Teilen ist auf der Arbeitsbereichsebene deaktiviert",
|
||||||
"Prevent pages in this space from being shared publicly.": "Verhindern Sie, dass Seiten in diesem Bereich öffentlich geteilt werden.",
|
"Prevent pages in this space from being shared publicly.": "Verhindern Sie, dass Seiten in diesem Bereich öffentlich geteilt werden.",
|
||||||
"Page permissions": "Seitenberechtigungen",
|
"Page permissions": "Seitenberechtigungen",
|
||||||
@@ -664,6 +674,24 @@
|
|||||||
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> hat Sie auf einer Seite erwähnt",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> hat Sie auf einer Seite erwähnt",
|
||||||
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> hat Ihnen Bearbeitungszugriff auf eine Seite gegeben",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> hat Ihnen Bearbeitungszugriff auf eine Seite gegeben",
|
||||||
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> hat Ihnen Ansichtsrechte für eine Seite gegeben",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> hat Ihnen Ansichtsrechte für eine Seite gegeben",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> hat eine Seite aktualisiert.",
|
||||||
|
"Watch page": "Seite beobachten",
|
||||||
|
"Stop watching": "Beobachtung beenden",
|
||||||
|
"Email notifications": "E-Mail-Benachrichtigungen",
|
||||||
|
"Page updates": "Seitenaktualisierungen",
|
||||||
|
"Get notified when pages you watch are updated.": "Erhalten Sie eine Benachrichtigung, wenn Seiten, die Sie beobachten, aktualisiert werden.",
|
||||||
|
"Page mentions": "Seiten-Erwähnungen",
|
||||||
|
"Get notified when someone mentions you on a page.": "Erhalten Sie eine Benachrichtigung, wenn Sie jemand auf einer Seite erwähnt.",
|
||||||
|
"Comment mentions": "Kommentar-Erwähnungen",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Erhalten Sie eine Benachrichtigung, wenn Sie jemand in einem Kommentar erwähnt.",
|
||||||
|
"New comments": "Neue Kommentare",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Erhalten Sie eine Benachrichtigung über neue Kommentare in Threads, an denen Sie teilnehmen.",
|
||||||
|
"Resolved comments": "Erledigte Kommentare",
|
||||||
|
"Get notified when your comment is resolved.": "Erhalten Sie eine Benachrichtigung, wenn Ihr Kommentar erledigt wurde.",
|
||||||
|
"You are now watching this page": "Sie beobachten diese Seite jetzt",
|
||||||
|
"You are no longer watching this page": "Sie beobachten diese Seite nicht mehr",
|
||||||
|
"Direct": "Direkt",
|
||||||
|
"Updates": "Aktualisierungen",
|
||||||
"Today": "Heute",
|
"Today": "Heute",
|
||||||
"Yesterday": "Gestern",
|
"Yesterday": "Gestern",
|
||||||
"This week": "Diese Woche",
|
"This week": "Diese Woche",
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"Account": "Account ",
|
"Account": "Account",
|
||||||
"Active": "Active",
|
"Active": "Active",
|
||||||
"Add": "Add.",
|
"Add": "Add",
|
||||||
"Add group members": "Add group members",
|
"Add group members": "Add group members",
|
||||||
"Add groups": "Add groups",
|
"Add groups": "Add groups",
|
||||||
"Add members": "Add members",
|
"Add members": "Add members",
|
||||||
@@ -44,24 +44,24 @@
|
|||||||
"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.",
|
"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",
|
"Description": "Description",
|
||||||
"Details": "Details",
|
"Details": "Details",
|
||||||
"e.g ACME": "e.g. ACME",
|
"e.g ACME": "e.g ACME",
|
||||||
"e.g ACME Inc": "e.g. ACME Inc",
|
"e.g ACME Inc": "e.g ACME Inc",
|
||||||
"e.g Developers": "e.g. Developers",
|
"e.g Developers": "e.g Developers",
|
||||||
"e.g Group for developers": "e.g. Group for developers",
|
"e.g Group for developers": "e.g Group for developers",
|
||||||
"e.g product": "e.g. product",
|
"e.g product": "e.g product",
|
||||||
"e.g Product Team": "e.g. Product Team",
|
"e.g Product Team": "e.g Product Team",
|
||||||
"e.g Sales": "e.g. Sales",
|
"e.g Sales": "e.g Sales",
|
||||||
"e.g Space for product team": "e.g. Space for product team",
|
"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",
|
"e.g Space for sales team to collaborate": "e.g Space for sales team to collaborate",
|
||||||
"Edit": "Edit",
|
"Edit": "Edit",
|
||||||
"Read": "Read.",
|
"Read": "Read",
|
||||||
"Edit group": "Edit group",
|
"Edit group": "Edit group",
|
||||||
"Email": "Email",
|
"Email": "Email",
|
||||||
"Enter a strong password": "Enter a strong password",
|
"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 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 email addresses",
|
"enter valid emails addresses": "enter valid emails addresses",
|
||||||
"Enter your current password": "Enter your current password",
|
"Enter your current password": "Enter your current password",
|
||||||
"enter your full name": "Enter your full name",
|
"enter your full name": "enter your full name",
|
||||||
"Enter your new password": "Enter your new password",
|
"Enter your new password": "Enter your new password",
|
||||||
"Enter your new preferred email": "Enter your new preferred email",
|
"Enter your new preferred email": "Enter your new preferred email",
|
||||||
"Enter your password": "Enter your password",
|
"Enter your password": "Enter your password",
|
||||||
@@ -87,7 +87,7 @@
|
|||||||
"Import pages": "Import pages",
|
"Import pages": "Import pages",
|
||||||
"Import pages & space settings": "Import pages & space settings",
|
"Import pages & space settings": "Import pages & space settings",
|
||||||
"Importing pages": "Importing pages",
|
"Importing pages": "Importing pages",
|
||||||
"invalid invitation link": "Invalid invitation link",
|
"invalid invitation link": "invalid invitation link",
|
||||||
"Invitation signup": "Invitation signup",
|
"Invitation signup": "Invitation signup",
|
||||||
"Invite by email": "Invite by email",
|
"Invite by email": "Invite by email",
|
||||||
"Invite members": "Invite members",
|
"Invite members": "Invite members",
|
||||||
@@ -113,7 +113,7 @@
|
|||||||
"New email": "New email",
|
"New email": "New email",
|
||||||
"New page": "New page",
|
"New page": "New page",
|
||||||
"New password": "New password",
|
"New password": "New password",
|
||||||
"No group found": "No group found.",
|
"No group found": "No group found",
|
||||||
"No page history saved yet.": "No page history saved yet.",
|
"No page history saved yet.": "No page history saved yet.",
|
||||||
"No pages yet": "No pages yet",
|
"No pages yet": "No pages yet",
|
||||||
"No shared pages": "No shared pages",
|
"No shared pages": "No shared pages",
|
||||||
@@ -149,56 +149,56 @@
|
|||||||
"Search for users": "Search for users",
|
"Search for users": "Search for users",
|
||||||
"Search for users and groups": "Search for users and groups",
|
"Search for users and groups": "Search for users and groups",
|
||||||
"Search...": "Search...",
|
"Search...": "Search...",
|
||||||
"Select language": "Select language.",
|
"Select language": "Select language",
|
||||||
"Select role": "Select role.",
|
"Select role": "Select role",
|
||||||
"Select role to assign to all invited members": "Select role to assign to all invited members.",
|
"Select role to assign to all invited members": "Select role to assign to all invited members",
|
||||||
"Select theme": "Select theme.",
|
"Select theme": "Select theme",
|
||||||
"Send invitation": "Send invitation.",
|
"Send invitation": "Send invitation",
|
||||||
"Invitation sent": "Invitation sent.",
|
"Invitation sent": "Invitation sent",
|
||||||
"Settings": "Settings.",
|
"Settings": "Settings",
|
||||||
"Setup workspace": "Setup workspace.",
|
"Setup workspace": "Setup workspace",
|
||||||
"Sign In": "Sign In.",
|
"Sign In": "Sign In",
|
||||||
"Sign Up": "Sign Up.",
|
"Sign Up": "Sign Up",
|
||||||
"Slug": "Slug.",
|
"Slug": "Slug",
|
||||||
"Space": "Space.",
|
"Space": "Space",
|
||||||
"Space description": "Space description.",
|
"Space description": "Space description",
|
||||||
"Space menu": "Space menu.",
|
"Space menu": "Space menu",
|
||||||
"Space name": "Space name.",
|
"Space name": "Space name",
|
||||||
"Space settings": "Space settings.",
|
"Space settings": "Space settings",
|
||||||
"Space slug": "Space slug.",
|
"Space slug": "Space slug",
|
||||||
"Spaces": "Spaces.",
|
"Spaces": "Spaces",
|
||||||
"Spaces you belong to": "Spaces you belong to.",
|
"Spaces you belong to": "Spaces you belong to",
|
||||||
"No space found": "No space found.",
|
"No space found": "No space found",
|
||||||
"Search for spaces": "Search for spaces.",
|
"Search for spaces": "Search for spaces",
|
||||||
"Start typing to search...": "Start typing to search...",
|
"Start typing to search...": "Start typing to search...",
|
||||||
"Status": "Status.",
|
"Status": "Status",
|
||||||
"Successfully imported": "Successfully imported.",
|
"Successfully imported": "Successfully imported",
|
||||||
"Successfully restored": "Successfully restored.",
|
"Successfully restored": "Successfully restored",
|
||||||
"System settings": "System settings.",
|
"System settings": "System settings",
|
||||||
"Theme": "Theme.",
|
"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.",
|
"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.",
|
"Toggle full page width": "Toggle full page width",
|
||||||
"Unable to import pages. Please try again.": "Unable to import pages. Please try again.",
|
"Unable to import pages. Please try again.": "Unable to import pages. Please try again.",
|
||||||
"untitled": "untitled.",
|
"untitled": "untitled",
|
||||||
"Untitled": "Untitled.",
|
"Untitled": "Untitled",
|
||||||
"Updated successfully": "Updated successfully.",
|
"Updated successfully": "Updated successfully",
|
||||||
"User": "User.",
|
"User": "User",
|
||||||
"Workspace": "Workspace.",
|
"Workspace": "Workspace",
|
||||||
"Workspace Name": "Workspace Name.",
|
"Workspace Name": "Workspace Name",
|
||||||
"Workspace settings": "Workspace settings.",
|
"Workspace settings": "Workspace settings",
|
||||||
"You can change your password here.": "You can change your password here.",
|
"You can change your password here.": "You can change your password here.",
|
||||||
"Your Email": "Your Email.",
|
"Your Email": "Your Email",
|
||||||
"Your import is complete.": "Your import is complete.",
|
"Your import is complete.": "Your import is complete.",
|
||||||
"Your name": "Your name.",
|
"Your name": "Your name",
|
||||||
"Your Name": "Your Name.",
|
"Your Name": "Your Name",
|
||||||
"Your password": "Your password.",
|
"Your password": "Your password",
|
||||||
"Your password must be a minimum of 8 characters.": "Your password must be a minimum of 8 characters.",
|
"Your password must be a minimum of 8 characters.": "Your password must be a minimum of 8 characters.",
|
||||||
"Sidebar toggle": "Sidebar toggle.",
|
"Sidebar toggle": "Sidebar toggle",
|
||||||
"Comments": "Comments.",
|
"Comments": "Comments",
|
||||||
"404 page not found": "404 page not found.",
|
"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.",
|
"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.",
|
"Take me back to homepage": "Take me back to homepage",
|
||||||
"Forgot password": "Forgot password.",
|
"Forgot password": "Forgot password",
|
||||||
"Forgot your password?": "Forgot your 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.",
|
"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",
|
"Send reset link": "Send reset link",
|
||||||
@@ -222,16 +222,16 @@
|
|||||||
"Comment deleted successfully": "Comment deleted successfully",
|
"Comment deleted successfully": "Comment deleted successfully",
|
||||||
"Failed to delete comment": "Failed to delete comment",
|
"Failed to delete comment": "Failed to delete comment",
|
||||||
"Comment resolved successfully": "Comment resolved successfully",
|
"Comment resolved successfully": "Comment resolved successfully",
|
||||||
"Comment re-opened successfully": "Comment re-opened successfully.",
|
"Comment re-opened successfully": "Comment re-opened successfully",
|
||||||
"Comment unresolved successfully": "Comment unresolved successfully.",
|
"Comment unresolved successfully": "Comment unresolved successfully",
|
||||||
"Failed to resolve comment": "Failed to resolve comment",
|
"Failed to resolve comment": "Failed to resolve comment",
|
||||||
"Resolve comment": "Resolve comment.",
|
"Resolve comment": "Resolve comment",
|
||||||
"Unresolve comment": "Unresolve comment.",
|
"Unresolve comment": "Unresolve comment",
|
||||||
"Resolve Comment Thread": "Resolve Comment Thread.",
|
"Resolve Comment Thread": "Resolve Comment Thread",
|
||||||
"Unresolve Comment Thread": "Unresolve Comment Thread.",
|
"Unresolve Comment Thread": "Unresolve Comment Thread",
|
||||||
"Are you sure you want to resolve this comment thread? This will mark it as completed.": "Are you sure you want to resolve this comment thread? This will mark it as completed.",
|
"Are you sure you want to resolve this comment thread? This will mark it as completed.": "Are you sure you want to resolve this comment thread? This will mark it as completed.",
|
||||||
"Are you sure you want to unresolve this comment thread?": "Are you sure you want to unresolve this comment thread?",
|
"Are you sure you want to unresolve this comment thread?": "Are you sure you want to unresolve this comment thread?",
|
||||||
"Resolved": "Resolved.",
|
"Resolved": "Resolved",
|
||||||
"No active comments.": "No active comments.",
|
"No active comments.": "No active comments.",
|
||||||
"Revoke invitation": "Revoke invitation",
|
"Revoke invitation": "Revoke invitation",
|
||||||
"Revoke": "Revoke",
|
"Revoke": "Revoke",
|
||||||
@@ -241,9 +241,9 @@
|
|||||||
"Anyone with this link can join this workspace.": "Anyone with this link can join this workspace.",
|
"Anyone with this link can join this workspace.": "Anyone with this link can join this workspace.",
|
||||||
"Invite link": "Invite link",
|
"Invite link": "Invite link",
|
||||||
"Copy": "Copy",
|
"Copy": "Copy",
|
||||||
"Copy to space": "Copy to space.",
|
"Copy to space": "Copy to space",
|
||||||
"Copied": "Copied",
|
"Copied": "Copied",
|
||||||
"Duplicate": "Duplicate.",
|
"Duplicate": "Duplicate",
|
||||||
"Select a user": "Select a user",
|
"Select a user": "Select a user",
|
||||||
"Select a group": "Select a group",
|
"Select a group": "Select a group",
|
||||||
"Export all pages and attachments in this space.": "Export all pages and attachments in this space.",
|
"Export all pages and attachments in this space.": "Export all pages and attachments in this space.",
|
||||||
@@ -251,7 +251,7 @@
|
|||||||
"Are you sure you want to delete this space?": "Are you sure you want to delete this space?",
|
"Are you sure you want to delete this space?": "Are you sure you want to delete this space?",
|
||||||
"Delete this space with all its pages and data.": "Delete this space with all its pages and data.",
|
"Delete this space with all its pages and data.": "Delete this space with all its pages and data.",
|
||||||
"All pages, comments, attachments and permissions in this space will be deleted irreversibly.": "All pages, comments, attachments and permissions in this space will be deleted irreversibly.",
|
"All pages, comments, attachments and permissions in this space will be deleted irreversibly.": "All pages, comments, attachments and permissions in this space will be deleted irreversibly.",
|
||||||
"Confirm space name": "Confirm space name.",
|
"Confirm space name": "Confirm space name",
|
||||||
"Type the space name <b>{{spaceName}}</b> to confirm your action.": "Type the space name <b>{{spaceName}}</b> to confirm your action.",
|
"Type the space name <b>{{spaceName}}</b> to confirm your action.": "Type the space name <b>{{spaceName}}</b> to confirm your action.",
|
||||||
"Format": "Format",
|
"Format": "Format",
|
||||||
"Include subpages": "Include subpages",
|
"Include subpages": "Include subpages",
|
||||||
@@ -267,7 +267,7 @@
|
|||||||
"Align left": "Align left",
|
"Align left": "Align left",
|
||||||
"Align right": "Align right",
|
"Align right": "Align right",
|
||||||
"Align center": "Align center",
|
"Align center": "Align center",
|
||||||
"Justify": "Justify.",
|
"Justify": "Justify",
|
||||||
"Merge cells": "Merge cells",
|
"Merge cells": "Merge cells",
|
||||||
"Split cell": "Split cell",
|
"Split cell": "Split cell",
|
||||||
"Delete column": "Delete column",
|
"Delete column": "Delete column",
|
||||||
@@ -312,7 +312,7 @@
|
|||||||
"Pink": "Pink",
|
"Pink": "Pink",
|
||||||
"Gray": "Gray",
|
"Gray": "Gray",
|
||||||
"Embed link": "Embed link",
|
"Embed link": "Embed link",
|
||||||
"Invalid {{provider}} embed link": "Invalid {{provider}} embed link.",
|
"Invalid {{provider}} embed link": "Invalid {{provider}} embed link",
|
||||||
"Embed {{provider}}": "Embed {{provider}}",
|
"Embed {{provider}}": "Embed {{provider}}",
|
||||||
"Enter {{provider}} link to embed": "Enter {{provider}} link to embed",
|
"Enter {{provider}} link to embed": "Enter {{provider}} link to embed",
|
||||||
"Bold": "Bold",
|
"Bold": "Bold",
|
||||||
@@ -345,41 +345,41 @@
|
|||||||
"Upload any file from your device.": "Upload any file from your device.",
|
"Upload any file from your device.": "Upload any file from your device.",
|
||||||
"Uploading {{name}}": "Uploading {{name}}",
|
"Uploading {{name}}": "Uploading {{name}}",
|
||||||
"Uploading file": "Uploading file",
|
"Uploading file": "Uploading file",
|
||||||
"Table": "Table.",
|
"Table": "Table",
|
||||||
"Insert a table.": "Insert a table.",
|
"Insert a table.": "Insert a table.",
|
||||||
"Insert collapsible block.": "Insert collapsible block.",
|
"Insert collapsible block.": "Insert collapsible block.",
|
||||||
"Video": "Video.",
|
"Video": "Video",
|
||||||
"Divider": "Divider.",
|
"Divider": "Divider",
|
||||||
"Quote": "Quote.",
|
"Quote": "Quote",
|
||||||
"Image": "Image.",
|
"Image": "Image",
|
||||||
"Audio": "Audio.",
|
"Audio": "Audio",
|
||||||
"Embed PDF": "Embed PDF",
|
"Embed PDF": "Embed PDF",
|
||||||
"Upload and embed a PDF file.": "Upload and embed a PDF file.",
|
"Upload and embed a PDF file.": "Upload and embed a PDF file.",
|
||||||
"Embed as PDF": "Embed as PDF",
|
"Embed as PDF": "Embed as PDF",
|
||||||
"Failed to load PDF": "Failed to load PDF",
|
"Failed to load PDF": "Failed to load PDF",
|
||||||
"Convert to attachment": "Convert to attachment",
|
"Convert to attachment": "Convert to attachment",
|
||||||
"File attachment": "File attachment.",
|
"File attachment": "File attachment",
|
||||||
"Toggle block": "Toggle block.",
|
"Toggle block": "Toggle block",
|
||||||
"Callout": "Callout.",
|
"Callout": "Callout",
|
||||||
"Insert callout notice.": "Insert callout notice.",
|
"Insert callout notice.": "Insert callout notice.",
|
||||||
"Math inline": "Math inline.",
|
"Math inline": "Math inline",
|
||||||
"Insert inline math equation.": "Insert inline math equation.",
|
"Insert inline math equation.": "Insert inline math equation.",
|
||||||
"Math block": "Math block.",
|
"Math block": "Math block",
|
||||||
"Insert math equation": "Insert math equation.",
|
"Insert math equation": "Insert math equation",
|
||||||
"Mermaid diagram": "Mermaid diagram.",
|
"Mermaid diagram": "Mermaid diagram",
|
||||||
"Insert mermaid diagram": "Insert mermaid diagram.",
|
"Insert mermaid diagram": "Insert mermaid diagram",
|
||||||
"Insert and design Drawio diagrams": "Insert and design Drawio diagrams.",
|
"Insert and design Drawio diagrams": "Insert and design Drawio diagrams",
|
||||||
"Insert current date": "Insert current date.",
|
"Insert current date": "Insert current date",
|
||||||
"Draw and sketch excalidraw diagrams": "Draw and sketch excalidraw diagrams.",
|
"Draw and sketch excalidraw diagrams": "Draw and sketch excalidraw diagrams",
|
||||||
"Multiple": "Multiple.",
|
"Multiple": "Multiple",
|
||||||
"Turn into": "Turn into",
|
"Turn into": "Turn into",
|
||||||
"Text align": "Text align",
|
"Text align": "Text align",
|
||||||
"This page may have been deleted, moved, or you may not have access.": "This page may have been deleted, moved, or you may not have access.",
|
"This page may have been deleted, moved, or you may not have access.": "This page may have been deleted, moved, or you may not have access.",
|
||||||
"Go to homepage": "Go to homepage",
|
"Go to homepage": "Go to homepage",
|
||||||
"Pages you create will show up here.": "Pages you create will show up here.",
|
"Pages you create will show up here.": "Pages you create will show up here.",
|
||||||
"Heading {{level}}": "Heading {{level}}.",
|
"Heading {{level}}": "Heading {{level}}",
|
||||||
"Toggle title": "Toggle title.",
|
"Toggle title": "Toggle title",
|
||||||
"Write anything. Enter \"/\" for commands": "Write anything. Enter \"/\" for commands.",
|
"Write anything. Enter \"/\" for commands": "Write anything. Enter \"/\" for commands",
|
||||||
"Write...": "Write...",
|
"Write...": "Write...",
|
||||||
"Column count": "Column count",
|
"Column count": "Column count",
|
||||||
"{{count}} Columns": "{{count}} Columns",
|
"{{count}} Columns": "{{count}} Columns",
|
||||||
@@ -389,27 +389,27 @@
|
|||||||
"Wide center": "Wide center",
|
"Wide center": "Wide center",
|
||||||
"Left wide": "Left wide",
|
"Left wide": "Left wide",
|
||||||
"Right wide": "Right wide",
|
"Right wide": "Right wide",
|
||||||
"Names do not match": "Names do not match.",
|
"Names do not match": "Names do not match",
|
||||||
"Today, {{time}}": "Today, {{time}}.",
|
"Today, {{time}}": "Today, {{time}}",
|
||||||
"Yesterday, {{time}}": "Yesterday, {{time}}.",
|
"Yesterday, {{time}}": "Yesterday, {{time}}",
|
||||||
"Space created successfully": "Space created successfully.",
|
"Space created successfully": "Space created successfully",
|
||||||
"Space updated successfully": "Space updated successfully.",
|
"Space updated successfully": "Space updated successfully",
|
||||||
"Space deleted successfully": "Space deleted successfully.",
|
"Space deleted successfully": "Space deleted successfully",
|
||||||
"Members added successfully": "Members added successfully.",
|
"Members added successfully": "Members added successfully",
|
||||||
"Member removed successfully": "Member removed successfully.",
|
"Member removed successfully": "Member removed successfully",
|
||||||
"Member role updated successfully": "Member role updated successfully.",
|
"Member role updated successfully": "Member role updated successfully",
|
||||||
"Created by: <b>{{creatorName}}</b>": "Created by: <b>{{creatorName}}</b>.",
|
"Created by: <b>{{creatorName}}</b>": "Created by: <b>{{creatorName}}</b>",
|
||||||
"Created at: {{time}}": "Created at: {{time}}.",
|
"Created at: {{time}}": "Created at: {{time}}",
|
||||||
"Edited by {{name}} {{time}}": "Edited by {{name}} {{time}}.",
|
"Edited by {{name}} {{time}}": "Edited by {{name}} {{time}}",
|
||||||
"Word count: {{wordCount}}": "Word count: {{wordCount}}.",
|
"Word count: {{wordCount}}": "Word count: {{wordCount}}",
|
||||||
"Character count: {{characterCount}}": "Character count: {{characterCount}}.",
|
"Character count: {{characterCount}}": "Character count: {{characterCount}}",
|
||||||
"New update": "New update.",
|
"New update": "New update",
|
||||||
"{{latestVersion}} is available": "{{latestVersion}} is available.",
|
"{{latestVersion}} is available": "{{latestVersion}} is available",
|
||||||
"Default page edit mode": "Default page edit mode.",
|
"Default page edit mode": "Default page edit mode",
|
||||||
"Choose your preferred page edit mode. Avoid accidental edits.": "Choose your preferred page edit mode. Avoid accidental edits.",
|
"Choose your preferred page edit mode. Avoid accidental edits.": "Choose your preferred page edit mode. Avoid accidental edits.",
|
||||||
"Reading": "Reading.",
|
"Reading": "Reading",
|
||||||
"Delete member": "Delete member.",
|
"Delete member": "Delete member",
|
||||||
"Member deleted successfully": "Member deleted successfully.",
|
"Member deleted successfully": "Member deleted successfully",
|
||||||
"Are you sure you want to delete this workspace member? This action is irreversible.": "Are you sure you want to delete this workspace member? This action is irreversible.",
|
"Are you sure you want to delete this workspace member? This action is irreversible.": "Are you sure you want to delete this workspace member? This action is irreversible.",
|
||||||
"Deactivate member": "Deactivate member",
|
"Deactivate member": "Deactivate member",
|
||||||
"Activate member": "Activate member",
|
"Activate member": "Activate member",
|
||||||
@@ -418,37 +418,40 @@
|
|||||||
"Deactivate": "Deactivate",
|
"Deactivate": "Deactivate",
|
||||||
"Activate": "Activate",
|
"Activate": "Activate",
|
||||||
"Deactivated": "Deactivated",
|
"Deactivated": "Deactivated",
|
||||||
"Move": "Move.",
|
"Move": "Move",
|
||||||
"Move page": "Move page.",
|
"Move page": "Move page",
|
||||||
"Move page to a different space.": "Move page to a different space.",
|
"Move page to a different space.": "Move page to a different space.",
|
||||||
"Real-time editor connection lost. Retrying...": "Real-time editor connection lost. Retrying...",
|
"Real-time editor connection lost. Retrying...": "Real-time editor connection lost. Retrying...",
|
||||||
"Table of contents": "Table of contents.",
|
"Table of contents": "Table of contents",
|
||||||
"Add headings (H1, H2, H3) to generate a table of contents.": "Add headings (H1, H2, H3) to generate a table of contents.",
|
"Add headings (H1, H2, H3) to generate a table of contents.": "Add headings (H1, H2, H3) to generate a table of contents.",
|
||||||
"Share": "Share.",
|
"Share": "Share",
|
||||||
"Public sharing": "Public sharing.",
|
"Public sharing": "Public sharing",
|
||||||
"Shared by": "Shared by.",
|
"Shared by": "Shared by",
|
||||||
"Shared at": "Shared at.",
|
"Shared at": "Shared at",
|
||||||
"Inherits public sharing from": "Inherits public sharing from.",
|
"Inherits public sharing from": "Inherits public sharing from",
|
||||||
"Share to web": "Share to web.",
|
"Share to web": "Share to web",
|
||||||
"Shared to web": "Shared to web.",
|
"Shared to web": "Shared to web",
|
||||||
"Anyone with the link can view this page": "Anyone with the link can view this page.",
|
"Anyone with the link can view this page": "Anyone with the link can view this page",
|
||||||
"Make this page publicly accessible": "Make this page publicly accessible.",
|
"Make this page publicly accessible": "Make this page publicly accessible",
|
||||||
"Include sub-pages": "Include sub-pages.",
|
"Include sub-pages": "Include sub-pages",
|
||||||
"Make sub-pages public too": "Make sub-pages public too.",
|
"Make sub-pages public too": "Make sub-pages public too",
|
||||||
"Allow search engines to index page": "Allow search engines to index page.",
|
"Allow search engines to index page": "Allow search engines to index page",
|
||||||
"Open page": "Open page.",
|
"Open page": "Open page",
|
||||||
"Page": "Page.",
|
"Page": "Page",
|
||||||
"Delete public share link": "Delete public share link.",
|
"Delete public share link": "Delete public share link",
|
||||||
"Delete share": "Delete share.",
|
"Delete share": "Delete share",
|
||||||
"Are you sure you want to delete this shared link?": "Are you sure you want to delete this shared link?",
|
"Are you sure you want to delete this shared link?": "Are you sure you want to delete this shared link?",
|
||||||
"Publicly shared pages from spaces you are a member of will appear here": "Publicly shared pages from spaces you are a member of will appear here.",
|
"Publicly shared pages from spaces you are a member of will appear here": "Publicly shared pages from spaces you are a member of will appear here",
|
||||||
"Share deleted successfully": "Share deleted successfully.",
|
"Share deleted successfully": "Share deleted successfully",
|
||||||
"Share not found": "Share not found.",
|
"Share not found": "Share not found",
|
||||||
"Failed to share page": "Failed to share page.",
|
"Failed to share page": "Failed to share page",
|
||||||
"Disable public sharing": "Disable public sharing",
|
"Disable public sharing": "Disable public sharing",
|
||||||
"Prevent members from sharing pages publicly.": "Prevent members from sharing pages publicly.",
|
"Prevent members from sharing pages publicly.": "Prevent members from sharing pages publicly.",
|
||||||
"Toggle public sharing": "Toggle public sharing",
|
"Toggle public sharing": "Toggle public sharing",
|
||||||
"Toggle space public sharing": "Toggle space public sharing",
|
"Toggle space public sharing": "Toggle space public sharing",
|
||||||
|
"Allow viewers to comment": "Allow viewers to comment",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Allow viewers to add comments on pages in this space.",
|
||||||
|
"Toggle viewer comments": "Toggle viewer comments",
|
||||||
"Public sharing is disabled at the workspace level": "Public sharing is disabled at the workspace level",
|
"Public sharing is disabled at the workspace level": "Public sharing is disabled at the workspace level",
|
||||||
"Prevent pages in this space from being shared publicly.": "Prevent pages in this space from being shared publicly.",
|
"Prevent pages in this space from being shared publicly.": "Prevent pages in this space from being shared publicly.",
|
||||||
"Page permissions": "Page permissions",
|
"Page permissions": "Page permissions",
|
||||||
@@ -461,135 +464,135 @@
|
|||||||
"Public sharing is disabled": "Public sharing is disabled",
|
"Public sharing is disabled": "Public sharing is disabled",
|
||||||
"Public sharing has been disabled at the workspace level.": "Public sharing has been disabled at the workspace level.",
|
"Public sharing has been disabled at the workspace level.": "Public sharing has been disabled at the workspace level.",
|
||||||
"Public sharing has been disabled for this space.": "Public sharing has been disabled for this space.",
|
"Public sharing has been disabled for this space.": "Public sharing has been disabled for this space.",
|
||||||
"Copy page": "Copy page.",
|
"Copy page": "Copy page",
|
||||||
"Copy page to a different space.": "Copy page to a different space.",
|
"Copy page to a different space.": "Copy page to a different space.",
|
||||||
"Page copied successfully": "Page copied successfully.",
|
"Page copied successfully": "Page copied successfully",
|
||||||
"Page duplicated successfully": "Page duplicated successfully.",
|
"Page duplicated successfully": "Page duplicated successfully",
|
||||||
"Find": "Find.",
|
"Find": "Find",
|
||||||
"Not found": "Not found.",
|
"Not found": "Not found",
|
||||||
"Previous Match (Shift+Enter)": "Previous Match (Shift+Enter).",
|
"Previous Match (Shift+Enter)": "Previous Match (Shift+Enter)",
|
||||||
"Next match (Enter)": "Next match (Enter).",
|
"Next match (Enter)": "Next match (Enter)",
|
||||||
"Match case (Alt+C)": "Match case (Alt+C).",
|
"Match case (Alt+C)": "Match case (Alt+C)",
|
||||||
"Replace": "Replace.",
|
"Replace": "Replace",
|
||||||
"Close (Escape)": "Close (Escape).",
|
"Close (Escape)": "Close (Escape)",
|
||||||
"Replace (Enter)": "Replace (Enter).",
|
"Replace (Enter)": "Replace (Enter)",
|
||||||
"Replace all (Ctrl+Alt+Enter)": "Replace all (Ctrl+Alt+Enter).",
|
"Replace all (Ctrl+Alt+Enter)": "Replace all (Ctrl+Alt+Enter)",
|
||||||
"Replace all": "Replace all.",
|
"Replace all": "Replace all",
|
||||||
"View all spaces": "View all spaces.",
|
"View all spaces": "View all spaces",
|
||||||
"Error": "Error.",
|
"Error": "Error",
|
||||||
"Failed to disable MFA": "Failed to disable MFA.",
|
"Failed to disable MFA": "Failed to disable MFA",
|
||||||
"Disable two-factor authentication": "Disable two-factor authentication.",
|
"Disable two-factor authentication": "Disable two-factor authentication",
|
||||||
"Disabling two-factor authentication will make your account less secure. You'll only need your password to sign in.": "Disabling two-factor authentication will make your account less secure. You'll only need your password to sign in.",
|
"Disabling two-factor authentication will make your account less secure. You'll only need your password to sign in.": "Disabling two-factor authentication will make your account less secure. You'll only need your password to sign in.",
|
||||||
"Please enter your password to disable two-factor authentication:": "Please enter your password to disable two-factor authentication:",
|
"Please enter your password to disable two-factor authentication:": "Please enter your password to disable two-factor authentication:",
|
||||||
"Two-factor authentication has been enabled": "Two-factor authentication has been enabled.",
|
"Two-factor authentication has been enabled": "Two-factor authentication has been enabled",
|
||||||
"Two-factor authentication has been disabled": "Two-factor authentication has been disabled.",
|
"Two-factor authentication has been disabled": "Two-factor authentication has been disabled",
|
||||||
"2-step verification": "2-step verification.",
|
"2-step verification": "2-step verification",
|
||||||
"Protect your account with an additional verification layer when signing in.": "Protect your account with an additional verification layer when signing in.",
|
"Protect your account with an additional verification layer when signing in.": "Protect your account with an additional verification layer when signing in.",
|
||||||
"Two-factor authentication is active on your account.": "Two-factor authentication is active on your account.",
|
"Two-factor authentication is active on your account.": "Two-factor authentication is active on your account.",
|
||||||
"Add 2FA method": "Add 2FA method.",
|
"Add 2FA method": "Add 2FA method",
|
||||||
"Backup codes": "Backup codes.",
|
"Backup codes": "Backup codes",
|
||||||
"Disable": "Disable.",
|
"Disable": "Disable",
|
||||||
"Invalid verification code": "Invalid verification code.",
|
"Invalid verification code": "Invalid verification code",
|
||||||
"New backup codes have been generated": "New backup codes have been generated.",
|
"New backup codes have been generated": "New backup codes have been generated",
|
||||||
"Failed to regenerate backup codes": "Failed to regenerate backup codes.",
|
"Failed to regenerate backup codes": "Failed to regenerate backup codes",
|
||||||
"About backup codes": "About backup codes.",
|
"About backup codes": "About backup codes",
|
||||||
"Backup codes can be used to access your account if you lose access to your authenticator app. Each code can only be used once.": "Backup codes can be used to access your account if you lose access to your authenticator app. Each code can only be used once.",
|
"Backup codes can be used to access your account if you lose access to your authenticator app. Each code can only be used once.": "Backup codes can be used to access your account if you lose access to your authenticator app. Each code can only be used once.",
|
||||||
"You can regenerate new backup codes at any time. This will invalidate all existing codes.": "You can regenerate new backup codes at any time. This will invalidate all existing codes.",
|
"You can regenerate new backup codes at any time. This will invalidate all existing codes.": "You can regenerate new backup codes at any time. This will invalidate all existing codes.",
|
||||||
"Confirm password": "Confirm password.",
|
"Confirm password": "Confirm password",
|
||||||
"Generate new backup codes": "Generate new backup codes.",
|
"Generate new backup codes": "Generate new backup codes",
|
||||||
"Save your new backup codes": "Save your new backup codes.",
|
"Save your new backup codes": "Save your new backup codes",
|
||||||
"Make sure to save these codes in a secure place. Your old backup codes are no longer valid.": "Make sure to save these codes in a secure place. Your old backup codes are no longer valid.",
|
"Make sure to save these codes in a secure place. Your old backup codes are no longer valid.": "Make sure to save these codes in a secure place. Your old backup codes are no longer valid.",
|
||||||
"Your new backup codes": "Your new backup codes.",
|
"Your new backup codes": "Your new backup codes",
|
||||||
"I've saved my backup codes": "I've saved my backup codes.",
|
"I've saved my backup codes": "I've saved my backup codes",
|
||||||
"Failed to setup MFA": "Failed to setup MFA.",
|
"Failed to setup MFA": "Failed to setup MFA",
|
||||||
"Setup & Verify": "Setup & Verify.",
|
"Setup & Verify": "Setup & Verify",
|
||||||
"Add to authenticator": "Add to authenticator.",
|
"Add to authenticator": "Add to authenticator",
|
||||||
"1. Scan this QR code with your authenticator app": "1. Scan this QR code with your authenticator app.",
|
"1. Scan this QR code with your authenticator app": "1. Scan this QR code with your authenticator app",
|
||||||
"Can't scan the code?": "Can't scan the code?",
|
"Can't scan the code?": "Can't scan the code?",
|
||||||
"Enter this code manually in your authenticator app:": "Enter this code manually in your authenticator app:",
|
"Enter this code manually in your authenticator app:": "Enter this code manually in your authenticator app:",
|
||||||
"2. Enter the 6-digit code from your authenticator": "2. Enter the 6-digit code from your authenticator.",
|
"2. Enter the 6-digit code from your authenticator": "2. Enter the 6-digit code from your authenticator",
|
||||||
"Verify and enable": "Verify and enable.",
|
"Verify and enable": "Verify and enable",
|
||||||
"Failed to generate QR code. Please try again.": "Failed to generate QR code. Please try again.",
|
"Failed to generate QR code. Please try again.": "Failed to generate QR code. Please try again.",
|
||||||
"Backup": "Backup.",
|
"Backup": "Backup",
|
||||||
"Save codes": "Save codes.",
|
"Save codes": "Save codes",
|
||||||
"Save your backup codes": "Save your backup codes.",
|
"Save your backup codes": "Save your backup codes",
|
||||||
"These codes can be used to access your account if you lose access to your authenticator app. Each code can only be used once.": "These codes can be used to access your account if you lose access to your authenticator app. Each code can only be used once.",
|
"These codes can be used to access your account if you lose access to your authenticator app. Each code can only be used once.": "These codes can be used to access your account if you lose access to your authenticator app. Each code can only be used once.",
|
||||||
"Print": "Print.",
|
"Print": "Print",
|
||||||
"Two-factor authentication has been set up. Please log in again.": "Two-factor authentication has been set up. Please log in again.",
|
"Two-factor authentication has been set up. Please log in again.": "Two-factor authentication has been set up. Please log in again.",
|
||||||
"Two-Factor authentication required": "Two-factor authentication required.",
|
"Two-Factor authentication required": "Two-factor authentication required",
|
||||||
"Your workspace requires two-factor authentication for all users": "Your workspace requires two-factor authentication for all users.",
|
"Your workspace requires two-factor authentication for all users": "Your workspace requires two-factor authentication for all users",
|
||||||
"To continue accessing your workspace, you must set up two-factor authentication. This adds an extra layer of security to your account.": "To continue accessing your workspace, you must set up two-factor authentication. This adds an extra layer of security to your account.",
|
"To continue accessing your workspace, you must set up two-factor authentication. This adds an extra layer of security to your account.": "To continue accessing your workspace, you must set up two-factor authentication. This adds an extra layer of security to your account.",
|
||||||
"Set up two-factor authentication": "Set up two-factor authentication.",
|
"Set up two-factor authentication": "Set up two-factor authentication",
|
||||||
"Cancel and logout": "Cancel and logout.",
|
"Cancel and logout": "Cancel and logout",
|
||||||
"Your workspace requires two-factor authentication. Please set it up to continue.": "Your workspace requires two-factor authentication. Please set it up to continue.",
|
"Your workspace requires two-factor authentication. Please set it up to continue.": "Your workspace requires two-factor authentication. Please set it up to continue.",
|
||||||
"This adds an extra layer of security to your account by requiring a verification code from your authenticator app.": "This adds an extra layer of security to your account by requiring a verification code from your authenticator app.",
|
"This adds an extra layer of security to your account by requiring a verification code from your authenticator app.": "This adds an extra layer of security to your account by requiring a verification code from your authenticator app.",
|
||||||
"Password is required": "Password is required.",
|
"Password is required": "Password is required",
|
||||||
"Password must be at least 8 characters": "Password must be at least 8 characters.",
|
"Password must be at least 8 characters": "Password must be at least 8 characters",
|
||||||
"Please enter a 6-digit code": "Please enter a 6-digit code.",
|
"Please enter a 6-digit code": "Please enter a 6-digit code",
|
||||||
"Code must be exactly 6 digits": "Code must be exactly 6 digits.",
|
"Code must be exactly 6 digits": "Code must be exactly 6 digits",
|
||||||
"Enter the 6-digit code found in your authenticator app": "Enter the 6-digit code found in your authenticator app.",
|
"Enter the 6-digit code found in your authenticator app": "Enter the 6-digit code found in your authenticator app",
|
||||||
"Need help authenticating?": "Need help authenticating?",
|
"Need help authenticating?": "Need help authenticating?",
|
||||||
"MFA QR Code": "MFA QR Code.",
|
"MFA QR Code": "MFA QR Code",
|
||||||
"Account created successfully. Please log in to set up two-factor authentication.": "Account created successfully. Please log in to set up two-factor authentication.",
|
"Account created successfully. Please log in to set up two-factor authentication.": "Account created successfully. Please log in to set up two-factor authentication.",
|
||||||
"Password reset successful. Please log in with your new password and complete two-factor authentication.": "Password reset successful. Please log in with your new password and complete two-factor authentication.",
|
"Password reset successful. Please log in with your new password and complete two-factor authentication.": "Password reset successful. Please log in with your new password and complete two-factor authentication.",
|
||||||
"Password reset successful. Please log in with your new password to set up two-factor authentication.": "Password reset successful. Please log in with your new password to set up two-factor authentication.",
|
"Password reset successful. Please log in with your new password to set up two-factor authentication.": "Password reset successful. Please log in with your new password to set up two-factor authentication.",
|
||||||
"Password reset was successful. Please log in with your new password.": "Password reset was successful. Please log in with your new password.",
|
"Password reset was successful. Please log in with your new password.": "Password reset was successful. Please log in with your new password.",
|
||||||
"Two-factor authentication": "Two-factor authentication.",
|
"Two-factor authentication": "Two-factor authentication",
|
||||||
"Use authenticator app instead": "Use authenticator app instead.",
|
"Use authenticator app instead": "Use authenticator app instead",
|
||||||
"Verify backup code": "Verify backup code.",
|
"Verify backup code": "Verify backup code",
|
||||||
"Use backup code": "Use backup code.",
|
"Use backup code": "Use backup code",
|
||||||
"Enter one of your backup codes": "Enter one of your backup codes.",
|
"Enter one of your backup codes": "Enter one of your backup codes",
|
||||||
"Backup code": "Backup code.",
|
"Backup code": "Backup code",
|
||||||
"Enter one of your backup codes. Each backup code can only be used once.": "Enter one of your backup codes. Each backup code can only be used once.",
|
"Enter one of your backup codes. Each backup code can only be used once.": "Enter one of your backup codes. Each backup code can only be used once.",
|
||||||
"Verify": "Verify.",
|
"Verify": "Verify",
|
||||||
"Trash": "Trash.",
|
"Trash": "Trash",
|
||||||
"Pages in trash will be permanently deleted after {{count}} days.": "Pages in trash will be permanently deleted after {{count}} days.",
|
"Pages in trash will be permanently deleted after {{count}} days.": "Pages in trash will be permanently deleted after {{count}} days.",
|
||||||
"Deleted": "Deleted.",
|
"Deleted": "Deleted",
|
||||||
"No pages in trash": "No pages in trash.",
|
"No pages in trash": "No pages in trash",
|
||||||
"Permanently delete page?": "Permanently delete page?",
|
"Permanently delete page?": "Permanently delete page?",
|
||||||
"Are you sure you want to permanently delete '{{title}}'? This action cannot be undone.": "Are you sure you want to permanently delete '{{title}}'? This action cannot be undone.",
|
"Are you sure you want to permanently delete '{{title}}'? This action cannot be undone.": "Are you sure you want to permanently delete '{{title}}'? This action cannot be undone.",
|
||||||
"Restore '{{title}}' and its sub-pages?": "Restore '{{title}}' and its sub-pages?",
|
"Restore '{{title}}' and its sub-pages?": "Restore '{{title}}' and its sub-pages?",
|
||||||
"Move to trash": "Move to trash.",
|
"Move to trash": "Move to trash",
|
||||||
"Move this page to trash?": "Move this page to trash?",
|
"Move this page to trash?": "Move this page to trash?",
|
||||||
"Restore page": "Restore page.",
|
"Restore page": "Restore page",
|
||||||
"Page moved to trash": "Page moved to trash.",
|
"Page moved to trash": "Page moved to trash",
|
||||||
"Page restored successfully": "Page restored successfully.",
|
"Page restored successfully": "Page restored successfully",
|
||||||
"Deleted by": "Deleted by.",
|
"Deleted by": "Deleted by",
|
||||||
"Deleted at": "Deleted at.",
|
"Deleted at": "Deleted at",
|
||||||
"Preview": "Preview.",
|
"Preview": "Preview",
|
||||||
"Subpages": "Subpages.",
|
"Subpages": "Subpages",
|
||||||
"Failed to load subpages": "Failed to load subpages.",
|
"Failed to load subpages": "Failed to load subpages",
|
||||||
"No subpages": "No subpages.",
|
"No subpages": "No subpages",
|
||||||
"Subpages (Child pages)": "Subpages (Child pages).",
|
"Subpages (Child pages)": "Subpages (Child pages)",
|
||||||
"List all subpages of the current page": "List all subpages of the current page.",
|
"List all subpages of the current page": "List all subpages of the current page",
|
||||||
"Attachments": "Attachments.",
|
"Attachments": "Attachments",
|
||||||
"All spaces": "All spaces.",
|
"All spaces": "All spaces",
|
||||||
"Unknown": "Unknown.",
|
"Unknown": "Unknown",
|
||||||
"Find a space": "Find a space.",
|
"Find a space": "Find a space",
|
||||||
"Search in all your spaces": "Search in all your spaces.",
|
"Search in all your spaces": "Search in all your spaces",
|
||||||
"Type": "Type.",
|
"Type": "Type",
|
||||||
"Enterprise": "Enterprise.",
|
"Enterprise": "Enterprise",
|
||||||
"Download attachment": "Download attachment.",
|
"Download attachment": "Download attachment",
|
||||||
"Allowed email domains": "Allowed email domains.",
|
"Allowed email domains": "Allowed email domains",
|
||||||
"Only users with email addresses from these domains can signup via SSO.": "Only users with email addresses from these domains can sign up via SSO.",
|
"Only users with email addresses from these domains can signup via SSO.": "Only users with email addresses from these domains can signup via SSO.",
|
||||||
"Enter valid domain names separated by comma or space": "Enter valid domain names separated by comma or space.",
|
"Enter valid domain names separated by comma or space": "Enter valid domain names separated by comma or space",
|
||||||
"Enforce two-factor authentication": "Enforce two-factor authentication.",
|
"Enforce two-factor authentication": "Enforce two-factor authentication",
|
||||||
"Once enforced, all members must enable two-factor authentication to access the workspace.": "Once enforced, all members must enable two-factor authentication to access the workspace.",
|
"Once enforced, all members must enable two-factor authentication to access the workspace.": "Once enforced, all members must enable two-factor authentication to access the workspace.",
|
||||||
"Toggle MFA enforcement": "Toggle MFA enforcement.",
|
"Toggle MFA enforcement": "Toggle MFA enforcement",
|
||||||
"Display name": "Display name.",
|
"Display name": "Display name",
|
||||||
"Allow signup": "Allow signup.",
|
"Allow signup": "Allow signup",
|
||||||
"Enabled": "Enabled.",
|
"Enabled": "Enabled",
|
||||||
"Advanced Settings": "Advanced Settings.",
|
"Advanced Settings": "Advanced Settings",
|
||||||
"Enable TLS/SSL": "Enable TLS/SSL.",
|
"Enable TLS/SSL": "Enable TLS/SSL",
|
||||||
"Use secure connection to LDAP server": "Use secure connection to LDAP server.",
|
"Use secure connection to LDAP server": "Use secure connection to LDAP server",
|
||||||
"Group sync": "Group sync.",
|
"Group sync": "Group sync",
|
||||||
"No SSO providers found.": "No SSO providers found.",
|
"No SSO providers found.": "No SSO providers found.",
|
||||||
"Delete SSO provider": "Delete SSO provider.",
|
"Delete SSO provider": "Delete SSO provider",
|
||||||
"Are you sure you want to delete this SSO provider?": "Are you sure you want to delete this SSO provider?",
|
"Are you sure you want to delete this SSO provider?": "Are you sure you want to delete this SSO provider?",
|
||||||
"Action": "Action.",
|
"Action": "Action",
|
||||||
"{{ssoProviderType}} configuration": "{{ssoProviderType}} configuration.",
|
"{{ssoProviderType}} configuration": "{{ssoProviderType}} configuration",
|
||||||
"Icon": "Icon.",
|
"Icon": "Icon",
|
||||||
"Upload image": "Upload image.",
|
"Upload image": "Upload image",
|
||||||
"Remove image": "Remove image",
|
"Remove image": "Remove image",
|
||||||
"Failed to remove image": "Failed to remove image",
|
"Failed to remove image": "Failed to remove image",
|
||||||
"Image exceeds 10MB limit.": "Image exceeds 10MB limit.",
|
"Image exceeds 10MB limit.": "Image exceeds 10MB limit.",
|
||||||
@@ -667,10 +670,28 @@
|
|||||||
"More options": "More options",
|
"More options": "More options",
|
||||||
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> mentioned you in a comment",
|
"<bold>{{name}}</bold> mentioned you in a comment": "<bold>{{name}}</bold> mentioned you in a comment",
|
||||||
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> commented on a page",
|
"<bold>{{name}}</bold> commented on a page": "<bold>{{name}}</bold> commented on a page",
|
||||||
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> resolved a comment.",
|
"<bold>{{name}}</bold> resolved a comment": "<bold>{{name}}</bold> resolved a comment",
|
||||||
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> mentioned you on a page.",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> mentioned you on a page",
|
||||||
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> gave you edit access to a page.",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> gave you edit access to a page",
|
||||||
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> gave you view access to a page.",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> gave you view access to a page",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> updated a page",
|
||||||
|
"Watch page": "Watch page",
|
||||||
|
"Stop watching": "Stop watching",
|
||||||
|
"Email notifications": "Email notifications",
|
||||||
|
"Page updates": "Page updates",
|
||||||
|
"Get notified when pages you watch are updated.": "Receive notifications when the pages you watch are updated.",
|
||||||
|
"Page mentions": "Page mentions",
|
||||||
|
"Get notified when someone mentions you on a page.": "Receive notifications when someone mentions you on a page.",
|
||||||
|
"Comment mentions": "Comment mentions",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Receive notifications when someone mentions you in a comment.",
|
||||||
|
"New comments": "New comments",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Receive notifications about new comments in threads you are participating in.",
|
||||||
|
"Resolved comments": "Resolved comments",
|
||||||
|
"Get notified when your comment is resolved.": "Receive a notification when your comment is resolved.",
|
||||||
|
"You are now watching this page": "You’re now watching this page",
|
||||||
|
"You are no longer watching this page": "You’re no longer watching this page",
|
||||||
|
"Direct": "Direct",
|
||||||
|
"Updates": "Updates",
|
||||||
"Today": "Today",
|
"Today": "Today",
|
||||||
"Yesterday": "Yesterday",
|
"Yesterday": "Yesterday",
|
||||||
"This week": "This week",
|
"This week": "This week",
|
||||||
@@ -705,32 +726,30 @@
|
|||||||
"Removed page restriction": "Removed page restriction",
|
"Removed page restriction": "Removed page restriction",
|
||||||
"Added page permission": "Added page permission",
|
"Added page permission": "Added page permission",
|
||||||
"Removed page permission": "Removed page permission",
|
"Removed page permission": "Removed page permission",
|
||||||
"Verifying your email": "Verifying your email.",
|
"Verifying your email": "Verifying your email",
|
||||||
"Please wait...": "Please wait...",
|
"Please wait...": "Please wait...",
|
||||||
"Verification failed. The link may have expired.": "Verification failed. The link may have expired.",
|
"Verification failed. The link may have expired.": "Verification failed. The link may have expired.",
|
||||||
"Check your email": "Check your email.",
|
"Check your email": "Check your email",
|
||||||
"We sent a verification link to {{email}}.": "We sent a verification link to {{email}}.",
|
"We sent a verification link to {{email}}.": "We sent a verification link to {{email}}.",
|
||||||
"We sent a verification link to your email.": "We sent a verification link to your email.",
|
"We sent a verification link to your email.": "We sent a verification link to your email.",
|
||||||
"Click the link to verify your email and access your workspace.": "Click the link to verify your email and access your workspace.",
|
"Click the link to verify your email and access your workspace.": "Click the link to verify your email and access your workspace.",
|
||||||
"Resend verification email": "Resend verification email.",
|
"Resend verification email": "Resend verification email",
|
||||||
"Verification email sent. Please check your inbox.": "Verification email sent. Please check your inbox.",
|
"Verification email sent. Please check your inbox.": "Verification email sent. Please check your inbox.",
|
||||||
"Failed to resend verification email. Please try again.": "Failed to resend verification email. Please try again.",
|
"Failed to resend verification email. Please try again.": "Failed to resend verification email. Please try again.",
|
||||||
"We've sent you an email with your associated workspaces.": "We've sent you an email with your associated workspaces.",
|
"We've sent you an email with your associated workspaces.": "We've sent you an email with your associated workspaces.",
|
||||||
"Load more": "Load more.",
|
"Load more": "Load more",
|
||||||
"Log out of all devices": "Log out of all devices.",
|
"Log out of all devices": "Log out of all devices",
|
||||||
"Log out of all sessions except this device": "Log out of all sessions except this device.",
|
"Log out of all sessions except this device": "Log out of all sessions except this device",
|
||||||
"This Device": "This Device.",
|
"This Device": "This Device",
|
||||||
"Unknown device": "Unknown device.",
|
"Unknown device": "Unknown device",
|
||||||
"No active sessions": "No active sessions.",
|
"No active sessions": "No active sessions",
|
||||||
"Session revoked": "Session revoked.",
|
"Session revoked": "Session revoked",
|
||||||
"All other sessions revoked": "All other sessions revoked.",
|
"All other sessions revoked": "All other sessions revoked",
|
||||||
"Last used": "Last used.",
|
"Last used": "Last used",
|
||||||
"Created": "Created.",
|
"Created": "Created",
|
||||||
"Rename": "Rename.",
|
"Rename": "Rename",
|
||||||
"Publish": "Publish.",
|
"Publish": "Publish",
|
||||||
"Security": "Security.",
|
"Security": "Security",
|
||||||
"Enforce SSO": "Enforce SSO.",
|
"Enforce SSO": "Enforce SSO",
|
||||||
"Once enforced, members will not be able to login with email and password.": "Once enforced, members will not be able to log in with email and password.",
|
"Once enforced, members will not be able to login with email and password.": "Once enforced, members will not be able to login with email and password."
|
||||||
"Uploading {{name}}": "Uploading {{name}}",
|
|
||||||
"Uploading file": "Uploading file"
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -341,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Insertar regla horizontal",
|
"Insert horizontal rule divider": "Insertar regla horizontal",
|
||||||
"Upload any image from your device.": "Sube cualquier imagen desde tu dispositivo.",
|
"Upload any image from your device.": "Sube cualquier imagen desde tu dispositivo.",
|
||||||
"Upload any video from your device.": "Sube cualquier video desde tu dispositivo.",
|
"Upload any video from your device.": "Sube cualquier video desde tu dispositivo.",
|
||||||
|
"Upload any audio from your device.": "Sube cualquier audio desde tu dispositivo.",
|
||||||
"Upload any file from your device.": "Sube cualquier archivo desde tu dispositivo.",
|
"Upload any file from your device.": "Sube cualquier archivo desde tu dispositivo.",
|
||||||
"Uploading {{name}}": "Subiendo {{name}}",
|
"Uploading {{name}}": "Subiendo {{name}}",
|
||||||
"Uploading file": "Subiendo archivo",
|
"Uploading file": "Subiendo archivo",
|
||||||
@@ -351,6 +352,12 @@
|
|||||||
"Divider": "Divisor",
|
"Divider": "Divisor",
|
||||||
"Quote": "Cita",
|
"Quote": "Cita",
|
||||||
"Image": "Imagen",
|
"Image": "Imagen",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "Adjuntar PDF",
|
||||||
|
"Upload and embed a PDF file.": "Sube y adjunta un archivo PDF.",
|
||||||
|
"Embed as PDF": "Adjuntar como PDF",
|
||||||
|
"Failed to load PDF": "Error al cargar el PDF",
|
||||||
|
"Convert to attachment": "Convertir en adjunto",
|
||||||
"File attachment": "Adjunto de archivo",
|
"File attachment": "Adjunto de archivo",
|
||||||
"Toggle block": "Alternar bloque",
|
"Toggle block": "Alternar bloque",
|
||||||
"Callout": "Aviso",
|
"Callout": "Aviso",
|
||||||
@@ -442,6 +449,9 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Evitar que los miembros compartan páginas públicamente.",
|
"Prevent members from sharing pages publicly.": "Evitar que los miembros compartan páginas públicamente.",
|
||||||
"Toggle public sharing": "Alternar el uso compartido público",
|
"Toggle public sharing": "Alternar el uso compartido público",
|
||||||
"Toggle space public sharing": "Alternar el uso compartido público del espacio",
|
"Toggle space public sharing": "Alternar el uso compartido público del espacio",
|
||||||
|
"Allow viewers to comment": "Permitir que los espectadores comenten",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Permitir que los espectadores agreguen comentarios en las páginas de este espacio.",
|
||||||
|
"Toggle viewer comments": "Activar/desactivar comentarios de los espectadores",
|
||||||
"Public sharing is disabled at the workspace level": "El uso compartido público está desactivado a nivel de espacio de trabajo",
|
"Public sharing is disabled at the workspace level": "El uso compartido público está desactivado a nivel de espacio de trabajo",
|
||||||
"Prevent pages in this space from being shared publicly.": "Evitar que las páginas en este espacio se compartan públicamente.",
|
"Prevent pages in this space from being shared publicly.": "Evitar que las páginas en este espacio se compartan públicamente.",
|
||||||
"Page permissions": "Permisos de la página},{",
|
"Page permissions": "Permisos de la página},{",
|
||||||
@@ -664,6 +674,24 @@
|
|||||||
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> te mencionó en una página",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> te mencionó en una página",
|
||||||
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> te dio acceso de edición a una página",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> te dio acceso de edición a una página",
|
||||||
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> te dio acceso de visualización a una página",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> te dio acceso de visualización a una página",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> actualizó una página.",
|
||||||
|
"Watch page": "Seguir página",
|
||||||
|
"Stop watching": "Dejar de seguir",
|
||||||
|
"Email notifications": "Notificaciones por correo electrónico",
|
||||||
|
"Page updates": "Actualizaciones de página",
|
||||||
|
"Get notified when pages you watch are updated.": "Recibe una notificación cuando se actualicen las páginas que sigues.",
|
||||||
|
"Page mentions": "Menciones en la página",
|
||||||
|
"Get notified when someone mentions you on a page.": "Recibe una notificación cuando alguien te mencione en una página.",
|
||||||
|
"Comment mentions": "Menciones en comentarios",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Recibe una notificación cuando alguien te mencione en un comentario.",
|
||||||
|
"New comments": "Nuevos comentarios",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Recibe una notificación sobre nuevos comentarios en los hilos donde participas.",
|
||||||
|
"Resolved comments": "Comentarios resueltos",
|
||||||
|
"Get notified when your comment is resolved.": "Recibe una notificación cuando tu comentario sea resuelto.",
|
||||||
|
"You are now watching this page": "Ahora sigues esta página",
|
||||||
|
"You are no longer watching this page": "Ya no sigues esta página",
|
||||||
|
"Direct": "Directo",
|
||||||
|
"Updates": "Actualizaciones",
|
||||||
"Today": "Hoy",
|
"Today": "Hoy",
|
||||||
"Yesterday": "Ayer",
|
"Yesterday": "Ayer",
|
||||||
"This week": "Esta semana",
|
"This week": "Esta semana",
|
||||||
|
|||||||
@@ -341,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Insérer un séparateur de règle horizontale",
|
"Insert horizontal rule divider": "Insérer un séparateur de règle horizontale",
|
||||||
"Upload any image from your device.": "Téléchargez n'importe quelle image depuis votre appareil.",
|
"Upload any image from your device.": "Téléchargez n'importe quelle image depuis votre appareil.",
|
||||||
"Upload any video from your device.": "Téléchargez n'importe quelle vidéo depuis votre appareil.",
|
"Upload any video from your device.": "Téléchargez n'importe quelle vidéo depuis votre appareil.",
|
||||||
|
"Upload any audio from your device.": "Téléchargez n'importe quel fichier audio depuis votre appareil.",
|
||||||
"Upload any file from your device.": "Téléchargez n'importe quel fichier depuis votre appareil.",
|
"Upload any file from your device.": "Téléchargez n'importe quel fichier depuis votre appareil.",
|
||||||
"Uploading {{name}}": "Téléchargement de {{name}}",
|
"Uploading {{name}}": "Téléchargement de {{name}}",
|
||||||
"Uploading file": "Téléchargement du fichier",
|
"Uploading file": "Téléchargement du fichier",
|
||||||
@@ -351,6 +352,12 @@
|
|||||||
"Divider": "Diviseur",
|
"Divider": "Diviseur",
|
||||||
"Quote": "Citation",
|
"Quote": "Citation",
|
||||||
"Image": "Image",
|
"Image": "Image",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "Intégrer un PDF",
|
||||||
|
"Upload and embed a PDF file.": "Téléchargez et intégrez un fichier PDF.",
|
||||||
|
"Embed as PDF": "Intégrer comme PDF",
|
||||||
|
"Failed to load PDF": "Échec du chargement du PDF",
|
||||||
|
"Convert to attachment": "Convertir en pièce jointe",
|
||||||
"File attachment": "Pièce jointe",
|
"File attachment": "Pièce jointe",
|
||||||
"Toggle block": "Basculer le bloc",
|
"Toggle block": "Basculer le bloc",
|
||||||
"Callout": "Appel",
|
"Callout": "Appel",
|
||||||
@@ -415,7 +422,7 @@
|
|||||||
"Move page": "Déplacer la page",
|
"Move page": "Déplacer la page",
|
||||||
"Move page to a different space.": "Déplacer la page vers un autre espace.",
|
"Move page to a different space.": "Déplacer la page vers un autre espace.",
|
||||||
"Real-time editor connection lost. Retrying...": "Connexion avec l'éditeur en temps réel perdue. Nouvelle tentative...",
|
"Real-time editor connection lost. Retrying...": "Connexion avec l'éditeur en temps réel perdue. Nouvelle tentative...",
|
||||||
"Table of contents": "",
|
"Table of contents": "Table des matières.",
|
||||||
"Add headings (H1, H2, H3) to generate a table of contents.": "Ajoutez des titres (H1, H2, H3) pour générer une table des matières.",
|
"Add headings (H1, H2, H3) to generate a table of contents.": "Ajoutez des titres (H1, H2, H3) pour générer une table des matières.",
|
||||||
"Share": "Partager",
|
"Share": "Partager",
|
||||||
"Public sharing": "Partage public",
|
"Public sharing": "Partage public",
|
||||||
@@ -442,6 +449,9 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Empêcher les membres de partager des pages publiquement.",
|
"Prevent members from sharing pages publicly.": "Empêcher les membres de partager des pages publiquement.",
|
||||||
"Toggle public sharing": "Basculer le partage public",
|
"Toggle public sharing": "Basculer le partage public",
|
||||||
"Toggle space public sharing": "Basculer le partage public de l'espace",
|
"Toggle space public sharing": "Basculer le partage public de l'espace",
|
||||||
|
"Allow viewers to comment": "Autoriser les spectateurs à commenter",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Autoriser les spectateurs à ajouter des commentaires sur les pages de cet espace.",
|
||||||
|
"Toggle viewer comments": "Basculer les commentaires des spectateurs",
|
||||||
"Public sharing is disabled at the workspace level": "Le partage public est désactivé au niveau de l'espace de travail",
|
"Public sharing is disabled at the workspace level": "Le partage public est désactivé au niveau de l'espace de travail",
|
||||||
"Prevent pages in this space from being shared publicly.": "Empêcher les pages de cet espace d'être partagées publiquement.",
|
"Prevent pages in this space from being shared publicly.": "Empêcher les pages de cet espace d'être partagées publiquement.",
|
||||||
"Page permissions": "Autorisations de la page",
|
"Page permissions": "Autorisations de la page",
|
||||||
@@ -664,6 +674,24 @@
|
|||||||
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> vous a mentionné sur une page",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> vous a mentionné sur une page",
|
||||||
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> vous a donné l'accès en modification à une page",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> vous a donné l'accès en modification à une page",
|
||||||
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> vous a donné l'accès en lecture à une page",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> vous a donné l'accès en lecture à une page",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> a mis à jour une page.",
|
||||||
|
"Watch page": "Surveiller la page",
|
||||||
|
"Stop watching": "Ne plus surveiller",
|
||||||
|
"Email notifications": "Notifications par e-mail",
|
||||||
|
"Page updates": "Mises à jour de la page",
|
||||||
|
"Get notified when pages you watch are updated.": "Recevez une notification lorsque les pages que vous surveillez sont mises à jour.",
|
||||||
|
"Page mentions": "Mentions sur la page",
|
||||||
|
"Get notified when someone mentions you on a page.": "Recevez une notification lorsqu'une personne vous mentionne sur une page.",
|
||||||
|
"Comment mentions": "Mentions dans les commentaires",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Recevez une notification lorsqu'une personne vous mentionne dans un commentaire.",
|
||||||
|
"New comments": "Nouveaux commentaires",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Recevez une notification concernant les nouveaux commentaires dans les fils auxquels vous participez.",
|
||||||
|
"Resolved comments": "Commentaires résolus",
|
||||||
|
"Get notified when your comment is resolved.": "Recevez une notification lorsque votre commentaire est résolu.",
|
||||||
|
"You are now watching this page": "Vous surveillez désormais cette page",
|
||||||
|
"You are no longer watching this page": "Vous ne surveillez plus cette page",
|
||||||
|
"Direct": "Direct",
|
||||||
|
"Updates": "Mises à jour",
|
||||||
"Today": "Aujourd'hui",
|
"Today": "Aujourd'hui",
|
||||||
"Yesterday": "Hier",
|
"Yesterday": "Hier",
|
||||||
"This week": "Cette semaine",
|
"This week": "Cette semaine",
|
||||||
|
|||||||
@@ -341,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Inserisci divisore di regola orizzontale",
|
"Insert horizontal rule divider": "Inserisci divisore di regola orizzontale",
|
||||||
"Upload any image from your device.": "Carica un'immagine dal tuo dispositivo.",
|
"Upload any image from your device.": "Carica un'immagine dal tuo dispositivo.",
|
||||||
"Upload any video from your device.": "Carica qualsiasi video dal tuo dispositivo.",
|
"Upload any video from your device.": "Carica qualsiasi video dal tuo dispositivo.",
|
||||||
|
"Upload any audio from your device.": "Carica qualsiasi audio dal tuo dispositivo.",
|
||||||
"Upload any file from your device.": "Carica qualsiasi file dal tuo dispositivo.",
|
"Upload any file from your device.": "Carica qualsiasi file dal tuo dispositivo.",
|
||||||
"Uploading {{name}}": "Caricamento di {{name}}",
|
"Uploading {{name}}": "Caricamento di {{name}}",
|
||||||
"Uploading file": "Caricamento file",
|
"Uploading file": "Caricamento file",
|
||||||
@@ -351,6 +352,12 @@
|
|||||||
"Divider": "Divisore",
|
"Divider": "Divisore",
|
||||||
"Quote": "Preventivo",
|
"Quote": "Preventivo",
|
||||||
"Image": "Immagine",
|
"Image": "Immagine",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "Incorpora PDF",
|
||||||
|
"Upload and embed a PDF file.": "Carica e incorpora un file PDF.",
|
||||||
|
"Embed as PDF": "Incorpora come PDF",
|
||||||
|
"Failed to load PDF": "Caricamento del PDF non riuscito",
|
||||||
|
"Convert to attachment": "Converti in allegato",
|
||||||
"File attachment": "Allegato file",
|
"File attachment": "Allegato file",
|
||||||
"Toggle block": "Attiva blocco",
|
"Toggle block": "Attiva blocco",
|
||||||
"Callout": "Avviso",
|
"Callout": "Avviso",
|
||||||
@@ -442,6 +449,9 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Impedisci ai membri di condividere pubblicamente le pagine.",
|
"Prevent members from sharing pages publicly.": "Impedisci ai membri di condividere pubblicamente le pagine.",
|
||||||
"Toggle public sharing": "Attiva/disattiva la condivisione pubblica",
|
"Toggle public sharing": "Attiva/disattiva la condivisione pubblica",
|
||||||
"Toggle space public sharing": "Attiva/disattiva la condivisione pubblica nello spazio",
|
"Toggle space public sharing": "Attiva/disattiva la condivisione pubblica nello spazio",
|
||||||
|
"Allow viewers to comment": "Consenti agli utenti di commentare",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Consenti agli utenti di aggiungere commenti alle pagine in questo spazio.",
|
||||||
|
"Toggle viewer comments": "Attiva/disattiva i commenti degli utenti",
|
||||||
"Public sharing is disabled at the workspace level": "La condivisione pubblica è disabilitata a livello di area di lavoro",
|
"Public sharing is disabled at the workspace level": "La condivisione pubblica è disabilitata a livello di area di lavoro",
|
||||||
"Prevent pages in this space from being shared publicly.": "Impedisci che le pagine in questo spazio vengano condivise pubblicamente.",
|
"Prevent pages in this space from being shared publicly.": "Impedisci che le pagine in questo spazio vengano condivise pubblicamente.",
|
||||||
"Page permissions": "Autorizzazioni della pagina.",
|
"Page permissions": "Autorizzazioni della pagina.",
|
||||||
@@ -664,6 +674,24 @@
|
|||||||
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> ti ha menzionato su una pagina",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> ti ha menzionato su una pagina",
|
||||||
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> ti ha dato l'accesso di modifica a una pagina",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> ti ha dato l'accesso di modifica a una pagina",
|
||||||
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> ti ha dato l'accesso di visualizzazione a una pagina",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> ti ha dato l'accesso di visualizzazione a una pagina",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> ha aggiornato una pagina.",
|
||||||
|
"Watch page": "Segui pagina",
|
||||||
|
"Stop watching": "Smetti di seguire",
|
||||||
|
"Email notifications": "Notifiche email",
|
||||||
|
"Page updates": "Aggiornamenti pagina",
|
||||||
|
"Get notified when pages you watch are updated.": "Ricevi una notifica quando le pagine che segui vengono aggiornate.",
|
||||||
|
"Page mentions": "Menzioni nella pagina",
|
||||||
|
"Get notified when someone mentions you on a page.": "Ricevi una notifica quando qualcuno ti menziona su una pagina.",
|
||||||
|
"Comment mentions": "Menzioni nei commenti",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Ricevi una notifica quando qualcuno ti menziona in un commento.",
|
||||||
|
"New comments": "Nuovi commenti",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Ricevi una notifica sui nuovi commenti nelle discussioni a cui partecipi.",
|
||||||
|
"Resolved comments": "Commenti risolti",
|
||||||
|
"Get notified when your comment is resolved.": "Ricevi una notifica quando il tuo commento viene risolto.",
|
||||||
|
"You are now watching this page": "Ora stai seguendo questa pagina",
|
||||||
|
"You are no longer watching this page": "Non stai più seguendo questa pagina",
|
||||||
|
"Direct": "Diretto",
|
||||||
|
"Updates": "Aggiornamenti",
|
||||||
"Today": "Oggi",
|
"Today": "Oggi",
|
||||||
"Yesterday": "Ieri",
|
"Yesterday": "Ieri",
|
||||||
"This week": "Questa settimana",
|
"This week": "Questa settimana",
|
||||||
|
|||||||
@@ -341,6 +341,7 @@
|
|||||||
"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 audio from your device.": "デバイスから音声ファイルをアップロードします。",
|
||||||
"Upload any file from your device.": "デバイスからファイルをアップロードします",
|
"Upload any file from your device.": "デバイスからファイルをアップロードします",
|
||||||
"Uploading {{name}}": "{{name}} をアップロード中",
|
"Uploading {{name}}": "{{name}} をアップロード中",
|
||||||
"Uploading file": "ファイルをアップロード中",
|
"Uploading file": "ファイルをアップロード中",
|
||||||
@@ -351,6 +352,12 @@
|
|||||||
"Divider": "区切り線",
|
"Divider": "区切り線",
|
||||||
"Quote": "引用",
|
"Quote": "引用",
|
||||||
"Image": "画像",
|
"Image": "画像",
|
||||||
|
"Audio": "音声。",
|
||||||
|
"Embed PDF": "PDFを埋め込む",
|
||||||
|
"Upload and embed a PDF file.": "PDFファイルをアップロードして埋め込みます。",
|
||||||
|
"Embed as PDF": "PDFとして埋め込む",
|
||||||
|
"Failed to load PDF": "PDFの読み込みに失敗しました",
|
||||||
|
"Convert to attachment": "添付ファイルに変換",
|
||||||
"File attachment": "ファイル添付",
|
"File attachment": "ファイル添付",
|
||||||
"Toggle block": "ブロックを切り替える",
|
"Toggle block": "ブロックを切り替える",
|
||||||
"Callout": "コールアウト",
|
"Callout": "コールアウト",
|
||||||
@@ -442,6 +449,9 @@
|
|||||||
"Prevent members from sharing pages publicly.": "メンバーがページを公開で共有するのを防ぐ。",
|
"Prevent members from sharing pages publicly.": "メンバーがページを公開で共有するのを防ぐ。",
|
||||||
"Toggle public sharing": "公開共有を切り替える",
|
"Toggle public sharing": "公開共有を切り替える",
|
||||||
"Toggle space public sharing": "スペースの公開共有を切り替える",
|
"Toggle space public sharing": "スペースの公開共有を切り替える",
|
||||||
|
"Allow viewers to comment": "閲覧者によるコメントを許可",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "このスペース内のページに閲覧者がコメントを追加できるようにします。",
|
||||||
|
"Toggle viewer comments": "閲覧者コメントの切り替え",
|
||||||
"Public sharing is disabled at the workspace level": "ワークスペースレベルで公開共有が無効になっています",
|
"Public sharing is disabled at the workspace level": "ワークスペースレベルで公開共有が無効になっています",
|
||||||
"Prevent pages in this space from being shared publicly.": "このスペース内のページが公開で共有されるのを防ぐ。",
|
"Prevent pages in this space from being shared publicly.": "このスペース内のページが公開で共有されるのを防ぐ。",
|
||||||
"Page permissions": "ページのアクセス権",
|
"Page permissions": "ページのアクセス権",
|
||||||
@@ -664,6 +674,24 @@
|
|||||||
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold>さんがページであなたに言及しました",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold>さんがページであなたに言及しました",
|
||||||
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold>さんがページの編集権限をあなたに付与しました",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold>さんがページの編集権限をあなたに付与しました",
|
||||||
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold>さんがページの閲覧権限をあなたに付与しました",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold>さんがページの閲覧権限をあなたに付与しました",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold>さんがページを更新しました。",
|
||||||
|
"Watch page": "ページをウォッチ",
|
||||||
|
"Stop watching": "ウォッチを解除",
|
||||||
|
"Email notifications": "メール通知",
|
||||||
|
"Page updates": "ページの更新",
|
||||||
|
"Get notified when pages you watch are updated.": "ウォッチしているページが更新されたときに通知を受け取ります。",
|
||||||
|
"Page mentions": "ページでの言及",
|
||||||
|
"Get notified when someone mentions you on a page.": "誰かがページであなたに言及したとき通知を受け取ります。",
|
||||||
|
"Comment mentions": "コメントでの言及",
|
||||||
|
"Get notified when someone mentions you in a comment.": "誰かがコメントであなたに言及したとき通知を受け取ります。",
|
||||||
|
"New comments": "新しいコメント",
|
||||||
|
"Get notified about new comments on threads you participate in.": "参加しているスレッドに新しいコメントがあると通知されます。",
|
||||||
|
"Resolved comments": "解決済みコメント",
|
||||||
|
"Get notified when your comment is resolved.": "あなたのコメントが解決されたとき通知を受け取ります。",
|
||||||
|
"You are now watching this page": "このページをウォッチしています",
|
||||||
|
"You are no longer watching this page": "このページのウォッチを解除しました",
|
||||||
|
"Direct": "直接",
|
||||||
|
"Updates": "アップデート",
|
||||||
"Today": "今日",
|
"Today": "今日",
|
||||||
"Yesterday": "昨日",
|
"Yesterday": "昨日",
|
||||||
"This week": "今週",
|
"This week": "今週",
|
||||||
|
|||||||
@@ -341,6 +341,7 @@
|
|||||||
"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 audio from your device.": "기기에서 오디오를 업로드하세요.",
|
||||||
"Upload any file from your device.": "기기에서 파일을 업로드하세요.",
|
"Upload any file from your device.": "기기에서 파일을 업로드하세요.",
|
||||||
"Uploading {{name}}": "{{name}} 업로드 중",
|
"Uploading {{name}}": "{{name}} 업로드 중",
|
||||||
"Uploading file": "파일 업로드 중",
|
"Uploading file": "파일 업로드 중",
|
||||||
@@ -351,6 +352,12 @@
|
|||||||
"Divider": "구분선",
|
"Divider": "구분선",
|
||||||
"Quote": "인용",
|
"Quote": "인용",
|
||||||
"Image": "이미지",
|
"Image": "이미지",
|
||||||
|
"Audio": "오디오.",
|
||||||
|
"Embed PDF": "PDF 임베드",
|
||||||
|
"Upload and embed a PDF file.": "PDF 파일을 업로드하고 임베드하세요.",
|
||||||
|
"Embed as PDF": "PDF로 임베드",
|
||||||
|
"Failed to load PDF": "PDF 로드 실패",
|
||||||
|
"Convert to attachment": "첨부 파일로 변환",
|
||||||
"File attachment": "파일 첨부",
|
"File attachment": "파일 첨부",
|
||||||
"Toggle block": "블록 토글",
|
"Toggle block": "블록 토글",
|
||||||
"Callout": "경고 상자",
|
"Callout": "경고 상자",
|
||||||
@@ -442,6 +449,9 @@
|
|||||||
"Prevent members from sharing pages publicly.": "멤버들이 페이지를 공개적으로 공유하지 못하도록 방지하십시오.",
|
"Prevent members from sharing pages publicly.": "멤버들이 페이지를 공개적으로 공유하지 못하도록 방지하십시오.",
|
||||||
"Toggle public sharing": "공유 전환",
|
"Toggle public sharing": "공유 전환",
|
||||||
"Toggle space public sharing": "공간 공유 전환",
|
"Toggle space public sharing": "공간 공유 전환",
|
||||||
|
"Allow viewers to comment": "뷰어가 댓글을 달 수 있도록 허용",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "이 공간 내 페이지에 뷰어가 댓글을 추가할 수 있도록 허용합니다.",
|
||||||
|
"Toggle viewer comments": "뷰어 댓글 전환",
|
||||||
"Public sharing is disabled at the workspace level": "워크스페이스 수준에서 공유가 비활성화되었습니다.",
|
"Public sharing is disabled at the workspace level": "워크스페이스 수준에서 공유가 비활성화되었습니다.",
|
||||||
"Prevent pages in this space from being shared publicly.": "이 공간의 페이지가 공개적으로 공유되지 않도록 방지하십시오.",
|
"Prevent pages in this space from being shared publicly.": "이 공간의 페이지가 공개적으로 공유되지 않도록 방지하십시오.",
|
||||||
"Page permissions": "페이지 권한},{",
|
"Page permissions": "페이지 권한},{",
|
||||||
@@ -664,6 +674,24 @@
|
|||||||
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold>님이 페이지에서 당신을 언급했습니다",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold>님이 페이지에서 당신을 언급했습니다",
|
||||||
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold>님이 페이지 편집 권한을 부여했습니다",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold>님이 페이지 편집 권한을 부여했습니다",
|
||||||
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold>님이 페이지 조회 권한을 부여했습니다",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold>님이 페이지 조회 권한을 부여했습니다",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold>님이 페이지를 업데이트했습니다.",
|
||||||
|
"Watch page": "페이지 구독",
|
||||||
|
"Stop watching": "구독 취소",
|
||||||
|
"Email notifications": "이메일 알림",
|
||||||
|
"Page updates": "페이지 업데이트",
|
||||||
|
"Get notified when pages you watch are updated.": "구독한 페이지가 업데이트될 때 알림을 받으세요.",
|
||||||
|
"Page mentions": "페이지 언급",
|
||||||
|
"Get notified when someone mentions you on a page.": "누군가가 페이지에서 당신을 언급하면 알림을 받으세요.",
|
||||||
|
"Comment mentions": "댓글 언급",
|
||||||
|
"Get notified when someone mentions you in a comment.": "누군가가 댓글에서 당신을 언급하면 알림을 받으세요.",
|
||||||
|
"New comments": "새 댓글",
|
||||||
|
"Get notified about new comments on threads you participate in.": "참여 중인 스레드에 새 댓글이 달리면 알림을 받으세요.",
|
||||||
|
"Resolved comments": "해결된 댓글",
|
||||||
|
"Get notified when your comment is resolved.": "내 댓글이 해결되었을 때 알림을 받으세요.",
|
||||||
|
"You are now watching this page": "이제 이 페이지를 주시합니다.",
|
||||||
|
"You are no longer watching this page": "더 이상 이 페이지를 주시하지 않습니다.",
|
||||||
|
"Direct": "직접",
|
||||||
|
"Updates": "업데이트",
|
||||||
"Today": "오늘",
|
"Today": "오늘",
|
||||||
"Yesterday": "어제",
|
"Yesterday": "어제",
|
||||||
"This week": "이번 주",
|
"This week": "이번 주",
|
||||||
|
|||||||
@@ -341,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Horizontale lijn invoegen",
|
"Insert horizontal rule divider": "Horizontale lijn invoegen",
|
||||||
"Upload any image from your device.": "Upload een afbeelding vanaf uw apparaat.",
|
"Upload any image from your device.": "Upload een afbeelding vanaf uw apparaat.",
|
||||||
"Upload any video from your device.": "Upload een video vanaf uw apparaat.",
|
"Upload any video from your device.": "Upload een video vanaf uw apparaat.",
|
||||||
|
"Upload any audio from your device.": "Upload een audio vanaf uw apparaat.",
|
||||||
"Upload any file from your device.": "Upload een bestand vanaf uw apparaat.",
|
"Upload any file from your device.": "Upload een bestand vanaf uw apparaat.",
|
||||||
"Uploading {{name}}": "Uploaden {{name}}",
|
"Uploading {{name}}": "Uploaden {{name}}",
|
||||||
"Uploading file": "Bestand uploaden",
|
"Uploading file": "Bestand uploaden",
|
||||||
@@ -351,6 +352,12 @@
|
|||||||
"Divider": "Scheidingslijn",
|
"Divider": "Scheidingslijn",
|
||||||
"Quote": "Quote",
|
"Quote": "Quote",
|
||||||
"Image": "Afbeelding",
|
"Image": "Afbeelding",
|
||||||
|
"Audio": "Audio.",
|
||||||
|
"Embed PDF": "PDF insluiten",
|
||||||
|
"Upload and embed a PDF file.": "Upload en sluit een PDF-bestand in.",
|
||||||
|
"Embed as PDF": "Insluiten als PDF",
|
||||||
|
"Failed to load PDF": "Laden van PDF mislukt",
|
||||||
|
"Convert to attachment": "Converteren naar bijlage",
|
||||||
"File attachment": "Bestand bijlage",
|
"File attachment": "Bestand bijlage",
|
||||||
"Toggle block": "Schakel blok in/uit",
|
"Toggle block": "Schakel blok in/uit",
|
||||||
"Callout": "Opmerking",
|
"Callout": "Opmerking",
|
||||||
@@ -442,6 +449,9 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Voorkom dat leden pagina's openbaar delen.",
|
"Prevent members from sharing pages publicly.": "Voorkom dat leden pagina's openbaar delen.",
|
||||||
"Toggle public sharing": "Wissel openbaar delen",
|
"Toggle public sharing": "Wissel openbaar delen",
|
||||||
"Toggle space public sharing": "Wissel openbaar delen van ruimte",
|
"Toggle space public sharing": "Wissel openbaar delen van ruimte",
|
||||||
|
"Allow viewers to comment": "Toestaan dat kijkers reageren",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Sta kijkers toe om reacties toe te voegen op pagina\u0019s in deze ruimte.",
|
||||||
|
"Toggle viewer comments": "Reacties van kijkers in- of uitschakelen",
|
||||||
"Public sharing is disabled at the workspace level": "Openbaar delen is uitgeschakeld op werkruimteniveau",
|
"Public sharing is disabled at the workspace level": "Openbaar delen is uitgeschakeld op werkruimteniveau",
|
||||||
"Prevent pages in this space from being shared publicly.": "Voorkom dat pagina's in deze ruimte openbaar worden gedeeld.",
|
"Prevent pages in this space from being shared publicly.": "Voorkom dat pagina's in deze ruimte openbaar worden gedeeld.",
|
||||||
"Page permissions": "Pagina rechten",
|
"Page permissions": "Pagina rechten",
|
||||||
@@ -664,6 +674,24 @@
|
|||||||
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> noemde je op een pagina",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> noemde je op een pagina",
|
||||||
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> heeft je toegang gegeven om een pagina te bewerken",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> heeft je toegang gegeven om een pagina te bewerken",
|
||||||
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> heeft je toegang gegeven om een pagina te bekijken",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> heeft je toegang gegeven om een pagina te bekijken",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> heeft een pagina bijgewerkt.",
|
||||||
|
"Watch page": "Pagina volgen",
|
||||||
|
"Stop watching": "Volgen stoppen",
|
||||||
|
"Email notifications": "E-mailmeldingen",
|
||||||
|
"Page updates": "Pagina-updates",
|
||||||
|
"Get notified when pages you watch are updated.": "Ontvang een melding wanneer pagina's die je volgt worden bijgewerkt.",
|
||||||
|
"Page mentions": "Pagina-vermeldingen",
|
||||||
|
"Get notified when someone mentions you on a page.": "Ontvang een melding wanneer iemand je noemt op een pagina.",
|
||||||
|
"Comment mentions": "Vermeldingen in opmerkingen",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Ontvang een melding wanneer iemand je noemt in een opmerking.",
|
||||||
|
"New comments": "Nieuwe opmerkingen",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Ontvang meldingen over nieuwe reacties in threads waaraan je deelneemt.",
|
||||||
|
"Resolved comments": "Opgeloste opmerkingen",
|
||||||
|
"Get notified when your comment is resolved.": "Ontvang een melding wanneer je reactie is opgelost.",
|
||||||
|
"You are now watching this page": "Je volgt nu deze pagina",
|
||||||
|
"You are no longer watching this page": "Je volgt deze pagina niet meer",
|
||||||
|
"Direct": "Direct",
|
||||||
|
"Updates": "Updates",
|
||||||
"Today": "Vandaag",
|
"Today": "Vandaag",
|
||||||
"Yesterday": "Gisteren",
|
"Yesterday": "Gisteren",
|
||||||
"This week": "Deze week",
|
"This week": "Deze week",
|
||||||
|
|||||||
@@ -341,6 +341,7 @@
|
|||||||
"Insert horizontal rule divider": "Insira um divisor horizontal",
|
"Insert horizontal rule divider": "Insira um divisor horizontal",
|
||||||
"Upload any image from your device.": "Envie qualquer imagem do seu dispositivo.",
|
"Upload any image from your device.": "Envie qualquer imagem do seu dispositivo.",
|
||||||
"Upload any video from your device.": "Envie qualquer vídeo do seu dispositivo.",
|
"Upload any video from your device.": "Envie qualquer vídeo do seu dispositivo.",
|
||||||
|
"Upload any audio from your device.": "Envie qualquer áudio do seu dispositivo.",
|
||||||
"Upload any file from your device.": "Envie qualquer arquivo do seu dispositivo.",
|
"Upload any file from your device.": "Envie qualquer arquivo do seu dispositivo.",
|
||||||
"Uploading {{name}}": "Enviando {{name}}",
|
"Uploading {{name}}": "Enviando {{name}}",
|
||||||
"Uploading file": "Enviando arquivo",
|
"Uploading file": "Enviando arquivo",
|
||||||
@@ -351,6 +352,12 @@
|
|||||||
"Divider": "Divisor",
|
"Divider": "Divisor",
|
||||||
"Quote": "Citação",
|
"Quote": "Citação",
|
||||||
"Image": "Imagem",
|
"Image": "Imagem",
|
||||||
|
"Audio": "Áudio.",
|
||||||
|
"Embed PDF": "Incorporar PDF",
|
||||||
|
"Upload and embed a PDF file.": "Envie e incorpore um arquivo PDF.",
|
||||||
|
"Embed as PDF": "Incorporar como PDF",
|
||||||
|
"Failed to load PDF": "Falha ao carregar PDF",
|
||||||
|
"Convert to attachment": "Converter em anexo",
|
||||||
"File attachment": "Anexo de arquivo",
|
"File attachment": "Anexo de arquivo",
|
||||||
"Toggle block": "Bloco colapsável",
|
"Toggle block": "Bloco colapsável",
|
||||||
"Callout": "Aviso",
|
"Callout": "Aviso",
|
||||||
@@ -442,6 +449,9 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Impedir que os membros compartilhem páginas publicamente.",
|
"Prevent members from sharing pages publicly.": "Impedir que os membros compartilhem páginas publicamente.",
|
||||||
"Toggle public sharing": "Alternar compartilhamento público",
|
"Toggle public sharing": "Alternar compartilhamento público",
|
||||||
"Toggle space public sharing": "Alternar compartilhamento público do espaço",
|
"Toggle space public sharing": "Alternar compartilhamento público do espaço",
|
||||||
|
"Allow viewers to comment": "Permitir que os visualizadores comentem",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Permitir que os visualizadores adicionem comentários em páginas deste espaço.",
|
||||||
|
"Toggle viewer comments": "Ativar/desativar comentários de visualizadores",
|
||||||
"Public sharing is disabled at the workspace level": "O compartilhamento público está desativado no nível do espaço de trabalho",
|
"Public sharing is disabled at the workspace level": "O compartilhamento público está desativado no nível do espaço de trabalho",
|
||||||
"Prevent pages in this space from being shared publicly.": "Impedir que as páginas neste espaço sejam compartilhadas publicamente.",
|
"Prevent pages in this space from being shared publicly.": "Impedir que as páginas neste espaço sejam compartilhadas publicamente.",
|
||||||
"Page permissions": "Permissões da página},{",
|
"Page permissions": "Permissões da página},{",
|
||||||
@@ -664,6 +674,24 @@
|
|||||||
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> mencionou você em uma página",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> mencionou você em uma página",
|
||||||
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> concedeu acesso de edição a uma página",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> concedeu acesso de edição a uma página",
|
||||||
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> concedeu acesso de visualização a uma página",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> concedeu acesso de visualização a uma página",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> atualizou uma página.",
|
||||||
|
"Watch page": "Observar página",
|
||||||
|
"Stop watching": "Parar de observar",
|
||||||
|
"Email notifications": "Notificações por e-mail",
|
||||||
|
"Page updates": "Atualizações da página",
|
||||||
|
"Get notified when pages you watch are updated.": "Receba notificações quando as páginas que você observa forem atualizadas.",
|
||||||
|
"Page mentions": "Menções na página",
|
||||||
|
"Get notified when someone mentions you on a page.": "Receba notificações quando alguém mencionar você em uma página.",
|
||||||
|
"Comment mentions": "Menções em comentários",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Receba notificações quando alguém mencionar você em um comentário.",
|
||||||
|
"New comments": "Novos comentários",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Receba notificações sobre novos comentários nas discussões em que você participa.",
|
||||||
|
"Resolved comments": "Comentários resolvidos",
|
||||||
|
"Get notified when your comment is resolved.": "Receba notificações quando seu comentário for resolvido.",
|
||||||
|
"You are now watching this page": "Agora você está observando esta página",
|
||||||
|
"You are no longer watching this page": "Você não está mais observando esta página",
|
||||||
|
"Direct": "Direto",
|
||||||
|
"Updates": "Atualizações",
|
||||||
"Today": "Hoje",
|
"Today": "Hoje",
|
||||||
"Yesterday": "Ontem",
|
"Yesterday": "Ontem",
|
||||||
"This week": "Esta semana",
|
"This week": "Esta semana",
|
||||||
|
|||||||
@@ -341,6 +341,7 @@
|
|||||||
"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 audio from your device.": "Загрузите любой аудиофайл с вашего устройства.",
|
||||||
"Upload any file from your device.": "Загрузить любой файл с вашего устройства.",
|
"Upload any file from your device.": "Загрузить любой файл с вашего устройства.",
|
||||||
"Uploading {{name}}": "Загрузка {{name}}",
|
"Uploading {{name}}": "Загрузка {{name}}",
|
||||||
"Uploading file": "Загрузка файла",
|
"Uploading file": "Загрузка файла",
|
||||||
@@ -351,6 +352,12 @@
|
|||||||
"Divider": "Разделитель",
|
"Divider": "Разделитель",
|
||||||
"Quote": "Цитата",
|
"Quote": "Цитата",
|
||||||
"Image": "Изображение",
|
"Image": "Изображение",
|
||||||
|
"Audio": "Аудио.",
|
||||||
|
"Embed PDF": "Встроить PDF",
|
||||||
|
"Upload and embed a PDF file.": "Загрузите и встроите PDF-файл.",
|
||||||
|
"Embed as PDF": "Встроить как PDF",
|
||||||
|
"Failed to load PDF": "Не удалось загрузить PDF",
|
||||||
|
"Convert to attachment": "Преобразовать в вложение",
|
||||||
"File attachment": "Прикрепленный файл",
|
"File attachment": "Прикрепленный файл",
|
||||||
"Toggle block": "Сворачиваемый блок",
|
"Toggle block": "Сворачиваемый блок",
|
||||||
"Callout": "Выноска",
|
"Callout": "Выноска",
|
||||||
@@ -442,6 +449,9 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Запретить участникам делиться страницами публично.",
|
"Prevent members from sharing pages publicly.": "Запретить участникам делиться страницами публично.",
|
||||||
"Toggle public sharing": "Переключить общий доступ",
|
"Toggle public sharing": "Переключить общий доступ",
|
||||||
"Toggle space public sharing": "Переключить общий доступ для пространства",
|
"Toggle space public sharing": "Переключить общий доступ для пространства",
|
||||||
|
"Allow viewers to comment": "Разрешить зрителям комментировать",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Разрешить зрителям добавлять комментарии на страницах в этом пространстве.",
|
||||||
|
"Toggle viewer comments": "Переключить комментарии зрителей",
|
||||||
"Public sharing is disabled at the workspace level": "Общий доступ отключен на уровне рабочего пространства",
|
"Public sharing is disabled at the workspace level": "Общий доступ отключен на уровне рабочего пространства",
|
||||||
"Prevent pages in this space from being shared publicly.": "Запретить делиться страницами в этом пространстве публично.",
|
"Prevent pages in this space from being shared publicly.": "Запретить делиться страницами в этом пространстве публично.",
|
||||||
"Page permissions": "Права доступа к странице},{",
|
"Page permissions": "Права доступа к странице},{",
|
||||||
@@ -664,6 +674,24 @@
|
|||||||
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> упомянул вас на странице",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> упомянул вас на странице",
|
||||||
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> предоставил вам доступ для редактирования страницы",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> предоставил вам доступ для редактирования страницы",
|
||||||
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> предоставил вам доступ к просмотру страницы",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> предоставил вам доступ к просмотру страницы",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> обновил страницу.",
|
||||||
|
"Watch page": "Следить за страницей",
|
||||||
|
"Stop watching": "Прекратить отслеживание",
|
||||||
|
"Email notifications": "Уведомления на email",
|
||||||
|
"Page updates": "Обновления страницы",
|
||||||
|
"Get notified when pages you watch are updated.": "Получайте уведомления, когда отслеживаемые вами страницы обновляются.",
|
||||||
|
"Page mentions": "Упоминания на странице",
|
||||||
|
"Get notified when someone mentions you on a page.": "Получайте уведомления, когда кто-то упоминает вас на странице.",
|
||||||
|
"Comment mentions": "Упоминания в комментариях",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Получайте уведомления, когда кто-то упоминает вас в комментарии.",
|
||||||
|
"New comments": "Новые комментарии",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Получайте уведомления о новых комментариях в цепочках, в которых вы участвуете.",
|
||||||
|
"Resolved comments": "Разрешённые комментарии",
|
||||||
|
"Get notified when your comment is resolved.": "Получайте уведомление, когда ваш комментарий разрешён.",
|
||||||
|
"You are now watching this page": "Вы теперь следите за этой страницей",
|
||||||
|
"You are no longer watching this page": "Вы больше не следите за этой страницей",
|
||||||
|
"Direct": "Прямые",
|
||||||
|
"Updates": "Обновления",
|
||||||
"Today": "Сегодня",
|
"Today": "Сегодня",
|
||||||
"Yesterday": "Вчера",
|
"Yesterday": "Вчера",
|
||||||
"This week": "На этой неделе",
|
"This week": "На этой неделе",
|
||||||
|
|||||||
@@ -341,6 +341,7 @@
|
|||||||
"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 audio from your device.": "Завантажте будь-який аудіофайл зі свого пристрою.",
|
||||||
"Upload any file from your device.": "Завантажити будь-який файл з вашого пристрою.",
|
"Upload any file from your device.": "Завантажити будь-який файл з вашого пристрою.",
|
||||||
"Uploading {{name}}": "Завантаження {{name}}",
|
"Uploading {{name}}": "Завантаження {{name}}",
|
||||||
"Uploading file": "Завантаження файлу",
|
"Uploading file": "Завантаження файлу",
|
||||||
@@ -351,6 +352,12 @@
|
|||||||
"Divider": "Роздільник",
|
"Divider": "Роздільник",
|
||||||
"Quote": "Цитата",
|
"Quote": "Цитата",
|
||||||
"Image": "Зображення",
|
"Image": "Зображення",
|
||||||
|
"Audio": "Аудіо.",
|
||||||
|
"Embed PDF": "Вбудувати PDF",
|
||||||
|
"Upload and embed a PDF file.": "Завантажте та вбудуйте файл PDF.",
|
||||||
|
"Embed as PDF": "Вбудувати як PDF",
|
||||||
|
"Failed to load PDF": "Не вдалося завантажити PDF",
|
||||||
|
"Convert to attachment": "Перетворити на вкладення",
|
||||||
"File attachment": "Прикріплений файл",
|
"File attachment": "Прикріплений файл",
|
||||||
"Toggle block": "Блок, що згортається",
|
"Toggle block": "Блок, що згортається",
|
||||||
"Callout": "Виноска",
|
"Callout": "Виноска",
|
||||||
@@ -442,6 +449,9 @@
|
|||||||
"Prevent members from sharing pages publicly.": "Перешкодити учасникам публічно ділитися сторінками.",
|
"Prevent members from sharing pages publicly.": "Перешкодити учасникам публічно ділитися сторінками.",
|
||||||
"Toggle public sharing": "Перемикання публічного доступу",
|
"Toggle public sharing": "Перемикання публічного доступу",
|
||||||
"Toggle space public sharing": "Перемикання публічного доступу до просторів",
|
"Toggle space public sharing": "Перемикання публічного доступу до просторів",
|
||||||
|
"Allow viewers to comment": "Дозволити глядачам коментувати",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "Дозволити глядачам додавати коментарі на сторінках у цьому просторі.",
|
||||||
|
"Toggle viewer comments": "Увімкнути або вимкнути коментарі глядачів",
|
||||||
"Public sharing is disabled at the workspace level": "Публічний доступ вимкнуто на рівні робочого простору",
|
"Public sharing is disabled at the workspace level": "Публічний доступ вимкнуто на рівні робочого простору",
|
||||||
"Prevent pages in this space from being shared publicly.": "Перешкодити публічному поширенню сторінок у цьому просторі.",
|
"Prevent pages in this space from being shared publicly.": "Перешкодити публічному поширенню сторінок у цьому просторі.",
|
||||||
"Page permissions": "Права доступу до сторінки.",
|
"Page permissions": "Права доступу до сторінки.",
|
||||||
@@ -664,6 +674,24 @@
|
|||||||
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> згадав вас на сторінці",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold> згадав вас на сторінці",
|
||||||
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> надав вам доступ до редагування сторінки",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold> надав вам доступ до редагування сторінки",
|
||||||
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> надав вам доступ до перегляду сторінки",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold> надав вам доступ до перегляду сторінки",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold> оновив сторінку.",
|
||||||
|
"Watch page": "Стежити за сторінкою",
|
||||||
|
"Stop watching": "Припинити стежити",
|
||||||
|
"Email notifications": "Сповіщення електронною поштою",
|
||||||
|
"Page updates": "Оновлення сторінки",
|
||||||
|
"Get notified when pages you watch are updated.": "Отримуйте сповіщення, коли сторінки, за якими ви стежите, оновлюються.",
|
||||||
|
"Page mentions": "Згадки на сторінці",
|
||||||
|
"Get notified when someone mentions you on a page.": "Отримуйте сповіщення, коли хтось згадує вас на сторінці.",
|
||||||
|
"Comment mentions": "Згадки у коментарях",
|
||||||
|
"Get notified when someone mentions you in a comment.": "Отримуйте сповіщення, коли хтось згадує вас у коментарі.",
|
||||||
|
"New comments": "Нові коментарі",
|
||||||
|
"Get notified about new comments on threads you participate in.": "Отримуйте сповіщення про нові коментарі у темах, у яких ви берете участь.",
|
||||||
|
"Resolved comments": "Вирішені коментарі",
|
||||||
|
"Get notified when your comment is resolved.": "Отримайте сповіщення, коли ваш коментар вирішено.",
|
||||||
|
"You are now watching this page": "Ви зараз стежите за цією сторінкою",
|
||||||
|
"You are no longer watching this page": "Ви більше не стежите за цією сторінкою",
|
||||||
|
"Direct": "Прямі",
|
||||||
|
"Updates": "Оновлення",
|
||||||
"Today": "Сьогодні",
|
"Today": "Сьогодні",
|
||||||
"Yesterday": "Вчора",
|
"Yesterday": "Вчора",
|
||||||
"This week": "Цього тижня",
|
"This week": "Цього тижня",
|
||||||
|
|||||||
@@ -341,6 +341,7 @@
|
|||||||
"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 audio from your device.": "从您的设备上传任意音频文件。",
|
||||||
"Upload any file from your device.": "从设备上传任何文件",
|
"Upload any file from your device.": "从设备上传任何文件",
|
||||||
"Uploading {{name}}": "正在上传{{name}}",
|
"Uploading {{name}}": "正在上传{{name}}",
|
||||||
"Uploading file": "正在上传文件",
|
"Uploading file": "正在上传文件",
|
||||||
@@ -351,6 +352,12 @@
|
|||||||
"Divider": "分割线",
|
"Divider": "分割线",
|
||||||
"Quote": "引用",
|
"Quote": "引用",
|
||||||
"Image": "图像",
|
"Image": "图像",
|
||||||
|
"Audio": "音频。",
|
||||||
|
"Embed PDF": "嵌入 PDF",
|
||||||
|
"Upload and embed a PDF file.": "上传并嵌入 PDF 文件。",
|
||||||
|
"Embed as PDF": "作为 PDF 嵌入",
|
||||||
|
"Failed to load PDF": "加载 PDF 失败",
|
||||||
|
"Convert to attachment": "转换为附件",
|
||||||
"File attachment": "文件附件",
|
"File attachment": "文件附件",
|
||||||
"Toggle block": "切换块",
|
"Toggle block": "切换块",
|
||||||
"Callout": "标注块",
|
"Callout": "标注块",
|
||||||
@@ -442,6 +449,9 @@
|
|||||||
"Prevent members from sharing pages publicly.": "阻止成员公开分享页面。",
|
"Prevent members from sharing pages publicly.": "阻止成员公开分享页面。",
|
||||||
"Toggle public sharing": "切换公开分享",
|
"Toggle public sharing": "切换公开分享",
|
||||||
"Toggle space public sharing": "切换空间公开分享",
|
"Toggle space public sharing": "切换空间公开分享",
|
||||||
|
"Allow viewers to comment": "允许观众评论",
|
||||||
|
"Allow viewers to add comments on pages in this space.": "允许观众在此空间的页面上添加评论。",
|
||||||
|
"Toggle viewer comments": "切换观众评论",
|
||||||
"Public sharing is disabled at the workspace level": "公开分享在工作区级别被禁用",
|
"Public sharing is disabled at the workspace level": "公开分享在工作区级别被禁用",
|
||||||
"Prevent pages in this space from being shared publicly.": "阻止此空间中的页面被公开分享。",
|
"Prevent pages in this space from being shared publicly.": "阻止此空间中的页面被公开分享。",
|
||||||
"Page permissions": "页面权限},{",
|
"Page permissions": "页面权限},{",
|
||||||
@@ -664,6 +674,24 @@
|
|||||||
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold>在页面上提到你",
|
"<bold>{{name}}</bold> mentioned you on a page": "<bold>{{name}}</bold>在页面上提到你",
|
||||||
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold>授予你页面编辑权限",
|
"<bold>{{name}}</bold> gave you edit access to a page": "<bold>{{name}}</bold>授予你页面编辑权限",
|
||||||
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold>授予你页面查看权限",
|
"<bold>{{name}}</bold> gave you view access to a page": "<bold>{{name}}</bold>授予你页面查看权限",
|
||||||
|
"<bold>{{name}}</bold> updated a page": "<bold>{{name}}</bold>更新了一个页面。",
|
||||||
|
"Watch page": "关注页面",
|
||||||
|
"Stop watching": "取消关注",
|
||||||
|
"Email notifications": "邮件通知",
|
||||||
|
"Page updates": "页面更新",
|
||||||
|
"Get notified when pages you watch are updated.": "当你关注的页面有更新时收到通知。",
|
||||||
|
"Page mentions": "页面提及",
|
||||||
|
"Get notified when someone mentions you on a page.": "当有人在页面上提到你时收到通知。",
|
||||||
|
"Comment mentions": "评论提及",
|
||||||
|
"Get notified when someone mentions you in a comment.": "当有人在评论中提到你时收到通知。",
|
||||||
|
"New comments": "新评论",
|
||||||
|
"Get notified about new comments on threads you participate in.": "当你参与的讨论有新评论时收到通知。",
|
||||||
|
"Resolved comments": "已解决的评论",
|
||||||
|
"Get notified when your comment is resolved.": "当你的评论被解决时收到通知。",
|
||||||
|
"You are now watching this page": "你现在正在关注此页面",
|
||||||
|
"You are no longer watching this page": "你已取消关注此页面",
|
||||||
|
"Direct": "直接",
|
||||||
|
"Updates": "更新",
|
||||||
"Today": "今天",
|
"Today": "今天",
|
||||||
"Yesterday": "昨天",
|
"Yesterday": "昨天",
|
||||||
"This week": "本周",
|
"This week": "本周",
|
||||||
|
|||||||
@@ -16,4 +16,5 @@ export const Feature = {
|
|||||||
AUDIT_LOGS: 'audit:logs',
|
AUDIT_LOGS: 'audit:logs',
|
||||||
RETENTION: 'retention',
|
RETENTION: 'retention',
|
||||||
SHARING_CONTROLS: 'sharing:controls',
|
SHARING_CONTROLS: 'sharing:controls',
|
||||||
|
VIEWER_COMMENTS: 'comment:viewer',
|
||||||
} as const;
|
} as const;
|
||||||
|
|||||||
@@ -0,0 +1,61 @@
|
|||||||
|
import { Group, Text, Switch, Tooltip } from "@mantine/core";
|
||||||
|
import React, { useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { ISpace } from "@/features/space/types/space.types.ts";
|
||||||
|
import { useUpdateSpaceMutation } from "@/features/space/queries/space-query.ts";
|
||||||
|
import { useHasFeature } from "@/ee/hooks/use-feature.ts";
|
||||||
|
import { Feature } from "@/ee/features.ts";
|
||||||
|
import { useUpgradeLabel } from "@/ee/hooks/use-upgrade-label.ts";
|
||||||
|
|
||||||
|
type SpaceViewerCommentsToggleProps = {
|
||||||
|
space: ISpace;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function SpaceViewerCommentsToggle({
|
||||||
|
space,
|
||||||
|
}: SpaceViewerCommentsToggleProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const hasViewerComments = useHasFeature(Feature.VIEWER_COMMENTS);
|
||||||
|
const upgradeLabel = useUpgradeLabel();
|
||||||
|
const isDisabled = !hasViewerComments;
|
||||||
|
const [checked, setChecked] = useState(
|
||||||
|
space.settings?.comments?.allowViewerComments === true,
|
||||||
|
);
|
||||||
|
const updateSpaceMutation = useUpdateSpaceMutation();
|
||||||
|
|
||||||
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const value = event.currentTarget.checked;
|
||||||
|
try {
|
||||||
|
await updateSpaceMutation.mutateAsync({
|
||||||
|
spaceId: space.id,
|
||||||
|
allowViewerComments: value,
|
||||||
|
});
|
||||||
|
setChecked(value);
|
||||||
|
} catch {
|
||||||
|
// error handled by mutation
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group justify="space-between" wrap="nowrap" gap="xl">
|
||||||
|
<div>
|
||||||
|
<Text size="md">{t("Allow viewers to comment")}</Text>
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{t("Allow viewers to add comments on pages in this space.")}
|
||||||
|
</Text>
|
||||||
|
</div>
|
||||||
|
<Tooltip
|
||||||
|
label={upgradeLabel}
|
||||||
|
disabled={!isDisabled}
|
||||||
|
refProp="rootRef"
|
||||||
|
>
|
||||||
|
<Switch
|
||||||
|
checked={checked}
|
||||||
|
onChange={handleChange}
|
||||||
|
disabled={isDisabled}
|
||||||
|
aria-label={t("Toggle viewer comments")}
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -3,3 +3,15 @@ import { atom } from 'jotai';
|
|||||||
export const showCommentPopupAtom = atom<boolean>(false);
|
export const showCommentPopupAtom = atom<boolean>(false);
|
||||||
export const activeCommentIdAtom = atom<string>('');
|
export const activeCommentIdAtom = atom<string>('');
|
||||||
export const draftCommentIdAtom = atom<string>('');
|
export const draftCommentIdAtom = atom<string>('');
|
||||||
|
|
||||||
|
// Read-only comment state
|
||||||
|
export const showReadOnlyCommentPopupAtom = atom<boolean>(false);
|
||||||
|
export type YjsSelection = {
|
||||||
|
anchor: any;
|
||||||
|
head: any;
|
||||||
|
};
|
||||||
|
export type ReadOnlyCommentData = {
|
||||||
|
yjsSelection: YjsSelection;
|
||||||
|
selectedText: string;
|
||||||
|
};
|
||||||
|
export const readOnlyCommentDataAtom = atom<ReadOnlyCommentData | null>(null);
|
||||||
|
|||||||
@@ -6,6 +6,8 @@ import {
|
|||||||
activeCommentIdAtom,
|
activeCommentIdAtom,
|
||||||
draftCommentIdAtom,
|
draftCommentIdAtom,
|
||||||
showCommentPopupAtom,
|
showCommentPopupAtom,
|
||||||
|
showReadOnlyCommentPopupAtom,
|
||||||
|
readOnlyCommentDataAtom,
|
||||||
} from "@/features/comment/atoms/comment-atom";
|
} from "@/features/comment/atoms/comment-atom";
|
||||||
import CommentEditor from "@/features/comment/components/comment-editor";
|
import CommentEditor from "@/features/comment/components/comment-editor";
|
||||||
import CommentActions from "@/features/comment/components/comment-actions";
|
import CommentActions from "@/features/comment/components/comment-actions";
|
||||||
@@ -19,12 +21,15 @@ import { useTranslation } from "react-i18next";
|
|||||||
interface CommentDialogProps {
|
interface CommentDialogProps {
|
||||||
editor: ReturnType<typeof useEditor>;
|
editor: ReturnType<typeof useEditor>;
|
||||||
pageId: string;
|
pageId: string;
|
||||||
|
readOnly?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
function CommentDialog({ editor, pageId, readOnly }: CommentDialogProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [comment, setComment] = useState("");
|
const [comment, setComment] = useState("");
|
||||||
const [, setShowCommentPopup] = useAtom(showCommentPopupAtom);
|
const [, setShowCommentPopup] = useAtom(showCommentPopupAtom);
|
||||||
|
const [, setShowReadOnlyCommentPopup] = useAtom(showReadOnlyCommentPopupAtom);
|
||||||
|
const [readOnlyCommentData, setReadOnlyCommentData] = useAtom(readOnlyCommentDataAtom);
|
||||||
const [, setActiveCommentId] = useAtom(activeCommentIdAtom);
|
const [, setActiveCommentId] = useAtom(activeCommentIdAtom);
|
||||||
const [draftCommentId, setDraftCommentId] = useAtom(draftCommentIdAtom);
|
const [draftCommentId, setDraftCommentId] = useAtom(draftCommentIdAtom);
|
||||||
const [currentUser] = useAtom(currentUserAtom);
|
const [currentUser] = useAtom(currentUserAtom);
|
||||||
@@ -34,11 +39,17 @@ function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
|||||||
handleDialogClose();
|
handleDialogClose();
|
||||||
});
|
});
|
||||||
const createCommentMutation = useCreateCommentMutation();
|
const createCommentMutation = useCreateCommentMutation();
|
||||||
const { isPending } = createCommentMutation;
|
const isPending = createCommentMutation.isPending;
|
||||||
|
|
||||||
const handleDialogClose = () => {
|
const handleDialogClose = () => {
|
||||||
setShowCommentPopup(false);
|
if (readOnly) {
|
||||||
editor.chain().focus().unsetCommentDecoration().run();
|
setShowReadOnlyCommentPopup(false);
|
||||||
|
// @ts-ignore
|
||||||
|
setReadOnlyCommentData(null);
|
||||||
|
} else {
|
||||||
|
setShowCommentPopup(false);
|
||||||
|
editor.chain().focus().unsetCommentDecoration().run();
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const getSelectedText = () => {
|
const getSelectedText = () => {
|
||||||
@@ -47,6 +58,11 @@ function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const handleAddComment = async () => {
|
const handleAddComment = async () => {
|
||||||
|
if (readOnly) {
|
||||||
|
await handleAddReadOnlyComment();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const selectedText = getSelectedText();
|
const selectedText = getSelectedText();
|
||||||
const commentData = {
|
const commentData = {
|
||||||
@@ -65,7 +81,6 @@ function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
|||||||
.run();
|
.run();
|
||||||
setActiveCommentId(createdComment.id);
|
setActiveCommentId(createdComment.id);
|
||||||
|
|
||||||
//unselect text to close bubble menu
|
|
||||||
editor.commands.setTextSelection({ from: editor.view.state.selection.from, to: editor.view.state.selection.from });
|
editor.commands.setTextSelection({ from: editor.view.state.selection.from, to: editor.view.state.selection.from });
|
||||||
|
|
||||||
setAsideState({ tab: "comments", isAsideOpen: true });
|
setAsideState({ tab: "comments", isAsideOpen: true });
|
||||||
@@ -85,6 +100,33 @@ function CommentDialog({ editor, pageId }: CommentDialogProps) {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleAddReadOnlyComment = async () => {
|
||||||
|
if (!readOnlyCommentData) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const createdComment = await createCommentMutation.mutateAsync({
|
||||||
|
pageId,
|
||||||
|
content: JSON.stringify(comment),
|
||||||
|
selection: readOnlyCommentData.selectedText,
|
||||||
|
type: "inline",
|
||||||
|
yjsSelection: readOnlyCommentData.yjsSelection,
|
||||||
|
});
|
||||||
|
|
||||||
|
setActiveCommentId(createdComment.id);
|
||||||
|
setAsideState({ tab: "comments", isAsideOpen: true });
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
const selector = `div[data-comment-id="${createdComment.id}"]`;
|
||||||
|
const commentElement = document.querySelector(selector);
|
||||||
|
commentElement?.scrollIntoView({ behavior: "smooth", block: "center" });
|
||||||
|
}, 400);
|
||||||
|
} finally {
|
||||||
|
setShowReadOnlyCommentPopup(false);
|
||||||
|
// @ts-ignore
|
||||||
|
setReadOnlyCommentData(null);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const handleCommentEditorChange = (newContent: any) => {
|
const handleCommentEditorChange = (newContent: any) => {
|
||||||
setComment(newContent);
|
setComment(newContent);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -44,7 +44,9 @@ function CommentListWithTabs() {
|
|||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const { data: space } = useGetSpaceBySlugQuery(page?.space?.slug);
|
const { data: space } = useGetSpaceBySlugQuery(page?.space?.slug);
|
||||||
|
|
||||||
const canComment = page?.permissions?.canEdit ?? false;
|
const canComment =
|
||||||
|
(page?.permissions?.canEdit ?? false) ||
|
||||||
|
(space?.settings?.comments?.allowViewerComments === true);
|
||||||
|
|
||||||
// Separate active and resolved comments
|
// Separate active and resolved comments
|
||||||
const { activeComments, resolvedComments } = useMemo(() => {
|
const { activeComments, resolvedComments } = useMemo(() => {
|
||||||
@@ -153,7 +155,7 @@ function CommentListWithTabs() {
|
|||||||
)}
|
)}
|
||||||
</Paper>
|
</Paper>
|
||||||
),
|
),
|
||||||
[comments, handleAddReply, isLoading, space?.membership?.role],
|
[comments, handleAddReply, isLoading, space?.membership?.role, canComment],
|
||||||
);
|
);
|
||||||
|
|
||||||
if (isCommentsLoading) {
|
if (isCommentsLoading) {
|
||||||
|
|||||||
@@ -75,7 +75,7 @@ function CommentMenu({
|
|||||||
{isResolved ? t("Re-open comment") : t("Resolve comment")}
|
{isResolved ? t("Re-open comment") : t("Resolve comment")}
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
) : (
|
) : (
|
||||||
<Tooltip label={upgradeLabel} position="left">
|
<Tooltip label={upgradeLabel} position="left" withinPortal={false}>
|
||||||
<Menu.Item disabled leftSection={<IconCircleCheck size={14} />}>
|
<Menu.Item disabled leftSection={<IconCircleCheck size={14} />}>
|
||||||
{t("Resolve comment")}
|
{t("Resolve comment")}
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
|
|||||||
@@ -65,6 +65,11 @@ export function useCreateCommentMutation() {
|
|||||||
) as InfiniteData<IPagination<IComment>> | undefined;
|
) as InfiniteData<IPagination<IComment>> | undefined;
|
||||||
|
|
||||||
if (cache && cache.pages.length > 0) {
|
if (cache && cache.pages.length > 0) {
|
||||||
|
const alreadyExists = cache.pages.some((page) =>
|
||||||
|
page.items.some((c) => c.id === newComment.id),
|
||||||
|
);
|
||||||
|
if (alreadyExists) return;
|
||||||
|
|
||||||
const lastIdx = cache.pages.length - 1;
|
const lastIdx = cache.pages.length - 1;
|
||||||
queryClient.setQueryData(RQ_KEY(newComment.pageId), {
|
queryClient.setQueryData(RQ_KEY(newComment.pageId), {
|
||||||
...cache,
|
...cache,
|
||||||
|
|||||||
@@ -17,6 +17,10 @@ export interface IComment {
|
|||||||
deletedAt?: Date;
|
deletedAt?: Date;
|
||||||
creator: IUser;
|
creator: IUser;
|
||||||
resolvedBy?: IUser;
|
resolvedBy?: IUser;
|
||||||
|
yjsSelection?: {
|
||||||
|
anchor: any;
|
||||||
|
head: any;
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ICommentData {
|
export interface ICommentData {
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ import { useCallback } from "react";
|
|||||||
export default function AttachmentView(props: NodeViewProps) {
|
export default function AttachmentView(props: NodeViewProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { editor, node, getPos, selected } = props;
|
const { editor, node, getPos, selected } = props;
|
||||||
const { url, name, size, mime, attachmentId } = node.attrs;
|
const { url, name, size, mime, attachmentId, placeholder } = node.attrs;
|
||||||
const { hovered, ref } = useHover();
|
const { hovered, ref } = useHover();
|
||||||
|
|
||||||
const isPdf = mime === "application/pdf" || name?.toLowerCase().endsWith(".pdf");
|
const isPdf = mime === "application/pdf" || name?.toLowerCase().endsWith(".pdf");
|
||||||
@@ -49,14 +49,14 @@ export default function AttachmentView(props: NodeViewProps) {
|
|||||||
h={25}
|
h={25}
|
||||||
>
|
>
|
||||||
<Group wrap="nowrap" gap="sm" style={{ minWidth: 0, flex: 1 }}>
|
<Group wrap="nowrap" gap="sm" style={{ minWidth: 0, flex: 1 }}>
|
||||||
{url ? (
|
{!url && placeholder ? (
|
||||||
<IconPaperclip size={20} style={{ flexShrink: 0 }} />
|
|
||||||
) : (
|
|
||||||
<Loader size={20} style={{ flexShrink: 0 }} />
|
<Loader size={20} style={{ flexShrink: 0 }} />
|
||||||
|
) : (
|
||||||
|
<IconPaperclip size={20} style={{ flexShrink: 0 }} />
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Text component="span" size="md" truncate="end" style={{ minWidth: 0 }}>
|
<Text component="span" size="md" truncate="end" style={{ minWidth: 0 }}>
|
||||||
{url ? name : t("Uploading {{name}}", { name })}
|
{!url && placeholder ? t("Uploading {{name}}", { name }) : name}
|
||||||
</Text>
|
</Text>
|
||||||
|
|
||||||
<Text component="span" size="sm" c="dimmed" style={{ flexShrink: 0 }}>
|
<Text component="span" size="sm" c="dimmed" style={{ flexShrink: 0 }}>
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ export default function AudioView(props: NodeViewProps) {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<NodeViewWrapper data-drag-handle>
|
<NodeViewWrapper data-drag-handle>
|
||||||
<div className={`${classes.audioWrapper} ${!safeSrc ? classes.skeleton : ''}`}>
|
<div className={`${classes.audioWrapper} ${!safeSrc && placeholder ? classes.skeleton : ''}`}>
|
||||||
{safeSrc && (
|
{safeSrc && (
|
||||||
<audio
|
<audio
|
||||||
className={classes.audio}
|
className={classes.audio}
|
||||||
@@ -49,7 +49,7 @@ export default function AudioView(props: NodeViewProps) {
|
|||||||
<Loader size={20} pos="absolute" top={6} right={6} />
|
<Loader size={20} pos="absolute" top={6} right={6} />
|
||||||
</Group>
|
</Group>
|
||||||
)}
|
)}
|
||||||
{!safeSrc && !previewSrc && (
|
{!safeSrc && !previewSrc && placeholder && (
|
||||||
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md" h={54}>
|
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md" h={54}>
|
||||||
<Loader size={20} style={{ flexShrink: 0 }} />
|
<Loader size={20} style={{ flexShrink: 0 }} />
|
||||||
<Text component="span" size="sm" truncate="end">
|
<Text component="span" size="sm" truncate="end">
|
||||||
@@ -59,6 +59,9 @@ export default function AudioView(props: NodeViewProps) {
|
|||||||
</Text>
|
</Text>
|
||||||
</Group>
|
</Group>
|
||||||
)}
|
)}
|
||||||
|
{!safeSrc && !previewSrc && !placeholder && (
|
||||||
|
<audio className={classes.audio} controls />
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</NodeViewWrapper>
|
</NodeViewWrapper>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -0,0 +1,159 @@
|
|||||||
|
import type { Editor } from "@tiptap/react";
|
||||||
|
import { TextSelection } from "@tiptap/pm/state";
|
||||||
|
import { FC, useCallback, useEffect, useRef, useState } from "react";
|
||||||
|
import { IconMessage } from "@tabler/icons-react";
|
||||||
|
import classes from "./bubble-menu.module.css";
|
||||||
|
import { ActionIcon, Tooltip } from "@mantine/core";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import {
|
||||||
|
showReadOnlyCommentPopupAtom,
|
||||||
|
readOnlyCommentDataAtom,
|
||||||
|
} from "@/features/comment/atoms/comment-atom";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { getRelativeSelection, ySyncPluginKey } from "@tiptap/y-tiptap";
|
||||||
|
|
||||||
|
type ReadonlyBubbleMenuProps = {
|
||||||
|
editor: Editor;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ReadonlyBubbleMenu: FC<ReadonlyBubbleMenuProps> = ({ editor }) => {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [showReadOnlyCommentPopup, setShowReadOnlyCommentPopup] = useAtom(
|
||||||
|
showReadOnlyCommentPopupAtom,
|
||||||
|
);
|
||||||
|
const [, setReadOnlyCommentData] = useAtom(readOnlyCommentDataAtom);
|
||||||
|
const menuRef = useRef<HTMLDivElement>(null);
|
||||||
|
const [visible, setVisible] = useState(false);
|
||||||
|
const [position, setPosition] = useState({ top: 0, left: 0 });
|
||||||
|
const isInteractingRef = useRef(false);
|
||||||
|
|
||||||
|
const updateMenuPosition = useCallback(() => {
|
||||||
|
if (isInteractingRef.current) return;
|
||||||
|
|
||||||
|
const pmSelection = editor.state.selection;
|
||||||
|
if (!(pmSelection instanceof TextSelection) || pmSelection.empty) {
|
||||||
|
setVisible(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const selection = window.getSelection();
|
||||||
|
if (
|
||||||
|
!selection ||
|
||||||
|
selection.isCollapsed ||
|
||||||
|
selection.rangeCount === 0 ||
|
||||||
|
showReadOnlyCommentPopup
|
||||||
|
) {
|
||||||
|
setVisible(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const editorDom = editor.view.dom;
|
||||||
|
if (
|
||||||
|
!editorDom.contains(selection.anchorNode) ||
|
||||||
|
!editorDom.contains(selection.focusNode)
|
||||||
|
) {
|
||||||
|
setVisible(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const range = selection.getRangeAt(0);
|
||||||
|
const rect = range.getBoundingClientRect();
|
||||||
|
|
||||||
|
if (rect.width === 0) {
|
||||||
|
setVisible(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const editorRect = editorDom
|
||||||
|
.closest(".editor-container")
|
||||||
|
?.getBoundingClientRect();
|
||||||
|
if (!editorRect) {
|
||||||
|
setVisible(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setPosition({
|
||||||
|
top: rect.top - editorRect.top - 44,
|
||||||
|
left: rect.left - editorRect.left + rect.width / 2,
|
||||||
|
});
|
||||||
|
setVisible(true);
|
||||||
|
}, [editor, showReadOnlyCommentPopup]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const handleSelectionChange = () => {
|
||||||
|
updateMenuPosition();
|
||||||
|
};
|
||||||
|
|
||||||
|
document.addEventListener("selectionchange", handleSelectionChange);
|
||||||
|
return () => {
|
||||||
|
document.removeEventListener("selectionchange", handleSelectionChange);
|
||||||
|
};
|
||||||
|
}, [updateMenuPosition]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (showReadOnlyCommentPopup) {
|
||||||
|
setVisible(false);
|
||||||
|
}
|
||||||
|
}, [showReadOnlyCommentPopup]);
|
||||||
|
|
||||||
|
const handleCommentClick = () => {
|
||||||
|
if (!editor) return;
|
||||||
|
|
||||||
|
const view = editor.view;
|
||||||
|
const ystate = ySyncPluginKey.getState(view.state);
|
||||||
|
|
||||||
|
if (ystate?.binding) {
|
||||||
|
const selection = getRelativeSelection(ystate.binding, view.state);
|
||||||
|
const { from, to } = editor.state.selection;
|
||||||
|
const selectedText = editor.state.doc.textBetween(from, to);
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
setReadOnlyCommentData({
|
||||||
|
yjsSelection: {
|
||||||
|
anchor: selection.anchor,
|
||||||
|
head: selection.head,
|
||||||
|
},
|
||||||
|
selectedText,
|
||||||
|
});
|
||||||
|
|
||||||
|
setShowReadOnlyCommentPopup(true);
|
||||||
|
setVisible(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!visible) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
ref={menuRef}
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
top: position.top,
|
||||||
|
left: position.left,
|
||||||
|
transform: "translateX(-50%)",
|
||||||
|
zIndex: 199,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div className={classes.bubbleMenu}>
|
||||||
|
<Tooltip label={t("Comment")} withArrow withinPortal={false}>
|
||||||
|
<ActionIcon
|
||||||
|
variant="default"
|
||||||
|
size="lg"
|
||||||
|
radius="6px"
|
||||||
|
aria-label={t("Comment")}
|
||||||
|
style={{ border: "none" }}
|
||||||
|
onMouseDown={(e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
isInteractingRef.current = true;
|
||||||
|
handleCommentClick();
|
||||||
|
isInteractingRef.current = false;
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<IconMessage size={16} stroke={2} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Tooltip>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -33,7 +33,7 @@ export default function ImageView(props: NodeViewProps) {
|
|||||||
className={clsx(
|
className={clsx(
|
||||||
selected && "ProseMirror-selectednode",
|
selected && "ProseMirror-selectednode",
|
||||||
classes.imageWrapper,
|
classes.imageWrapper,
|
||||||
!src && classes.skeleton,
|
!src && placeholder && classes.skeleton,
|
||||||
alignClass,
|
alignClass,
|
||||||
)}
|
)}
|
||||||
style={{
|
style={{
|
||||||
@@ -55,7 +55,7 @@ export default function ImageView(props: NodeViewProps) {
|
|||||||
<Loader size={20} pos="absolute" bottom={6} right={6} />
|
<Loader size={20} pos="absolute" bottom={6} right={6} />
|
||||||
</Group>
|
</Group>
|
||||||
)}
|
)}
|
||||||
{!src && !previewSrc && (
|
{!src && !previewSrc && placeholder && (
|
||||||
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md">
|
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md">
|
||||||
<Loader size={20} style={{ flexShrink: 0 }} />
|
<Loader size={20} style={{ flexShrink: 0 }} />
|
||||||
<Text component="span" size="sm" truncate="end">
|
<Text component="span" size="sm" truncate="end">
|
||||||
|
|||||||
@@ -294,6 +294,7 @@ const MentionList = forwardRef<any, MentionListProps>((props, ref) => {
|
|||||||
w={popupWidth}
|
w={popupWidth}
|
||||||
scrollbars={"y"}
|
scrollbars={"y"}
|
||||||
scrollbarSize={6}
|
scrollbarSize={6}
|
||||||
|
overscrollBehavior={"contain"}
|
||||||
styles={{ content: { minWidth: 0 } }}
|
styles={{ content: { minWidth: 0 } }}
|
||||||
>
|
>
|
||||||
{renderItems?.map((item, index) => {
|
{renderItems?.map((item, index) => {
|
||||||
|
|||||||
@@ -73,15 +73,17 @@ export default function PdfView(props: NodeViewProps) {
|
|||||||
if (!src || !safeSrc) {
|
if (!src || !safeSrc) {
|
||||||
return (
|
return (
|
||||||
<NodeViewWrapper data-drag-handle>
|
<NodeViewWrapper data-drag-handle>
|
||||||
<div className={`${classes.pdfWrapper} ${classes.skeleton}`} style={{ height: 600 }}>
|
<div className={`${classes.pdfWrapper} ${placeholder ? classes.skeleton : ''}`} style={{ height: placeholder ? 600 : undefined }}>
|
||||||
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md">
|
{placeholder && (
|
||||||
<Loader size={20} style={{ flexShrink: 0 }} />
|
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md">
|
||||||
<Text component="span" size="sm" truncate="end">
|
<Loader size={20} style={{ flexShrink: 0 }} />
|
||||||
{placeholder?.name
|
<Text component="span" size="sm" truncate="end">
|
||||||
? t("Uploading {{name}}", { name: placeholder.name })
|
{placeholder?.name
|
||||||
: t("Uploading file")}
|
? t("Uploading {{name}}", { name: placeholder.name })
|
||||||
</Text>
|
: t("Uploading file")}
|
||||||
</Group>
|
</Text>
|
||||||
|
</Group>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</NodeViewWrapper>
|
</NodeViewWrapper>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -87,7 +87,13 @@ const CommandList = ({
|
|||||||
|
|
||||||
return flatItems.length > 0 ? (
|
return flatItems.length > 0 ? (
|
||||||
<Paper id="slash-command" shadow="md" p="xs" withBorder>
|
<Paper id="slash-command" shadow="md" p="xs" withBorder>
|
||||||
<ScrollArea viewportRef={viewportRef} h={350} w={270} scrollbarSize={8}>
|
<ScrollArea
|
||||||
|
viewportRef={viewportRef}
|
||||||
|
h={350}
|
||||||
|
w={270}
|
||||||
|
scrollbarSize={8}
|
||||||
|
overscrollBehavior="contain"
|
||||||
|
>
|
||||||
{Object.entries(items).map(([category, categoryItems]) => (
|
{Object.entries(items).map(([category, categoryItems]) => (
|
||||||
<div key={category}>
|
<div key={category}>
|
||||||
<Text c="dimmed" mb={4} fw={500} tt="capitalize">
|
<Text c="dimmed" mb={4} fw={500} tt="capitalize">
|
||||||
@@ -103,10 +109,7 @@ const CommandList = ({
|
|||||||
})}
|
})}
|
||||||
>
|
>
|
||||||
<Group>
|
<Group>
|
||||||
<ActionIcon
|
<ActionIcon variant="default" component="div">
|
||||||
variant="default"
|
|
||||||
component="div"
|
|
||||||
>
|
|
||||||
<item.icon size={18} />
|
<item.icon size={18} />
|
||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
|
|
||||||
|
|||||||
@@ -49,7 +49,7 @@ const renderItems = () => {
|
|||||||
getReferenceClientRect = props.clientRect;
|
getReferenceClientRect = props.clientRect;
|
||||||
|
|
||||||
popup = document.createElement("div");
|
popup = document.createElement("div");
|
||||||
popup.style.zIndex = "9999";
|
popup.style.zIndex = "199";
|
||||||
popup.style.position = "absolute";
|
popup.style.position = "absolute";
|
||||||
popup.style.top = "0";
|
popup.style.top = "0";
|
||||||
popup.style.left = "0";
|
popup.style.left = "0";
|
||||||
|
|||||||
@@ -34,7 +34,7 @@ export const TableMenu = React.memo(
|
|||||||
if (isTextSelected(editor)) return false;
|
if (isTextSelected(editor)) return false;
|
||||||
return editor.isActive("table") && !isCellSelection(state.selection);
|
return editor.isActive("table") && !isCellSelection(state.selection);
|
||||||
},
|
},
|
||||||
[editor]
|
[editor],
|
||||||
);
|
);
|
||||||
|
|
||||||
const getReferencedVirtualElement = useCallback(() => {
|
const getReferencedVirtualElement = useCallback(() => {
|
||||||
@@ -121,7 +121,11 @@ export const TableMenu = React.memo(
|
|||||||
shouldShow={shouldShow}
|
shouldShow={shouldShow}
|
||||||
>
|
>
|
||||||
<div className={classes.toolbar}>
|
<div className={classes.toolbar}>
|
||||||
<Tooltip position="top" label={t("Add left column")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Add left column")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={addColumnLeft}
|
onClick={addColumnLeft}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -132,7 +136,11 @@ export const TableMenu = React.memo(
|
|||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Add right column")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Add right column")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={addColumnRight}
|
onClick={addColumnRight}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -143,7 +151,11 @@ export const TableMenu = React.memo(
|
|||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Delete column")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Delete column")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={deleteColumn}
|
onClick={deleteColumn}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -156,7 +168,11 @@ export const TableMenu = React.memo(
|
|||||||
|
|
||||||
<div className={classes.divider} />
|
<div className={classes.divider} />
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Add row above")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Add row above")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={addRowAbove}
|
onClick={addRowAbove}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -167,7 +183,11 @@ export const TableMenu = React.memo(
|
|||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Add row below")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Add row below")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={addRowBelow}
|
onClick={addRowBelow}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -178,7 +198,7 @@ export const TableMenu = React.memo(
|
|||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Delete row")}>
|
<Tooltip position="top" label={t("Delete row")} withinPortal={false}>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={deleteRow}
|
onClick={deleteRow}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -191,7 +211,11 @@ export const TableMenu = React.memo(
|
|||||||
|
|
||||||
<div className={classes.divider} />
|
<div className={classes.divider} />
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Toggle header row")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Toggle header row")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={toggleHeaderRow}
|
onClick={toggleHeaderRow}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -202,7 +226,11 @@ export const TableMenu = React.memo(
|
|||||||
</ActionIcon>
|
</ActionIcon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Toggle header column")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Toggle header column")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={toggleHeaderColumn}
|
onClick={toggleHeaderColumn}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -215,7 +243,11 @@ export const TableMenu = React.memo(
|
|||||||
|
|
||||||
<div className={classes.divider} />
|
<div className={classes.divider} />
|
||||||
|
|
||||||
<Tooltip position="top" label={t("Delete table")}>
|
<Tooltip
|
||||||
|
position="top"
|
||||||
|
label={t("Delete table")}
|
||||||
|
withinPortal={false}
|
||||||
|
>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
onClick={deleteTable}
|
onClick={deleteTable}
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
@@ -228,7 +260,7 @@ export const TableMenu = React.memo(
|
|||||||
</div>
|
</div>
|
||||||
</BubbleMenu>
|
</BubbleMenu>
|
||||||
);
|
);
|
||||||
}
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
export default TableMenu;
|
export default TableMenu;
|
||||||
|
|||||||
@@ -33,7 +33,7 @@ export default function VideoView(props: NodeViewProps) {
|
|||||||
className={clsx(
|
className={clsx(
|
||||||
selected && "ProseMirror-selectednode",
|
selected && "ProseMirror-selectednode",
|
||||||
classes.videoWrapper,
|
classes.videoWrapper,
|
||||||
!src && classes.skeleton,
|
!src && placeholder && classes.skeleton,
|
||||||
alignClass,
|
alignClass,
|
||||||
)}
|
)}
|
||||||
style={{
|
style={{
|
||||||
@@ -60,7 +60,7 @@ export default function VideoView(props: NodeViewProps) {
|
|||||||
<Loader size={20} pos="absolute" top={6} right={6} />
|
<Loader size={20} pos="absolute" top={6} right={6} />
|
||||||
</Group>
|
</Group>
|
||||||
)}
|
)}
|
||||||
{!src && !previewSrc && (
|
{!src && !previewSrc && placeholder && (
|
||||||
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md">
|
<Group justify="center" wrap="nowrap" gap="xs" maw="100%" px="md">
|
||||||
<Loader size={20} style={{ flexShrink: 0 }} />
|
<Loader size={20} style={{ flexShrink: 0 }} />
|
||||||
<Text component="span" size="sm" truncate="end">
|
<Text component="span" size="sm" truncate="end">
|
||||||
@@ -70,6 +70,9 @@ export default function VideoView(props: NodeViewProps) {
|
|||||||
</Text>
|
</Text>
|
||||||
</Group>
|
</Group>
|
||||||
)}
|
)}
|
||||||
|
{!src && !previewSrc && !placeholder && (
|
||||||
|
<video className={classes.video} controls />
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</NodeViewWrapper>
|
</NodeViewWrapper>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -0,0 +1,105 @@
|
|||||||
|
// https://github.com/NiclasDev63/tiptap-extension-auto-joiner - MIT
|
||||||
|
import { Extension } from "@tiptap/core";
|
||||||
|
import { Plugin, PluginKey } from "@tiptap/pm/state";
|
||||||
|
import { canJoin } from "@tiptap/pm/transform";
|
||||||
|
import { getNodeType } from "@tiptap/react";
|
||||||
|
import { NodeType } from "@tiptap/pm/model";
|
||||||
|
import { Transaction } from "@tiptap/pm/state";
|
||||||
|
|
||||||
|
// https://discuss.prosemirror.net/t/how-to-autojoin-all-the-time/2957/4
|
||||||
|
// Adapted from prosemirror-commands wrapDispatchForJoin
|
||||||
|
function autoJoin(
|
||||||
|
transactions: readonly Transaction[],
|
||||||
|
newTr: Transaction,
|
||||||
|
nodeTypes: NodeType[]
|
||||||
|
) {
|
||||||
|
// Collect changed ranges across all transactions, mapping earlier ranges
|
||||||
|
// forward through later mappings so every position lands in newTr.doc space.
|
||||||
|
let ranges: number[] = [];
|
||||||
|
for (const tr of transactions) {
|
||||||
|
for (let i = 0; i < tr.mapping.maps.length; i++) {
|
||||||
|
let map = tr.mapping.maps[i];
|
||||||
|
if (!map) continue;
|
||||||
|
for (let j = 0; j < ranges.length; j++) ranges[j] = map.map(ranges[j]!);
|
||||||
|
map.forEach((_s, _e, from, to) => ranges.push(from, to));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Figure out which joinable points exist inside those ranges,
|
||||||
|
// by checking all node boundaries in their parent nodes.
|
||||||
|
// Resolve against newTr.doc — the same document we will join on.
|
||||||
|
let joinable: number[] = [];
|
||||||
|
for (let i = 0; i < ranges.length; i += 2) {
|
||||||
|
let from = ranges[i]!,
|
||||||
|
to = ranges[i + 1]!;
|
||||||
|
let $from = newTr.doc.resolve(from),
|
||||||
|
depth = $from.sharedDepth(to),
|
||||||
|
parent = $from.node(depth);
|
||||||
|
for (
|
||||||
|
let index = $from.indexAfter(depth), pos = $from.after(depth + 1);
|
||||||
|
pos <= to;
|
||||||
|
++index
|
||||||
|
) {
|
||||||
|
let after = parent.maybeChild(index);
|
||||||
|
if (!after) break;
|
||||||
|
if (index && joinable.indexOf(pos) == -1) {
|
||||||
|
let before = parent.child(index - 1);
|
||||||
|
if (before.type == after.type && nodeTypes.includes(before.type))
|
||||||
|
joinable.push(pos);
|
||||||
|
}
|
||||||
|
pos += after.nodeSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Join the joinable points (reverse order to preserve earlier positions)
|
||||||
|
let joined = false;
|
||||||
|
joinable.sort((a, b) => a - b);
|
||||||
|
for (let i = joinable.length - 1; i >= 0; i--) {
|
||||||
|
if (canJoin(newTr.doc, joinable[i]!)) {
|
||||||
|
newTr.join(joinable[i]!);
|
||||||
|
joined = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return joined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface AutoJoinerOptions {
|
||||||
|
elementsToJoin: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
const AutoJoiner = Extension.create<AutoJoinerOptions>({
|
||||||
|
name: "autoJoiner",
|
||||||
|
|
||||||
|
addOptions() {
|
||||||
|
return {
|
||||||
|
elementsToJoin: [],
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
addProseMirrorPlugins() {
|
||||||
|
const plugin = new PluginKey(this.name);
|
||||||
|
const joinableNodes = [
|
||||||
|
this.editor.schema.nodes.bulletList,
|
||||||
|
this.editor.schema.nodes.orderedList,
|
||||||
|
];
|
||||||
|
this.options.elementsToJoin.forEach((element) => {
|
||||||
|
const nodeTyp = getNodeType(element, this.editor.schema);
|
||||||
|
joinableNodes.push(nodeTyp);
|
||||||
|
});
|
||||||
|
|
||||||
|
return [
|
||||||
|
new Plugin({
|
||||||
|
key: plugin,
|
||||||
|
appendTransaction(transactions, _, newState) {
|
||||||
|
let newTr = newState.tr;
|
||||||
|
if (autoJoin(transactions, newTr, joinableNodes as NodeType[])) {
|
||||||
|
return newTr;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
];
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export default AutoJoiner;
|
||||||
@@ -49,7 +49,7 @@ import {
|
|||||||
SharedStorage,
|
SharedStorage,
|
||||||
Columns,
|
Columns,
|
||||||
Column,
|
Column,
|
||||||
Status
|
Status,
|
||||||
} from "@docmost/editor-ext";
|
} from "@docmost/editor-ext";
|
||||||
import {
|
import {
|
||||||
randomElement,
|
randomElement,
|
||||||
@@ -97,6 +97,7 @@ import i18n from "@/i18n.ts";
|
|||||||
import { MarkdownClipboard } from "@/features/editor/extensions/markdown-clipboard.ts";
|
import { MarkdownClipboard } from "@/features/editor/extensions/markdown-clipboard.ts";
|
||||||
import EmojiCommand from "./emoji-command";
|
import EmojiCommand from "./emoji-command";
|
||||||
import { countWords } from "alfaaz";
|
import { countWords } from "alfaaz";
|
||||||
|
import AutoJoiner from "@/features/editor/extensions/autojoiner.ts";
|
||||||
|
|
||||||
const lowlight = createLowlight(common);
|
const lowlight = createLowlight(common);
|
||||||
lowlight.register("mermaid", plaintext);
|
lowlight.register("mermaid", plaintext);
|
||||||
@@ -141,6 +142,25 @@ export const mainExtensions = [
|
|||||||
}),
|
}),
|
||||||
];
|
];
|
||||||
},
|
},
|
||||||
|
addKeyboardShortcuts() {
|
||||||
|
return {
|
||||||
|
Enter: ({ editor }) => {
|
||||||
|
const { from, to } = editor.state.selection;
|
||||||
|
if (from !== to) return false;
|
||||||
|
if (!editor.isActive("code")) return false;
|
||||||
|
|
||||||
|
const $from = editor.state.doc.resolve(from);
|
||||||
|
const codeType = editor.state.schema.marks.code;
|
||||||
|
const nodeAfter = $from.nodeAfter;
|
||||||
|
|
||||||
|
if (nodeAfter && codeType.isInSet(nodeAfter.marks)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return editor.chain().unsetCode().splitBlock().run();
|
||||||
|
},
|
||||||
|
};
|
||||||
|
},
|
||||||
}),
|
}),
|
||||||
SharedStorage,
|
SharedStorage,
|
||||||
Heading,
|
Heading,
|
||||||
@@ -252,8 +272,8 @@ export const mainExtensions = [
|
|||||||
resize: {
|
resize: {
|
||||||
enabled: true,
|
enabled: true,
|
||||||
directions: ["left", "right"],
|
directions: ["left", "right"],
|
||||||
minWidth: 80,
|
minWidth: 24,
|
||||||
minHeight: 40,
|
minHeight: 16,
|
||||||
alwaysPreserveAspectRatio: true,
|
alwaysPreserveAspectRatio: true,
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
createCustomHandle: createImageHandle,
|
createCustomHandle: createImageHandle,
|
||||||
@@ -265,8 +285,8 @@ export const mainExtensions = [
|
|||||||
resize: {
|
resize: {
|
||||||
enabled: true,
|
enabled: true,
|
||||||
directions: ["left", "right"],
|
directions: ["left", "right"],
|
||||||
minWidth: 80,
|
minWidth: 24,
|
||||||
minHeight: 40,
|
minHeight: 16,
|
||||||
alwaysPreserveAspectRatio: true,
|
alwaysPreserveAspectRatio: true,
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
createCustomHandle: createResizeHandle,
|
createCustomHandle: createResizeHandle,
|
||||||
@@ -296,8 +316,8 @@ export const mainExtensions = [
|
|||||||
resize: {
|
resize: {
|
||||||
enabled: true,
|
enabled: true,
|
||||||
directions: ["left", "right"],
|
directions: ["left", "right"],
|
||||||
minWidth: 80,
|
minWidth: 24,
|
||||||
minHeight: 40,
|
minHeight: 16,
|
||||||
alwaysPreserveAspectRatio: true,
|
alwaysPreserveAspectRatio: true,
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
createCustomHandle: createResizeHandle,
|
createCustomHandle: createResizeHandle,
|
||||||
@@ -309,8 +329,8 @@ export const mainExtensions = [
|
|||||||
resize: {
|
resize: {
|
||||||
enabled: true,
|
enabled: true,
|
||||||
directions: ["left", "right"],
|
directions: ["left", "right"],
|
||||||
minWidth: 80,
|
minWidth: 24,
|
||||||
minHeight: 40,
|
minHeight: 16,
|
||||||
alwaysPreserveAspectRatio: true,
|
alwaysPreserveAspectRatio: true,
|
||||||
//@ts-ignore
|
//@ts-ignore
|
||||||
createCustomHandle: createResizeHandle,
|
createCustomHandle: createResizeHandle,
|
||||||
@@ -353,6 +373,9 @@ export const mainExtensions = [
|
|||||||
}).configure(),
|
}).configure(),
|
||||||
Columns,
|
Columns,
|
||||||
Column,
|
Column,
|
||||||
|
AutoJoiner.configure({
|
||||||
|
elementsToJoin: [],
|
||||||
|
}),
|
||||||
] as any;
|
] as any;
|
||||||
|
|
||||||
type CollabExtensions = (provider: HocuspocusProvider, user: IUser) => any[];
|
type CollabExtensions = (provider: HocuspocusProvider, user: IUser) => any[];
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
// adapted from: https://github.com/aguingand/tiptap-markdown/blob/main/src/extensions/tiptap/clipboard.js - MIT
|
// adapted from: https://github.com/aguingand/tiptap-markdown/blob/main/src/extensions/tiptap/clipboard.js - MIT
|
||||||
import { Extension } from "@tiptap/core";
|
import { Extension } from "@tiptap/core";
|
||||||
import { Plugin, PluginKey } from "@tiptap/pm/state";
|
import { Plugin, PluginKey, TextSelection } from "@tiptap/pm/state";
|
||||||
import { DOMParser } from "@tiptap/pm/model";
|
import { DOMParser, DOMSerializer, Fragment, Slice } from "@tiptap/pm/model";
|
||||||
import { find } from "linkifyjs";
|
import { find } from "linkifyjs";
|
||||||
import { markdownToHtml } from "@docmost/editor-ext";
|
import { markdownToHtml, htmlToMarkdown } from "@docmost/editor-ext";
|
||||||
|
|
||||||
export const MarkdownClipboard = Extension.create({
|
export const MarkdownClipboard = Extension.create({
|
||||||
name: "markdownClipboard",
|
name: "markdownClipboard",
|
||||||
@@ -19,6 +19,27 @@ export const MarkdownClipboard = Extension.create({
|
|||||||
new Plugin({
|
new Plugin({
|
||||||
key: new PluginKey("markdownClipboard"),
|
key: new PluginKey("markdownClipboard"),
|
||||||
props: {
|
props: {
|
||||||
|
clipboardTextSerializer: (slice) => {
|
||||||
|
const listTypes = ["bulletList", "orderedList", "taskList"];
|
||||||
|
let topLevelCount = 0;
|
||||||
|
let hasList = false;
|
||||||
|
slice.content.forEach((node) => {
|
||||||
|
if (listTypes.includes(node.type.name)) {
|
||||||
|
hasList = true;
|
||||||
|
topLevelCount += node.childCount;
|
||||||
|
} else {
|
||||||
|
topLevelCount++;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!hasList || topLevelCount < 2) return null;
|
||||||
|
|
||||||
|
const div = document.createElement("div");
|
||||||
|
const serializer = DOMSerializer.fromSchema(this.editor.schema);
|
||||||
|
const fragment = serializer.serializeFragment(slice.content);
|
||||||
|
div.appendChild(fragment);
|
||||||
|
return htmlToMarkdown(div.innerHTML);
|
||||||
|
},
|
||||||
handlePaste: (view, event, slice) => {
|
handlePaste: (view, event, slice) => {
|
||||||
if (!event.clipboardData) {
|
if (!event.clipboardData) {
|
||||||
return false;
|
return false;
|
||||||
@@ -29,49 +50,80 @@ export const MarkdownClipboard = Extension.create({
|
|||||||
}
|
}
|
||||||
|
|
||||||
const text = event.clipboardData.getData("text/plain");
|
const text = event.clipboardData.getData("text/plain");
|
||||||
|
const html = event.clipboardData.getData("text/html");
|
||||||
const vscode = event.clipboardData.getData("vscode-editor-data");
|
const vscode = event.clipboardData.getData("vscode-editor-data");
|
||||||
const vscodeData = vscode ? JSON.parse(vscode) : undefined;
|
const vscodeData = vscode ? JSON.parse(vscode) : undefined;
|
||||||
const language = vscodeData?.mode;
|
const language = vscodeData?.mode;
|
||||||
|
|
||||||
if (language !== "markdown") {
|
const isVscodeMarkdown = language === "markdown";
|
||||||
|
const isPlainTextOnly = !html && !vscode && !!text;
|
||||||
|
|
||||||
|
if (!isVscodeMarkdown && !isPlainTextOnly) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (isPlainTextOnly) {
|
||||||
|
if ((view as any).input?.shiftKey || !this.options.transformPastedText) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const link = find(text, {
|
||||||
|
defaultProtocol: "http",
|
||||||
|
}).find((item) => item.isLink && item.value === text);
|
||||||
|
|
||||||
|
if (link) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const { tr } = view.state;
|
const { tr } = view.state;
|
||||||
const { from, to } = view.state.selection;
|
const { from, to } = view.state.selection;
|
||||||
|
|
||||||
const html = markdownToHtml(text);
|
const parsed = markdownToHtml(text.replace(/\n+$/, ""));
|
||||||
|
|
||||||
const contentNodes = DOMParser.fromSchema(
|
const contentNodes = DOMParser.fromSchema(
|
||||||
this.editor.schema,
|
this.editor.schema,
|
||||||
).parseSlice(elementFromString(html), {
|
).parseSlice(elementFromString(parsed), {
|
||||||
preserveWhitespace: true,
|
preserveWhitespace: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
tr.replaceRange(from, to, contentNodes);
|
tr.replaceRange(from, to, contentNodes);
|
||||||
|
const insertEnd = tr.mapping.map(from, 1);
|
||||||
|
tr.setSelection(TextSelection.near(tr.doc.resolve(Math.max(from, insertEnd - 2)), -1));
|
||||||
tr.setMeta('paste', true)
|
tr.setMeta('paste', true)
|
||||||
view.dispatch(tr);
|
view.dispatch(tr);
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
clipboardTextParser: (text, context, plainText) => {
|
// Strip trailing whitespace-only paragraphs from pasted content.
|
||||||
const link = find(text, {
|
// Terminals (GNOME Terminal, etc.) often include trailing
|
||||||
defaultProtocol: "http",
|
// whitespace in their HTML clipboard data, which ProseMirror
|
||||||
}).find((item) => item.isLink && item.value === text);
|
// parses as an extra paragraph. Inside a list item this creates
|
||||||
|
// an orphan empty line that breaks the list structure.
|
||||||
|
transformPasted: (slice) => {
|
||||||
|
let { content, openStart, openEnd } = slice;
|
||||||
|
|
||||||
if (plainText || !this.options.transformPastedText || link) {
|
// Remove trailing paragraphs that contain only whitespace
|
||||||
// don't parse plaintext link to allow link paste handler to work
|
while (content.childCount > 1) {
|
||||||
// pasting with shift key prevents formatting
|
const lastChild = content.lastChild;
|
||||||
return null;
|
if (
|
||||||
|
lastChild?.type.name === "paragraph" &&
|
||||||
|
lastChild.textContent.trim() === ""
|
||||||
|
) {
|
||||||
|
const children = [];
|
||||||
|
for (let i = 0; i < content.childCount - 1; i++) {
|
||||||
|
children.push(content.child(i));
|
||||||
|
}
|
||||||
|
content = Fragment.from(children);
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const parsed = markdownToHtml(text);
|
if (content !== slice.content) {
|
||||||
return DOMParser.fromSchema(this.editor.schema).parseSlice(
|
return new Slice(content, openStart, Math.max(openEnd, 1));
|
||||||
elementFromString(parsed),
|
}
|
||||||
{
|
|
||||||
preserveWhitespace: true,
|
return slice;
|
||||||
context,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ export interface FullEditorProps {
|
|||||||
content: string;
|
content: string;
|
||||||
spaceSlug: string;
|
spaceSlug: string;
|
||||||
editable: boolean;
|
editable: boolean;
|
||||||
|
canComment?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function FullEditor({
|
export function FullEditor({
|
||||||
@@ -25,6 +26,7 @@ export function FullEditor({
|
|||||||
content,
|
content,
|
||||||
spaceSlug,
|
spaceSlug,
|
||||||
editable,
|
editable,
|
||||||
|
canComment,
|
||||||
}: FullEditorProps) {
|
}: FullEditorProps) {
|
||||||
const [user] = useAtom(userAtom);
|
const [user] = useAtom(userAtom);
|
||||||
const fullPageWidth = user.settings?.preferences?.fullPageWidth;
|
const fullPageWidth = user.settings?.preferences?.fullPageWidth;
|
||||||
@@ -46,6 +48,7 @@ export function FullEditor({
|
|||||||
pageId={pageId}
|
pageId={pageId}
|
||||||
editable={editable}
|
editable={editable}
|
||||||
content={content}
|
content={content}
|
||||||
|
canComment={canComment}
|
||||||
/>
|
/>
|
||||||
</Container>
|
</Container>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -37,9 +37,11 @@ import { asideStateAtom } from "@/components/layouts/global/hooks/atoms/sidebar-
|
|||||||
import {
|
import {
|
||||||
activeCommentIdAtom,
|
activeCommentIdAtom,
|
||||||
showCommentPopupAtom,
|
showCommentPopupAtom,
|
||||||
|
showReadOnlyCommentPopupAtom,
|
||||||
} from "@/features/comment/atoms/comment-atom";
|
} from "@/features/comment/atoms/comment-atom";
|
||||||
import CommentDialog from "@/features/comment/components/comment-dialog";
|
import CommentDialog from "@/features/comment/components/comment-dialog";
|
||||||
import { EditorBubbleMenu } from "@/features/editor/components/bubble-menu/bubble-menu";
|
import { EditorBubbleMenu } from "@/features/editor/components/bubble-menu/bubble-menu";
|
||||||
|
import { ReadonlyBubbleMenu } from "@/features/editor/components/bubble-menu/readonly-bubble-menu";
|
||||||
import TableCellMenu from "@/features/editor/components/table/table-cell-menu.tsx";
|
import TableCellMenu from "@/features/editor/components/table/table-cell-menu.tsx";
|
||||||
import TableMenu from "@/features/editor/components/table/table-menu.tsx";
|
import TableMenu from "@/features/editor/components/table/table-menu.tsx";
|
||||||
import ImageMenu from "@/features/editor/components/image/image-menu.tsx";
|
import ImageMenu from "@/features/editor/components/image/image-menu.tsx";
|
||||||
@@ -74,12 +76,14 @@ interface PageEditorProps {
|
|||||||
pageId: string;
|
pageId: string;
|
||||||
editable: boolean;
|
editable: boolean;
|
||||||
content: any;
|
content: any;
|
||||||
|
canComment?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function PageEditor({
|
export default function PageEditor({
|
||||||
pageId,
|
pageId,
|
||||||
editable,
|
editable,
|
||||||
content,
|
content,
|
||||||
|
canComment,
|
||||||
}: PageEditorProps) {
|
}: PageEditorProps) {
|
||||||
const collaborationURL = useCollaborationUrl();
|
const collaborationURL = useCollaborationUrl();
|
||||||
const isComponentMounted = useRef(false);
|
const isComponentMounted = useRef(false);
|
||||||
@@ -94,6 +98,7 @@ export default function PageEditor({
|
|||||||
const [, setAsideState] = useAtom(asideStateAtom);
|
const [, setAsideState] = useAtom(asideStateAtom);
|
||||||
const [, setActiveCommentId] = useAtom(activeCommentIdAtom);
|
const [, setActiveCommentId] = useAtom(activeCommentIdAtom);
|
||||||
const [showCommentPopup, setShowCommentPopup] = useAtom(showCommentPopupAtom);
|
const [showCommentPopup, setShowCommentPopup] = useAtom(showCommentPopupAtom);
|
||||||
|
const [showReadOnlyCommentPopup] = useAtom(showReadOnlyCommentPopupAtom);
|
||||||
const [isLocalSynced, setIsLocalSynced] = useState(false);
|
const [isLocalSynced, setIsLocalSynced] = useState(false);
|
||||||
const [isRemoteSynced, setIsRemoteSynced] = useState(false);
|
const [isRemoteSynced, setIsRemoteSynced] = useState(false);
|
||||||
const [yjsConnectionStatus, setYjsConnectionStatus] = useAtom(
|
const [yjsConnectionStatus, setYjsConnectionStatus] = useAtom(
|
||||||
@@ -423,7 +428,13 @@ export default function PageEditor({
|
|||||||
<ColumnsMenu editor={editor} />
|
<ColumnsMenu editor={editor} />
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
{editor && !editorIsEditable && (editable || canComment) && providersRef.current && (
|
||||||
|
<ReadonlyBubbleMenu editor={editor} />
|
||||||
|
)}
|
||||||
{showCommentPopup && <CommentDialog editor={editor} pageId={pageId} />}
|
{showCommentPopup && <CommentDialog editor={editor} pageId={pageId} />}
|
||||||
|
{showReadOnlyCommentPopup && (
|
||||||
|
<CommentDialog editor={editor} pageId={pageId} readOnly />
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
onClick={() => editor.commands.focus("end")}
|
onClick={() => editor.commands.focus("end")}
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ import {
|
|||||||
import { CustomAvatar } from "@/components/ui/custom-avatar";
|
import { CustomAvatar } from "@/components/ui/custom-avatar";
|
||||||
import { INotification } from "../types/notification.types";
|
import { INotification } from "../types/notification.types";
|
||||||
import { Trans, useTranslation } from "react-i18next";
|
import { Trans, useTranslation } from "react-i18next";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { Link } from "react-router-dom";
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
import { useMarkReadMutation } from "../queries/notification-query";
|
import { useMarkReadMutation } from "../queries/notification-query";
|
||||||
import { buildPageUrl } from "@/features/page/page.utils";
|
import { buildPageUrl } from "@/features/page/page.utils";
|
||||||
@@ -30,7 +30,6 @@ export function NotificationItem({
|
|||||||
onNavigate,
|
onNavigate,
|
||||||
}: NotificationItemProps) {
|
}: NotificationItemProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const navigate = useNavigate();
|
|
||||||
const markRead = useMarkReadMutation();
|
const markRead = useMarkReadMutation();
|
||||||
const [hovered, setHovered] = useState(false);
|
const [hovered, setHovered] = useState(false);
|
||||||
|
|
||||||
@@ -50,37 +49,47 @@ export function NotificationItem({
|
|||||||
return notification.data?.role === "writer"
|
return notification.data?.role === "writer"
|
||||||
? "<bold>{{name}}</bold> gave you edit access to a page"
|
? "<bold>{{name}}</bold> gave you edit access to a page"
|
||||||
: "<bold>{{name}}</bold> gave you view access to a page";
|
: "<bold>{{name}}</bold> gave you view access to a page";
|
||||||
|
case "page.updated":
|
||||||
|
return "<bold>{{name}}</bold> updated a page";
|
||||||
default:
|
default:
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleClick = () => {
|
const pageUrl =
|
||||||
if (notification.page && notification.space) {
|
notification.page && notification.space
|
||||||
if (isUnread) {
|
? buildPageUrl(
|
||||||
markRead.mutate([notification.id]);
|
|
||||||
}
|
|
||||||
navigate(
|
|
||||||
buildPageUrl(
|
|
||||||
notification.space.slug,
|
notification.space.slug,
|
||||||
notification.page.slugId,
|
notification.page.slugId,
|
||||||
notification.page.title,
|
notification.page.title,
|
||||||
),
|
)
|
||||||
);
|
: undefined;
|
||||||
onNavigate();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleMarkRead = (e: React.MouseEvent) => {
|
const markReadIfNeeded = () => {
|
||||||
e.stopPropagation();
|
|
||||||
if (isUnread) {
|
if (isUnread) {
|
||||||
markRead.mutate([notification.id]);
|
markRead.mutate([notification.id]);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleClick = () => {
|
||||||
|
markReadIfNeeded();
|
||||||
|
onNavigate();
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleMarkRead = (e: React.MouseEvent) => {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
markReadIfNeeded();
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<UnstyledButton
|
<UnstyledButton
|
||||||
|
component={Link}
|
||||||
|
to={pageUrl ?? ""}
|
||||||
onClick={handleClick}
|
onClick={handleClick}
|
||||||
|
// auxclick fires for all non-primary buttons; guard to middle-click only (button 1)
|
||||||
|
// so that right-click (button 2, context menu) does not mark as read
|
||||||
|
onAuxClick={(e: React.MouseEvent) => e.button === 1 && markReadIfNeeded()}
|
||||||
onMouseEnter={() => setHovered(true)}
|
onMouseEnter={() => setHovered(true)}
|
||||||
onMouseLeave={() => setHovered(false)}
|
onMouseLeave={() => setHovered(false)}
|
||||||
w="100%"
|
w="100%"
|
||||||
|
|||||||
@@ -3,17 +3,23 @@ import { IconBellOff } from "@tabler/icons-react";
|
|||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { useEffect, useRef } from "react";
|
import { useEffect, useRef } from "react";
|
||||||
import { NotificationItem } from "./notification-item";
|
import { NotificationItem } from "./notification-item";
|
||||||
import { INotification, NotificationFilter } from "../types/notification.types";
|
import {
|
||||||
|
INotification,
|
||||||
|
NotificationFilter,
|
||||||
|
NotificationTab,
|
||||||
|
} from "../types/notification.types";
|
||||||
import { groupNotificationsByTime } from "../notification.utils";
|
import { groupNotificationsByTime } from "../notification.utils";
|
||||||
import { useNotificationsQuery } from "../queries/notification-query";
|
import { useNotificationsQuery } from "../queries/notification-query";
|
||||||
import classes from "../notification.module.css";
|
import classes from "../notification.module.css";
|
||||||
|
|
||||||
type NotificationListProps = {
|
type NotificationListProps = {
|
||||||
|
tab: NotificationTab;
|
||||||
filter: NotificationFilter;
|
filter: NotificationFilter;
|
||||||
onNavigate: () => void;
|
onNavigate: () => void;
|
||||||
};
|
};
|
||||||
|
|
||||||
export function NotificationList({
|
export function NotificationList({
|
||||||
|
tab,
|
||||||
filter,
|
filter,
|
||||||
onNavigate,
|
onNavigate,
|
||||||
}: NotificationListProps) {
|
}: NotificationListProps) {
|
||||||
@@ -24,7 +30,7 @@ export function NotificationList({
|
|||||||
hasNextPage,
|
hasNextPage,
|
||||||
fetchNextPage,
|
fetchNextPage,
|
||||||
isFetchingNextPage,
|
isFetchingNextPage,
|
||||||
} = useNotificationsQuery();
|
} = useNotificationsQuery(tab as string);
|
||||||
|
|
||||||
const sentinelRef = useRef<HTMLDivElement>(null);
|
const sentinelRef = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import {
|
|||||||
Menu,
|
Menu,
|
||||||
Popover,
|
Popover,
|
||||||
ScrollArea,
|
ScrollArea,
|
||||||
|
Tabs,
|
||||||
Text,
|
Text,
|
||||||
Tooltip,
|
Tooltip,
|
||||||
} from "@mantine/core";
|
} from "@mantine/core";
|
||||||
@@ -18,15 +19,20 @@ import {
|
|||||||
} from "@tabler/icons-react";
|
} from "@tabler/icons-react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { NotificationList } from "./notification-list";
|
import { NotificationList } from "./notification-list";
|
||||||
import { NotificationFilter } from "../types/notification.types";
|
import {
|
||||||
|
NotificationFilter,
|
||||||
|
NotificationTab,
|
||||||
|
} from "../types/notification.types";
|
||||||
import {
|
import {
|
||||||
useMarkAllReadMutation,
|
useMarkAllReadMutation,
|
||||||
useUnreadCountQuery,
|
useUnreadCountQuery,
|
||||||
} from "../queries/notification-query";
|
} from "../queries/notification-query";
|
||||||
|
import classes from "../notification.module.css";
|
||||||
|
|
||||||
export function NotificationPopover() {
|
export function NotificationPopover() {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [opened, setOpened] = useState(false);
|
const [opened, setOpened] = useState(false);
|
||||||
|
const [tab, setTab] = useState<NotificationTab>("direct");
|
||||||
const [filter, setFilter] = useState<NotificationFilter>("all");
|
const [filter, setFilter] = useState<NotificationFilter>("all");
|
||||||
|
|
||||||
const { data: unreadData } = useUnreadCountQuery();
|
const { data: unreadData } = useUnreadCountQuery();
|
||||||
@@ -125,13 +131,27 @@ export function NotificationPopover() {
|
|||||||
</Group>
|
</Group>
|
||||||
</Group>
|
</Group>
|
||||||
|
|
||||||
|
<Tabs
|
||||||
|
value={tab}
|
||||||
|
onChange={(value) => setTab(value as NotificationTab)}
|
||||||
|
variant="default"
|
||||||
|
color="dark"
|
||||||
|
>
|
||||||
|
<Tabs.List px="md">
|
||||||
|
<Tabs.Tab value="direct">{t("Direct")}</Tabs.Tab>
|
||||||
|
<Tabs.Tab value="updates">{t("Updates")}</Tabs.Tab>
|
||||||
|
</Tabs.List>
|
||||||
|
</Tabs>
|
||||||
|
|
||||||
<ScrollArea.Autosize
|
<ScrollArea.Autosize
|
||||||
mah={500}
|
mah={500}
|
||||||
type="auto"
|
type="auto"
|
||||||
offsetScrollbars
|
offsetScrollbars
|
||||||
scrollbarSize={6}
|
scrollbarSize={6}
|
||||||
|
style={{ overscrollBehavior: "contain" }}
|
||||||
>
|
>
|
||||||
<NotificationList
|
<NotificationList
|
||||||
|
tab={tab}
|
||||||
filter={filter}
|
filter={filter}
|
||||||
onNavigate={() => setOpened(false)}
|
onNavigate={() => setOpened(false)}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
.notificationItem {
|
.notificationItem {
|
||||||
|
display: block;
|
||||||
padding: 8px 12px;
|
padding: 8px 12px;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
user-select: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.notificationItem:hover {
|
.notificationItem:hover {
|
||||||
@@ -11,3 +13,4 @@
|
|||||||
.divider {
|
.divider {
|
||||||
border-color: light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-5));
|
border-color: light-dark(var(--mantine-color-gray-2), var(--mantine-color-dark-5));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -15,10 +15,10 @@ import {
|
|||||||
export const NOTIFICATION_KEY = ["notifications"];
|
export const NOTIFICATION_KEY = ["notifications"];
|
||||||
export const UNREAD_COUNT_KEY = ["notifications", "unread-count"];
|
export const UNREAD_COUNT_KEY = ["notifications", "unread-count"];
|
||||||
|
|
||||||
export function useNotificationsQuery() {
|
export function useNotificationsQuery(type?: string) {
|
||||||
return useInfiniteQuery({
|
return useInfiniteQuery({
|
||||||
queryKey: NOTIFICATION_KEY,
|
queryKey: [...NOTIFICATION_KEY, type],
|
||||||
queryFn: ({ pageParam }) => getNotifications({ cursor: pageParam }),
|
queryFn: ({ pageParam }) => getNotifications({ cursor: pageParam, type }),
|
||||||
initialPageParam: undefined as string | undefined,
|
initialPageParam: undefined as string | undefined,
|
||||||
getNextPageParam: (lastPage) =>
|
getNextPageParam: (lastPage) =>
|
||||||
lastPage.meta.hasNextPage ? lastPage.meta.nextCursor : undefined,
|
lastPage.meta.hasNextPage ? lastPage.meta.nextCursor : undefined,
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import { IPagination } from "@/lib/types";
|
|||||||
export async function getNotifications(params: {
|
export async function getNotifications(params: {
|
||||||
limit?: number;
|
limit?: number;
|
||||||
cursor?: string;
|
cursor?: string;
|
||||||
|
type?: string;
|
||||||
}): Promise<IPagination<INotification>> {
|
}): Promise<IPagination<INotification>> {
|
||||||
const req = await api.post<IPagination<INotification>>(
|
const req = await api.post<IPagination<INotification>>(
|
||||||
"/notifications",
|
"/notifications",
|
||||||
|
|||||||
@@ -3,7 +3,8 @@ export type NotificationType =
|
|||||||
| "comment.created"
|
| "comment.created"
|
||||||
| "comment.resolved"
|
| "comment.resolved"
|
||||||
| "page.user_mention"
|
| "page.user_mention"
|
||||||
| "page.permission_granted";
|
| "page.permission_granted"
|
||||||
|
| "page.updated";
|
||||||
|
|
||||||
export type INotification = {
|
export type INotification = {
|
||||||
id: string;
|
id: string;
|
||||||
@@ -38,3 +39,5 @@ export type INotification = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export type NotificationFilter = "all" | "unread";
|
export type NotificationFilter = "all" | "unread";
|
||||||
|
|
||||||
|
export type NotificationTab = "direct" | "updates" | "all";
|
||||||
|
|||||||
@@ -3,6 +3,8 @@ import {
|
|||||||
IconArrowRight,
|
IconArrowRight,
|
||||||
IconArrowsHorizontal,
|
IconArrowsHorizontal,
|
||||||
IconDots,
|
IconDots,
|
||||||
|
IconEye,
|
||||||
|
IconEyeOff,
|
||||||
IconFileExport,
|
IconFileExport,
|
||||||
IconHistory,
|
IconHistory,
|
||||||
IconLink,
|
IconLink,
|
||||||
@@ -40,6 +42,11 @@ import { PageStateSegmentedControl } from "@/features/user/components/page-state
|
|||||||
import MovePageModal from "@/features/page/components/move-page-modal.tsx";
|
import MovePageModal from "@/features/page/components/move-page-modal.tsx";
|
||||||
import { useTimeAgo } from "@/hooks/use-time-ago.tsx";
|
import { useTimeAgo } from "@/hooks/use-time-ago.tsx";
|
||||||
import { PageShareModal } from "@/ee/page-permission";
|
import { PageShareModal } from "@/ee/page-permission";
|
||||||
|
import {
|
||||||
|
useWatchStatusQuery,
|
||||||
|
useWatchPageMutation,
|
||||||
|
useUnwatchPageMutation,
|
||||||
|
} from "@/features/page/queries/watcher-query";
|
||||||
|
|
||||||
interface PageHeaderMenuProps {
|
interface PageHeaderMenuProps {
|
||||||
readOnly?: boolean;
|
readOnly?: boolean;
|
||||||
@@ -123,6 +130,9 @@ function PageActionMenu({ readOnly }: PageActionMenuProps) {
|
|||||||
] = useDisclosure(false);
|
] = useDisclosure(false);
|
||||||
const [pageEditor] = useAtom(pageEditorAtom);
|
const [pageEditor] = useAtom(pageEditorAtom);
|
||||||
const pageUpdatedAt = useTimeAgo(page?.updatedAt);
|
const pageUpdatedAt = useTimeAgo(page?.updatedAt);
|
||||||
|
const { data: watchStatus } = useWatchStatusQuery(page?.id);
|
||||||
|
const watchPage = useWatchPageMutation();
|
||||||
|
const unwatchPage = useUnwatchPageMutation();
|
||||||
|
|
||||||
const handleCopyLink = () => {
|
const handleCopyLink = () => {
|
||||||
const pageUrl =
|
const pageUrl =
|
||||||
@@ -185,6 +195,23 @@ function PageActionMenu({ readOnly }: PageActionMenuProps) {
|
|||||||
>
|
>
|
||||||
{t("Copy as Markdown")}
|
{t("Copy as Markdown")}
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
|
|
||||||
|
{watchStatus?.watching ? (
|
||||||
|
<Menu.Item
|
||||||
|
leftSection={<IconEyeOff size={16} />}
|
||||||
|
onClick={() => unwatchPage.mutate(page.id)}
|
||||||
|
>
|
||||||
|
{t("Stop watching")}
|
||||||
|
</Menu.Item>
|
||||||
|
) : (
|
||||||
|
<Menu.Item
|
||||||
|
leftSection={<IconEye size={16} />}
|
||||||
|
onClick={() => watchPage.mutate(page.id)}
|
||||||
|
>
|
||||||
|
{t("Watch page")}
|
||||||
|
</Menu.Item>
|
||||||
|
)}
|
||||||
|
|
||||||
<Menu.Divider />
|
<Menu.Divider />
|
||||||
|
|
||||||
<Menu.Item leftSection={<IconArrowsHorizontal size={16} />}>
|
<Menu.Item leftSection={<IconArrowsHorizontal size={16} />}>
|
||||||
|
|||||||
@@ -0,0 +1,43 @@
|
|||||||
|
import { useMutation, useQuery, useQueryClient } from "@tanstack/react-query";
|
||||||
|
import {
|
||||||
|
watchPage,
|
||||||
|
unwatchPage,
|
||||||
|
getWatchStatus,
|
||||||
|
} from "@/features/page/services/watcher-service";
|
||||||
|
import { notifications } from "@mantine/notifications";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
|
const WATCHER_KEY = "watcher";
|
||||||
|
|
||||||
|
export function useWatchStatusQuery(pageId: string) {
|
||||||
|
return useQuery({
|
||||||
|
queryKey: [WATCHER_KEY, pageId],
|
||||||
|
queryFn: () => getWatchStatus(pageId),
|
||||||
|
enabled: !!pageId,
|
||||||
|
staleTime: 60_000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useWatchPageMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const { t } = useTranslation();
|
||||||
|
return useMutation({
|
||||||
|
mutationFn: (pageId: string) => watchPage(pageId),
|
||||||
|
onSuccess: (_data, pageId) => {
|
||||||
|
queryClient.setQueryData([WATCHER_KEY, pageId], { watching: true });
|
||||||
|
notifications.show({ message: t("You are now watching this page") });
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useUnwatchPageMutation() {
|
||||||
|
const queryClient = useQueryClient();
|
||||||
|
const { t } = useTranslation();
|
||||||
|
return useMutation({
|
||||||
|
mutationFn: (pageId: string) => unwatchPage(pageId),
|
||||||
|
onSuccess: (_data, pageId) => {
|
||||||
|
queryClient.setQueryData([WATCHER_KEY, pageId], { watching: false });
|
||||||
|
notifications.show({ message: t("You are no longer watching this page") });
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
import api from "@/lib/api-client";
|
||||||
|
|
||||||
|
export async function watchPage(pageId: string): Promise<{ watching: boolean }> {
|
||||||
|
const req = await api.post<{ watching: boolean }>("/pages/watch", { pageId });
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function unwatchPage(pageId: string): Promise<{ watching: boolean }> {
|
||||||
|
const req = await api.post<{ watching: boolean }>("/pages/unwatch", { pageId });
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getWatchStatus(pageId: string): Promise<{ watching: boolean }> {
|
||||||
|
const req = await api.post<{ watching: boolean }>("/pages/watch-status", { pageId });
|
||||||
|
return req.data;
|
||||||
|
}
|
||||||
@@ -3,6 +3,7 @@ import SpaceMembersList from "@/features/space/components/space-members.tsx";
|
|||||||
import AddSpaceMembersModal from "@/features/space/components/add-space-members-modal.tsx";
|
import AddSpaceMembersModal from "@/features/space/components/add-space-members-modal.tsx";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import SpaceDetails from "@/features/space/components/space-details.tsx";
|
import SpaceDetails from "@/features/space/components/space-details.tsx";
|
||||||
|
import SpaceSecuritySettings from "@/features/space/components/space-security-settings.tsx";
|
||||||
import { useSpaceQuery } from "@/features/space/queries/space-query.ts";
|
import { useSpaceQuery } from "@/features/space/queries/space-query.ts";
|
||||||
import { useSpaceAbility } from "@/features/space/permissions/use-space-ability.ts";
|
import { useSpaceAbility } from "@/features/space/permissions/use-space-ability.ts";
|
||||||
import {
|
import {
|
||||||
@@ -59,6 +60,14 @@ export default function SpaceSettingsModal({
|
|||||||
<Tabs.Tab fw={500} value="members">
|
<Tabs.Tab fw={500} value="members">
|
||||||
{t("Members")}
|
{t("Members")}
|
||||||
</Tabs.Tab>
|
</Tabs.Tab>
|
||||||
|
{spaceAbility.can(
|
||||||
|
SpaceCaslAction.Manage,
|
||||||
|
SpaceCaslSubject.Settings,
|
||||||
|
) && (
|
||||||
|
<Tabs.Tab fw={500} value="security">
|
||||||
|
{t("Security")}
|
||||||
|
</Tabs.Tab>
|
||||||
|
)}
|
||||||
</Tabs.List>
|
</Tabs.List>
|
||||||
|
|
||||||
<Tabs.Panel value="general">
|
<Tabs.Panel value="general">
|
||||||
@@ -91,6 +100,20 @@ export default function SpaceSettingsModal({
|
|||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
</Tabs.Panel>
|
</Tabs.Panel>
|
||||||
|
|
||||||
|
<Tabs.Panel value="security">
|
||||||
|
<ScrollArea h={580} scrollbarSize={5} pr={8}>
|
||||||
|
<div style={{ paddingBottom: "100px" }}>
|
||||||
|
<SpaceSecuritySettings
|
||||||
|
space={space}
|
||||||
|
readOnly={spaceAbility.cannot(
|
||||||
|
SpaceCaslAction.Manage,
|
||||||
|
SpaceCaslSubject.Settings,
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</ScrollArea>
|
||||||
|
</Tabs.Panel>
|
||||||
</Tabs>
|
</Tabs>
|
||||||
</div>
|
</div>
|
||||||
</Modal.Body>
|
</Modal.Body>
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ import {
|
|||||||
ResponsiveSettingsControl,
|
ResponsiveSettingsControl,
|
||||||
ResponsiveSettingsRow,
|
ResponsiveSettingsRow,
|
||||||
} from "@/components/ui/responsive-settings-row.tsx";
|
} from "@/components/ui/responsive-settings-row.tsx";
|
||||||
import SpacePublicSharingToggle from "@/ee/security/components/space-public-sharing-toggle.tsx";
|
|
||||||
|
|
||||||
interface SpaceDetailsProps {
|
interface SpaceDetailsProps {
|
||||||
spaceId: string;
|
spaceId: string;
|
||||||
@@ -27,7 +27,6 @@ interface SpaceDetailsProps {
|
|||||||
export default function SpaceDetails({ spaceId, readOnly }: SpaceDetailsProps) {
|
export default function SpaceDetails({ spaceId, readOnly }: SpaceDetailsProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const { data: space, isLoading, refetch } = useSpaceQuery(spaceId);
|
const { data: space, isLoading, refetch } = useSpaceQuery(spaceId);
|
||||||
const showSharingToggle = !readOnly;
|
|
||||||
const [exportOpened, { open: openExportModal, close: closeExportModal }] =
|
const [exportOpened, { open: openExportModal, close: closeExportModal }] =
|
||||||
useDisclosure(false);
|
useDisclosure(false);
|
||||||
const [isIconUploading, setIsIconUploading] = useState(false);
|
const [isIconUploading, setIsIconUploading] = useState(false);
|
||||||
@@ -89,13 +88,6 @@ export default function SpaceDetails({ spaceId, readOnly }: SpaceDetailsProps) {
|
|||||||
|
|
||||||
<EditSpaceForm space={space} readOnly={readOnly} />
|
<EditSpaceForm space={space} readOnly={readOnly} />
|
||||||
|
|
||||||
{showSharingToggle && (
|
|
||||||
<>
|
|
||||||
<Divider my="lg" />
|
|
||||||
<SpacePublicSharingToggle space={space} />
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{!readOnly && (
|
{!readOnly && (
|
||||||
<>
|
<>
|
||||||
<Divider my="lg" />
|
<Divider my="lg" />
|
||||||
|
|||||||
@@ -0,0 +1,34 @@
|
|||||||
|
import { Text, Divider } from "@mantine/core";
|
||||||
|
import React from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { ISpace } from "@/features/space/types/space.types.ts";
|
||||||
|
import SpacePublicSharingToggle from "@/ee/security/components/space-public-sharing-toggle.tsx";
|
||||||
|
import SpaceViewerCommentsToggle from "@/ee/security/components/space-viewer-comments-toggle.tsx";
|
||||||
|
|
||||||
|
type SpaceSecuritySettingsProps = {
|
||||||
|
space: ISpace;
|
||||||
|
readOnly?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function SpaceSecuritySettings({
|
||||||
|
space,
|
||||||
|
readOnly,
|
||||||
|
}: SpaceSecuritySettingsProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
if (readOnly) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<Text my="md" fw={600}>
|
||||||
|
{t("Security")}
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<SpacePublicSharingToggle space={space} />
|
||||||
|
|
||||||
|
<Divider my="lg" />
|
||||||
|
|
||||||
|
<SpaceViewerCommentsToggle space={space} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -9,8 +9,13 @@ export interface ISpaceSharingSettings {
|
|||||||
disabled?: boolean;
|
disabled?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface ISpaceCommentsSettings {
|
||||||
|
allowViewerComments?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
export interface ISpaceSettings {
|
export interface ISpaceSettings {
|
||||||
sharing?: ISpaceSharingSettings;
|
sharing?: ISpaceSharingSettings;
|
||||||
|
comments?: ISpaceCommentsSettings;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ISpace {
|
export interface ISpace {
|
||||||
@@ -29,6 +34,7 @@ export interface ISpace {
|
|||||||
settings?: ISpaceSettings;
|
settings?: ISpaceSettings;
|
||||||
// for updates
|
// for updates
|
||||||
disablePublicSharing?: boolean;
|
disablePublicSharing?: boolean;
|
||||||
|
allowViewerComments?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface IMembership {
|
interface IMembership {
|
||||||
|
|||||||
@@ -0,0 +1,117 @@
|
|||||||
|
import { userAtom } from "@/features/user/atoms/current-user-atom.ts";
|
||||||
|
import { updateUser } from "@/features/user/services/user-service.ts";
|
||||||
|
import { IUser, IUserSettings } from "@/features/user/types/user.types.ts";
|
||||||
|
import { Switch, Text, Title, Stack } from "@mantine/core";
|
||||||
|
import { useAtom } from "jotai";
|
||||||
|
import React, { useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import {
|
||||||
|
ResponsiveSettingsRow,
|
||||||
|
ResponsiveSettingsContent,
|
||||||
|
ResponsiveSettingsControl,
|
||||||
|
} from "@/components/ui/responsive-settings-row";
|
||||||
|
|
||||||
|
type NotificationKey = keyof NonNullable<IUserSettings["notifications"]>;
|
||||||
|
|
||||||
|
const notificationItems: {
|
||||||
|
key: NotificationKey;
|
||||||
|
dtoField: keyof IUser;
|
||||||
|
label: string;
|
||||||
|
description: string;
|
||||||
|
}[] = [
|
||||||
|
{
|
||||||
|
key: "page.updated",
|
||||||
|
dtoField: "notificationPageUpdates",
|
||||||
|
label: "Page updates",
|
||||||
|
description: "Get notified when pages you watch are updated.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "page.userMention",
|
||||||
|
dtoField: "notificationPageUserMention",
|
||||||
|
label: "Page mentions",
|
||||||
|
description: "Get notified when someone mentions you on a page.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "comment.userMention",
|
||||||
|
dtoField: "notificationCommentUserMention",
|
||||||
|
label: "Comment mentions",
|
||||||
|
description: "Get notified when someone mentions you in a comment.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "comment.created",
|
||||||
|
dtoField: "notificationCommentCreated",
|
||||||
|
label: "New comments",
|
||||||
|
description:
|
||||||
|
"Get notified about new comments on threads you participate in.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "comment.resolved",
|
||||||
|
dtoField: "notificationCommentResolved",
|
||||||
|
label: "Resolved comments",
|
||||||
|
description: "Get notified when your comment is resolved.",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
function NotificationToggle({
|
||||||
|
settingKey,
|
||||||
|
dtoField,
|
||||||
|
label,
|
||||||
|
description,
|
||||||
|
}: {
|
||||||
|
settingKey: NotificationKey;
|
||||||
|
dtoField: keyof IUser;
|
||||||
|
label: string;
|
||||||
|
description: string;
|
||||||
|
}) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const [user, setUser] = useAtom(userAtom);
|
||||||
|
const [checked, setChecked] = useState(
|
||||||
|
user.settings?.notifications?.[settingKey] !== false,
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleChange = async (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const value = event.currentTarget.checked;
|
||||||
|
setChecked(value);
|
||||||
|
try {
|
||||||
|
const updatedUser = await updateUser({ [dtoField]: value } as any);
|
||||||
|
setUser(updatedUser);
|
||||||
|
} catch {
|
||||||
|
setChecked(!value);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ResponsiveSettingsRow>
|
||||||
|
<ResponsiveSettingsContent>
|
||||||
|
<Text size="md">{t(label)}</Text>
|
||||||
|
<Text size="sm" c="dimmed">
|
||||||
|
{t(description)}
|
||||||
|
</Text>
|
||||||
|
</ResponsiveSettingsContent>
|
||||||
|
|
||||||
|
<ResponsiveSettingsControl>
|
||||||
|
<Switch checked={checked} onChange={handleChange} />
|
||||||
|
</ResponsiveSettingsControl>
|
||||||
|
</ResponsiveSettingsRow>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function NotificationPref() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Stack gap="xs">
|
||||||
|
<Title order={5}>{t("Email notifications")}</Title>
|
||||||
|
|
||||||
|
{notificationItems.map((item) => (
|
||||||
|
<NotificationToggle
|
||||||
|
key={item.key}
|
||||||
|
settingKey={item.key}
|
||||||
|
dtoField={item.dtoField}
|
||||||
|
label={item.label}
|
||||||
|
description={item.description}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</Stack>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -20,6 +20,11 @@ export interface IUser {
|
|||||||
deletedAt: Date;
|
deletedAt: Date;
|
||||||
fullPageWidth: boolean; // used for update
|
fullPageWidth: boolean; // used for update
|
||||||
pageEditMode: string; // used for update
|
pageEditMode: string; // used for update
|
||||||
|
notificationPageUpdates: boolean; // used for update
|
||||||
|
notificationPageUserMention: boolean; // used for update
|
||||||
|
notificationCommentUserMention: boolean; // used for update
|
||||||
|
notificationCommentCreated: boolean; // used for update
|
||||||
|
notificationCommentResolved: boolean; // used for update
|
||||||
hasGeneratedPassword?: boolean;
|
hasGeneratedPassword?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -33,6 +38,13 @@ export interface IUserSettings {
|
|||||||
fullPageWidth: boolean;
|
fullPageWidth: boolean;
|
||||||
pageEditMode: string;
|
pageEditMode: string;
|
||||||
};
|
};
|
||||||
|
notifications?: {
|
||||||
|
"page.updated"?: boolean;
|
||||||
|
"page.userMention"?: boolean;
|
||||||
|
"comment.userMention"?: boolean;
|
||||||
|
"comment.created"?: boolean;
|
||||||
|
"comment.resolved"?: boolean;
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum PageEditMode {
|
export enum PageEditMode {
|
||||||
|
|||||||
@@ -53,6 +53,9 @@ function PageContent({ pageSlug }: { pageSlug: string | undefined }) {
|
|||||||
const { data: space } = useGetSpaceBySlugQuery(page?.space?.slug);
|
const { data: space } = useGetSpaceBySlugQuery(page?.space?.slug);
|
||||||
|
|
||||||
const canEdit = page?.permissions?.canEdit ?? false;
|
const canEdit = page?.permissions?.canEdit ?? false;
|
||||||
|
const canComment =
|
||||||
|
canEdit ||
|
||||||
|
(space?.settings?.comments?.allowViewerComments === true);
|
||||||
|
|
||||||
if (isLoading) {
|
if (isLoading) {
|
||||||
return <></>;
|
return <></>;
|
||||||
@@ -104,6 +107,7 @@ function PageContent({ pageSlug }: { pageSlug: string | undefined }) {
|
|||||||
slugId={page.slugId}
|
slugId={page.slugId}
|
||||||
spaceSlug={page?.space?.slug}
|
spaceSlug={page?.space?.slug}
|
||||||
editable={canEdit}
|
editable={canEdit}
|
||||||
|
canComment={canComment}
|
||||||
/>
|
/>
|
||||||
<MemoizedHistoryModal pageId={page.id} />
|
<MemoizedHistoryModal pageId={page.id} />
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ import AccountLanguage from "@/features/user/components/account-language.tsx";
|
|||||||
import AccountTheme from "@/features/user/components/account-theme.tsx";
|
import AccountTheme from "@/features/user/components/account-theme.tsx";
|
||||||
import PageWidthPref from "@/features/user/components/page-width-pref.tsx";
|
import PageWidthPref from "@/features/user/components/page-width-pref.tsx";
|
||||||
import PageEditPref from "@/features/user/components/page-state-pref";
|
import PageEditPref from "@/features/user/components/page-state-pref";
|
||||||
|
import NotificationPref from "@/features/user/components/notification-pref";
|
||||||
import { getAppName } from "@/lib/config.ts";
|
import { getAppName } from "@/lib/config.ts";
|
||||||
import { Divider } from "@mantine/core";
|
import { Divider } from "@mantine/core";
|
||||||
import { Helmet } from "react-helmet-async";
|
import { Helmet } from "react-helmet-async";
|
||||||
@@ -33,6 +34,10 @@ export default function AccountPreferences() {
|
|||||||
<Divider my={"md"} />
|
<Divider my={"md"} />
|
||||||
|
|
||||||
<PageEditPref />
|
<PageEditPref />
|
||||||
|
|
||||||
|
<Divider my={"md"} />
|
||||||
|
|
||||||
|
<NotificationPref />
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "server",
|
"name": "server",
|
||||||
"version": "0.70.3",
|
"version": "0.71.1",
|
||||||
"description": "",
|
"description": "",
|
||||||
"author": "",
|
"author": "",
|
||||||
"private": true,
|
"private": true,
|
||||||
@@ -44,6 +44,7 @@
|
|||||||
"@langchain/core": "1.1.34",
|
"@langchain/core": "1.1.34",
|
||||||
"@langchain/textsplitters": "1.0.1",
|
"@langchain/textsplitters": "1.0.1",
|
||||||
"@modelcontextprotocol/sdk": "^1.27.1",
|
"@modelcontextprotocol/sdk": "^1.27.1",
|
||||||
|
"@nest-lab/throttler-storage-redis": "^1.2.0",
|
||||||
"@nestjs-labs/nestjs-ioredis": "^11.0.4",
|
"@nestjs-labs/nestjs-ioredis": "^11.0.4",
|
||||||
"@nestjs/bullmq": "^11.0.4",
|
"@nestjs/bullmq": "^11.0.4",
|
||||||
"@nestjs/cache-manager": "^3.1.0",
|
"@nestjs/cache-manager": "^3.1.0",
|
||||||
@@ -58,6 +59,7 @@
|
|||||||
"@nestjs/platform-socket.io": "^11.1.17",
|
"@nestjs/platform-socket.io": "^11.1.17",
|
||||||
"@nestjs/schedule": "^6.1.1",
|
"@nestjs/schedule": "^6.1.1",
|
||||||
"@nestjs/terminus": "^11.1.1",
|
"@nestjs/terminus": "^11.1.1",
|
||||||
|
"@nestjs/throttler": "^6.5.0",
|
||||||
"@nestjs/websockets": "^11.1.17",
|
"@nestjs/websockets": "^11.1.17",
|
||||||
"@node-saml/passport-saml": "^5.1.0",
|
"@node-saml/passport-saml": "^5.1.0",
|
||||||
"@react-email/components": "1.0.10",
|
"@react-email/components": "1.0.10",
|
||||||
@@ -73,8 +75,9 @@
|
|||||||
"class-transformer": "^0.5.1",
|
"class-transformer": "^0.5.1",
|
||||||
"class-validator": "^0.15.1",
|
"class-validator": "^0.15.1",
|
||||||
"cookie": "^1.1.1",
|
"cookie": "^1.1.1",
|
||||||
|
"fastify-ip": "^2.0.0",
|
||||||
"fs-extra": "^11.3.4",
|
"fs-extra": "^11.3.4",
|
||||||
"happy-dom": "20.8.4",
|
"happy-dom": "20.8.9",
|
||||||
"ioredis": "^5.10.1",
|
"ioredis": "^5.10.1",
|
||||||
"jsonwebtoken": "^9.0.3",
|
"jsonwebtoken": "^9.0.3",
|
||||||
"kysely": "^0.28.14",
|
"kysely": "^0.28.14",
|
||||||
@@ -89,7 +92,7 @@
|
|||||||
"nestjs-cls": "^6.2.0",
|
"nestjs-cls": "^6.2.0",
|
||||||
"nestjs-kysely": "^3.1.2",
|
"nestjs-kysely": "^3.1.2",
|
||||||
"nestjs-pino": "^4.6.1",
|
"nestjs-pino": "^4.6.1",
|
||||||
"nodemailer": "^8.0.3",
|
"nodemailer": "^8.0.4",
|
||||||
"openid-client": "^6.8.2",
|
"openid-client": "^6.8.2",
|
||||||
"otpauth": "^9.5.0",
|
"otpauth": "^9.5.0",
|
||||||
"p-limit": "^7.3.0",
|
"p-limit": "^7.3.0",
|
||||||
|
|||||||
@@ -26,6 +26,7 @@ import KeyvRedis from '@keyv/redis';
|
|||||||
import { LoggerModule } from './common/logger/logger.module';
|
import { LoggerModule } from './common/logger/logger.module';
|
||||||
import { ClsModule } from 'nestjs-cls';
|
import { ClsModule } from 'nestjs-cls';
|
||||||
import { NoopAuditModule } from './integrations/audit/audit.module';
|
import { NoopAuditModule } from './integrations/audit/audit.module';
|
||||||
|
import { ThrottleModule } from './integrations/throttle/throttle.module';
|
||||||
|
|
||||||
const enterpriseModules = [];
|
const enterpriseModules = [];
|
||||||
try {
|
try {
|
||||||
@@ -83,6 +84,7 @@ try {
|
|||||||
EventEmitterModule.forRoot(),
|
EventEmitterModule.forRoot(),
|
||||||
SecurityModule,
|
SecurityModule,
|
||||||
TelemetryModule,
|
TelemetryModule,
|
||||||
|
ThrottleModule,
|
||||||
...enterpriseModules,
|
...enterpriseModules,
|
||||||
],
|
],
|
||||||
controllers: [AppController],
|
controllers: [AppController],
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import {
|
|||||||
prosemirrorNodeToYElement,
|
prosemirrorNodeToYElement,
|
||||||
tiptapExtensions,
|
tiptapExtensions,
|
||||||
} from './collaboration.util';
|
} from './collaboration.util';
|
||||||
|
import { setYjsMark, updateYjsMarkAttribute, YjsSelection } from './yjs.util';
|
||||||
import * as Y from 'yjs';
|
import * as Y from 'yjs';
|
||||||
import { User } from '@docmost/db/types/entity.types';
|
import { User } from '@docmost/db/types/entity.types';
|
||||||
|
|
||||||
@@ -27,6 +28,53 @@ export class CollaborationHandler {
|
|||||||
// const fragment = doc.getXmlFragment('default');
|
// const fragment = doc.getXmlFragment('default');
|
||||||
//});
|
//});
|
||||||
},
|
},
|
||||||
|
setCommentMark: async (
|
||||||
|
documentName: string,
|
||||||
|
payload: {
|
||||||
|
yjsSelection: YjsSelection;
|
||||||
|
commentId: string;
|
||||||
|
resolved: boolean;
|
||||||
|
user: User;
|
||||||
|
},
|
||||||
|
) => {
|
||||||
|
const { yjsSelection, commentId, resolved, user } = payload;
|
||||||
|
await this.withYdocConnection(
|
||||||
|
hocuspocus,
|
||||||
|
documentName,
|
||||||
|
{ user },
|
||||||
|
(doc) => {
|
||||||
|
const fragment = doc.getXmlFragment('default');
|
||||||
|
setYjsMark(doc, fragment, yjsSelection, 'comment', {
|
||||||
|
commentId,
|
||||||
|
resolved,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
|
resolveCommentMark: async (
|
||||||
|
documentName: string,
|
||||||
|
payload: {
|
||||||
|
commentId: string;
|
||||||
|
resolved: boolean;
|
||||||
|
user: User;
|
||||||
|
},
|
||||||
|
) => {
|
||||||
|
const { commentId, resolved, user } = payload;
|
||||||
|
await this.withYdocConnection(
|
||||||
|
hocuspocus,
|
||||||
|
documentName,
|
||||||
|
{ user },
|
||||||
|
(doc) => {
|
||||||
|
const fragment = doc.getXmlFragment('default');
|
||||||
|
updateYjsMarkAttribute(
|
||||||
|
fragment,
|
||||||
|
'comment',
|
||||||
|
{ name: 'commentId', value: commentId },
|
||||||
|
{ resolved },
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
updatePageContent: async (
|
updatePageContent: async (
|
||||||
documentName: string,
|
documentName: string,
|
||||||
payload: {
|
payload: {
|
||||||
@@ -58,8 +106,7 @@ export class CollaborationHandler {
|
|||||||
} else {
|
} else {
|
||||||
const newContent = prosemirrorJson.content || [];
|
const newContent = prosemirrorJson.content || [];
|
||||||
const yElements = newContent.map(prosemirrorNodeToYElement);
|
const yElements = newContent.map(prosemirrorNodeToYElement);
|
||||||
const position =
|
const position = operation === 'prepend' ? 0 : fragment.length;
|
||||||
operation === 'prepend' ? 0 : fragment.length;
|
|
||||||
fragment.insert(position, yElements);
|
fragment.insert(position, yElements);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -18,12 +18,10 @@ import { QueueJob, QueueName } from '../../integrations/queue/constants';
|
|||||||
import { Queue } from 'bullmq';
|
import { Queue } from 'bullmq';
|
||||||
import {
|
import {
|
||||||
extractMentions,
|
extractMentions,
|
||||||
extractPageMentions,
|
|
||||||
extractUserMentions,
|
extractUserMentions,
|
||||||
} from '../../common/helpers/prosemirror/utils';
|
} from '../../common/helpers/prosemirror/utils';
|
||||||
import { isDeepStrictEqual } from 'node:util';
|
import { isDeepStrictEqual } from 'node:util';
|
||||||
import {
|
import {
|
||||||
IPageBacklinkJob,
|
|
||||||
IPageHistoryJob,
|
IPageHistoryJob,
|
||||||
IPageMentionNotificationJob,
|
IPageMentionNotificationJob,
|
||||||
} from '../../integrations/queue/constants/queue.interface';
|
} from '../../integrations/queue/constants/queue.interface';
|
||||||
@@ -43,7 +41,6 @@ export class PersistenceExtension implements Extension {
|
|||||||
constructor(
|
constructor(
|
||||||
private readonly pageRepo: PageRepo,
|
private readonly pageRepo: PageRepo,
|
||||||
@InjectKysely() private readonly db: KyselyDB,
|
@InjectKysely() private readonly db: KyselyDB,
|
||||||
@InjectQueue(QueueName.GENERAL_QUEUE) private generalQueue: Queue,
|
|
||||||
@InjectQueue(QueueName.AI_QUEUE) private aiQueue: Queue,
|
@InjectQueue(QueueName.AI_QUEUE) private aiQueue: Queue,
|
||||||
@InjectQueue(QueueName.HISTORY_QUEUE) private historyQueue: Queue,
|
@InjectQueue(QueueName.HISTORY_QUEUE) private historyQueue: Queue,
|
||||||
@InjectQueue(QueueName.NOTIFICATION_QUEUE) private notificationQueue: Queue,
|
@InjectQueue(QueueName.NOTIFICATION_QUEUE) private notificationQueue: Queue,
|
||||||
@@ -165,13 +162,6 @@ export class PersistenceExtension implements Extension {
|
|||||||
await this.collabHistory.addContributors(pageId, editingUserIds);
|
await this.collabHistory.addContributors(pageId, editingUserIds);
|
||||||
|
|
||||||
const mentions = extractMentions(tiptapJson);
|
const mentions = extractMentions(tiptapJson);
|
||||||
const pageMentions = extractPageMentions(mentions);
|
|
||||||
|
|
||||||
await this.generalQueue.add(QueueJob.PAGE_BACKLINKS, {
|
|
||||||
pageId: pageId,
|
|
||||||
workspaceId: page.workspaceId,
|
|
||||||
mentions: pageMentions,
|
|
||||||
} as IPageBacklinkJob);
|
|
||||||
|
|
||||||
const userMentions = extractUserMentions(mentions);
|
const userMentions = extractUserMentions(mentions);
|
||||||
const oldMentions = page.content ? extractMentions(page.content) : [];
|
const oldMentions = page.content ? extractMentions(page.content) : [];
|
||||||
|
|||||||
@@ -1,8 +1,18 @@
|
|||||||
import { Logger, OnModuleDestroy } from '@nestjs/common';
|
import { Logger, OnModuleDestroy } from '@nestjs/common';
|
||||||
import { OnWorkerEvent, Processor, WorkerHost } from '@nestjs/bullmq';
|
import { OnWorkerEvent, Processor, WorkerHost } from '@nestjs/bullmq';
|
||||||
import { Job } from 'bullmq';
|
import { InjectQueue } from '@nestjs/bullmq';
|
||||||
|
import { Job, Queue } from 'bullmq';
|
||||||
import { QueueJob, QueueName } from '../../integrations/queue/constants';
|
import { QueueJob, QueueName } from '../../integrations/queue/constants';
|
||||||
import { IPageHistoryJob } from '../../integrations/queue/constants/queue.interface';
|
import {
|
||||||
|
IPageBacklinkJob,
|
||||||
|
IPageHistoryJob,
|
||||||
|
IPageUpdateNotificationJob,
|
||||||
|
} from '../../integrations/queue/constants/queue.interface';
|
||||||
|
import {
|
||||||
|
extractMentions,
|
||||||
|
extractPageMentions,
|
||||||
|
extractInternalLinkSlugIds,
|
||||||
|
} from '../../common/helpers/prosemirror/utils';
|
||||||
import { PageHistoryRepo } from '@docmost/db/repos/page/page-history.repo';
|
import { PageHistoryRepo } from '@docmost/db/repos/page/page-history.repo';
|
||||||
import { PageRepo } from '@docmost/db/repos/page/page.repo';
|
import { PageRepo } from '@docmost/db/repos/page/page.repo';
|
||||||
import { isDeepStrictEqual } from 'node:util';
|
import { isDeepStrictEqual } from 'node:util';
|
||||||
@@ -18,6 +28,8 @@ export class HistoryProcessor extends WorkerHost implements OnModuleDestroy {
|
|||||||
private readonly pageRepo: PageRepo,
|
private readonly pageRepo: PageRepo,
|
||||||
private readonly collabHistory: CollabHistoryService,
|
private readonly collabHistory: CollabHistoryService,
|
||||||
private readonly watcherService: WatcherService,
|
private readonly watcherService: WatcherService,
|
||||||
|
@InjectQueue(QueueName.NOTIFICATION_QUEUE) private notificationQueue: Queue,
|
||||||
|
@InjectQueue(QueueName.GENERAL_QUEUE) private generalQueue: Queue,
|
||||||
) {
|
) {
|
||||||
super();
|
super();
|
||||||
}
|
}
|
||||||
@@ -47,8 +59,7 @@ export class HistoryProcessor extends WorkerHost implements OnModuleDestroy {
|
|||||||
!lastHistory ||
|
!lastHistory ||
|
||||||
!isDeepStrictEqual(lastHistory.content, page.content)
|
!isDeepStrictEqual(lastHistory.content, page.content)
|
||||||
) {
|
) {
|
||||||
const contributorIds =
|
const contributorIds = await this.collabHistory.popContributors(pageId);
|
||||||
await this.collabHistory.popContributors(pageId);
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await this.watcherService.addPageWatchers(
|
await this.watcherService.addPageWatchers(
|
||||||
@@ -61,12 +72,41 @@ export class HistoryProcessor extends WorkerHost implements OnModuleDestroy {
|
|||||||
await this.pageHistoryRepo.saveHistory(page, { contributorIds });
|
await this.pageHistoryRepo.saveHistory(page, { contributorIds });
|
||||||
this.logger.debug(`History created for page: ${pageId}`);
|
this.logger.debug(`History created for page: ${pageId}`);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
await this.collabHistory.addContributors(
|
await this.collabHistory.addContributors(pageId, contributorIds);
|
||||||
pageId,
|
|
||||||
contributorIds,
|
|
||||||
);
|
|
||||||
throw err;
|
throw err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const mentions = extractMentions(page.content);
|
||||||
|
const pageMentions = extractPageMentions(mentions);
|
||||||
|
const internalLinkSlugIds = extractInternalLinkSlugIds(page.content);
|
||||||
|
|
||||||
|
await this.generalQueue
|
||||||
|
.add(QueueJob.PAGE_BACKLINKS, {
|
||||||
|
pageId,
|
||||||
|
workspaceId: page.workspaceId,
|
||||||
|
mentions: pageMentions,
|
||||||
|
internalLinkSlugIds,
|
||||||
|
} as IPageBacklinkJob)
|
||||||
|
.catch((err) => {
|
||||||
|
this.logger.error(
|
||||||
|
`Failed to queue backlinks for ${pageId}: ${err.message}`,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
if (contributorIds.length > 0 && lastHistory?.content) {
|
||||||
|
await this.notificationQueue
|
||||||
|
.add(QueueJob.PAGE_UPDATED, {
|
||||||
|
pageId,
|
||||||
|
spaceId: page.spaceId,
|
||||||
|
workspaceId: page.workspaceId,
|
||||||
|
actorIds: contributorIds,
|
||||||
|
} as IPageUpdateNotificationJob)
|
||||||
|
.catch((err) => {
|
||||||
|
this.logger.error(
|
||||||
|
`Failed to queue page update notification for ${pageId}: ${err.message}`,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
throw err;
|
throw err;
|
||||||
|
|||||||
@@ -11,12 +11,14 @@ import { CollaborationController } from './collaboration.controller';
|
|||||||
import { LoggerModule } from '../../common/logger/logger.module';
|
import { LoggerModule } from '../../common/logger/logger.module';
|
||||||
import { RedisModule } from '@nestjs-labs/nestjs-ioredis';
|
import { RedisModule } from '@nestjs-labs/nestjs-ioredis';
|
||||||
import { RedisConfigService } from '../../integrations/redis/redis-config.service';
|
import { RedisConfigService } from '../../integrations/redis/redis-config.service';
|
||||||
|
import { CaslModule } from '../../core/casl/casl.module';
|
||||||
|
|
||||||
@Module({
|
@Module({
|
||||||
imports: [
|
imports: [
|
||||||
LoggerModule,
|
LoggerModule,
|
||||||
DatabaseModule,
|
DatabaseModule,
|
||||||
EnvironmentModule,
|
EnvironmentModule,
|
||||||
|
CaslModule,
|
||||||
CollaborationModule,
|
CollaborationModule,
|
||||||
QueueModule,
|
QueueModule,
|
||||||
HealthModule,
|
HealthModule,
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import {
|
import {
|
||||||
initProseMirrorDoc,
|
initProseMirrorDoc,
|
||||||
relativePositionToAbsolutePosition,
|
relativePositionToAbsolutePosition,
|
||||||
} from 'y-prosemirror';
|
} from '@tiptap/y-tiptap';
|
||||||
import * as Y from 'yjs';
|
import * as Y from 'yjs';
|
||||||
import { Document } from '@hocuspocus/server';
|
import { Document } from '@hocuspocus/server';
|
||||||
import { getSchema } from '@tiptap/core';
|
import { getSchema } from '@tiptap/core';
|
||||||
|
|||||||
@@ -0,0 +1,22 @@
|
|||||||
|
export const Feature = {
|
||||||
|
SSO_CUSTOM: 'sso:custom',
|
||||||
|
SSO_GOOGLE: 'sso:google',
|
||||||
|
MFA: 'mfa',
|
||||||
|
API_KEYS: 'api:keys',
|
||||||
|
COMMENT_RESOLUTION: 'comment:resolution',
|
||||||
|
PAGE_PERMISSIONS: 'page:permissions',
|
||||||
|
AI: 'ai',
|
||||||
|
CONFLUENCE_IMPORT: 'import:confluence',
|
||||||
|
DOCX_IMPORT: 'import:docx',
|
||||||
|
ATTACHMENT_INDEXING: 'attachment:indexing',
|
||||||
|
SECURITY_SETTINGS: 'security:settings',
|
||||||
|
MCP: 'mcp',
|
||||||
|
SCIM: 'scim',
|
||||||
|
PAGE_VERIFICATION: 'page:verification',
|
||||||
|
AUDIT_LOGS: 'audit:logs',
|
||||||
|
RETENTION: 'retention',
|
||||||
|
SHARING_CONTROLS: 'sharing:controls',
|
||||||
|
VIEWER_COMMENTS: 'comment:viewer',
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
export type FeatureKey = (typeof Feature)[keyof typeof Feature];
|
||||||
@@ -7,6 +7,10 @@ import { validate as isValidUUID } from 'uuid';
|
|||||||
import { Transform } from '@tiptap/pm/transform';
|
import { Transform } from '@tiptap/pm/transform';
|
||||||
import { TiptapTransformer } from '@hocuspocus/transformer';
|
import { TiptapTransformer } from '@hocuspocus/transformer';
|
||||||
import * as Y from 'yjs';
|
import * as Y from 'yjs';
|
||||||
|
import {
|
||||||
|
INTERNAL_LINK_REGEX,
|
||||||
|
extractPageSlugId,
|
||||||
|
} from '../../../integrations/export/utils';
|
||||||
|
|
||||||
export interface MentionNode {
|
export interface MentionNode {
|
||||||
id: string;
|
id: string;
|
||||||
@@ -64,6 +68,27 @@ export function extractPageMentions(mentionList: MentionNode[]): MentionNode[] {
|
|||||||
return pageMentionList as MentionNode[];
|
return pageMentionList as MentionNode[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function extractInternalLinkSlugIds(prosemirrorJson: any): string[] {
|
||||||
|
const slugIds: string[] = [];
|
||||||
|
const doc = jsonToNode(prosemirrorJson);
|
||||||
|
|
||||||
|
doc.descendants((node: Node) => {
|
||||||
|
for (const mark of node.marks) {
|
||||||
|
if (mark.type.name === 'link' && mark.attrs.internal && mark.attrs.href) {
|
||||||
|
const match = mark.attrs.href.match(INTERNAL_LINK_REGEX);
|
||||||
|
if (match) {
|
||||||
|
const slugId = extractPageSlugId(match[5]);
|
||||||
|
if (slugId && !slugIds.includes(slugId)) {
|
||||||
|
slugIds.push(slugId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return slugIds;
|
||||||
|
}
|
||||||
|
|
||||||
export function extractUserMentionIdsFromJson(json: any): string[] {
|
export function extractUserMentionIdsFromJson(json: any): string[] {
|
||||||
const userIds: string[] = [];
|
const userIds: string[] = [];
|
||||||
|
|
||||||
|
|||||||
@@ -50,20 +50,12 @@ export function createPinoConfig(): Params {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
serializers: {
|
serializers: {
|
||||||
req: (req) => {
|
req: (req) => ({
|
||||||
const forwardedFor = req.headers?.['x-forwarded-for'];
|
method: req.method,
|
||||||
const ip =
|
url: req.url,
|
||||||
req.headers?.['cf-connecting-ip'] ||
|
ip: req.ip || req.remoteAddress,
|
||||||
(typeof forwardedFor === 'string' ? forwardedFor.split(',')[0]?.trim() : undefined) ||
|
userAgent: req.headers?.['user-agent'],
|
||||||
req.remoteAddress;
|
}),
|
||||||
|
|
||||||
return {
|
|
||||||
method: req.method,
|
|
||||||
url: req.url,
|
|
||||||
ip,
|
|
||||||
userAgent: req.headers?.['user-agent'],
|
|
||||||
};
|
|
||||||
},
|
|
||||||
res: (res) => ({
|
res: (res) => ({
|
||||||
statusCode: res.statusCode,
|
statusCode: res.statusCode,
|
||||||
}),
|
}),
|
||||||
|
|||||||
@@ -18,7 +18,8 @@ export class AuditContextMiddleware implements NestMiddleware {
|
|||||||
|
|
||||||
use(req: FastifyRequest['raw'], res: FastifyReply['raw'], next: () => void) {
|
use(req: FastifyRequest['raw'], res: FastifyReply['raw'], next: () => void) {
|
||||||
const workspaceId = (req as any).workspaceId ?? null;
|
const workspaceId = (req as any).workspaceId ?? null;
|
||||||
const ipAddress = this.extractIpAddress(req);
|
|
||||||
|
const ipAddress = (req as any).ip ?? (req as any).socket?.remoteAddress ?? null;
|
||||||
|
|
||||||
const userAgent =
|
const userAgent =
|
||||||
(req.headers['user-agent'] as string) ?? null;
|
(req.headers['user-agent'] as string) ?? null;
|
||||||
@@ -35,21 +36,4 @@ export class AuditContextMiddleware implements NestMiddleware {
|
|||||||
|
|
||||||
next();
|
next();
|
||||||
}
|
}
|
||||||
|
|
||||||
private extractIpAddress(req: FastifyRequest['raw']): string | null {
|
|
||||||
const xForwardedFor = req.headers['x-forwarded-for'];
|
|
||||||
if (xForwardedFor) {
|
|
||||||
const ips = Array.isArray(xForwardedFor)
|
|
||||||
? xForwardedFor[0]
|
|
||||||
: xForwardedFor.split(',')[0];
|
|
||||||
return ips?.trim() ?? null;
|
|
||||||
}
|
|
||||||
|
|
||||||
const xRealIp = req.headers['x-real-ip'];
|
|
||||||
if (xRealIp) {
|
|
||||||
return Array.isArray(xRealIp) ? xRealIp[0] : xRealIp;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (req as any).socket?.remoteAddress ?? null;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ import {
|
|||||||
UseGuards,
|
UseGuards,
|
||||||
Logger,
|
Logger,
|
||||||
} from '@nestjs/common';
|
} from '@nestjs/common';
|
||||||
|
import { SkipThrottle, ThrottlerGuard } from '@nestjs/throttler';
|
||||||
import { LoginDto } from './dto/login.dto';
|
import { LoginDto } from './dto/login.dto';
|
||||||
import { AuthService } from './services/auth.service';
|
import { AuthService } from './services/auth.service';
|
||||||
import { SessionService } from '../session/session.service';
|
import { SessionService } from '../session/session.service';
|
||||||
@@ -33,6 +34,7 @@ import {
|
|||||||
IAuditService,
|
IAuditService,
|
||||||
} from '../../integrations/audit/audit.service';
|
} from '../../integrations/audit/audit.service';
|
||||||
|
|
||||||
|
@UseGuards(ThrottlerGuard)
|
||||||
@Controller('auth')
|
@Controller('auth')
|
||||||
export class AuthController {
|
export class AuthController {
|
||||||
private readonly logger = new Logger(AuthController.name);
|
private readonly logger = new Logger(AuthController.name);
|
||||||
@@ -111,6 +113,7 @@ export class AuthController {
|
|||||||
return workspace;
|
return workspace;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SkipThrottle()
|
||||||
@UseGuards(JwtAuthGuard)
|
@UseGuards(JwtAuthGuard)
|
||||||
@HttpCode(HttpStatus.OK)
|
@HttpCode(HttpStatus.OK)
|
||||||
@Post('change-password')
|
@Post('change-password')
|
||||||
@@ -173,6 +176,7 @@ export class AuthController {
|
|||||||
return this.authService.verifyUserToken(verifyUserTokenDto, workspace.id);
|
return this.authService.verifyUserToken(verifyUserTokenDto, workspace.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SkipThrottle()
|
||||||
@UseGuards(JwtAuthGuard)
|
@UseGuards(JwtAuthGuard)
|
||||||
@HttpCode(HttpStatus.OK)
|
@HttpCode(HttpStatus.OK)
|
||||||
@Post('collab-token')
|
@Post('collab-token')
|
||||||
@@ -183,6 +187,7 @@ export class AuthController {
|
|||||||
return this.authService.getCollabToken(user, workspace.id);
|
return this.authService.getCollabToken(user, workspace.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SkipThrottle()
|
||||||
@UseGuards(JwtAuthGuard)
|
@UseGuards(JwtAuthGuard)
|
||||||
@HttpCode(HttpStatus.OK)
|
@HttpCode(HttpStatus.OK)
|
||||||
@Post('logout')
|
@Post('logout')
|
||||||
|
|||||||
@@ -58,13 +58,13 @@ export class CommentController {
|
|||||||
throw new NotFoundException('Page not found');
|
throw new NotFoundException('Page not found');
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.pageAccessService.validateCanEdit(page, user);
|
await this.pageAccessService.validateCanComment(page, user, workspace.id);
|
||||||
|
|
||||||
const comment = await this.commentService.create(
|
const comment = await this.commentService.create(
|
||||||
{
|
{
|
||||||
userId: user.id,
|
|
||||||
page,
|
page,
|
||||||
workspaceId: workspace.id,
|
workspaceId: workspace.id,
|
||||||
|
user,
|
||||||
},
|
},
|
||||||
createCommentDto,
|
createCommentDto,
|
||||||
);
|
);
|
||||||
@@ -120,7 +120,7 @@ export class CommentController {
|
|||||||
|
|
||||||
@HttpCode(HttpStatus.OK)
|
@HttpCode(HttpStatus.OK)
|
||||||
@Post('update')
|
@Post('update')
|
||||||
async update(@Body() dto: UpdateCommentDto, @AuthUser() user: User) {
|
async update(@Body() dto: UpdateCommentDto, @AuthUser() user: User, @AuthWorkspace() workspace: Workspace) {
|
||||||
const comment = await this.commentRepo.findById(dto.commentId, {
|
const comment = await this.commentRepo.findById(dto.commentId, {
|
||||||
includeCreator: true,
|
includeCreator: true,
|
||||||
includeResolvedBy: true,
|
includeResolvedBy: true,
|
||||||
@@ -134,14 +134,14 @@ export class CommentController {
|
|||||||
throw new NotFoundException('Page not found');
|
throw new NotFoundException('Page not found');
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.pageAccessService.validateCanEdit(page, user);
|
await this.pageAccessService.validateCanComment(page, user, workspace.id);
|
||||||
|
|
||||||
return this.commentService.update(comment, dto, user);
|
return this.commentService.update(comment, dto, user);
|
||||||
}
|
}
|
||||||
|
|
||||||
@HttpCode(HttpStatus.OK)
|
@HttpCode(HttpStatus.OK)
|
||||||
@Post('delete')
|
@Post('delete')
|
||||||
async delete(@Body() input: CommentIdDto, @AuthUser() user: User) {
|
async delete(@Body() input: CommentIdDto, @AuthUser() user: User, @AuthWorkspace() workspace: Workspace) {
|
||||||
const comment = await this.commentRepo.findById(input.commentId);
|
const comment = await this.commentRepo.findById(input.commentId);
|
||||||
if (!comment) {
|
if (!comment) {
|
||||||
throw new NotFoundException('Comment not found');
|
throw new NotFoundException('Comment not found');
|
||||||
@@ -152,8 +152,7 @@ export class CommentController {
|
|||||||
throw new NotFoundException('Page not found');
|
throw new NotFoundException('Page not found');
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check page-level edit permission first
|
await this.pageAccessService.validateCanComment(page, user, workspace.id);
|
||||||
await this.pageAccessService.validateCanEdit(page, user);
|
|
||||||
|
|
||||||
// Check if user is the comment owner
|
// Check if user is the comment owner
|
||||||
const isOwner = comment.creatorId === user.id;
|
const isOwner = comment.creatorId === user.id;
|
||||||
@@ -169,7 +168,7 @@ export class CommentController {
|
|||||||
// Space admin can delete any comment
|
// Space admin can delete any comment
|
||||||
if (ability.cannot(SpaceCaslAction.Manage, SpaceCaslSubject.Settings)) {
|
if (ability.cannot(SpaceCaslAction.Manage, SpaceCaslSubject.Settings)) {
|
||||||
throw new ForbiddenException(
|
throw new ForbiddenException(
|
||||||
'You can only delete your own comments or must be a space admin',
|
'You can only delete your own comments',
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
await this.commentRepo.deleteComment(comment.id);
|
await this.commentRepo.deleteComment(comment.id);
|
||||||
|
|||||||
@@ -1,8 +1,10 @@
|
|||||||
import { Module } from '@nestjs/common';
|
import { Module } from '@nestjs/common';
|
||||||
import { CommentService } from './comment.service';
|
import { CommentService } from './comment.service';
|
||||||
import { CommentController } from './comment.controller';
|
import { CommentController } from './comment.controller';
|
||||||
|
import { CollaborationModule } from '../../collaboration/collaboration.module';
|
||||||
|
|
||||||
@Module({
|
@Module({
|
||||||
|
imports: [CollaborationModule],
|
||||||
controllers: [CommentController],
|
controllers: [CommentController],
|
||||||
providers: [CommentService],
|
providers: [CommentService],
|
||||||
exports: [CommentService],
|
exports: [CommentService],
|
||||||
|
|||||||
@@ -7,7 +7,8 @@ import {
|
|||||||
} from '@nestjs/common';
|
} from '@nestjs/common';
|
||||||
import { InjectQueue } from '@nestjs/bullmq';
|
import { InjectQueue } from '@nestjs/bullmq';
|
||||||
import { Queue } from 'bullmq';
|
import { Queue } from 'bullmq';
|
||||||
import { CreateCommentDto } from './dto/create-comment.dto';
|
import { CreateCommentDto, yjsSelectionSchema } from './dto/create-comment.dto';
|
||||||
|
import { CollaborationGateway } from '../../collaboration/collaboration.gateway';
|
||||||
import { UpdateCommentDto } from './dto/update-comment.dto';
|
import { UpdateCommentDto } from './dto/update-comment.dto';
|
||||||
import { CommentRepo } from '@docmost/db/repos/comment/comment.repo';
|
import { CommentRepo } from '@docmost/db/repos/comment/comment.repo';
|
||||||
import { Comment, Page, User } from '@docmost/db/types/entity.types';
|
import { Comment, Page, User } from '@docmost/db/types/entity.types';
|
||||||
@@ -27,6 +28,7 @@ export class CommentService {
|
|||||||
private commentRepo: CommentRepo,
|
private commentRepo: CommentRepo,
|
||||||
private pageRepo: PageRepo,
|
private pageRepo: PageRepo,
|
||||||
private wsService: WsService,
|
private wsService: WsService,
|
||||||
|
private collaborationGateway: CollaborationGateway,
|
||||||
@InjectQueue(QueueName.GENERAL_QUEUE)
|
@InjectQueue(QueueName.GENERAL_QUEUE)
|
||||||
private generalQueue: Queue,
|
private generalQueue: Queue,
|
||||||
@InjectQueue(QueueName.NOTIFICATION_QUEUE)
|
@InjectQueue(QueueName.NOTIFICATION_QUEUE)
|
||||||
@@ -45,10 +47,10 @@ export class CommentService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async create(
|
async create(
|
||||||
opts: { userId: string; page: Page; workspaceId: string },
|
opts: { page: Page; workspaceId: string; user: User },
|
||||||
createCommentDto: CreateCommentDto,
|
createCommentDto: CreateCommentDto,
|
||||||
) {
|
) {
|
||||||
const { userId, page, workspaceId } = opts;
|
const { page, workspaceId, user } = opts;
|
||||||
const commentContent = JSON.parse(createCommentDto.content);
|
const commentContent = JSON.parse(createCommentDto.content);
|
||||||
|
|
||||||
if (createCommentDto.parentCommentId) {
|
if (createCommentDto.parentCommentId) {
|
||||||
@@ -71,11 +73,39 @@ export class CommentService {
|
|||||||
selection: createCommentDto?.selection?.substring(0, 250) ?? null,
|
selection: createCommentDto?.selection?.substring(0, 250) ?? null,
|
||||||
type: createCommentDto.type ?? 'page',
|
type: createCommentDto.type ?? 'page',
|
||||||
parentCommentId: createCommentDto?.parentCommentId,
|
parentCommentId: createCommentDto?.parentCommentId,
|
||||||
creatorId: userId,
|
creatorId: user.id,
|
||||||
workspaceId: workspaceId,
|
workspaceId: workspaceId,
|
||||||
spaceId: page.spaceId,
|
spaceId: page.spaceId,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (createCommentDto.yjsSelection) {
|
||||||
|
const parsed = yjsSelectionSchema.safeParse(createCommentDto.yjsSelection);
|
||||||
|
if (!parsed.success) {
|
||||||
|
this.logger.warn(
|
||||||
|
`Invalid yjsSelection for comment ${inserted.id}: ${parsed.error.message}`,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
const documentName = `page.${page.id}`;
|
||||||
|
try {
|
||||||
|
await this.collaborationGateway.handleYjsEvent(
|
||||||
|
'setCommentMark',
|
||||||
|
documentName,
|
||||||
|
{
|
||||||
|
yjsSelection: parsed.data,
|
||||||
|
commentId: inserted.id,
|
||||||
|
resolved: false,
|
||||||
|
user,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
this.logger.warn(
|
||||||
|
`Failed to apply comment mark for comment ${inserted.id}, comment saved without inline highlight`,
|
||||||
|
error,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const comment = await this.commentRepo.findById(inserted.id, {
|
const comment = await this.commentRepo.findById(inserted.id, {
|
||||||
includeCreator: true,
|
includeCreator: true,
|
||||||
includeResolvedBy: true,
|
includeResolvedBy: true,
|
||||||
@@ -83,7 +113,7 @@ export class CommentService {
|
|||||||
|
|
||||||
this.generalQueue
|
this.generalQueue
|
||||||
.add(QueueJob.ADD_PAGE_WATCHERS, {
|
.add(QueueJob.ADD_PAGE_WATCHERS, {
|
||||||
userIds: [userId],
|
userIds: [user.id],
|
||||||
pageId: page.id,
|
pageId: page.id,
|
||||||
spaceId: page.spaceId,
|
spaceId: page.spaceId,
|
||||||
workspaceId,
|
workspaceId,
|
||||||
@@ -101,7 +131,7 @@ export class CommentService {
|
|||||||
page.id,
|
page.id,
|
||||||
page.spaceId,
|
page.spaceId,
|
||||||
workspaceId,
|
workspaceId,
|
||||||
userId,
|
user.id,
|
||||||
!isReply,
|
!isReply,
|
||||||
createCommentDto.parentCommentId,
|
createCommentDto.parentCommentId,
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,4 +1,22 @@
|
|||||||
import { IsIn, IsJSON, IsOptional, IsString, IsUUID } from 'class-validator';
|
import { IsIn, IsJSON, IsObject, IsOptional, IsString, IsUUID } from 'class-validator';
|
||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
const yjsIdSchema = z.object({
|
||||||
|
client: z.number().int().nonnegative(),
|
||||||
|
clock: z.number().int().nonnegative(),
|
||||||
|
});
|
||||||
|
|
||||||
|
const yjsRelativePositionSchema = z.object({
|
||||||
|
type: yjsIdSchema,
|
||||||
|
tname: z.string().nullable(),
|
||||||
|
item: yjsIdSchema.nullable(),
|
||||||
|
assoc: z.number().int(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const yjsSelectionSchema = z.object({
|
||||||
|
anchor: yjsRelativePositionSchema,
|
||||||
|
head: yjsRelativePositionSchema,
|
||||||
|
});
|
||||||
|
|
||||||
export class CreateCommentDto {
|
export class CreateCommentDto {
|
||||||
@IsString()
|
@IsString()
|
||||||
@@ -18,4 +36,11 @@ export class CreateCommentDto {
|
|||||||
@IsOptional()
|
@IsOptional()
|
||||||
@IsUUID()
|
@IsUUID()
|
||||||
parentCommentId: string;
|
parentCommentId: string;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsObject()
|
||||||
|
yjsSelection?: {
|
||||||
|
anchor: any;
|
||||||
|
head: any;
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { IsArray, IsOptional, IsUUID } from 'class-validator';
|
import { IsArray, IsIn, IsOptional, IsString, IsUUID } from 'class-validator';
|
||||||
|
import { PaginationOptions } from '@docmost/db/pagination/pagination-options';
|
||||||
|
|
||||||
export class NotificationIdDto {
|
export class NotificationIdDto {
|
||||||
@IsUUID()
|
@IsUUID()
|
||||||
@@ -11,3 +12,10 @@ export class MarkNotificationsReadDto {
|
|||||||
@IsOptional()
|
@IsOptional()
|
||||||
notificationIds?: string[];
|
notificationIds?: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class ListNotificationsDto extends PaginationOptions {
|
||||||
|
@IsOptional()
|
||||||
|
@IsString()
|
||||||
|
@IsIn(['direct', 'updates', 'all'])
|
||||||
|
type?: 'direct' | 'updates' | 'all' = 'all';
|
||||||
|
}
|
||||||
|
|||||||
@@ -4,7 +4,45 @@ export const NotificationType = {
|
|||||||
COMMENT_RESOLVED: 'comment.resolved',
|
COMMENT_RESOLVED: 'comment.resolved',
|
||||||
PAGE_USER_MENTION: 'page.user_mention',
|
PAGE_USER_MENTION: 'page.user_mention',
|
||||||
PAGE_PERMISSION_GRANTED: 'page.permission_granted',
|
PAGE_PERMISSION_GRANTED: 'page.permission_granted',
|
||||||
|
PAGE_UPDATED: 'page.updated',
|
||||||
} as const;
|
} as const;
|
||||||
|
|
||||||
export type NotificationType =
|
export type NotificationType =
|
||||||
(typeof NotificationType)[keyof typeof NotificationType];
|
(typeof NotificationType)[keyof typeof NotificationType];
|
||||||
|
|
||||||
|
export type NotificationSettingKey =
|
||||||
|
| 'page.updated'
|
||||||
|
| 'page.userMention'
|
||||||
|
| 'comment.userMention'
|
||||||
|
| 'comment.created'
|
||||||
|
| 'comment.resolved';
|
||||||
|
|
||||||
|
export const NotificationTypeToSettingKey: Partial<
|
||||||
|
Record<NotificationType, NotificationSettingKey>
|
||||||
|
> = {
|
||||||
|
[NotificationType.PAGE_UPDATED]: 'page.updated',
|
||||||
|
[NotificationType.PAGE_USER_MENTION]: 'page.userMention',
|
||||||
|
[NotificationType.COMMENT_USER_MENTION]: 'comment.userMention',
|
||||||
|
[NotificationType.COMMENT_CREATED]: 'comment.created',
|
||||||
|
[NotificationType.COMMENT_RESOLVED]: 'comment.resolved',
|
||||||
|
};
|
||||||
|
|
||||||
|
export type NotificationTab = 'direct' | 'updates' | 'all';
|
||||||
|
|
||||||
|
export const DIRECT_NOTIFICATION_TYPES: NotificationType[] = [
|
||||||
|
NotificationType.COMMENT_USER_MENTION,
|
||||||
|
NotificationType.COMMENT_CREATED,
|
||||||
|
NotificationType.COMMENT_RESOLVED,
|
||||||
|
NotificationType.PAGE_USER_MENTION,
|
||||||
|
NotificationType.PAGE_PERMISSION_GRANTED,
|
||||||
|
];
|
||||||
|
|
||||||
|
export const UPDATES_NOTIFICATION_TYPES: NotificationType[] = [
|
||||||
|
NotificationType.PAGE_UPDATED,
|
||||||
|
];
|
||||||
|
|
||||||
|
export function getTypesForTab(tab: NotificationTab): NotificationType[] | undefined {
|
||||||
|
if (tab === 'direct') return DIRECT_NOTIFICATION_TYPES;
|
||||||
|
if (tab === 'updates') return UPDATES_NOTIFICATION_TYPES;
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|||||||
@@ -9,9 +9,8 @@ import {
|
|||||||
import { NotificationService } from './notification.service';
|
import { NotificationService } from './notification.service';
|
||||||
import { AuthUser } from '../../common/decorators/auth-user.decorator';
|
import { AuthUser } from '../../common/decorators/auth-user.decorator';
|
||||||
import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard';
|
import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard';
|
||||||
import { PaginationOptions } from '@docmost/db/pagination/pagination-options';
|
|
||||||
import { User } from '@docmost/db/types/entity.types';
|
import { User } from '@docmost/db/types/entity.types';
|
||||||
import { MarkNotificationsReadDto } from './dto/notification.dto';
|
import { ListNotificationsDto, MarkNotificationsReadDto } from './dto/notification.dto';
|
||||||
|
|
||||||
@UseGuards(JwtAuthGuard)
|
@UseGuards(JwtAuthGuard)
|
||||||
@Controller('notifications')
|
@Controller('notifications')
|
||||||
@@ -21,10 +20,10 @@ export class NotificationController {
|
|||||||
@HttpCode(HttpStatus.OK)
|
@HttpCode(HttpStatus.OK)
|
||||||
@Post('/')
|
@Post('/')
|
||||||
async getNotifications(
|
async getNotifications(
|
||||||
@Body() pagination: PaginationOptions,
|
@Body() dto: ListNotificationsDto,
|
||||||
@AuthUser() user: User,
|
@AuthUser() user: User,
|
||||||
) {
|
) {
|
||||||
return this.notificationService.findByUserId(user.id, pagination);
|
return this.notificationService.findByUserId(user.id, dto, dto.type);
|
||||||
}
|
}
|
||||||
|
|
||||||
@HttpCode(HttpStatus.OK)
|
@HttpCode(HttpStatus.OK)
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import { NotificationController } from './notification.controller';
|
|||||||
import { NotificationProcessor } from './notification.processor';
|
import { NotificationProcessor } from './notification.processor';
|
||||||
import { CommentNotificationService } from './services/comment.notification';
|
import { CommentNotificationService } from './services/comment.notification';
|
||||||
import { PageNotificationService } from './services/page.notification';
|
import { PageNotificationService } from './services/page.notification';
|
||||||
|
import { PageUpdateEmailRateLimiter } from './services/page-update-email-rate-limiter';
|
||||||
|
|
||||||
@Module({
|
@Module({
|
||||||
imports: [],
|
imports: [],
|
||||||
@@ -13,6 +14,7 @@ import { PageNotificationService } from './services/page.notification';
|
|||||||
NotificationProcessor,
|
NotificationProcessor,
|
||||||
CommentNotificationService,
|
CommentNotificationService,
|
||||||
PageNotificationService,
|
PageNotificationService,
|
||||||
|
PageUpdateEmailRateLimiter,
|
||||||
],
|
],
|
||||||
exports: [NotificationService],
|
exports: [NotificationService],
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ import {
|
|||||||
ICommentNotificationJob,
|
ICommentNotificationJob,
|
||||||
ICommentResolvedNotificationJob,
|
ICommentResolvedNotificationJob,
|
||||||
IPageMentionNotificationJob,
|
IPageMentionNotificationJob,
|
||||||
|
IPageUpdateNotificationJob,
|
||||||
IPermissionGrantedNotificationJob,
|
IPermissionGrantedNotificationJob,
|
||||||
} from '../../integrations/queue/constants/queue.interface';
|
} from '../../integrations/queue/constants/queue.interface';
|
||||||
import { CommentNotificationService } from './services/comment.notification';
|
import { CommentNotificationService } from './services/comment.notification';
|
||||||
@@ -35,6 +36,7 @@ export class NotificationProcessor
|
|||||||
| ICommentNotificationJob
|
| ICommentNotificationJob
|
||||||
| ICommentResolvedNotificationJob
|
| ICommentResolvedNotificationJob
|
||||||
| IPageMentionNotificationJob
|
| IPageMentionNotificationJob
|
||||||
|
| IPageUpdateNotificationJob
|
||||||
| IPermissionGrantedNotificationJob,
|
| IPermissionGrantedNotificationJob,
|
||||||
void
|
void
|
||||||
>,
|
>,
|
||||||
@@ -76,6 +78,20 @@ export class NotificationProcessor
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case QueueJob.PAGE_UPDATED: {
|
||||||
|
await this.pageNotificationService.processPageUpdate(
|
||||||
|
job.data as IPageUpdateNotificationJob,
|
||||||
|
appUrl,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case QueueJob.PAGE_UPDATE_DIGEST: {
|
||||||
|
const { userId } = job.data as unknown as { userId: string };
|
||||||
|
await this.pageNotificationService.processDigest(userId, appUrl);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
this.logger.warn(`Unknown notification job: ${job.name}`);
|
this.logger.warn(`Unknown notification job: ${job.name}`);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,8 @@ import { InsertableNotification } from '@docmost/db/types/entity.types';
|
|||||||
import { PaginationOptions } from '@docmost/db/pagination/pagination-options';
|
import { PaginationOptions } from '@docmost/db/pagination/pagination-options';
|
||||||
import { WsGateway } from '../../ws/ws.gateway';
|
import { WsGateway } from '../../ws/ws.gateway';
|
||||||
import { MailService } from '../../integrations/mail/mail.service';
|
import { MailService } from '../../integrations/mail/mail.service';
|
||||||
|
import { NotificationTab, NotificationType, NotificationTypeToSettingKey } from './notification.constants';
|
||||||
|
import { PagePermissionRepo } from '@docmost/db/repos/page/page-permission.repo';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class NotificationService {
|
export class NotificationService {
|
||||||
@@ -13,12 +15,23 @@ export class NotificationService {
|
|||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private readonly notificationRepo: NotificationRepo,
|
private readonly notificationRepo: NotificationRepo,
|
||||||
|
private readonly pagePermissionRepo: PagePermissionRepo,
|
||||||
private readonly wsGateway: WsGateway,
|
private readonly wsGateway: WsGateway,
|
||||||
private readonly mailService: MailService,
|
private readonly mailService: MailService,
|
||||||
@InjectKysely() private readonly db: KyselyDB,
|
@InjectKysely() private readonly db: KyselyDB,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
async create(data: InsertableNotification) {
|
async create(data: InsertableNotification) {
|
||||||
|
const user = await this.db
|
||||||
|
.selectFrom('users')
|
||||||
|
.select(['id'])
|
||||||
|
.where('id', '=', data.userId)
|
||||||
|
.where('deletedAt', 'is', null)
|
||||||
|
.where('deactivatedAt', 'is', null)
|
||||||
|
.executeTakeFirst();
|
||||||
|
|
||||||
|
if (!user) return null;
|
||||||
|
|
||||||
const notification = await this.notificationRepo.insert(data);
|
const notification = await this.notificationRepo.insert(data);
|
||||||
|
|
||||||
this.wsGateway.server
|
this.wsGateway.server
|
||||||
@@ -28,8 +41,35 @@ export class NotificationService {
|
|||||||
return notification;
|
return notification;
|
||||||
}
|
}
|
||||||
|
|
||||||
async findByUserId(userId: string, pagination: PaginationOptions) {
|
async findByUserId(
|
||||||
return this.notificationRepo.findByUserId(userId, pagination);
|
userId: string,
|
||||||
|
pagination: PaginationOptions,
|
||||||
|
type: NotificationTab = 'all',
|
||||||
|
) {
|
||||||
|
const result = await this.notificationRepo.findByUserId(
|
||||||
|
userId,
|
||||||
|
pagination,
|
||||||
|
type,
|
||||||
|
);
|
||||||
|
|
||||||
|
const pageIds = result.items
|
||||||
|
.map((n: any) => n.pageId)
|
||||||
|
.filter(Boolean);
|
||||||
|
|
||||||
|
if (pageIds.length > 0) {
|
||||||
|
const accessiblePageIds =
|
||||||
|
await this.pagePermissionRepo.filterAccessiblePageIds({
|
||||||
|
pageIds,
|
||||||
|
userId,
|
||||||
|
});
|
||||||
|
const accessibleSet = new Set(accessiblePageIds);
|
||||||
|
|
||||||
|
result.items = result.items.filter(
|
||||||
|
(n: any) => !n.pageId || accessibleSet.has(n.pageId),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getUnreadCount(userId: string) {
|
async getUnreadCount(userId: string) {
|
||||||
@@ -53,17 +93,27 @@ export class NotificationService {
|
|||||||
notificationId: string,
|
notificationId: string,
|
||||||
subject: string,
|
subject: string,
|
||||||
template: any,
|
template: any,
|
||||||
|
type?: NotificationType,
|
||||||
) {
|
) {
|
||||||
try {
|
try {
|
||||||
const user = await this.db
|
const user = await this.db
|
||||||
.selectFrom('users')
|
.selectFrom('users')
|
||||||
.select(['email'])
|
.select(['email', 'settings'])
|
||||||
.where('id', '=', userId)
|
.where('id', '=', userId)
|
||||||
.where('deletedAt', 'is', null)
|
.where('deletedAt', 'is', null)
|
||||||
|
.where('deactivatedAt', 'is', null)
|
||||||
.executeTakeFirst();
|
.executeTakeFirst();
|
||||||
|
|
||||||
if (!user?.email) return;
|
if (!user?.email) return;
|
||||||
|
|
||||||
|
if (type) {
|
||||||
|
const settingKey = NotificationTypeToSettingKey[type];
|
||||||
|
if (settingKey) {
|
||||||
|
const settings = user.settings as any;
|
||||||
|
if (settings?.notifications?.[settingKey] === false) return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
await this.mailService.sendToQueue({
|
await this.mailService.sendToQueue({
|
||||||
to: user.email,
|
to: user.email,
|
||||||
subject,
|
subject,
|
||||||
|
|||||||
@@ -86,12 +86,14 @@ export class CommentNotificationService {
|
|||||||
spaceId,
|
spaceId,
|
||||||
commentId,
|
commentId,
|
||||||
});
|
});
|
||||||
|
if (!notification) continue;
|
||||||
|
|
||||||
await this.notificationService.queueEmail(
|
await this.notificationService.queueEmail(
|
||||||
userId,
|
userId,
|
||||||
notification.id,
|
notification.id,
|
||||||
`${actor.name} mentioned you in a comment`,
|
`${actor.name} mentioned you in a comment`,
|
||||||
CommentMentionEmail({ actorName: actor.name, pageTitle, pageUrl }),
|
CommentMentionEmail({ actorName: actor.name, pageTitle, pageUrl }),
|
||||||
|
NotificationType.COMMENT_USER_MENTION,
|
||||||
);
|
);
|
||||||
|
|
||||||
notifiedUserIds.add(userId);
|
notifiedUserIds.add(userId);
|
||||||
@@ -110,12 +112,14 @@ export class CommentNotificationService {
|
|||||||
spaceId,
|
spaceId,
|
||||||
commentId,
|
commentId,
|
||||||
});
|
});
|
||||||
|
if (!notification) continue;
|
||||||
|
|
||||||
await this.notificationService.queueEmail(
|
await this.notificationService.queueEmail(
|
||||||
recipientId,
|
recipientId,
|
||||||
notification.id,
|
notification.id,
|
||||||
`${actor.name} commented on ${pageTitle}`,
|
`${actor.name} commented on ${pageTitle}`,
|
||||||
CommentCreateEmail({ actorName: actor.name, pageTitle, pageUrl }),
|
CommentCreateEmail({ actorName: actor.name, pageTitle, pageUrl }),
|
||||||
|
NotificationType.COMMENT_CREATED,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -171,6 +175,7 @@ export class CommentNotificationService {
|
|||||||
spaceId,
|
spaceId,
|
||||||
commentId,
|
commentId,
|
||||||
});
|
});
|
||||||
|
if (!notification) return;
|
||||||
|
|
||||||
const subject = `${actor.name} resolved a comment on ${pageTitle}`;
|
const subject = `${actor.name} resolved a comment on ${pageTitle}`;
|
||||||
|
|
||||||
@@ -179,6 +184,7 @@ export class CommentNotificationService {
|
|||||||
notification.id,
|
notification.id,
|
||||||
subject,
|
subject,
|
||||||
CommentResolvedEmail({ actorName: actor.name, pageTitle, pageUrl }),
|
CommentResolvedEmail({ actorName: actor.name, pageTitle, pageUrl }),
|
||||||
|
NotificationType.COMMENT_RESOLVED,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,43 @@
|
|||||||
|
import { Injectable } from '@nestjs/common';
|
||||||
|
import { RedisService } from '@nestjs-labs/nestjs-ioredis';
|
||||||
|
import type { Redis } from 'ioredis';
|
||||||
|
|
||||||
|
const KEY_PREFIX = 'page-update:emails:';
|
||||||
|
const DIGEST_PREFIX = 'page-update:digest:';
|
||||||
|
const TTL_SECONDS = 86400; // 24 hours
|
||||||
|
const MAX_IMMEDIATE_EMAILS = 4;
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class PageUpdateEmailRateLimiter {
|
||||||
|
private readonly redis: Redis;
|
||||||
|
|
||||||
|
constructor(private readonly redisService: RedisService) {
|
||||||
|
this.redis = this.redisService.getOrThrow();
|
||||||
|
}
|
||||||
|
|
||||||
|
async canSendEmail(userId: string): Promise<boolean> {
|
||||||
|
const key = KEY_PREFIX + userId;
|
||||||
|
const count = await this.redis.incr(key);
|
||||||
|
await this.redis.expire(key, TTL_SECONDS, 'NX');
|
||||||
|
return count <= MAX_IMMEDIATE_EMAILS;
|
||||||
|
}
|
||||||
|
|
||||||
|
async addToDigest(userId: string, notificationId: string): Promise<boolean> {
|
||||||
|
const key = DIGEST_PREFIX + userId;
|
||||||
|
const len = await this.redis.rpush(key, notificationId);
|
||||||
|
await this.redis.expire(key, TTL_SECONDS);
|
||||||
|
return len === 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
async popDigest(userId: string): Promise<string[]> {
|
||||||
|
const key = DIGEST_PREFIX + userId;
|
||||||
|
const [ids] = await this.redis
|
||||||
|
.multi()
|
||||||
|
.lrange(key, 0, -1)
|
||||||
|
.del(key)
|
||||||
|
.exec();
|
||||||
|
|
||||||
|
return (ids?.[1] as string[]) ?? [];
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -1,25 +1,43 @@
|
|||||||
import { Injectable } from '@nestjs/common';
|
import { Injectable, Logger } from '@nestjs/common';
|
||||||
import { InjectKysely } from 'nestjs-kysely';
|
import { InjectKysely } from 'nestjs-kysely';
|
||||||
|
import { InjectQueue } from '@nestjs/bullmq';
|
||||||
|
import { Queue } from 'bullmq';
|
||||||
import { KyselyDB } from '@docmost/db/types/kysely.types';
|
import { KyselyDB } from '@docmost/db/types/kysely.types';
|
||||||
import {
|
import {
|
||||||
IPageMentionNotificationJob,
|
IPageMentionNotificationJob,
|
||||||
|
IPageUpdateNotificationJob,
|
||||||
IPermissionGrantedNotificationJob,
|
IPermissionGrantedNotificationJob,
|
||||||
} from '../../../integrations/queue/constants/queue.interface';
|
} from '../../../integrations/queue/constants/queue.interface';
|
||||||
import { NotificationService } from '../notification.service';
|
import { NotificationService } from '../notification.service';
|
||||||
import { NotificationType } from '../notification.constants';
|
import { NotificationType } from '../notification.constants';
|
||||||
|
import { NotificationRepo } from '@docmost/db/repos/notification/notification.repo';
|
||||||
import { SpaceMemberRepo } from '@docmost/db/repos/space/space-member.repo';
|
import { SpaceMemberRepo } from '@docmost/db/repos/space/space-member.repo';
|
||||||
import { PagePermissionRepo } from '@docmost/db/repos/page/page-permission.repo';
|
import { PagePermissionRepo } from '@docmost/db/repos/page/page-permission.repo';
|
||||||
|
import { WatcherRepo } from '@docmost/db/repos/watcher/watcher.repo';
|
||||||
|
import { PageUpdateEmailRateLimiter } from './page-update-email-rate-limiter';
|
||||||
import { PageMentionEmail } from '@docmost/transactional/emails/page-mention-email';
|
import { PageMentionEmail } from '@docmost/transactional/emails/page-mention-email';
|
||||||
|
import { PageUpdateEmail } from '@docmost/transactional/emails/page-update-email';
|
||||||
|
import { PageUpdateDigestEmail } from '@docmost/transactional/emails/page-update-digest-email';
|
||||||
import { PermissionGrantedEmail } from '@docmost/transactional/emails/permission-granted-email';
|
import { PermissionGrantedEmail } from '@docmost/transactional/emails/permission-granted-email';
|
||||||
import { getPageTitle } from '../../../common/helpers';
|
import { getPageTitle } from '../../../common/helpers';
|
||||||
|
import { QueueJob, QueueName } from '../../../integrations/queue/constants';
|
||||||
|
|
||||||
|
const PAGE_UPDATE_COOLDOWN_HOURS = 7;
|
||||||
|
const DIGEST_DELAY_MS = 12 * 60 * 60 * 1000; // 12 hours
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class PageNotificationService {
|
export class PageNotificationService {
|
||||||
|
private readonly logger = new Logger(PageNotificationService.name);
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
@InjectKysely() private readonly db: KyselyDB,
|
@InjectKysely() private readonly db: KyselyDB,
|
||||||
private readonly notificationService: NotificationService,
|
private readonly notificationService: NotificationService,
|
||||||
|
private readonly notificationRepo: NotificationRepo,
|
||||||
private readonly spaceMemberRepo: SpaceMemberRepo,
|
private readonly spaceMemberRepo: SpaceMemberRepo,
|
||||||
private readonly pagePermissionRepo: PagePermissionRepo,
|
private readonly pagePermissionRepo: PagePermissionRepo,
|
||||||
|
private readonly watcherRepo: WatcherRepo,
|
||||||
|
private readonly rateLimiter: PageUpdateEmailRateLimiter,
|
||||||
|
@InjectQueue(QueueName.NOTIFICATION_QUEUE) private notificationQueue: Queue,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
async processPageMention(data: IPageMentionNotificationJob, appUrl: string) {
|
async processPageMention(data: IPageMentionNotificationJob, appUrl: string) {
|
||||||
@@ -41,10 +59,9 @@ export class PageNotificationService {
|
|||||||
);
|
);
|
||||||
|
|
||||||
const usersWithPageAccess =
|
const usersWithPageAccess =
|
||||||
await this.pagePermissionRepo.getUserIdsWithPageAccess(
|
await this.pagePermissionRepo.getUserIdsWithPageAccess(pageId, [
|
||||||
pageId,
|
...usersWithSpaceAccess,
|
||||||
[...usersWithSpaceAccess],
|
]);
|
||||||
);
|
|
||||||
const usersWithAccess = new Set(usersWithPageAccess);
|
const usersWithAccess = new Set(usersWithPageAccess);
|
||||||
|
|
||||||
const accessibleMentions = newMentions.filter((m) =>
|
const accessibleMentions = newMentions.filter((m) =>
|
||||||
@@ -97,6 +114,7 @@ export class PageNotificationService {
|
|||||||
spaceId,
|
spaceId,
|
||||||
data: { mentionId },
|
data: { mentionId },
|
||||||
});
|
});
|
||||||
|
if (!notification) continue;
|
||||||
|
|
||||||
const pageUrl = `${basePageUrl}`;
|
const pageUrl = `${basePageUrl}`;
|
||||||
const subject = `${actor.name} mentioned you in ${pageTitle}`;
|
const subject = `${actor.name} mentioned you in ${pageTitle}`;
|
||||||
@@ -106,6 +124,7 @@ export class PageNotificationService {
|
|||||||
notification.id,
|
notification.id,
|
||||||
subject,
|
subject,
|
||||||
PageMentionEmail({ actorName: actor.name, pageTitle, pageUrl }),
|
PageMentionEmail({ actorName: actor.name, pageTitle, pageUrl }),
|
||||||
|
NotificationType.PAGE_USER_MENTION,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -139,6 +158,7 @@ export class PageNotificationService {
|
|||||||
spaceId,
|
spaceId,
|
||||||
data: { role },
|
data: { role },
|
||||||
});
|
});
|
||||||
|
if (!notification) continue;
|
||||||
|
|
||||||
const subject = `${actor.name} gave you ${accessLabel} access to ${pageTitle}`;
|
const subject = `${actor.name} gave you ${accessLabel} access to ${pageTitle}`;
|
||||||
|
|
||||||
@@ -156,6 +176,232 @@ export class PageNotificationService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async processPageUpdate(data: IPageUpdateNotificationJob, appUrl: string) {
|
||||||
|
const { pageId, spaceId, workspaceId, actorIds } = data;
|
||||||
|
|
||||||
|
const watcherIds = await this.watcherRepo.getPageWatcherIds(pageId);
|
||||||
|
if (watcherIds.length === 0) return;
|
||||||
|
|
||||||
|
const actorSet = new Set(actorIds);
|
||||||
|
const candidateIds = watcherIds.filter((id) => !actorSet.has(id));
|
||||||
|
if (candidateIds.length === 0) return;
|
||||||
|
|
||||||
|
const eligibleUsers = await this.getEligiblePageUpdateUsers(candidateIds);
|
||||||
|
if (eligibleUsers.size === 0) return;
|
||||||
|
|
||||||
|
const afterPrefs = [...eligibleUsers.keys()];
|
||||||
|
|
||||||
|
const recentlyNotified =
|
||||||
|
await this.notificationRepo.getRecentlyNotifiedUserIds(
|
||||||
|
afterPrefs,
|
||||||
|
pageId,
|
||||||
|
NotificationType.PAGE_UPDATED,
|
||||||
|
PAGE_UPDATE_COOLDOWN_HOURS,
|
||||||
|
);
|
||||||
|
const afterCooldown = afterPrefs.filter((id) => !recentlyNotified.has(id));
|
||||||
|
if (afterCooldown.length === 0) return;
|
||||||
|
|
||||||
|
const usersWithSpaceAccess =
|
||||||
|
await this.spaceMemberRepo.getUserIdsWithSpaceAccess(
|
||||||
|
afterCooldown,
|
||||||
|
spaceId,
|
||||||
|
);
|
||||||
|
|
||||||
|
const usersWithPageAccess =
|
||||||
|
await this.pagePermissionRepo.getUserIdsWithPageAccess(pageId, [
|
||||||
|
...usersWithSpaceAccess,
|
||||||
|
]);
|
||||||
|
if (usersWithPageAccess.length === 0) return;
|
||||||
|
|
||||||
|
const recipientIds = new Set(usersWithPageAccess);
|
||||||
|
const actorId = actorIds[0];
|
||||||
|
|
||||||
|
const context = await this.getPageContext(actorId, pageId, spaceId, appUrl);
|
||||||
|
if (!context) return;
|
||||||
|
|
||||||
|
const { actor, pageTitle, basePageUrl } = context;
|
||||||
|
|
||||||
|
for (const userId of recipientIds) {
|
||||||
|
const notification = await this.notificationService.create({
|
||||||
|
userId,
|
||||||
|
workspaceId,
|
||||||
|
type: NotificationType.PAGE_UPDATED,
|
||||||
|
actorId,
|
||||||
|
pageId,
|
||||||
|
spaceId,
|
||||||
|
});
|
||||||
|
if (!notification) continue;
|
||||||
|
|
||||||
|
const canSend = await this.rateLimiter.canSendEmail(userId);
|
||||||
|
if (canSend) {
|
||||||
|
await this.notificationService.queueEmail(
|
||||||
|
userId,
|
||||||
|
notification.id,
|
||||||
|
`${actor.name} updated ${pageTitle}`,
|
||||||
|
PageUpdateEmail({
|
||||||
|
userName: eligibleUsers.get(userId) ?? '',
|
||||||
|
actorName: actor.name,
|
||||||
|
pageTitle,
|
||||||
|
pageUrl: basePageUrl,
|
||||||
|
}),
|
||||||
|
NotificationType.PAGE_UPDATED,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
const isFirst = await this.rateLimiter.addToDigest(
|
||||||
|
userId,
|
||||||
|
notification.id,
|
||||||
|
);
|
||||||
|
if (isFirst) {
|
||||||
|
await this.scheduleDigest(userId, workspaceId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async getEligiblePageUpdateUsers(
|
||||||
|
userIds: string[],
|
||||||
|
): Promise<Map<string, string>> {
|
||||||
|
if (userIds.length === 0) return new Map();
|
||||||
|
|
||||||
|
const users = await this.db
|
||||||
|
.selectFrom('users')
|
||||||
|
.select(['id', 'name', 'settings'])
|
||||||
|
.where('id', 'in', userIds)
|
||||||
|
.where('deletedAt', 'is', null)
|
||||||
|
.where('deactivatedAt', 'is', null)
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
const eligible = new Map<string, string>();
|
||||||
|
for (const u of users) {
|
||||||
|
const settings = u.settings as any;
|
||||||
|
if (settings?.notifications?.['page.updated'] !== false) {
|
||||||
|
eligible.set(u.id, u.name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return eligible;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async scheduleDigest(
|
||||||
|
userId: string,
|
||||||
|
workspaceId: string,
|
||||||
|
): Promise<void> {
|
||||||
|
await this.notificationQueue
|
||||||
|
.add(
|
||||||
|
QueueJob.PAGE_UPDATE_DIGEST,
|
||||||
|
{ userId, workspaceId },
|
||||||
|
{ delay: DIGEST_DELAY_MS, removeOnComplete: true },
|
||||||
|
)
|
||||||
|
.catch((err) => {
|
||||||
|
this.logger.error(
|
||||||
|
`Failed to schedule digest for ${userId}: ${err.message}`,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async processDigest(userId: string, appUrl: string): Promise<void> {
|
||||||
|
const notificationIds = await this.rateLimiter.popDigest(userId);
|
||||||
|
if (notificationIds.length === 0) return;
|
||||||
|
|
||||||
|
const [user, notifications] = await Promise.all([
|
||||||
|
this.db
|
||||||
|
.selectFrom('users')
|
||||||
|
.select(['id', 'name'])
|
||||||
|
.where('id', '=', userId)
|
||||||
|
.executeTakeFirst(),
|
||||||
|
this.db
|
||||||
|
.selectFrom('notifications')
|
||||||
|
.select(['id', 'pageId', 'actorId'])
|
||||||
|
.where('id', 'in', notificationIds)
|
||||||
|
.execute(),
|
||||||
|
]);
|
||||||
|
|
||||||
|
if (!user || notifications.length === 0) return;
|
||||||
|
|
||||||
|
const pageIds = [
|
||||||
|
...new Set(notifications.map((n) => n.pageId).filter(Boolean)),
|
||||||
|
];
|
||||||
|
const actorIds = [
|
||||||
|
...new Set(notifications.map((n) => n.actorId).filter(Boolean)),
|
||||||
|
];
|
||||||
|
|
||||||
|
const allPages = await this.db
|
||||||
|
.selectFrom('pages')
|
||||||
|
.innerJoin('spaces', 'spaces.id', 'pages.spaceId')
|
||||||
|
.select([
|
||||||
|
'pages.id',
|
||||||
|
'pages.title',
|
||||||
|
'pages.slugId',
|
||||||
|
'pages.spaceId',
|
||||||
|
'spaces.slug as spaceSlug',
|
||||||
|
])
|
||||||
|
.where('pages.id', 'in', pageIds)
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
if (allPages.length === 0) return;
|
||||||
|
|
||||||
|
const spaceIds = [...new Set(allPages.map((p) => p.spaceId))];
|
||||||
|
|
||||||
|
const accessibleSpaceIds = new Set<string>();
|
||||||
|
for (const spaceId of spaceIds) {
|
||||||
|
const usersWithAccess =
|
||||||
|
await this.spaceMemberRepo.getUserIdsWithSpaceAccess([userId], spaceId);
|
||||||
|
if (usersWithAccess.has(userId)) accessibleSpaceIds.add(spaceId);
|
||||||
|
}
|
||||||
|
|
||||||
|
const spaceFilteredPages = allPages.filter((p) =>
|
||||||
|
accessibleSpaceIds.has(p.spaceId),
|
||||||
|
);
|
||||||
|
if (spaceFilteredPages.length === 0) return;
|
||||||
|
|
||||||
|
const accessiblePageIds = new Set<string>();
|
||||||
|
for (const p of spaceFilteredPages) {
|
||||||
|
const hasAccess = await this.pagePermissionRepo.getUserIdsWithPageAccess(
|
||||||
|
p.id,
|
||||||
|
[userId],
|
||||||
|
);
|
||||||
|
if (hasAccess.includes(userId)) accessiblePageIds.add(p.id);
|
||||||
|
}
|
||||||
|
|
||||||
|
const pages = spaceFilteredPages.filter((p) => accessiblePageIds.has(p.id));
|
||||||
|
if (pages.length === 0) return;
|
||||||
|
|
||||||
|
const actors = actorIds.length > 0
|
||||||
|
? await this.db
|
||||||
|
.selectFrom('users')
|
||||||
|
.select(['id', 'name'])
|
||||||
|
.where('id', 'in', actorIds)
|
||||||
|
.execute()
|
||||||
|
: [];
|
||||||
|
|
||||||
|
const actorMap = new Map(actors.map((a) => [a.id, a.name]));
|
||||||
|
const pageActors = new Map<string, Set<string>>();
|
||||||
|
for (const n of notifications) {
|
||||||
|
if (!n.pageId || !n.actorId) continue;
|
||||||
|
const names = pageActors.get(n.pageId) ?? new Set();
|
||||||
|
const name = actorMap.get(n.actorId);
|
||||||
|
if (name) names.add(name);
|
||||||
|
pageActors.set(n.pageId, names);
|
||||||
|
}
|
||||||
|
|
||||||
|
const pageUpdates = pages.map((p) => ({
|
||||||
|
title: getPageTitle(p.title),
|
||||||
|
url: `${appUrl}/s/${p.spaceSlug}/p/${p.slugId}`,
|
||||||
|
updatedBy: [...(pageActors.get(p.id) ?? [])],
|
||||||
|
}));
|
||||||
|
|
||||||
|
await this.notificationService.queueEmail(
|
||||||
|
userId,
|
||||||
|
notificationIds[0],
|
||||||
|
`Your digest: ${pageUpdates.length} page ${pageUpdates.length === 1 ? 'update' : 'updates'}`,
|
||||||
|
PageUpdateDigestEmail({
|
||||||
|
userName: user.name,
|
||||||
|
pageUpdates,
|
||||||
|
totalUpdates: pageUpdates.length,
|
||||||
|
}),
|
||||||
|
NotificationType.PAGE_UPDATED,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
private async getPageContext(
|
private async getPageContext(
|
||||||
actorId: string,
|
actorId: string,
|
||||||
pageId: string,
|
pageId: string,
|
||||||
|
|||||||
@@ -6,12 +6,14 @@ import {
|
|||||||
SpaceCaslAction,
|
SpaceCaslAction,
|
||||||
SpaceCaslSubject,
|
SpaceCaslSubject,
|
||||||
} from '../../casl/interfaces/space-ability.type';
|
} from '../../casl/interfaces/space-ability.type';
|
||||||
|
import { SpaceRepo } from '@docmost/db/repos/space/space.repo';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class PageAccessService {
|
export class PageAccessService {
|
||||||
constructor(
|
constructor(
|
||||||
private readonly pagePermissionRepo: PagePermissionRepo,
|
private readonly pagePermissionRepo: PagePermissionRepo,
|
||||||
private readonly spaceAbility: SpaceAbilityFactory,
|
private readonly spaceAbility: SpaceAbilityFactory,
|
||||||
|
private readonly spaceRepo: SpaceRepo,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -99,4 +101,25 @@ export class PageAccessService {
|
|||||||
|
|
||||||
return { hasRestriction: hasAnyRestriction };
|
return { hasRestriction: hasAnyRestriction };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async validateCanComment(
|
||||||
|
page: Page,
|
||||||
|
user: User,
|
||||||
|
workspaceId: string,
|
||||||
|
): Promise<void> {
|
||||||
|
try {
|
||||||
|
await this.validateCanEdit(page, user);
|
||||||
|
return;
|
||||||
|
} catch {
|
||||||
|
// User cannot edit — check if reader commenting is enabled
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.validateCanView(page, user);
|
||||||
|
|
||||||
|
const space = await this.spaceRepo.findById(page.spaceId, workspaceId);
|
||||||
|
const settings = space?.settings as Record<string, any> | null;
|
||||||
|
if (!settings?.comments?.allowViewerComments) {
|
||||||
|
throw new ForbiddenException();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -47,6 +47,10 @@ import { QueueJob, QueueName } from '../../../integrations/queue/constants';
|
|||||||
import { EventName } from '../../../common/events/event.contants';
|
import { EventName } from '../../../common/events/event.contants';
|
||||||
import { EventEmitter2 } from '@nestjs/event-emitter';
|
import { EventEmitter2 } from '@nestjs/event-emitter';
|
||||||
import { CollaborationGateway } from '../../../collaboration/collaboration.gateway';
|
import { CollaborationGateway } from '../../../collaboration/collaboration.gateway';
|
||||||
|
import {
|
||||||
|
INTERNAL_LINK_REGEX,
|
||||||
|
extractPageSlugId,
|
||||||
|
} from '../../../integrations/export/utils';
|
||||||
import { markdownToHtml } from '@docmost/editor-ext';
|
import { markdownToHtml } from '@docmost/editor-ext';
|
||||||
import { WatcherService } from '../../watcher/watcher.service';
|
import { WatcherService } from '../../watcher/watcher.service';
|
||||||
import { sql } from 'kysely';
|
import { sql } from 'kysely';
|
||||||
@@ -510,6 +514,11 @@ export class PageService {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const slugIdMap = new Map<string, CopyPageMapEntry>();
|
||||||
|
for (const [, entry] of pageMap) {
|
||||||
|
slugIdMap.set(entry.oldSlugId, entry);
|
||||||
|
}
|
||||||
|
|
||||||
const attachmentMap = new Map<string, ICopyPageAttachment>();
|
const attachmentMap = new Map<string, ICopyPageAttachment>();
|
||||||
|
|
||||||
const insertablePages: InsertablePage[] = await Promise.all(
|
const insertablePages: InsertablePage[] = await Promise.all(
|
||||||
@@ -576,6 +585,28 @@ export class PageService {
|
|||||||
node.attrs.slugId = mappedPage.newSlugId;
|
node.attrs.slugId = mappedPage.newSlugId;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Update internal page links in link marks
|
||||||
|
for (const mark of node.marks) {
|
||||||
|
if (
|
||||||
|
mark.type.name === 'link' &&
|
||||||
|
mark.attrs.internal &&
|
||||||
|
mark.attrs.href
|
||||||
|
) {
|
||||||
|
const match = mark.attrs.href.match(INTERNAL_LINK_REGEX);
|
||||||
|
if (match) {
|
||||||
|
const slugId = extractPageSlugId(match[5]);
|
||||||
|
if (slugId && slugIdMap.has(slugId)) {
|
||||||
|
const mappedPage = slugIdMap.get(slugId);
|
||||||
|
//@ts-ignore
|
||||||
|
mark.attrs.href = mark.attrs.href.replace(
|
||||||
|
slugId,
|
||||||
|
mappedPage.newSlugId,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const prosemirrorJson = prosemirrorDoc.toJSON();
|
const prosemirrorJson = prosemirrorDoc.toJSON();
|
||||||
|
|||||||
@@ -11,4 +11,8 @@ export class UpdateSpaceDto extends PartialType(CreateSpaceDto) {
|
|||||||
@IsOptional()
|
@IsOptional()
|
||||||
@IsBoolean()
|
@IsBoolean()
|
||||||
disablePublicSharing: boolean;
|
disablePublicSharing: boolean;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsBoolean()
|
||||||
|
allowViewerComments: boolean;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import { Space, User } from '@docmost/db/types/entity.types';
|
|||||||
import { UpdateSpaceDto } from '../dto/update-space.dto';
|
import { UpdateSpaceDto } from '../dto/update-space.dto';
|
||||||
import { executeTx } from '@docmost/db/utils';
|
import { executeTx } from '@docmost/db/utils';
|
||||||
import { InjectKysely } from 'nestjs-kysely';
|
import { InjectKysely } from 'nestjs-kysely';
|
||||||
|
import { Feature } from '../../../common/features';
|
||||||
import { SpaceMemberService } from './space-member.service';
|
import { SpaceMemberService } from './space-member.service';
|
||||||
import { SpaceRole } from '../../../common/helpers/types/permission';
|
import { SpaceRole } from '../../../common/helpers/types/permission';
|
||||||
import { QueueJob, QueueName } from 'src/integrations/queue/constants';
|
import { QueueJob, QueueName } from 'src/integrations/queue/constants';
|
||||||
@@ -133,17 +134,34 @@ export class SpaceService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (typeof updateSpaceDto.disablePublicSharing !== 'undefined') {
|
if (
|
||||||
|
typeof updateSpaceDto.disablePublicSharing !== 'undefined' ||
|
||||||
|
typeof updateSpaceDto.allowViewerComments !== 'undefined'
|
||||||
|
) {
|
||||||
const workspace = await this.workspaceRepo.findById(workspaceId, {
|
const workspace = await this.workspaceRepo.findById(workspaceId, {
|
||||||
withLicenseKey: true,
|
withLicenseKey: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
if (
|
if (
|
||||||
!this.licenseCheckService.hasFeature(workspace.licenseKey, 'security:settings', workspace.plan)
|
typeof updateSpaceDto.disablePublicSharing !== 'undefined' &&
|
||||||
|
!this.licenseCheckService.hasFeature(
|
||||||
|
workspace.licenseKey,
|
||||||
|
Feature.SECURITY_SETTINGS,
|
||||||
|
workspace.plan,
|
||||||
|
)
|
||||||
) {
|
) {
|
||||||
throw new ForbiddenException(
|
throw new ForbiddenException('This feature requires a valid license');
|
||||||
'This feature requires a valid license',
|
}
|
||||||
);
|
|
||||||
|
if (
|
||||||
|
typeof updateSpaceDto.allowViewerComments !== 'undefined' &&
|
||||||
|
!this.licenseCheckService.hasFeature(
|
||||||
|
workspace.licenseKey,
|
||||||
|
Feature.VIEWER_COMMENTS,
|
||||||
|
workspace.plan,
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
throw new ForbiddenException('This feature requires a valid license');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -179,6 +197,22 @@ export class SpaceService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (typeof updateSpaceDto.allowViewerComments !== 'undefined') {
|
||||||
|
const prev = settingsBefore?.comments?.allowViewerComments ?? false;
|
||||||
|
if (prev !== updateSpaceDto.allowViewerComments) {
|
||||||
|
before.allowViewerComments = prev;
|
||||||
|
after.allowViewerComments = updateSpaceDto.allowViewerComments;
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.spaceRepo.updateCommentSettings(
|
||||||
|
updateSpaceDto.spaceId,
|
||||||
|
workspaceId,
|
||||||
|
'allowViewerComments',
|
||||||
|
updateSpaceDto.allowViewerComments,
|
||||||
|
trx,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
updatedSpace = await this.spaceRepo.updateSpace(
|
updatedSpace = await this.spaceRepo.updateSpace(
|
||||||
{
|
{
|
||||||
name: updateSpaceDto.name,
|
name: updateSpaceDto.name,
|
||||||
|
|||||||
@@ -35,4 +35,24 @@ export class UpdateUserDto extends PartialType(
|
|||||||
@MaxLength(70)
|
@MaxLength(70)
|
||||||
@IsString()
|
@IsString()
|
||||||
confirmPassword: string;
|
confirmPassword: string;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsBoolean()
|
||||||
|
notificationPageUpdates: boolean;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsBoolean()
|
||||||
|
notificationPageUserMention: boolean;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsBoolean()
|
||||||
|
notificationCommentUserMention: boolean;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsBoolean()
|
||||||
|
notificationCommentCreated: boolean;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@IsBoolean()
|
||||||
|
notificationCommentResolved: boolean;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import {
|
|||||||
UnauthorizedException,
|
UnauthorizedException,
|
||||||
} from '@nestjs/common';
|
} from '@nestjs/common';
|
||||||
import { UpdateUserDto } from './dto/update-user.dto';
|
import { UpdateUserDto } from './dto/update-user.dto';
|
||||||
|
import { NotificationSettingKey } from '../notification/notification.constants';
|
||||||
import { comparePasswordHash, diffAuditTrackedFields } from 'src/common/helpers/utils';
|
import { comparePasswordHash, diffAuditTrackedFields } from 'src/common/helpers/utils';
|
||||||
import { Workspace } from '@docmost/db/types/entity.types';
|
import { Workspace } from '@docmost/db/types/entity.types';
|
||||||
import { validateSsoEnforcement } from '../auth/auth.util';
|
import { validateSsoEnforcement } from '../auth/auth.util';
|
||||||
@@ -60,6 +61,24 @@ export class UserService {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const notificationSettings: Record<string, NotificationSettingKey> = {
|
||||||
|
notificationPageUpdates: 'page.updated',
|
||||||
|
notificationPageUserMention: 'page.userMention',
|
||||||
|
notificationCommentUserMention: 'comment.userMention',
|
||||||
|
notificationCommentCreated: 'comment.created',
|
||||||
|
notificationCommentResolved: 'comment.resolved',
|
||||||
|
};
|
||||||
|
|
||||||
|
for (const [dtoField, settingKey] of Object.entries(notificationSettings)) {
|
||||||
|
if (typeof updateUserDto[dtoField] !== 'undefined') {
|
||||||
|
return this.userRepo.updateNotificationSetting(
|
||||||
|
userId,
|
||||||
|
settingKey,
|
||||||
|
updateUserDto[dtoField],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const userBefore = { name: user.name, email: user.email, locale: user.locale };
|
const userBefore = { name: user.name, email: user.email, locale: user.locale };
|
||||||
|
|
||||||
if (updateUserDto.name) {
|
if (updateUserDto.name) {
|
||||||
|
|||||||
@@ -1,8 +1,6 @@
|
|||||||
/***
|
import {
|
||||||
import {
|
|
||||||
Body,
|
Body,
|
||||||
Controller,
|
Controller,
|
||||||
ForbiddenException,
|
|
||||||
HttpCode,
|
HttpCode,
|
||||||
HttpStatus,
|
HttpStatus,
|
||||||
NotFoundException,
|
NotFoundException,
|
||||||
@@ -16,12 +14,7 @@ import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard';
|
|||||||
import { User, Workspace } from '@docmost/db/types/entity.types';
|
import { User, Workspace } from '@docmost/db/types/entity.types';
|
||||||
import { WatcherPageDto } from './dto/watcher.dto';
|
import { WatcherPageDto } from './dto/watcher.dto';
|
||||||
import { PageRepo } from '@docmost/db/repos/page/page.repo';
|
import { PageRepo } from '@docmost/db/repos/page/page.repo';
|
||||||
import SpaceAbilityFactory from '../casl/abilities/space-ability.factory';
|
import { PageAccessService } from '../page/page-access/page-access.service';
|
||||||
import {
|
|
||||||
SpaceCaslAction,
|
|
||||||
SpaceCaslSubject,
|
|
||||||
} from '../casl/interfaces/space-ability.type';
|
|
||||||
|
|
||||||
|
|
||||||
@UseGuards(JwtAuthGuard)
|
@UseGuards(JwtAuthGuard)
|
||||||
@Controller('pages')
|
@Controller('pages')
|
||||||
@@ -29,7 +22,7 @@ export class WatcherController {
|
|||||||
constructor(
|
constructor(
|
||||||
private readonly watcherService: WatcherService,
|
private readonly watcherService: WatcherService,
|
||||||
private readonly pageRepo: PageRepo,
|
private readonly pageRepo: PageRepo,
|
||||||
private readonly spaceAbility: SpaceAbilityFactory,
|
private readonly pageAccessService: PageAccessService,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
@HttpCode(HttpStatus.OK)
|
@HttpCode(HttpStatus.OK)
|
||||||
@@ -44,10 +37,7 @@ export class WatcherController {
|
|||||||
throw new NotFoundException('Page not found');
|
throw new NotFoundException('Page not found');
|
||||||
}
|
}
|
||||||
|
|
||||||
const ability = await this.spaceAbility.createForUser(user, page.spaceId);
|
await this.pageAccessService.validateCanView(page, user);
|
||||||
if (ability.cannot(SpaceCaslAction.Read, SpaceCaslSubject.Page)) {
|
|
||||||
throw new ForbiddenException();
|
|
||||||
}
|
|
||||||
|
|
||||||
await this.watcherService.watchPage(
|
await this.watcherService.watchPage(
|
||||||
user.id,
|
user.id,
|
||||||
@@ -67,10 +57,7 @@ export class WatcherController {
|
|||||||
throw new NotFoundException('Page not found');
|
throw new NotFoundException('Page not found');
|
||||||
}
|
}
|
||||||
|
|
||||||
const ability = await this.spaceAbility.createForUser(user, page.spaceId);
|
await this.pageAccessService.validateCanView(page, user);
|
||||||
if (ability.cannot(SpaceCaslAction.Read, SpaceCaslSubject.Page)) {
|
|
||||||
throw new ForbiddenException();
|
|
||||||
}
|
|
||||||
|
|
||||||
await this.watcherService.unwatchPage(user.id, page.id);
|
await this.watcherService.unwatchPage(user.id, page.id);
|
||||||
|
|
||||||
@@ -85,15 +72,10 @@ export class WatcherController {
|
|||||||
throw new NotFoundException('Page not found');
|
throw new NotFoundException('Page not found');
|
||||||
}
|
}
|
||||||
|
|
||||||
const ability = await this.spaceAbility.createForUser(user, page.spaceId);
|
await this.pageAccessService.validateCanView(page, user);
|
||||||
if (ability.cannot(SpaceCaslAction.Read, SpaceCaslSubject.Page)) {
|
|
||||||
throw new ForbiddenException();
|
|
||||||
}
|
|
||||||
|
|
||||||
const watching = await this.watcherService.isWatchingPage(user.id, page.id);
|
const watching = await this.watcherService.isWatchingPage(user.id, page.id);
|
||||||
|
|
||||||
return { watching };
|
return { watching };
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
***/
|
|
||||||
|
|||||||
@@ -1,10 +1,11 @@
|
|||||||
import { Module } from '@nestjs/common';
|
import { Module } from '@nestjs/common';
|
||||||
import { WatcherService } from './watcher.service';
|
import { WatcherService } from './watcher.service';
|
||||||
import { CaslModule } from '../casl/casl.module';
|
import { WatcherController } from './watcher.controller';
|
||||||
|
import { PageAccessModule } from '../page/page-access/page-access.module';
|
||||||
|
|
||||||
@Module({
|
@Module({
|
||||||
imports: [CaslModule],
|
imports: [PageAccessModule],
|
||||||
controllers: [],
|
controllers: [WatcherController],
|
||||||
providers: [WatcherService],
|
providers: [WatcherService],
|
||||||
exports: [WatcherService],
|
exports: [WatcherService],
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ import { WorkspaceRepo } from '@docmost/db/repos/workspace/workspace.repo';
|
|||||||
import { KyselyDB, KyselyTransaction } from '@docmost/db/types/kysely.types';
|
import { KyselyDB, KyselyTransaction } from '@docmost/db/types/kysely.types';
|
||||||
import { executeTx } from '@docmost/db/utils';
|
import { executeTx } from '@docmost/db/utils';
|
||||||
import { InjectKysely } from 'nestjs-kysely';
|
import { InjectKysely } from 'nestjs-kysely';
|
||||||
|
import { Feature } from '../../../common/features';
|
||||||
import { User } from '@docmost/db/types/entity.types';
|
import { User } from '@docmost/db/types/entity.types';
|
||||||
import { GroupUserRepo } from '@docmost/db/repos/group/group-user.repo';
|
import { GroupUserRepo } from '@docmost/db/repos/group/group-user.repo';
|
||||||
import { GroupRepo } from '@docmost/db/repos/group/group.repo';
|
import { GroupRepo } from '@docmost/db/repos/group/group.repo';
|
||||||
@@ -352,7 +353,7 @@ export class WorkspaceService {
|
|||||||
typeof updateWorkspaceDto.trashRetentionDays !== 'undefined' ||
|
typeof updateWorkspaceDto.trashRetentionDays !== 'undefined' ||
|
||||||
typeof updateWorkspaceDto.restrictApiToAdmins !== 'undefined'
|
typeof updateWorkspaceDto.restrictApiToAdmins !== 'undefined'
|
||||||
) {
|
) {
|
||||||
if (!this.licenseCheckService.hasFeature(ws.licenseKey, 'security:settings', ws.plan)) {
|
if (!this.licenseCheckService.hasFeature(ws.licenseKey, Feature.SECURITY_SETTINGS, ws.plan)) {
|
||||||
throw new ForbiddenException(
|
throw new ForbiddenException(
|
||||||
'This feature requires a valid license',
|
'This feature requires a valid license',
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -0,0 +1,333 @@
|
|||||||
|
import { type Kysely, sql } from 'kysely';
|
||||||
|
|
||||||
|
export async function up(db: Kysely<any>): Promise<void> {
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_group_users_user_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('group_users')
|
||||||
|
.column('user_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_space_members_user_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('space_members')
|
||||||
|
.column('user_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_space_members_group_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('space_members')
|
||||||
|
.column('group_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
// Page tree
|
||||||
|
await sql`
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_pages_space_parent_position
|
||||||
|
ON pages (space_id, parent_page_id, position COLLATE "C")
|
||||||
|
WHERE deleted_at IS NULL
|
||||||
|
`.execute(db);
|
||||||
|
|
||||||
|
await sql`
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_pages_parent_page_id
|
||||||
|
ON pages (parent_page_id)
|
||||||
|
WHERE deleted_at IS NULL
|
||||||
|
`.execute(db);
|
||||||
|
|
||||||
|
// Recent pages query
|
||||||
|
await sql`
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_pages_space_updated
|
||||||
|
ON pages (space_id, updated_at DESC)
|
||||||
|
WHERE deleted_at IS NULL
|
||||||
|
`.execute(db);
|
||||||
|
|
||||||
|
// Trash view
|
||||||
|
await sql`
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_pages_space_deleted
|
||||||
|
ON pages (space_id, deleted_at DESC)
|
||||||
|
WHERE deleted_at IS NOT NULL
|
||||||
|
`.execute(db);
|
||||||
|
|
||||||
|
await sql`
|
||||||
|
CREATE UNIQUE INDEX IF NOT EXISTS idx_workspaces_hostname_lower
|
||||||
|
ON workspaces (LOWER(hostname))
|
||||||
|
`.execute(db);
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_workspaces_created_at')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('workspaces')
|
||||||
|
.column('created_at')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_users_workspace_deleted')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('users')
|
||||||
|
.columns(['workspace_id', 'deleted_at'])
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await sql`
|
||||||
|
CREATE UNIQUE INDEX IF NOT EXISTS idx_spaces_slug_lower_workspace
|
||||||
|
ON spaces (LOWER(slug), workspace_id)
|
||||||
|
`.execute(db);
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_spaces_workspace_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('spaces')
|
||||||
|
.column('workspace_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await sql`
|
||||||
|
CREATE UNIQUE INDEX IF NOT EXISTS idx_groups_name_lower_workspace
|
||||||
|
ON groups (LOWER(name), workspace_id)
|
||||||
|
`.execute(db);
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_groups_workspace_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('groups')
|
||||||
|
.column('workspace_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_shares_page_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('shares')
|
||||||
|
.column('page_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_attachments_page_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('attachments')
|
||||||
|
.column('page_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_attachments_space_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('attachments')
|
||||||
|
.column('space_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_comments_page_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('comments')
|
||||||
|
.column('page_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_comments_parent_comment_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('comments')
|
||||||
|
.column('parent_comment_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await sql`
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_page_history_page_created
|
||||||
|
ON page_history (page_id, created_at DESC)
|
||||||
|
`.execute(db);
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_attachments_workspace_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('attachments')
|
||||||
|
.column('workspace_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_backlinks_target_page_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('backlinks')
|
||||||
|
.column('target_page_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_pages_workspace_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('pages')
|
||||||
|
.column('workspace_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_pages_creator_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('pages')
|
||||||
|
.column('creator_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
// Notifications: FK cascade from pages, spaces, comments
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_notifications_page_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('notifications')
|
||||||
|
.column('page_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_notifications_space_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('notifications')
|
||||||
|
.column('space_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_notifications_comment_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('notifications')
|
||||||
|
.column('comment_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
// Watchers: cleanup queries and FK cascade
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_watchers_user_workspace')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('watchers')
|
||||||
|
.columns(['user_id', 'workspace_id'])
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_watchers_space_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('watchers')
|
||||||
|
.column('space_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
// Auth providers: all queries filter by workspaceId
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_auth_providers_workspace_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('auth_providers')
|
||||||
|
.column('workspace_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
// Auth accounts: SSO login lookup by provider user
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_auth_accounts_provider_user_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('auth_accounts')
|
||||||
|
.columns(['provider_user_id', 'auth_provider_id'])
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
// Workspace invitations: listing and SSO lookup
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_workspace_invitations_workspace_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('workspace_invitations')
|
||||||
|
.column('workspace_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
// API keys: query and FK cascade
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_api_keys_workspace_id')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('api_keys')
|
||||||
|
.column('workspace_id')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
// User sessions: delete queries and FK cascade on all session states
|
||||||
|
await db.schema
|
||||||
|
.createIndex('idx_user_sessions_user_workspace')
|
||||||
|
.ifNotExists()
|
||||||
|
.on('user_sessions')
|
||||||
|
.columns(['user_id', 'workspace_id'])
|
||||||
|
.execute();
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function down(db: Kysely<any>): Promise<void> {
|
||||||
|
await db.schema.dropIndex('idx_group_users_user_id').ifExists().execute();
|
||||||
|
await db.schema.dropIndex('idx_space_members_user_id').ifExists().execute();
|
||||||
|
await db.schema.dropIndex('idx_space_members_group_id').ifExists().execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_pages_space_parent_position')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema.dropIndex('idx_pages_parent_page_id').ifExists().execute();
|
||||||
|
await db.schema.dropIndex('idx_pages_space_updated').ifExists().execute();
|
||||||
|
await db.schema.dropIndex('idx_pages_space_deleted').ifExists().execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_workspaces_hostname_lower')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema.dropIndex('idx_workspaces_created_at').ifExists().execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_users_workspace_deleted')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_spaces_slug_lower_workspace')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_spaces_workspace_id')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_groups_name_lower_workspace')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema.dropIndex('idx_groups_workspace_id').ifExists().execute();
|
||||||
|
await db.schema.dropIndex('idx_shares_page_id').ifExists().execute();
|
||||||
|
await db.schema.dropIndex('idx_attachments_page_id').ifExists().execute();
|
||||||
|
await db.schema.dropIndex('idx_attachments_space_id').ifExists().execute();
|
||||||
|
await db.schema.dropIndex('idx_comments_page_id').ifExists().execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_comments_parent_comment_id')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_page_history_page_created')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_attachments_workspace_id')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_backlinks_target_page_id')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema.dropIndex('idx_pages_workspace_id').ifExists().execute();
|
||||||
|
await db.schema.dropIndex('idx_pages_creator_id').ifExists().execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_notifications_page_id')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_notifications_space_id')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_notifications_comment_id')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_watchers_user_workspace')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema.dropIndex('idx_watchers_space_id').ifExists().execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_auth_providers_workspace_id')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_auth_accounts_provider_user_id')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_workspace_invitations_workspace_id')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_api_keys_workspace_id')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
await db.schema
|
||||||
|
.dropIndex('idx_user_sessions_user_workspace')
|
||||||
|
.ifExists()
|
||||||
|
.execute();
|
||||||
|
}
|
||||||
@@ -11,6 +11,7 @@ import { ExpressionBuilder } from 'kysely';
|
|||||||
import { DB } from '@docmost/db/types/db';
|
import { DB } from '@docmost/db/types/db';
|
||||||
import { jsonObjectFrom } from 'kysely/helpers/postgres';
|
import { jsonObjectFrom } from 'kysely/helpers/postgres';
|
||||||
import { SpaceMemberRepo } from '@docmost/db/repos/space/space-member.repo';
|
import { SpaceMemberRepo } from '@docmost/db/repos/space/space-member.repo';
|
||||||
|
import { NotificationTab, NotificationType } from '../../../core/notification/notification.constants';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class NotificationRepo {
|
export class NotificationRepo {
|
||||||
@@ -27,8 +28,12 @@ export class NotificationRepo {
|
|||||||
.executeTakeFirst();
|
.executeTakeFirst();
|
||||||
}
|
}
|
||||||
|
|
||||||
async findByUserId(userId: string, pagination: PaginationOptions) {
|
async findByUserId(
|
||||||
const query = this.db
|
userId: string,
|
||||||
|
pagination: PaginationOptions,
|
||||||
|
type: NotificationTab = 'all',
|
||||||
|
) {
|
||||||
|
let query = this.db
|
||||||
.selectFrom('notifications')
|
.selectFrom('notifications')
|
||||||
.selectAll('notifications')
|
.selectAll('notifications')
|
||||||
.select((eb) => this.withActor(eb))
|
.select((eb) => this.withActor(eb))
|
||||||
@@ -42,6 +47,12 @@ export class NotificationRepo {
|
|||||||
]),
|
]),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if (type === 'direct') {
|
||||||
|
query = query.where('type', '!=', NotificationType.PAGE_UPDATED);
|
||||||
|
} else if (type === 'updates') {
|
||||||
|
query = query.where('type', '=', NotificationType.PAGE_UPDATED);
|
||||||
|
}
|
||||||
|
|
||||||
return executeWithCursorPagination(query, {
|
return executeWithCursorPagination(query, {
|
||||||
perPage: pagination.limit,
|
perPage: pagination.limit,
|
||||||
cursor: pagination.cursor,
|
cursor: pagination.cursor,
|
||||||
@@ -138,6 +149,29 @@ export class NotificationRepo {
|
|||||||
.execute();
|
.execute();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async getRecentlyNotifiedUserIds(
|
||||||
|
userIds: string[],
|
||||||
|
pageId: string,
|
||||||
|
type: string,
|
||||||
|
withinHours: number,
|
||||||
|
): Promise<Set<string>> {
|
||||||
|
if (userIds.length === 0) return new Set();
|
||||||
|
|
||||||
|
const cutoff = new Date(Date.now() - withinHours * 60 * 60 * 1000);
|
||||||
|
|
||||||
|
const rows = await this.db
|
||||||
|
.selectFrom('notifications')
|
||||||
|
.select('userId')
|
||||||
|
.where('userId', 'in', userIds)
|
||||||
|
.where('pageId', '=', pageId)
|
||||||
|
.where('type', '=', type)
|
||||||
|
.where('createdAt', '>', cutoff)
|
||||||
|
.groupBy('userId')
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
return new Set(rows.map((r) => r.userId));
|
||||||
|
}
|
||||||
|
|
||||||
withActor(eb: ExpressionBuilder<DB, 'notifications'>) {
|
withActor(eb: ExpressionBuilder<DB, 'notifications'>) {
|
||||||
return jsonObjectFrom(
|
return jsonObjectFrom(
|
||||||
eb
|
eb
|
||||||
|
|||||||
@@ -111,6 +111,28 @@ export class SpaceRepo {
|
|||||||
.executeTakeFirst();
|
.executeTakeFirst();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async updateCommentSettings(
|
||||||
|
spaceId: string,
|
||||||
|
workspaceId: string,
|
||||||
|
prefKey: string,
|
||||||
|
prefValue: string | boolean,
|
||||||
|
trx?: KyselyTransaction,
|
||||||
|
) {
|
||||||
|
const db = dbOrTx(this.db, trx);
|
||||||
|
return db
|
||||||
|
.updateTable('spaces')
|
||||||
|
.set({
|
||||||
|
settings: sql`COALESCE(settings, '{}'::jsonb)
|
||||||
|
|| jsonb_build_object('comments', COALESCE(settings->'comments', '{}'::jsonb)
|
||||||
|
|| jsonb_build_object('${sql.raw(prefKey)}', ${sql.lit(prefValue)}))`,
|
||||||
|
updatedAt: new Date(),
|
||||||
|
})
|
||||||
|
.where('id', '=', spaceId)
|
||||||
|
.where('workspaceId', '=', workspaceId)
|
||||||
|
.returningAll()
|
||||||
|
.executeTakeFirst();
|
||||||
|
}
|
||||||
|
|
||||||
async insertSpace(
|
async insertSpace(
|
||||||
insertableSpace: InsertableSpace,
|
insertableSpace: InsertableSpace,
|
||||||
trx?: KyselyTransaction,
|
trx?: KyselyTransaction,
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import { PaginationOptions } from '../../pagination/pagination-options';
|
|||||||
import { executeWithCursorPagination } from '@docmost/db/pagination/cursor-pagination';
|
import { executeWithCursorPagination } from '@docmost/db/pagination/cursor-pagination';
|
||||||
import { ExpressionBuilder, sql } from 'kysely';
|
import { ExpressionBuilder, sql } from 'kysely';
|
||||||
import { jsonObjectFrom } from 'kysely/helpers/postgres';
|
import { jsonObjectFrom } from 'kysely/helpers/postgres';
|
||||||
|
import { NotificationSettingKey } from '../../../core/notification/notification.constants';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class UserRepo {
|
export class UserRepo {
|
||||||
@@ -191,6 +192,24 @@ export class UserRepo {
|
|||||||
.executeTakeFirst();
|
.executeTakeFirst();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async updateNotificationSetting(
|
||||||
|
userId: string,
|
||||||
|
settingKey: NotificationSettingKey,
|
||||||
|
settingValue: boolean,
|
||||||
|
) {
|
||||||
|
return await this.db
|
||||||
|
.updateTable('users')
|
||||||
|
.set({
|
||||||
|
settings: sql`COALESCE(settings, '{}'::jsonb)
|
||||||
|
|| jsonb_build_object('notifications', COALESCE(settings->'notifications', '{}'::jsonb)
|
||||||
|
|| jsonb_build_object(${sql.lit(settingKey)}, ${sql.lit(settingValue)}))`,
|
||||||
|
updatedAt: new Date(),
|
||||||
|
})
|
||||||
|
.where('id', '=', userId)
|
||||||
|
.returning(this.baseFields)
|
||||||
|
.executeTakeFirst();
|
||||||
|
}
|
||||||
|
|
||||||
withUserMfa(eb: ExpressionBuilder<DB, 'users'>) {
|
withUserMfa(eb: ExpressionBuilder<DB, 'users'>) {
|
||||||
return jsonObjectFrom(
|
return jsonObjectFrom(
|
||||||
eb
|
eb
|
||||||
|
|||||||
+1
-1
Submodule apps/server/src/ee updated: a258ca3660...38158a5ab6
@@ -259,6 +259,12 @@ export class EnvironmentService {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getAiEmbeddingSupportsMrl(): boolean | undefined {
|
||||||
|
const val = this.configService.get<string>('AI_EMBEDDING_SUPPORTS_MRL');
|
||||||
|
if (val === undefined || val === null || val === '') return undefined;
|
||||||
|
return val === 'true';
|
||||||
|
}
|
||||||
|
|
||||||
getOpenAiApiKey(): string {
|
getOpenAiApiKey(): string {
|
||||||
return this.configService.get<string>('OPENAI_API_KEY');
|
return this.configService.get<string>('OPENAI_API_KEY');
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -117,6 +117,12 @@ export class EnvironmentVariables {
|
|||||||
@IsString()
|
@IsString()
|
||||||
AI_EMBEDDING_DIMENSION: string;
|
AI_EMBEDDING_DIMENSION: string;
|
||||||
|
|
||||||
|
@IsOptional()
|
||||||
|
@ValidateIf((obj) => obj.AI_EMBEDDING_SUPPORTS_MRL)
|
||||||
|
@IsIn(['true', 'false'])
|
||||||
|
@IsString()
|
||||||
|
AI_EMBEDDING_SUPPORTS_MRL: string;
|
||||||
|
|
||||||
@ValidateIf((obj) => obj.AI_DRIVER)
|
@ValidateIf((obj) => obj.AI_DRIVER)
|
||||||
@IsString()
|
@IsString()
|
||||||
@IsNotEmpty()
|
@IsNotEmpty()
|
||||||
|
|||||||
@@ -61,7 +61,7 @@ export class ExportController {
|
|||||||
|
|
||||||
await this.pageAccessService.validateCanView(page, user);
|
await this.pageAccessService.validateCanView(page, user);
|
||||||
|
|
||||||
const zipFileStream = await this.exportService.exportPages(
|
const result = await this.exportService.exportPages(
|
||||||
dto.pageId,
|
dto.pageId,
|
||||||
dto.format,
|
dto.format,
|
||||||
dto.includeAttachments,
|
dto.includeAttachments,
|
||||||
@@ -83,15 +83,29 @@ export class ExportController {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const fileName = sanitize(page.title || 'untitled') + '.zip';
|
if (result.type === 'file') {
|
||||||
|
const ext = getExportExtension(dto.format);
|
||||||
|
const fileName = sanitize(page.title || 'untitled') + ext;
|
||||||
|
const contentType = getMimeType(path.extname(fileName));
|
||||||
|
|
||||||
res.headers({
|
res.headers({
|
||||||
'Content-Type': 'application/zip',
|
'Content-Type': contentType,
|
||||||
'Content-Disposition':
|
'Content-Disposition':
|
||||||
'attachment; filename="' + encodeURIComponent(fileName) + '"',
|
'attachment; filename="' + encodeURIComponent(fileName) + '"',
|
||||||
});
|
});
|
||||||
|
|
||||||
res.send(zipFileStream);
|
res.send(result.content);
|
||||||
|
} else {
|
||||||
|
const fileName = sanitize(page.title || 'untitled') + '.zip';
|
||||||
|
|
||||||
|
res.headers({
|
||||||
|
'Content-Type': 'application/zip',
|
||||||
|
'Content-Disposition':
|
||||||
|
'attachment; filename="' + encodeURIComponent(fileName) + '"',
|
||||||
|
});
|
||||||
|
|
||||||
|
res.send(result.stream);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@UseGuards(JwtAuthGuard)
|
@UseGuards(JwtAuthGuard)
|
||||||
|
|||||||
@@ -150,6 +150,13 @@ export class ExportService {
|
|||||||
// set to null to make export of pages with parentId work
|
// set to null to make export of pages with parentId work
|
||||||
pages[parentPageIndex].parentPageId = null;
|
pages[parentPageIndex].parentPageId = null;
|
||||||
|
|
||||||
|
const isSinglePage = pages.length === 1 && !includeAttachments;
|
||||||
|
|
||||||
|
if (isSinglePage) {
|
||||||
|
const pageContent = await this.exportPage(format, pages[0], true);
|
||||||
|
return { type: 'file' as const, content: pageContent, page: pages[0] };
|
||||||
|
}
|
||||||
|
|
||||||
const tree = buildTree(pages as Page[]);
|
const tree = buildTree(pages as Page[]);
|
||||||
|
|
||||||
const baseUrl = await this.getWorkspaceBaseUrl(pages[0].workspaceId);
|
const baseUrl = await this.getWorkspaceBaseUrl(pages[0].workspaceId);
|
||||||
@@ -170,7 +177,7 @@ export class ExportService {
|
|||||||
compression: 'DEFLATE',
|
compression: 'DEFLATE',
|
||||||
});
|
});
|
||||||
|
|
||||||
return zipFile;
|
return { type: 'zip' as const, stream: zipFile, page: pages[0] };
|
||||||
}
|
}
|
||||||
|
|
||||||
async exportSpace(
|
async exportSpace(
|
||||||
|
|||||||
@@ -193,6 +193,8 @@ export class ImportAttachmentService {
|
|||||||
// Build a map from resolved archive path → real filename from Confluence
|
// Build a map from resolved archive path → real filename from Confluence
|
||||||
// metadata. Confluence Server archives often store files under numeric IDs
|
// metadata. Confluence Server archives often store files under numeric IDs
|
||||||
// (e.g. "attachments/65601/65602") instead of the original filename.
|
// (e.g. "attachments/65601/65602") instead of the original filename.
|
||||||
|
// Also register aliases so HTML references using the original filename
|
||||||
|
// (e.g. "attachments/pageId/original.mp3") resolve to the numeric path.
|
||||||
const pageDir = path.dirname(pageRelativePath);
|
const pageDir = path.dirname(pageRelativePath);
|
||||||
const attachmentNameByRelPath = new Map<string, string>();
|
const attachmentNameByRelPath = new Map<string, string>();
|
||||||
for (const attachment of pageAttachments) {
|
for (const attachment of pageAttachments) {
|
||||||
@@ -203,6 +205,13 @@ export class ImportAttachmentService {
|
|||||||
);
|
);
|
||||||
if (relPath && attachment.fileName) {
|
if (relPath && attachment.fileName) {
|
||||||
attachmentNameByRelPath.set(relPath, attachment.fileName);
|
attachmentNameByRelPath.set(relPath, attachment.fileName);
|
||||||
|
|
||||||
|
const dir = path.posix.dirname(relPath);
|
||||||
|
const aliasKey = `${dir}/${attachment.fileName}`;
|
||||||
|
if (!attachmentCandidates.has(aliasKey)) {
|
||||||
|
attachmentCandidates.set(aliasKey, attachmentCandidates.get(relPath)!);
|
||||||
|
attachmentNameByRelPath.set(aliasKey, attachment.fileName);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -562,18 +571,31 @@ export class ImportAttachmentService {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if already processed (was referenced in HTML)
|
// Resolve the metadata href to the actual archive path
|
||||||
if (processed.has(href)) {
|
const resolvedHref = resolveRelativeAttachmentPath(
|
||||||
continue;
|
href,
|
||||||
}
|
pageDir,
|
||||||
|
attachmentCandidates,
|
||||||
|
);
|
||||||
|
if (!resolvedHref) continue;
|
||||||
|
|
||||||
// Skip if the file doesn't exist
|
// Check if already processed (was referenced in HTML).
|
||||||
if (!attachmentCandidates.has(href)) {
|
// Inline elements may have been processed under an alias key (original
|
||||||
|
// filename) rather than the numeric archive path, so also check whether
|
||||||
|
// the underlying absolute file path has already been uploaded.
|
||||||
|
const absPath = attachmentCandidates.get(resolvedHref);
|
||||||
|
const alreadyProcessed =
|
||||||
|
processed.has(resolvedHref) ||
|
||||||
|
(absPath &&
|
||||||
|
Array.from(processed.values()).some(
|
||||||
|
(entry) => entry.abs === absPath,
|
||||||
|
));
|
||||||
|
if (alreadyProcessed) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// This attachment was in the list but not referenced in HTML - add it
|
// This attachment was in the list but not referenced in HTML - add it
|
||||||
const { attachmentId, apiFilePath, abs } = processFile(href);
|
const { attachmentId, apiFilePath, abs } = processFile(resolvedHref);
|
||||||
const mime = mimeType || getMimeType(abs);
|
const mime = mimeType || getMimeType(abs);
|
||||||
|
|
||||||
// Add as attachment node at the end
|
// Add as attachment node at the end
|
||||||
|
|||||||
@@ -69,6 +69,7 @@ export enum QueueJob {
|
|||||||
COMMENT_RESOLVED_NOTIFICATION = 'comment-resolved-notification',
|
COMMENT_RESOLVED_NOTIFICATION = 'comment-resolved-notification',
|
||||||
PAGE_MENTION_NOTIFICATION = 'page-mention-notification',
|
PAGE_MENTION_NOTIFICATION = 'page-mention-notification',
|
||||||
PAGE_PERMISSION_GRANTED = 'page-permission-granted',
|
PAGE_PERMISSION_GRANTED = 'page-permission-granted',
|
||||||
|
PAGE_UPDATE_DIGEST = 'page-update-digest',
|
||||||
|
|
||||||
AUDIT_LOG = 'audit-log',
|
AUDIT_LOG = 'audit-log',
|
||||||
AUDIT_CLEANUP = 'audit-cleanup',
|
AUDIT_CLEANUP = 'audit-cleanup',
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ export interface IPageBacklinkJob {
|
|||||||
pageId: string;
|
pageId: string;
|
||||||
workspaceId: string;
|
workspaceId: string;
|
||||||
mentions: MentionNode[];
|
mentions: MentionNode[];
|
||||||
|
internalLinkSlugIds?: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IAddPageWatchersJob {
|
export interface IAddPageWatchersJob {
|
||||||
@@ -60,6 +61,13 @@ export interface IPageMentionNotificationJob {
|
|||||||
workspaceId: string;
|
workspaceId: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface IPageUpdateNotificationJob {
|
||||||
|
pageId: string;
|
||||||
|
spaceId: string;
|
||||||
|
workspaceId: string;
|
||||||
|
actorIds: string[];
|
||||||
|
}
|
||||||
|
|
||||||
export interface IPermissionGrantedNotificationJob {
|
export interface IPermissionGrantedNotificationJob {
|
||||||
userIds: string[];
|
userIds: string[];
|
||||||
pageId: string;
|
pageId: string;
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user