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:
parent
a6e9057609
commit
4cce564b35
|
@ -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;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -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">
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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"
|
||||||
|
|
|
@ -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);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
Loading…
Reference in New Issue