@hmcts/media-viewer 4.1.9 → 4.1.10-exui-3053-1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (30) hide show
  1. package/esm2022/lib/icp/confirm-exit/confirm-action-dialog.component.mjs +13 -5
  2. package/esm2022/lib/media-viewer.component.mjs +98 -4
  3. package/esm2022/lib/shared/directives/keyboard-nav.directive.mjs +243 -0
  4. package/esm2022/lib/shared/shared.module.mjs +10 -5
  5. package/esm2022/lib/toolbar/highlight-toolbar/highlight-toolbar.component.mjs +38 -10
  6. package/esm2022/lib/toolbar/main-toolbar/main-toolbar.component.mjs +82 -18
  7. package/esm2022/lib/toolbar/redaction-toolbar/redaction-toolbar.component.mjs +35 -10
  8. package/esm2022/lib/toolbar/toolbar-focus.service.mjs +33 -0
  9. package/esm2022/lib/toolbar/toolbar.module.mjs +12 -6
  10. package/fesm2022/hmcts-media-viewer.mjs +609 -117
  11. package/fesm2022/hmcts-media-viewer.mjs.map +1 -1
  12. package/lib/icp/confirm-exit/confirm-action-dialog.component.d.ts +4 -1
  13. package/lib/icp/confirm-exit/confirm-action-dialog.component.d.ts.map +1 -1
  14. package/lib/media-viewer.component.d.ts +8 -0
  15. package/lib/media-viewer.component.d.ts.map +1 -1
  16. package/lib/shared/directives/keyboard-nav.directive.d.ts +30 -0
  17. package/lib/shared/directives/keyboard-nav.directive.d.ts.map +1 -0
  18. package/lib/shared/shared.module.d.ts +5 -4
  19. package/lib/shared/shared.module.d.ts.map +1 -1
  20. package/lib/toolbar/highlight-toolbar/highlight-toolbar.component.d.ts +7 -1
  21. package/lib/toolbar/highlight-toolbar/highlight-toolbar.component.d.ts.map +1 -1
  22. package/lib/toolbar/main-toolbar/main-toolbar.component.d.ts +12 -1
  23. package/lib/toolbar/main-toolbar/main-toolbar.component.d.ts.map +1 -1
  24. package/lib/toolbar/redaction-toolbar/redaction-toolbar.component.d.ts +6 -1
  25. package/lib/toolbar/redaction-toolbar/redaction-toolbar.component.d.ts.map +1 -1
  26. package/lib/toolbar/toolbar-focus.service.d.ts +7 -0
  27. package/lib/toolbar/toolbar-focus.service.d.ts.map +1 -0
  28. package/lib/toolbar/toolbar.module.d.ts +2 -1
  29. package/lib/toolbar/toolbar.module.d.ts.map +1 -1
  30. package/package.json +1 -1
@@ -0,0 +1,243 @@
1
+ import { Directive, EventEmitter, HostListener, Input, Output } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class KeyboardNavDirective {
4
+ constructor(elementRef) {
5
+ this.elementRef = elementRef;
6
+ this.orientation = 'horizontal';
7
+ this.itemFocused = new EventEmitter();
8
+ this.itemActivated = new EventEmitter();
9
+ this.focusableItems = [];
10
+ this.currentFocusIndex = -1;
11
+ this.isUsingArrowKeys = false;
12
+ }
13
+ ngOnInit() {
14
+ this.updateFocusableItems();
15
+ this.setupMutationObserver();
16
+ }
17
+ ngOnDestroy() {
18
+ if (this.mutationObserver) {
19
+ this.mutationObserver.disconnect();
20
+ }
21
+ }
22
+ setupMutationObserver() {
23
+ this.mutationObserver = new MutationObserver(() => {
24
+ this.updateFocusableItems();
25
+ });
26
+ this.mutationObserver.observe(this.elementRef.nativeElement, {
27
+ childList: true,
28
+ subtree: true,
29
+ attributes: true,
30
+ attributeFilter: ['class', 'disabled', 'hidden']
31
+ });
32
+ }
33
+ updateFocusableItems() {
34
+ const allItems = Array.from(this.elementRef.nativeElement.querySelectorAll('button:not([disabled])'));
35
+ this.focusableItems = allItems.filter(item => this.isVisible(item));
36
+ if (this.isUsingArrowKeys) {
37
+ this.applyArrowKeyTabindex();
38
+ }
39
+ else {
40
+ this.focusableItems.forEach(item => {
41
+ if (!item.hasAttribute('tabindex') || item.getAttribute('tabindex') === '-1') {
42
+ item.setAttribute('tabindex', '0');
43
+ }
44
+ });
45
+ }
46
+ }
47
+ isVisible(element) {
48
+ if (!element) {
49
+ return false;
50
+ }
51
+ if (element.hasAttribute('disabled')) {
52
+ return false;
53
+ }
54
+ const computedStyle = window.getComputedStyle(element);
55
+ if (computedStyle.display === 'none' || computedStyle.visibility === 'hidden') {
56
+ return false;
57
+ }
58
+ // check if any parent element is hidden (up to the directive's host element)
59
+ let parent = element.parentElement;
60
+ while (parent && parent !== this.elementRef.nativeElement) {
61
+ const parentStyle = window.getComputedStyle(parent);
62
+ if (parentStyle.display === 'none' || parentStyle.visibility === 'hidden') {
63
+ return false;
64
+ }
65
+ parent = parent.parentElement;
66
+ }
67
+ return true;
68
+ }
69
+ applyArrowKeyTabindex() {
70
+ if (this.focusableItems.length === 0) {
71
+ return;
72
+ }
73
+ let indexToMakeTabbable = 0;
74
+ if (this.currentFocusIndex >= 0 && this.currentFocusIndex < this.focusableItems.length) {
75
+ indexToMakeTabbable = this.currentFocusIndex;
76
+ }
77
+ else {
78
+ const focusedElement = document.activeElement;
79
+ const focusedIndex = this.focusableItems.indexOf(focusedElement);
80
+ if (focusedIndex !== -1) {
81
+ indexToMakeTabbable = focusedIndex;
82
+ this.currentFocusIndex = focusedIndex;
83
+ }
84
+ }
85
+ this.focusableItems.forEach((item, index) => {
86
+ item.setAttribute('tabindex', index === indexToMakeTabbable ? '0' : '-1');
87
+ });
88
+ }
89
+ onKeyDown(event) {
90
+ const target = event.target;
91
+ const currentIndex = this.focusableItems.indexOf(target);
92
+ if (currentIndex === -1) {
93
+ return;
94
+ }
95
+ let handled = false;
96
+ let isArrowKey = false;
97
+ switch (event.key) {
98
+ case 'Tab':
99
+ if (this.isUsingArrowKeys) {
100
+ this.isUsingArrowKeys = false;
101
+ this.focusableItems.forEach(item => {
102
+ item.setAttribute('tabindex', '0');
103
+ });
104
+ }
105
+ return;
106
+ case 'ArrowRight':
107
+ if (this.orientation === 'horizontal') {
108
+ isArrowKey = true;
109
+ if (!this.isUsingArrowKeys) {
110
+ this.isUsingArrowKeys = true;
111
+ this.applyArrowKeyTabindex();
112
+ }
113
+ this.focusNext(currentIndex);
114
+ handled = true;
115
+ }
116
+ break;
117
+ case 'ArrowLeft':
118
+ if (this.orientation === 'horizontal') {
119
+ isArrowKey = true;
120
+ if (!this.isUsingArrowKeys) {
121
+ this.isUsingArrowKeys = true;
122
+ this.applyArrowKeyTabindex();
123
+ }
124
+ this.focusPrevious(currentIndex);
125
+ handled = true;
126
+ }
127
+ break;
128
+ case 'ArrowDown':
129
+ if (this.orientation === 'vertical') {
130
+ isArrowKey = true;
131
+ if (!this.isUsingArrowKeys) {
132
+ this.isUsingArrowKeys = true;
133
+ this.applyArrowKeyTabindex();
134
+ }
135
+ this.focusNext(currentIndex);
136
+ handled = true;
137
+ }
138
+ break;
139
+ case 'ArrowUp':
140
+ if (this.orientation === 'vertical') {
141
+ isArrowKey = true;
142
+ if (!this.isUsingArrowKeys) {
143
+ this.isUsingArrowKeys = true;
144
+ this.applyArrowKeyTabindex();
145
+ }
146
+ this.focusPrevious(currentIndex);
147
+ handled = true;
148
+ }
149
+ break;
150
+ case 'Home':
151
+ isArrowKey = true;
152
+ if (!this.isUsingArrowKeys) {
153
+ this.isUsingArrowKeys = true;
154
+ this.applyArrowKeyTabindex();
155
+ }
156
+ this.focusFirst();
157
+ handled = true;
158
+ break;
159
+ case 'End':
160
+ isArrowKey = true;
161
+ if (!this.isUsingArrowKeys) {
162
+ this.isUsingArrowKeys = true;
163
+ this.applyArrowKeyTabindex();
164
+ }
165
+ this.focusLast();
166
+ handled = true;
167
+ break;
168
+ case 'Enter':
169
+ case ' ':
170
+ this.activateItem(target);
171
+ handled = true;
172
+ break;
173
+ }
174
+ if (handled) {
175
+ event.preventDefault();
176
+ event.stopPropagation();
177
+ }
178
+ }
179
+ focusNext(currentIndex) {
180
+ if (this.focusableItems.length === 0) {
181
+ return;
182
+ }
183
+ let nextIndex = currentIndex + 1;
184
+ if (nextIndex >= this.focusableItems.length) {
185
+ nextIndex = 0;
186
+ }
187
+ this.focusItemAtIndex(nextIndex);
188
+ }
189
+ focusPrevious(currentIndex) {
190
+ if (this.focusableItems.length === 0) {
191
+ return;
192
+ }
193
+ let previousIndex = currentIndex - 1;
194
+ if (previousIndex < 0) {
195
+ previousIndex = this.focusableItems.length - 1;
196
+ }
197
+ this.focusItemAtIndex(previousIndex);
198
+ }
199
+ focusFirst() {
200
+ if (this.focusableItems.length > 0) {
201
+ this.focusItemAtIndex(0);
202
+ }
203
+ }
204
+ focusLast() {
205
+ if (this.focusableItems.length > 0) {
206
+ this.focusItemAtIndex(this.focusableItems.length - 1);
207
+ }
208
+ }
209
+ focusItemAtIndex(index) {
210
+ if (index < 0 || index >= this.focusableItems.length) {
211
+ return;
212
+ }
213
+ this.currentFocusIndex = index;
214
+ this.focusableItems.forEach((item, i) => {
215
+ item.setAttribute('tabindex', i === index ? '0' : '-1');
216
+ });
217
+ this.focusableItems[index].focus();
218
+ this.itemFocused.emit(this.focusableItems[index]);
219
+ }
220
+ activateItem(item) {
221
+ item.click();
222
+ this.itemActivated.emit(item);
223
+ }
224
+ /** @nocollapse */ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: KeyboardNavDirective, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive }); }
225
+ /** @nocollapse */ static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: KeyboardNavDirective, selector: "[mvKeyboardNav]", inputs: { orientation: ["mvKeyboardNav", "orientation"] }, outputs: { itemFocused: "itemFocused", itemActivated: "itemActivated" }, host: { listeners: { "keydown": "onKeyDown($event)" } }, ngImport: i0 }); }
226
+ }
227
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: KeyboardNavDirective, decorators: [{
228
+ type: Directive,
229
+ args: [{
230
+ selector: '[mvKeyboardNav]'
231
+ }]
232
+ }], ctorParameters: () => [{ type: i0.ElementRef }], propDecorators: { orientation: [{
233
+ type: Input,
234
+ args: ['mvKeyboardNav']
235
+ }], itemFocused: [{
236
+ type: Output
237
+ }], itemActivated: [{
238
+ type: Output
239
+ }], onKeyDown: [{
240
+ type: HostListener,
241
+ args: ['keydown', ['$event']]
242
+ }] } });
243
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"keyboard-nav.directive.js","sourceRoot":"","sources":["../../../../../../projects/media-viewer/src/lib/shared/directives/keyboard-nav.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  ElementRef,\n  EventEmitter,\n  HostListener,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output\n} from '@angular/core';\n\nexport type KeyboardNavOrientation = 'horizontal' | 'vertical';\n\n@Directive({\n  selector: '[mvKeyboardNav]'\n})\nexport class KeyboardNavDirective implements OnInit, OnDestroy {\n  @Input('mvKeyboardNav') orientation: KeyboardNavOrientation = 'horizontal';\n  @Output() itemFocused = new EventEmitter<HTMLElement>();\n  @Output() itemActivated = new EventEmitter<HTMLElement>();\n\n  private focusableItems: HTMLElement[] = [];\n  private currentFocusIndex = -1;\n  private mutationObserver: MutationObserver;\n  private isUsingArrowKeys = false;\n\n  constructor(private elementRef: ElementRef<HTMLElement>) {}\n\n  ngOnInit(): void {\n    this.updateFocusableItems();\n    this.setupMutationObserver();\n  }\n\n  ngOnDestroy(): void {\n    if (this.mutationObserver) {\n      this.mutationObserver.disconnect();\n    }\n  }\n\n  private setupMutationObserver(): void {\n    this.mutationObserver = new MutationObserver(() => {\n      this.updateFocusableItems();\n    });\n\n    this.mutationObserver.observe(this.elementRef.nativeElement, {\n      childList: true,\n      subtree: true,\n      attributes: true,\n      attributeFilter: ['class', 'disabled', 'hidden']\n    });\n  }\n\n  private updateFocusableItems(): void {\n    const allItems = Array.from(\n      this.elementRef.nativeElement.querySelectorAll<HTMLElement>('button:not([disabled])')\n    );\n\n    this.focusableItems = allItems.filter(item => this.isVisible(item));\n\n    if (this.isUsingArrowKeys) {\n      this.applyArrowKeyTabindex();\n    } else {\n      this.focusableItems.forEach(item => {\n        if (!item.hasAttribute('tabindex') || item.getAttribute('tabindex') === '-1') {\n          item.setAttribute('tabindex', '0');\n        }\n      });\n    }\n  }\n\n  private isVisible(element: HTMLElement): boolean {\n    if (!element) {\n      return false;\n    }\n\n    if (element.hasAttribute('disabled')) {\n      return false;\n    }\n\n    const computedStyle = window.getComputedStyle(element);\n    if (computedStyle.display === 'none' || computedStyle.visibility === 'hidden') {\n      return false;\n    }\n\n    // check if any parent element is hidden (up to the directive's host element)\n    let parent = element.parentElement;\n    while (parent && parent !== this.elementRef.nativeElement) {\n      const parentStyle = window.getComputedStyle(parent);\n      if (parentStyle.display === 'none' || parentStyle.visibility === 'hidden') {\n        return false;\n      }\n      parent = parent.parentElement;\n    }\n\n    return true;\n  }\n\n  private applyArrowKeyTabindex(): void {\n    if (this.focusableItems.length === 0) {\n      return;\n    }\n\n    let indexToMakeTabbable = 0;\n\n    if (this.currentFocusIndex >= 0 && this.currentFocusIndex < this.focusableItems.length) {\n      indexToMakeTabbable = this.currentFocusIndex;\n    } else {\n      const focusedElement = document.activeElement as HTMLElement;\n      const focusedIndex = this.focusableItems.indexOf(focusedElement);\n      if (focusedIndex !== -1) {\n        indexToMakeTabbable = focusedIndex;\n        this.currentFocusIndex = focusedIndex;\n      }\n    }\n\n    this.focusableItems.forEach((item, index) => {\n      item.setAttribute('tabindex', index === indexToMakeTabbable ? '0' : '-1');\n    });\n  }\n\n  @HostListener('keydown', ['$event'])\n  onKeyDown(event: KeyboardEvent): void {\n    const target = event.target as HTMLElement;\n    const currentIndex = this.focusableItems.indexOf(target);\n\n    if (currentIndex === -1) {\n      return;\n    }\n\n    let handled = false;\n    let isArrowKey = false;\n\n    switch (event.key) {\n      case 'Tab':\n        if (this.isUsingArrowKeys) {\n          this.isUsingArrowKeys = false;\n          this.focusableItems.forEach(item => {\n            item.setAttribute('tabindex', '0');\n          });\n        }\n        return;\n      case 'ArrowRight':\n        if (this.orientation === 'horizontal') {\n          isArrowKey = true;\n          if (!this.isUsingArrowKeys) {\n            this.isUsingArrowKeys = true;\n            this.applyArrowKeyTabindex();\n          }\n          this.focusNext(currentIndex);\n          handled = true;\n        }\n        break;\n\n      case 'ArrowLeft':\n        if (this.orientation === 'horizontal') {\n          isArrowKey = true;\n          if (!this.isUsingArrowKeys) {\n            this.isUsingArrowKeys = true;\n            this.applyArrowKeyTabindex();\n          }\n          this.focusPrevious(currentIndex);\n          handled = true;\n        }\n        break;\n\n      case 'ArrowDown':\n        if (this.orientation === 'vertical') {\n          isArrowKey = true;\n          if (!this.isUsingArrowKeys) {\n            this.isUsingArrowKeys = true;\n            this.applyArrowKeyTabindex();\n          }\n          this.focusNext(currentIndex);\n          handled = true;\n        }\n        break;\n\n      case 'ArrowUp':\n        if (this.orientation === 'vertical') {\n          isArrowKey = true;\n          if (!this.isUsingArrowKeys) {\n            this.isUsingArrowKeys = true;\n            this.applyArrowKeyTabindex();\n          }\n          this.focusPrevious(currentIndex);\n          handled = true;\n        }\n        break;\n\n      case 'Home':\n        isArrowKey = true;\n        if (!this.isUsingArrowKeys) {\n          this.isUsingArrowKeys = true;\n          this.applyArrowKeyTabindex();\n        }\n        this.focusFirst();\n        handled = true;\n        break;\n\n      case 'End':\n        isArrowKey = true;\n        if (!this.isUsingArrowKeys) {\n          this.isUsingArrowKeys = true;\n          this.applyArrowKeyTabindex();\n        }\n        this.focusLast();\n        handled = true;\n        break;\n\n      case 'Enter':\n      case ' ':\n        this.activateItem(target);\n        handled = true;\n        break;\n    }\n\n    if (handled) {\n      event.preventDefault();\n      event.stopPropagation();\n    }\n  }\n\n  private focusNext(currentIndex: number): void {\n    if (this.focusableItems.length === 0) {\n      return;\n    }\n\n    let nextIndex = currentIndex + 1;\n\n    if (nextIndex >= this.focusableItems.length) {\n      nextIndex = 0;\n    }\n\n    this.focusItemAtIndex(nextIndex);\n  }\n\n  private focusPrevious(currentIndex: number): void {\n    if (this.focusableItems.length === 0) {\n      return;\n    }\n\n    let previousIndex = currentIndex - 1;\n\n    if (previousIndex < 0) {\n      previousIndex = this.focusableItems.length - 1;\n    }\n\n    this.focusItemAtIndex(previousIndex);\n  }\n\n  private focusFirst(): void {\n    if (this.focusableItems.length > 0) {\n      this.focusItemAtIndex(0);\n    }\n  }\n\n  private focusLast(): void {\n    if (this.focusableItems.length > 0) {\n      this.focusItemAtIndex(this.focusableItems.length - 1);\n    }\n  }\n\n  private focusItemAtIndex(index: number): void {\n    if (index < 0 || index >= this.focusableItems.length) {\n      return;\n    }\n\n    this.currentFocusIndex = index;\n\n    this.focusableItems.forEach((item, i) => {\n      item.setAttribute('tabindex', i === index ? '0' : '-1');\n    });\n\n    this.focusableItems[index].focus();\n    this.itemFocused.emit(this.focusableItems[index]);\n  }\n\n  private activateItem(item: HTMLElement): void {\n    item.click();\n    this.itemActivated.emit(item);\n  }\n}"]}
@@ -6,20 +6,23 @@ import { GovUkFieldsetComponent } from './gov-uk-fieldset/gov-uk-fieldset.compon
6
6
  import { GovUkLabelComponent } from './gov-uk-label/gov-uk-label.component';
7
7
  import { FormsModule, ReactiveFormsModule } from '@angular/forms';
8
8
  import { RouterModule } from '@angular/router';
9
+ import { KeyboardNavDirective } from './directives/keyboard-nav.directive';
9
10
  import * as i0 from "@angular/core";
10
11
  export class SharedModule {
11
12
  /** @nocollapse */ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SharedModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule }); }
12
13
  /** @nocollapse */ static { this.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "18.2.13", ngImport: i0, type: SharedModule, declarations: [GovUkDateComponent,
13
14
  GovUkErrorMessageComponent,
14
15
  GovUkFieldsetComponent,
15
- GovUkLabelComponent], imports: [CommonModule,
16
+ GovUkLabelComponent,
17
+ KeyboardNavDirective], imports: [CommonModule,
16
18
  FormsModule,
17
19
  ReactiveFormsModule,
18
20
  CommonModule,
19
21
  RouterModule], exports: [GovUkDateComponent,
20
22
  GovUkErrorMessageComponent,
21
23
  GovUkFieldsetComponent,
22
- GovUkLabelComponent] }); }
24
+ GovUkLabelComponent,
25
+ KeyboardNavDirective] }); }
23
26
  /** @nocollapse */ static { this.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SharedModule, imports: [CommonModule,
24
27
  FormsModule,
25
28
  ReactiveFormsModule,
@@ -40,14 +43,16 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
40
43
  GovUkDateComponent,
41
44
  GovUkErrorMessageComponent,
42
45
  GovUkFieldsetComponent,
43
- GovUkLabelComponent
46
+ GovUkLabelComponent,
47
+ KeyboardNavDirective
44
48
  ],
45
49
  exports: [
46
50
  GovUkDateComponent,
47
51
  GovUkErrorMessageComponent,
48
52
  GovUkFieldsetComponent,
49
- GovUkLabelComponent
53
+ GovUkLabelComponent,
54
+ KeyboardNavDirective
50
55
  ]
51
56
  }]
52
57
  }] });
53
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2hhcmVkLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL21lZGlhLXZpZXdlci9zcmMvbGliL3NoYXJlZC9zaGFyZWQubW9kdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBQyxrQkFBa0IsRUFBQyxNQUFNLHFDQUFxQyxDQUFDO0FBQ3ZFLE9BQU8sRUFBQywwQkFBMEIsRUFBQyxNQUFNLHVEQUF1RCxDQUFDO0FBQ2pHLE9BQU8sRUFBQyxzQkFBc0IsRUFBQyxNQUFNLDZDQUE2QyxDQUFDO0FBQ25GLE9BQU8sRUFBQyxtQkFBbUIsRUFBQyxNQUFNLHVDQUF1QyxDQUFDO0FBQzFFLE9BQU8sRUFBQyxXQUFXLEVBQUUsbUJBQW1CLEVBQUMsTUFBTSxnQkFBZ0IsQ0FBQztBQUNoRSxPQUFPLEVBQUMsWUFBWSxFQUFDLE1BQU0saUJBQWlCLENBQUM7O0FBMEI3QyxNQUFNLE9BQU8sWUFBWTtrSUFBWixZQUFZO21JQUFaLFlBQVksaUJBYnJCLGtCQUFrQjtZQUNsQiwwQkFBMEI7WUFDMUIsc0JBQXNCO1lBQ3RCLG1CQUFtQixhQVZuQixZQUFZO1lBQ1osV0FBVztZQUNYLG1CQUFtQjtZQUNuQixZQUFZO1lBQ1osWUFBWSxhQVVaLGtCQUFrQjtZQUNsQiwwQkFBMEI7WUFDMUIsc0JBQXNCO1lBQ3RCLG1CQUFtQjttSUFHVixZQUFZLFlBcEJyQixZQUFZO1lBQ1osV0FBVztZQUNYLG1CQUFtQjtZQUNuQixZQUFZO1lBQ1osWUFBWTs7NEZBZ0JILFlBQVk7a0JBdEJ4QixRQUFRO21CQUFDO29CQUNSLE9BQU8sRUFBRTt3QkFDUCxZQUFZO3dCQUNaLFdBQVc7d0JBQ1gsbUJBQW1CO3dCQUNuQixZQUFZO3dCQUNaLFlBQVk7cUJBQ2I7b0JBQ0QsWUFBWSxFQUFFO3dCQUNaLGtCQUFrQjt3QkFDbEIsMEJBQTBCO3dCQUMxQixzQkFBc0I7d0JBQ3RCLG1CQUFtQjtxQkFFcEI7b0JBQ0QsT0FBTyxFQUFFO3dCQUNQLGtCQUFrQjt3QkFDbEIsMEJBQTBCO3dCQUMxQixzQkFBc0I7d0JBQ3RCLG1CQUFtQjtxQkFDcEI7aUJBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7R292VWtEYXRlQ29tcG9uZW50fSBmcm9tICcuL2dvdi11ay1kYXRlL2dvdi11ay1kYXRlLmNvbXBvbmVudCc7XG5pbXBvcnQge0dvdlVrRXJyb3JNZXNzYWdlQ29tcG9uZW50fSBmcm9tICcuL2dvdi11ay1lcnJvci1tZXNzYWdlL2dvdi11ay1lcnJvci1tZXNzYWdlLmNvbXBvbmVudCc7XG5pbXBvcnQge0dvdlVrRmllbGRzZXRDb21wb25lbnR9IGZyb20gJy4vZ292LXVrLWZpZWxkc2V0L2dvdi11ay1maWVsZHNldC5jb21wb25lbnQnO1xuaW1wb3J0IHtHb3ZVa0xhYmVsQ29tcG9uZW50fSBmcm9tICcuL2dvdi11ay1sYWJlbC9nb3YtdWstbGFiZWwuY29tcG9uZW50JztcbmltcG9ydCB7Rm9ybXNNb2R1bGUsIFJlYWN0aXZlRm9ybXNNb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcbmltcG9ydCB7Um91dGVyTW9kdWxlfSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xuXG5cblxuQE5nTW9kdWxlKHtcbiAgaW1wb3J0czogW1xuICAgIENvbW1vbk1vZHVsZSxcbiAgICBGb3Jtc01vZHVsZSxcbiAgICBSZWFjdGl2ZUZvcm1zTW9kdWxlLFxuICAgIENvbW1vbk1vZHVsZSxcbiAgICBSb3V0ZXJNb2R1bGVcbiAgXSxcbiAgZGVjbGFyYXRpb25zOiBbXG4gICAgR292VWtEYXRlQ29tcG9uZW50LFxuICAgIEdvdlVrRXJyb3JNZXNzYWdlQ29tcG9uZW50LFxuICAgIEdvdlVrRmllbGRzZXRDb21wb25lbnQsXG4gICAgR292VWtMYWJlbENvbXBvbmVudFxuXG4gIF0sXG4gIGV4cG9ydHM6IFtcbiAgICBHb3ZVa0RhdGVDb21wb25lbnQsXG4gICAgR292VWtFcnJvck1lc3NhZ2VDb21wb25lbnQsXG4gICAgR292VWtGaWVsZHNldENvbXBvbmVudCxcbiAgICBHb3ZVa0xhYmVsQ29tcG9uZW50XG4gIF1cbn0pXG5leHBvcnQgY2xhc3MgU2hhcmVkTW9kdWxlIHt9XG4iXX0=
58
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2hhcmVkLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL21lZGlhLXZpZXdlci9zcmMvbGliL3NoYXJlZC9zaGFyZWQubW9kdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLHFDQUFxQyxDQUFDO0FBQ3pFLE9BQU8sRUFBRSwwQkFBMEIsRUFBRSxNQUFNLHVEQUF1RCxDQUFDO0FBQ25HLE9BQU8sRUFBRSxzQkFBc0IsRUFBRSxNQUFNLDZDQUE2QyxDQUFDO0FBQ3JGLE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLHVDQUF1QyxDQUFDO0FBQzVFLE9BQU8sRUFBRSxXQUFXLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUNsRSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0scUNBQXFDLENBQUM7O0FBeUIzRSxNQUFNLE9BQU8sWUFBWTtrSUFBWixZQUFZO21JQUFaLFlBQVksaUJBZHJCLGtCQUFrQjtZQUNsQiwwQkFBMEI7WUFDMUIsc0JBQXNCO1lBQ3RCLG1CQUFtQjtZQUNuQixvQkFBb0IsYUFYcEIsWUFBWTtZQUNaLFdBQVc7WUFDWCxtQkFBbUI7WUFDbkIsWUFBWTtZQUNaLFlBQVksYUFVWixrQkFBa0I7WUFDbEIsMEJBQTBCO1lBQzFCLHNCQUFzQjtZQUN0QixtQkFBbUI7WUFDbkIsb0JBQW9CO21JQUdYLFlBQVksWUFyQnJCLFlBQVk7WUFDWixXQUFXO1lBQ1gsbUJBQW1CO1lBQ25CLFlBQVk7WUFDWixZQUFZOzs0RkFpQkgsWUFBWTtrQkF2QnhCLFFBQVE7bUJBQUM7b0JBQ1IsT0FBTyxFQUFFO3dCQUNQLFlBQVk7d0JBQ1osV0FBVzt3QkFDWCxtQkFBbUI7d0JBQ25CLFlBQVk7d0JBQ1osWUFBWTtxQkFDYjtvQkFDRCxZQUFZLEVBQUU7d0JBQ1osa0JBQWtCO3dCQUNsQiwwQkFBMEI7d0JBQzFCLHNCQUFzQjt3QkFDdEIsbUJBQW1CO3dCQUNuQixvQkFBb0I7cUJBQ3JCO29CQUNELE9BQU8sRUFBRTt3QkFDUCxrQkFBa0I7d0JBQ2xCLDBCQUEwQjt3QkFDMUIsc0JBQXNCO3dCQUN0QixtQkFBbUI7d0JBQ25CLG9CQUFvQjtxQkFDckI7aUJBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IEdvdlVrRGF0ZUNvbXBvbmVudCB9IGZyb20gJy4vZ292LXVrLWRhdGUvZ292LXVrLWRhdGUuY29tcG9uZW50JztcbmltcG9ydCB7IEdvdlVrRXJyb3JNZXNzYWdlQ29tcG9uZW50IH0gZnJvbSAnLi9nb3YtdWstZXJyb3ItbWVzc2FnZS9nb3YtdWstZXJyb3ItbWVzc2FnZS5jb21wb25lbnQnO1xuaW1wb3J0IHsgR292VWtGaWVsZHNldENvbXBvbmVudCB9IGZyb20gJy4vZ292LXVrLWZpZWxkc2V0L2dvdi11ay1maWVsZHNldC5jb21wb25lbnQnO1xuaW1wb3J0IHsgR292VWtMYWJlbENvbXBvbmVudCB9IGZyb20gJy4vZ292LXVrLWxhYmVsL2dvdi11ay1sYWJlbC5jb21wb25lbnQnO1xuaW1wb3J0IHsgRm9ybXNNb2R1bGUsIFJlYWN0aXZlRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQgeyBSb3V0ZXJNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xuaW1wb3J0IHsgS2V5Ym9hcmROYXZEaXJlY3RpdmUgfSBmcm9tICcuL2RpcmVjdGl2ZXMva2V5Ym9hcmQtbmF2LmRpcmVjdGl2ZSc7XG5cbkBOZ01vZHVsZSh7XG4gIGltcG9ydHM6IFtcbiAgICBDb21tb25Nb2R1bGUsXG4gICAgRm9ybXNNb2R1bGUsXG4gICAgUmVhY3RpdmVGb3Jtc01vZHVsZSxcbiAgICBDb21tb25Nb2R1bGUsXG4gICAgUm91dGVyTW9kdWxlXG4gIF0sXG4gIGRlY2xhcmF0aW9uczogW1xuICAgIEdvdlVrRGF0ZUNvbXBvbmVudCxcbiAgICBHb3ZVa0Vycm9yTWVzc2FnZUNvbXBvbmVudCxcbiAgICBHb3ZVa0ZpZWxkc2V0Q29tcG9uZW50LFxuICAgIEdvdlVrTGFiZWxDb21wb25lbnQsXG4gICAgS2V5Ym9hcmROYXZEaXJlY3RpdmVcbiAgXSxcbiAgZXhwb3J0czogW1xuICAgIEdvdlVrRGF0ZUNvbXBvbmVudCxcbiAgICBHb3ZVa0Vycm9yTWVzc2FnZUNvbXBvbmVudCxcbiAgICBHb3ZVa0ZpZWxkc2V0Q29tcG9uZW50LFxuICAgIEdvdlVrTGFiZWxDb21wb25lbnQsXG4gICAgS2V5Ym9hcmROYXZEaXJlY3RpdmVcbiAgXVxufSlcbmV4cG9ydCBjbGFzcyBTaGFyZWRNb2R1bGUge31cbiJdfQ==
@@ -1,15 +1,18 @@
1
- import { Component } from '@angular/core';
1
+ import { Component, HostListener } from '@angular/core';
2
2
  import { SearchType } from '../toolbar-event.service';
3
3
  import * as i0 from "@angular/core";
4
4
  import * as i1 from "../toolbar-event.service";
5
5
  import * as i2 from "../toolbar-button-visibility.service";
6
- import * as i3 from "../../shared/directives/tooltip-dismiss.directive";
7
- import * as i4 from "@angular/common";
8
- import * as i5 from "rpx-xui-translation";
6
+ import * as i3 from "../toolbar-focus.service";
7
+ import * as i4 from "../../shared/directives/keyboard-nav.directive";
8
+ import * as i5 from "../../shared/directives/tooltip-dismiss.directive";
9
+ import * as i6 from "@angular/common";
10
+ import * as i7 from "rpx-xui-translation";
9
11
  export class HighlightToolbarComponent {
10
- constructor(toolbarEventService, toolbarButtons) {
12
+ constructor(toolbarEventService, toolbarButtons, toolbarFocusService) {
11
13
  this.toolbarEventService = toolbarEventService;
12
14
  this.toolbarButtons = toolbarButtons;
15
+ this.toolbarFocusService = toolbarFocusService;
13
16
  this.subscriptions = [];
14
17
  }
15
18
  ngOnInit() {
@@ -31,16 +34,41 @@ export class HighlightToolbarComponent {
31
34
  this.toolbarEventService.highlightModeSubject.next(false);
32
35
  this.toolbarEventService.openRedactionSearch.next({ modeType: SearchType.Highlight, isOpen: false });
33
36
  }
37
+ onEscapeKey(event) {
38
+ event.preventDefault();
39
+ event.stopPropagation();
40
+ this.closeAndReturnFocus();
41
+ }
42
+ onArrowUp(event) {
43
+ const target = event.target;
44
+ const highlightToolbar = target.closest('.redaction');
45
+ if (highlightToolbar) {
46
+ this.returnFocusToMainToolbar();
47
+ }
48
+ }
49
+ closeAndReturnFocus() {
50
+ this.onClose();
51
+ this.returnFocusToMainToolbar();
52
+ }
53
+ returnFocusToMainToolbar() {
54
+ this.toolbarFocusService.focusToolbarButton('#mvHighlightBtn');
55
+ }
34
56
  ngOnDestroy() {
35
57
  for (const subscription of this.subscriptions) {
36
58
  subscription.unsubscribe();
37
59
  }
38
60
  }
39
- /** @nocollapse */ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: HighlightToolbarComponent, deps: [{ token: i1.ToolbarEventService }, { token: i2.ToolbarButtonVisibilityService }], target: i0.ɵɵFactoryTarget.Component }); }
40
- /** @nocollapse */ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: HighlightToolbarComponent, selector: "mv-highlight-toolbar", ngImport: i0, template: "<div class=\"redaction\">\n <label class=\"govuk-label redaction-title\" data-l10n-id=\"redaction_options\"\n >{{ \"Highlight options\" | rpxTranslate }}</label\n >\n\n <button\n id=\"mvDrawBtn\"\n #mvDrawBtn\n class=\"mv-button mv-toolbar__menu-button--draw mv-tooltip\"\n [attr.data-tooltip]=\"'Draw a box' | rpxTranslate\"\n [class.toggled]=\"toolbarEventService.drawModeSubject | async\"\n aria-pressed=\"false\"\n data-l10n-id=\"mvDrawBtn\"\n (click)=\"onClickDrawToggle()\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span data-l10n-id=\"draw_label\">{{ \"Draw a box\" | rpxTranslate }}</span>\n </button>\n\n <button\n id=\"highlightTextBtn\"\n class=\"mv-button mv-toolbar__menu-button--highlight mv-tooltip\"\n [attr.data-tooltip]=\"'Highlight text' | rpxTranslate\"\n data-l10n-id=\"highlightTextBtn\"\n (click)=\"onHighlight()\"\n [class.toggled]=\"toolbarEventService.highlightModeSubject | async\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span data-l10n-id=\"highlightTextBtn_label\">{{ \"Highlight text\" | rpxTranslate }}</span>\n </button>\n\n <button\n id=\"mvHighlightFromSearchBtn\"\n [attr.data-tooltip]=\"'From search' | rpxTranslate\"\n data-l10n-id=\"fromSearchButton\"\n class=\"mv-button redaction-button--search mv-tooltip\"\n (click)=\"onAllSearch()\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span style=\"width: 5rem\" data-l10n-id=\"fromSearchButton_label\"\n >{{ \"From search\" | rpxTranslate }}</span\n >\n </button>\n\n <button\n id=\"mvCloseBtn\"\n #mvCloseBtn\n class=\"mv-button redaction-button--close mv-tooltip\"\n [attr.data-tooltip]=\"'Close highlight' | rpxTranslate\"\n data-l10n-id=\"mvRedactBtn\"\n (click)=\"onClose()\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span data-l10n-id=\"Close Redaction\">{{ \"Close Highlight\" | rpxTranslate }}</span>\n </button>\n</div>\n", styles: [""], dependencies: [{ kind: "directive", type: i3.TooltipDismissDirective, selector: ".mv-tooltip, [mvTooltipDismiss]" }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }, { kind: "pipe", type: i5.RpxTranslatePipe, name: "rpxTranslate" }] }); }
61
+ /** @nocollapse */ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: HighlightToolbarComponent, deps: [{ token: i1.ToolbarEventService }, { token: i2.ToolbarButtonVisibilityService }, { token: i3.ToolbarFocusService }], target: i0.ɵɵFactoryTarget.Component }); }
62
+ /** @nocollapse */ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: HighlightToolbarComponent, selector: "mv-highlight-toolbar", host: { listeners: { "keydown.escape": "onEscapeKey($event)", "keydown.arrowup": "onArrowUp($event)" } }, ngImport: i0, template: "<div class=\"redaction\" [mvKeyboardNav]=\"'horizontal'\">\n <label class=\"govuk-label redaction-title\" data-l10n-id=\"redaction_options\"\n >{{ \"Highlight options\" | rpxTranslate }}</label\n >\n\n <button\n id=\"mvDrawBtn\"\n #mvDrawBtn\n class=\"mv-button mv-toolbar__menu-button--draw mv-tooltip\"\n [attr.data-tooltip]=\"'Draw a box' | rpxTranslate\"\n [class.toggled]=\"toolbarEventService.drawModeSubject | async\"\n aria-pressed=\"false\"\n data-l10n-id=\"mvDrawBtn\"\n (click)=\"onClickDrawToggle()\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span data-l10n-id=\"draw_label\">{{ \"Draw a box\" | rpxTranslate }}</span>\n </button>\n\n <button\n id=\"highlightTextBtn\"\n class=\"mv-button mv-toolbar__menu-button--highlight mv-tooltip\"\n [attr.data-tooltip]=\"'Highlight text' | rpxTranslate\"\n data-l10n-id=\"highlightTextBtn\"\n (click)=\"onHighlight()\"\n [class.toggled]=\"toolbarEventService.highlightModeSubject | async\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span data-l10n-id=\"highlightTextBtn_label\">{{ \"Highlight text\" | rpxTranslate }}</span>\n </button>\n\n <button\n id=\"mvHighlightFromSearchBtn\"\n [attr.data-tooltip]=\"'From search' | rpxTranslate\"\n data-l10n-id=\"fromSearchButton\"\n class=\"mv-button redaction-button--search mv-tooltip\"\n (click)=\"onAllSearch()\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span style=\"width: 5rem\" data-l10n-id=\"fromSearchButton_label\"\n >{{ \"From search\" | rpxTranslate }}</span\n >\n </button>\n\n <button\n id=\"mvCloseBtn\"\n #mvCloseBtn\n class=\"mv-button redaction-button--close mv-tooltip\"\n [attr.data-tooltip]=\"'Close highlight' | rpxTranslate\"\n data-l10n-id=\"mvRedactBtn\"\n (click)=\"onClose()\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span data-l10n-id=\"Close Redaction\">{{ \"Close Highlight\" | rpxTranslate }}</span>\n </button>\n</div>\n", styles: [""], dependencies: [{ kind: "directive", type: i4.KeyboardNavDirective, selector: "[mvKeyboardNav]", inputs: ["mvKeyboardNav"], outputs: ["itemFocused", "itemActivated"] }, { kind: "directive", type: i5.TooltipDismissDirective, selector: ".mv-tooltip, [mvTooltipDismiss]" }, { kind: "pipe", type: i6.AsyncPipe, name: "async" }, { kind: "pipe", type: i7.RpxTranslatePipe, name: "rpxTranslate" }] }); }
41
63
  }
42
64
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: HighlightToolbarComponent, decorators: [{
43
65
  type: Component,
44
- args: [{ selector: 'mv-highlight-toolbar', template: "<div class=\"redaction\">\n <label class=\"govuk-label redaction-title\" data-l10n-id=\"redaction_options\"\n >{{ \"Highlight options\" | rpxTranslate }}</label\n >\n\n <button\n id=\"mvDrawBtn\"\n #mvDrawBtn\n class=\"mv-button mv-toolbar__menu-button--draw mv-tooltip\"\n [attr.data-tooltip]=\"'Draw a box' | rpxTranslate\"\n [class.toggled]=\"toolbarEventService.drawModeSubject | async\"\n aria-pressed=\"false\"\n data-l10n-id=\"mvDrawBtn\"\n (click)=\"onClickDrawToggle()\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span data-l10n-id=\"draw_label\">{{ \"Draw a box\" | rpxTranslate }}</span>\n </button>\n\n <button\n id=\"highlightTextBtn\"\n class=\"mv-button mv-toolbar__menu-button--highlight mv-tooltip\"\n [attr.data-tooltip]=\"'Highlight text' | rpxTranslate\"\n data-l10n-id=\"highlightTextBtn\"\n (click)=\"onHighlight()\"\n [class.toggled]=\"toolbarEventService.highlightModeSubject | async\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span data-l10n-id=\"highlightTextBtn_label\">{{ \"Highlight text\" | rpxTranslate }}</span>\n </button>\n\n <button\n id=\"mvHighlightFromSearchBtn\"\n [attr.data-tooltip]=\"'From search' | rpxTranslate\"\n data-l10n-id=\"fromSearchButton\"\n class=\"mv-button redaction-button--search mv-tooltip\"\n (click)=\"onAllSearch()\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span style=\"width: 5rem\" data-l10n-id=\"fromSearchButton_label\"\n >{{ \"From search\" | rpxTranslate }}</span\n >\n </button>\n\n <button\n id=\"mvCloseBtn\"\n #mvCloseBtn\n class=\"mv-button redaction-button--close mv-tooltip\"\n [attr.data-tooltip]=\"'Close highlight' | rpxTranslate\"\n data-l10n-id=\"mvRedactBtn\"\n (click)=\"onClose()\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span data-l10n-id=\"Close Redaction\">{{ \"Close Highlight\" | rpxTranslate }}</span>\n </button>\n</div>\n" }]
45
- }], ctorParameters: () => [{ type: i1.ToolbarEventService }, { type: i2.ToolbarButtonVisibilityService }] });
46
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGlnaGxpZ2h0LXRvb2xiYXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbWVkaWEtdmlld2VyL3NyYy9saWIvdG9vbGJhci9oaWdobGlnaHQtdG9vbGJhci9oaWdobGlnaHQtdG9vbGJhci5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9tZWRpYS12aWV3ZXIvc3JjL2xpYi90b29sYmFyL2hpZ2hsaWdodC10b29sYmFyL2hpZ2hsaWdodC10b29sYmFyLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQXFCLE1BQU0sZUFBZSxDQUFDO0FBQzdELE9BQU8sRUFBYyxVQUFVLEVBQXVCLE1BQU0sMEJBQTBCLENBQUM7Ozs7Ozs7QUFZdkYsTUFBTSxPQUFPLHlCQUF5QjtJQUtwQyxZQUE0QixtQkFBd0MsRUFDbEQsY0FBOEM7UUFEcEMsd0JBQW1CLEdBQW5CLG1CQUFtQixDQUFxQjtRQUNsRCxtQkFBYyxHQUFkLGNBQWMsQ0FBZ0M7UUFKeEQsa0JBQWEsR0FBbUIsRUFBRSxDQUFDO0lBSXlCLENBQUM7SUFFckUsUUFBUTtRQUNOLElBQUksQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxtQkFBbUIsQ0FBQywwQkFBMEIsQ0FBQyxTQUFTLENBQUMsVUFBVSxDQUFDLEVBQUU7WUFDakcsSUFBSSxDQUFDLHNCQUFzQixHQUFHLFVBQVUsQ0FBQztRQUMzQyxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ04sQ0FBQztJQUVELFdBQVc7UUFDVCxJQUFJLENBQUMsbUJBQW1CLENBQUMsbUJBQW1CLENBQUMsSUFBSSxDQUFDLEVBQUUsUUFBUSxFQUFFLFVBQVUsQ0FBQyxTQUFTLEVBQUUsTUFBTSxFQUFFLElBQUksRUFBZ0IsQ0FBQyxDQUFDO0lBQ3BILENBQUM7SUFFRCxXQUFXO1FBQ1QsSUFBSSxDQUFDLG1CQUFtQixDQUFDLG1CQUFtQixFQUFFLENBQUM7SUFDakQsQ0FBQztJQUVELGlCQUFpQjtRQUNmLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxjQUFjLEVBQUUsQ0FBQztJQUM1QyxDQUFDO0lBRUQsT0FBTztRQUNMLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyx1QkFBdUIsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDN0QsSUFBSSxDQUFDLG1CQUFtQixDQUFDLG9CQUFvQixDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUMxRCxJQUFJLENBQUMsbUJBQW1CLENBQUMsbUJBQW1CLENBQUMsSUFBSSxDQUFDLEVBQUUsUUFBUSxFQUFFLFVBQVUsQ0FBQyxTQUFTLEVBQUUsTUFBTSxFQUFFLEtBQUssRUFBZ0IsQ0FBQyxDQUFDO0lBQ3JILENBQUM7SUFFRCxXQUFXO1FBQ1QsS0FBSyxNQUFNLFlBQVksSUFBSSxJQUFJLENBQUMsYUFBYSxFQUFFLENBQUM7WUFDOUMsWUFBWSxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQzdCLENBQUM7SUFDSCxDQUFDO2tJQXBDVSx5QkFBeUI7c0hBQXpCLHlCQUF5Qiw0RENidEMseTZEQXdEQTs7NEZEM0NhLHlCQUF5QjtrQkFMckMsU0FBUzsrQkFDRSxzQkFBc0IiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIE9uRGVzdHJveSwgT25Jbml0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBTZWFyY2hNb2RlLCBTZWFyY2hUeXBlLCBUb29sYmFyRXZlbnRTZXJ2aWNlIH0gZnJvbSAnLi4vdG9vbGJhci1ldmVudC5zZXJ2aWNlJztcbmltcG9ydCB7IFRvb2xiYXJCdXR0b25WaXNpYmlsaXR5U2VydmljZSB9IGZyb20gJy4uL3Rvb2xiYXItYnV0dG9uLXZpc2liaWxpdHkuc2VydmljZSc7XG5pbXBvcnQgKiBhcyBmcm9tU3RvcmUgZnJvbSAnLi4vLi4vc3RvcmUvcmVkdWNlcnMvcmVkdWNlcnMnO1xuaW1wb3J0IHsgU3RvcmUsIHNlbGVjdCB9IGZyb20gJ0BuZ3J4L3N0b3JlJztcbmltcG9ydCB7IFN1YnNjcmlwdGlvbiB9IGZyb20gJ3J4anMnO1xuaW1wb3J0ICogYXMgZnJvbVJlZGFjdFNlbGVjdG9ycyBmcm9tICcuLi8uLi9zdG9yZS9zZWxlY3RvcnMvcmVkYWN0aW9uLnNlbGVjdG9ycyc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ212LWhpZ2hsaWdodC10b29sYmFyJyxcbiAgdGVtcGxhdGVVcmw6ICcuL2hpZ2hsaWdodC10b29sYmFyLmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVVcmxzOiBbJy4vaGlnaGxpZ2h0LXRvb2xiYXIuY29tcG9uZW50LnNjc3MnXVxufSlcbmV4cG9ydCBjbGFzcyBIaWdobGlnaHRUb29sYmFyQ29tcG9uZW50IGltcGxlbWVudHMgT25Jbml0LCBPbkRlc3Ryb3kge1xuXG4gIHByaXZhdGUgc3Vic2NyaXB0aW9uczogU3Vic2NyaXB0aW9uW10gPSBbXTtcbiAgcmVkYWN0aW9uQWxsSW5Qcm9ncmVzczogYm9vbGVhbjtcblxuICBjb25zdHJ1Y3RvcihwdWJsaWMgcmVhZG9ubHkgdG9vbGJhckV2ZW50U2VydmljZTogVG9vbGJhckV2ZW50U2VydmljZSxcbiAgICBwdWJsaWMgcmVhZG9ubHkgdG9vbGJhckJ1dHRvbnM6IFRvb2xiYXJCdXR0b25WaXNpYmlsaXR5U2VydmljZSkgeyB9XG5cbiAgbmdPbkluaXQoKTogdm9pZCB7XG4gICAgdGhpcy5zdWJzY3JpcHRpb25zLnB1c2godGhpcy50b29sYmFyRXZlbnRTZXJ2aWNlLnJlZGFjdEFsbEluUHJvZ3Jlc3NTdWJqZWN0LnN1YnNjcmliZShpbnByb2dyZXNzID0+IHtcbiAgICAgIHRoaXMucmVkYWN0aW9uQWxsSW5Qcm9ncmVzcyA9IGlucHJvZ3Jlc3M7XG4gICAgfSkpO1xuICB9XG5cbiAgb25BbGxTZWFyY2goKSB7XG4gICAgdGhpcy50b29sYmFyRXZlbnRTZXJ2aWNlLm9wZW5SZWRhY3Rpb25TZWFyY2gubmV4dCh7IG1vZGVUeXBlOiBTZWFyY2hUeXBlLkhpZ2hsaWdodCwgaXNPcGVuOiB0cnVlIH0gYXMgU2VhcmNoTW9kZSk7XG4gIH1cblxuICBvbkhpZ2hsaWdodCgpIHtcbiAgICB0aGlzLnRvb2xiYXJFdmVudFNlcnZpY2UudG9nZ2xlSGlnaGxpZ2h0TW9kZSgpO1xuICB9XG5cbiAgb25DbGlja0RyYXdUb2dnbGUoKSB7XG4gICAgdGhpcy50b29sYmFyRXZlbnRTZXJ2aWNlLnRvZ2dsZURyYXdNb2RlKCk7XG4gIH1cblxuICBvbkNsb3NlKCkge1xuICAgIHRoaXMudG9vbGJhckV2ZW50U2VydmljZS5oaWdobGlnaHRUb29sYmFyU3ViamVjdC5uZXh0KGZhbHNlKTtcbiAgICB0aGlzLnRvb2xiYXJFdmVudFNlcnZpY2UuaGlnaGxpZ2h0TW9kZVN1YmplY3QubmV4dChmYWxzZSk7XG4gICAgdGhpcy50b29sYmFyRXZlbnRTZXJ2aWNlLm9wZW5SZWRhY3Rpb25TZWFyY2gubmV4dCh7IG1vZGVUeXBlOiBTZWFyY2hUeXBlLkhpZ2hsaWdodCwgaXNPcGVuOiBmYWxzZSB9IGFzIFNlYXJjaE1vZGUpO1xuICB9XG5cbiAgbmdPbkRlc3Ryb3koKTogdm9pZCB7XG4gICAgZm9yIChjb25zdCBzdWJzY3JpcHRpb24gb2YgdGhpcy5zdWJzY3JpcHRpb25zKSB7XG4gICAgICBzdWJzY3JpcHRpb24udW5zdWJzY3JpYmUoKTtcbiAgICB9XG4gIH1cblxufVxuIiwiPGRpdiBjbGFzcz1cInJlZGFjdGlvblwiPlxuICA8bGFiZWwgY2xhc3M9XCJnb3Z1ay1sYWJlbCByZWRhY3Rpb24tdGl0bGVcIiBkYXRhLWwxMG4taWQ9XCJyZWRhY3Rpb25fb3B0aW9uc1wiXG4gICAgPnt7IFwiSGlnaGxpZ2h0IG9wdGlvbnNcIiB8IHJweFRyYW5zbGF0ZSB9fTwvbGFiZWxcbiAgPlxuXG4gIDxidXR0b25cbiAgICBpZD1cIm12RHJhd0J0blwiXG4gICAgI212RHJhd0J0blxuICAgIGNsYXNzPVwibXYtYnV0dG9uIG12LXRvb2xiYXJfX21lbnUtYnV0dG9uLS1kcmF3IG12LXRvb2x0aXBcIlxuICAgIFthdHRyLmRhdGEtdG9vbHRpcF09XCInRHJhdyBhIGJveCcgfCBycHhUcmFuc2xhdGVcIlxuICAgIFtjbGFzcy50b2dnbGVkXT1cInRvb2xiYXJFdmVudFNlcnZpY2UuZHJhd01vZGVTdWJqZWN0IHwgYXN5bmNcIlxuICAgIGFyaWEtcHJlc3NlZD1cImZhbHNlXCJcbiAgICBkYXRhLWwxMG4taWQ9XCJtdkRyYXdCdG5cIlxuICAgIChjbGljayk9XCJvbkNsaWNrRHJhd1RvZ2dsZSgpXCJcbiAgICBbZGlzYWJsZWRdPVwicmVkYWN0aW9uQWxsSW5Qcm9ncmVzc1wiXG4gID5cbiAgICA8c3BhbiBkYXRhLWwxMG4taWQ9XCJkcmF3X2xhYmVsXCI+e3sgXCJEcmF3IGEgYm94XCIgfCBycHhUcmFuc2xhdGUgfX08L3NwYW4+XG4gIDwvYnV0dG9uPlxuXG4gIDxidXR0b25cbiAgICBpZD1cImhpZ2hsaWdodFRleHRCdG5cIlxuICAgIGNsYXNzPVwibXYtYnV0dG9uIG12LXRvb2xiYXJfX21lbnUtYnV0dG9uLS1oaWdobGlnaHQgbXYtdG9vbHRpcFwiXG4gICAgW2F0dHIuZGF0YS10b29sdGlwXT1cIidIaWdobGlnaHQgdGV4dCcgfCBycHhUcmFuc2xhdGVcIlxuICAgIGRhdGEtbDEwbi1pZD1cImhpZ2hsaWdodFRleHRCdG5cIlxuICAgIChjbGljayk9XCJvbkhpZ2hsaWdodCgpXCJcbiAgICBbY2xhc3MudG9nZ2xlZF09XCJ0b29sYmFyRXZlbnRTZXJ2aWNlLmhpZ2hsaWdodE1vZGVTdWJqZWN0IHwgYXN5bmNcIlxuICAgIFtkaXNhYmxlZF09XCJyZWRhY3Rpb25BbGxJblByb2dyZXNzXCJcbiAgPlxuICAgIDxzcGFuIGRhdGEtbDEwbi1pZD1cImhpZ2hsaWdodFRleHRCdG5fbGFiZWxcIj57eyBcIkhpZ2hsaWdodCB0ZXh0XCIgfCBycHhUcmFuc2xhdGUgfX08L3NwYW4+XG4gIDwvYnV0dG9uPlxuXG4gIDxidXR0b25cbiAgICBpZD1cIm12SGlnaGxpZ2h0RnJvbVNlYXJjaEJ0blwiXG4gICAgW2F0dHIuZGF0YS10b29sdGlwXT1cIidGcm9tIHNlYXJjaCcgfCBycHhUcmFuc2xhdGVcIlxuICAgIGRhdGEtbDEwbi1pZD1cImZyb21TZWFyY2hCdXR0b25cIlxuICAgIGNsYXNzPVwibXYtYnV0dG9uIHJlZGFjdGlvbi1idXR0b24tLXNlYXJjaCBtdi10b29sdGlwXCJcbiAgICAoY2xpY2spPVwib25BbGxTZWFyY2goKVwiXG4gICAgW2Rpc2FibGVkXT1cInJlZGFjdGlvbkFsbEluUHJvZ3Jlc3NcIlxuICA+XG4gICAgPHNwYW4gc3R5bGU9XCJ3aWR0aDogNXJlbVwiIGRhdGEtbDEwbi1pZD1cImZyb21TZWFyY2hCdXR0b25fbGFiZWxcIlxuICAgICAgPnt7IFwiRnJvbSBzZWFyY2hcIiB8IHJweFRyYW5zbGF0ZSB9fTwvc3BhblxuICAgID5cbiAgPC9idXR0b24+XG5cbiAgPGJ1dHRvblxuICAgIGlkPVwibXZDbG9zZUJ0blwiXG4gICAgI212Q2xvc2VCdG5cbiAgICBjbGFzcz1cIm12LWJ1dHRvbiByZWRhY3Rpb24tYnV0dG9uLS1jbG9zZSBtdi10b29sdGlwXCJcbiAgICBbYXR0ci5kYXRhLXRvb2x0aXBdPVwiJ0Nsb3NlIGhpZ2hsaWdodCcgfCBycHhUcmFuc2xhdGVcIlxuICAgIGRhdGEtbDEwbi1pZD1cIm12UmVkYWN0QnRuXCJcbiAgICAoY2xpY2spPVwib25DbG9zZSgpXCJcbiAgICBbZGlzYWJsZWRdPVwicmVkYWN0aW9uQWxsSW5Qcm9ncmVzc1wiXG4gID5cbiAgICA8c3BhbiBkYXRhLWwxMG4taWQ9XCJDbG9zZSBSZWRhY3Rpb25cIj57eyBcIkNsb3NlIEhpZ2hsaWdodFwiIHwgcnB4VHJhbnNsYXRlIH19PC9zcGFuPlxuICA8L2J1dHRvbj5cbjwvZGl2PlxuIl19
66
+ args: [{ selector: 'mv-highlight-toolbar', template: "<div class=\"redaction\" [mvKeyboardNav]=\"'horizontal'\">\n <label class=\"govuk-label redaction-title\" data-l10n-id=\"redaction_options\"\n >{{ \"Highlight options\" | rpxTranslate }}</label\n >\n\n <button\n id=\"mvDrawBtn\"\n #mvDrawBtn\n class=\"mv-button mv-toolbar__menu-button--draw mv-tooltip\"\n [attr.data-tooltip]=\"'Draw a box' | rpxTranslate\"\n [class.toggled]=\"toolbarEventService.drawModeSubject | async\"\n aria-pressed=\"false\"\n data-l10n-id=\"mvDrawBtn\"\n (click)=\"onClickDrawToggle()\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span data-l10n-id=\"draw_label\">{{ \"Draw a box\" | rpxTranslate }}</span>\n </button>\n\n <button\n id=\"highlightTextBtn\"\n class=\"mv-button mv-toolbar__menu-button--highlight mv-tooltip\"\n [attr.data-tooltip]=\"'Highlight text' | rpxTranslate\"\n data-l10n-id=\"highlightTextBtn\"\n (click)=\"onHighlight()\"\n [class.toggled]=\"toolbarEventService.highlightModeSubject | async\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span data-l10n-id=\"highlightTextBtn_label\">{{ \"Highlight text\" | rpxTranslate }}</span>\n </button>\n\n <button\n id=\"mvHighlightFromSearchBtn\"\n [attr.data-tooltip]=\"'From search' | rpxTranslate\"\n data-l10n-id=\"fromSearchButton\"\n class=\"mv-button redaction-button--search mv-tooltip\"\n (click)=\"onAllSearch()\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span style=\"width: 5rem\" data-l10n-id=\"fromSearchButton_label\"\n >{{ \"From search\" | rpxTranslate }}</span\n >\n </button>\n\n <button\n id=\"mvCloseBtn\"\n #mvCloseBtn\n class=\"mv-button redaction-button--close mv-tooltip\"\n [attr.data-tooltip]=\"'Close highlight' | rpxTranslate\"\n data-l10n-id=\"mvRedactBtn\"\n (click)=\"onClose()\"\n [disabled]=\"redactionAllInProgress\"\n >\n <span data-l10n-id=\"Close Redaction\">{{ \"Close Highlight\" | rpxTranslate }}</span>\n </button>\n</div>\n" }]
67
+ }], ctorParameters: () => [{ type: i1.ToolbarEventService }, { type: i2.ToolbarButtonVisibilityService }, { type: i3.ToolbarFocusService }], propDecorators: { onEscapeKey: [{
68
+ type: HostListener,
69
+ args: ['keydown.escape', ['$event']]
70
+ }], onArrowUp: [{
71
+ type: HostListener,
72
+ args: ['keydown.arrowup', ['$event']]
73
+ }] } });
74
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"highlight-toolbar.component.js","sourceRoot":"","sources":["../../../../../../projects/media-viewer/src/lib/toolbar/highlight-toolbar/highlight-toolbar.component.ts","../../../../../../projects/media-viewer/src/lib/toolbar/highlight-toolbar/highlight-toolbar.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,YAAY,EAAqB,MAAM,eAAe,CAAC;AAC3E,OAAO,EAAc,UAAU,EAAuB,MAAM,0BAA0B,CAAC;;;;;;;;;AAavF,MAAM,OAAO,yBAAyB;IAKpC,YAA4B,mBAAwC,EAClD,cAA8C,EAC7C,mBAAwC;QAF/B,wBAAmB,GAAnB,mBAAmB,CAAqB;QAClD,mBAAc,GAAd,cAAc,CAAgC;QAC7C,wBAAmB,GAAnB,mBAAmB,CAAqB;QALnD,kBAAa,GAAmB,EAAE,CAAC;IAMvC,CAAC;IAEL,QAAQ;QACN,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,mBAAmB,CAAC,0BAA0B,CAAC,SAAS,CAAC,UAAU,CAAC,EAAE;YACjG,IAAI,CAAC,sBAAsB,GAAG,UAAU,CAAC;QAC3C,CAAC,CAAC,CAAC,CAAC;IACN,CAAC;IAED,WAAW;QACT,IAAI,CAAC,mBAAmB,CAAC,mBAAmB,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,UAAU,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAgB,CAAC,CAAC;IACpH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,mBAAmB,CAAC,mBAAmB,EAAE,CAAC;IACjD,CAAC;IAED,iBAAiB;QACf,IAAI,CAAC,mBAAmB,CAAC,cAAc,EAAE,CAAC;IAC5C,CAAC;IAED,OAAO;QACL,IAAI,CAAC,mBAAmB,CAAC,uBAAuB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC7D,IAAI,CAAC,mBAAmB,CAAC,oBAAoB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC1D,IAAI,CAAC,mBAAmB,CAAC,mBAAmB,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,UAAU,CAAC,SAAS,EAAE,MAAM,EAAE,KAAK,EAAgB,CAAC,CAAC;IACrH,CAAC;IAGD,WAAW,CAAC,KAAoB;QAC9B,KAAK,CAAC,cAAc,EAAE,CAAC;QACvB,KAAK,CAAC,eAAe,EAAE,CAAC;QACxB,IAAI,CAAC,mBAAmB,EAAE,CAAC;IAC7B,CAAC;IAGD,SAAS,CAAC,KAAoB;QAC5B,MAAM,MAAM,GAAG,KAAK,CAAC,MAAqB,CAAC;QAC3C,MAAM,gBAAgB,GAAG,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;QACtD,IAAI,gBAAgB,EAAE,CAAC;YACrB,IAAI,CAAC,wBAAwB,EAAE,CAAC;QAClC,CAAC;IACH,CAAC;IAEO,mBAAmB;QACzB,IAAI,CAAC,OAAO,EAAE,CAAC;QACf,IAAI,CAAC,wBAAwB,EAAE,CAAC;IAClC,CAAC;IAEO,wBAAwB;QAC5B,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,CAAC,iBAAiB,CAAC,CAAC;IACnE,CAAC;IAED,WAAW;QACT,KAAK,MAAM,YAAY,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YAC9C,YAAY,CAAC,WAAW,EAAE,CAAC;QAC7B,CAAC;IACH,CAAC;kIA/DU,yBAAyB;sHAAzB,yBAAyB,sKCdtC,08DAwDA;;4FD1Ca,yBAAyB;kBALrC,SAAS;+BACE,sBAAsB;uKAuChC,WAAW;sBADV,YAAY;uBAAC,gBAAgB,EAAE,CAAC,QAAQ,CAAC;gBAQ1C,SAAS;sBADR,YAAY;uBAAC,iBAAiB,EAAE,CAAC,QAAQ,CAAC","sourcesContent":["import { Component, HostListener, OnDestroy, OnInit } from '@angular/core';\nimport { SearchMode, SearchType, ToolbarEventService } from '../toolbar-event.service';\nimport { ToolbarButtonVisibilityService } from '../toolbar-button-visibility.service';\nimport { ToolbarFocusService } from '../toolbar-focus.service';\nimport * as fromStore from '../../store/reducers/reducers';\nimport { Store, select } from '@ngrx/store';\nimport { Subscription } from 'rxjs';\nimport * as fromRedactSelectors from '../../store/selectors/redaction.selectors';\n\n@Component({\n  selector: 'mv-highlight-toolbar',\n  templateUrl: './highlight-toolbar.component.html',\n  styleUrls: ['./highlight-toolbar.component.scss']\n})\nexport class HighlightToolbarComponent implements OnInit, OnDestroy {\n\n  private subscriptions: Subscription[] = [];\n  redactionAllInProgress: boolean;\n\n  constructor(public readonly toolbarEventService: ToolbarEventService,\n    public readonly toolbarButtons: ToolbarButtonVisibilityService,\n    private readonly toolbarFocusService: ToolbarFocusService\n  ) { }\n\n  ngOnInit(): void {\n    this.subscriptions.push(this.toolbarEventService.redactAllInProgressSubject.subscribe(inprogress => {\n      this.redactionAllInProgress = inprogress;\n    }));\n  }\n\n  onAllSearch() {\n    this.toolbarEventService.openRedactionSearch.next({ modeType: SearchType.Highlight, isOpen: true } as SearchMode);\n  }\n\n  onHighlight() {\n    this.toolbarEventService.toggleHighlightMode();\n  }\n\n  onClickDrawToggle() {\n    this.toolbarEventService.toggleDrawMode();\n  }\n\n  onClose() {\n    this.toolbarEventService.highlightToolbarSubject.next(false);\n    this.toolbarEventService.highlightModeSubject.next(false);\n    this.toolbarEventService.openRedactionSearch.next({ modeType: SearchType.Highlight, isOpen: false } as SearchMode);\n  }\n\n  @HostListener('keydown.escape', ['$event'])\n  onEscapeKey(event: KeyboardEvent) {\n    event.preventDefault();\n    event.stopPropagation();\n    this.closeAndReturnFocus();\n  }\n\n  @HostListener('keydown.arrowup', ['$event'])\n  onArrowUp(event: KeyboardEvent) {\n    const target = event.target as HTMLElement;\n    const highlightToolbar = target.closest('.redaction');\n    if (highlightToolbar) {\n      this.returnFocusToMainToolbar();\n    }\n  }\n\n  private closeAndReturnFocus() {\n    this.onClose();\n    this.returnFocusToMainToolbar();\n  }\n\n  private returnFocusToMainToolbar() {\n      this.toolbarFocusService.focusToolbarButton('#mvHighlightBtn');\n  }\n\n  ngOnDestroy(): void {\n    for (const subscription of this.subscriptions) {\n      subscription.unsubscribe();\n    }\n  }\n\n}\n","<div class=\"redaction\" [mvKeyboardNav]=\"'horizontal'\">\n  <label class=\"govuk-label redaction-title\" data-l10n-id=\"redaction_options\"\n    >{{ \"Highlight options\" | rpxTranslate }}</label\n  >\n\n  <button\n    id=\"mvDrawBtn\"\n    #mvDrawBtn\n    class=\"mv-button mv-toolbar__menu-button--draw mv-tooltip\"\n    [attr.data-tooltip]=\"'Draw a box' | rpxTranslate\"\n    [class.toggled]=\"toolbarEventService.drawModeSubject | async\"\n    aria-pressed=\"false\"\n    data-l10n-id=\"mvDrawBtn\"\n    (click)=\"onClickDrawToggle()\"\n    [disabled]=\"redactionAllInProgress\"\n  >\n    <span data-l10n-id=\"draw_label\">{{ \"Draw a box\" | rpxTranslate }}</span>\n  </button>\n\n  <button\n    id=\"highlightTextBtn\"\n    class=\"mv-button mv-toolbar__menu-button--highlight mv-tooltip\"\n    [attr.data-tooltip]=\"'Highlight text' | rpxTranslate\"\n    data-l10n-id=\"highlightTextBtn\"\n    (click)=\"onHighlight()\"\n    [class.toggled]=\"toolbarEventService.highlightModeSubject | async\"\n    [disabled]=\"redactionAllInProgress\"\n  >\n    <span data-l10n-id=\"highlightTextBtn_label\">{{ \"Highlight text\" | rpxTranslate }}</span>\n  </button>\n\n  <button\n    id=\"mvHighlightFromSearchBtn\"\n    [attr.data-tooltip]=\"'From search' | rpxTranslate\"\n    data-l10n-id=\"fromSearchButton\"\n    class=\"mv-button redaction-button--search mv-tooltip\"\n    (click)=\"onAllSearch()\"\n    [disabled]=\"redactionAllInProgress\"\n  >\n    <span style=\"width: 5rem\" data-l10n-id=\"fromSearchButton_label\"\n      >{{ \"From search\" | rpxTranslate }}</span\n    >\n  </button>\n\n  <button\n    id=\"mvCloseBtn\"\n    #mvCloseBtn\n    class=\"mv-button redaction-button--close mv-tooltip\"\n    [attr.data-tooltip]=\"'Close highlight' | rpxTranslate\"\n    data-l10n-id=\"mvRedactBtn\"\n    (click)=\"onClose()\"\n    [disabled]=\"redactionAllInProgress\"\n  >\n    <span data-l10n-id=\"Close Redaction\">{{ \"Close Highlight\" | rpxTranslate }}</span>\n  </button>\n</div>\n"]}