@ckeditor/ckeditor5-table 0.0.0-nightly-next-20260127.0 → 0.0.0-nightly-20260128.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.
- package/build/table.js +5 -0
- package/build/translations/af.js +1 -0
- package/build/translations/ar.js +1 -0
- package/build/translations/ast.js +1 -0
- package/build/translations/az.js +1 -0
- package/build/translations/be.js +1 -0
- package/build/translations/bg.js +1 -0
- package/build/translations/bn.js +1 -0
- package/build/translations/bs.js +1 -0
- package/build/translations/ca.js +1 -0
- package/build/translations/cs.js +1 -0
- package/build/translations/da.js +1 -0
- package/build/translations/de-ch.js +1 -0
- package/build/translations/de.js +1 -0
- package/build/translations/el.js +1 -0
- package/build/translations/en-au.js +1 -0
- package/build/translations/en-gb.js +1 -0
- package/build/translations/eo.js +1 -0
- package/build/translations/es-co.js +1 -0
- package/build/translations/es.js +1 -0
- package/build/translations/et.js +1 -0
- package/build/translations/eu.js +1 -0
- package/build/translations/fa.js +1 -0
- package/build/translations/fi.js +1 -0
- package/build/translations/fr.js +1 -0
- package/build/translations/gl.js +1 -0
- package/build/translations/gu.js +1 -0
- package/build/translations/he.js +1 -0
- package/build/translations/hi.js +1 -0
- package/build/translations/hr.js +1 -0
- package/build/translations/hu.js +1 -0
- package/build/translations/hy.js +1 -0
- package/build/translations/id.js +1 -0
- package/build/translations/it.js +1 -0
- package/build/translations/ja.js +1 -0
- package/build/translations/jv.js +1 -0
- package/build/translations/kk.js +1 -0
- package/build/translations/km.js +1 -0
- package/build/translations/kn.js +1 -0
- package/build/translations/ko.js +1 -0
- package/build/translations/ku.js +1 -0
- package/build/translations/lt.js +1 -0
- package/build/translations/lv.js +1 -0
- package/build/translations/ms.js +1 -0
- package/build/translations/nb.js +1 -0
- package/build/translations/ne.js +1 -0
- package/build/translations/nl.js +1 -0
- package/build/translations/no.js +1 -0
- package/build/translations/oc.js +1 -0
- package/build/translations/pl.js +1 -0
- package/build/translations/pt-br.js +1 -0
- package/build/translations/pt.js +1 -0
- package/build/translations/ro.js +1 -0
- package/build/translations/ru.js +1 -0
- package/build/translations/si.js +1 -0
- package/build/translations/sk.js +1 -0
- package/build/translations/sl.js +1 -0
- package/build/translations/sq.js +1 -0
- package/build/translations/sr-latn.js +1 -0
- package/build/translations/sr.js +1 -0
- package/build/translations/sv.js +1 -0
- package/build/translations/th.js +1 -0
- package/build/translations/ti.js +1 -0
- package/build/translations/tk.js +1 -0
- package/build/translations/tr.js +1 -0
- package/build/translations/tt.js +1 -0
- package/build/translations/ug.js +1 -0
- package/build/translations/uk.js +1 -0
- package/build/translations/ur.js +1 -0
- package/build/translations/uz.js +1 -0
- package/build/translations/vi.js +1 -0
- package/build/translations/zh-cn.js +1 -0
- package/build/translations/zh.js +1 -0
- package/ckeditor5-metadata.json +7 -16
- package/dist/index-content.css +30 -30
- package/dist/index-editor.css +170 -104
- package/dist/index.css +237 -147
- package/dist/index.css.map +1 -1
- package/dist/index.js +2433 -448
- package/dist/index.js.map +1 -1
- package/lang/contexts.json +82 -0
- package/lang/translations/af.po +332 -0
- package/lang/translations/ar.po +332 -0
- package/lang/translations/ast.po +332 -0
- package/lang/translations/az.po +332 -0
- package/lang/translations/be.po +332 -0
- package/lang/translations/bg.po +332 -0
- package/lang/translations/bn.po +334 -0
- package/lang/translations/bs.po +332 -0
- package/lang/translations/ca.po +332 -0
- package/lang/translations/cs.po +332 -0
- package/lang/translations/da.po +332 -0
- package/lang/translations/de-ch.po +332 -0
- package/lang/translations/de.po +332 -0
- package/lang/translations/el.po +332 -0
- package/lang/translations/en-au.po +332 -0
- package/lang/translations/en-gb.po +332 -0
- package/lang/translations/en.po +332 -0
- package/lang/translations/eo.po +332 -0
- package/lang/translations/es-co.po +332 -0
- package/lang/translations/es.po +332 -0
- package/lang/translations/et.po +332 -0
- package/lang/translations/eu.po +332 -0
- package/lang/translations/fa.po +332 -0
- package/lang/translations/fi.po +332 -0
- package/lang/translations/fr.po +332 -0
- package/lang/translations/gl.po +332 -0
- package/lang/translations/gu.po +332 -0
- package/lang/translations/he.po +332 -0
- package/lang/translations/hi.po +332 -0
- package/lang/translations/hr.po +332 -0
- package/lang/translations/hu.po +332 -0
- package/lang/translations/hy.po +332 -0
- package/lang/translations/id.po +332 -0
- package/lang/translations/it.po +332 -0
- package/lang/translations/ja.po +332 -0
- package/lang/translations/jv.po +332 -0
- package/lang/translations/kk.po +332 -0
- package/lang/translations/km.po +332 -0
- package/lang/translations/kn.po +332 -0
- package/lang/translations/ko.po +332 -0
- package/lang/translations/ku.po +332 -0
- package/lang/translations/lt.po +332 -0
- package/lang/translations/lv.po +332 -0
- package/lang/translations/ms.po +332 -0
- package/lang/translations/nb.po +332 -0
- package/lang/translations/ne.po +332 -0
- package/lang/translations/nl.po +332 -0
- package/lang/translations/no.po +332 -0
- package/lang/translations/oc.po +332 -0
- package/lang/translations/pl.po +332 -0
- package/lang/translations/pt-br.po +332 -0
- package/lang/translations/pt.po +332 -0
- package/lang/translations/ro.po +332 -0
- package/lang/translations/ru.po +332 -0
- package/lang/translations/si.po +332 -0
- package/lang/translations/sk.po +332 -0
- package/lang/translations/sl.po +332 -0
- package/lang/translations/sq.po +332 -0
- package/lang/translations/sr-latn.po +332 -0
- package/lang/translations/sr.po +332 -0
- package/lang/translations/sv.po +332 -0
- package/lang/translations/th.po +332 -0
- package/lang/translations/ti.po +332 -0
- package/lang/translations/tk.po +332 -0
- package/lang/translations/tr.po +332 -0
- package/lang/translations/tt.po +332 -0
- package/lang/translations/ug.po +332 -0
- package/lang/translations/uk.po +332 -0
- package/lang/translations/ur.po +332 -0
- package/lang/translations/uz.po +332 -0
- package/lang/translations/vi.po +332 -0
- package/lang/translations/zh-cn.po +332 -0
- package/lang/translations/zh.po +332 -0
- package/package.json +49 -26
- package/{dist → src}/augmentation.d.ts +15 -0
- package/src/augmentation.js +5 -0
- package/{dist → src}/commands/insertcolumncommand.d.ts +1 -1
- package/src/commands/insertcolumncommand.js +71 -0
- package/{dist → src}/commands/insertrowcommand.d.ts +1 -1
- package/src/commands/insertrowcommand.js +70 -0
- package/{dist → src}/commands/inserttablecommand.d.ts +1 -1
- package/src/commands/inserttablecommand.js +69 -0
- package/{dist → src}/commands/inserttablelayoutcommand.d.ts +1 -1
- package/src/commands/inserttablelayoutcommand.js +65 -0
- package/{dist → src}/commands/mergecellcommand.d.ts +3 -3
- package/src/commands/mergecellcommand.js +206 -0
- package/{dist → src}/commands/mergecellscommand.d.ts +1 -1
- package/src/commands/mergecellscommand.js +94 -0
- package/{dist → src}/commands/removecolumncommand.d.ts +1 -1
- package/src/commands/removecolumncommand.js +109 -0
- package/{dist → src}/commands/removerowcommand.d.ts +1 -1
- package/src/commands/removerowcommand.js +82 -0
- package/{dist → src}/commands/selectcolumncommand.d.ts +1 -1
- package/src/commands/selectcolumncommand.js +60 -0
- package/{dist → src}/commands/selectrowcommand.d.ts +1 -1
- package/src/commands/selectrowcommand.js +56 -0
- package/{dist → src}/commands/setheadercolumncommand.d.ts +1 -1
- package/src/commands/setheadercolumncommand.js +76 -0
- package/{dist → src}/commands/setheaderrowcommand.d.ts +1 -1
- package/src/commands/setheaderrowcommand.js +83 -0
- package/{dist → src}/commands/splitcellcommand.d.ts +1 -1
- package/src/commands/splitcellcommand.js +58 -0
- package/{dist → src}/converters/downcast.d.ts +2 -2
- package/src/converters/downcast.js +298 -0
- package/{dist → src}/converters/table-caption-post-fixer.d.ts +1 -1
- package/src/converters/table-caption-post-fixer.js +55 -0
- package/{dist → src}/converters/table-cell-paragraph-post-fixer.d.ts +1 -1
- package/src/converters/table-cell-paragraph-post-fixer.js +109 -0
- package/{dist → src}/converters/table-cell-refresh-handler.d.ts +1 -1
- package/src/converters/table-cell-refresh-handler.js +47 -0
- package/{dist → src}/converters/table-headings-refresh-handler.d.ts +1 -1
- package/src/converters/table-headings-refresh-handler.js +51 -0
- package/{dist → src}/converters/table-layout-post-fixer.d.ts +1 -1
- package/src/converters/table-layout-post-fixer.js +369 -0
- package/{dist → src}/converters/tableproperties.d.ts +2 -2
- package/src/converters/tableproperties.js +444 -0
- package/{dist → src}/converters/upcasttable.d.ts +1 -1
- package/src/converters/upcasttable.js +385 -0
- package/{dist → src}/index.d.ts +5 -2
- package/src/index.js +98 -0
- package/{dist → src}/plaintableoutput.d.ts +1 -1
- package/src/plaintableoutput.js +49 -0
- package/{dist → src}/table.d.ts +2 -2
- package/src/table.js +50 -0
- package/{dist → src}/tablecaption/tablecaptionediting.d.ts +2 -2
- package/src/tablecaption/tablecaptionediting.js +136 -0
- package/{dist → src}/tablecaption/tablecaptionui.d.ts +1 -1
- package/src/tablecaption/tablecaptionui.js +64 -0
- package/{dist → src}/tablecaption/toggletablecaptioncommand.d.ts +1 -1
- package/src/tablecaption/toggletablecaptioncommand.js +105 -0
- package/{dist → src}/tablecaption/utils.d.ts +1 -1
- package/src/tablecaption/utils.js +61 -0
- package/{dist → src}/tablecaption.d.ts +1 -1
- package/src/tablecaption.js +34 -0
- package/{dist → src}/tablecellproperties/commands/tablecellbackgroundcolorcommand.d.ts +1 -1
- package/src/tablecellproperties/commands/tablecellbackgroundcolorcommand.js +30 -0
- package/{dist → src}/tablecellproperties/commands/tablecellbordercolorcommand.d.ts +2 -2
- package/src/tablecellproperties/commands/tablecellbordercolorcommand.js +44 -0
- package/{dist → src}/tablecellproperties/commands/tablecellborderstylecommand.d.ts +2 -2
- package/src/tablecellproperties/commands/tablecellborderstylecommand.js +44 -0
- package/{dist → src}/tablecellproperties/commands/tablecellborderwidthcommand.d.ts +2 -2
- package/src/tablecellproperties/commands/tablecellborderwidthcommand.js +64 -0
- package/{dist → src}/tablecellproperties/commands/tablecellheightcommand.d.ts +1 -1
- package/src/tablecellproperties/commands/tablecellheightcommand.js +51 -0
- package/{dist → src}/tablecellproperties/commands/tablecellhorizontalalignmentcommand.d.ts +1 -1
- package/src/tablecellproperties/commands/tablecellhorizontalalignmentcommand.js +30 -0
- package/{dist → src}/tablecellproperties/commands/tablecellpaddingcommand.d.ts +2 -2
- package/src/tablecellproperties/commands/tablecellpaddingcommand.js +64 -0
- package/{dist → src}/tablecellproperties/commands/tablecellpropertycommand.d.ts +2 -2
- package/src/tablecellproperties/commands/tablecellpropertycommand.js +138 -0
- package/{dist → src}/tablecellproperties/commands/tablecelltypecommand.d.ts +6 -2
- package/src/tablecellproperties/commands/tablecelltypecommand.js +167 -0
- package/{dist → src}/tablecellproperties/commands/tablecellverticalalignmentcommand.d.ts +1 -1
- package/src/tablecellproperties/commands/tablecellverticalalignmentcommand.js +38 -0
- package/{dist → src}/tablecellproperties/tablecellpropertiesediting.d.ts +1 -1
- package/src/tablecellproperties/tablecellpropertiesediting.js +412 -0
- package/{dist → src}/tablecellproperties/tablecellpropertiesui.d.ts +2 -2
- package/src/tablecellproperties/tablecellpropertiesui.js +385 -0
- package/src/tablecellproperties/tablecellpropertiesuiexperimental.d.ts +128 -0
- package/src/tablecellproperties/tablecellpropertiesuiexperimental.js +408 -0
- package/src/tablecellproperties/ui/tablecellpropertiesview.d.ts +229 -0
- package/src/tablecellproperties/ui/tablecellpropertiesview.js +612 -0
- package/{dist/tablecellproperties/ui/tablecellpropertiesview.d.ts → src/tablecellproperties/ui/tablecellpropertiesviewexperimental.d.ts} +12 -11
- package/src/tablecellproperties/ui/tablecellpropertiesviewexperimental.js +744 -0
- package/{dist → src}/tablecellproperties.d.ts +1 -1
- package/src/tablecellproperties.js +40 -0
- package/{dist → src}/tablecellwidth/commands/tablecellwidthcommand.d.ts +1 -1
- package/src/tablecellwidth/commands/tablecellwidthcommand.js +51 -0
- package/{dist → src}/tablecellwidth/tablecellwidthediting.d.ts +1 -1
- package/src/tablecellwidth/tablecellwidthediting.js +53 -0
- package/{dist → src}/tableclipboard.d.ts +3 -3
- package/src/tableclipboard.js +500 -0
- package/src/tablecolumnresize/constants.js +33 -0
- package/{dist → src}/tablecolumnresize/converters.d.ts +1 -1
- package/src/tablecolumnresize/converters.js +62 -0
- package/{dist → src}/tablecolumnresize/tablecolumnresizeediting.d.ts +2 -2
- package/src/tablecolumnresize/tablecolumnresizeediting.js +734 -0
- package/{dist → src}/tablecolumnresize/tablewidthscommand.d.ts +2 -2
- package/src/tablecolumnresize/tablewidthscommand.js +61 -0
- package/{dist → src}/tablecolumnresize/utils.d.ts +2 -2
- package/src/tablecolumnresize/utils.js +370 -0
- package/{dist → src}/tablecolumnresize.d.ts +1 -1
- package/src/tablecolumnresize.js +36 -0
- package/{dist → src}/tableconfig.d.ts +6 -26
- package/src/tableconfig.js +5 -0
- package/{dist → src}/tableediting.d.ts +3 -3
- package/src/tableediting.js +246 -0
- package/{dist → src}/tablekeyboard.d.ts +3 -3
- package/src/tablekeyboard.js +273 -0
- package/{dist → src}/tablelayout/commands/tabletypecommand.d.ts +1 -1
- package/src/tablelayout/commands/tabletypecommand.js +68 -0
- package/{dist → src}/tablelayout/tablelayoutediting.d.ts +1 -1
- package/src/tablelayout/tablelayoutediting.js +295 -0
- package/{dist → src}/tablelayout/tablelayoutui.d.ts +1 -1
- package/src/tablelayout/tablelayoutui.js +196 -0
- package/{dist → src}/tablelayout.d.ts +1 -1
- package/src/tablelayout.js +37 -0
- package/{dist → src}/tablemouse/mouseeventsobserver.d.ts +1 -1
- package/src/tablemouse/mouseeventsobserver.js +34 -0
- package/{dist → src}/tablemouse.d.ts +1 -1
- package/src/tablemouse.js +178 -0
- package/{dist → src}/tableproperties/commands/tablealignmentcommand.d.ts +1 -1
- package/src/tableproperties/commands/tablealignmentcommand.js +30 -0
- package/{dist → src}/tableproperties/commands/tablebackgroundcolorcommand.d.ts +1 -1
- package/src/tableproperties/commands/tablebackgroundcolorcommand.js +30 -0
- package/{dist → src}/tableproperties/commands/tablebordercolorcommand.d.ts +2 -2
- package/src/tableproperties/commands/tablebordercolorcommand.js +44 -0
- package/{dist → src}/tableproperties/commands/tableborderstylecommand.d.ts +2 -2
- package/src/tableproperties/commands/tableborderstylecommand.js +44 -0
- package/{dist → src}/tableproperties/commands/tableborderwidthcommand.d.ts +2 -2
- package/src/tableproperties/commands/tableborderwidthcommand.js +64 -0
- package/{dist → src}/tableproperties/commands/tableheightcommand.d.ts +1 -1
- package/src/tableproperties/commands/tableheightcommand.js +54 -0
- package/{dist → src}/tableproperties/commands/tablepropertycommand.d.ts +2 -2
- package/src/tableproperties/commands/tablepropertycommand.js +103 -0
- package/{dist → src}/tableproperties/commands/tablewidthcommand.d.ts +1 -1
- package/src/tableproperties/commands/tablewidthcommand.js +54 -0
- package/{dist → src}/tableproperties/tablepropertiesediting.d.ts +1 -1
- package/src/tableproperties/tablepropertiesediting.js +546 -0
- package/{dist → src}/tableproperties/tablepropertiesui.d.ts +2 -2
- package/src/tableproperties/tablepropertiesui.js +374 -0
- package/src/tableproperties/tablepropertiesuiexperimental.d.ts +136 -0
- package/src/tableproperties/tablepropertiesuiexperimental.js +375 -0
- package/{dist → src}/tableproperties/ui/tablepropertiesview.d.ts +2 -10
- package/src/tableproperties/ui/tablepropertiesview.js +520 -0
- package/src/tableproperties/ui/tablepropertiesviewexperimental.d.ts +216 -0
- package/src/tableproperties/ui/tablepropertiesviewexperimental.js +544 -0
- package/{dist → src}/tableproperties.d.ts +1 -1
- package/src/tableproperties.js +40 -0
- package/{dist → src}/tableselection.d.ts +2 -2
- package/src/tableselection.js +323 -0
- package/{dist → src}/tabletoolbar.d.ts +2 -2
- package/src/tabletoolbar.js +63 -0
- package/{dist → src}/tableui.d.ts +1 -1
- package/src/tableui.js +335 -0
- package/{dist → src}/tableutils.d.ts +2 -2
- package/src/tableutils.js +1282 -0
- package/{dist → src}/tablewalker.d.ts +1 -1
- package/src/tablewalker.js +489 -0
- package/{dist → src}/ui/colorinputview.d.ts +2 -2
- package/src/ui/colorinputview.js +305 -0
- package/{dist → src}/ui/inserttableview.d.ts +2 -2
- package/src/ui/inserttableview.js +192 -0
- package/{dist → src}/utils/common.d.ts +2 -2
- package/src/utils/common.js +118 -0
- package/{dist → src}/utils/structure.d.ts +1 -1
- package/src/utils/structure.js +452 -0
- package/{dist → src}/utils/table-properties.d.ts +1 -1
- package/src/utils/table-properties.js +121 -0
- package/{dist → src}/utils/ui/contextualballoon.d.ts +2 -2
- package/src/utils/ui/contextualballoon.js +111 -0
- package/{dist → src}/utils/ui/table-properties.d.ts +2 -2
- package/src/utils/ui/table-properties.js +390 -0
- package/src/utils/ui/table-propertiesexperimental.d.ts +215 -0
- package/src/utils/ui/table-propertiesexperimental.js +391 -0
- package/{dist → src}/utils/ui/widget.d.ts +1 -1
- package/src/utils/ui/widget.js +56 -0
- package/theme/colorinput.css +39 -0
- package/theme/formrow-experimental.css +15 -0
- package/theme/formrow.css +13 -0
- package/theme/inserttable.css +10 -0
- package/theme/table.css +144 -0
- package/theme/tablecaption.css +66 -0
- package/theme/tablecellproperties-experimental.css +4 -0
- package/theme/tablecellproperties.css +28 -0
- package/theme/tablecolumnresize.css +62 -0
- package/theme/tableediting.css +10 -0
- package/theme/tableform-experimental.css +61 -0
- package/theme/tableform.css +64 -0
- package/theme/tablelayout.css +74 -0
- package/theme/tableproperties-experimental.css +78 -0
- package/theme/tableproperties.css +18 -0
- package/theme/tableselection.css +10 -0
- package/dist/tablecellproperties/tablecellpropertiesutils.d.ts +0 -18
- /package/{dist → src}/tablecolumnresize/constants.d.ts +0 -0
|
@@ -0,0 +1,58 @@
|
|
|
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 table/commands/splitcellcommand
|
|
7
|
+
*/
|
|
8
|
+
import { Command } from 'ckeditor5/src/core.js';
|
|
9
|
+
/**
|
|
10
|
+
* The split cell command.
|
|
11
|
+
*
|
|
12
|
+
* The command is registered by {@link module:table/tableediting~TableEditing} as the `'splitTableCellVertically'`
|
|
13
|
+
* and `'splitTableCellHorizontally'` editor commands.
|
|
14
|
+
*
|
|
15
|
+
* You can split any cell vertically or horizontally by executing this command. For example, to split the selected table cell vertically:
|
|
16
|
+
*
|
|
17
|
+
* ```ts
|
|
18
|
+
* editor.execute( 'splitTableCellVertically' );
|
|
19
|
+
* ```
|
|
20
|
+
*/
|
|
21
|
+
export class SplitCellCommand extends Command {
|
|
22
|
+
/**
|
|
23
|
+
* The direction that indicates which cell will be split.
|
|
24
|
+
*/
|
|
25
|
+
direction;
|
|
26
|
+
/**
|
|
27
|
+
* Creates a new `SplitCellCommand` instance.
|
|
28
|
+
*
|
|
29
|
+
* @param editor The editor on which this command will be used.
|
|
30
|
+
* @param options.direction Indicates whether the command should split cells `'horizontally'` or `'vertically'`.
|
|
31
|
+
*/
|
|
32
|
+
constructor(editor, options = {}) {
|
|
33
|
+
super(editor);
|
|
34
|
+
this.direction = options.direction || 'horizontally';
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* @inheritDoc
|
|
38
|
+
*/
|
|
39
|
+
refresh() {
|
|
40
|
+
const tableUtils = this.editor.plugins.get('TableUtils');
|
|
41
|
+
const selectedCells = tableUtils.getSelectionAffectedTableCells(this.editor.model.document.selection);
|
|
42
|
+
this.isEnabled = selectedCells.length === 1;
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* @inheritDoc
|
|
46
|
+
*/
|
|
47
|
+
execute() {
|
|
48
|
+
const tableUtils = this.editor.plugins.get('TableUtils');
|
|
49
|
+
const tableCell = tableUtils.getSelectionAffectedTableCells(this.editor.model.document.selection)[0];
|
|
50
|
+
const isHorizontal = this.direction === 'horizontally';
|
|
51
|
+
if (isHorizontal) {
|
|
52
|
+
tableUtils.splitCellHorizontally(tableCell, 2);
|
|
53
|
+
}
|
|
54
|
+
else {
|
|
55
|
+
tableUtils.splitCellVertically(tableCell, 2);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
}
|
|
@@ -5,8 +5,8 @@
|
|
|
5
5
|
/**
|
|
6
6
|
* @module table/converters/downcast
|
|
7
7
|
*/
|
|
8
|
-
import { type Editor } from '
|
|
9
|
-
import type { ViewElement, ModelElement, DowncastElementCreatorFunction, DowncastConversionApi } from '
|
|
8
|
+
import { type Editor } from 'ckeditor5/src/core.js';
|
|
9
|
+
import type { ViewElement, ModelElement, DowncastElementCreatorFunction, DowncastConversionApi } from 'ckeditor5/src/engine.js';
|
|
10
10
|
import { type TableUtils } from '../tableutils.js';
|
|
11
11
|
import type { TableConversionAdditionalSlot } from '../tableediting.js';
|
|
12
12
|
/**
|
|
@@ -0,0 +1,298 @@
|
|
|
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
|
+
import { toWidget, toWidgetEditable } from 'ckeditor5/src/widget.js';
|
|
6
|
+
import { downcastTableAlignmentConfig } from './tableproperties.js';
|
|
7
|
+
import { getNormalizedDefaultTableProperties } from '../utils/table-properties.js';
|
|
8
|
+
import { TableWalker } from '../tablewalker.js';
|
|
9
|
+
/**
|
|
10
|
+
* Model table element to view table element conversion helper.
|
|
11
|
+
*
|
|
12
|
+
* @internal
|
|
13
|
+
*/
|
|
14
|
+
export function downcastTable(tableUtils, options) {
|
|
15
|
+
return (table, { writer }) => {
|
|
16
|
+
const headingRows = table.getAttribute('headingRows') || 0;
|
|
17
|
+
const tableElement = writer.createContainerElement('table', null, []);
|
|
18
|
+
const figureElement = writer.createContainerElement('figure', { class: 'table' }, tableElement);
|
|
19
|
+
// Table head slot.
|
|
20
|
+
if (headingRows > 0) {
|
|
21
|
+
writer.insert(writer.createPositionAt(tableElement, 'end'), writer.createContainerElement('thead', null, writer.createSlot(element => element.is('element', 'tableRow') && element.index < headingRows)));
|
|
22
|
+
}
|
|
23
|
+
// Table body slot.
|
|
24
|
+
if (headingRows < tableUtils.getRows(table)) {
|
|
25
|
+
writer.insert(writer.createPositionAt(tableElement, 'end'), writer.createContainerElement('tbody', null, writer.createSlot(element => element.is('element', 'tableRow') && element.index >= headingRows)));
|
|
26
|
+
}
|
|
27
|
+
// Dynamic slots.
|
|
28
|
+
for (const { positionOffset, filter } of options.additionalSlots) {
|
|
29
|
+
writer.insert(writer.createPositionAt(tableElement, positionOffset), writer.createSlot(filter));
|
|
30
|
+
}
|
|
31
|
+
// Create a slot with items that don't fit into the table.
|
|
32
|
+
writer.insert(writer.createPositionAt(tableElement, 'after'), writer.createSlot(element => {
|
|
33
|
+
if (element.is('element', 'tableRow')) {
|
|
34
|
+
return false;
|
|
35
|
+
}
|
|
36
|
+
return !options.additionalSlots.some(({ filter }) => filter(element));
|
|
37
|
+
}));
|
|
38
|
+
return options.asWidget ? toTableWidget(figureElement, writer) : figureElement;
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Model table row element to view `<tr>` element conversion helper.
|
|
43
|
+
*
|
|
44
|
+
* @internal
|
|
45
|
+
* @returns Element creator.
|
|
46
|
+
*/
|
|
47
|
+
export function downcastRow() {
|
|
48
|
+
return (tableRow, { writer }) => {
|
|
49
|
+
return tableRow.isEmpty ?
|
|
50
|
+
writer.createEmptyElement('tr') :
|
|
51
|
+
writer.createContainerElement('tr');
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
/**
|
|
55
|
+
* Model table cell element to view `<td>` or `<th>` element conversion helper.
|
|
56
|
+
*
|
|
57
|
+
* This conversion helper will create proper `<th>` elements for table cells that are in the heading section (heading row or column)
|
|
58
|
+
* and `<td>` otherwise.
|
|
59
|
+
*
|
|
60
|
+
* @internal
|
|
61
|
+
* @param options.asWidget If set to `true`, the downcast conversion will produce a widget.
|
|
62
|
+
* @param options.cellTypeEnabled If returns `true`, the downcast conversion will use the `tableCellType` attribute to determine cell type.
|
|
63
|
+
* @returns Element creator.
|
|
64
|
+
*/
|
|
65
|
+
export function downcastCell(options) {
|
|
66
|
+
return (tableCell, { writer }) => {
|
|
67
|
+
// If the table cell type feature is enabled, then we can simply check the cell type attribute.
|
|
68
|
+
if (options.cellTypeEnabled?.()) {
|
|
69
|
+
const cellElementName = (tableCell.getAttribute('tableCellType') === 'header' ?
|
|
70
|
+
'th' :
|
|
71
|
+
'td');
|
|
72
|
+
return createCellElement(writer, cellElementName);
|
|
73
|
+
}
|
|
74
|
+
// If the table cell type feature is not enabled, we should iterate through the table structure
|
|
75
|
+
// to determine whether the cell is in the heading section.
|
|
76
|
+
const tableRow = tableCell.parent;
|
|
77
|
+
const table = tableRow.parent;
|
|
78
|
+
const rowIndex = table.getChildIndex(tableRow);
|
|
79
|
+
const tableWalker = new TableWalker(table, { row: rowIndex });
|
|
80
|
+
const headingRows = table.getAttribute('headingRows') || 0;
|
|
81
|
+
const headingColumns = table.getAttribute('headingColumns') || 0;
|
|
82
|
+
let result = null;
|
|
83
|
+
// We need to iterate over a table in order to get proper row & column values from a walker.
|
|
84
|
+
for (const tableSlot of tableWalker) {
|
|
85
|
+
if (tableSlot.cell == tableCell) {
|
|
86
|
+
const isHeading = tableSlot.row < headingRows || tableSlot.column < headingColumns;
|
|
87
|
+
result = createCellElement(writer, isHeading ? 'th' : 'td');
|
|
88
|
+
break;
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
return result;
|
|
92
|
+
};
|
|
93
|
+
function createCellElement(writer, name) {
|
|
94
|
+
return (options.asWidget ?
|
|
95
|
+
toWidgetEditable(writer.createEditableElement(name), writer, { withAriaRole: false }) :
|
|
96
|
+
writer.createContainerElement(name));
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* Overrides paragraph inside table cell conversion.
|
|
101
|
+
*
|
|
102
|
+
* This converter:
|
|
103
|
+
* * should be used to override default paragraph conversion.
|
|
104
|
+
* * It will only convert `<paragraph>` placed directly inside `<tableCell>`.
|
|
105
|
+
* * For a single paragraph without attributes it returns `<span>` to simulate data table.
|
|
106
|
+
* * For all other cases it returns `<p>` element.
|
|
107
|
+
*
|
|
108
|
+
* @internal
|
|
109
|
+
* @param options.asWidget If set to `true`, the downcast conversion will produce a widget.
|
|
110
|
+
* @returns Element creator.
|
|
111
|
+
*/
|
|
112
|
+
export function convertParagraphInTableCell(options = {}) {
|
|
113
|
+
return (modelElement, { writer }) => {
|
|
114
|
+
if (!modelElement.parent.is('element', 'tableCell')) {
|
|
115
|
+
return null;
|
|
116
|
+
}
|
|
117
|
+
if (!isSingleParagraphWithoutAttributes(modelElement)) {
|
|
118
|
+
return null;
|
|
119
|
+
}
|
|
120
|
+
if (options.asWidget) {
|
|
121
|
+
return writer.createContainerElement('span', { class: 'ck-table-bogus-paragraph' });
|
|
122
|
+
}
|
|
123
|
+
else {
|
|
124
|
+
// Using `<p>` in case there are some markers on it and transparentRendering will render it anyway.
|
|
125
|
+
const viewElement = writer.createContainerElement('p');
|
|
126
|
+
writer.setCustomProperty('dataPipeline:transparentRendering', true, viewElement);
|
|
127
|
+
return viewElement;
|
|
128
|
+
}
|
|
129
|
+
};
|
|
130
|
+
}
|
|
131
|
+
/**
|
|
132
|
+
* Checks if given model `<paragraph>` is an only child of a parent (`<tableCell>`) and if it has any attribute set.
|
|
133
|
+
*
|
|
134
|
+
* The paragraph should be converted in the editing view to:
|
|
135
|
+
*
|
|
136
|
+
* * If returned `true` - to a `<span class="ck-table-bogus-paragraph">`
|
|
137
|
+
* * If returned `false` - to a `<p>`
|
|
138
|
+
*
|
|
139
|
+
* @internal
|
|
140
|
+
*/
|
|
141
|
+
export function isSingleParagraphWithoutAttributes(modelElement) {
|
|
142
|
+
const tableCell = modelElement.parent;
|
|
143
|
+
const isSingleParagraph = tableCell.childCount == 1;
|
|
144
|
+
return isSingleParagraph && !hasAnyAttribute(modelElement);
|
|
145
|
+
}
|
|
146
|
+
/**
|
|
147
|
+
* Converts a given {@link module:engine/view/element~ViewElement} to a table widget:
|
|
148
|
+
* * Adds a {@link module:engine/view/element~ViewElement#_setCustomProperty custom property}
|
|
149
|
+
* allowing to recognize the table widget element.
|
|
150
|
+
* * Calls the {@link module:widget/utils~toWidget} function with the proper element's label creator.
|
|
151
|
+
*
|
|
152
|
+
* @param writer An instance of the view writer.
|
|
153
|
+
* @param label The element's label. It will be concatenated with the table `alt` attribute if one is present.
|
|
154
|
+
*/
|
|
155
|
+
function toTableWidget(viewElement, writer) {
|
|
156
|
+
writer.setCustomProperty('table', true, viewElement);
|
|
157
|
+
return toWidget(viewElement, writer, { hasSelectionHandle: true });
|
|
158
|
+
}
|
|
159
|
+
/**
|
|
160
|
+
* Checks if an element has any attributes set.
|
|
161
|
+
*/
|
|
162
|
+
function hasAnyAttribute(element) {
|
|
163
|
+
for (const attributeKey of element.getAttributeKeys()) {
|
|
164
|
+
// Ignore selection attributes stored on block elements.
|
|
165
|
+
if (attributeKey.startsWith('selection:') || attributeKey == 'htmlEmptyBlock') {
|
|
166
|
+
continue;
|
|
167
|
+
}
|
|
168
|
+
return true;
|
|
169
|
+
}
|
|
170
|
+
return false;
|
|
171
|
+
}
|
|
172
|
+
/**
|
|
173
|
+
* Downcasts a plain table (also used in the clipboard pipeline).
|
|
174
|
+
*/
|
|
175
|
+
export function convertPlainTable(editor) {
|
|
176
|
+
return (table, conversionApi) => {
|
|
177
|
+
const hasPlainTableOutput = editor.plugins.has('PlainTableOutput');
|
|
178
|
+
const isClipboardPipeline = conversionApi.options.isClipboardPipeline;
|
|
179
|
+
const useExtendedAlignment = editor.config.get('experimentalFlags.useExtendedTableBlockAlignment');
|
|
180
|
+
if (!hasPlainTableOutput && !(useExtendedAlignment && isClipboardPipeline)) {
|
|
181
|
+
return null;
|
|
182
|
+
}
|
|
183
|
+
return downcastPlainTable(table, conversionApi, editor);
|
|
184
|
+
};
|
|
185
|
+
}
|
|
186
|
+
/**
|
|
187
|
+
* Downcasts a plain table caption (also used in the clipboard pipeline).
|
|
188
|
+
*/
|
|
189
|
+
export function convertPlainTableCaption(editor) {
|
|
190
|
+
return (modelElement, { writer, options }) => {
|
|
191
|
+
const hasPlainTableOutput = editor.plugins.has('PlainTableOutput');
|
|
192
|
+
const isClipboardPipeline = options.isClipboardPipeline;
|
|
193
|
+
const useExtendedAlignment = editor.config.get('experimentalFlags.useExtendedTableBlockAlignment');
|
|
194
|
+
if (!hasPlainTableOutput && !(useExtendedAlignment && isClipboardPipeline)) {
|
|
195
|
+
return null;
|
|
196
|
+
}
|
|
197
|
+
if (modelElement.parent.name === 'table') {
|
|
198
|
+
return writer.createContainerElement('caption');
|
|
199
|
+
}
|
|
200
|
+
return null;
|
|
201
|
+
};
|
|
202
|
+
}
|
|
203
|
+
/**
|
|
204
|
+
* Downcasts a plain table.
|
|
205
|
+
*
|
|
206
|
+
* @param table Table model element.
|
|
207
|
+
* @param conversionApi The conversion API object.
|
|
208
|
+
* @param editor The editor instance.
|
|
209
|
+
* @returns Created element.
|
|
210
|
+
*/
|
|
211
|
+
export function downcastPlainTable(table, conversionApi, editor) {
|
|
212
|
+
const writer = conversionApi.writer;
|
|
213
|
+
const headingRows = table.getAttribute('headingRows') || 0;
|
|
214
|
+
// Table head rows slot.
|
|
215
|
+
const headRowsSlot = writer.createSlot((element) => element.is('element', 'tableRow') && element.index < headingRows);
|
|
216
|
+
// Table body rows slot.
|
|
217
|
+
const bodyRowsSlot = writer.createSlot((element) => element.is('element', 'tableRow') && element.index >= headingRows);
|
|
218
|
+
// Table children slot.
|
|
219
|
+
const childrenSlot = writer.createSlot((element) => !element.is('element', 'tableRow'));
|
|
220
|
+
// Table <thead> element with all the heading rows.
|
|
221
|
+
const theadElement = writer.createContainerElement('thead', null, headRowsSlot);
|
|
222
|
+
// Table <tbody> element with all the body rows.
|
|
223
|
+
const tbodyElement = writer.createContainerElement('tbody', null, bodyRowsSlot);
|
|
224
|
+
// Table contents element containing <thead> and <tbody> when necessary.
|
|
225
|
+
const tableContentElements = [];
|
|
226
|
+
if (headingRows) {
|
|
227
|
+
tableContentElements.push(theadElement);
|
|
228
|
+
}
|
|
229
|
+
if (headingRows < table.childCount) {
|
|
230
|
+
tableContentElements.push(tbodyElement);
|
|
231
|
+
}
|
|
232
|
+
const tableAttributes = { class: 'table' };
|
|
233
|
+
if (editor.plugins.has('TablePropertiesEditing') && conversionApi.options.isClipboardPipeline) {
|
|
234
|
+
const defaultTableProperties = getNormalizedDefaultTableProperties(editor.config.get('table.tableProperties.defaultProperties'), {
|
|
235
|
+
includeAlignmentProperty: true
|
|
236
|
+
});
|
|
237
|
+
const tableAlignment = table.getAttribute('tableAlignment');
|
|
238
|
+
let localDefaultValue = defaultTableProperties.alignment;
|
|
239
|
+
if (table.getAttribute('tableType') === 'layout') {
|
|
240
|
+
localDefaultValue = '';
|
|
241
|
+
}
|
|
242
|
+
const tableAlignmentValue = tableAlignment || localDefaultValue;
|
|
243
|
+
if (tableAlignmentValue) {
|
|
244
|
+
tableAttributes.class += ' ' + downcastTableAlignmentConfig[tableAlignmentValue].className;
|
|
245
|
+
tableAttributes.style = downcastTableAlignmentConfig[tableAlignmentValue].style;
|
|
246
|
+
if (downcastTableAlignmentConfig[tableAlignmentValue].align !== undefined) {
|
|
247
|
+
tableAttributes.align = downcastTableAlignmentConfig[tableAlignmentValue].align;
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
// Create table structure.
|
|
252
|
+
//
|
|
253
|
+
// <table>
|
|
254
|
+
// {children-slot-like-caption}
|
|
255
|
+
// <thead>
|
|
256
|
+
// {table-head-rows-slot}
|
|
257
|
+
// </thead>
|
|
258
|
+
// <tbody>
|
|
259
|
+
// {table-body-rows-slot}
|
|
260
|
+
// </tbody>
|
|
261
|
+
// </table>
|
|
262
|
+
return writer.createContainerElement('table', tableAttributes, [childrenSlot, ...tableContentElements]);
|
|
263
|
+
}
|
|
264
|
+
/**
|
|
265
|
+
* Registers border and background attributes converters for plain tables or when the clipboard pipeline is used.
|
|
266
|
+
*/
|
|
267
|
+
export function downcastTableBorderAndBackgroundAttributes(editor) {
|
|
268
|
+
const modelAttributes = {
|
|
269
|
+
'border-width': 'tableBorderWidth',
|
|
270
|
+
'border-color': 'tableBorderColor',
|
|
271
|
+
'border-style': 'tableBorderStyle',
|
|
272
|
+
'background-color': 'tableBackgroundColor'
|
|
273
|
+
};
|
|
274
|
+
for (const [styleName, modelAttribute] of Object.entries(modelAttributes)) {
|
|
275
|
+
editor.conversion.for('dataDowncast').add(dispatcher => {
|
|
276
|
+
return dispatcher.on(`attribute:${modelAttribute}:table`, (evt, data, conversionApi) => {
|
|
277
|
+
const { item, attributeNewValue } = data;
|
|
278
|
+
const { mapper, writer } = conversionApi;
|
|
279
|
+
const hasPlainTableOutput = editor.plugins.has('PlainTableOutput');
|
|
280
|
+
const isClipboardPipeline = conversionApi.options.isClipboardPipeline;
|
|
281
|
+
const useExtendedAlignment = editor.config.get('experimentalFlags.useExtendedTableBlockAlignment');
|
|
282
|
+
if (!hasPlainTableOutput && !(useExtendedAlignment && isClipboardPipeline)) {
|
|
283
|
+
return;
|
|
284
|
+
}
|
|
285
|
+
if (!conversionApi.consumable.consume(item, evt.name)) {
|
|
286
|
+
return;
|
|
287
|
+
}
|
|
288
|
+
const table = mapper.toViewElement(item);
|
|
289
|
+
if (attributeNewValue) {
|
|
290
|
+
writer.setStyle(styleName, attributeNewValue, table);
|
|
291
|
+
}
|
|
292
|
+
else {
|
|
293
|
+
writer.removeStyle(styleName, table);
|
|
294
|
+
}
|
|
295
|
+
}, { priority: 'high' });
|
|
296
|
+
});
|
|
297
|
+
}
|
|
298
|
+
}
|
|
@@ -0,0 +1,55 @@
|
|
|
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
|
+
* Injects a table caption post-fixer into the model.
|
|
7
|
+
*
|
|
8
|
+
* The role of the table caption post-fixer is to ensure that the table with caption have the correct structure
|
|
9
|
+
* after a {@link module:engine/model/model~Model#change `change()`} block was executed.
|
|
10
|
+
*
|
|
11
|
+
* The correct structure means that:
|
|
12
|
+
*
|
|
13
|
+
* * If there are many caption model element, they are merged into one model.
|
|
14
|
+
* * A final, merged caption model is placed at the end of the table.
|
|
15
|
+
*
|
|
16
|
+
* @internal
|
|
17
|
+
*/
|
|
18
|
+
export function injectTableCaptionPostFixer(model) {
|
|
19
|
+
model.document.registerPostFixer(writer => tableCaptionPostFixer(writer, model));
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* The table caption post-fixer.
|
|
23
|
+
*/
|
|
24
|
+
function tableCaptionPostFixer(writer, model) {
|
|
25
|
+
const changes = model.document.differ.getChanges();
|
|
26
|
+
let wasFixed = false;
|
|
27
|
+
for (const entry of changes) {
|
|
28
|
+
if (entry.type != 'insert') {
|
|
29
|
+
continue;
|
|
30
|
+
}
|
|
31
|
+
const positionParent = entry.position.parent;
|
|
32
|
+
if (positionParent.is('element', 'table') || entry.name == 'table') {
|
|
33
|
+
const table = (entry.name == 'table' ? entry.position.nodeAfter : positionParent);
|
|
34
|
+
const captionsToMerge = Array.from(table.getChildren())
|
|
35
|
+
.filter((child) => child.is('element', 'caption'));
|
|
36
|
+
const firstCaption = captionsToMerge.shift();
|
|
37
|
+
if (!firstCaption) {
|
|
38
|
+
continue;
|
|
39
|
+
}
|
|
40
|
+
// Move all the contents of the captions to the first one.
|
|
41
|
+
for (const caption of captionsToMerge) {
|
|
42
|
+
writer.move(writer.createRangeIn(caption), firstCaption, 'end');
|
|
43
|
+
writer.remove(caption);
|
|
44
|
+
}
|
|
45
|
+
// Make sure the final caption is at the end of the table.
|
|
46
|
+
if (firstCaption.nextSibling) {
|
|
47
|
+
writer.move(writer.createRangeOn(firstCaption), table, 'end');
|
|
48
|
+
wasFixed = true;
|
|
49
|
+
}
|
|
50
|
+
// Do we merged captions and/or moved the single caption to the end of the table?
|
|
51
|
+
wasFixed = !!captionsToMerge.length || wasFixed;
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
return wasFixed;
|
|
55
|
+
}
|
|
@@ -5,7 +5,7 @@
|
|
|
5
5
|
/**
|
|
6
6
|
* @module table/converters/table-cell-paragraph-post-fixer
|
|
7
7
|
*/
|
|
8
|
-
import type { Model } from '
|
|
8
|
+
import type { Model } from 'ckeditor5/src/engine.js';
|
|
9
9
|
/**
|
|
10
10
|
* Injects a table cell post-fixer into the model which inserts a `paragraph` element into empty table cells.
|
|
11
11
|
*
|
|
@@ -0,0 +1,109 @@
|
|
|
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
|
+
* Injects a table cell post-fixer into the model which inserts a `paragraph` element into empty table cells.
|
|
7
|
+
*
|
|
8
|
+
* A table cell must contain at least one block element as a child. An empty table cell will have an empty `paragraph` as a child.
|
|
9
|
+
*
|
|
10
|
+
* ```xml
|
|
11
|
+
* <table>
|
|
12
|
+
* <tableRow>
|
|
13
|
+
* <tableCell></tableCell>
|
|
14
|
+
* </tableRow>
|
|
15
|
+
* </table>
|
|
16
|
+
* ```
|
|
17
|
+
*
|
|
18
|
+
* Will be fixed to:
|
|
19
|
+
*
|
|
20
|
+
* ```xml
|
|
21
|
+
* <table>
|
|
22
|
+
* <tableRow>
|
|
23
|
+
* <tableCell><paragraph></paragraph></tableCell>
|
|
24
|
+
* </tableRow>
|
|
25
|
+
* </table>
|
|
26
|
+
* ```
|
|
27
|
+
*
|
|
28
|
+
* @internal
|
|
29
|
+
*/
|
|
30
|
+
export function injectTableCellParagraphPostFixer(model) {
|
|
31
|
+
model.document.registerPostFixer(writer => tableCellContentsPostFixer(writer, model));
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* The table cell contents post-fixer.
|
|
35
|
+
*/
|
|
36
|
+
function tableCellContentsPostFixer(writer, model) {
|
|
37
|
+
const changes = model.document.differ.getChanges();
|
|
38
|
+
let wasFixed = false;
|
|
39
|
+
for (const entry of changes) {
|
|
40
|
+
if (entry.type == 'insert' && entry.name == 'table') {
|
|
41
|
+
wasFixed = fixTable(entry.position.nodeAfter, writer) || wasFixed;
|
|
42
|
+
}
|
|
43
|
+
if (entry.type == 'insert' && entry.name == 'tableRow') {
|
|
44
|
+
wasFixed = fixTableRow(entry.position.nodeAfter, writer) || wasFixed;
|
|
45
|
+
}
|
|
46
|
+
if (entry.type == 'insert' && entry.name == 'tableCell') {
|
|
47
|
+
wasFixed = fixTableCellContent(entry.position.nodeAfter, writer) || wasFixed;
|
|
48
|
+
}
|
|
49
|
+
if ((entry.type == 'remove' || entry.type == 'insert') && checkTableCellChange(entry)) {
|
|
50
|
+
wasFixed = fixTableCellContent(entry.position.parent, writer) || wasFixed;
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
return wasFixed;
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Fixes all table cells in a table.
|
|
57
|
+
*/
|
|
58
|
+
function fixTable(table, writer) {
|
|
59
|
+
let wasFixed = false;
|
|
60
|
+
for (const row of table.getChildren()) {
|
|
61
|
+
if (row.is('element', 'tableRow')) {
|
|
62
|
+
wasFixed = fixTableRow(row, writer) || wasFixed;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
return wasFixed;
|
|
66
|
+
}
|
|
67
|
+
/**
|
|
68
|
+
* Fixes all table cells in a table row.
|
|
69
|
+
*/
|
|
70
|
+
function fixTableRow(tableRow, writer) {
|
|
71
|
+
let wasFixed = false;
|
|
72
|
+
for (const tableCell of tableRow.getChildren()) {
|
|
73
|
+
wasFixed = fixTableCellContent(tableCell, writer) || wasFixed;
|
|
74
|
+
}
|
|
75
|
+
return wasFixed;
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Fixes all table cell content by:
|
|
79
|
+
* - Adding a paragraph to a table cell without any child.
|
|
80
|
+
* - Wrapping direct $text in a `<paragraph>`.
|
|
81
|
+
*/
|
|
82
|
+
function fixTableCellContent(tableCell, writer) {
|
|
83
|
+
// Insert paragraph to an empty table cell.
|
|
84
|
+
if (tableCell.childCount == 0) {
|
|
85
|
+
// @if CK_DEBUG_TABLE // console.log( 'Post-fixing table: insert paragraph in empty cell.' );
|
|
86
|
+
writer.insertElement('paragraph', tableCell);
|
|
87
|
+
return true;
|
|
88
|
+
}
|
|
89
|
+
// Check table cell children for directly placed text nodes.
|
|
90
|
+
// Temporary solution. See https://github.com/ckeditor/ckeditor5/issues/1464.
|
|
91
|
+
const textNodes = Array.from(tableCell.getChildren()).filter(child => child.is('$text'));
|
|
92
|
+
// @if CK_DEBUG_TABLE // textNodes.length && console.log( 'Post-fixing table: wrap cell content with paragraph.' );
|
|
93
|
+
for (const child of textNodes) {
|
|
94
|
+
writer.wrap(writer.createRangeOn(child), 'paragraph');
|
|
95
|
+
}
|
|
96
|
+
// Return true when there were text nodes to fix.
|
|
97
|
+
return !!textNodes.length;
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* Checks if a differ change should fix the table cell. This happens on:
|
|
101
|
+
* - Removing content from the table cell (i.e. `tableCell` can be left empty).
|
|
102
|
+
* - Adding a text node directly into a table cell.
|
|
103
|
+
*/
|
|
104
|
+
function checkTableCellChange(entry) {
|
|
105
|
+
if (!entry.position.parent.is('element', 'tableCell')) {
|
|
106
|
+
return false;
|
|
107
|
+
}
|
|
108
|
+
return entry.type == 'insert' && entry.name == '$text' || entry.type == 'remove';
|
|
109
|
+
}
|
|
@@ -5,7 +5,7 @@
|
|
|
5
5
|
/**
|
|
6
6
|
* @module table/converters/table-cell-refresh-handler
|
|
7
7
|
*/
|
|
8
|
-
import type { EditingController, Model } from '
|
|
8
|
+
import type { EditingController, Model } from 'ckeditor5/src/engine.js';
|
|
9
9
|
/**
|
|
10
10
|
* A table cell refresh handler which marks the table cell in the differ to have it re-rendered.
|
|
11
11
|
*
|
|
@@ -0,0 +1,47 @@
|
|
|
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
|
+
import { isSingleParagraphWithoutAttributes } from './downcast.js';
|
|
6
|
+
/**
|
|
7
|
+
* A table cell refresh handler which marks the table cell in the differ to have it re-rendered.
|
|
8
|
+
*
|
|
9
|
+
* Model `paragraph` inside a table cell can be rendered as `<span>` or `<p>`. It is rendered as `<span>` if this is the only block
|
|
10
|
+
* element in that table cell and it does not have any attributes. It is rendered as `<p>` otherwise.
|
|
11
|
+
*
|
|
12
|
+
* When table cell content changes, for example a second `paragraph` element is added, we need to ensure that the first `paragraph` is
|
|
13
|
+
* re-rendered so it changes from `<span>` to `<p>`. The easiest way to do it is to re-render the entire table cell.
|
|
14
|
+
*
|
|
15
|
+
* @internal
|
|
16
|
+
*/
|
|
17
|
+
export function tableCellRefreshHandler(model, editing) {
|
|
18
|
+
const differ = model.document.differ;
|
|
19
|
+
// Stores cells to be refreshed, so the table cell will be refreshed once for multiple changes.
|
|
20
|
+
const cellsToCheck = new Set();
|
|
21
|
+
for (const change of differ.getChanges()) {
|
|
22
|
+
const parent = change.type == 'attribute' ? change.range.start.parent : change.position.parent;
|
|
23
|
+
if (parent.is('element', 'tableCell')) {
|
|
24
|
+
cellsToCheck.add(parent);
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
for (const tableCell of cellsToCheck.values()) {
|
|
28
|
+
const paragraphsToRefresh = Array.from(tableCell.getChildren())
|
|
29
|
+
.filter(child => shouldRefresh(child, editing.mapper));
|
|
30
|
+
for (const paragraph of paragraphsToRefresh) {
|
|
31
|
+
editing.reconvertItem(paragraph);
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Check if given model element needs refreshing.
|
|
37
|
+
*/
|
|
38
|
+
function shouldRefresh(child, mapper) {
|
|
39
|
+
if (!child.is('element', 'paragraph')) {
|
|
40
|
+
return false;
|
|
41
|
+
}
|
|
42
|
+
const viewElement = mapper.toViewElement(child);
|
|
43
|
+
if (!viewElement) {
|
|
44
|
+
return false;
|
|
45
|
+
}
|
|
46
|
+
return isSingleParagraphWithoutAttributes(child) !== viewElement.is('element', 'span');
|
|
47
|
+
}
|
|
@@ -5,7 +5,7 @@
|
|
|
5
5
|
/**
|
|
6
6
|
* @module table/converters/table-headings-refresh-handler
|
|
7
7
|
*/
|
|
8
|
-
import type { EditingController, Model } from '
|
|
8
|
+
import type { EditingController, Model } from 'ckeditor5/src/engine.js';
|
|
9
9
|
/**
|
|
10
10
|
* A table headings refresh handler which marks the table cells or rows in the differ to have it re-rendered
|
|
11
11
|
* if the headings attribute changed.
|
|
@@ -0,0 +1,51 @@
|
|
|
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
|
+
import { TableWalker } from '../tablewalker.js';
|
|
6
|
+
/**
|
|
7
|
+
* A table headings refresh handler which marks the table cells or rows in the differ to have it re-rendered
|
|
8
|
+
* if the headings attribute changed.
|
|
9
|
+
*
|
|
10
|
+
* Table heading rows and heading columns are represented in the model by a `headingRows` and `headingColumns` attributes.
|
|
11
|
+
*
|
|
12
|
+
* When table headings attribute changes, all the cells/rows are marked to re-render to change between `<td>` and `<th>`.
|
|
13
|
+
*
|
|
14
|
+
* @internal
|
|
15
|
+
*/
|
|
16
|
+
export function tableHeadingsRefreshHandler(model, editing) {
|
|
17
|
+
const differ = model.document.differ;
|
|
18
|
+
for (const change of differ.getChanges()) {
|
|
19
|
+
let table;
|
|
20
|
+
let isRowChange = false;
|
|
21
|
+
if (change.type == 'attribute') {
|
|
22
|
+
const element = change.range.start.nodeAfter;
|
|
23
|
+
if (!element || !element.is('element', 'table')) {
|
|
24
|
+
continue;
|
|
25
|
+
}
|
|
26
|
+
if (change.attributeKey != 'headingRows' && change.attributeKey != 'headingColumns') {
|
|
27
|
+
continue;
|
|
28
|
+
}
|
|
29
|
+
table = element;
|
|
30
|
+
isRowChange = change.attributeKey == 'headingRows';
|
|
31
|
+
}
|
|
32
|
+
else if (change.name == 'tableRow' || change.name == 'tableCell') {
|
|
33
|
+
table = change.position.findAncestor('table');
|
|
34
|
+
isRowChange = change.name == 'tableRow';
|
|
35
|
+
}
|
|
36
|
+
if (!table) {
|
|
37
|
+
continue;
|
|
38
|
+
}
|
|
39
|
+
const headingRows = table.getAttribute('headingRows') || 0;
|
|
40
|
+
const headingColumns = table.getAttribute('headingColumns') || 0;
|
|
41
|
+
const tableWalker = new TableWalker(table);
|
|
42
|
+
for (const tableSlot of tableWalker) {
|
|
43
|
+
const isHeading = tableSlot.row < headingRows || tableSlot.column < headingColumns;
|
|
44
|
+
const expectedElementName = isHeading ? 'th' : 'td';
|
|
45
|
+
const viewElement = editing.mapper.toViewElement(tableSlot.cell);
|
|
46
|
+
if (viewElement && viewElement.is('element') && viewElement.name != expectedElementName) {
|
|
47
|
+
editing.reconvertItem((isRowChange ? tableSlot.cell.parent : tableSlot.cell));
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
}
|