Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 5 additions & 1 deletion examples/07-collaboration/05-comments/src/App.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -100,7 +100,11 @@ function Document() {
label={"User"}
items={HARDCODED_USERS.map((user) => ({
text: `${user.username} (${
user.role === "editor" ? "Editor" : "Commenter"
user.role === "editor"
? "Editor"
: user.role === "comment"
? "Commenter"
: "Content-Only Viewer"
})`,
icon: null,
onClick: () => setActiveUser(user),
Expand Down
8 changes: 7 additions & 1 deletion examples/07-collaboration/05-comments/src/userdata.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ const getRandomElement = (list: any[]) =>
export const getRandomColor = () => getRandomElement(colors);

export type MyUserType = User & {
role: "editor" | "comment";
role: "editor" | "comment" | "document-only";
};

export const HARDCODED_USERS: MyUserType[] = [
Expand Down Expand Up @@ -44,4 +44,10 @@ export const HARDCODED_USERS: MyUserType[] = [
avatarUrl: "https://placehold.co/100x100?text=Betty",
role: "comment",
},
{
id: "5",
username: "Donna Document",
avatarUrl: "https://placehold.co/100x100?text=Donna",
role: "document-only",
},
];
Original file line number Diff line number Diff line change
Expand Up @@ -125,7 +125,11 @@ function Document() {
label={"User"}
items={HARDCODED_USERS.map((user) => ({
text: `${user.username} (${
user.role === "editor" ? "Editor" : "Commenter"
user.role === "editor"
? "Editor"
: user.role === "comment"
? "Commenter"
: "Content-Only Viewer"
})`,
icon: null,
onClick: () => {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ const getRandomElement = (list: any[]) =>
export const getRandomColor = () => getRandomElement(colors);

export type MyUserType = User & {
role: "editor" | "comment";
role: "editor" | "comment" | "document-only";
};

export const HARDCODED_USERS: MyUserType[] = [
Expand Down Expand Up @@ -44,4 +44,10 @@ export const HARDCODED_USERS: MyUserType[] = [
avatarUrl: "https://placehold.co/100x100?text=Betty",
role: "comment",
},
{
id: "5",
username: "Donna Document",
avatarUrl: "https://placehold.co/100x100?text=Donna",
role: "document-only",
},
];
30 changes: 20 additions & 10 deletions packages/core/src/comments/threadstore/DefaultThreadStoreAuth.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,58 +18,68 @@ import { ThreadStoreAuth } from "./ThreadStoreAuth.js";
export class DefaultThreadStoreAuth extends ThreadStoreAuth {
constructor(
private readonly userId: string,
private readonly role: "comment" | "editor",
private readonly role: "document-only" | "comment" | "editor",
) {
super();
}

/**
* Auth: should be possible by anyone with comment access
*/
canViewComments(): boolean {
return this.role !== "document-only";
}

/**
* Auth: should be possible by anyone with comment access
*/
canCreateThread(): boolean {
return true;
return this.canViewComments();
}

/**
* Auth: should be possible by anyone with comment access
*/
canAddComment(_thread: ThreadData): boolean {
return true;
return this.canViewComments();
}

/**
* Auth: should only be possible by the comment author
*/
canUpdateComment(comment: CommentData): boolean {
return comment.userId === this.userId;
return this.canViewComments() && comment.userId === this.userId;
}

/**
* Auth: should be possible by the comment author OR an editor of the document
*/
canDeleteComment(comment: CommentData): boolean {
return comment.userId === this.userId || this.role === "editor";
return (
this.canViewComments() &&
(comment.userId === this.userId || this.role === "editor")
);
}

/**
* Auth: should only be possible by an editor of the document
*/
canDeleteThread(_thread: ThreadData): boolean {
return this.role === "editor";
return this.canViewComments() && this.role === "editor";
}

/**
* Auth: should be possible by anyone with comment access
*/
canResolveThread(_thread: ThreadData): boolean {
return true;
return this.canViewComments();
}

/**
* Auth: should be possible by anyone with comment access
*/
canUnresolveThread(_thread: ThreadData): boolean {
return true;
return this.canViewComments();
}

/**
Expand All @@ -79,7 +89,7 @@ export class DefaultThreadStoreAuth extends ThreadStoreAuth {
*/
canAddReaction(comment: CommentData, emoji?: string): boolean {
if (!emoji) {
return true;
return this.canViewComments();
}

return !comment.reactions.some(
Expand All @@ -95,7 +105,7 @@ export class DefaultThreadStoreAuth extends ThreadStoreAuth {
*/
canDeleteReaction(comment: CommentData, emoji?: string): boolean {
if (!emoji) {
return true;
return this.canViewComments();
}

return comment.reactions.some(
Expand Down
1 change: 1 addition & 0 deletions packages/core/src/comments/threadstore/ThreadStoreAuth.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ import { CommentData, ThreadData } from "../types.js";

export abstract class ThreadStoreAuth {
abstract canCreateThread(): boolean;
abstract canViewComments(): boolean;
abstract canAddComment(thread: ThreadData): boolean;
abstract canUpdateComment(comment: CommentData): boolean;
abstract canDeleteComment(comment: CommentData): boolean;
Expand Down
4 changes: 2 additions & 2 deletions packages/core/src/editor/Block.css
Original file line number Diff line number Diff line change
Expand Up @@ -717,10 +717,10 @@ NESTED BLOCKS
padding-right: 0;
}

.bn-thread-mark:not([data-orphan="true"]) {
.bn-editor:not([data-hide-comments]) .bn-thread-mark:not([data-orphan="true"]) {
background: rgba(255, 200, 0, 0.15);
}

.bn-thread-mark .bn-thread-mark-selected {
.bn-editor:not([data-hide-comments]) .bn-thread-mark .bn-thread-mark-selected {
background: rgba(255, 200, 0, 0.25);
}
3 changes: 3 additions & 0 deletions packages/core/src/editor/BlockNoteEditor.ts
Original file line number Diff line number Diff line change
Expand Up @@ -848,6 +848,9 @@ export class BlockNoteEditor<
editorProps: {
...newOptions._tiptapOptions?.editorProps,
attributes: {
...(this.comments?.threadStore.auth.canViewComments()
? {}
: { "data-hide-comments": "true" }),
// As of TipTap v2.5.0 the tabIndex is removed when the editor is not
// editable, so you can't focus it. We want to revert this as we have
// UI behaviour that relies on it.
Expand Down
11 changes: 10 additions & 1 deletion packages/core/src/extensions/Comments/CommentsPlugin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -180,6 +180,12 @@ export class CommentsPlugin extends BlockNoteExtension {
if (!tr.docChanged && !action) {
return state;
}
if (!self.threadStore.auth.canViewComments()) {
// if the user doesn't have comment access, don't display the marks in the document
return {
decorations: DecorationSet.empty,
};
}

// only update threadPositions if the doc changed
const threadPositions = tr.docChanged
Expand Down Expand Up @@ -228,7 +234,10 @@ export class CommentsPlugin extends BlockNoteExtension {
* Handle click on a thread mark and mark it as selected
*/
handleClick: (view, pos, event) => {
if (event.button !== 0) {
if (
event.button !== 0 ||
!self.threadStore.auth.canViewComments()
) {
return;
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,8 @@ export const AddCommentButton = () => {
if (
// We manually check if a comment extension (like liveblocks) is installed
// By adding default support for this, the user doesn't need to customize the formatting toolbar
!editor.comments
!editor.comments ||
!editor.comments.threadStore.auth.canViewComments()
) {
return null;
}
Expand Down
Loading