MOBILE-4724 ux: Change confirm message when leaving page
parent
9e006424f6
commit
c51cca9fd3
|
@ -1561,9 +1561,9 @@
|
|||
"core.completion-alt-manual-n-override": "completion",
|
||||
"core.completion-alt-manual-y": "completion",
|
||||
"core.completion-alt-manual-y-override": "completion",
|
||||
"core.confirmcanceledit": "local_moodlemobileapp",
|
||||
"core.confirmdeletefile": "repository",
|
||||
"core.confirmleaveunknownchanges": "local_moodlemobileapp",
|
||||
"core.confirmleavepagedescription": "local_moodlemobileapp",
|
||||
"core.confirmleavepagetitle": "local_moodlemobileapp",
|
||||
"core.confirmloss": "local_moodlemobileapp",
|
||||
"core.confirmopeninbrowser": "local_moodlemobileapp",
|
||||
"core.confirmremoveselectedfile": "local_moodlemobileapp",
|
||||
|
@ -2091,6 +2091,7 @@
|
|||
"core.lastmodified": "moodle",
|
||||
"core.lastsync": "local_moodlemobileapp",
|
||||
"core.layoutgrid": "workshopform_rubric",
|
||||
"core.leave": "local_moodlemobileapp",
|
||||
"core.list": "moodle",
|
||||
"core.listsep": "langconfig",
|
||||
"core.loading": "moodle",
|
||||
|
|
|
@ -41,7 +41,6 @@ import { CoreNetwork } from '@services/network';
|
|||
import { CoreSites, CoreSitesReadingStrategy } from '@services/sites';
|
||||
import { CoreSync } from '@services/sync';
|
||||
import { CoreWSError } from '@classes/errors/wserror';
|
||||
import { Translate } from '@singletons';
|
||||
import { CoreEvents } from '@singletons/events';
|
||||
import { CoreForms } from '@singletons/form';
|
||||
import { CoreFileEntry } from '@services/file-helper';
|
||||
|
@ -412,7 +411,7 @@ export default class AddonBlogEditEntryPage implements CanLeave, OnInit, OnDestr
|
|||
|
||||
if ((!this.entry && this.hasDataChangedForNewEntry) || (this.entry && this.hasDataChangedForEdit)) {
|
||||
// Modified, confirm user wants to go back.
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
}
|
||||
|
||||
CoreForms.triggerFormCancelledEvent(this.formElement, CoreSites.getCurrentSiteId());
|
||||
|
|
|
@ -593,7 +593,7 @@ export class AddonCalendarEditEventPage implements OnInit, OnDestroy, CanLeave {
|
|||
async canLeave(): Promise<boolean> {
|
||||
if (AddonCalendarHelper.hasEventDataChanged(this.form.value, this.originalData)) {
|
||||
// Show confirmation if some data has been modified.
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
}
|
||||
|
||||
CoreForms.triggerFormCancelledEvent(this.formElement, this.currentSite.getId());
|
||||
|
|
|
@ -16,7 +16,7 @@ import { Component, Input, ViewChild, ElementRef } from '@angular/core';
|
|||
import { CoreSites } from '@services/sites';
|
||||
import { CoreFormFields, CoreForms } from '@singletons/form';
|
||||
import { CorePromiseUtils } from '@singletons/promise-utils';
|
||||
import { ModalController, Translate } from '@singletons';
|
||||
import { ModalController } from '@singletons';
|
||||
import { AddonModAssignAssign, AddonModAssignPlugin, AddonModAssignSubmission } from '../../services/assign';
|
||||
import { AddonModAssignFeedbackDelegate } from '../../services/feedback-delegate';
|
||||
import { CoreSharedModule } from '@/core/shared.module';
|
||||
|
@ -50,7 +50,7 @@ export class AddonModAssignEditFeedbackModalComponent {
|
|||
async closeModal(): Promise<void> {
|
||||
const changed = await this.hasDataChanged();
|
||||
if (changed) {
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
}
|
||||
|
||||
CoreForms.triggerFormCancelledEvent(this.formElement, CoreSites.getCurrentSiteId());
|
||||
|
|
|
@ -296,7 +296,7 @@ export class AddonModAssignSubmissionComponent implements OnInit, OnDestroy, Can
|
|||
|
||||
if (modified) {
|
||||
// Modified, confirm user wants to go back.
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
|
||||
await this.discardDrafts();
|
||||
}
|
||||
|
|
|
@ -125,7 +125,7 @@ export class AddonModAssignEditPage implements OnInit, OnDestroy, CanLeave {
|
|||
// Check if data has changed.
|
||||
const changed = await this.hasDataChanged();
|
||||
if (changed) {
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
}
|
||||
|
||||
// Nothing has changed or user confirmed to leave. Clear temporary data from plugins.
|
||||
|
|
|
@ -156,7 +156,7 @@ export class AddonModDataEditPage implements OnInit {
|
|||
|
||||
if (changed) {
|
||||
// Show confirmation if some data has been modified.
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
}
|
||||
|
||||
// Delete the local files from the tmp folder.
|
||||
|
|
|
@ -160,7 +160,7 @@ export class AddonModFeedbackFormPage implements OnInit, OnDestroy, CanLeave {
|
|||
if (this.items && !this.completed && this.originalData) {
|
||||
// Form submitted. Check if there is any change.
|
||||
if (!CoreObject.basicLeftCompare(responses, this.originalData, 3)) {
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -303,7 +303,7 @@ export class AddonModForumDiscussionPage implements OnInit, AfterViewInit, OnDes
|
|||
async canLeave(): Promise<boolean> {
|
||||
if (AddonModForumHelper.hasPostDataChanged(this.formData, this.originalData)) {
|
||||
// Show confirmation if some data has been modified.
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
}
|
||||
|
||||
// Delete the local files from the tmp folder.
|
||||
|
|
|
@ -662,7 +662,7 @@ export class AddonModForumNewDiscussionPage implements OnInit, OnDestroy, CanLea
|
|||
|
||||
if (AddonModForumHelper.hasPostDataChanged(this.newDiscussion, this.originalData)) {
|
||||
// Show confirmation if some data has been modified.
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
}
|
||||
|
||||
// Delete the local files from the tmp folder.
|
||||
|
|
|
@ -187,7 +187,7 @@ export class AddonModGlossaryEditPage implements OnInit, CanLeave {
|
|||
|
||||
if (this.hasDataChanged()) {
|
||||
// Show confirmation if some data has been modified.
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
}
|
||||
|
||||
// Delete the local files from the tmp folder.
|
||||
|
|
|
@ -16,7 +16,6 @@ import { Component, OnDestroy, ViewChild } from '@angular/core';
|
|||
|
||||
import { CoreCourseModuleMainActivityPage } from '@features/course/classes/main-activity-page';
|
||||
import { CanLeave } from '@guards/can-leave';
|
||||
import { Translate } from '@singletons';
|
||||
import { AddonModH5PActivityIndexComponent } from '../../components/index';
|
||||
import { CoreAlerts } from '@services/overlays/alerts';
|
||||
|
||||
|
@ -45,7 +44,7 @@ export class AddonModH5PActivityIndexPage extends CoreCourseModuleMainActivityPa
|
|||
|
||||
if (!this.canLeaveSafely) {
|
||||
try {
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmleaveunknownchanges'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
|
||||
return true;
|
||||
} catch {
|
||||
|
|
|
@ -173,7 +173,7 @@ export class AddonModLessonPlayerPage implements OnInit, OnDestroy, CanLeave {
|
|||
if (this.question && !this.eolData && !this.processData && this.originalData) {
|
||||
// Question shown. Check if there is any change.
|
||||
if (!CoreObject.basicLeftCompare(this.questionForm.getRawValue(), this.originalData, 3)) {
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -337,7 +337,7 @@ export class AddonModWikiEditPage implements OnInit, OnDestroy, CanLeave {
|
|||
|
||||
// Check if data has changed.
|
||||
if (this.hasDataChanged()) {
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
}
|
||||
|
||||
CoreForms.triggerFormCancelledEvent(this.formElement, CoreSites.getCurrentSiteId());
|
||||
|
|
|
@ -172,7 +172,7 @@ export class AddonModWorkshopAssessmentPage implements OnInit, OnDestroy, CanLea
|
|||
}
|
||||
|
||||
// Show confirmation if some data has been modified.
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
|
||||
CoreForms.triggerFormCancelledEvent(this.formElement, this.siteId);
|
||||
|
||||
|
|
|
@ -147,7 +147,7 @@ export class AddonModWorkshopEditSubmissionPage implements OnInit, OnDestroy, Ca
|
|||
// Check if data has changed.
|
||||
if (this.hasDataChanged()) {
|
||||
// Show confirmation if some data has been modified.
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
}
|
||||
|
||||
if (this.submission?.attachmentfiles) {
|
||||
|
|
|
@ -184,7 +184,7 @@ export class AddonModWorkshopSubmissionPage implements OnInit, OnDestroy, CanLea
|
|||
}
|
||||
|
||||
// Show confirmation if some data has been modified.
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
|
||||
CoreForms.triggerFormCancelledEvent(this.formElement, this.siteId);
|
||||
|
||||
|
|
|
@ -18,7 +18,7 @@ import { MediaFile } from '@awesome-cordova-plugins/media-capture/ngx';
|
|||
import { CoreFile, CoreFileProvider } from '@services/file';
|
||||
import { CoreMimetypeUtils } from '@services/utils/mimetype';
|
||||
import { CoreTimeUtils } from '@services/utils/time';
|
||||
import { ModalController, Translate } from '@singletons';
|
||||
import { ModalController } from '@singletons';
|
||||
import { CoreError } from '@classes/errors/error';
|
||||
import { CoreCaptureError } from '@classes/errors/captureerror';
|
||||
import { CoreCanceledError } from '@classes/errors/cancelederror';
|
||||
|
@ -235,7 +235,7 @@ export class CoreEmulatorCaptureMediaComponent implements OnInit, OnDestroy {
|
|||
async cancel(): Promise<void> {
|
||||
if (this.hasCaptured) {
|
||||
try {
|
||||
await CoreAlerts.confirm(Translate.instant('core.confirmcanceledit'));
|
||||
await CoreAlerts.confirmLeaveWithChanges();
|
||||
} catch {
|
||||
// Canceled.
|
||||
return;
|
||||
|
|
|
@ -87,12 +87,13 @@ Feature: Test different cases of logout and switch account
|
|||
| Message | An awesome message |
|
||||
And I press the user menu button in the app
|
||||
And I press "Log out" in the app
|
||||
Then I should find "Are you sure you want to leave this page?" in the app
|
||||
Then I should find "Leave page?" in the app
|
||||
And I should find "Unsaved changes will be lost." in the app
|
||||
|
||||
# Check that the app continues working fine if the user cancels the logout.
|
||||
When I press "Cancel" in the app
|
||||
And I press "Forum topic 1" in the app
|
||||
And I press "OK" in the app
|
||||
And I press "Leave" in the app
|
||||
Then I should find "Forum message 1" in the app
|
||||
|
||||
When I press "Forum topic 2" in the app
|
||||
|
@ -105,7 +106,7 @@ Feature: Test different cases of logout and switch account
|
|||
| Message | An awesome message |
|
||||
And I press the user menu button in the app
|
||||
And I press "Log out" in the app
|
||||
And I press "OK" in the app
|
||||
And I press "Leave" in the app
|
||||
And I wait the app to restart
|
||||
Then the header should be "Accounts" in the app
|
||||
|
||||
|
@ -135,12 +136,13 @@ Feature: Test different cases of logout and switch account
|
|||
And I press the user menu button in the app
|
||||
And I press "Switch account" in the app
|
||||
And I press "pau student2" in the app
|
||||
Then I should find "Are you sure you want to leave this page?" in the app
|
||||
Then I should find "Leave page?" in the app
|
||||
And I should find "Unsaved changes will be lost." in the app
|
||||
|
||||
# Check that the app continues working fine if the user cancels the switch account.
|
||||
When I press "Cancel" in the app
|
||||
And I press "Forum topic 1" in the app
|
||||
And I press "OK" in the app
|
||||
And I press "Leave" in the app
|
||||
Then I should find "Forum message 1" in the app
|
||||
|
||||
When I press "Forum topic 2" in the app
|
||||
|
@ -154,7 +156,7 @@ Feature: Test different cases of logout and switch account
|
|||
And I press the user menu button in the app
|
||||
And I press "Switch account" in the app
|
||||
And I press "pau student2" in the app
|
||||
And I press "OK" in the app
|
||||
And I press "Leave" in the app
|
||||
And I wait the app to restart
|
||||
And I press the user menu button in the app
|
||||
Then I should find "pau student2" in the app
|
||||
|
|
|
@ -48,9 +48,9 @@
|
|||
"completion-alt-manual-n-override": "Not completed: {{$a.modname}} (set by {{$a.overrideuser}}). Select to mark as complete.",
|
||||
"completion-alt-manual-y": "Completed: {{$a}}. Select to mark as not complete.",
|
||||
"completion-alt-manual-y-override": "Completed: {{$a.modname}} (set by {{$a.overrideuser}}). Select to mark as not complete.",
|
||||
"confirmcanceledit": "Are you sure you want to leave this page? All changes will be lost.",
|
||||
"confirmdeletefile": "Are you sure you want to delete this file?",
|
||||
"confirmleaveunknownchanges": "Are you sure you want to leave this page? If you have unsaved changes they will be lost.",
|
||||
"confirmleavepagedescription": "Unsaved changes will be lost.",
|
||||
"confirmleavepagetitle": "Leave page?",
|
||||
"confirmloss": "Are you sure? All changes will be lost.",
|
||||
"confirmopeninbrowser": "Do you want to open it in a web browser?",
|
||||
"confirmremoveselectedfile": "This will permanently delete '{{filename}}'. You can't undo this.",
|
||||
|
@ -168,6 +168,7 @@
|
|||
"lastmodified": "Last modified",
|
||||
"lastsync": "Last synchronisation",
|
||||
"layoutgrid": "Grid",
|
||||
"leave": "Leave",
|
||||
"list": "List",
|
||||
"listsep": ",",
|
||||
"loading": "Loading",
|
||||
|
|
|
@ -70,7 +70,7 @@ export class CoreAlertsService {
|
|||
*/
|
||||
confirm<T>(message: string, options: CoreAlertsConfirmOptions = {}): Promise<T> {
|
||||
return new Promise<T>((resolve, reject): void => {
|
||||
const { okText, cancelText, ...alertOptions } = options;
|
||||
const { okText, cancelText, isDestructive, ...alertOptions } = options;
|
||||
const buttons = [
|
||||
{
|
||||
text: cancelText || Translate.instant('core.cancel'),
|
||||
|
@ -81,6 +81,7 @@ export class CoreAlertsService {
|
|||
},
|
||||
{
|
||||
text: okText || Translate.instant('core.ok'),
|
||||
role: isDestructive ? 'destructive' : undefined,
|
||||
handler: (data: T) => {
|
||||
resolve(data);
|
||||
},
|
||||
|
@ -111,34 +112,22 @@ export class CoreAlertsService {
|
|||
async confirmDelete(message: string, options: Omit<AlertOptions, 'message'|'buttons'> = {}): Promise<void> {
|
||||
message = await CoreLang.filterMultilang(message);
|
||||
|
||||
const alertOptions: AlertOptions = {
|
||||
await this.confirm(message, {
|
||||
...options,
|
||||
message,
|
||||
};
|
||||
|
||||
return new Promise((resolve, reject): void => {
|
||||
alertOptions.buttons = [
|
||||
{
|
||||
text: Translate.instant('core.cancel'),
|
||||
role: 'cancel',
|
||||
handler: () => {
|
||||
reject(new CoreCanceledError());
|
||||
},
|
||||
},
|
||||
{
|
||||
text: Translate.instant('core.delete'),
|
||||
role: 'destructive',
|
||||
handler: () => {
|
||||
resolve();
|
||||
},
|
||||
},
|
||||
];
|
||||
|
||||
if (!alertOptions.header) {
|
||||
alertOptions.cssClass = (alertOptions.cssClass || '') + ' core-nohead';
|
||||
okText: Translate.instant('core.delete'),
|
||||
isDestructive: true,
|
||||
});
|
||||
}
|
||||
|
||||
this.show(alertOptions);
|
||||
/**
|
||||
* Show a confirmation modal to confirm leaving a page with unsaves changes.
|
||||
*
|
||||
* @returns Promise resolved if the user confirms and rejected with a canceled error if he cancels.
|
||||
*/
|
||||
async confirmLeaveWithChanges(): Promise<void> {
|
||||
await this.confirm(Translate.instant('core.confirmleavepagedescription'), {
|
||||
header: Translate.instant('core.confirmleavepagetitle'),
|
||||
okText: Translate.instant('core.leave'),
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -504,6 +493,7 @@ export const CoreAlerts = makeSingleton(CoreAlertsService);
|
|||
export type CoreAlertsConfirmOptions = Omit<AlertOptions, 'message'|'buttons'> & {
|
||||
okText?: string; // Text of the OK button. By default, 'OK'.
|
||||
cancelText?: string; // Text of the Cancel button. By default, 'Cancel'.
|
||||
isDestructive?: boolean; // Whether confirming is destructive (will remove data), so the button will have a danger color.
|
||||
};
|
||||
|
||||
/**
|
||||
|
|
|
@ -155,7 +155,8 @@ Feature: It navigates properly using deep links.
|
|||
Then I should find "This link belongs to another site" in the app
|
||||
|
||||
When I press "OK" in the app
|
||||
Then I should find "Are you sure you want to leave this page?" in the app
|
||||
Then I should find "Leave page?" in the app
|
||||
And I should find "Unsaved changes will be lost." in the app
|
||||
|
||||
When I press "Cancel" in the app
|
||||
Then I should not find "Forum message" in the app
|
||||
|
@ -164,7 +165,7 @@ Feature: It navigates properly using deep links.
|
|||
| discussion | user |
|
||||
| Forum topic | student2 |
|
||||
And I press "OK" in the app
|
||||
And I press "OK" in the app
|
||||
And I press "Leave" in the app
|
||||
And I wait the app to restart
|
||||
Then I should find "Forum topic" in the app
|
||||
And I should find "Forum message" in the app
|
||||
|
|
Loading…
Reference in New Issue