2024-05-31 09:38:54 +02:00

380 lines
14 KiB
TypeScript

// (C) Copyright 2015 Moodle Pty Ltd.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
import { ContextLevel } from '@/core/constants';
import { CoreSharedModule } from '@/core/shared.module';
import { ADDON_BLOG_ENTRY_UPDATED } from '@addons/blog/constants';
import {
AddonBlog,
AddonBlogAddEntryOption,
AddonBlogFilter,
AddonBlogPost,
AddonBlogProvider,
AddonBlogPublishState,
} from '@addons/blog/services/blog';
import { Component, ElementRef, OnInit, ViewChild } from '@angular/core';
import { FormControl, FormGroup, Validators } from '@angular/forms';
import { CoreError } from '@classes/errors/error';
import { CoreCommentsComponentsModule } from '@features/comments/components/components.module';
import { CoreCourse } from '@features/course/services/course';
import { CoreCourseHelper, CoreCourseModuleData } from '@features/course/services/course-helper';
import { CoreCourseBasicData } from '@features/courses/services/courses';
import { CoreEditorComponentsModule } from '@features/editor/components/components.module';
import { CoreFileUploader } from '@features/fileuploader/services/fileuploader';
import { CoreTagComponentsModule } from '@features/tag/components/components.module';
import { CanLeave } from '@guards/can-leave';
import { CoreNavigator } from '@services/navigator';
import { CoreSites, CoreSitesReadingStrategy } from '@services/sites';
import { CoreDomUtils } from '@services/utils/dom';
import { CoreUtils } from '@services/utils/utils';
import { CoreWSFile } from '@services/ws';
import { Translate } from '@singletons';
import { CoreEvents } from '@singletons/events';
import { CoreForms } from '@singletons/form';
@Component({
selector: 'addon-blog-edit-entry',
templateUrl: './edit-entry.html',
standalone: true,
imports: [
CoreEditorComponentsModule,
CoreSharedModule,
CoreCommentsComponentsModule,
CoreTagComponentsModule,
],
})
export class AddonBlogEditEntryPage implements CanLeave, OnInit {
@ViewChild('editEntryForm') formElement!: ElementRef;
publishState = AddonBlogPublishState;
form = new FormGroup({
subject: new FormControl<string>('', { nonNullable: true, validators: [Validators.required] }),
summary: new FormControl<string>('', { nonNullable: true, validators: [Validators.required] }),
publishState: new FormControl<AddonBlogPublishState>(
AddonBlogPublishState.site,
{ nonNullable: true, validators: [Validators.required] },
),
associateWithCourse: new FormControl<boolean>(false, { nonNullable: true, validators: [Validators.required] }),
associateWithModule: new FormControl<boolean>(false, { nonNullable: true, validators: [Validators.required] }),
});
entry?: AddonBlogPost;
loaded = false;
maxFiles = 99;
initialFiles: CoreWSFile[] = [];
files: CoreWSFile[] = [];
courseId?: number;
modId?: number;
userId?: number;
associatedCourse?: CoreCourseBasicData;
associatedModule?: CoreCourseModuleData;
associationsExpanded = false;
moduleContext: ContextLevel = ContextLevel.MODULE;
courseContext: ContextLevel = ContextLevel.COURSE;
contextLevel: ContextLevel = ContextLevel.SYSTEM;
contextInstanceId = 0;
component = AddonBlogProvider.COMPONENT;
siteHomeId?: number;
forceLeave = false;
/**
* Gives if the form is not pristine. (only for existing entries)
*
* @returns Data has changed or not.
*/
get hasDataChangedForEdit(): boolean {
const form = this.form.controls;
return form.summary.value !== this.entry?.summary ||
form.subject.value !== this.entry?.subject ||
form.publishState.value !== this.entry?.publishstate ||
CoreFileUploader.areFileListDifferent(this.files, this.initialFiles) ||
form.associateWithModule.value !== (this.entry?.moduleid !== 0) ||
form.associateWithCourse.value !== (this.entry?.courseid !== 0);
}
/**
* Gives if the form is not pristine. (only for new entries)
*
* @returns Data has changed or not.
*/
get hasDataChangedForNewEntry(): boolean {
const form = this.form.controls;
return form.subject.value !== '' ||
form.summary.value !== '' ||
form.publishState.value !== AddonBlogPublishState.site ||
CoreFileUploader.areFileListDifferent(this.files, this.initialFiles);
}
/**
* @inheritdoc
*/
async ngOnInit(): Promise<void> {
const site = await CoreSites.getSite();
const isEditingEnabled = await AddonBlog.isEditingEnabled();
if (!site || !isEditingEnabled) {
return CoreNavigator.back();
}
const entryId = CoreNavigator.getRouteNumberParam('id');
const lastModified = CoreNavigator.getRouteNumberParam('lastModified');
const filters: AddonBlogFilter | undefined = CoreNavigator.getRouteParam('filters');
const courseId = CoreNavigator.getRouteNumberParam('courseId');
this.userId = CoreNavigator.getRouteNumberParam('userId');
this.siteHomeId = CoreSites.getCurrentSiteHomeId();
if (courseId) {
this.courseId = courseId;
this.form.controls.associateWithCourse.setValue(true);
const { course } = await CoreCourseHelper.getCourse(this.courseId);
this.associatedCourse = course;
}
if (!entryId) {
this.loaded = true;
return;
}
try {
this.entry = await this.getEntry({ filters, lastModified, entryId });
this.files = this.entry.attachmentfiles ?? [];
this.initialFiles = [...this.files];
this.courseId = this.courseId || this.entry.courseid;
this.modId = this.entry.coursemoduleid ? this.entry.coursemoduleid : CoreNavigator.getRouteNumberParam('cmId');
if (this.courseId) {
this.form.controls.associateWithCourse.setValue(true);
const { course } = await CoreCourseHelper.getCourse(this.courseId);
this.associatedCourse = course;
}
if (this.modId) {
this.form.controls.associateWithModule.setValue(true);
this.associatedModule = await CoreCourse.getModule(this.modId);
}
} catch (error) {
CoreDomUtils.showErrorModalDefault(error, 'Error retrieving data.');
this.forceLeave = true;
CoreNavigator.back();
return;
}
this.form.setValue({
subject: this.entry?.subject ?? '',
summary: this.entry?.summary ?? '',
publishState: this.entry?.publishstate ?? AddonBlogPublishState.site,
associateWithCourse: this.form.controls.associateWithCourse.value,
associateWithModule: this.form.controls.associateWithModule.value,
});
this.calculateContext();
this.loaded = true;
}
/**
* Retrieves blog entry.
*
* @returns Blog entry.
*/
protected async getEntry(params: AddonBlogEditEntryGetEntryParams): Promise<AddonBlogPost> {
try {
const { entries } = await AddonBlog.getEntries(
{ entryid: params.entryId },
{ readingStrategy: CoreSitesReadingStrategy.PREFER_NETWORK },
);
const selectedEntry = entries.find(entry => entry.id === params.entryId);
if (!selectedEntry) {
throw new CoreError('Entry not found');
}
if (params.filters && params.lastModified && selectedEntry.lastmodified < params.lastModified) {
throw new CoreError('Entry is outdated');
}
return selectedEntry;
} catch (error) {
if (!params.filters || CoreUtils.isWebServiceError(error)) {
// Cannot get the entry, reject.
throw error;
}
const updatedEntries = await AddonBlog.getEntries(params.filters);
const entry = updatedEntries.entries.find(entry => entry.id === params.entryId);
if (!entry) {
throw error;
}
return entry;
}
}
/**
* Calculate context level and context instance.
*/
calculateContext(): void {
// Calculate the context level.
if (this.userId && !this.courseId && !this.modId) {
this.contextLevel = ContextLevel.USER;
this.contextInstanceId = this.userId;
} else if (this.courseId && this.courseId != this.siteHomeId) {
this.contextLevel = ContextLevel.COURSE;
this.contextInstanceId = this.courseId;
} else {
this.contextLevel = ContextLevel.SYSTEM;
this.contextInstanceId = 0;
}
}
/**
* Update or create entry.
*
* @returns Promise resolved when done.
*/
async save(): Promise<void> {
const { summary, subject, publishState } = this.form.value;
if (!subject || !summary || !publishState) {
return;
}
const loading = await CoreDomUtils.showModalLoading('core.sending', true);
if (this.entry) {
try {
if (!CoreFileUploader.areFileListDifferent(this.files, this.initialFiles)) {
return await this.saveEntry();
}
const { attachmentsid } = await AddonBlog.prepareEntryForEdition({ entryid: this.entry.id });
const removedFiles = CoreFileUploader.getFilesToDelete(this.initialFiles, this.files);
if (removedFiles.length) {
await CoreFileUploader.deleteDraftFiles(attachmentsid, removedFiles);
}
await CoreFileUploader.uploadFiles(attachmentsid, this.files);
return await this.saveEntry(attachmentsid);
} catch (error) {
CoreDomUtils.showErrorModalDefault(error, 'Error updating entry.');
} finally {
await loading.dismiss();
}
return;
}
try {
if (!this.files.length) {
return await this.saveEntry();
}
const attachmentId = await CoreFileUploader.uploadOrReuploadFiles(this.files, this.component);
await this.saveEntry(attachmentId);
} catch (error) {
CoreDomUtils.showErrorModalDefault(error, 'Error creating entry.');
} finally {
await loading.dismiss();
}
}
/**
* Expand or collapse associations.
*/
toggleAssociations(): void {
this.associationsExpanded = !this.associationsExpanded;
}
/**
* Check if the user can leave the view. If there are changes to be saved, it will ask for confirm.
*
* @returns Promise resolved with true if can leave the view, rejected otherwise.
*/
async canLeave(): Promise<boolean> {
if (this.forceLeave) {
return true;
}
if ((!this.entry && this.hasDataChangedForNewEntry) || (this.entry && this.hasDataChangedForEdit)) {
// Modified, confirm user wants to go back.
await CoreDomUtils.showConfirm(Translate.instant('core.confirmcanceledit'));
}
CoreForms.triggerFormCancelledEvent(this.formElement, CoreSites.getCurrentSiteId());
return true;
}
/**
* Add attachment to options list.
*
* @param attachmentsId Attachment ID.
* @param options Options list.
*/
addAttachments(attachmentsId: number | undefined, options: AddonBlogAddEntryOption[]): void {
if (attachmentsId === undefined) {
return;
}
options.push({ name: 'attachmentsid', value: attachmentsId });
}
/**
* Create or update entry.
*
* @param attachmentsId Attachments.
* @returns Promise resolved when done.
*/
async saveEntry(attachmentsId?: number): Promise<void> {
const { summary, subject, publishState } = this.form.value;
if (!summary || !subject || !publishState) {
return;
}
const options: AddonBlogAddEntryOption[] = [
{ name: 'publishstate', value: publishState },
{ name: 'courseassoc', value: this.form.controls.associateWithCourse.value && this.courseId ? this.courseId : 0 },
{ name: 'modassoc', value: this.form.controls.associateWithModule.value && this.modId ? this.modId : 0 },
];
this.addAttachments(attachmentsId, options);
this.entry
? await AddonBlog.updateEntry({ subject, summary, summaryformat: 1, options , entryid: this.entry.id })
: await AddonBlog.addEntry({ subject, summary, summaryformat: 1, options });
CoreEvents.trigger(ADDON_BLOG_ENTRY_UPDATED);
this.forceLeave = true;
CoreForms.triggerFormSubmittedEvent(this.formElement, true, CoreSites.getCurrentSiteId());
return CoreNavigator.back();
}
}
type AddonBlogEditEntryGetEntryParams = {
entryId: number;
filters?: AddonBlogFilter;
lastModified?: number;
};