From 04d76933553938d039ba145bece8147dfb0c1441 Mon Sep 17 00:00:00 2001
From: Ahmed Gagan <35112518+Ahmedgagan@users.noreply.github.com>
Date: Thu, 2 Jul 2020 15:36:00 +0530
Subject: [PATCH] FIX: Filter read/unread notifications on the server side
(#10152)
https://meta.discourse.org/t/notifications-unread-only-filter/37621/32
---
.../components/user-notifications-large.js | 8 ++---
.../app/controllers/user-notifications.js | 14 ++------
.../app/routes/user-notifications.js | 10 ++++--
.../templates/user/notifications-index.hbs | 5 +--
.../app/widgets/user-notifications-large.js | 8 ++---
app/assets/stylesheets/common/base/user.scss | 4 +--
app/controllers/notifications_controller.rb | 6 +++-
.../requests/notifications_controller_spec.rb | 18 +++++++++++
.../acceptance/notifications-filter-test.js | 32 +++++++++++++++++++
9 files changed, 76 insertions(+), 29 deletions(-)
create mode 100644 test/javascripts/acceptance/notifications-filter-test.js
diff --git a/app/assets/javascripts/discourse/app/components/user-notifications-large.js b/app/assets/javascripts/discourse/app/components/user-notifications-large.js
index 5f508780999..a81412266a1 100644
--- a/app/assets/javascripts/discourse/app/components/user-notifications-large.js
+++ b/app/assets/javascripts/discourse/app/components/user-notifications-large.js
@@ -4,20 +4,18 @@ import { observes } from "discourse-common/utils/decorators";
export default MountWidget.extend({
widget: "user-notifications-large",
notifications: null,
- filter: null,
args: null,
init() {
this._super(...arguments);
- this.args = { notifications: this.notifications, filter: this.filter };
+ this.args = { notifications: this.notifications };
},
- @observes("notifications.length", "notifications.@each.read", "filter")
+ @observes("notifications.length", "notifications.@each.read")
_triggerRefresh() {
this.set("args", {
- notifications: this.notifications,
- filter: this.filter
+ notifications: this.notifications
});
this.queueRerender();
diff --git a/app/assets/javascripts/discourse/app/controllers/user-notifications.js b/app/assets/javascripts/discourse/app/controllers/user-notifications.js
index c0761e70dcb..6f6012c090e 100644
--- a/app/assets/javascripts/discourse/app/controllers/user-notifications.js
+++ b/app/assets/javascripts/discourse/app/controllers/user-notifications.js
@@ -6,6 +6,7 @@ import { inject as service } from "@ember/service";
export default Controller.extend({
application: controller(),
+ queryParams: ["filter"],
router: service(),
currentPath: readOnly("router._router.currentPath"),
filter: "all",
@@ -15,13 +16,8 @@ export default Controller.extend({
this.set("application.showFooter", !this.get("model.canLoadMore"));
},
- @discourseComputed("model.content.length", "filter")
- hasFilteredNotifications(length, filter) {
- if (filter === "read") {
- return this.model.filterBy("read", true).length > 0;
- } else if (filter === "unread") {
- return this.model.filterBy("read", false).length > 0;
- }
+ @discourseComputed("model.content.length")
+ hasFilteredNotifications(length) {
return length > 0;
},
@@ -41,10 +37,6 @@ export default Controller.extend({
loadMore() {
this.model.loadMore();
- },
-
- filterNotifications(value) {
- this.set("filter", value);
}
}
});
diff --git a/app/assets/javascripts/discourse/app/routes/user-notifications.js b/app/assets/javascripts/discourse/app/routes/user-notifications.js
index 3feb837746c..040cacb8c47 100644
--- a/app/assets/javascripts/discourse/app/routes/user-notifications.js
+++ b/app/assets/javascripts/discourse/app/routes/user-notifications.js
@@ -2,6 +2,9 @@ import DiscourseRoute from "discourse/routes/discourse";
import ViewingActionType from "discourse/mixins/viewing-action-type";
export default DiscourseRoute.extend(ViewingActionType, {
+ controllerName: "user-notifications",
+ queryParams: { filter: { refreshModel: true } },
+
renderTemplate() {
this.render("user/notifications");
},
@@ -13,14 +16,17 @@ export default DiscourseRoute.extend(ViewingActionType, {
}
},
- model() {
+ model(params) {
const username = this.modelFor("user").get("username");
if (
this.get("currentUser.username") === username ||
this.get("currentUser.admin")
) {
- return this.store.find("notification", { username });
+ return this.store.find("notification", {
+ username: username,
+ filter: params.filter
+ });
}
},
diff --git a/app/assets/javascripts/discourse/app/templates/user/notifications-index.hbs b/app/assets/javascripts/discourse/app/templates/user/notifications-index.hbs
index b2c9185ec47..14638f5f781 100644
--- a/app/assets/javascripts/discourse/app/templates/user/notifications-index.hbs
+++ b/app/assets/javascripts/discourse/app/templates/user/notifications-index.hbs
@@ -8,8 +8,9 @@
{{/if}}
-{{notifications-filter value=filter onChange=(action "filterNotifications")}}
-
+
+ {{notifications-filter value=filter onChange=(action (mut filter))}}
+
{{#if hasFilteredNotifications}}
{{user-notifications-large notifications=model filter=filter}}
diff --git a/app/assets/javascripts/discourse/app/widgets/user-notifications-large.js b/app/assets/javascripts/discourse/app/widgets/user-notifications-large.js
index 53a4d3ad527..76bf1da6875 100644
--- a/app/assets/javascripts/discourse/app/widgets/user-notifications-large.js
+++ b/app/assets/javascripts/discourse/app/widgets/user-notifications-large.js
@@ -30,13 +30,9 @@ createWidget("large-notification-item", {
export default createWidget("user-notifications-large", {
html(attrs) {
- let notifications = attrs.notifications;
+ const notifications = attrs.notifications;
const username = notifications.findArgs.username;
- if (attrs.filter === "read") {
- notifications = notifications.filterBy("read", true);
- } else if (attrs.filter === "unread") {
- notifications = notifications.filterBy("read", false);
- }
+
return notifications.map(n => {
n.username = username;
return this.attach("large-notification-item", n);
diff --git a/app/assets/stylesheets/common/base/user.scss b/app/assets/stylesheets/common/base/user.scss
index 4d5e9f42886..c42df5ea818 100644
--- a/app/assets/stylesheets/common/base/user.scss
+++ b/app/assets/stylesheets/common/base/user.scss
@@ -65,10 +65,10 @@
color: $love;
}
- .user-notifications-filter-separator {
+ .user-notifications-filter-select-kit {
display: block;
width: 100%;
- border: 0.5px solid $primary-low;
+ border-bottom: 0.5px solid $primary-low;
}
}
diff --git a/app/controllers/notifications_controller.rb b/app/controllers/notifications_controller.rb
index a928b493bc5..1d28fb4caf0 100644
--- a/app/controllers/notifications_controller.rb
+++ b/app/controllers/notifications_controller.rb
@@ -44,12 +44,16 @@ class NotificationsController < ApplicationController
.includes(:topic)
.order(created_at: :desc)
+ notifications = notifications.where(read: true) if params[:filter] == "read"
+
+ notifications = notifications.where(read: false) if params[:filter] == "unread"
+
total_rows = notifications.dup.count
notifications = notifications.offset(offset).limit(60)
render_json_dump(notifications: serialize_data(notifications, NotificationSerializer),
total_rows_notifications: total_rows,
seen_notification_id: user.seen_notification_id,
- load_more_notifications: notifications_path(username: user.username, offset: offset + 60))
+ load_more_notifications: notifications_path(username: user.username, offset: offset + 60, filter: params[:filter]))
end
end
diff --git a/spec/requests/notifications_controller_spec.rb b/spec/requests/notifications_controller_spec.rb
index 4270afb728c..495ae5afbe5 100644
--- a/spec/requests/notifications_controller_spec.rb
+++ b/spec/requests/notifications_controller_spec.rb
@@ -74,6 +74,24 @@ describe NotificationsController do
Discourse.clear_redis_readonly!
end
+ it "get notifications with all filters" do
+ notification = Fabricate(:notification, user: user)
+ notification2 = Fabricate(:notification, user: user)
+ put "/notifications/mark-read.json", params: { id: notification.id }
+ expect(response.status).to eq(200)
+
+ get "/notifications.json"
+ expect(JSON.parse(response.body)['notifications'].length).to be >= 2
+
+ get "/notifications.json", params: { filter: "read" }
+ expect(JSON.parse(response.body)['notifications'].length).to be >= 1
+ expect(JSON.parse(response.body)['notifications'][0]['read']).to eq(true)
+
+ get "/notifications.json", params: { filter: "unread" }
+ expect(JSON.parse(response.body)['notifications'].length).to be >= 1
+ expect(JSON.parse(response.body)['notifications'][0]['read']).to eq(false)
+ end
+
context 'when username params is not valid' do
it 'should raise the right error' do
get "/notifications.json", params: { username: 'somedude' }
diff --git a/test/javascripts/acceptance/notifications-filter-test.js b/test/javascripts/acceptance/notifications-filter-test.js
new file mode 100644
index 00000000000..76234f3f653
--- /dev/null
+++ b/test/javascripts/acceptance/notifications-filter-test.js
@@ -0,0 +1,32 @@
+import { acceptance } from "helpers/qunit-helpers";
+import selectKit from "helpers/select-kit-helper";
+
+acceptance("NotificationsFilter", {
+ loggedIn: true
+});
+
+test("Notifications filter true", async assert => {
+ await visit("/u/eviltrout/notifications");
+
+ assert.ok(find(".large-notification").length >= 0);
+});
+
+test("Notifications filter read", async assert => {
+ await visit("/u/eviltrout/notifications");
+
+ const dropdown = selectKit(".notifications-filter");
+ await dropdown.expand();
+ await dropdown.selectRowByValue("read");
+
+ assert.ok(find(".large-notification").length >= 0);
+});
+
+test("Notifications filter unread", async assert => {
+ await visit("/u/eviltrout/notifications");
+
+ const dropdown = selectKit(".notifications-filter");
+ await dropdown.expand();
+ await dropdown.selectRowByValue("unread");
+
+ assert.ok(find(".large-notification").length >= 0);
+});