@hmcts/media-viewer
Version:
91 lines • 21.8 kB
JavaScript
import { Component, Input } from '@angular/core';
import { select } from '@ngrx/store';
import { v4 as uuid } from 'uuid';
import { filter, take } from 'rxjs/operators';
import * as fromSelectors from '../../store/selectors/redaction.selectors';
import * as fromRedaSelectors from '../../store/selectors/redaction.selectors';
import * as fromDocument from '../../store/selectors/document.selectors';
import * as fromActions from '../../store/actions/redaction.actions';
import * as fromRedactionActions from '../../store/actions/redaction.actions';
import * as fromRedaActions from '../../store/actions/redaction.actions';
import * as i0 from "@angular/core";
import * as i1 from "@ngrx/store";
import * as i2 from "../../viewers/viewer-event.service";
import * as i3 from "../../toolbar/toolbar-event.service";
import * as i4 from "@angular/common";
import * as i5 from "../../annotations/annotation-set/annotation-view/annotation-view.component";
import * as i6 from "../../annotations/annotation-set/annotation-create/box-highlight-create/box-highlight-create.component";
export class RedactionComponent {
constructor(store, viewerEvents, toolbarEvents) {
this.store = store;
this.viewerEvents = viewerEvents;
this.toolbarEvents = toolbarEvents;
}
ngOnInit() {
this.redactionsPerPage$ = this.store.pipe(select(fromSelectors.getRedactionsPerPage));
this.selectedRedaction$ = this.store.pipe(select(fromSelectors.getSelected));
this.$subscription = this.toolbarEvents.drawModeSubject.subscribe(drawMode => this.drawMode = drawMode);
this.$subscription.add(this.store.pipe(select(fromSelectors.getRedactedDocumentInfo), filter(value => !!value))
.subscribe(redactedDocInfo => this.downloadDocument(redactedDocInfo)));
this.$subscription.add(this.store.pipe(select(fromDocument.getDocumentId)).subscribe(docId => this.documentId = docId));
this.$subscription.add(this.viewerEvents.textHighlight.subscribe(highlight => this.markTextRedaction(highlight)));
this.toolbarEvents.applyRedactToDocument.subscribe(() => {
this.store.pipe(select(fromRedaSelectors.getRedactionArray), take(1)).subscribe(redactions => {
this.store.dispatch(new fromRedaActions.Redact(redactions));
});
});
this.toolbarEvents.clearAllRedactMarkers.subscribe(() => {
this.store.dispatch(new fromRedaActions.UnmarkAll(this.documentId));
});
}
ngOnDestroy() {
this.$subscription.unsubscribe();
}
markTextRedaction(highlight) {
const redactionHighlight = highlight.rectangles;
if (redactionHighlight && redactionHighlight.length) {
this.saveRedaction(highlight.page, [...redactionHighlight]);
}
this.toolbarEvents.highlightModeSubject.next(false);
}
markBoxRedaction({ rectangles, page }) {
this.saveRedaction(page, rectangles);
this.toolbarEvents.drawModeSubject.next(false);
}
saveRedaction(page, rectangles) {
const redaction = { page, rectangles, redactionId: uuid(), documentId: this.documentId };
this.store.dispatch(new fromRedactionActions.SaveRedaction(redaction));
}
onMarkerDelete(event) {
this.store.dispatch(new fromActions.DeleteRedaction(event));
}
selectRedaction(event) {
this.store.dispatch(new fromActions.SelectRedaction(event));
}
onMarkerUpdate(redaction) {
this.store.dispatch(new fromActions.SaveRedaction(redaction));
}
downloadDocument({ blob, filename }) {
const url = URL.createObjectURL(blob);
const a = document.createElement('a');
document.body.appendChild(a);
a.setAttribute('style', 'display: none');
a.href = url;
a.download = filename;
a.click();
a.remove();
URL.revokeObjectURL(url);
this.store.dispatch(new fromRedactionActions.ResetRedactedDocument());
}
/** @nocollapse */ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RedactionComponent, deps: [{ token: i1.Store }, { token: i2.ViewerEventService }, { token: i3.ToolbarEventService }], target: i0.ɵɵFactoryTarget.Component }); }
/** @nocollapse */ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: RedactionComponent, selector: "mv-redactions", inputs: { zoom: "zoom", rotate: "rotate" }, ngImport: i0, template: "<div class=\"pageContainer\">\n <div *ngFor=\"let redaction of (redactionsPerPage$ | async); index as i\"\n class=\"pageContainer__page\"\n [ngStyle]=\"{\n 'width.px': redaction.styles.width,\n 'height.px': redaction.styles.height\n }\"\n [attr.redaction-page-num]=\"i+1\"\n [ngClass]=\"{ 'pageContainer__page--draw' : drawMode }\">\n <mv-box-highlight-create\n [page]=\"i + 1\"\n [pageHeight]=\"redaction.styles.height\"\n [pageWidth]=\"redaction.styles.width\"\n [rotate]=\"rotate\"\n [zoom]=\"zoom\"\n (saveSelection)=\"markBoxRedaction($event)\">\n </mv-box-highlight-create>\n <div class=\"pageContainer__page-item\">\n <ng-container *ngFor=\"let anno of redaction.anno\">\n <!-- TODO rename this to selection -->\n <mv-annotation [annotation]=\"anno\"\n [zoom]=\"zoom\"\n [rotate]=\"rotate\"\n [selectedAnnoId]=\"selectedRedaction$ | async\"\n [pageHeight]=\"redaction.styles.height\"\n [pageWidth]=\"redaction.styles.width\"\n (update)=\"onMarkerUpdate($event)\"\n (delete)=\"onMarkerDelete($event)\"\n (annotationClick)=\"selectRedaction($event)\">\n </mv-annotation>\n </ng-container>\n </div>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i5.AnnotationViewComponent, selector: "mv-annotation", inputs: ["annotation", "zoom", "rotate", "selectedAnnoId", "pageHeight", "pageWidth"], outputs: ["update", "delete", "annotationClick"] }, { kind: "component", type: i6.BoxHighlightCreateComponent, selector: "mv-box-highlight-create", inputs: ["page", "pageHeight", "pageWidth", "rotate", "zoom", "container"], outputs: ["saveSelection"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }] }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RedactionComponent, decorators: [{
type: Component,
args: [{ selector: 'mv-redactions', template: "<div class=\"pageContainer\">\n <div *ngFor=\"let redaction of (redactionsPerPage$ | async); index as i\"\n class=\"pageContainer__page\"\n [ngStyle]=\"{\n 'width.px': redaction.styles.width,\n 'height.px': redaction.styles.height\n }\"\n [attr.redaction-page-num]=\"i+1\"\n [ngClass]=\"{ 'pageContainer__page--draw' : drawMode }\">\n <mv-box-highlight-create\n [page]=\"i + 1\"\n [pageHeight]=\"redaction.styles.height\"\n [pageWidth]=\"redaction.styles.width\"\n [rotate]=\"rotate\"\n [zoom]=\"zoom\"\n (saveSelection)=\"markBoxRedaction($event)\">\n </mv-box-highlight-create>\n <div class=\"pageContainer__page-item\">\n <ng-container *ngFor=\"let anno of redaction.anno\">\n <!-- TODO rename this to selection -->\n <mv-annotation [annotation]=\"anno\"\n [zoom]=\"zoom\"\n [rotate]=\"rotate\"\n [selectedAnnoId]=\"selectedRedaction$ | async\"\n [pageHeight]=\"redaction.styles.height\"\n [pageWidth]=\"redaction.styles.width\"\n (update)=\"onMarkerUpdate($event)\"\n (delete)=\"onMarkerDelete($event)\"\n (annotationClick)=\"selectRedaction($event)\">\n </mv-annotation>\n </ng-container>\n </div>\n </div>\n</div>\n" }]
}], ctorParameters: () => [{ type: i1.Store }, { type: i2.ViewerEventService }, { type: i3.ToolbarEventService }], propDecorators: { zoom: [{
type: Input
}], rotate: [{
type: Input
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"redaction.component.js","sourceRoot":"","sources":["../../../../../../projects/media-viewer/src/lib/redaction/components/redaction.component.ts","../../../../../../projects/media-viewer/src/lib/redaction/components/redaction.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnDestroy, OnInit } from '@angular/core';\nimport { Observable, Subscription } from 'rxjs';\nimport { select, Store } from '@ngrx/store';\nimport { v4 as uuid } from 'uuid';\nimport { filter, take } from 'rxjs/operators';\n\nimport { Rectangle } from '../../annotations/annotation-set/annotation-view/rectangle/rectangle.model';\nimport * as fromStore from '../../store/reducers/reducers';\nimport * as fromSelectors from '../../store/selectors/redaction.selectors';\nimport * as fromRedaSelectors from '../../store/selectors/redaction.selectors';\nimport * as fromDocument from '../../store/selectors/document.selectors';\nimport * as fromActions from '../../store/actions/redaction.actions';\nimport * as fromRedactionActions from '../../store/actions/redaction.actions';\nimport * as fromRedaActions from '../../store/actions/redaction.actions';\nimport { SelectionAnnotation } from '../../annotations/models/event-select.model';\nimport { ViewerEventService } from '../../viewers/viewer-event.service';\nimport { Redaction } from '../services/redaction.model';\nimport { ToolbarEventService } from '../../toolbar/toolbar-event.service';\n\n@Component({\n  selector: 'mv-redactions',\n  templateUrl: './redaction.component.html'\n})\nexport class RedactionComponent implements OnInit, OnDestroy {\n\n  @Input() zoom: number;\n  @Input() rotate: number;\n\n  redactionsPerPage$: Observable<any>; // todo add type\n  selectedRedaction$: Observable<SelectionAnnotation | {}>;\n  rectangles: Rectangle[];\n  drawMode: boolean;\n  documentId: string;\n\n  private $subscription: Subscription;\n\n  constructor(private store: Store<fromStore.State>,\n    private readonly viewerEvents: ViewerEventService,\n    private toolbarEvents: ToolbarEventService) { }\n\n  ngOnInit(): void {\n    this.redactionsPerPage$ = this.store.pipe(select(fromSelectors.getRedactionsPerPage));\n    this.selectedRedaction$ = this.store.pipe(select(fromSelectors.getSelected));\n    this.$subscription = this.toolbarEvents.drawModeSubject.subscribe(drawMode => this.drawMode = drawMode);\n    this.$subscription.add(this.store.pipe(select(fromSelectors.getRedactedDocumentInfo), filter(value => !!value))\n      .subscribe(redactedDocInfo => this.downloadDocument(redactedDocInfo)));\n    this.$subscription.add(this.store.pipe(select(fromDocument.getDocumentId)).subscribe(docId => this.documentId = docId));\n    this.$subscription.add(this.viewerEvents.textHighlight.subscribe(highlight => this.markTextRedaction(highlight)));\n    this.toolbarEvents.applyRedactToDocument.subscribe(() => {\n      this.store.pipe(select(fromRedaSelectors.getRedactionArray), take(1)).subscribe(redactions => {\n        this.store.dispatch(new fromRedaActions.Redact(redactions));\n      });\n    });\n    this.toolbarEvents.clearAllRedactMarkers.subscribe(() => {\n      this.store.dispatch(new fromRedaActions.UnmarkAll(this.documentId));\n    });\n  }\n\n  ngOnDestroy(): void {\n    this.$subscription.unsubscribe();\n  }\n\n  markTextRedaction(highlight) {\n    const redactionHighlight = highlight.rectangles;\n    if (redactionHighlight && redactionHighlight.length) {\n      this.saveRedaction(highlight.page, [...redactionHighlight]);\n    }\n    this.toolbarEvents.highlightModeSubject.next(false);\n  }\n\n  markBoxRedaction({ rectangles, page }) {\n    this.saveRedaction(page, rectangles);\n    this.toolbarEvents.drawModeSubject.next(false);\n  }\n\n  saveRedaction(page: number, rectangles: Rectangle[]) {\n    const redaction = { page, rectangles, redactionId: uuid(), documentId: this.documentId };\n    this.store.dispatch(new fromRedactionActions.SaveRedaction(redaction));\n  }\n\n  onMarkerDelete(event) {\n    this.store.dispatch(new fromActions.DeleteRedaction(event));\n  }\n\n  selectRedaction(event) {\n    this.store.dispatch(new fromActions.SelectRedaction(event));\n  }\n\n  onMarkerUpdate(redaction: Redaction) {\n    this.store.dispatch(new fromActions.SaveRedaction(redaction));\n  }\n\n  downloadDocument({ blob, filename }) {\n    const url = URL.createObjectURL(blob);\n    const a = document.createElement('a');\n    document.body.appendChild(a);\n    a.setAttribute('style', 'display: none');\n    a.href = url;\n    a.download = filename;\n    a.click();\n    a.remove();\n    URL.revokeObjectURL(url);\n    this.store.dispatch(new fromRedactionActions.ResetRedactedDocument());\n  }\n}\n","<div class=\"pageContainer\">\n  <div *ngFor=\"let redaction of (redactionsPerPage$ | async);  index as i\"\n       class=\"pageContainer__page\"\n       [ngStyle]=\"{\n       'width.px': redaction.styles.width,\n       'height.px': redaction.styles.height\n       }\"\n      [attr.redaction-page-num]=\"i+1\"\n       [ngClass]=\"{ 'pageContainer__page--draw' : drawMode }\">\n    <mv-box-highlight-create\n      [page]=\"i + 1\"\n      [pageHeight]=\"redaction.styles.height\"\n      [pageWidth]=\"redaction.styles.width\"\n      [rotate]=\"rotate\"\n      [zoom]=\"zoom\"\n      (saveSelection)=\"markBoxRedaction($event)\">\n    </mv-box-highlight-create>\n    <div class=\"pageContainer__page-item\">\n      <ng-container *ngFor=\"let anno of redaction.anno\">\n        <!--  TODO rename this to selection -->\n        <mv-annotation [annotation]=\"anno\"\n                       [zoom]=\"zoom\"\n                       [rotate]=\"rotate\"\n                       [selectedAnnoId]=\"selectedRedaction$ | async\"\n                       [pageHeight]=\"redaction.styles.height\"\n                       [pageWidth]=\"redaction.styles.width\"\n                       (update)=\"onMarkerUpdate($event)\"\n                       (delete)=\"onMarkerDelete($event)\"\n                       (annotationClick)=\"selectRedaction($event)\">\n        </mv-annotation>\n      </ng-container>\n    </div>\n  </div>\n</div>\n"]}