// (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 { Component, ElementRef, EventEmitter, Input, OnChanges, OnDestroy, OnInit, Output, SimpleChange, ViewChild, } from '@angular/core'; import { FormControl } from '@angular/forms'; import { CoreDomUtils } from '@services/utils/dom'; import { CoreEvents } from '@singletons/events'; import { CoreSites } from '@services/sites'; import { AddonModForum, AddonModForumAccessInformation, AddonModForumData, AddonModForumDiscussion, AddonModForumPost, AddonModForumProvider, AddonModForumPostFormData, } from '../../services/forum'; import { CoreTag } from '@features/tag/services/tag'; import { Translate } from '@singletons'; import { CoreFileUploader } from '@features/fileuploader/services/fileuploader'; import { AddonModForumSync } from '../../services/forum-sync'; import { CoreSync } from '@services/sync'; import { CoreTextUtils } from '@services/utils/text'; import { AddonModForumHelper } from '../../services/forum-helper'; import { AddonModForumOffline } from '../../services/forum-offline'; import { CoreUtils } from '@services/utils/utils'; import { AddonModForumPostOptionsMenuComponent } from '../post-options-menu/post-options-menu'; import { CoreRatingInfo } from '@features/rating/services/rating'; import { CoreForms } from '@singletons/form'; import { CoreFileEntry } from '@services/file-helper'; import { AddonModForumSharedPostFormData } from '../../pages/discussion/discussion.page'; import { CoreDom } from '@singletons/dom'; /** * Components that shows a discussion post, its attachments and the action buttons allowed (reply, etc.). */ @Component({ selector: 'addon-mod-forum-post', templateUrl: 'post.html', styleUrls: ['post.scss'], }) export class AddonModForumPostComponent implements OnInit, OnDestroy, OnChanges { @Input() post!: AddonModForumPost; // Post. @Input() courseId!: number; // Post's course ID. @Input() discussionId!: number; // Post's' discussion ID. @Input() discussion?: AddonModForumDiscussion; // Post's' discussion, only for starting posts. @Input() component!: string; // Component this post belong to. @Input() componentId!: number; // Component ID. @Input() formData!: AddonModForumSharedPostFormData; // Object with the new post data. Usually shared between posts. @Input() originalData!: Omit; // Original post data. Usually shared between posts. @Input() trackPosts!: boolean; // True if post is being tracked. @Input() forum!: AddonModForumData; // The forum the post belongs to. Required for attachments and offline posts. @Input() accessInfo!: AddonModForumAccessInformation; // Forum access information. @Input() parentSubject?: string; // Subject of parent post. @Input() ratingInfo?: CoreRatingInfo; // Rating info item. @Input() leavingPage?: boolean; // Whether the page that contains this post is being left and will be destroyed. @Input() highlight = false; @Output() onPostChange: EventEmitter = new EventEmitter(); // Event emitted when a reply is posted or modified. @ViewChild('replyFormEl') formElement!: ElementRef; messageControl = new FormControl(); uniqueId!: string; defaultReplySubject!: string; advanced = false; // Display all form fields. tagsEnabled!: boolean; displaySubject = true; optionsMenuEnabled = false; constructor( protected elementRef: ElementRef, ) {} get showForm(): boolean { return this.post.id > 0 ? (!this.formData.isEditing && this.formData.replyingTo === this.post.id) || (!!this.formData.isEditing && this.formData.id === this.post.id) : !!this.formData.isEditing && this.formData.replyingTo === this.post.parentid; } /** * Component being initialized. */ ngOnInit(): void { this.tagsEnabled = CoreTag.areTagsAvailableInSite(); this.uniqueId = this.post.id > 0 ? 'reply' + this.post.id : 'edit' + this.post.parentid; const reTranslated = Translate.instant('addon.mod_forum.re'); this.displaySubject = !this.parentSubject || (this.post.subject != this.parentSubject && this.post.subject != `Re: ${this.parentSubject}` && this.post.subject != `${reTranslated} ${this.parentSubject}`); this.defaultReplySubject = this.post.replysubject || ((this.post.subject.startsWith('Re: ') || this.post.subject.startsWith(reTranslated)) ? this.post.subject : `${reTranslated} ${this.post.subject}`); if (this.post.id < 0) { this.optionsMenuEnabled = true; } else if (this.post.capabilities.delete !== undefined) { this.optionsMenuEnabled = this.post.capabilities.delete === true || this.post.capabilities.edit === true; } else { // Cannot know if the user can edit/delete or not, display the menu if the WebServices are available. this.optionsMenuEnabled = this.post.id < 0 || (AddonModForum.isGetDiscussionPostAvailable() && (AddonModForum.isDeletePostAvailable() || AddonModForum.isUpdatePostAvailable())); } } /** * Detect changes on input properties. */ ngOnChanges(changes: {[name: string]: SimpleChange}): void { if (changes.leavingPage && this.leavingPage) { // Download all courses is enabled now, initialize it. CoreForms.triggerFormCancelledEvent(this.formElement, CoreSites.getCurrentSiteId()); } } /** * Deletes an online post. */ async deletePost(): Promise { try { await CoreDomUtils.showDeleteConfirm('addon.mod_forum.deletesure'); const modal = await CoreDomUtils.showModalLoading('core.deleting', true); try { const response = await AddonModForum.deletePost(this.post.id); const data = { forumId: this.forum.id, discussionId: this.discussionId, cmId: this.forum.cmid, deleted: response.status, post: this.post, }; CoreEvents.trigger( AddonModForumProvider.CHANGE_DISCUSSION_EVENT, data, CoreSites.getCurrentSiteId(), ); CoreDomUtils.showToast('addon.mod_forum.deletedpost', true); } catch (error) { CoreDomUtils.showErrorModal(error); } finally { modal.dismiss(); } } catch (error) { // Do nothing. } } /** * Set data to new/edit post, clearing temporary files and updating original data. * * @param replyingTo Id of post beeing replied. * @param isEditing True it's an offline reply beeing edited, false otherwise. * @param subject Subject of the reply. * @param message Message of the reply. * @param files Reply attachments. * @param isPrivate True if it's private reply. * @param postId The post ID if user is editing an online post. */ protected setFormData( replyingTo?: number, isEditing?: boolean, subject?: string, message?: string, files?: CoreFileEntry[], isPrivate?: boolean, postId?: number, ): void { // Delete the local files from the tmp folder if any. CoreFileUploader.clearTmpFiles(this.formData.files); this.formData.replyingTo = replyingTo || 0; this.formData.isEditing = !!isEditing; this.formData.subject = subject || this.defaultReplySubject || ''; this.formData.message = message || null; this.formData.files = files || []; this.formData.isprivatereply = !!isPrivate; this.formData.id = postId; // Update rich text editor. this.messageControl.setValue(this.formData.message); // Update original data. this.originalData.subject = this.formData.subject; this.originalData.message = this.formData.message; this.originalData.files = this.formData.files.slice(); this.originalData.isprivatereply = this.formData.isprivatereply; // Show advanced fields if any of them has not the default value. this.advanced = this.formData.files.length > 0; } /** * Show the context menu. * * @param event Click Event. */ async showOptionsMenu(event: Event): Promise { const popoverData = await CoreDomUtils.openPopover<{ action?: string }>({ component: AddonModForumPostOptionsMenuComponent, componentProps: { post: this.post, forumId: this.forum.id, cmId: this.forum.cmid, }, event, waitForDismissCompleted: true, }); if (popoverData && popoverData.action) { switch (popoverData.action) { case 'edit': this.editPost(); break; case 'delete': this.deletePost(); break; case 'deleteoffline': this.discardOfflineReply(); break; } } } /** * Set this post as being replied to. * * @param event Click event. */ async showReplyForm(event: Event): Promise { event.preventDefault(); event.stopPropagation(); if (this.formData.isEditing) { // User is editing a post, data needs to be resetted. Ask confirm if there is unsaved data. try { await this.confirmDiscard(); this.setFormData(this.post.id); this.scrollToForm(); } catch { // Cancelled. } return; } if (!this.formData.replyingTo) { // User isn't replying, it's a brand new reply. Initialize the data. this.setFormData(this.post.id); } else { // The post being replied has changed but the data will be kept. this.formData.replyingTo = this.post.id; if (this.formData.subject == this.originalData.subject) { // Update subject only if it hadn't been modified this.formData.subject = this.defaultReplySubject; this.originalData.subject = this.defaultReplySubject; } this.messageControl.setValue(this.formData.message); } this.scrollToForm(); } /** * Set this post as being edited to. */ async editPost(): Promise { // Ask confirm if there is unsaved data. try { await this.confirmDiscard(); this.formData.syncId = AddonModForumSync.getDiscussionSyncId(this.discussionId); CoreSync.blockOperation(AddonModForumProvider.COMPONENT, this.formData.syncId); this.setFormData( this.post.parentid, true, this.post.subject, this.post.message, this.post.attachments, this.post.isprivatereply, this.post.id > 0 ? this.post.id : undefined, ); this.scrollToForm(); } catch { // Cancelled. } } /** * Message changed. * * @param text The new text. */ onMessageChange(text: string): void { this.formData.message = text; } /** * Reply to this post or edit post data. */ async send(): Promise { if (!this.formData.subject) { CoreDomUtils.showErrorModal('addon.mod_forum.erroremptysubject', true); return; } if (!this.formData.message) { CoreDomUtils.showErrorModal('addon.mod_forum.erroremptymessage', true); return; } let saveOffline = false; let message = this.formData.message; const subject = this.formData.subject; const replyingTo = this.formData.replyingTo!; const files = this.formData.files || []; const isEditOnline = this.formData.id && this.formData.id > 0; const modal = await CoreDomUtils.showModalLoading('core.sending', true); // Add some HTML to the message if needed. message = CoreTextUtils.formatHtmlLines(message); // Upload attachments first if any. let attachments; try { if (files.length) { try { attachments = await AddonModForumHelper.uploadOrStoreReplyFiles( this.forum.id, isEditOnline ? this.formData.id! : replyingTo, files, false, ); } catch (error) { // Cannot upload them in online, save them in offline. if (!this.forum.id || isEditOnline || CoreUtils.isWebServiceError(error)) { // Cannot store them in offline. Reject. throw error; } saveOffline = true; attachments = await AddonModForumHelper.uploadOrStoreReplyFiles(this.forum.id, replyingTo, files, true); } } let sent = false; if (isEditOnline) { sent = await AddonModForum.updatePost(this.formData.id!, subject, message, { attachmentsid: attachments, }); } else if (saveOffline) { // Save post in offline. await AddonModForumOffline.replyPost( replyingTo, this.discussionId, this.forum.id, this.forum.name, this.courseId, subject, message, { attachmentsid: attachments, private: !!this.formData.isprivatereply, }, ); // Set sent to false since it wasn't sent to server. sent = false; } else { // Try to send it to server. // Don't allow offline if there are attachments since they were uploaded fine. sent = await AddonModForum.replyPost( replyingTo, this.discussionId, this.forum.id, this.forum.name, this.courseId, subject, message, { attachmentsid: attachments, private: !!this.formData.isprivatereply, }, undefined, !files.length, ); } if (sent && this.forum.id) { // Data sent to server, delete stored files (if any). AddonModForumHelper.deleteReplyStoredFiles(this.forum.id, replyingTo); } // Reset data. this.setFormData(); this.onPostChange.emit(); CoreForms.triggerFormSubmittedEvent(this.formElement, sent, CoreSites.getCurrentSiteId()); this.unblockOperation(); } catch (error) { CoreDomUtils.showErrorModalDefault( error, isEditOnline ? 'addon.mod_forum.couldnotupdate' : 'addon.mod_forum.couldnotadd', true, ); } finally { modal.dismiss(); } } /** * Cancel reply. */ async cancel(): Promise { try { await this.confirmDiscard(); // Reset data. this.setFormData(); CoreForms.triggerFormCancelledEvent(this.formElement, CoreSites.getCurrentSiteId()); this.unblockOperation(); } catch (error) { // Cancelled. } } /** * Discard offline reply. */ async discardOfflineReply(): Promise { try { await CoreDomUtils.showDeleteConfirm(); const promises: Promise[] = []; promises.push(AddonModForumOffline.deleteReply(this.post.parentid!)); if (this.forum.id) { promises.push(AddonModForumHelper.deleteReplyStoredFiles(this.forum.id, this.post.parentid!).catch(() => { // Ignore errors, maybe there are no files. })); } await CoreUtils.ignoreErrors(Promise.all(promises)); // Reset data. this.setFormData(); this.onPostChange.emit(); this.unblockOperation(); } catch (error) { // Cancelled. } } /** * Function called when rating is updated online. */ ratingUpdated(): void { AddonModForum.invalidateDiscussionPosts(this.discussionId, this.forum.id); } /** * Show or hide advanced form fields. */ toggleAdvanced(): void { this.advanced = !this.advanced; } /** * Component being destroyed. */ ngOnDestroy(): void { this.unblockOperation(); } /** * Confirm discard changes if any. * * @return Promise resolved if the user confirms or data was not changed and rejected otherwise. */ protected async confirmDiscard(): Promise { if (AddonModForumHelper.hasPostDataChanged(this.formData, this.originalData)) { // Show confirmation if some data has been modified. await CoreDomUtils.showConfirm(Translate.instant('core.confirmloss')); } this.unblockOperation(); } /** * Unblock operation if there's any blocked operation. */ protected unblockOperation(): void { if (!this.formData.syncId) { return; } CoreSync.unblockOperation(AddonModForumProvider.COMPONENT, this.formData.syncId); delete this.formData.syncId; } /** * Scroll to reply/edit form. * * @return Promise resolved when done. */ protected async scrollToForm(): Promise { await CoreDom.scrollToElement( this.elementRef.nativeElement, '#addon-forum-reply-edit-form-' + this.uniqueId, ); } }