UX: Changes to new features section in admin dashboard (#12029)

This commit is contained in:
Penar Musaraj 2021-02-10 13:12:04 -05:00 committed by GitHub
parent 43948f6a10
commit 544a4e4b48
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 92 additions and 52 deletions

View File

@ -1,9 +1,11 @@
import Component from "@ember/component"; import Component from "@ember/component";
import { action } from "@ember/object"; import { action, computed } from "@ember/object";
import { ajax } from "discourse/lib/ajax"; import { ajax } from "discourse/lib/ajax";
export default Component.extend({ export default Component.extend({
newFeatures: null, newFeatures: null,
classNames: ["section", "dashboard-new-features"],
classNameBindings: ["hasUnseenFeatures:ordered-first"],
releaseNotesLink: null, releaseNotesLink: null,
init() { init() {
@ -12,15 +14,20 @@ export default Component.extend({
ajax("/admin/dashboard/new-features.json").then((json) => { ajax("/admin/dashboard/new-features.json").then((json) => {
this.setProperties({ this.setProperties({
newFeatures: json.new_features, newFeatures: json.new_features,
hasUnseenFeatures: json.has_unseen_features,
releaseNotesLink: json.release_notes_link, releaseNotesLink: json.release_notes_link,
}); });
}); });
}, },
columnCountClass: computed("newFeatures", function () {
return this.newFeatures.length > 2 ? "three-or-more-items" : "";
}),
@action @action
dismissNewFeatures() { dismissNewFeatures() {
ajax("/admin/dashboard/mark-new-features-as-seen.json", { ajax("/admin/dashboard/mark-new-features-as-seen.json", {
type: "PUT", type: "PUT",
}).then(() => this.set("newFeatures", null)); }).then(() => this.set("hasUnseenFeatures", false));
}, },
}); });

View File

@ -1,12 +1,11 @@
{{#if newFeatures}} {{#if newFeatures}}
<div class="section dashboard-new-features">
<div class="section-title"> <div class="section-title">
<h2>{{replace-emoji (i18n "admin.dashboard.new_features.title") }}</h2> <h2>{{replace-emoji (i18n "admin.dashboard.new_features.title") }}</h2>
</div> </div>
<div class="section-body"> <div class="section-body {{columnCountClass}}">
{{#each newFeatures as |feature|}} {{#each newFeatures as |feature|}}
{{dashboard-new-feature-item item=feature}} {{dashboard-new-feature-item item=feature tagName=""}}
{{/each}} {{/each}}
</div> </div>
<div class="section-footer"> <div class="section-footer">
@ -17,5 +16,4 @@
{{/if}} {{/if}}
{{d-button label="admin.dashboard.new_features.dismiss" class="new-features-dismiss" action=dismissNewFeatures }} {{d-button label="admin.dashboard.new_features.dismiss" class="new-features-dismiss" action=dismissNewFeatures }}
</div> </div>
</div>
{{/if}} {{/if}}

View File

@ -1,5 +1,3 @@
{{dashboard-new-features}}
{{plugin-outlet name="admin-dashboard-top"}} {{plugin-outlet name="admin-dashboard-top"}}
{{#if showVersionChecks}} {{#if showVersionChecks}}
@ -52,4 +50,6 @@
{{outlet}} {{outlet}}
{{dashboard-new-features tagName="div"}}
{{plugin-outlet name="admin-dashboard-bottom"}} {{plugin-outlet name="admin-dashboard-bottom"}}

View File

@ -613,11 +613,32 @@
} }
} }
.dashboard-next.general {
display: flex;
flex-direction: column;
}
.dashboard-new-features { .dashboard-new-features {
&.ordered-first {
order: -1;
}
&:not(.ordered-first) {
.section-title {
margin-top: 1.5em;
}
.new-features-dismiss {
display: none;
}
}
.section-body { .section-body {
display: grid; display: grid;
grid-template-columns: repeat(auto-fill, minmax(300px, 1fr)); grid-template-columns: repeat(auto-fill, minmax(360px, 1fr));
grid-gap: 1.5em; grid-gap: 1.5em;
&.three-or-more-items {
grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));
}
} }
.section-footer { .section-footer {

View File

@ -6,4 +6,7 @@
.navigation a.navigation-link { .navigation a.navigation-link {
padding: 0.5em; padding: 0.5em;
} }
.dashboard-new-features .section-body {
grid-template-columns: none;
}
} }

View File

@ -24,8 +24,11 @@ class Admin::DashboardController < Admin::AdminController
end end
def new_features def new_features
data = { new_features: DiscourseUpdates.unseen_new_features(current_user.id) } data = {
data.merge!(release_notes_link: AdminDashboardGeneralData.fetch_cached_stats["release_notes_link"]) new_features: DiscourseUpdates.new_features,
has_unseen_features: DiscourseUpdates.has_unseen_features?(current_user.id),
release_notes_link: AdminDashboardGeneralData.fetch_cached_stats["release_notes_link"]
}
render json: data render json: data
end end

View File

@ -121,21 +121,28 @@ module DiscourseUpdates
Discourse.redis.set(new_features_key, response.body) Discourse.redis.set(new_features_key, response.body)
end end
def unseen_new_features(user_id) def new_features
entries = JSON.parse(Discourse.redis.get(new_features_key)) rescue nil entries = JSON.parse(Discourse.redis.get(new_features_key)) rescue nil
return nil if entries.nil? return nil if entries.nil?
entries.select! do |item|
item["discourse_version"].nil? || Discourse.has_needed_version?(Discourse::VERSION::STRING, item["discourse_version"]) rescue nil
end
entries.sort { |item| Time.zone.parse(item["created_at"]) }
end
def has_unseen_features?(user_id)
entries = new_features
return false if entries.nil?
last_seen = new_features_last_seen(user_id) last_seen = new_features_last_seen(user_id)
if last_seen.present? if last_seen.present?
entries.select! { |item| Time.zone.parse(item["created_at"]) > last_seen } entries.select! { |item| Time.zone.parse(item["created_at"]) > last_seen }
end end
entries.select! do |item| entries.size > 0
item["discourse_version"].nil? || Discourse.has_needed_version?(Discourse::VERSION::STRING, item["discourse_version"]) rescue nil
end
entries.sort { |item| Time.zone.parse(item["created_at"]) }
end end
def new_features_last_seen(user_id) def new_features_last_seen(user_id)

View File

@ -158,46 +158,37 @@ describe DiscourseUpdates do
before(:each) do before(:each) do
Discourse.redis.del "new_features_last_seen_user_#{admin.id}" Discourse.redis.del "new_features_last_seen_user_#{admin.id}"
Discourse.redis.del "new_features_last_seen_user_#{admin2.id}" Discourse.redis.del "new_features_last_seen_user_#{admin2.id}"
Discourse.redis.del "new_features"
Discourse.redis.set('new_features', MultiJson.dump(sample_features)) Discourse.redis.set('new_features', MultiJson.dump(sample_features))
end end
it 'returns all items on the first run' do it 'returns all items on the first run' do
result = DiscourseUpdates.unseen_new_features(admin.id) result = DiscourseUpdates.new_features
expect(result.length).to eq(3) expect(result.length).to eq(3)
expect(result[2]["title"]).to eq("Super Fruits") expect(result[2]["title"]).to eq("Super Fruits")
end end
it 'returns only unseen items by user' do it 'correctly marks unseen items by user' do
DiscourseUpdates.stubs(:new_features_last_seen).with(admin.id).returns(10.minutes.ago) DiscourseUpdates.stubs(:new_features_last_seen).with(admin.id).returns(10.minutes.ago)
DiscourseUpdates.stubs(:new_features_last_seen).with(admin2.id).returns(30.minutes.ago) DiscourseUpdates.stubs(:new_features_last_seen).with(admin2.id).returns(30.minutes.ago)
result = DiscourseUpdates.unseen_new_features(admin.id) expect(DiscourseUpdates.has_unseen_features?(admin.id)).to eq(true)
expect(result.length).to eq(1) expect(DiscourseUpdates.has_unseen_features?(admin2.id)).to eq(true)
expect(result[0]["title"]).to eq("Quality Veggies")
result2 = DiscourseUpdates.unseen_new_features(admin2.id)
expect(result2.length).to eq(2)
expect(result2[0]["title"]).to eq("Quality Veggies")
expect(result2[1]["title"]).to eq("Fancy Legumes")
end end
it 'can mark features as seen for a given user' do it 'can mark features as seen for a given user' do
expect(DiscourseUpdates.unseen_new_features(admin.id)).to be_present expect(DiscourseUpdates.has_unseen_features?(admin.id)).to be_truthy
DiscourseUpdates.mark_new_features_as_seen(admin.id) DiscourseUpdates.mark_new_features_as_seen(admin.id)
expect(DiscourseUpdates.unseen_new_features(admin.id)).to be_empty expect(DiscourseUpdates.has_unseen_features?(admin.id)).to eq(false)
# doesn't affect another user # doesn't affect another user
expect(DiscourseUpdates.unseen_new_features(admin2.id)).to be_present expect(DiscourseUpdates.has_unseen_features?(admin2.id)).to eq(true)
end end
it 'correctly sees newly added features as unseen' do it 'correctly sees newly added features as unseen' do
DiscourseUpdates.mark_new_features_as_seen(admin.id) DiscourseUpdates.mark_new_features_as_seen(admin.id)
expect(DiscourseUpdates.unseen_new_features(admin.id)).to be_empty expect(DiscourseUpdates.has_unseen_features?(admin.id)).to eq(false)
expect(DiscourseUpdates.new_features_last_seen(admin.id)).to be_within(1.second).of (last_item_date) expect(DiscourseUpdates.new_features_last_seen(admin.id)).to be_within(1.second).of (last_item_date)
updated_features = [ updated_features = [
@ -206,10 +197,7 @@ describe DiscourseUpdates do
updated_features += sample_features updated_features += sample_features
Discourse.redis.set('new_features', MultiJson.dump(updated_features)) Discourse.redis.set('new_features', MultiJson.dump(updated_features))
expect(DiscourseUpdates.has_unseen_features?(admin.id)).to eq(true)
result = DiscourseUpdates.unseen_new_features(admin.id)
expect(result.length).to eq(1)
expect(result[0]["title"]).to eq("Brand New Item")
end end
it 'correctly shows features by Discourse version' do it 'correctly shows features by Discourse version' do
@ -224,7 +212,7 @@ describe DiscourseUpdates do
Discourse.redis.set('new_features', MultiJson.dump(features_with_versions)) Discourse.redis.set('new_features', MultiJson.dump(features_with_versions))
DiscourseUpdates.stubs(:last_installed_version).returns("2.7.0.beta2") DiscourseUpdates.stubs(:last_installed_version).returns("2.7.0.beta2")
result = DiscourseUpdates.unseen_new_features(admin.id) result = DiscourseUpdates.new_features
expect(result.length).to eq(3) expect(result.length).to eq(3)
expect(result[0]["title"]).to eq("Confetti") expect(result[0]["title"]).to eq("Confetti")

View File

@ -118,6 +118,18 @@ describe Admin::DashboardController do
expect(json['new_features'].length).to eq(2) expect(json['new_features'].length).to eq(2)
expect(json['new_features'][0]["emoji"]).to eq("🙈") expect(json['new_features'][0]["emoji"]).to eq("🙈")
expect(json['new_features'][0]["title"]).to eq("Fancy Legumes") expect(json['new_features'][0]["title"]).to eq("Fancy Legumes")
expect(json['has_unseen_features']).to eq(true)
end
it 'passes unseen feature state' do
populate_new_features
DiscourseUpdates.mark_new_features_as_seen(admin.id)
get "/admin/dashboard/new-features.json"
expect(response.status).to eq(200)
json = response.parsed_body
expect(json['has_unseen_features']).to eq(false)
end end
end end
@ -128,6 +140,7 @@ describe Admin::DashboardController do
expect(response.status).to eq(200) expect(response.status).to eq(200)
expect(DiscourseUpdates.new_features_last_seen(admin.id)).not_to eq(nil) expect(DiscourseUpdates.new_features_last_seen(admin.id)).not_to eq(nil)
expect(DiscourseUpdates.has_unseen_features?(admin.id)).to eq(false)
end end
end end
end end