FIX: Bookmark reminders and improvements changes (#9213)

* Cosmetic fixes for the bookmark modal
* Do not show "later today" when the later time will be > 5pm
* When a custom reminder time is selected, store it in localStorage. The next time the modal is opened, if the last datetime is > now, then a new tile with "Last" will be shown that lets the user reselect that same time.
* Also add an explicit "No Reminder" option that is selected by default
This commit is contained in:
Martin Brennan 2020-03-16 16:05:44 +10:00 committed by GitHub
parent a6e9057609
commit 4cce564b35
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 134 additions and 30 deletions

View File

@ -7,6 +7,7 @@ import { htmlSafe } from "@ember/template";
import { ajax } from "discourse/lib/ajax"; import { ajax } from "discourse/lib/ajax";
const START_OF_DAY_HOUR = 8; const START_OF_DAY_HOUR = 8;
const LATER_TODAY_CUTOFF_HOUR = 17;
const REMINDER_TYPES = { const REMINDER_TYPES = {
AT_DESKTOP: "at_desktop", AT_DESKTOP: "at_desktop",
LATER_TODAY: "later_today", LATER_TODAY: "later_today",
@ -14,7 +15,9 @@ const REMINDER_TYPES = {
TOMORROW: "tomorrow", TOMORROW: "tomorrow",
NEXT_WEEK: "next_week", NEXT_WEEK: "next_week",
NEXT_MONTH: "next_month", NEXT_MONTH: "next_month",
CUSTOM: "custom" CUSTOM: "custom",
LAST_CUSTOM: "last_custom",
NONE: "none"
}; };
export default Controller.extend(ModalFunctionality, { export default Controller.extend(ModalFunctionality, {
@ -27,17 +30,42 @@ export default Controller.extend(ModalFunctionality, {
onCloseWithoutSaving: null, onCloseWithoutSaving: null,
customReminderDate: null, customReminderDate: null,
customReminderTime: null, customReminderTime: null,
lastCustomReminderDate: null,
lastCustomReminderTime: null,
onShow() { onShow() {
this.setProperties({ this.setProperties({
errorMessage: null, errorMessage: null,
name: null, name: null,
selectedReminderType: null, selectedReminderType: REMINDER_TYPES.NONE,
closeWithoutSaving: false, closeWithoutSaving: false,
isSavingBookmarkManually: false, isSavingBookmarkManually: false,
customReminderDate: null, customReminderDate: null,
customReminderTime: null customReminderTime: null,
lastCustomReminderDate: null,
lastCustomReminderTime: null
}); });
this.loadLastUsedCustomReminderDatetime();
},
loadLastUsedCustomReminderDatetime() {
let lastTime = localStorage.lastCustomBookmarkReminderTime;
let lastDate = localStorage.lastCustomBookmarkReminderDate;
if (lastTime && lastDate) {
let parsed = this.parseCustomDateTime(lastDate, lastTime);
if (parsed < this.now()) {
return;
}
this.setProperties({
lastCustomReminderDate: lastDate,
lastCustomReminderTime: lastTime,
parsedLastCustomReminderDatetime: parsed
});
}
}, },
// we always want to save the bookmark unless the user specifically // we always want to save the bookmark unless the user specifically
@ -71,9 +99,29 @@ export default Controller.extend(ModalFunctionality, {
return REMINDER_TYPES; return REMINDER_TYPES;
}, },
@discourseComputed()
showLastCustom() {
return this.lastCustomReminderTime && this.lastCustomReminderDate;
},
@discourseComputed() @discourseComputed()
showLaterToday() { showLaterToday() {
return !this.laterToday().isSame(this.tomorrow(), "date"); let later = this.laterToday();
return (
!later.isSame(this.tomorrow(), "date") &&
later.hour() <= LATER_TODAY_CUTOFF_HOUR
);
},
@discourseComputed("parsedLastCustomReminderDatetime")
lastCustomFormatted(parsedLastCustomReminderDatetime) {
return htmlSafe(
I18n.t("bookmarks.reminders.last_custom", {
date: parsedLastCustomReminderDatetime.format(
I18n.t("dates.long_no_year")
)
})
);
}, },
@discourseComputed() @discourseComputed()
@ -130,12 +178,26 @@ export default Controller.extend(ModalFunctionality, {
const reminderAt = this.reminderAt(); const reminderAt = this.reminderAt();
const reminderAtISO = reminderAt ? reminderAt.toISOString() : null; const reminderAtISO = reminderAt ? reminderAt.toISOString() : null;
if (!reminderAt && this.selectedReminderType === REMINDER_TYPES.CUSTOM) { if (this.selectedReminderType === REMINDER_TYPES.CUSTOM) {
if (!reminderAt) {
return Promise.reject(I18n.t("bookmarks.invalid_custom_datetime")); return Promise.reject(I18n.t("bookmarks.invalid_custom_datetime"));
} }
localStorage.lastCustomBookmarkReminderTime = this.customReminderTime;
localStorage.lastCustomBookmarkReminderDate = this.customReminderDate;
}
let reminderType;
if (this.selectedReminderType === REMINDER_TYPES.NONE) {
reminderType = null;
} else if (this.selectedReminderType === REMINDER_TYPES.LAST_CUSTOM) {
reminderType = REMINDER_TYPES.CUSTOM;
} else {
reminderType = this.selectedReminderType;
}
const data = { const data = {
reminder_type: this.selectedReminderType, reminder_type: reminderType,
reminder_at: reminderAtISO, reminder_at: reminderAtISO,
name: this.name, name: this.name,
post_id: this.model.postId post_id: this.model.postId
@ -148,6 +210,10 @@ export default Controller.extend(ModalFunctionality, {
}); });
}, },
parseCustomDateTime(date, time) {
return moment.tz(date + " " + time, this.userTimezone());
},
reminderAt() { reminderAt() {
if (!this.selectedReminderType) { if (!this.selectedReminderType) {
return; return;
@ -167,9 +233,9 @@ export default Controller.extend(ModalFunctionality, {
case REMINDER_TYPES.NEXT_MONTH: case REMINDER_TYPES.NEXT_MONTH:
return this.nextMonth(); return this.nextMonth();
case REMINDER_TYPES.CUSTOM: case REMINDER_TYPES.CUSTOM:
const customDateTime = moment.tz( const customDateTime = this.parseCustomDateTime(
this.customReminderDate + " " + this.customReminderTime, this.customReminderDate,
this.userTimezone() this.customReminderTime
); );
if (!customDateTime.isValid()) { if (!customDateTime.isValid()) {
this.setProperties({ this.setProperties({
@ -179,6 +245,8 @@ export default Controller.extend(ModalFunctionality, {
return; return;
} }
return customDateTime; return customDateTime;
case REMINDER_TYPES.LAST_CUSTOM:
return this.parsedLastCustomReminderDatetime;
} }
}, },

View File

@ -9,10 +9,6 @@
{{/if}} {{/if}}
<div class="control-group"> <div class="control-group">
<label class="control-label" for="name">
{{i18n 'post.bookmarks.name'}}
</label>
{{input value=name name="name" class="bookmark-name" enter=(action "saveAndClose") placeholder=(i18n "post.bookmarks.name_placeholder")}} {{input value=name name="name" class="bookmark-name" enter=(action "saveAndClose") placeholder=(i18n "post.bookmarks.name_placeholder")}}
</div> </div>
@ -36,6 +32,10 @@
{{tap-tile icon="far-clock" text=nextWeekFormatted tileId=reminderTypes.NEXT_WEEK activeTile=grid.activeTile onChange=(action "selectReminderType")}} {{tap-tile icon="far-clock" text=nextWeekFormatted tileId=reminderTypes.NEXT_WEEK activeTile=grid.activeTile onChange=(action "selectReminderType")}}
{{tap-tile icon="far-calendar-plus" text=nextMonthFormatted tileId=reminderTypes.NEXT_MONTH activeTile=grid.activeTile onChange=(action "selectReminderType")}} {{tap-tile icon="far-calendar-plus" text=nextMonthFormatted tileId=reminderTypes.NEXT_MONTH activeTile=grid.activeTile onChange=(action "selectReminderType")}}
{{tap-tile icon="calendar-alt" text=(I18n "bookmarks.reminders.custom") tileId=reminderTypes.CUSTOM activeTile=grid.activeTile onChange=(action "selectReminderType")}} {{tap-tile icon="calendar-alt" text=(I18n "bookmarks.reminders.custom") tileId=reminderTypes.CUSTOM activeTile=grid.activeTile onChange=(action "selectReminderType")}}
{{#if showLastCustom}}
{{tap-tile icon="undo" text=lastCustomFormatted tileId=reminderTypes.LAST_CUSTOM activeTile=grid.activeTile onChange=(action "selectReminderType")}}
{{/if}}
{{tap-tile icon="ban" text=(I18n "bookmarks.reminders.none") tileId=reminderTypes.NONE activeTile=grid.activeTile onChange=(action "selectReminderType")}}
{{/tap-tile-grid}} {{/tap-tile-grid}}
{{#if customDateTimeSelected}} {{#if customDateTimeSelected}}
<div class="control-group"> <div class="control-group">

View File

@ -7,6 +7,7 @@
$horizontal-tile-padding: 5px; $horizontal-tile-padding: 5px;
.tap-tile { .tap-tile {
color: $primary-high;
padding: 10px $horizontal-tile-padding; padding: 10px $horizontal-tile-padding;
display: flex; display: flex;
flex-direction: column; flex-direction: column;

View File

@ -321,6 +321,8 @@ en:
next_week: "Next week <br/>{{date}}" next_week: "Next week <br/>{{date}}"
next_month: "Next month <br/>{{date}}" next_month: "Next month <br/>{{date}}"
custom: "Custom date and time" custom: "Custom date and time"
last_custom: "Last <br/>{{date}}"
none: "No reminder needed"
drafts: drafts:
resume: "Resume" resume: "Resume"
@ -2676,8 +2678,8 @@ en:
create: "Create bookmark" create: "Create bookmark"
created: "Created" created: "Created"
name: "Name" name: "Name"
name_placeholder: "Name the bookmark to help jog your memory" name_placeholder: "What is this bookmark for?"
set_reminder: "Set a reminder" set_reminder: "Remind me"
actions: actions:
delete_bookmark: delete_bookmark:
name: "Delete bookmark" name: "Delete bookmark"

View File

@ -19,22 +19,27 @@ function mockMomentTz(dateString) {
QUnit.test("showLaterToday when later today is tomorrow do not show", function( QUnit.test("showLaterToday when later today is tomorrow do not show", function(
assert assert
) { ) {
mockMomentTz("2019-12-11T13:00:00Z"); mockMomentTz("2019-12-11T22:00:00");
assert.equal(BookmarkController.get("showLaterToday"), false); assert.equal(BookmarkController.get("showLaterToday"), false);
}); });
QUnit.test("showLaterToday when later today is after 5pm", function(assert) {
mockMomentTz("2019-12-11T15:00:00");
assert.equal(BookmarkController.get("showLaterToday"), false);
});
QUnit.test( QUnit.test(
"showLaterToday when later today is before the end of the day, show", "showLaterToday when later today is before the end of the day, show",
function(assert) { function(assert) {
mockMomentTz("2019-12-11T08:00:00Z"); mockMomentTz("2019-12-11T10:00:00");
assert.equal(BookmarkController.get("showLaterToday"), true); assert.equal(BookmarkController.get("showLaterToday"), true);
} }
); );
QUnit.test("nextWeek gets next week correctly", function(assert) { QUnit.test("nextWeek gets next week correctly", function(assert) {
mockMomentTz("2019-12-11T08:00:00Z"); mockMomentTz("2019-12-11T08:00:00");
assert.equal( assert.equal(
BookmarkController.nextWeek().format("YYYY-MM-DD"), BookmarkController.nextWeek().format("YYYY-MM-DD"),
@ -43,7 +48,7 @@ QUnit.test("nextWeek gets next week correctly", function(assert) {
}); });
QUnit.test("nextMonth gets next month correctly", function(assert) { QUnit.test("nextMonth gets next month correctly", function(assert) {
mockMomentTz("2019-12-11T08:00:00Z"); mockMomentTz("2019-12-11T08:00:00");
assert.equal( assert.equal(
BookmarkController.nextMonth().format("YYYY-MM-DD"), BookmarkController.nextMonth().format("YYYY-MM-DD"),
@ -54,7 +59,7 @@ QUnit.test("nextMonth gets next month correctly", function(assert) {
QUnit.test( QUnit.test(
"nextBusinessDay gets next business day of monday correctly if today is friday", "nextBusinessDay gets next business day of monday correctly if today is friday",
function(assert) { function(assert) {
mockMomentTz("2019-12-13T08:00:00Z"); mockMomentTz("2019-12-13T08:00:00");
assert.equal( assert.equal(
BookmarkController.nextBusinessDay().format("YYYY-MM-DD"), BookmarkController.nextBusinessDay().format("YYYY-MM-DD"),
@ -66,7 +71,7 @@ QUnit.test(
QUnit.test( QUnit.test(
"nextBusinessDay gets next business day of monday correctly if today is saturday", "nextBusinessDay gets next business day of monday correctly if today is saturday",
function(assert) { function(assert) {
mockMomentTz("2019-12-14T08:00:00Z"); mockMomentTz("2019-12-14T08:00:00");
assert.equal( assert.equal(
BookmarkController.nextBusinessDay().format("YYYY-MM-DD"), BookmarkController.nextBusinessDay().format("YYYY-MM-DD"),
@ -78,7 +83,7 @@ QUnit.test(
QUnit.test( QUnit.test(
"nextBusinessDay gets next business day of monday correctly if today is sunday", "nextBusinessDay gets next business day of monday correctly if today is sunday",
function(assert) { function(assert) {
mockMomentTz("2019-12-15T08:00:00Z"); mockMomentTz("2019-12-15T08:00:00");
assert.equal( assert.equal(
BookmarkController.nextBusinessDay().format("YYYY-MM-DD"), BookmarkController.nextBusinessDay().format("YYYY-MM-DD"),
@ -90,7 +95,7 @@ QUnit.test(
QUnit.test( QUnit.test(
"nextBusinessDay gets next business day of thursday correctly if today is wednesday", "nextBusinessDay gets next business day of thursday correctly if today is wednesday",
function(assert) { function(assert) {
mockMomentTz("2019-12-11T08:00:00Z"); mockMomentTz("2019-12-11T08:00:00");
assert.equal( assert.equal(
BookmarkController.nextBusinessDay().format("YYYY-MM-DD"), BookmarkController.nextBusinessDay().format("YYYY-MM-DD"),
@ -100,7 +105,7 @@ QUnit.test(
); );
QUnit.test("tomorrow gets tomorrow correctly", function(assert) { QUnit.test("tomorrow gets tomorrow correctly", function(assert) {
mockMomentTz("2019-12-11T08:00:00Z"); mockMomentTz("2019-12-11T08:00:00");
assert.equal( assert.equal(
BookmarkController.tomorrow().format("YYYY-MM-DD"), BookmarkController.tomorrow().format("YYYY-MM-DD"),
@ -112,7 +117,7 @@ QUnit.test(
"startOfDay changes the time of the provided date to 8:00am correctly", "startOfDay changes the time of the provided date to 8:00am correctly",
function(assert) { function(assert) {
let dt = moment.tz( let dt = moment.tz(
"2019-12-11T11:37:16Z", "2019-12-11T11:37:16",
BookmarkController.currentUser.timezone BookmarkController.currentUser.timezone
); );
@ -126,11 +131,11 @@ QUnit.test(
QUnit.test( QUnit.test(
"laterToday gets 3 hours from now and if before half-past, it sets the time to half-past", "laterToday gets 3 hours from now and if before half-past, it sets the time to half-past",
function(assert) { function(assert) {
mockMomentTz("2019-12-11T08:13:00Z"); mockMomentTz("2019-12-11T08:13:00");
assert.equal( assert.equal(
BookmarkController.laterToday().format("YYYY-MM-DD HH:mm:ss"), BookmarkController.laterToday().format("YYYY-MM-DD HH:mm:ss"),
"2019-12-11 21:30:00" "2019-12-11 11:30:00"
); );
} }
); );
@ -138,11 +143,39 @@ QUnit.test(
QUnit.test( QUnit.test(
"laterToday gets 3 hours from now and if after half-past, it rounds up to the next hour", "laterToday gets 3 hours from now and if after half-past, it rounds up to the next hour",
function(assert) { function(assert) {
mockMomentTz("2019-12-11T08:43:00Z"); mockMomentTz("2019-12-11T08:43:00");
assert.equal( assert.equal(
BookmarkController.laterToday().format("YYYY-MM-DD HH:mm:ss"), BookmarkController.laterToday().format("YYYY-MM-DD HH:mm:ss"),
"2019-12-11 22:00:00" "2019-12-11 12:00:00"
); );
} }
); );
QUnit.test(
"loadLastUsedCustomReminderDatetime fills the custom reminder date + time if present in localStorage",
function(assert) {
mockMomentTz("2019-12-11T08:00:00");
localStorage.lastCustomBookmarkReminderDate = "2019-12-12";
localStorage.lastCustomBookmarkReminderTime = "08:00";
BookmarkController.loadLastUsedCustomReminderDatetime();
assert.equal(BookmarkController.lastCustomReminderDate, "2019-12-12");
assert.equal(BookmarkController.lastCustomReminderTime, "08:00");
}
);
QUnit.test(
"loadLastUsedCustomReminderDatetime does not fills the custom reminder date + time if the datetime in localStorage is < now",
function(assert) {
mockMomentTz("2019-12-11T08:00:00");
localStorage.lastCustomBookmarkReminderDate = "2019-12-11";
localStorage.lastCustomBookmarkReminderTime = "07:00";
BookmarkController.loadLastUsedCustomReminderDatetime();
assert.equal(BookmarkController.lastCustomReminderDate, null);
assert.equal(BookmarkController.lastCustomReminderTime, null);
}
);