@ckeditor/ckeditor5-ui 47.6.1 → 48.0.0-alpha.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/LICENSE.md +1 -1
- package/ckeditor5-metadata.json +3 -3
- package/{src → dist}/colorselector/documentcolorcollection.d.ts +4 -4
- package/{src → dist}/dialog/dialog.d.ts +1 -1
- package/dist/index-editor.css +2728 -587
- package/dist/index.css +2673 -850
- package/dist/index.css.map +1 -1
- package/dist/index.js +11 -1
- package/dist/index.js.map +1 -1
- package/{src → dist}/menubar/menubarmenuview.d.ts +4 -0
- package/{src → dist}/view.d.ts +1 -1
- package/package.json +25 -47
- package/lang/contexts.json +0 -51
- package/lang/translations/af.po +0 -208
- package/lang/translations/ar.po +0 -208
- package/lang/translations/ast.po +0 -208
- package/lang/translations/az.po +0 -208
- package/lang/translations/be.po +0 -208
- package/lang/translations/bg.po +0 -208
- package/lang/translations/bn.po +0 -208
- package/lang/translations/bs.po +0 -208
- package/lang/translations/ca.po +0 -208
- package/lang/translations/cs.po +0 -208
- package/lang/translations/da.po +0 -208
- package/lang/translations/de-ch.po +0 -208
- package/lang/translations/de.po +0 -208
- package/lang/translations/el.po +0 -208
- package/lang/translations/en-au.po +0 -208
- package/lang/translations/en-gb.po +0 -208
- package/lang/translations/en.po +0 -208
- package/lang/translations/eo.po +0 -208
- package/lang/translations/es-co.po +0 -208
- package/lang/translations/es.po +0 -208
- package/lang/translations/et.po +0 -208
- package/lang/translations/eu.po +0 -208
- package/lang/translations/fa.po +0 -208
- package/lang/translations/fi.po +0 -208
- package/lang/translations/fr.po +0 -208
- package/lang/translations/gl.po +0 -208
- package/lang/translations/gu.po +0 -208
- package/lang/translations/he.po +0 -208
- package/lang/translations/hi.po +0 -208
- package/lang/translations/hr.po +0 -208
- package/lang/translations/hu.po +0 -208
- package/lang/translations/hy.po +0 -208
- package/lang/translations/id.po +0 -208
- package/lang/translations/it.po +0 -208
- package/lang/translations/ja.po +0 -208
- package/lang/translations/jv.po +0 -208
- package/lang/translations/kk.po +0 -208
- package/lang/translations/km.po +0 -208
- package/lang/translations/kn.po +0 -208
- package/lang/translations/ko.po +0 -208
- package/lang/translations/ku.po +0 -208
- package/lang/translations/lt.po +0 -208
- package/lang/translations/lv.po +0 -208
- package/lang/translations/ms.po +0 -208
- package/lang/translations/nb.po +0 -208
- package/lang/translations/ne.po +0 -208
- package/lang/translations/nl.po +0 -208
- package/lang/translations/no.po +0 -208
- package/lang/translations/oc.po +0 -208
- package/lang/translations/pl.po +0 -208
- package/lang/translations/pt-br.po +0 -208
- package/lang/translations/pt.po +0 -208
- package/lang/translations/ro.po +0 -208
- package/lang/translations/ru.po +0 -208
- package/lang/translations/si.po +0 -208
- package/lang/translations/sk.po +0 -208
- package/lang/translations/sl.po +0 -208
- package/lang/translations/sq.po +0 -208
- package/lang/translations/sr-latn.po +0 -208
- package/lang/translations/sr.po +0 -208
- package/lang/translations/sv.po +0 -208
- package/lang/translations/th.po +0 -208
- package/lang/translations/ti.po +0 -208
- package/lang/translations/tk.po +0 -208
- package/lang/translations/tr.po +0 -208
- package/lang/translations/tt.po +0 -208
- package/lang/translations/ug.po +0 -208
- package/lang/translations/uk.po +0 -208
- package/lang/translations/ur.po +0 -208
- package/lang/translations/uz.po +0 -208
- package/lang/translations/vi.po +0 -208
- package/lang/translations/zh-cn.po +0 -208
- package/lang/translations/zh.po +0 -208
- package/src/arialiveannouncer.js +0 -189
- package/src/augmentation.js +0 -5
- package/src/autocomplete/autocompleteview.js +0 -157
- package/src/badge/badge.js +0 -226
- package/src/bindings/addkeyboardhandlingforgrid.js +0 -107
- package/src/bindings/clickoutsidehandler.js +0 -36
- package/src/bindings/csstransitiondisablermixin.js +0 -58
- package/src/bindings/draggableviewmixin.js +0 -144
- package/src/bindings/preventdefault.js +0 -35
- package/src/bindings/submithandler.js +0 -47
- package/src/button/button.js +0 -5
- package/src/button/buttonlabel.js +0 -5
- package/src/button/buttonlabelview.js +0 -42
- package/src/button/buttonview.js +0 -278
- package/src/button/filedialogbuttonview.js +0 -147
- package/src/button/listitembuttonview.js +0 -136
- package/src/button/switchbuttonview.js +0 -79
- package/src/collapsible/collapsibleview.js +0 -106
- package/src/colorgrid/colorgridview.js +0 -140
- package/src/colorgrid/colortileview.js +0 -42
- package/src/colorgrid/utils.js +0 -84
- package/src/colorpicker/colorpickerview.js +0 -356
- package/src/colorpicker/utils.js +0 -108
- package/src/colorselector/colorgridsfragmentview.js +0 -368
- package/src/colorselector/colorpickerfragmentview.js +0 -254
- package/src/colorselector/colorselectorview.js +0 -294
- package/src/colorselector/documentcolorcollection.js +0 -42
- package/src/componentfactory.js +0 -108
- package/src/dialog/dialog.js +0 -325
- package/src/dialog/dialogactionsview.js +0 -118
- package/src/dialog/dialogcontentview.js +0 -39
- package/src/dialog/dialogview.js +0 -507
- package/src/dropdown/button/dropdownbutton.js +0 -5
- package/src/dropdown/button/dropdownbuttonview.js +0 -70
- package/src/dropdown/button/splitbuttonview.js +0 -178
- package/src/dropdown/dropdownpanelfocusable.js +0 -5
- package/src/dropdown/dropdownpanelview.js +0 -106
- package/src/dropdown/dropdownview.js +0 -438
- package/src/dropdown/menu/dropdownmenubehaviors.js +0 -125
- package/src/dropdown/menu/dropdownmenubuttonview.js +0 -69
- package/src/dropdown/menu/dropdownmenulistitembuttonview.js +0 -30
- package/src/dropdown/menu/dropdownmenulistitemview.js +0 -38
- package/src/dropdown/menu/dropdownmenulistview.js +0 -29
- package/src/dropdown/menu/dropdownmenunestedmenupanelview.js +0 -63
- package/src/dropdown/menu/dropdownmenunestedmenuview.js +0 -214
- package/src/dropdown/menu/dropdownmenurootlistview.js +0 -168
- package/src/dropdown/menu/utils.js +0 -61
- package/src/dropdown/utils.js +0 -654
- package/src/editableui/editableuiview.js +0 -130
- package/src/editableui/inline/inlineeditableuiview.js +0 -75
- package/src/editorui/accessibilityhelp/accessibilityhelp.js +0 -142
- package/src/editorui/accessibilityhelp/accessibilityhelpcontentview.js +0 -112
- package/src/editorui/bodycollection.js +0 -128
- package/src/editorui/boxed/boxededitoruiview.js +0 -95
- package/src/editorui/editorui.js +0 -586
- package/src/editorui/editoruiview.js +0 -60
- package/src/editorui/evaluationbadge.js +0 -99
- package/src/editorui/poweredby.js +0 -120
- package/src/focuscycler.js +0 -383
- package/src/formheader/formheaderview.js +0 -77
- package/src/formrow/formrowview.js +0 -56
- package/src/highlightedtext/buttonlabelwithhighlightview.js +0 -31
- package/src/highlightedtext/highlightedtextview.js +0 -102
- package/src/highlightedtext/labelwithhighlightview.js +0 -37
- package/src/icon/iconview.js +0 -123
- package/src/iframe/iframeview.js +0 -63
- package/src/index.js +0 -132
- package/src/input/inputbase.js +0 -119
- package/src/input/inputview.js +0 -24
- package/src/inputnumber/inputnumberview.js +0 -40
- package/src/inputtext/inputtextview.js +0 -27
- package/src/label/labelview.js +0 -46
- package/src/labeledfield/labeledfieldview.js +0 -177
- package/src/labeledfield/utils.js +0 -176
- package/src/labeledinput/labeledinputview.js +0 -138
- package/src/legacyerrors.js +0 -20
- package/src/list/listitemgroupview.js +0 -82
- package/src/list/listitemview.js +0 -46
- package/src/list/listseparatorview.js +0 -28
- package/src/list/listview.js +0 -210
- package/src/menubar/menubarmenubuttonview.js +0 -68
- package/src/menubar/menubarmenulistitembuttonview.js +0 -30
- package/src/menubar/menubarmenulistitemfiledialogbuttonview.js +0 -32
- package/src/menubar/menubarmenulistitemview.js +0 -34
- package/src/menubar/menubarmenulistview.js +0 -72
- package/src/menubar/menubarmenupanelview.js +0 -64
- package/src/menubar/menubarmenuview.js +0 -198
- package/src/menubar/menubarview.js +0 -281
- package/src/menubar/utils.js +0 -1432
- package/src/model.js +0 -31
- package/src/notification/notification.js +0 -192
- package/src/panel/balloon/balloonpanelview.js +0 -1077
- package/src/panel/balloon/contextualballoon.js +0 -616
- package/src/panel/sticky/stickypanelview.js +0 -246
- package/src/search/filteredview.js +0 -5
- package/src/search/filtergroupanditemnames.js +0 -38
- package/src/search/searchinfoview.js +0 -59
- package/src/search/searchresultsview.js +0 -83
- package/src/search/text/searchtextqueryview.js +0 -87
- package/src/search/text/searchtextview.js +0 -242
- package/src/spinner/spinnerview.js +0 -38
- package/src/template.js +0 -1396
- package/src/textarea/textareaview.js +0 -189
- package/src/toolbar/balloon/balloontoolbar.js +0 -358
- package/src/toolbar/block/blockbuttonview.js +0 -41
- package/src/toolbar/block/blocktoolbar.js +0 -507
- package/src/toolbar/normalizetoolbarconfig.js +0 -52
- package/src/toolbar/toolbarlinebreakview.js +0 -28
- package/src/toolbar/toolbarseparatorview.js +0 -28
- package/src/toolbar/toolbarview.js +0 -873
- package/src/tooltipmanager.js +0 -454
- package/src/view.js +0 -471
- package/src/viewcollection.js +0 -210
- package/theme/components/arialiveannouncer/arialiveannouncer.css +0 -14
- package/theme/components/autocomplete/autocomplete.css +0 -22
- package/theme/components/button/button.css +0 -39
- package/theme/components/button/listitembutton.css +0 -48
- package/theme/components/button/switchbutton.css +0 -14
- package/theme/components/collapsible/collapsible.css +0 -10
- package/theme/components/colorgrid/colorgrid.css +0 -8
- package/theme/components/colorpicker/colorpicker.css +0 -34
- package/theme/components/colorselector/colorselector.css +0 -35
- package/theme/components/dialog/dialog.css +0 -42
- package/theme/components/dialog/dialogactions.css +0 -11
- package/theme/components/dropdown/dropdown.css +0 -95
- package/theme/components/dropdown/listdropdown.css +0 -10
- package/theme/components/dropdown/menu/dropdownmenu.css +0 -8
- package/theme/components/dropdown/menu/dropdownmenubutton.css +0 -9
- package/theme/components/dropdown/menu/dropdownmenulistitem.css +0 -10
- package/theme/components/dropdown/menu/dropdownmenulistitembutton.css +0 -10
- package/theme/components/dropdown/menu/dropdownmenupanel.css +0 -11
- package/theme/components/dropdown/splitbutton.css +0 -14
- package/theme/components/dropdown/toolbardropdown.css +0 -20
- package/theme/components/editorui/accessibilityhelp.css +0 -10
- package/theme/components/editorui/editorui.css +0 -10
- package/theme/components/form/form.css +0 -87
- package/theme/components/formheader/formheader.css +0 -18
- package/theme/components/formrow/formrow.css +0 -32
- package/theme/components/highlightedtext/highlightedtext.css +0 -12
- package/theme/components/icon/icon.css +0 -8
- package/theme/components/input/input.css +0 -10
- package/theme/components/label/label.css +0 -12
- package/theme/components/labeledfield/labeledfieldview.css +0 -16
- package/theme/components/labeledinput/labeledinput.css +0 -10
- package/theme/components/list/list.css +0 -26
- package/theme/components/menubar/menubar.css +0 -10
- package/theme/components/menubar/menubarmenu.css +0 -9
- package/theme/components/menubar/menubarmenubutton.css +0 -11
- package/theme/components/menubar/menubarmenulistitem.css +0 -10
- package/theme/components/menubar/menubarmenulistitembutton.css +0 -10
- package/theme/components/menubar/menubarmenupanel.css +0 -62
- package/theme/components/panel/balloonpanel.css +0 -56
- package/theme/components/panel/balloonrotator.css +0 -17
- package/theme/components/panel/fakepanel.css +0 -23
- package/theme/components/panel/stickypanel.css +0 -17
- package/theme/components/responsive-form/responsiveform.css +0 -42
- package/theme/components/search/search.css +0 -43
- package/theme/components/spinner/spinner.css +0 -23
- package/theme/components/textarea/textarea.css +0 -10
- package/theme/components/toolbar/blocktoolbar.css +0 -9
- package/theme/components/toolbar/toolbar.css +0 -58
- package/theme/components/tooltip/tooltip.css +0 -12
- package/theme/globals/_evaluationbadge.css +0 -54
- package/theme/globals/_hidden.css +0 -13
- package/theme/globals/_poweredby.css +0 -84
- package/theme/globals/_transition.css +0 -12
- package/theme/globals/_zindex.css +0 -10
- package/theme/globals/globals.css +0 -10
- package/theme/mixins/_dir.css +0 -10
- package/theme/mixins/_mediacolors.css +0 -20
- package/theme/mixins/_rwd.css +0 -10
- package/theme/mixins/_unselectable.css +0 -14
- /package/{src → dist}/arialiveannouncer.d.ts +0 -0
- /package/{src → dist}/augmentation.d.ts +0 -0
- /package/{src → dist}/autocomplete/autocompleteview.d.ts +0 -0
- /package/{src → dist}/badge/badge.d.ts +0 -0
- /package/{src → dist}/bindings/addkeyboardhandlingforgrid.d.ts +0 -0
- /package/{src → dist}/bindings/clickoutsidehandler.d.ts +0 -0
- /package/{src → dist}/bindings/csstransitiondisablermixin.d.ts +0 -0
- /package/{src → dist}/bindings/draggableviewmixin.d.ts +0 -0
- /package/{src → dist}/bindings/preventdefault.d.ts +0 -0
- /package/{src → dist}/bindings/submithandler.d.ts +0 -0
- /package/{src → dist}/button/button.d.ts +0 -0
- /package/{src → dist}/button/buttonlabel.d.ts +0 -0
- /package/{src → dist}/button/buttonlabelview.d.ts +0 -0
- /package/{src → dist}/button/buttonview.d.ts +0 -0
- /package/{src → dist}/button/filedialogbuttonview.d.ts +0 -0
- /package/{src → dist}/button/listitembuttonview.d.ts +0 -0
- /package/{src → dist}/button/switchbuttonview.d.ts +0 -0
- /package/{src → dist}/collapsible/collapsibleview.d.ts +0 -0
- /package/{src → dist}/colorgrid/colorgridview.d.ts +0 -0
- /package/{src → dist}/colorgrid/colortileview.d.ts +0 -0
- /package/{src → dist}/colorgrid/utils.d.ts +0 -0
- /package/{src → dist}/colorpicker/colorpickerview.d.ts +0 -0
- /package/{src → dist}/colorpicker/utils.d.ts +0 -0
- /package/{src → dist}/colorselector/colorgridsfragmentview.d.ts +0 -0
- /package/{src → dist}/colorselector/colorpickerfragmentview.d.ts +0 -0
- /package/{src → dist}/colorselector/colorselectorview.d.ts +0 -0
- /package/{src → dist}/componentfactory.d.ts +0 -0
- /package/{src → dist}/dialog/dialogactionsview.d.ts +0 -0
- /package/{src → dist}/dialog/dialogcontentview.d.ts +0 -0
- /package/{src → dist}/dialog/dialogview.d.ts +0 -0
- /package/{src → dist}/dropdown/button/dropdownbutton.d.ts +0 -0
- /package/{src → dist}/dropdown/button/dropdownbuttonview.d.ts +0 -0
- /package/{src → dist}/dropdown/button/splitbuttonview.d.ts +0 -0
- /package/{src → dist}/dropdown/dropdownpanelfocusable.d.ts +0 -0
- /package/{src → dist}/dropdown/dropdownpanelview.d.ts +0 -0
- /package/{src → dist}/dropdown/dropdownview.d.ts +0 -0
- /package/{src → dist}/dropdown/menu/dropdownmenubehaviors.d.ts +0 -0
- /package/{src → dist}/dropdown/menu/dropdownmenubuttonview.d.ts +0 -0
- /package/{src → dist}/dropdown/menu/dropdownmenulistitembuttonview.d.ts +0 -0
- /package/{src → dist}/dropdown/menu/dropdownmenulistitemview.d.ts +0 -0
- /package/{src → dist}/dropdown/menu/dropdownmenulistview.d.ts +0 -0
- /package/{src → dist}/dropdown/menu/dropdownmenunestedmenupanelview.d.ts +0 -0
- /package/{src → dist}/dropdown/menu/dropdownmenunestedmenuview.d.ts +0 -0
- /package/{src → dist}/dropdown/menu/dropdownmenurootlistview.d.ts +0 -0
- /package/{src → dist}/dropdown/menu/utils.d.ts +0 -0
- /package/{src → dist}/dropdown/utils.d.ts +0 -0
- /package/{src → dist}/editableui/editableuiview.d.ts +0 -0
- /package/{src → dist}/editableui/inline/inlineeditableuiview.d.ts +0 -0
- /package/{src → dist}/editorui/accessibilityhelp/accessibilityhelp.d.ts +0 -0
- /package/{src → dist}/editorui/accessibilityhelp/accessibilityhelpcontentview.d.ts +0 -0
- /package/{src → dist}/editorui/bodycollection.d.ts +0 -0
- /package/{src → dist}/editorui/boxed/boxededitoruiview.d.ts +0 -0
- /package/{src → dist}/editorui/editorui.d.ts +0 -0
- /package/{src → dist}/editorui/editoruiview.d.ts +0 -0
- /package/{src → dist}/editorui/evaluationbadge.d.ts +0 -0
- /package/{src → dist}/editorui/poweredby.d.ts +0 -0
- /package/{src → dist}/focuscycler.d.ts +0 -0
- /package/{src → dist}/formheader/formheaderview.d.ts +0 -0
- /package/{src → dist}/formrow/formrowview.d.ts +0 -0
- /package/{src → dist}/highlightedtext/buttonlabelwithhighlightview.d.ts +0 -0
- /package/{src → dist}/highlightedtext/highlightedtextview.d.ts +0 -0
- /package/{src → dist}/highlightedtext/labelwithhighlightview.d.ts +0 -0
- /package/{src → dist}/icon/iconview.d.ts +0 -0
- /package/{src → dist}/iframe/iframeview.d.ts +0 -0
- /package/{src → dist}/index.d.ts +0 -0
- /package/{src → dist}/input/inputbase.d.ts +0 -0
- /package/{src → dist}/input/inputview.d.ts +0 -0
- /package/{src → dist}/inputnumber/inputnumberview.d.ts +0 -0
- /package/{src → dist}/inputtext/inputtextview.d.ts +0 -0
- /package/{src → dist}/label/labelview.d.ts +0 -0
- /package/{src → dist}/labeledfield/labeledfieldview.d.ts +0 -0
- /package/{src → dist}/labeledfield/utils.d.ts +0 -0
- /package/{src → dist}/labeledinput/labeledinputview.d.ts +0 -0
- /package/{src → dist}/legacyerrors.d.ts +0 -0
- /package/{src → dist}/list/listitemgroupview.d.ts +0 -0
- /package/{src → dist}/list/listitemview.d.ts +0 -0
- /package/{src → dist}/list/listseparatorview.d.ts +0 -0
- /package/{src → dist}/list/listview.d.ts +0 -0
- /package/{src → dist}/menubar/menubarmenubuttonview.d.ts +0 -0
- /package/{src → dist}/menubar/menubarmenulistitembuttonview.d.ts +0 -0
- /package/{src → dist}/menubar/menubarmenulistitemfiledialogbuttonview.d.ts +0 -0
- /package/{src → dist}/menubar/menubarmenulistitemview.d.ts +0 -0
- /package/{src → dist}/menubar/menubarmenulistview.d.ts +0 -0
- /package/{src → dist}/menubar/menubarmenupanelview.d.ts +0 -0
- /package/{src → dist}/menubar/menubarview.d.ts +0 -0
- /package/{src → dist}/menubar/utils.d.ts +0 -0
- /package/{src → dist}/model.d.ts +0 -0
- /package/{src → dist}/notification/notification.d.ts +0 -0
- /package/{src → dist}/panel/balloon/balloonpanelview.d.ts +0 -0
- /package/{src → dist}/panel/balloon/contextualballoon.d.ts +0 -0
- /package/{src → dist}/panel/sticky/stickypanelview.d.ts +0 -0
- /package/{src → dist}/search/filteredview.d.ts +0 -0
- /package/{src → dist}/search/filtergroupanditemnames.d.ts +0 -0
- /package/{src → dist}/search/searchinfoview.d.ts +0 -0
- /package/{src → dist}/search/searchresultsview.d.ts +0 -0
- /package/{src → dist}/search/text/searchtextqueryview.d.ts +0 -0
- /package/{src → dist}/search/text/searchtextview.d.ts +0 -0
- /package/{src → dist}/spinner/spinnerview.d.ts +0 -0
- /package/{src → dist}/template.d.ts +0 -0
- /package/{src → dist}/textarea/textareaview.d.ts +0 -0
- /package/{src → dist}/toolbar/balloon/balloontoolbar.d.ts +0 -0
- /package/{src → dist}/toolbar/block/blockbuttonview.d.ts +0 -0
- /package/{src → dist}/toolbar/block/blocktoolbar.d.ts +0 -0
- /package/{src → dist}/toolbar/normalizetoolbarconfig.d.ts +0 -0
- /package/{src → dist}/toolbar/toolbarlinebreakview.d.ts +0 -0
- /package/{src → dist}/toolbar/toolbarseparatorview.d.ts +0 -0
- /package/{src → dist}/toolbar/toolbarview.d.ts +0 -0
- /package/{src → dist}/tooltipmanager.d.ts +0 -0
- /package/{src → dist}/viewcollection.d.ts +0 -0
package/src/dropdown/utils.js
DELETED
|
@@ -1,654 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license Copyright (c) 2003-2026, CKSource Holding sp. z o.o. All rights reserved.
|
|
3
|
-
* For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-licensing-options
|
|
4
|
-
*/
|
|
5
|
-
/**
|
|
6
|
-
* @module ui/dropdown/utils
|
|
7
|
-
*/
|
|
8
|
-
import { DropdownPanelView } from './dropdownpanelview.js';
|
|
9
|
-
import { DropdownView } from './dropdownview.js';
|
|
10
|
-
import { DropdownButtonView } from './button/dropdownbuttonview.js';
|
|
11
|
-
import { DropdownMenuRootListView } from './menu/dropdownmenurootlistview.js';
|
|
12
|
-
import { ToolbarView } from '../toolbar/toolbarview.js';
|
|
13
|
-
import { ListView } from '../list/listview.js';
|
|
14
|
-
import { ListItemView } from '../list/listitemview.js';
|
|
15
|
-
import { ListSeparatorView } from '../list/listseparatorview.js';
|
|
16
|
-
import { SplitButtonView } from './button/splitbuttonview.js';
|
|
17
|
-
import { SwitchButtonView } from '../button/switchbuttonview.js';
|
|
18
|
-
import { ViewCollection } from '../viewcollection.js';
|
|
19
|
-
import { clickOutsideHandler } from '../bindings/clickoutsidehandler.js';
|
|
20
|
-
import { global, priorities, logWarning } from '@ckeditor/ckeditor5-utils';
|
|
21
|
-
import '../../theme/components/dropdown/toolbardropdown.css';
|
|
22
|
-
import '../../theme/components/dropdown/listdropdown.css';
|
|
23
|
-
import { ListItemGroupView } from '../list/listitemgroupview.js';
|
|
24
|
-
import { ListItemButtonView } from '../button/listitembuttonview.js';
|
|
25
|
-
/**
|
|
26
|
-
* A helper for creating dropdowns. It creates an instance of a {@link module:ui/dropdown/dropdownview~DropdownView dropdown},
|
|
27
|
-
* with a {@link module:ui/dropdown/button/dropdownbutton~DropdownButton button},
|
|
28
|
-
* {@link module:ui/dropdown/dropdownpanelview~DropdownPanelView panel} and all standard dropdown's behaviors.
|
|
29
|
-
*
|
|
30
|
-
* # Creating dropdowns
|
|
31
|
-
*
|
|
32
|
-
* By default, the default {@link module:ui/dropdown/button/dropdownbuttonview~DropdownButtonView} class is used as
|
|
33
|
-
* definition of the button:
|
|
34
|
-
*
|
|
35
|
-
* ```ts
|
|
36
|
-
* const dropdown = createDropdown( model );
|
|
37
|
-
*
|
|
38
|
-
* // Configure dropdown's button properties:
|
|
39
|
-
* dropdown.buttonView.set( {
|
|
40
|
-
* label: 'A dropdown',
|
|
41
|
-
* withText: true
|
|
42
|
-
* } );
|
|
43
|
-
*
|
|
44
|
-
* dropdown.render();
|
|
45
|
-
*
|
|
46
|
-
* // Will render a dropdown labeled "A dropdown" with an empty panel.
|
|
47
|
-
* document.body.appendChild( dropdown.element );
|
|
48
|
-
* ```
|
|
49
|
-
*
|
|
50
|
-
* You can also provide other button views (they need to implement the
|
|
51
|
-
* {@link module:ui/dropdown/button/dropdownbutton~DropdownButton} interface). For instance, you can use
|
|
52
|
-
* {@link module:ui/dropdown/button/splitbuttonview~SplitButtonView} to create a dropdown with a split button.
|
|
53
|
-
*
|
|
54
|
-
* ```ts
|
|
55
|
-
* const dropdown = createDropdown( locale, SplitButtonView );
|
|
56
|
-
*
|
|
57
|
-
* // Configure dropdown's button properties:
|
|
58
|
-
* dropdown.buttonView.set( {
|
|
59
|
-
* label: 'A dropdown',
|
|
60
|
-
* withText: true
|
|
61
|
-
* } );
|
|
62
|
-
*
|
|
63
|
-
* dropdown.buttonView.on( 'execute', () => {
|
|
64
|
-
* // Add the behavior of the "action part" of the split button.
|
|
65
|
-
* // Split button consists of the "action part" and "arrow part".
|
|
66
|
-
* // The arrow opens the dropdown while the action part can have some other behavior.
|
|
67
|
-
* } );
|
|
68
|
-
*
|
|
69
|
-
* dropdown.render();
|
|
70
|
-
*
|
|
71
|
-
* // Will render a dropdown labeled "A dropdown" with an empty panel.
|
|
72
|
-
* document.body.appendChild( dropdown.element );
|
|
73
|
-
* ```
|
|
74
|
-
*
|
|
75
|
-
* # Adding content to the dropdown's panel
|
|
76
|
-
*
|
|
77
|
-
* The content of the panel can be inserted directly into the `dropdown.panelView.element`:
|
|
78
|
-
*
|
|
79
|
-
* ```ts
|
|
80
|
-
* dropdown.panelView.element.textContent = 'Content of the panel';
|
|
81
|
-
* ```
|
|
82
|
-
*
|
|
83
|
-
* However, most of the time you will want to add there either a {@link module:ui/list/listview~ListView list of options}
|
|
84
|
-
* or a list of buttons (i.e. a {@link module:ui/toolbar/toolbarview~ToolbarView toolbar}).
|
|
85
|
-
* To simplify the task, you can use, respectively, {@link module:ui/dropdown/utils~addListToDropdown} or
|
|
86
|
-
* {@link module:ui/dropdown/utils~addToolbarToDropdown} utils.
|
|
87
|
-
*
|
|
88
|
-
* @param locale The locale instance.
|
|
89
|
-
* @param ButtonClassOrInstance The dropdown button view class. Needs to implement the
|
|
90
|
-
* {@link module:ui/dropdown/button/dropdownbutton~DropdownButton} interface.
|
|
91
|
-
* @returns The dropdown view instance.
|
|
92
|
-
*/
|
|
93
|
-
export function createDropdown(locale, ButtonClassOrInstance = DropdownButtonView) {
|
|
94
|
-
const buttonView = typeof ButtonClassOrInstance == 'function' ? new ButtonClassOrInstance(locale) : ButtonClassOrInstance;
|
|
95
|
-
const panelView = new DropdownPanelView(locale);
|
|
96
|
-
const dropdownView = new DropdownView(locale, buttonView, panelView);
|
|
97
|
-
buttonView.bind('isEnabled').to(dropdownView);
|
|
98
|
-
if (buttonView instanceof SplitButtonView) {
|
|
99
|
-
buttonView.arrowView.bind('isOn').to(dropdownView, 'isOpen');
|
|
100
|
-
}
|
|
101
|
-
else {
|
|
102
|
-
buttonView.bind('isOn').to(dropdownView, 'isOpen');
|
|
103
|
-
}
|
|
104
|
-
addDefaultBehaviors(dropdownView);
|
|
105
|
-
return dropdownView;
|
|
106
|
-
}
|
|
107
|
-
/**
|
|
108
|
-
* Adds a menu UI component to a dropdown and sets all common behaviors and interactions between the dropdown and the menu.
|
|
109
|
-
*
|
|
110
|
-
* Use this helper to create multi-level dropdown menus that are displayed in a toolbar.
|
|
111
|
-
*
|
|
112
|
-
* Internally, it creates an instance of {@link module:ui/dropdown/menu/dropdownmenurootlistview~DropdownMenuRootListView}.
|
|
113
|
-
*
|
|
114
|
-
* Example:
|
|
115
|
-
*
|
|
116
|
-
* ```ts
|
|
117
|
-
* const definitions = [
|
|
118
|
-
* {
|
|
119
|
-
* id: 'menu_1',
|
|
120
|
-
* menu: 'Menu 1',
|
|
121
|
-
* children: [
|
|
122
|
-
* {
|
|
123
|
-
* id: 'menu_1_a',
|
|
124
|
-
* label: 'Item A'
|
|
125
|
-
* },
|
|
126
|
-
* {
|
|
127
|
-
* id: 'menu_1_b',
|
|
128
|
-
* label: 'Item B'
|
|
129
|
-
* }
|
|
130
|
-
* ]
|
|
131
|
-
* },
|
|
132
|
-
* {
|
|
133
|
-
* id: 'top_a',
|
|
134
|
-
* label: 'Top Item A'
|
|
135
|
-
* },
|
|
136
|
-
* {
|
|
137
|
-
* id: 'top_b',
|
|
138
|
-
* label: 'Top Item B'
|
|
139
|
-
* }
|
|
140
|
-
* ];
|
|
141
|
-
*
|
|
142
|
-
* const dropdownView = createDropdown( editor.locale );
|
|
143
|
-
*
|
|
144
|
-
* addMenuToDropdown( dropdownView, editor.ui.view.body, definitions );
|
|
145
|
-
* ```
|
|
146
|
-
*
|
|
147
|
-
* After using this helper, the `dropdown` will fire {@link module:ui/dropdown/dropdownview~DropdownViewEvent `execute`} event when
|
|
148
|
-
* a nested menu button is pressed.
|
|
149
|
-
*
|
|
150
|
-
* The helper will make sure that the `dropdownMenuRootListView` is lazy loaded, i.e., the menu component structure will be initialized
|
|
151
|
-
* and rendered only after the `dropdown` is opened for the first time.
|
|
152
|
-
*
|
|
153
|
-
* @param dropdownView A dropdown instance to which the menu component will be added.
|
|
154
|
-
* @param body Body collection to which floating menu panels will be added.
|
|
155
|
-
* @param definition The menu component definition.
|
|
156
|
-
* @param options.ariaLabel Label used by assistive technologies to describe the top-level menu.
|
|
157
|
-
*/
|
|
158
|
-
export function addMenuToDropdown(dropdownView, body, definition, options = {}) {
|
|
159
|
-
dropdownView.menuView = new DropdownMenuRootListView(dropdownView.locale, body, definition);
|
|
160
|
-
dropdownView.focusTracker.add(dropdownView.menuView);
|
|
161
|
-
if (dropdownView.isOpen) {
|
|
162
|
-
addMenuToOpenDropdown(dropdownView, options);
|
|
163
|
-
}
|
|
164
|
-
else {
|
|
165
|
-
// Load the UI elements only after the dropdown is opened for the first time - lazy loading.
|
|
166
|
-
dropdownView.once('change:isOpen', () => {
|
|
167
|
-
addMenuToOpenDropdown(dropdownView, options);
|
|
168
|
-
}, { priority: 'highest' });
|
|
169
|
-
}
|
|
170
|
-
}
|
|
171
|
-
function addMenuToOpenDropdown(dropdownView, options) {
|
|
172
|
-
const dropdownMenuRootListView = dropdownView.menuView;
|
|
173
|
-
const t = dropdownView.locale.t;
|
|
174
|
-
dropdownMenuRootListView.delegate('menu:execute').to(dropdownView, 'execute');
|
|
175
|
-
dropdownMenuRootListView.listenTo(dropdownView, 'change:isOpen', (evt, name, isOpen) => {
|
|
176
|
-
if (!isOpen) {
|
|
177
|
-
dropdownMenuRootListView.closeMenus();
|
|
178
|
-
}
|
|
179
|
-
}, { priority: 'low' }); // Make sure this is fired after `focusDropdownButtonOnClose` behavior.
|
|
180
|
-
// When `dropdownMenuRootListView` is added as a `panelView` child, it becomes rendered (`panelView` is rendered at this point).
|
|
181
|
-
dropdownView.panelView.children.add(dropdownMenuRootListView);
|
|
182
|
-
// Nested menu panels are added to body collection, so they are not children of the `dropdownView` from DOM perspective.
|
|
183
|
-
// Add these panels to `dropdownView` focus tracker, so they are treated like part of the `dropdownView` for focus-related purposes.
|
|
184
|
-
for (const menu of dropdownMenuRootListView.menus) {
|
|
185
|
-
dropdownView.focusTracker.add(menu);
|
|
186
|
-
}
|
|
187
|
-
dropdownMenuRootListView.ariaLabel = options.ariaLabel || t('Dropdown menu');
|
|
188
|
-
}
|
|
189
|
-
/**
|
|
190
|
-
* Adds an instance of {@link module:ui/toolbar/toolbarview~ToolbarView} to a dropdown.
|
|
191
|
-
*
|
|
192
|
-
* ```ts
|
|
193
|
-
* const buttonsCreator = () => {
|
|
194
|
-
* const buttons = [];
|
|
195
|
-
*
|
|
196
|
-
* // Either create a new ButtonView instance or create existing.
|
|
197
|
-
* buttons.push( new ButtonView() );
|
|
198
|
-
* buttons.push( editor.ui.componentFactory.create( 'someButton' ) );
|
|
199
|
-
* };
|
|
200
|
-
*
|
|
201
|
-
* const dropdown = createDropdown( locale );
|
|
202
|
-
*
|
|
203
|
-
* addToolbarToDropdown( dropdown, buttonsCreator, { isVertical: true } );
|
|
204
|
-
*
|
|
205
|
-
* // Will render a vertical button dropdown labeled "A button dropdown"
|
|
206
|
-
* // with a button group in the panel containing two buttons.
|
|
207
|
-
* // Buttons inside the dropdown will be created on first dropdown panel open.
|
|
208
|
-
* dropdown.render()
|
|
209
|
-
* document.body.appendChild( dropdown.element );
|
|
210
|
-
* ```
|
|
211
|
-
*
|
|
212
|
-
* **Note:** To improve the accessibility, you can tell the dropdown to focus the first active button of the toolbar when the dropdown
|
|
213
|
-
* {@link module:ui/dropdown/dropdownview~DropdownView#isOpen gets open}. See the documentation of `options` to learn more.
|
|
214
|
-
*
|
|
215
|
-
* **Note:** Toolbar view will be created on first open of the dropdown.
|
|
216
|
-
*
|
|
217
|
-
* See {@link module:ui/dropdown/utils~createDropdown} and {@link module:ui/toolbar/toolbarview~ToolbarView}.
|
|
218
|
-
*
|
|
219
|
-
* @param dropdownView A dropdown instance to which `ToolbarView` will be added.
|
|
220
|
-
* @param options.enableActiveItemFocusOnDropdownOpen When set `true`, the focus will automatically move to the first
|
|
221
|
-
* active {@link module:ui/toolbar/toolbarview~ToolbarView#items item} of the toolbar upon
|
|
222
|
-
* {@link module:ui/dropdown/dropdownview~DropdownView#isOpen opening} the dropdown. Active items are those with the `isOn` property set
|
|
223
|
-
* `true` (for instance {@link module:ui/button/buttonview~ButtonView buttons}). If no active items is found, the toolbar will be focused
|
|
224
|
-
* as a whole resulting in the focus moving to its first focusable item (default behavior of
|
|
225
|
-
* {@link module:ui/dropdown/dropdownview~DropdownView}).
|
|
226
|
-
* @param options.ariaLabel Label used by assistive technologies to describe toolbar element.
|
|
227
|
-
* @param options.maxWidth The maximum width of the toolbar element.
|
|
228
|
-
* Details: {@link module:ui/toolbar/toolbarview~ToolbarView#maxWidth}.
|
|
229
|
-
* @param options.class An additional CSS class added to the toolbar element.
|
|
230
|
-
* @param options.isCompact When set true, makes the toolbar look compact with toolbar element.
|
|
231
|
-
* @param options.isVertical Controls the orientation of toolbar items.
|
|
232
|
-
*/
|
|
233
|
-
export function addToolbarToDropdown(dropdownView, buttonsOrCallback, options = {}) {
|
|
234
|
-
dropdownView.extendTemplate({
|
|
235
|
-
attributes: {
|
|
236
|
-
class: ['ck-toolbar-dropdown']
|
|
237
|
-
}
|
|
238
|
-
});
|
|
239
|
-
if (dropdownView.isOpen) {
|
|
240
|
-
addToolbarToOpenDropdown(dropdownView, buttonsOrCallback, options);
|
|
241
|
-
}
|
|
242
|
-
else {
|
|
243
|
-
dropdownView.once('change:isOpen', () => addToolbarToOpenDropdown(dropdownView, buttonsOrCallback, options), { priority: 'highest' });
|
|
244
|
-
}
|
|
245
|
-
if (options.enableActiveItemFocusOnDropdownOpen) {
|
|
246
|
-
// Accessibility: Focus the first active button in the toolbar when the dropdown gets open.
|
|
247
|
-
focusChildOnDropdownOpen(dropdownView, () => dropdownView.toolbarView.items.find((item) => item.isOn));
|
|
248
|
-
}
|
|
249
|
-
}
|
|
250
|
-
/**
|
|
251
|
-
* Adds an instance of {@link module:ui/toolbar/toolbarview~ToolbarView} to a dropdown.
|
|
252
|
-
*/
|
|
253
|
-
function addToolbarToOpenDropdown(dropdownView, buttonsOrCallback, options) {
|
|
254
|
-
const locale = dropdownView.locale;
|
|
255
|
-
const t = locale.t;
|
|
256
|
-
const toolbarView = dropdownView.toolbarView = new ToolbarView(locale);
|
|
257
|
-
const buttons = typeof buttonsOrCallback == 'function' ? buttonsOrCallback() : buttonsOrCallback;
|
|
258
|
-
toolbarView.ariaLabel = options.ariaLabel || t('Dropdown toolbar');
|
|
259
|
-
if (options.maxWidth) {
|
|
260
|
-
toolbarView.maxWidth = options.maxWidth;
|
|
261
|
-
}
|
|
262
|
-
if (options.class) {
|
|
263
|
-
toolbarView.class = options.class;
|
|
264
|
-
}
|
|
265
|
-
if (options.isCompact) {
|
|
266
|
-
toolbarView.isCompact = options.isCompact;
|
|
267
|
-
}
|
|
268
|
-
if (options.isVertical) {
|
|
269
|
-
toolbarView.isVertical = true;
|
|
270
|
-
}
|
|
271
|
-
if (buttons instanceof ViewCollection) {
|
|
272
|
-
toolbarView.items.bindTo(buttons).using(item => item);
|
|
273
|
-
}
|
|
274
|
-
else {
|
|
275
|
-
toolbarView.items.addMany(buttons);
|
|
276
|
-
}
|
|
277
|
-
dropdownView.panelView.children.add(toolbarView);
|
|
278
|
-
dropdownView.focusTracker.add(toolbarView);
|
|
279
|
-
toolbarView.items.delegate('execute').to(dropdownView);
|
|
280
|
-
}
|
|
281
|
-
/**
|
|
282
|
-
* Adds an instance of {@link module:ui/list/listview~ListView} to a dropdown.
|
|
283
|
-
*
|
|
284
|
-
* ```ts
|
|
285
|
-
* const items = new Collection<ListDropdownItemDefinition>();
|
|
286
|
-
*
|
|
287
|
-
* items.add( {
|
|
288
|
-
* type: 'button',
|
|
289
|
-
* model: new Model( {
|
|
290
|
-
* withText: true,
|
|
291
|
-
* label: 'First item',
|
|
292
|
-
* labelStyle: 'color: red'
|
|
293
|
-
* } )
|
|
294
|
-
* } );
|
|
295
|
-
*
|
|
296
|
-
* items.add( {
|
|
297
|
-
* type: 'button',
|
|
298
|
-
* model: new Model( {
|
|
299
|
-
* withText: true,
|
|
300
|
-
* label: 'Second item',
|
|
301
|
-
* labelStyle: 'color: green',
|
|
302
|
-
* class: 'foo'
|
|
303
|
-
* } )
|
|
304
|
-
* } );
|
|
305
|
-
*
|
|
306
|
-
* const dropdown = createDropdown( locale );
|
|
307
|
-
*
|
|
308
|
-
* addListToDropdown( dropdown, items );
|
|
309
|
-
*
|
|
310
|
-
* // Will render a dropdown with a list in the panel containing two items.
|
|
311
|
-
* dropdown.render()
|
|
312
|
-
* document.body.appendChild( dropdown.element );
|
|
313
|
-
* ```
|
|
314
|
-
*
|
|
315
|
-
* The `items` collection passed to this methods controls the presence and attributes of respective
|
|
316
|
-
* {@link module:ui/list/listitemview~ListItemView list items}.
|
|
317
|
-
*
|
|
318
|
-
* **Note:** To improve the accessibility, when a list is added to the dropdown using this helper the dropdown will automatically attempt
|
|
319
|
-
* to focus the first active item (a host to a {@link module:ui/button/buttonview~ButtonView} with
|
|
320
|
-
* {@link module:ui/button/buttonview~ButtonView#isOn} set `true`) or the very first item when none are active.
|
|
321
|
-
*
|
|
322
|
-
* **Note:** List view will be created on first open of the dropdown.
|
|
323
|
-
*
|
|
324
|
-
* See {@link module:ui/dropdown/utils~createDropdown} and {@link module:list/list~List}.
|
|
325
|
-
*
|
|
326
|
-
* @param dropdownView A dropdown instance to which `ListVIew` will be added.
|
|
327
|
-
* @param itemsOrCallback A collection of the list item definitions or a callback returning a list item definitions to populate the list.
|
|
328
|
-
* @param options.ariaLabel Label used by assistive technologies to describe list element.
|
|
329
|
-
* @param options.role Will be reflected by the `role` DOM attribute in `ListVIew` and used by assistive technologies.
|
|
330
|
-
*/
|
|
331
|
-
export function addListToDropdown(dropdownView, itemsOrCallback, options = {}) {
|
|
332
|
-
if (dropdownView.isOpen) {
|
|
333
|
-
addListToOpenDropdown(dropdownView, itemsOrCallback, options);
|
|
334
|
-
}
|
|
335
|
-
else {
|
|
336
|
-
dropdownView.once('change:isOpen', () => addListToOpenDropdown(dropdownView, itemsOrCallback, options), { priority: 'highest' });
|
|
337
|
-
}
|
|
338
|
-
// Accessibility: Focus the first active button in the list when the dropdown gets open.
|
|
339
|
-
focusChildOnDropdownOpen(dropdownView, () => dropdownView.listView.items.find(item => {
|
|
340
|
-
if (item instanceof ListItemView) {
|
|
341
|
-
return item.children.first.isOn;
|
|
342
|
-
}
|
|
343
|
-
return false;
|
|
344
|
-
}));
|
|
345
|
-
}
|
|
346
|
-
/**
|
|
347
|
-
* Adds an instance of {@link module:ui/list/listview~ListView} to a dropdown.
|
|
348
|
-
*/
|
|
349
|
-
function addListToOpenDropdown(dropdownView, itemsOrCallback, options) {
|
|
350
|
-
const locale = dropdownView.locale;
|
|
351
|
-
const listView = dropdownView.listView = new ListView(locale);
|
|
352
|
-
const items = typeof itemsOrCallback == 'function' ? itemsOrCallback() : itemsOrCallback;
|
|
353
|
-
listView.ariaLabel = options.ariaLabel;
|
|
354
|
-
listView.role = options.role;
|
|
355
|
-
bindViewCollectionItemsToDefinitions(dropdownView, listView.items, items, locale);
|
|
356
|
-
dropdownView.panelView.children.add(listView);
|
|
357
|
-
listView.items.delegate('execute').to(dropdownView);
|
|
358
|
-
}
|
|
359
|
-
/**
|
|
360
|
-
* A helper to be used on an existing {@link module:ui/dropdown/dropdownview~DropdownView} that focuses
|
|
361
|
-
* a specific child in DOM when the dropdown {@link module:ui/dropdown/dropdownview~DropdownView#isOpen gets open}.
|
|
362
|
-
*
|
|
363
|
-
* @param dropdownView A dropdown instance to which the focus behavior will be added.
|
|
364
|
-
* @param childSelectorCallback A callback executed when the dropdown gets open. It should return a {@link module:ui/view~View}
|
|
365
|
-
* instance (child of {@link module:ui/dropdown/dropdownview~DropdownView#panelView}) that will get focused or a falsy value.
|
|
366
|
-
* If falsy value is returned, a default behavior of the dropdown will engage focusing the first focusable child in
|
|
367
|
-
* the {@link module:ui/dropdown/dropdownview~DropdownView#panelView}.
|
|
368
|
-
*/
|
|
369
|
-
export function focusChildOnDropdownOpen(dropdownView, childSelectorCallback) {
|
|
370
|
-
dropdownView.on('change:isOpen', () => {
|
|
371
|
-
if (!dropdownView.isOpen) {
|
|
372
|
-
return;
|
|
373
|
-
}
|
|
374
|
-
const childToFocus = childSelectorCallback();
|
|
375
|
-
if (!childToFocus) {
|
|
376
|
-
return;
|
|
377
|
-
}
|
|
378
|
-
if (typeof childToFocus.focus === 'function') {
|
|
379
|
-
childToFocus.focus();
|
|
380
|
-
}
|
|
381
|
-
else {
|
|
382
|
-
/**
|
|
383
|
-
* The child view of a {@link module:ui/dropdown/dropdownview~DropdownView dropdown} is missing the `focus()` method
|
|
384
|
-
* and could not be focused when the dropdown got {@link module:ui/dropdown/dropdownview~DropdownView#isOpen open}.
|
|
385
|
-
*
|
|
386
|
-
* Making the content of a dropdown focusable in this case greatly improves the accessibility. Please make the view instance
|
|
387
|
-
* implements the {@link module:ui/dropdown/dropdownpanelfocusable~DropdownPanelFocusable focusable interface} for the best user
|
|
388
|
-
* experience.
|
|
389
|
-
*
|
|
390
|
-
* @error ui-dropdown-focus-child-on-open-child-missing-focus
|
|
391
|
-
* @param {module:ui/view~View} view Child to focus.
|
|
392
|
-
*/
|
|
393
|
-
logWarning('ui-dropdown-focus-child-on-open-child-missing-focus', { view: childToFocus });
|
|
394
|
-
}
|
|
395
|
-
// * Let the panel show up first (do not focus an invisible element).
|
|
396
|
-
// * Execute after focusDropdownPanelOnOpen(). See focusDropdownPanelOnOpen() to learn more.
|
|
397
|
-
}, { priority: priorities.low - 10 });
|
|
398
|
-
}
|
|
399
|
-
/**
|
|
400
|
-
* Add a set of default behaviors to dropdown view.
|
|
401
|
-
*/
|
|
402
|
-
function addDefaultBehaviors(dropdownView) {
|
|
403
|
-
closeDropdownOnClickOutside(dropdownView);
|
|
404
|
-
closeDropdownOnExecute(dropdownView);
|
|
405
|
-
closeDropdownOnBlur(dropdownView);
|
|
406
|
-
focusDropdownContentsOnArrows(dropdownView);
|
|
407
|
-
focusDropdownButtonOnClose(dropdownView);
|
|
408
|
-
focusDropdownPanelOnOpen(dropdownView);
|
|
409
|
-
}
|
|
410
|
-
/**
|
|
411
|
-
* Adds a behavior to a dropdownView that closes opened dropdown when user clicks outside the dropdown.
|
|
412
|
-
*/
|
|
413
|
-
function closeDropdownOnClickOutside(dropdownView) {
|
|
414
|
-
clickOutsideHandler({
|
|
415
|
-
emitter: dropdownView,
|
|
416
|
-
activator: () => dropdownView.isRendered && dropdownView.isOpen,
|
|
417
|
-
callback: () => {
|
|
418
|
-
dropdownView.isOpen = false;
|
|
419
|
-
},
|
|
420
|
-
contextElements: () => [
|
|
421
|
-
dropdownView.element,
|
|
422
|
-
// Include all elements connected to the dropdown's focus tracker, but exclude those that are direct children
|
|
423
|
-
// of DropdownView#element. They would be identified as descendants of #element anyway upon clicking and would
|
|
424
|
-
// not contribute to the logic.
|
|
425
|
-
...getFocusTrackerTreeElements(dropdownView.focusTracker).filter(element => !dropdownView.element.contains(element))
|
|
426
|
-
]
|
|
427
|
-
});
|
|
428
|
-
}
|
|
429
|
-
/**
|
|
430
|
-
* Returns all DOM elements connected to a DropdownView's focus tracker, either directly (same DOM sub-tree)
|
|
431
|
-
* or indirectly (external views registered in the focus tracker).
|
|
432
|
-
*/
|
|
433
|
-
function getFocusTrackerTreeElements(focusTracker) {
|
|
434
|
-
return [
|
|
435
|
-
...focusTracker.elements,
|
|
436
|
-
...focusTracker.externalViews.flatMap(view => getFocusTrackerTreeElements(view.focusTracker))
|
|
437
|
-
];
|
|
438
|
-
}
|
|
439
|
-
/**
|
|
440
|
-
* Adds a behavior to a dropdownView that closes the dropdown view on "execute" event.
|
|
441
|
-
*/
|
|
442
|
-
function closeDropdownOnExecute(dropdownView) {
|
|
443
|
-
// Close the dropdown when one of the list items has been executed.
|
|
444
|
-
dropdownView.on('execute', evt => {
|
|
445
|
-
// Toggling a switch button view should not close the dropdown.
|
|
446
|
-
if (evt.source instanceof SwitchButtonView) {
|
|
447
|
-
return;
|
|
448
|
-
}
|
|
449
|
-
dropdownView.isOpen = false;
|
|
450
|
-
});
|
|
451
|
-
}
|
|
452
|
-
/**
|
|
453
|
-
* Adds a behavior to a dropdown view that closes opened dropdown when it loses focus.
|
|
454
|
-
*/
|
|
455
|
-
function closeDropdownOnBlur(dropdownView) {
|
|
456
|
-
dropdownView.focusTracker.on('change:isFocused', (evt, name, isFocused) => {
|
|
457
|
-
if (isFocused || !dropdownView.isOpen) {
|
|
458
|
-
return;
|
|
459
|
-
}
|
|
460
|
-
dropdownView.isOpen = false;
|
|
461
|
-
});
|
|
462
|
-
}
|
|
463
|
-
/**
|
|
464
|
-
* Adds a behavior to a dropdownView that focuses the dropdown's panel view contents on keystrokes.
|
|
465
|
-
*/
|
|
466
|
-
function focusDropdownContentsOnArrows(dropdownView) {
|
|
467
|
-
// If the dropdown panel is already open, the arrow down key should focus the first child of the #panelView.
|
|
468
|
-
dropdownView.keystrokes.set('arrowdown', (data, cancel) => {
|
|
469
|
-
if (dropdownView.isOpen) {
|
|
470
|
-
dropdownView.panelView.focus();
|
|
471
|
-
cancel();
|
|
472
|
-
}
|
|
473
|
-
});
|
|
474
|
-
// If the dropdown panel is already open, the arrow up key should focus the last child of the #panelView.
|
|
475
|
-
dropdownView.keystrokes.set('arrowup', (data, cancel) => {
|
|
476
|
-
if (dropdownView.isOpen) {
|
|
477
|
-
dropdownView.panelView.focusLast();
|
|
478
|
-
cancel();
|
|
479
|
-
}
|
|
480
|
-
});
|
|
481
|
-
}
|
|
482
|
-
/**
|
|
483
|
-
* Adds a behavior that focuses the #buttonView when the dropdown was closed but focus was within the #panelView element.
|
|
484
|
-
* This makes sure the focus is never lost.
|
|
485
|
-
*/
|
|
486
|
-
function focusDropdownButtonOnClose(dropdownView) {
|
|
487
|
-
dropdownView.on('change:isOpen', (evt, name, isOpen) => {
|
|
488
|
-
if (isOpen) {
|
|
489
|
-
return;
|
|
490
|
-
}
|
|
491
|
-
const elements = dropdownView.focusTracker.elements;
|
|
492
|
-
// If the dropdown was closed, move the focus back to the button (#12125).
|
|
493
|
-
// Don't touch the focus, if it moved somewhere else (e.g. moved to the editing root on #execute) (#12178).
|
|
494
|
-
// Note: Don't use the state of the DropdownView#focusTracker here. It fires #blur with the timeout.
|
|
495
|
-
if (elements.some(element => element.contains(global.document.activeElement))) {
|
|
496
|
-
dropdownView.buttonView.focus();
|
|
497
|
-
}
|
|
498
|
-
});
|
|
499
|
-
}
|
|
500
|
-
/**
|
|
501
|
-
* Adds a behavior that focuses the #panelView when dropdown gets open (accessibility).
|
|
502
|
-
*/
|
|
503
|
-
function focusDropdownPanelOnOpen(dropdownView) {
|
|
504
|
-
dropdownView.on('change:isOpen', (evt, name, isOpen) => {
|
|
505
|
-
if (!isOpen) {
|
|
506
|
-
return;
|
|
507
|
-
}
|
|
508
|
-
// Focus the first item in the dropdown when the dropdown opened.
|
|
509
|
-
dropdownView.panelView.focus();
|
|
510
|
-
// * Let the panel show up first (do not focus an invisible element).
|
|
511
|
-
// * Also, execute before focusChildOnDropdownOpen() to make sure this helper does not break the
|
|
512
|
-
// focus of a specific child by kicking in too late and resetting the focus in the panel.
|
|
513
|
-
}, { priority: 'low' });
|
|
514
|
-
}
|
|
515
|
-
/**
|
|
516
|
-
* This helper populates a dropdown list with items and groups according to the
|
|
517
|
-
* collection of item definitions. A permanent binding is created in this process allowing
|
|
518
|
-
* dynamic management of the dropdown list content.
|
|
519
|
-
*
|
|
520
|
-
* @param dropdownView
|
|
521
|
-
* @param listItems
|
|
522
|
-
* @param definitions
|
|
523
|
-
* @param locale
|
|
524
|
-
*/
|
|
525
|
-
function bindViewCollectionItemsToDefinitions(dropdownView, listItems, definitions, locale) {
|
|
526
|
-
bindDropdownToggleableButtonsAlignment(listItems);
|
|
527
|
-
listItems.bindTo(definitions).using(def => {
|
|
528
|
-
if (def.type === 'separator') {
|
|
529
|
-
return new ListSeparatorView(locale);
|
|
530
|
-
}
|
|
531
|
-
else if (def.type === 'group') {
|
|
532
|
-
const groupView = new ListItemGroupView(locale);
|
|
533
|
-
groupView.set({ label: def.label });
|
|
534
|
-
bindViewCollectionItemsToDefinitions(dropdownView, groupView.items, def.items, locale);
|
|
535
|
-
groupView.items.delegate('execute').to(dropdownView);
|
|
536
|
-
return groupView;
|
|
537
|
-
}
|
|
538
|
-
else if (def.type === 'button' || def.type === 'switchbutton') {
|
|
539
|
-
const isToggleable = def.model.role === 'menuitemcheckbox' || def.model.role === 'menuitemradio';
|
|
540
|
-
const listItemView = new ListItemView(locale);
|
|
541
|
-
let buttonView;
|
|
542
|
-
if (def.type === 'button') {
|
|
543
|
-
buttonView = new ListItemButtonView(locale, def.labelView);
|
|
544
|
-
buttonView.set({
|
|
545
|
-
isToggleable
|
|
546
|
-
});
|
|
547
|
-
}
|
|
548
|
-
else {
|
|
549
|
-
buttonView = new SwitchButtonView(locale);
|
|
550
|
-
}
|
|
551
|
-
// Bind all model properties to the button view.
|
|
552
|
-
buttonView.bind(...Object.keys(def.model)).to(def.model);
|
|
553
|
-
buttonView.delegate('execute').to(listItemView);
|
|
554
|
-
listItemView.children.add(buttonView);
|
|
555
|
-
return listItemView;
|
|
556
|
-
}
|
|
557
|
-
return null;
|
|
558
|
-
});
|
|
559
|
-
}
|
|
560
|
-
/**
|
|
561
|
-
* Sets up alignment handling for toggleable buttons in a dropdown list.
|
|
562
|
-
*
|
|
563
|
-
* Buttons in dropdowns have reserved space for a check icon when they are toggleable.
|
|
564
|
-
* When at least one button in the list is toggleable, all other buttons (even non-toggleable ones)
|
|
565
|
-
* will have space on their left side to align with toggleable buttons.
|
|
566
|
-
*
|
|
567
|
-
* This function handles a special case where a new toggleable button is added (or removed) to a list
|
|
568
|
-
* where previous buttons weren't toggleable. In that case, those previous buttons will
|
|
569
|
-
* automatically allocate space to align with the new toggleable button.
|
|
570
|
-
*
|
|
571
|
-
* Example:
|
|
572
|
-
* ```
|
|
573
|
-
* Before adding toggleable button:
|
|
574
|
-
* +----------------+
|
|
575
|
-
* | Normal Button |
|
|
576
|
-
* +----------------+
|
|
577
|
-
* | Another Button |
|
|
578
|
-
* +----------------+
|
|
579
|
-
*
|
|
580
|
-
* After adding toggleable button:
|
|
581
|
-
* +-------------------+
|
|
582
|
-
* | Normal Button |
|
|
583
|
-
* +-------------------+
|
|
584
|
-
* | Another Button |
|
|
585
|
-
* +-------------------+
|
|
586
|
-
* | ✓ Toggle Button |
|
|
587
|
-
* +-------------------+
|
|
588
|
-
* ```
|
|
589
|
-
*
|
|
590
|
-
* @param listItems Collection of list items to observe for toggleable buttons.
|
|
591
|
-
*/
|
|
592
|
-
function bindDropdownToggleableButtonsAlignment(listItems) {
|
|
593
|
-
// Keep track of how many toggleable buttons are in the list.
|
|
594
|
-
let toggleableButtonsCount = 0;
|
|
595
|
-
// Helper function that checks if a view item is a list item button.
|
|
596
|
-
const pickListItemButtonIfPresent = (item) => {
|
|
597
|
-
// Check if the item is a ListItemView with a ListItemButtonView as its first child.
|
|
598
|
-
if (!(item instanceof ListItemView) || !(item.children.first instanceof ListItemButtonView)) {
|
|
599
|
-
return null;
|
|
600
|
-
}
|
|
601
|
-
return item.children.first;
|
|
602
|
-
};
|
|
603
|
-
// Helper function that checks if a view item is a toggleable button.
|
|
604
|
-
// Returns the button if it's toggleable - otherwise, returns null.
|
|
605
|
-
const pickListItemToggleableButtonIfPresent = (item) => {
|
|
606
|
-
const listItemButtonView = pickListItemButtonIfPresent(item);
|
|
607
|
-
// Only return buttons that are configured as toggleable.
|
|
608
|
-
if (!listItemButtonView || !listItemButtonView.isToggleable) {
|
|
609
|
-
return null;
|
|
610
|
-
}
|
|
611
|
-
return listItemButtonView;
|
|
612
|
-
};
|
|
613
|
-
// Updates all buttons in the list to either allocate space for check marks or not.
|
|
614
|
-
// This ensures all buttons are properly aligned regardless of their toggleable state.
|
|
615
|
-
const updateAllButtonsCheckSpace = (hasSpace) => {
|
|
616
|
-
for (const listItem of listItems) {
|
|
617
|
-
const listItemButton = pickListItemButtonIfPresent(listItem);
|
|
618
|
-
if (listItemButton) {
|
|
619
|
-
listItemButton.hasCheckSpace = hasSpace;
|
|
620
|
-
}
|
|
621
|
-
}
|
|
622
|
-
};
|
|
623
|
-
// Listen for changes in the list items collection.
|
|
624
|
-
listItems.on('change', (evt, data) => {
|
|
625
|
-
// Remember the current state - whether we have any toggleable buttons.
|
|
626
|
-
const prevToggleable = toggleableButtonsCount > 0;
|
|
627
|
-
// Process removed items - decrease count for each toggleable button removed.
|
|
628
|
-
for (const item of data.removed) {
|
|
629
|
-
if (pickListItemToggleableButtonIfPresent(item)) {
|
|
630
|
-
toggleableButtonsCount--;
|
|
631
|
-
}
|
|
632
|
-
}
|
|
633
|
-
// Process added items - increase count for each toggleable button added.
|
|
634
|
-
for (const item of data.added) {
|
|
635
|
-
const button = pickListItemButtonIfPresent(item);
|
|
636
|
-
if (!button) {
|
|
637
|
-
continue;
|
|
638
|
-
}
|
|
639
|
-
if (button.isToggleable) {
|
|
640
|
-
// Check if the button is toggleable and increase the count.
|
|
641
|
-
toggleableButtonsCount++;
|
|
642
|
-
}
|
|
643
|
-
// Depending on the current state, set the check space for the button.
|
|
644
|
-
button.hasCheckSpace = toggleableButtonsCount > 0;
|
|
645
|
-
}
|
|
646
|
-
// Check if the current state has changed.
|
|
647
|
-
const currentToggleable = toggleableButtonsCount > 0;
|
|
648
|
-
// Only update button alignment if we've crossed the threshold between
|
|
649
|
-
// having no toggleable buttons and having at least one.
|
|
650
|
-
if (prevToggleable !== currentToggleable) {
|
|
651
|
-
updateAllButtonsCheckSpace(currentToggleable);
|
|
652
|
-
}
|
|
653
|
-
});
|
|
654
|
-
}
|