@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.
- package/esm2022/lib/icp/confirm-exit/confirm-action-dialog.component.mjs +13 -5
- package/esm2022/lib/media-viewer.component.mjs +98 -4
- package/esm2022/lib/shared/directives/keyboard-nav.directive.mjs +243 -0
- package/esm2022/lib/shared/shared.module.mjs +10 -5
- package/esm2022/lib/toolbar/highlight-toolbar/highlight-toolbar.component.mjs +38 -10
- package/esm2022/lib/toolbar/main-toolbar/main-toolbar.component.mjs +82 -18
- package/esm2022/lib/toolbar/redaction-toolbar/redaction-toolbar.component.mjs +35 -10
- package/esm2022/lib/toolbar/toolbar-focus.service.mjs +33 -0
- package/esm2022/lib/toolbar/toolbar.module.mjs +12 -6
- package/fesm2022/hmcts-media-viewer.mjs +609 -117
- package/fesm2022/hmcts-media-viewer.mjs.map +1 -1
- package/lib/icp/confirm-exit/confirm-action-dialog.component.d.ts +4 -1
- package/lib/icp/confirm-exit/confirm-action-dialog.component.d.ts.map +1 -1
- package/lib/media-viewer.component.d.ts +8 -0
- package/lib/media-viewer.component.d.ts.map +1 -1
- package/lib/shared/directives/keyboard-nav.directive.d.ts +30 -0
- package/lib/shared/directives/keyboard-nav.directive.d.ts.map +1 -0
- package/lib/shared/shared.module.d.ts +5 -4
- package/lib/shared/shared.module.d.ts.map +1 -1
- package/lib/toolbar/highlight-toolbar/highlight-toolbar.component.d.ts +7 -1
- package/lib/toolbar/highlight-toolbar/highlight-toolbar.component.d.ts.map +1 -1
- package/lib/toolbar/main-toolbar/main-toolbar.component.d.ts +12 -1
- package/lib/toolbar/main-toolbar/main-toolbar.component.d.ts.map +1 -1
- package/lib/toolbar/redaction-toolbar/redaction-toolbar.component.d.ts +6 -1
- package/lib/toolbar/redaction-toolbar/redaction-toolbar.component.d.ts.map +1 -1
- package/lib/toolbar/toolbar-focus.service.d.ts +7 -0
- package/lib/toolbar/toolbar-focus.service.d.ts.map +1 -0
- package/lib/toolbar/toolbar.module.d.ts +2 -1
- package/lib/toolbar/toolbar.module.d.ts.map +1 -1
- 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
|
|
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,
|
|
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 "
|
|
7
|
-
import * as i4 from "
|
|
8
|
-
import * as i5 from "
|
|
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:
|
|
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
|
-
|
|
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"]}
|