mirror of
https://github.com/docmost/docmost.git
synced 2026-05-20 08:34:04 +08:00
Compare commits
8 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 2fe2c0e6c1 | |||
| 388572f689 | |||
| a8335475fd | |||
| 6a90b318e5 | |||
| 8b4cc82e5a | |||
| cda7cc9a57 | |||
| 59c5f25502 | |||
| b9d58081b8 |
@@ -733,5 +733,7 @@
|
|||||||
"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 log in with email and password.",
|
||||||
|
"Uploading {{name}}": "Uploading {{name}}",
|
||||||
|
"Uploading file": "Uploading file"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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" withinPortal={false}>
|
<Tooltip label={upgradeLabel} position="left" withPortal={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>
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
// 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, TextSelection } from "@tiptap/pm/state";
|
import { Plugin, PluginKey } from "@tiptap/pm/state";
|
||||||
import { DOMParser, DOMSerializer, Fragment, Slice } from "@tiptap/pm/model";
|
import { DOMParser, DOMSerializer, Fragment, Slice } from "@tiptap/pm/model";
|
||||||
import { find } from "linkifyjs";
|
import { find } from "linkifyjs";
|
||||||
import { markdownToHtml, htmlToMarkdown } from "@docmost/editor-ext";
|
import { markdownToHtml, htmlToMarkdown } from "@docmost/editor-ext";
|
||||||
@@ -50,46 +50,26 @@ 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;
|
||||||
|
|
||||||
const isVscodeMarkdown = language === "markdown";
|
if (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 parsed = markdownToHtml(text.replace(/\n+$/, ""));
|
const html = markdownToHtml(text.replace(/\n+$/, ""));
|
||||||
|
|
||||||
const contentNodes = DOMParser.fromSchema(
|
const contentNodes = DOMParser.fromSchema(
|
||||||
this.editor.schema,
|
this.editor.schema,
|
||||||
).parseSlice(elementFromString(parsed), {
|
).parseSlice(elementFromString(html), {
|
||||||
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;
|
||||||
@@ -125,6 +105,26 @@ export const MarkdownClipboard = Extension.create({
|
|||||||
|
|
||||||
return slice;
|
return slice;
|
||||||
},
|
},
|
||||||
|
clipboardTextParser: (text, context, plainText) => {
|
||||||
|
const link = find(text, {
|
||||||
|
defaultProtocol: "http",
|
||||||
|
}).find((item) => item.isLink && item.value === text);
|
||||||
|
|
||||||
|
if (plainText || !this.options.transformPastedText || link) {
|
||||||
|
// don't parse plaintext link to allow link paste handler to work
|
||||||
|
// pasting with shift key prevents formatting
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const parsed = markdownToHtml(text.replace(/\n+$/, ""));
|
||||||
|
return DOMParser.fromSchema(this.editor.schema).parseSlice(
|
||||||
|
elementFromString(parsed),
|
||||||
|
{
|
||||||
|
preserveWhitespace: true,
|
||||||
|
context,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -1,333 +0,0 @@
|
|||||||
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();
|
|
||||||
}
|
|
||||||
@@ -9,7 +9,5 @@
|
|||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
"module": "./src/index.ts",
|
"module": "./src/index.ts",
|
||||||
"types": "dist/index.d.ts",
|
"types": "dist/index.d.ts",
|
||||||
"dependencies": {
|
"dependencies": {}
|
||||||
"marked": "17.0.5"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
Generated
+1
-5
@@ -801,11 +801,7 @@ importers:
|
|||||||
specifier: ^8.57.1
|
specifier: ^8.57.1
|
||||||
version: 8.57.1(eslint@9.39.4(jiti@2.4.2))(typescript@5.9.3)
|
version: 8.57.1(eslint@9.39.4(jiti@2.4.2))(typescript@5.9.3)
|
||||||
|
|
||||||
packages/editor-ext:
|
packages/editor-ext: {}
|
||||||
dependencies:
|
|
||||||
marked:
|
|
||||||
specifier: 17.0.5
|
|
||||||
version: 17.0.5
|
|
||||||
|
|
||||||
packages:
|
packages:
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user