@ckeditor/ckeditor5-ui 39.0.2 → 40.0.0

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 (226) hide show
  1. package/lang/contexts.json +5 -1
  2. package/lang/translations/ar.po +16 -0
  3. package/lang/translations/ast.po +16 -0
  4. package/lang/translations/az.po +16 -0
  5. package/lang/translations/bg.po +16 -0
  6. package/lang/translations/bn.po +16 -0
  7. package/lang/translations/ca.po +16 -0
  8. package/lang/translations/cs.po +16 -0
  9. package/lang/translations/da.po +16 -0
  10. package/lang/translations/de-ch.po +16 -0
  11. package/lang/translations/de.po +16 -0
  12. package/lang/translations/el.po +16 -0
  13. package/lang/translations/en-au.po +16 -0
  14. package/lang/translations/en-gb.po +16 -0
  15. package/lang/translations/en.po +16 -0
  16. package/lang/translations/eo.po +16 -0
  17. package/lang/translations/es.po +16 -0
  18. package/lang/translations/et.po +16 -0
  19. package/lang/translations/eu.po +16 -0
  20. package/lang/translations/fa.po +16 -0
  21. package/lang/translations/fi.po +16 -0
  22. package/lang/translations/fr.po +16 -0
  23. package/lang/translations/gl.po +16 -0
  24. package/lang/translations/he.po +16 -0
  25. package/lang/translations/hi.po +16 -0
  26. package/lang/translations/hr.po +16 -0
  27. package/lang/translations/hu.po +16 -0
  28. package/lang/translations/id.po +16 -0
  29. package/lang/translations/it.po +16 -0
  30. package/lang/translations/ja.po +16 -0
  31. package/lang/translations/km.po +16 -0
  32. package/lang/translations/kn.po +16 -0
  33. package/lang/translations/ko.po +16 -0
  34. package/lang/translations/ku.po +16 -0
  35. package/lang/translations/lt.po +16 -0
  36. package/lang/translations/lv.po +16 -0
  37. package/lang/translations/ms.po +16 -0
  38. package/lang/translations/nb.po +16 -0
  39. package/lang/translations/ne.po +16 -0
  40. package/lang/translations/nl.po +16 -0
  41. package/lang/translations/no.po +16 -0
  42. package/lang/translations/pl.po +16 -0
  43. package/lang/translations/pt-br.po +16 -0
  44. package/lang/translations/pt.po +16 -0
  45. package/lang/translations/ro.po +16 -0
  46. package/lang/translations/ru.po +16 -0
  47. package/lang/translations/sk.po +16 -0
  48. package/lang/translations/sl.po +16 -0
  49. package/lang/translations/sq.po +16 -0
  50. package/lang/translations/sr-latn.po +16 -0
  51. package/lang/translations/sr.po +16 -0
  52. package/lang/translations/sv.po +16 -0
  53. package/lang/translations/th.po +16 -0
  54. package/lang/translations/tk.po +16 -0
  55. package/lang/translations/tr.po +16 -0
  56. package/lang/translations/tt.po +16 -0
  57. package/lang/translations/ug.po +16 -0
  58. package/lang/translations/uk.po +16 -0
  59. package/lang/translations/ur.po +16 -0
  60. package/lang/translations/uz.po +16 -0
  61. package/lang/translations/vi.po +16 -0
  62. package/lang/translations/zh-cn.po +16 -0
  63. package/lang/translations/zh.po +16 -0
  64. package/package.json +3 -3
  65. package/src/augmentation.d.ts +86 -86
  66. package/src/augmentation.js +5 -5
  67. package/src/autocomplete/autocompleteview.d.ts +81 -0
  68. package/src/autocomplete/autocompleteview.js +146 -0
  69. package/src/bindings/addkeyboardhandlingforgrid.d.ts +27 -27
  70. package/src/bindings/addkeyboardhandlingforgrid.js +107 -107
  71. package/src/bindings/clickoutsidehandler.d.ts +28 -28
  72. package/src/bindings/clickoutsidehandler.js +36 -36
  73. package/src/bindings/csstransitiondisablermixin.d.ts +40 -40
  74. package/src/bindings/csstransitiondisablermixin.js +55 -55
  75. package/src/bindings/injectcsstransitiondisabler.d.ts +59 -59
  76. package/src/bindings/injectcsstransitiondisabler.js +71 -71
  77. package/src/bindings/preventdefault.d.ts +33 -33
  78. package/src/bindings/preventdefault.js +34 -34
  79. package/src/bindings/submithandler.d.ts +57 -57
  80. package/src/bindings/submithandler.js +47 -47
  81. package/src/button/button.d.ts +178 -178
  82. package/src/button/button.js +5 -5
  83. package/src/button/buttonlabel.d.ts +34 -0
  84. package/src/button/buttonlabel.js +5 -0
  85. package/src/button/buttonlabelview.d.ts +31 -0
  86. package/src/button/buttonlabelview.js +42 -0
  87. package/src/button/buttonview.d.ts +185 -177
  88. package/src/button/buttonview.js +219 -231
  89. package/src/button/switchbuttonview.d.ts +45 -45
  90. package/src/button/switchbuttonview.js +75 -75
  91. package/src/colorgrid/colorgridview.d.ts +132 -132
  92. package/src/colorgrid/colorgridview.js +124 -124
  93. package/src/colorgrid/colortileview.d.ts +28 -28
  94. package/src/colorgrid/colortileview.js +40 -40
  95. package/src/colorgrid/utils.d.ts +47 -47
  96. package/src/colorgrid/utils.js +84 -84
  97. package/src/colorpicker/colorpickerview.d.ts +137 -137
  98. package/src/colorpicker/colorpickerview.js +270 -270
  99. package/src/colorpicker/utils.d.ts +43 -43
  100. package/src/colorpicker/utils.js +99 -99
  101. package/src/colorselector/colorgridsfragmentview.d.ts +194 -194
  102. package/src/colorselector/colorgridsfragmentview.js +289 -289
  103. package/src/colorselector/colorpickerfragmentview.d.ts +128 -128
  104. package/src/colorselector/colorpickerfragmentview.js +205 -205
  105. package/src/colorselector/colorselectorview.d.ts +242 -242
  106. package/src/colorselector/colorselectorview.js +256 -256
  107. package/src/colorselector/documentcolorcollection.d.ts +70 -70
  108. package/src/colorselector/documentcolorcollection.js +42 -42
  109. package/src/componentfactory.d.ts +81 -81
  110. package/src/componentfactory.js +104 -104
  111. package/src/dropdown/button/dropdownbutton.d.ts +25 -25
  112. package/src/dropdown/button/dropdownbutton.js +5 -5
  113. package/src/dropdown/button/dropdownbuttonview.d.ts +48 -48
  114. package/src/dropdown/button/dropdownbuttonview.js +66 -66
  115. package/src/dropdown/button/splitbuttonview.d.ts +161 -161
  116. package/src/dropdown/button/splitbuttonview.js +152 -152
  117. package/src/dropdown/dropdownpanelfocusable.d.ts +21 -21
  118. package/src/dropdown/dropdownpanelfocusable.js +5 -5
  119. package/src/dropdown/dropdownpanelview.d.ts +62 -62
  120. package/src/dropdown/dropdownpanelview.js +97 -97
  121. package/src/dropdown/dropdownview.d.ts +315 -315
  122. package/src/dropdown/dropdownview.js +379 -378
  123. package/src/dropdown/utils.d.ts +235 -221
  124. package/src/dropdown/utils.js +458 -437
  125. package/src/editableui/editableuiview.d.ts +72 -72
  126. package/src/editableui/editableuiview.js +112 -112
  127. package/src/editableui/inline/inlineeditableuiview.d.ts +40 -40
  128. package/src/editableui/inline/inlineeditableuiview.js +48 -48
  129. package/src/editorui/bodycollection.d.ts +55 -55
  130. package/src/editorui/bodycollection.js +84 -84
  131. package/src/editorui/boxed/boxededitoruiview.d.ts +40 -40
  132. package/src/editorui/boxed/boxededitoruiview.js +81 -81
  133. package/src/editorui/editorui.d.ts +282 -282
  134. package/src/editorui/editorui.js +410 -410
  135. package/src/editorui/editoruiview.d.ts +39 -39
  136. package/src/editorui/editoruiview.js +38 -38
  137. package/src/editorui/poweredby.d.ts +71 -71
  138. package/src/editorui/poweredby.js +276 -299
  139. package/src/focuscycler.d.ts +226 -183
  140. package/src/focuscycler.js +245 -220
  141. package/src/formheader/formheaderview.d.ts +59 -53
  142. package/src/formheader/formheaderview.js +69 -63
  143. package/src/highlightedtext/highlightedtextview.d.ts +38 -0
  144. package/src/highlightedtext/highlightedtextview.js +102 -0
  145. package/src/icon/iconview.d.ts +85 -78
  146. package/src/icon/iconview.js +114 -112
  147. package/src/iframe/iframeview.d.ts +50 -50
  148. package/src/iframe/iframeview.js +63 -63
  149. package/src/index.d.ts +73 -63
  150. package/src/index.js +70 -62
  151. package/src/input/inputbase.d.ts +107 -0
  152. package/src/input/inputbase.js +110 -0
  153. package/src/input/inputview.d.ts +36 -121
  154. package/src/input/inputview.js +24 -106
  155. package/src/inputnumber/inputnumberview.d.ts +49 -49
  156. package/src/inputnumber/inputnumberview.js +40 -40
  157. package/src/inputtext/inputtextview.d.ts +18 -18
  158. package/src/inputtext/inputtextview.js +27 -27
  159. package/src/label/labelview.d.ts +36 -36
  160. package/src/label/labelview.js +41 -41
  161. package/src/labeledfield/labeledfieldview.d.ts +187 -182
  162. package/src/labeledfield/labeledfieldview.js +157 -157
  163. package/src/labeledfield/utils.d.ts +123 -93
  164. package/src/labeledfield/utils.js +176 -131
  165. package/src/labeledinput/labeledinputview.d.ts +125 -125
  166. package/src/labeledinput/labeledinputview.js +125 -125
  167. package/src/list/listitemgroupview.d.ts +51 -0
  168. package/src/list/listitemgroupview.js +75 -0
  169. package/src/list/listitemview.d.ts +36 -35
  170. package/src/list/listitemview.js +42 -40
  171. package/src/list/listseparatorview.d.ts +18 -18
  172. package/src/list/listseparatorview.js +28 -28
  173. package/src/list/listview.d.ts +122 -65
  174. package/src/list/listview.js +187 -90
  175. package/src/model.d.ts +22 -22
  176. package/src/model.js +31 -31
  177. package/src/notification/notification.d.ts +211 -211
  178. package/src/notification/notification.js +187 -187
  179. package/src/panel/balloon/balloonpanelview.d.ts +685 -685
  180. package/src/panel/balloon/balloonpanelview.js +1010 -988
  181. package/src/panel/balloon/contextualballoon.d.ts +299 -299
  182. package/src/panel/balloon/contextualballoon.js +572 -572
  183. package/src/panel/sticky/stickypanelview.d.ts +156 -158
  184. package/src/panel/sticky/stickypanelview.js +234 -231
  185. package/src/search/filteredview.d.ts +31 -0
  186. package/src/search/filteredview.js +5 -0
  187. package/src/search/searchinfoview.d.ts +45 -0
  188. package/src/search/searchinfoview.js +59 -0
  189. package/src/search/searchresultsview.d.ts +54 -0
  190. package/src/search/searchresultsview.js +65 -0
  191. package/src/search/text/searchtextqueryview.d.ts +76 -0
  192. package/src/search/text/searchtextqueryview.js +75 -0
  193. package/src/search/text/searchtextview.d.ts +219 -0
  194. package/src/search/text/searchtextview.js +201 -0
  195. package/src/spinner/spinnerview.d.ts +25 -0
  196. package/src/spinner/spinnerview.js +38 -0
  197. package/src/template.d.ts +942 -942
  198. package/src/template.js +1294 -1294
  199. package/src/textarea/textareaview.d.ts +88 -0
  200. package/src/textarea/textareaview.js +140 -0
  201. package/src/toolbar/balloon/balloontoolbar.d.ts +122 -122
  202. package/src/toolbar/balloon/balloontoolbar.js +300 -300
  203. package/src/toolbar/block/blockbuttonview.d.ts +35 -35
  204. package/src/toolbar/block/blockbuttonview.js +41 -41
  205. package/src/toolbar/block/blocktoolbar.d.ts +161 -161
  206. package/src/toolbar/block/blocktoolbar.js +395 -391
  207. package/src/toolbar/normalizetoolbarconfig.d.ts +40 -39
  208. package/src/toolbar/normalizetoolbarconfig.js +51 -51
  209. package/src/toolbar/toolbarlinebreakview.d.ts +18 -18
  210. package/src/toolbar/toolbarlinebreakview.js +28 -28
  211. package/src/toolbar/toolbarseparatorview.d.ts +18 -18
  212. package/src/toolbar/toolbarseparatorview.js +28 -28
  213. package/src/toolbar/toolbarview.d.ts +266 -265
  214. package/src/toolbar/toolbarview.js +719 -717
  215. package/src/tooltipmanager.d.ts +180 -180
  216. package/src/tooltipmanager.js +353 -353
  217. package/src/view.d.ts +422 -422
  218. package/src/view.js +396 -396
  219. package/src/viewcollection.d.ts +139 -139
  220. package/src/viewcollection.js +206 -206
  221. package/theme/components/autocomplete/autocomplete.css +22 -0
  222. package/theme/components/formheader/formheader.css +8 -0
  223. package/theme/components/highlightedtext/highlightedtext.css +12 -0
  224. package/theme/components/search/search.css +43 -0
  225. package/theme/components/spinner/spinner.css +23 -0
  226. package/theme/components/textarea/textarea.css +10 -0
@@ -1,378 +1,379 @@
1
- /**
2
- * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.
3
- * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
4
- */
5
- /**
6
- * @module ui/dropdown/dropdownview
7
- */
8
- import View from '../view';
9
- import { KeystrokeHandler, FocusTracker, getOptimalPosition } from '@ckeditor/ckeditor5-utils';
10
- import '../../theme/components/dropdown/dropdown.css';
11
- /**
12
- * The dropdown view class. It manages the dropdown button and dropdown panel.
13
- *
14
- * In most cases, the easiest way to create a dropdown is by using the {@link module:ui/dropdown/utils~createDropdown}
15
- * util:
16
- *
17
- * ```ts
18
- * const dropdown = createDropdown( locale );
19
- *
20
- * // Configure dropdown's button properties:
21
- * dropdown.buttonView.set( {
22
- * label: 'A dropdown',
23
- * withText: true
24
- * } );
25
- *
26
- * dropdown.render();
27
- *
28
- * dropdown.panelView.element.textContent = 'Content of the panel';
29
- *
30
- * // Will render a dropdown with a panel containing a "Content of the panel" text.
31
- * document.body.appendChild( dropdown.element );
32
- * ```
33
- *
34
- * If you want to add a richer content to the dropdown panel, you can use the {@link module:ui/dropdown/utils~addListToDropdown}
35
- * and {@link module:ui/dropdown/utils~addToolbarToDropdown} helpers. See more examples in
36
- * {@link module:ui/dropdown/utils~createDropdown} documentation.
37
- *
38
- * If you want to create a completely custom dropdown, then you can compose it manually:
39
- *
40
- * ```ts
41
- * const button = new DropdownButtonView( locale );
42
- * const panel = new DropdownPanelView( locale );
43
- * const dropdown = new DropdownView( locale, button, panel );
44
- *
45
- * button.set( {
46
- * label: 'A dropdown',
47
- * withText: true
48
- * } );
49
- *
50
- * dropdown.render();
51
- *
52
- * panel.element.textContent = 'Content of the panel';
53
- *
54
- * // Will render a dropdown with a panel containing a "Content of the panel" text.
55
- * document.body.appendChild( dropdown.element );
56
- * ```
57
- *
58
- * However, dropdown created this way will contain little behavior. You will need to implement handlers for actions
59
- * such as {@link module:ui/bindings/clickoutsidehandler~clickOutsideHandler clicking outside an open dropdown}
60
- * (which should close it) and support for arrow keys inside the panel. Therefore, unless you really know what
61
- * you do and you really need to do it, it is recommended to use the {@link module:ui/dropdown/utils~createDropdown} helper.
62
- */
63
- export default class DropdownView extends View {
64
- /**
65
- * Creates an instance of the dropdown.
66
- *
67
- * Also see {@link #render}.
68
- *
69
- * @param locale The localization services instance.
70
- */
71
- constructor(locale, buttonView, panelView) {
72
- super(locale);
73
- const bind = this.bindTemplate;
74
- this.buttonView = buttonView;
75
- this.panelView = panelView;
76
- this.set('isOpen', false);
77
- this.set('isEnabled', true);
78
- this.set('class', undefined);
79
- this.set('id', undefined);
80
- this.set('panelPosition', 'auto');
81
- this.keystrokes = new KeystrokeHandler();
82
- this.focusTracker = new FocusTracker();
83
- this.setTemplate({
84
- tag: 'div',
85
- attributes: {
86
- class: [
87
- 'ck',
88
- 'ck-dropdown',
89
- bind.to('class'),
90
- bind.if('isEnabled', 'ck-disabled', value => !value)
91
- ],
92
- id: bind.to('id'),
93
- 'aria-describedby': bind.to('ariaDescribedById')
94
- },
95
- children: [
96
- buttonView,
97
- panelView
98
- ]
99
- });
100
- buttonView.extendTemplate({
101
- attributes: {
102
- class: [
103
- 'ck-dropdown__button'
104
- ],
105
- 'data-cke-tooltip-disabled': bind.to('isOpen')
106
- }
107
- });
108
- }
109
- /**
110
- * @inheritDoc
111
- */
112
- render() {
113
- super.render();
114
- this.focusTracker.add(this.buttonView.element);
115
- this.focusTracker.add(this.panelView.element);
116
- // Toggle the dropdown when its button has been clicked.
117
- this.listenTo(this.buttonView, 'open', () => {
118
- this.isOpen = !this.isOpen;
119
- });
120
- // Toggle the visibility of the panel when the dropdown becomes open.
121
- this.panelView.bind('isVisible').to(this, 'isOpen');
122
- // Let the dropdown control the position of the panel. The position must
123
- // be updated every time the dropdown is open.
124
- this.on('change:isOpen', (evt, name, isOpen) => {
125
- if (!isOpen) {
126
- return;
127
- }
128
- // If "auto", find the best position of the panel to fit into the viewport.
129
- // Otherwise, simply assign the static position.
130
- if (this.panelPosition === 'auto') {
131
- this.panelView.position = DropdownView._getOptimalPosition({
132
- element: this.panelView.element,
133
- target: this.buttonView.element,
134
- fitInViewport: true,
135
- positions: this._panelPositions
136
- }).name;
137
- }
138
- else {
139
- this.panelView.position = this.panelPosition;
140
- }
141
- });
142
- // Listen for keystrokes coming from within #element.
143
- this.keystrokes.listenTo(this.element);
144
- const closeDropdown = (data, cancel) => {
145
- if (this.isOpen) {
146
- this.isOpen = false;
147
- cancel();
148
- }
149
- };
150
- // Open the dropdown panel using the arrow down key, just like with return or space.
151
- this.keystrokes.set('arrowdown', (data, cancel) => {
152
- // Don't open if the dropdown is disabled or already open.
153
- if (this.buttonView.isEnabled && !this.isOpen) {
154
- this.isOpen = true;
155
- cancel();
156
- }
157
- });
158
- // Block the right arrow key (until nested dropdowns are implemented).
159
- this.keystrokes.set('arrowright', (data, cancel) => {
160
- if (this.isOpen) {
161
- cancel();
162
- }
163
- });
164
- // Close the dropdown using the arrow left/escape key.
165
- this.keystrokes.set('arrowleft', closeDropdown);
166
- this.keystrokes.set('esc', closeDropdown);
167
- }
168
- /**
169
- * Focuses the {@link #buttonView}.
170
- */
171
- focus() {
172
- this.buttonView.focus();
173
- }
174
- /**
175
- * Returns {@link #panelView panel} positions to be used by the
176
- * {@link module:utils/dom/position~getOptimalPosition `getOptimalPosition()`}
177
- * utility considering the direction of the language the UI of the editor is displayed in.
178
- */
179
- get _panelPositions() {
180
- const { south, north, southEast, southWest, northEast, northWest, southMiddleEast, southMiddleWest, northMiddleEast, northMiddleWest } = DropdownView.defaultPanelPositions;
181
- if (this.locale.uiLanguageDirection !== 'rtl') {
182
- return [
183
- southEast, southWest, southMiddleEast, southMiddleWest, south,
184
- northEast, northWest, northMiddleEast, northMiddleWest, north
185
- ];
186
- }
187
- else {
188
- return [
189
- southWest, southEast, southMiddleWest, southMiddleEast, south,
190
- northWest, northEast, northMiddleWest, northMiddleEast, north
191
- ];
192
- }
193
- }
194
- }
195
- /**
196
- * A set of positioning functions used by the dropdown view to determine
197
- * the optimal position (i.e. fitting into the browser viewport) of its
198
- * {@link module:ui/dropdown/dropdownview~DropdownView#panelView panel} when
199
- * {@link module:ui/dropdown/dropdownview~DropdownView#panelPosition} is set to 'auto'`.
200
- *
201
- * The available positioning functions are as follow:
202
- *
203
- * **South**
204
- *
205
- * * `south`
206
- *
207
- * ```
208
- * [ Button ]
209
- * +-----------------+
210
- * | Panel |
211
- * +-----------------+
212
- * ```
213
- *
214
- * * `southEast`
215
- *
216
- * ```
217
- * [ Button ]
218
- * +-----------------+
219
- * | Panel |
220
- * +-----------------+
221
- * ```
222
- *
223
- * * `southWest`
224
- *
225
- * ```
226
- * [ Button ]
227
- * +-----------------+
228
- * | Panel |
229
- * +-----------------+
230
- * ```
231
- *
232
- * * `southMiddleEast`
233
- *
234
- * ```
235
- * [ Button ]
236
- * +-----------------+
237
- * | Panel |
238
- * +-----------------+
239
- * ```
240
- *
241
- * * `southMiddleWest`
242
- *
243
- * ```
244
- * [ Button ]
245
- * +-----------------+
246
- * | Panel |
247
- * +-----------------+
248
- * ```
249
- *
250
- * **North**
251
- *
252
- * * `north`
253
- *
254
- * ```
255
- * +-----------------+
256
- * | Panel |
257
- * +-----------------+
258
- * [ Button ]
259
- * ```
260
- *
261
- * * `northEast`
262
- *
263
- * ```
264
- * +-----------------+
265
- * | Panel |
266
- * +-----------------+
267
- * [ Button ]
268
- * ```
269
- *
270
- * * `northWest`
271
- *
272
- * ```
273
- * +-----------------+
274
- * | Panel |
275
- * +-----------------+
276
- * [ Button ]
277
- * ```
278
- *
279
- * * `northMiddleEast`
280
- *
281
- * ```
282
- * +-----------------+
283
- * | Panel |
284
- * +-----------------+
285
- * [ Button ]
286
- * ```
287
- *
288
- * * `northMiddleWest`
289
- *
290
- * ```
291
- * +-----------------+
292
- * | Panel |
293
- * +-----------------+
294
- * [ Button ]
295
- * ```
296
- *
297
- * Positioning functions are compatible with {@link module:utils/dom/position~Position}.
298
- *
299
- * The name that position function returns will be reflected in dropdown panel's class that
300
- * controls its placement. See {@link module:ui/dropdown/dropdownview~DropdownView#panelPosition}
301
- * to learn more.
302
- */
303
- DropdownView.defaultPanelPositions = {
304
- south: (buttonRect, panelRect) => {
305
- return {
306
- top: buttonRect.bottom,
307
- left: buttonRect.left - (panelRect.width - buttonRect.width) / 2,
308
- name: 's'
309
- };
310
- },
311
- southEast: buttonRect => {
312
- return {
313
- top: buttonRect.bottom,
314
- left: buttonRect.left,
315
- name: 'se'
316
- };
317
- },
318
- southWest: (buttonRect, panelRect) => {
319
- return {
320
- top: buttonRect.bottom,
321
- left: buttonRect.left - panelRect.width + buttonRect.width,
322
- name: 'sw'
323
- };
324
- },
325
- southMiddleEast: (buttonRect, panelRect) => {
326
- return {
327
- top: buttonRect.bottom,
328
- left: buttonRect.left - (panelRect.width - buttonRect.width) / 4,
329
- name: 'sme'
330
- };
331
- },
332
- southMiddleWest: (buttonRect, panelRect) => {
333
- return {
334
- top: buttonRect.bottom,
335
- left: buttonRect.left - (panelRect.width - buttonRect.width) * 3 / 4,
336
- name: 'smw'
337
- };
338
- },
339
- north: (buttonRect, panelRect) => {
340
- return {
341
- top: buttonRect.top - panelRect.height,
342
- left: buttonRect.left - (panelRect.width - buttonRect.width) / 2,
343
- name: 'n'
344
- };
345
- },
346
- northEast: (buttonRect, panelRect) => {
347
- return {
348
- top: buttonRect.top - panelRect.height,
349
- left: buttonRect.left,
350
- name: 'ne'
351
- };
352
- },
353
- northWest: (buttonRect, panelRect) => {
354
- return {
355
- top: buttonRect.top - panelRect.height,
356
- left: buttonRect.left - panelRect.width + buttonRect.width,
357
- name: 'nw'
358
- };
359
- },
360
- northMiddleEast: (buttonRect, panelRect) => {
361
- return {
362
- top: buttonRect.top - panelRect.height,
363
- left: buttonRect.left - (panelRect.width - buttonRect.width) / 4,
364
- name: 'nme'
365
- };
366
- },
367
- northMiddleWest: (buttonRect, panelRect) => {
368
- return {
369
- top: buttonRect.top - panelRect.height,
370
- left: buttonRect.left - (panelRect.width - buttonRect.width) * 3 / 4,
371
- name: 'nmw'
372
- };
373
- }
374
- };
375
- /**
376
- * A function used to calculate the optimal position for the dropdown panel.
377
- */
378
- DropdownView._getOptimalPosition = getOptimalPosition;
1
+ /**
2
+ * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.
3
+ * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
4
+ */
5
+ /**
6
+ * @module ui/dropdown/dropdownview
7
+ */
8
+ import View from '../view';
9
+ import { KeystrokeHandler, FocusTracker, getOptimalPosition } from '@ckeditor/ckeditor5-utils';
10
+ import '../../theme/components/dropdown/dropdown.css';
11
+ /**
12
+ * The dropdown view class. It manages the dropdown button and dropdown panel.
13
+ *
14
+ * In most cases, the easiest way to create a dropdown is by using the {@link module:ui/dropdown/utils~createDropdown}
15
+ * util:
16
+ *
17
+ * ```ts
18
+ * const dropdown = createDropdown( locale );
19
+ *
20
+ * // Configure dropdown's button properties:
21
+ * dropdown.buttonView.set( {
22
+ * label: 'A dropdown',
23
+ * withText: true
24
+ * } );
25
+ *
26
+ * dropdown.render();
27
+ *
28
+ * dropdown.panelView.element.textContent = 'Content of the panel';
29
+ *
30
+ * // Will render a dropdown with a panel containing a "Content of the panel" text.
31
+ * document.body.appendChild( dropdown.element );
32
+ * ```
33
+ *
34
+ * If you want to add a richer content to the dropdown panel, you can use the {@link module:ui/dropdown/utils~addListToDropdown}
35
+ * and {@link module:ui/dropdown/utils~addToolbarToDropdown} helpers. See more examples in
36
+ * {@link module:ui/dropdown/utils~createDropdown} documentation.
37
+ *
38
+ * If you want to create a completely custom dropdown, then you can compose it manually:
39
+ *
40
+ * ```ts
41
+ * const button = new DropdownButtonView( locale );
42
+ * const panel = new DropdownPanelView( locale );
43
+ * const dropdown = new DropdownView( locale, button, panel );
44
+ *
45
+ * button.set( {
46
+ * label: 'A dropdown',
47
+ * withText: true
48
+ * } );
49
+ *
50
+ * dropdown.render();
51
+ *
52
+ * panel.element.textContent = 'Content of the panel';
53
+ *
54
+ * // Will render a dropdown with a panel containing a "Content of the panel" text.
55
+ * document.body.appendChild( dropdown.element );
56
+ * ```
57
+ *
58
+ * However, dropdown created this way will contain little behavior. You will need to implement handlers for actions
59
+ * such as {@link module:ui/bindings/clickoutsidehandler~clickOutsideHandler clicking outside an open dropdown}
60
+ * (which should close it) and support for arrow keys inside the panel. Therefore, unless you really know what
61
+ * you do and you really need to do it, it is recommended to use the {@link module:ui/dropdown/utils~createDropdown} helper.
62
+ */
63
+ export default class DropdownView extends View {
64
+ /**
65
+ * Creates an instance of the dropdown.
66
+ *
67
+ * Also see {@link #render}.
68
+ *
69
+ * @param locale The localization services instance.
70
+ */
71
+ constructor(locale, buttonView, panelView) {
72
+ super(locale);
73
+ const bind = this.bindTemplate;
74
+ this.buttonView = buttonView;
75
+ this.panelView = panelView;
76
+ this.set('isOpen', false);
77
+ this.set('isEnabled', true);
78
+ this.set('class', undefined);
79
+ this.set('id', undefined);
80
+ this.set('panelPosition', 'auto');
81
+ this.keystrokes = new KeystrokeHandler();
82
+ this.focusTracker = new FocusTracker();
83
+ this.setTemplate({
84
+ tag: 'div',
85
+ attributes: {
86
+ class: [
87
+ 'ck',
88
+ 'ck-dropdown',
89
+ bind.to('class'),
90
+ bind.if('isEnabled', 'ck-disabled', value => !value)
91
+ ],
92
+ id: bind.to('id'),
93
+ 'aria-describedby': bind.to('ariaDescribedById')
94
+ },
95
+ children: [
96
+ buttonView,
97
+ panelView
98
+ ]
99
+ });
100
+ buttonView.extendTemplate({
101
+ attributes: {
102
+ class: [
103
+ 'ck-dropdown__button'
104
+ ],
105
+ 'data-cke-tooltip-disabled': bind.to('isOpen')
106
+ }
107
+ });
108
+ }
109
+ /**
110
+ * @inheritDoc
111
+ */
112
+ render() {
113
+ super.render();
114
+ this.focusTracker.add(this.buttonView.element);
115
+ this.focusTracker.add(this.panelView.element);
116
+ // Toggle the dropdown when its button has been clicked.
117
+ this.listenTo(this.buttonView, 'open', () => {
118
+ this.isOpen = !this.isOpen;
119
+ });
120
+ // Toggle the visibility of the panel when the dropdown becomes open.
121
+ this.panelView.bind('isVisible').to(this, 'isOpen');
122
+ // Let the dropdown control the position of the panel. The position must
123
+ // be updated every time the dropdown is open.
124
+ this.on('change:isOpen', (evt, name, isOpen) => {
125
+ if (!isOpen) {
126
+ return;
127
+ }
128
+ // If "auto", find the best position of the panel to fit into the viewport.
129
+ // Otherwise, simply assign the static position.
130
+ if (this.panelPosition === 'auto') {
131
+ const optimalPanelPosition = DropdownView._getOptimalPosition({
132
+ element: this.panelView.element,
133
+ target: this.buttonView.element,
134
+ fitInViewport: true,
135
+ positions: this._panelPositions
136
+ });
137
+ this.panelView.position = (optimalPanelPosition ? optimalPanelPosition.name : this._panelPositions[0].name);
138
+ }
139
+ else {
140
+ this.panelView.position = this.panelPosition;
141
+ }
142
+ });
143
+ // Listen for keystrokes coming from within #element.
144
+ this.keystrokes.listenTo(this.element);
145
+ const closeDropdown = (data, cancel) => {
146
+ if (this.isOpen) {
147
+ this.isOpen = false;
148
+ cancel();
149
+ }
150
+ };
151
+ // Open the dropdown panel using the arrow down key, just like with return or space.
152
+ this.keystrokes.set('arrowdown', (data, cancel) => {
153
+ // Don't open if the dropdown is disabled or already open.
154
+ if (this.buttonView.isEnabled && !this.isOpen) {
155
+ this.isOpen = true;
156
+ cancel();
157
+ }
158
+ });
159
+ // Block the right arrow key (until nested dropdowns are implemented).
160
+ this.keystrokes.set('arrowright', (data, cancel) => {
161
+ if (this.isOpen) {
162
+ cancel();
163
+ }
164
+ });
165
+ // Close the dropdown using the arrow left/escape key.
166
+ this.keystrokes.set('arrowleft', closeDropdown);
167
+ this.keystrokes.set('esc', closeDropdown);
168
+ }
169
+ /**
170
+ * Focuses the {@link #buttonView}.
171
+ */
172
+ focus() {
173
+ this.buttonView.focus();
174
+ }
175
+ /**
176
+ * Returns {@link #panelView panel} positions to be used by the
177
+ * {@link module:utils/dom/position~getOptimalPosition `getOptimalPosition()`}
178
+ * utility considering the direction of the language the UI of the editor is displayed in.
179
+ */
180
+ get _panelPositions() {
181
+ const { south, north, southEast, southWest, northEast, northWest, southMiddleEast, southMiddleWest, northMiddleEast, northMiddleWest } = DropdownView.defaultPanelPositions;
182
+ if (this.locale.uiLanguageDirection !== 'rtl') {
183
+ return [
184
+ southEast, southWest, southMiddleEast, southMiddleWest, south,
185
+ northEast, northWest, northMiddleEast, northMiddleWest, north
186
+ ];
187
+ }
188
+ else {
189
+ return [
190
+ southWest, southEast, southMiddleWest, southMiddleEast, south,
191
+ northWest, northEast, northMiddleWest, northMiddleEast, north
192
+ ];
193
+ }
194
+ }
195
+ }
196
+ /**
197
+ * A set of positioning functions used by the dropdown view to determine
198
+ * the optimal position (i.e. fitting into the browser viewport) of its
199
+ * {@link module:ui/dropdown/dropdownview~DropdownView#panelView panel} when
200
+ * {@link module:ui/dropdown/dropdownview~DropdownView#panelPosition} is set to 'auto'`.
201
+ *
202
+ * The available positioning functions are as follow:
203
+ *
204
+ * **South**
205
+ *
206
+ * * `south`
207
+ *
208
+ * ```
209
+ * [ Button ]
210
+ * +-----------------+
211
+ * | Panel |
212
+ * +-----------------+
213
+ * ```
214
+ *
215
+ * * `southEast`
216
+ *
217
+ * ```
218
+ * [ Button ]
219
+ * +-----------------+
220
+ * | Panel |
221
+ * +-----------------+
222
+ * ```
223
+ *
224
+ * * `southWest`
225
+ *
226
+ * ```
227
+ * [ Button ]
228
+ * +-----------------+
229
+ * | Panel |
230
+ * +-----------------+
231
+ * ```
232
+ *
233
+ * * `southMiddleEast`
234
+ *
235
+ * ```
236
+ * [ Button ]
237
+ * +-----------------+
238
+ * | Panel |
239
+ * +-----------------+
240
+ * ```
241
+ *
242
+ * * `southMiddleWest`
243
+ *
244
+ * ```
245
+ * [ Button ]
246
+ * +-----------------+
247
+ * | Panel |
248
+ * +-----------------+
249
+ * ```
250
+ *
251
+ * **North**
252
+ *
253
+ * * `north`
254
+ *
255
+ * ```
256
+ * +-----------------+
257
+ * | Panel |
258
+ * +-----------------+
259
+ * [ Button ]
260
+ * ```
261
+ *
262
+ * * `northEast`
263
+ *
264
+ * ```
265
+ * +-----------------+
266
+ * | Panel |
267
+ * +-----------------+
268
+ * [ Button ]
269
+ * ```
270
+ *
271
+ * * `northWest`
272
+ *
273
+ * ```
274
+ * +-----------------+
275
+ * | Panel |
276
+ * +-----------------+
277
+ * [ Button ]
278
+ * ```
279
+ *
280
+ * * `northMiddleEast`
281
+ *
282
+ * ```
283
+ * +-----------------+
284
+ * | Panel |
285
+ * +-----------------+
286
+ * [ Button ]
287
+ * ```
288
+ *
289
+ * * `northMiddleWest`
290
+ *
291
+ * ```
292
+ * +-----------------+
293
+ * | Panel |
294
+ * +-----------------+
295
+ * [ Button ]
296
+ * ```
297
+ *
298
+ * Positioning functions are compatible with {@link module:utils/dom/position~Position}.
299
+ *
300
+ * The name that position function returns will be reflected in dropdown panel's class that
301
+ * controls its placement. See {@link module:ui/dropdown/dropdownview~DropdownView#panelPosition}
302
+ * to learn more.
303
+ */
304
+ DropdownView.defaultPanelPositions = {
305
+ south: (buttonRect, panelRect) => {
306
+ return {
307
+ top: buttonRect.bottom,
308
+ left: buttonRect.left - (panelRect.width - buttonRect.width) / 2,
309
+ name: 's'
310
+ };
311
+ },
312
+ southEast: buttonRect => {
313
+ return {
314
+ top: buttonRect.bottom,
315
+ left: buttonRect.left,
316
+ name: 'se'
317
+ };
318
+ },
319
+ southWest: (buttonRect, panelRect) => {
320
+ return {
321
+ top: buttonRect.bottom,
322
+ left: buttonRect.left - panelRect.width + buttonRect.width,
323
+ name: 'sw'
324
+ };
325
+ },
326
+ southMiddleEast: (buttonRect, panelRect) => {
327
+ return {
328
+ top: buttonRect.bottom,
329
+ left: buttonRect.left - (panelRect.width - buttonRect.width) / 4,
330
+ name: 'sme'
331
+ };
332
+ },
333
+ southMiddleWest: (buttonRect, panelRect) => {
334
+ return {
335
+ top: buttonRect.bottom,
336
+ left: buttonRect.left - (panelRect.width - buttonRect.width) * 3 / 4,
337
+ name: 'smw'
338
+ };
339
+ },
340
+ north: (buttonRect, panelRect) => {
341
+ return {
342
+ top: buttonRect.top - panelRect.height,
343
+ left: buttonRect.left - (panelRect.width - buttonRect.width) / 2,
344
+ name: 'n'
345
+ };
346
+ },
347
+ northEast: (buttonRect, panelRect) => {
348
+ return {
349
+ top: buttonRect.top - panelRect.height,
350
+ left: buttonRect.left,
351
+ name: 'ne'
352
+ };
353
+ },
354
+ northWest: (buttonRect, panelRect) => {
355
+ return {
356
+ top: buttonRect.top - panelRect.height,
357
+ left: buttonRect.left - panelRect.width + buttonRect.width,
358
+ name: 'nw'
359
+ };
360
+ },
361
+ northMiddleEast: (buttonRect, panelRect) => {
362
+ return {
363
+ top: buttonRect.top - panelRect.height,
364
+ left: buttonRect.left - (panelRect.width - buttonRect.width) / 4,
365
+ name: 'nme'
366
+ };
367
+ },
368
+ northMiddleWest: (buttonRect, panelRect) => {
369
+ return {
370
+ top: buttonRect.top - panelRect.height,
371
+ left: buttonRect.left - (panelRect.width - buttonRect.width) * 3 / 4,
372
+ name: 'nmw'
373
+ };
374
+ }
375
+ };
376
+ /**
377
+ * A function used to calculate the optimal position for the dropdown panel.
378
+ */
379
+ DropdownView._getOptimalPosition = getOptimalPosition;