FEATURE: Open the edit bookmark modal when clicking on the topic level bookmark button (#13407)
If you click on a bookmark in the post stream you get an Edit Bookmark modal. This does not happen if you click the topic bookmark button. We want to open the Edit modal too if there is only one bookmark on a topic (it doesn't matter on the first post or not). The other behaviour if there are > 1 bookmarks in the topic is to prompt the user to confirm delete of all the bookmarks in the topic. This behaviour will stay as-is. I have done some refactoring in this PR, and still, there is a place for improvement. For example, we don't call post.deleteBookmark() method when deleting several bookmarks. I just don't want to refactor too much in one PR.
This commit is contained in:
parent
0c42a29dc4
commit
3b87271647
|
@ -1219,57 +1219,43 @@ export default Controller.extend(bufferedProperty("model"), {
|
||||||
return Promise.resolve();
|
return Promise.resolve();
|
||||||
}
|
}
|
||||||
this.model.set("bookmarking", true);
|
this.model.set("bookmarking", true);
|
||||||
const bookmark = !this.model.bookmarked;
|
const alreadyBookmarkedPosts = this.model.bookmarkedPosts;
|
||||||
let posts = this.model.postStream.posts;
|
|
||||||
|
|
||||||
return this.model.firstPost().then((firstPost) => {
|
return this.model.firstPost().then((firstPost) => {
|
||||||
const toggleBookmarkOnServer = () => {
|
const bookmarkPost = async (post) => {
|
||||||
if (bookmark) {
|
const opts = await this._togglePostBookmark(post);
|
||||||
return this._togglePostBookmark(firstPost).then((opts) => {
|
|
||||||
this.model.set("bookmarking", false);
|
this.model.set("bookmarking", false);
|
||||||
if (opts && opts.closedWithoutSaving) {
|
if (opts.closedWithoutSaving) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
return this.model.afterTopicBookmarked(firstPost);
|
this.model.afterPostBookmarked(post);
|
||||||
});
|
return [post.id];
|
||||||
|
};
|
||||||
|
|
||||||
|
const toggleBookmarkOnServer = () => {
|
||||||
|
if (alreadyBookmarkedPosts.length === 0) {
|
||||||
|
return bookmarkPost(firstPost);
|
||||||
|
} else if (alreadyBookmarkedPosts.length === 1) {
|
||||||
|
const post = alreadyBookmarkedPosts[0];
|
||||||
|
return bookmarkPost(post);
|
||||||
} else {
|
} else {
|
||||||
return this.model
|
return this.model
|
||||||
.deleteBookmark()
|
.deleteBookmark()
|
||||||
.then(() => {
|
.then(() => {
|
||||||
this.model.toggleProperty("bookmarked");
|
this.model.toggleProperty("bookmarked");
|
||||||
this.model.set("bookmark_reminder_at", null);
|
this.model.set("bookmark_reminder_at", null);
|
||||||
let clearedBookmarkProps = {
|
alreadyBookmarkedPosts.forEach((post) => {
|
||||||
bookmarked: false,
|
post.clearBookmark();
|
||||||
bookmark_id: null,
|
|
||||||
bookmark_name: null,
|
|
||||||
bookmark_reminder_at: null,
|
|
||||||
};
|
|
||||||
if (posts) {
|
|
||||||
const updated = [];
|
|
||||||
posts.forEach((post) => {
|
|
||||||
if (post.bookmarked) {
|
|
||||||
post.setProperties(clearedBookmarkProps);
|
|
||||||
updated.push(post.id);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
firstPost.setProperties(clearedBookmarkProps);
|
return alreadyBookmarkedPosts.mapBy("id");
|
||||||
return updated;
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
.catch(popupAjaxError)
|
.catch(popupAjaxError)
|
||||||
.finally(() => this.model.set("bookmarking", false));
|
.finally(() => this.model.set("bookmarking", false));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const unbookmarkedPosts = [];
|
|
||||||
if (!bookmark && posts) {
|
|
||||||
posts.forEach(
|
|
||||||
(post) => post.bookmarked && unbookmarkedPosts.push(post)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return new Promise((resolve) => {
|
return new Promise((resolve) => {
|
||||||
if (unbookmarkedPosts.length > 1) {
|
if (alreadyBookmarkedPosts.length > 1) {
|
||||||
bootbox.confirm(
|
bootbox.confirm(
|
||||||
I18n.t("bookmarks.confirm_clear"),
|
I18n.t("bookmarks.confirm_clear"),
|
||||||
I18n.t("no_value"),
|
I18n.t("no_value"),
|
||||||
|
|
|
@ -66,7 +66,7 @@ export default {
|
||||||
});
|
});
|
||||||
|
|
||||||
registerTopicFooterButton({
|
registerTopicFooterButton({
|
||||||
dependentKeys: ["topic.bookmarked"],
|
dependentKeys: ["topic.bookmarked", "topic.bookmarkedPosts"],
|
||||||
id: "bookmark",
|
id: "bookmark",
|
||||||
icon() {
|
icon() {
|
||||||
if (this.get("topic.bookmark_reminder_at")) {
|
if (this.get("topic.bookmark_reminder_at")) {
|
||||||
|
@ -81,8 +81,15 @@ export default {
|
||||||
},
|
},
|
||||||
label() {
|
label() {
|
||||||
if (!this.get("topic.isPrivateMessage") || this.site.mobileView) {
|
if (!this.get("topic.isPrivateMessage") || this.site.mobileView) {
|
||||||
const bookmarked = this.get("topic.bookmarked");
|
const bookmarkedPostsCount = this.get("topic.bookmarkedPosts").length;
|
||||||
return bookmarked ? "bookmarked.clear_bookmarks" : "bookmarked.title";
|
|
||||||
|
if (bookmarkedPostsCount === 0) {
|
||||||
|
return "bookmarked.title";
|
||||||
|
} else if (bookmarkedPostsCount === 1) {
|
||||||
|
return "bookmarked.edit_bookmark";
|
||||||
|
} else {
|
||||||
|
return "bookmarked.clear_bookmarks";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
translatedTitle() {
|
translatedTitle() {
|
||||||
|
|
|
@ -314,6 +314,7 @@ const Post = RestModel.extend({
|
||||||
bookmark_name: data.name,
|
bookmark_name: data.name,
|
||||||
bookmark_id: data.id,
|
bookmark_id: data.id,
|
||||||
});
|
});
|
||||||
|
this.topic.incrementProperty("bookmarksWereChanged");
|
||||||
this.appEvents.trigger("page:bookmark-post-toggled", this);
|
this.appEvents.trigger("page:bookmark-post-toggled", this);
|
||||||
this.appEvents.trigger("post-stream:refresh", { id: this.id });
|
this.appEvents.trigger("post-stream:refresh", { id: this.id });
|
||||||
},
|
},
|
||||||
|
@ -333,6 +334,7 @@ const Post = RestModel.extend({
|
||||||
bookmarked: false,
|
bookmarked: false,
|
||||||
bookmark_auto_delete_preference: null,
|
bookmark_auto_delete_preference: null,
|
||||||
});
|
});
|
||||||
|
this.topic.incrementProperty("bookmarksWereChanged");
|
||||||
},
|
},
|
||||||
|
|
||||||
updateActionsSummary(json) {
|
updateActionsSummary(json) {
|
||||||
|
|
|
@ -322,6 +322,11 @@ const Topic = RestModel.extend({
|
||||||
return Site.currentProp("archetypes").findBy("id", archetype);
|
return Site.currentProp("archetypes").findBy("id", archetype);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@discourseComputed("bookmarksWereChanged")
|
||||||
|
bookmarkedPosts() {
|
||||||
|
return this.postStream.posts.filterBy("bookmarked", true);
|
||||||
|
},
|
||||||
|
|
||||||
isPrivateMessage: equal("archetype", "private_message"),
|
isPrivateMessage: equal("archetype", "private_message"),
|
||||||
isBanner: equal("archetype", "banner"),
|
isBanner: equal("archetype", "banner"),
|
||||||
|
|
||||||
|
@ -356,12 +361,9 @@ const Topic = RestModel.extend({
|
||||||
}).then(() => this.set("archetype", "regular"));
|
}).then(() => this.set("archetype", "regular"));
|
||||||
},
|
},
|
||||||
|
|
||||||
afterTopicBookmarked(firstPost) {
|
afterPostBookmarked(post) {
|
||||||
if (firstPost) {
|
post.set("bookmarked", true);
|
||||||
firstPost.set("bookmarked", true);
|
this.set("bookmark_reminder_at", post.bookmark_reminder_at);
|
||||||
this.set("bookmark_reminder_at", firstPost.bookmark_reminder_at);
|
|
||||||
return [firstPost.id];
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
firstPost() {
|
firstPost() {
|
||||||
|
|
|
@ -2,6 +2,7 @@ import {
|
||||||
acceptance,
|
acceptance,
|
||||||
exists,
|
exists,
|
||||||
loggedInUser,
|
loggedInUser,
|
||||||
|
query,
|
||||||
queryAll,
|
queryAll,
|
||||||
} from "discourse/tests/helpers/qunit-helpers";
|
} from "discourse/tests/helpers/qunit-helpers";
|
||||||
import { click, fillIn, visit } from "@ember/test-helpers";
|
import { click, fillIn, visit } from "@ember/test-helpers";
|
||||||
|
@ -315,4 +316,42 @@ acceptance("Bookmarking", function (needs) {
|
||||||
"the second bookmark is deleted"
|
"the second bookmark is deleted"
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("The topic level bookmark button opens the edit modal if only the first post on the topic is bookmarked", async function (assert) {
|
||||||
|
await visit("/t/internationalization-localization/280");
|
||||||
|
await openBookmarkModal(1);
|
||||||
|
await click("#save-bookmark");
|
||||||
|
|
||||||
|
assert.equal(
|
||||||
|
query("#topic-footer-button-bookmark").innerText,
|
||||||
|
I18n.t("bookmarked.edit_bookmark"),
|
||||||
|
"A topic level bookmark button has a label 'Edit Bookmark'"
|
||||||
|
);
|
||||||
|
|
||||||
|
await click("#topic-footer-button-bookmark");
|
||||||
|
|
||||||
|
assert.ok(
|
||||||
|
exists("div.modal.bookmark-with-reminder"),
|
||||||
|
"The edit modal is opened"
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("The topic level bookmark button opens the edit modal if only one post in the post stream is bookmarked", async function (assert) {
|
||||||
|
await visit("/t/internationalization-localization/280");
|
||||||
|
await openBookmarkModal(2);
|
||||||
|
await click("#save-bookmark");
|
||||||
|
|
||||||
|
assert.equal(
|
||||||
|
query("#topic-footer-button-bookmark").innerText,
|
||||||
|
I18n.t("bookmarked.edit_bookmark"),
|
||||||
|
"A topic level bookmark button has a label 'Edit Bookmark'"
|
||||||
|
);
|
||||||
|
|
||||||
|
await click("#topic-footer-button-bookmark");
|
||||||
|
|
||||||
|
assert.ok(
|
||||||
|
exists("div.modal.bookmark-with-reminder"),
|
||||||
|
"The edit modal is opened"
|
||||||
|
);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -297,6 +297,7 @@ en:
|
||||||
|
|
||||||
bookmarked:
|
bookmarked:
|
||||||
title: "Bookmark"
|
title: "Bookmark"
|
||||||
|
edit_bookmark: "Edit Bookmark"
|
||||||
clear_bookmarks: "Clear Bookmarks"
|
clear_bookmarks: "Clear Bookmarks"
|
||||||
help:
|
help:
|
||||||
bookmark: "Click to bookmark the first post on this topic"
|
bookmark: "Click to bookmark the first post on this topic"
|
||||||
|
|
Loading…
Reference in New Issue