UNPKG

@hmcts/media-viewer

Version:
177 lines 33.1 kB
import { Component, EventEmitter, Input, Output, ViewChild } from '@angular/core'; import { distinctUntilChanged } from 'rxjs/operators'; import * as fromSelector from '../../../store/selectors/annotation.selectors'; import * as i0 from "@angular/core"; import * as i1 from "@ngrx/store"; import * as i2 from "./comment.service"; import * as i3 from "../../services/tags/tags.services"; import * as i4 from "@angular/common"; import * as i5 from "@angular/forms"; import * as i6 from "./text-highlight/text-highlight.directive"; import * as i7 from "./textarea-auto-expand/textarea-auto-expand.directive"; import * as i8 from "../../tags/tags.component"; import * as i9 from "rpx-xui-translation"; import * as i10 from "../../pipes/date/date.pipe"; export class CommentComponent { constructor(store, commentService, tagsServices) { this.store = store; this.commentService = commentService; this.tagsServices = tagsServices; this.CHAR_LIMIT = 5000; this.totalPrevPagesHeight = 0; this.hasUnsavedChanges = false; this.commentClick = new EventEmitter(); this.renderComments = new EventEmitter(); this.delete = new EventEmitter(); this.updated = new EventEmitter(); this.changes = new EventEmitter(); this.rotate = 0; this.zoom = 1; } ngOnInit() { this.subscriptions = this.store.select(fromSelector.getComponentSearchText) .pipe(distinctUntilChanged()).subscribe(searchString => this.searchString = searchString); this.reRenderComments(); this.marginToComment$ = this.commentService.marginToCommentEmitter.asObservable(); } ngAfterContentInit() { if (this.tagItems && this.tagItems.length) { this.tagsServices.updateTagItems(this.tagItems, this._comment.annotationId); } } ngOnDestroy() { this.subscriptions.unsubscribe(); } set comment(comment) { this._comment = { ...comment }; this.page = this._comment.page; this.lastUpdate = comment.lastModifiedDate ? comment.lastModifiedDate : comment.createdDate; this.author = comment.createdByDetails; this.createdBy = comment.createdBy; this.editor = comment.lastModifiedByDetails; this.originalComment = comment.content; this.fullComment = this.originalComment; this.selected = this._comment.selected; this._editable = this._comment.editable; this.tagItems = this._comment.tags; const pageMarginBottom = 10; this.totalPrevPagesHeight = 0; for (let i = 0; i < this.page - 1; i++) { const height = this._comment.pages[i + 1] ? this._comment.pages[i + 1].styles.height : undefined; if (height) { this.totalPrevPagesHeight += height + pageMarginBottom; } } } get comment() { return this._comment; } set annotation(annotation) { this._rectangle = annotation.rectangles .reduce((prev, current) => prev.y < current.y ? prev : current); const actualHeight = this._comment.pages[this.page].styles.height / this.zoom; switch (this.rotate) { case 90: this.rectTop = this._rectangle.x; break; case 180: this.rectTop = actualHeight - (this._rectangle.y + this._rectangle.height); break; case 270: this.rectTop = actualHeight - (this._rectangle.x + this._rectangle.width); break; default: this.rectTop = this._rectangle.y; } this.rectLeft = this._rectangle.x; } get editable() { return this._editable; } onCommentChange(updatedComment) { this.hasUnsavedChanges = this.originalComment.substring(0, this.CHAR_LIMIT) !== updatedComment.substring(0, this.CHAR_LIMIT); this.commentService.onCommentChange(this.hasUnsavedChanges); } deleteOrCancel() { if (!this.editable) { this.delete.emit(this._comment); } else { this.hasUnsavedChanges = false; this._editable = false; this.fullComment = this.originalComment; this.changes.emit(false); if (!this.author && !this.fullComment) { this.delete.emit(this._comment); } } } editOrSave() { if (!this.editable) { this._editable = true; } else { this._comment.content = this.fullComment.substring(0, this.CHAR_LIMIT); const tags = this.tagsServices.getNewTags(this._comment.annotationId); const payload = { comment: this._comment, tags }; this.updated.emit(payload); this.hasUnsavedChanges = false; this._editable = false; this.changes.emit(false); } } onCommentClick() { if (!this.selected) { this.selected = true; this._editable = false; this.commentClick.emit({ annotationId: this._comment.annotationId, editable: this._editable, selected: true }); } } reRenderComments() { this.renderComments.emit(this._comment); } get commentTop() { return this.totalPrevPagesHeight + (this.rectTop * this.zoom); } get height() { return this.form.nativeElement.getBoundingClientRect().height / this.zoom; } /** @nocollapse */ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: CommentComponent, deps: [{ token: i1.Store }, { token: i2.CommentService }, { token: i3.TagsServices }], target: i0.ɵɵFactoryTarget.Component }); } /** @nocollapse */ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: CommentComponent, selector: "mv-anno-comment", inputs: { rotate: "rotate", zoom: "zoom", index: "index", page: "page", comment: "comment", annotation: "annotation" }, outputs: { commentClick: "commentClick", renderComments: "renderComments", delete: "delete", updated: "updated", changes: "changes" }, viewQueries: [{ propertyName: "form", first: true, predicate: ["form"], descendants: true }, { propertyName: "editableComment", first: true, predicate: ["editableComment"], descendants: true }], ngImport: i0, template: "<div\n #form\n (click)=\"onCommentClick()\"\n class=\"aui-comment\"\n [ngClass]=\"{ stylestoggle: (marginToComment$ | async) }\"\n [style.top.px]=\"commentTop\"\n [style.zIndex]=\"selected ? 100 : 0\"\n>\n <div id=\"detailsWrapper {{ index }}\" class=\"aui-comment__header\">\n <span *ngIf=\"author && !editor\" class=\"aui-comment__author\">\n {{ author.forename }} {{ author.surname }}\n </span>\n <span *ngIf=\"editor\" class=\"aui-comment__author\">\n {{ editor.forename }} {{ editor.surname }}\n </span>\n <time [hidden]=\"!selected && !this.editable\" class=\"aui-comment__meta\">\n {{ lastUpdate | momentDate : \"d MMMM y h:mm a\" }}\n </time>\n </div>\n <mv-tags\n [tagItems]=\"tagItems\"\n [userId]=\"createdBy\"\n [editable]=\"editable\"\n [annoId]=\"_comment.annotationId\"\n >\n </mv-tags>\n <textarea\n *ngIf=\"selected && editable\"\n #editableComment\n mvTextAreaAutoExpand\n type=\"text\"\n required\n name=\"content\"\n [maxlength]=\"CHAR_LIMIT\"\n class=\"aui-comment__content form-control mimic-focus edit-mode expanded\"\n [(ngModel)]=\"fullComment\"\n (ngModelChange)=\"reRenderComments(); onCommentChange($event)\"\n aria-label=\"comment\"\n >\n </textarea>\n <p\n *ngIf=\"!editable\"\n mvTextHighlight\n class=\"commentText\"\n [textToHighlight]=\"searchString\"\n >\n {{ fullComment }}\n </p>\n <div\n *ngIf=\"\n selected ||\n this.editable ||\n (!fullComment.length && tagItems && !tagItems.length)\n \"\n class=\"aui-comment__footer commentBtns\"\n >\n <button\n class=\"govuk-button\"\n type=\"button\"\n role=\"button\"\n (click)=\"editOrSave()\"\n >\n {{ !editable ? (\"Edit\" | rpxTranslate) : (\"Save\" | rpxTranslate) }}\n </button>\n <button\n type=\"button\"\n role=\"button\"\n class=\"govuk-button govuk-button--secondary\"\n (click)=\"deleteOrCancel()\"\n >\n {{ !editable ? (\"Delete\" | rpxTranslate) : (\"Cancel\" | rpxTranslate) }}\n </button>\n </div>\n <span class=\"aui-comment__private\">{{ \"private\" | rpxTranslate }}</span>\n</div>\n", dependencies: [{ kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i5.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i5.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i5.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i5.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i5.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i6.TextHighlightDirective, selector: "[mvTextHighlight]", inputs: ["textToHighlight"] }, { kind: "directive", type: i7.TextareaAutoExpandDirective, selector: "[mvTextAreaAutoExpand]" }, { kind: "component", type: i8.TagsComponent, selector: "mv-tags", inputs: ["tagItems", "userId", "editable", "annoId"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }, { kind: "pipe", type: i9.RpxTranslatePipe, name: "rpxTranslate" }, { kind: "pipe", type: i10.MomentDatePipe, name: "momentDate" }] }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: CommentComponent, decorators: [{ type: Component, args: [{ selector: 'mv-anno-comment', template: "<div\n #form\n (click)=\"onCommentClick()\"\n class=\"aui-comment\"\n [ngClass]=\"{ stylestoggle: (marginToComment$ | async) }\"\n [style.top.px]=\"commentTop\"\n [style.zIndex]=\"selected ? 100 : 0\"\n>\n <div id=\"detailsWrapper {{ index }}\" class=\"aui-comment__header\">\n <span *ngIf=\"author && !editor\" class=\"aui-comment__author\">\n {{ author.forename }} {{ author.surname }}\n </span>\n <span *ngIf=\"editor\" class=\"aui-comment__author\">\n {{ editor.forename }} {{ editor.surname }}\n </span>\n <time [hidden]=\"!selected && !this.editable\" class=\"aui-comment__meta\">\n {{ lastUpdate | momentDate : \"d MMMM y h:mm a\" }}\n </time>\n </div>\n <mv-tags\n [tagItems]=\"tagItems\"\n [userId]=\"createdBy\"\n [editable]=\"editable\"\n [annoId]=\"_comment.annotationId\"\n >\n </mv-tags>\n <textarea\n *ngIf=\"selected && editable\"\n #editableComment\n mvTextAreaAutoExpand\n type=\"text\"\n required\n name=\"content\"\n [maxlength]=\"CHAR_LIMIT\"\n class=\"aui-comment__content form-control mimic-focus edit-mode expanded\"\n [(ngModel)]=\"fullComment\"\n (ngModelChange)=\"reRenderComments(); onCommentChange($event)\"\n aria-label=\"comment\"\n >\n </textarea>\n <p\n *ngIf=\"!editable\"\n mvTextHighlight\n class=\"commentText\"\n [textToHighlight]=\"searchString\"\n >\n {{ fullComment }}\n </p>\n <div\n *ngIf=\"\n selected ||\n this.editable ||\n (!fullComment.length && tagItems && !tagItems.length)\n \"\n class=\"aui-comment__footer commentBtns\"\n >\n <button\n class=\"govuk-button\"\n type=\"button\"\n role=\"button\"\n (click)=\"editOrSave()\"\n >\n {{ !editable ? (\"Edit\" | rpxTranslate) : (\"Save\" | rpxTranslate) }}\n </button>\n <button\n type=\"button\"\n role=\"button\"\n class=\"govuk-button govuk-button--secondary\"\n (click)=\"deleteOrCancel()\"\n >\n {{ !editable ? (\"Delete\" | rpxTranslate) : (\"Cancel\" | rpxTranslate) }}\n </button>\n </div>\n <span class=\"aui-comment__private\">{{ \"private\" | rpxTranslate }}</span>\n</div>\n" }] }], ctorParameters: () => [{ type: i1.Store }, { type: i2.CommentService }, { type: i3.TagsServices }], propDecorators: { commentClick: [{ type: Output }], renderComments: [{ type: Output }], delete: [{ type: Output }], updated: [{ type: Output }], changes: [{ type: Output }], rotate: [{ type: Input }], zoom: [{ type: Input }], index: [{ type: Input }], page: [{ type: Input }], form: [{ type: ViewChild, args: ['form', { static: false }] }], editableComment: [{ type: ViewChild, args: ['editableComment', { static: false }] }], comment: [{ type: Input }], annotation: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"comment.component.js","sourceRoot":"","sources":["../../../../../../../projects/media-viewer/src/lib/annotations/comment-set/comment/comment.component.ts","../../../../../../../projects/media-viewer/src/lib/annotations/comment-set/comment/comment.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterContentInit,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output,\n  ViewChild\n} from '@angular/core';\nimport {Comment} from './comment.model';\nimport {User} from '../../models/user.model';\nimport {Rectangle} from '../../annotation-set/annotation-view/rectangle/rectangle.model';\nimport {SelectionAnnotation} from '../../models/event-select.model';\nimport {CommentService} from './comment.service';\nimport {TagsModel} from '../../models/tags.model';\nimport {TagsServices} from '../../services/tags/tags.services';\nimport { Observable, Subscription } from 'rxjs';\nimport { distinctUntilChanged } from 'rxjs/operators';\nimport {Store} from '@ngrx/store';\nimport * as fromStore from '../../../store/reducers/reducers';\nimport * as fromSelector from '../../../store/selectors/annotation.selectors';\nimport { AnnotationSet } from '../../annotation-set/annotation-set.model';\nimport { Annotation } from '../../annotation-set/annotation-view/annotation.model';\n\n@Component({\n  selector: 'mv-anno-comment',\n  templateUrl: './comment.component.html'\n})\nexport class CommentComponent implements OnInit, OnDestroy, AfterContentInit {\n\n  CHAR_LIMIT = 5000;\n  lastUpdate: string;\n  originalComment: string;\n  fullComment: string;\n  author: User;\n  createdBy: string;\n  editor: User;\n  _comment: Comment;\n  _editable: boolean;\n  _rectangle;\n  totalPrevPagesHeight = 0;\n  rectTop;\n  rectLeft;\n  pageHeight: number;\n  hasUnsavedChanges = false;\n  selected: boolean;\n  searchString: string;\n  public tagItems: TagsModel[];\n  marginToComment$: Observable<boolean> ;\n\n\n\n  @Output() commentClick = new EventEmitter<SelectionAnnotation>();\n  @Output() renderComments = new EventEmitter<Comment>();\n  @Output() delete = new EventEmitter<Comment>();\n  @Output() updated = new EventEmitter<{comment: Comment, tags: TagsModel[]}>();\n  @Output() changes = new EventEmitter<boolean>();\n  @Input() rotate = 0;\n  @Input() zoom = 1;\n  @Input() index: number;\n  @Input() page: number;\n\n  @ViewChild('form', {static: false}) form: ElementRef;\n  @ViewChild('editableComment', {static: false}) editableComment: ElementRef<HTMLElement>;\n\n  private subscriptions: Subscription;\n\n  constructor(\n    private store: Store<fromStore.AnnotationSetState>,\n    private readonly commentService: CommentService,\n    private tagsServices: TagsServices\n  ) {}\n\n\n  ngOnInit(): void {\n    this.subscriptions = this.store.select(fromSelector.getComponentSearchText)\n      .pipe(distinctUntilChanged()).subscribe(searchString => this.searchString = searchString);\n    this.reRenderComments();\n    this.marginToComment$ = this.commentService.marginToCommentEmitter.asObservable();\n  }\n\n  ngAfterContentInit(): void {\n    if (this.tagItems && this.tagItems.length) {\n      this.tagsServices.updateTagItems(this.tagItems, this._comment.annotationId);\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.unsubscribe();\n  }\n\n  @Input()\n  set comment(comment: Comment) {\n    this._comment = {...comment};\n    this.page = this._comment.page;\n    this.lastUpdate = comment.lastModifiedDate ? comment.lastModifiedDate : comment.createdDate;\n    this.author = comment.createdByDetails;\n    this.createdBy = comment.createdBy;\n    this.editor = comment.lastModifiedByDetails;\n    this.originalComment = comment.content;\n    this.fullComment = this.originalComment;\n    this.selected = this._comment.selected;\n    this._editable = this._comment.editable;\n    this.tagItems = this._comment.tags;\n    const pageMarginBottom = 10;\n    this.totalPrevPagesHeight = 0;\n    for (let i = 0; i < this.page - 1; i++) {\n      const height = this._comment.pages[i + 1] ? this._comment.pages[i + 1].styles.height : undefined;\n      if (height) {\n        this.totalPrevPagesHeight += height + pageMarginBottom;\n      }\n    }\n\n  }\n\n  get comment() {\n    return this._comment;\n  }\n\n  @Input()\n  set annotation(annotation: Annotation) {\n    this._rectangle = annotation.rectangles\n      .reduce((prev, current) => prev.y < current.y ? prev : current);\n    const actualHeight = this._comment.pages[this.page].styles.height / this.zoom;\n    switch (this.rotate) {\n      case 90: this.rectTop = this._rectangle.x;\n      break;\n      case 180: this.rectTop = actualHeight - (this._rectangle.y + this._rectangle.height);\n      break;\n      case 270: this.rectTop = actualHeight - (this._rectangle.x + this._rectangle.width);\n      break;\n      default: this.rectTop = this._rectangle.y;\n    }\n    this.rectLeft = this._rectangle.x;\n  }\n\n  get editable(): boolean {\n    return this._editable;\n  }\n\n  onCommentChange(updatedComment) {\n    this.hasUnsavedChanges =\n      this.originalComment.substring(0, this.CHAR_LIMIT) !== updatedComment.substring(0, this.CHAR_LIMIT);\n    this.commentService.onCommentChange(this.hasUnsavedChanges);\n  }\n\n  deleteOrCancel() {\n    if (!this.editable) {\n      this.delete.emit(this._comment);\n    } else {\n      this.hasUnsavedChanges = false;\n      this._editable = false;\n      this.fullComment = this.originalComment;\n      this.changes.emit(false);\n      if (!this.author && !this.fullComment) {\n        this.delete.emit(this._comment);\n      }\n    }\n  }\n\n  public editOrSave() {\n    if (!this.editable) {\n      this._editable = true;\n    } else {\n      this._comment.content = this.fullComment.substring(0, this.CHAR_LIMIT);\n      const tags = this.tagsServices.getNewTags(this._comment.annotationId);\n      const payload = {\n        comment: this._comment,\n        tags\n      };\n      this.updated.emit(payload);\n      this.hasUnsavedChanges = false;\n      this._editable = false;\n      this.changes.emit(false);\n    }\n  }\n\n  onCommentClick() {\n    if (!this.selected) {\n      this.selected = true;\n      this._editable = false;\n      this.commentClick.emit({ annotationId: this._comment.annotationId, editable: this._editable, selected: true });\n    }\n  }\n\n  reRenderComments() {\n    this.renderComments.emit(this._comment);\n  }\n\n  get commentTop(): number {\n    return this.totalPrevPagesHeight + (this.rectTop * this.zoom);\n  }\n\n\n  get height() {\n    return this.form.nativeElement.getBoundingClientRect().height / this.zoom;\n  }\n}\n","<div\n  #form\n  (click)=\"onCommentClick()\"\n  class=\"aui-comment\"\n  [ngClass]=\"{ stylestoggle: (marginToComment$ | async) }\"\n  [style.top.px]=\"commentTop\"\n  [style.zIndex]=\"selected ? 100 : 0\"\n>\n  <div id=\"detailsWrapper {{ index }}\" class=\"aui-comment__header\">\n    <span *ngIf=\"author && !editor\" class=\"aui-comment__author\">\n      {{ author.forename }} {{ author.surname }}\n    </span>\n    <span *ngIf=\"editor\" class=\"aui-comment__author\">\n      {{ editor.forename }} {{ editor.surname }}\n    </span>\n    <time [hidden]=\"!selected && !this.editable\" class=\"aui-comment__meta\">\n      {{ lastUpdate | momentDate : \"d MMMM y h:mm a\" }}\n    </time>\n  </div>\n  <mv-tags\n    [tagItems]=\"tagItems\"\n    [userId]=\"createdBy\"\n    [editable]=\"editable\"\n    [annoId]=\"_comment.annotationId\"\n  >\n  </mv-tags>\n  <textarea\n    *ngIf=\"selected && editable\"\n    #editableComment\n    mvTextAreaAutoExpand\n    type=\"text\"\n    required\n    name=\"content\"\n    [maxlength]=\"CHAR_LIMIT\"\n    class=\"aui-comment__content form-control mimic-focus edit-mode expanded\"\n    [(ngModel)]=\"fullComment\"\n    (ngModelChange)=\"reRenderComments(); onCommentChange($event)\"\n    aria-label=\"comment\"\n  >\n  </textarea>\n  <p\n    *ngIf=\"!editable\"\n    mvTextHighlight\n    class=\"commentText\"\n    [textToHighlight]=\"searchString\"\n  >\n    {{ fullComment }}\n  </p>\n  <div\n    *ngIf=\"\n      selected ||\n      this.editable ||\n      (!fullComment.length && tagItems && !tagItems.length)\n    \"\n    class=\"aui-comment__footer commentBtns\"\n  >\n    <button\n      class=\"govuk-button\"\n      type=\"button\"\n      role=\"button\"\n      (click)=\"editOrSave()\"\n    >\n      {{ !editable ? (\"Edit\" | rpxTranslate) : (\"Save\" | rpxTranslate) }}\n    </button>\n    <button\n      type=\"button\"\n      role=\"button\"\n      class=\"govuk-button govuk-button--secondary\"\n      (click)=\"deleteOrCancel()\"\n    >\n      {{ !editable ? (\"Delete\" | rpxTranslate) : (\"Cancel\" | rpxTranslate) }}\n    </button>\n  </div>\n  <span class=\"aui-comment__private\">{{ \"private\" | rpxTranslate }}</span>\n</div>\n"]}