-
-
Notifications
You must be signed in to change notification settings - Fork 3.7k
/
title.ts
576 lines (475 loc) · 19.1 KB
/
title.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
/**
* @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.
* For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
*/
/**
* @module heading/title
*/
import { Plugin, type Editor, type ElementApi } from 'ckeditor5/src/core';
import { first, type GetCallback } from 'ckeditor5/src/utils';
import {
DowncastWriter,
enablePlaceholder,
hidePlaceholder,
needsPlaceholder,
showPlaceholder,
type DowncastInsertEvent,
type Element,
type MapperModelToViewPositionEvent,
type Model,
type RootElement,
type UpcastConversionApi,
type UpcastConversionData,
type UpcastElementEvent,
type View,
type ViewElement,
type Writer
} from 'ckeditor5/src/engine';
// A list of element names that should be treated by the Title plugin as title-like.
// This means that an element of a type from this list will be changed to a title element
// when it is the first element in the root.
const titleLikeElements = new Set( [ 'paragraph', 'heading1', 'heading2', 'heading3', 'heading4', 'heading5', 'heading6' ] );
/**
* The Title plugin.
*
* It splits the document into `Title` and `Body` sections.
*/
export default class Title extends Plugin {
/**
* A reference to an empty paragraph in the body
* created when there is no element in the body for the placeholder purposes.
*/
private _bodyPlaceholder?: null | Element;
/**
* @inheritDoc
*/
public static get pluginName(): 'Title' {
return 'Title';
}
/**
* @inheritDoc
*/
public static get requires() {
return [ 'Paragraph' ] as const;
}
/**
* @inheritDoc
*/
public init(): void {
const editor = this.editor;
const model = editor.model;
this._bodyPlaceholder = null;
// To use the schema for disabling some features when the selection is inside the title element
// it is needed to create the following structure:
//
// <title>
// <title-content>The title text</title-content>
// </title>
//
// See: https://github.com/ckeditor/ckeditor5/issues/2005.
model.schema.register( 'title', { isBlock: true, allowIn: '$root' } );
model.schema.register( 'title-content', { isBlock: true, allowIn: 'title', allowAttributes: [ 'alignment' ] } );
model.schema.extend( '$text', { allowIn: 'title-content' } );
// Disallow all attributes in `title-content`.
model.schema.addAttributeCheck( context => {
if ( context.endsWith( 'title-content $text' ) ) {
return false;
}
} );
// Because `title` is represented by two elements in the model
// but only one in the view, it is needed to adjust Mapper.
editor.editing.mapper.on<MapperModelToViewPositionEvent>( 'modelToViewPosition', mapModelPositionToView( editor.editing.view ) );
editor.data.mapper.on<MapperModelToViewPositionEvent>( 'modelToViewPosition', mapModelPositionToView( editor.editing.view ) );
// Conversion.
editor.conversion.for( 'downcast' ).elementToElement( { model: 'title-content', view: 'h1' } );
editor.conversion.for( 'downcast' ).add( dispatcher => dispatcher.on<DowncastInsertEvent>(
'insert:title',
( evt, data, conversionApi ) => {
conversionApi.consumable.consume( data.item, evt.name );
}
) );
// Custom converter is used for data v -> m conversion to avoid calling post-fixer when setting data.
// See https://github.com/ckeditor/ckeditor5/issues/2036.
editor.data.upcastDispatcher.on<UpcastElementEvent>( 'element:h1', dataViewModelH1Insertion, { priority: 'high' } );
editor.data.upcastDispatcher.on<UpcastElementEvent>( 'element:h2', dataViewModelH1Insertion, { priority: 'high' } );
editor.data.upcastDispatcher.on<UpcastElementEvent>( 'element:h3', dataViewModelH1Insertion, { priority: 'high' } );
// Take care about correct `title` element structure.
model.document.registerPostFixer( writer => this._fixTitleContent( writer ) );
// Create and take care of correct position of a `title` element.
model.document.registerPostFixer( writer => this._fixTitleElement( writer ) );
// Create element for `Body` placeholder if it is missing.
model.document.registerPostFixer( writer => this._fixBodyElement( writer ) );
// Prevent from adding extra at the end of the document.
model.document.registerPostFixer( writer => this._fixExtraParagraph( writer ) );
// Attach `Title` and `Body` placeholders to the empty title and/or content.
this._attachPlaceholders();
// Attach Tab handling.
this._attachTabPressHandling();
}
/**
* Returns the title of the document. Note that because this plugin does not allow any formatting inside
* the title element, the output of this method will be a plain text, with no HTML tags.
*
* It is not recommended to use this method together with features that insert markers to the
* data output, like comments or track changes features. If such markers start in the title and end in the
* body, the result of this method might be incorrect.
*
* @param options Additional configuration passed to the conversion process.
* See {@link module:engine/controller/datacontroller~DataController#get `DataController#get`}.
* @returns The title of the document.
*/
public getTitle( options: Record<string, unknown> = {} ): string {
const titleElement = this._getTitleElement();
const titleContentElement = titleElement!.getChild( 0 ) as Element;
return this.editor.data.stringify( titleContentElement, options );
}
/**
* Returns the body of the document.
*
* Note that it is not recommended to use this method together with features that insert markers to the
* data output, like comments or track changes features. If such markers start in the title and end in the
* body, the result of this method might be incorrect.
*
* @param options Additional configuration passed to the conversion process.
* See {@link module:engine/controller/datacontroller~DataController#get `DataController#get`}.
* @returns The body of the document.
*/
public getBody( options: Record<string, unknown> = {} ): string {
const editor = this.editor;
const data = editor.data;
const model = editor.model;
const root = editor.model.document.getRoot()!;
const view = editor.editing.view;
const viewWriter = new DowncastWriter( view.document );
const rootRange = model.createRangeIn( root );
const viewDocumentFragment = viewWriter.createDocumentFragment();
// Find all markers that intersects with body.
const bodyStartPosition = model.createPositionAfter( root.getChild( 0 )! );
const bodyRange = model.createRange( bodyStartPosition, model.createPositionAt( root, 'end' ) );
const markers = new Map();
for ( const marker of model.markers ) {
const intersection = bodyRange.getIntersection( marker.getRange() );
if ( intersection ) {
markers.set( marker.name, intersection );
}
}
// Convert the entire root to view.
data.mapper.clearBindings();
data.mapper.bindElements( root, viewDocumentFragment );
data.downcastDispatcher.convert( rootRange, markers, viewWriter, options );
// Remove title element from view.
viewWriter.remove( viewWriter.createRangeOn( viewDocumentFragment.getChild( 0 ) ) );
// view -> data
return editor.data.processor.toData( viewDocumentFragment );
}
/**
* Returns the `title` element when it is in the document. Returns `undefined` otherwise.
*/
private _getTitleElement(): Element | undefined {
const root = this.editor.model.document.getRoot()!;
for ( const child of root.getChildren() as IterableIterator<Element> ) {
if ( isTitle( child ) ) {
return child;
}
}
}
/**
* Model post-fixer callback that ensures that `title` has only one `title-content` child.
* All additional children should be moved after the `title` element and renamed to a paragraph.
*/
private _fixTitleContent( writer: Writer ) {
const title = this._getTitleElement();
// There's no title in the content - it will be created by _fixTitleElement post-fixer.
if ( !title || title.maxOffset === 1 ) {
return false;
}
const titleChildren = Array.from( title.getChildren() as IterableIterator<Element> );
// Skip first child because it is an allowed element.
titleChildren.shift();
for ( const titleChild of titleChildren ) {
writer.move( writer.createRangeOn( titleChild ), title, 'after' );
writer.rename( titleChild, 'paragraph' );
}
return true;
}
/**
* Model post-fixer callback that creates a title element when it is missing,
* takes care of the correct position of it and removes additional title elements.
*/
private _fixTitleElement( writer: Writer ) {
const model = this.editor.model;
const modelRoot = model.document.getRoot()!;
const titleElements = Array.from( modelRoot.getChildren() as IterableIterator<Element> ).filter( isTitle );
const firstTitleElement = titleElements[ 0 ];
const firstRootChild = modelRoot.getChild( 0 ) as Element;
// When title element is at the beginning of the document then try to fix additional
// title elements (if there are any) and stop post-fixer as soon as possible.
if ( firstRootChild.is( 'element', 'title' ) ) {
return fixAdditionalTitleElements( titleElements, writer, model );
}
// When there is no title in the document and first element in the document cannot be changed
// to the title then create an empty title element at the beginning of the document.
if ( !firstTitleElement && !titleLikeElements.has( firstRootChild.name ) ) {
const title = writer.createElement( 'title' );
writer.insert( title, modelRoot );
writer.insertElement( 'title-content', title );
return true;
}
// At this stage, we are sure the title is somewhere in the content. It has to be fixed.
// Change the first element in the document to the title if it can be changed (is title-like).
if ( titleLikeElements.has( firstRootChild.name ) ) {
changeElementToTitle( firstRootChild, writer, model );
// Otherwise, move the first occurrence of the title element to the beginning of the document.
} else {
writer.move( writer.createRangeOn( firstTitleElement ), modelRoot, 0 );
}
fixAdditionalTitleElements( titleElements, writer, model );
return true;
}
/**
* Model post-fixer callback that adds an empty paragraph at the end of the document
* when it is needed for the placeholder purposes.
*/
private _fixBodyElement( writer: Writer ) {
const modelRoot = this.editor.model.document.getRoot()!;
if ( modelRoot.childCount < 2 ) {
this._bodyPlaceholder = writer.createElement( 'paragraph' );
writer.insert( this._bodyPlaceholder, modelRoot, 1 );
return true;
}
return false;
}
/**
* Model post-fixer callback that removes a paragraph from the end of the document
* if it was created for the placeholder purposes and is not needed anymore.
*/
private _fixExtraParagraph( writer: Writer ) {
const root = this.editor.model.document.getRoot()!;
const placeholder = this._bodyPlaceholder!;
if ( shouldRemoveLastParagraph( placeholder, root ) ) {
this._bodyPlaceholder = null;
writer.remove( placeholder );
return true;
}
return false;
}
/**
* Attaches the `Title` and `Body` placeholders to the title and/or content.
*/
private _attachPlaceholders() {
const editor: Editor & Partial<ElementApi> = this.editor;
const t = editor.t;
const view = editor.editing.view;
const viewRoot = view.document.getRoot();
const sourceElement = editor.sourceElement;
const titlePlaceholder = editor.config.get( 'title.placeholder' ) || t( 'Type your title' );
const bodyPlaceholder = editor.config.get( 'placeholder' ) ||
sourceElement && sourceElement.tagName.toLowerCase() === 'textarea' && sourceElement.getAttribute( 'placeholder' ) ||
t( 'Type or paste your content here.' );
// Attach placeholder to the view title element.
editor.editing.downcastDispatcher.on<DowncastInsertEvent<Element>>( 'insert:title-content', ( evt, data, conversionApi ) => {
enablePlaceholder( {
view,
element: conversionApi.mapper.toViewElement( data.item )!,
text: titlePlaceholder,
keepOnFocus: true
} );
} );
// Attach placeholder to first element after a title element and remove it if it's not needed anymore.
// First element after title can change so we need to observe all changes keep placeholder in sync.
let oldBody: ViewElement;
// This post-fixer runs after the model post-fixer so we can assume that
// the second child in view root will always exist.
view.document.registerPostFixer( writer => {
const body = viewRoot!.getChild( 1 ) as ViewElement;
let hasChanged = false;
// If body element has changed we need to disable placeholder on the previous element
// and enable on the new one.
if ( body !== oldBody ) {
if ( oldBody ) {
hidePlaceholder( writer, oldBody );
writer.removeAttribute( 'data-placeholder', oldBody );
}
writer.setAttribute( 'data-placeholder', bodyPlaceholder, body );
oldBody = body;
hasChanged = true;
}
// Then we need to display placeholder if it is needed.
// See: https://github.com/ckeditor/ckeditor5/issues/8689.
if ( needsPlaceholder( body, true ) && viewRoot!.childCount === 2 && body!.name === 'p' ) {
hasChanged = showPlaceholder( writer, body ) ? true : hasChanged;
// Or hide if it is not needed.
} else {
hasChanged = hidePlaceholder( writer, body ) ? true : hasChanged;
}
return hasChanged;
} );
}
/**
* Creates navigation between the title and body sections using <kbd>Tab</kbd> and <kbd>Shift</kbd>+<kbd>Tab</kbd> keys.
*/
private _attachTabPressHandling() {
const editor = this.editor;
const model = editor.model;
// Pressing <kbd>Tab</kbd> inside the title should move the caret to the body.
editor.keystrokes.set( 'TAB', ( data, cancel ) => {
model.change( writer => {
const selection = model.document.selection;
const selectedElements = Array.from( selection.getSelectedBlocks() );
if ( selectedElements.length === 1 && selectedElements[ 0 ].is( 'element', 'title-content' ) ) {
const firstBodyElement = model.document.getRoot()!.getChild( 1 );
writer.setSelection( firstBodyElement!, 0 );
cancel();
}
} );
} );
// Pressing <kbd>Shift</kbd>+<kbd>Tab</kbd> at the beginning of the body should move the caret to the title.
editor.keystrokes.set( 'SHIFT + TAB', ( data, cancel ) => {
model.change( writer => {
const selection = model.document.selection;
if ( !selection.isCollapsed ) {
return;
}
const root = editor.model.document.getRoot()!;
const selectedElement = first( selection.getSelectedBlocks() );
const selectionPosition = selection.getFirstPosition()!;
const title = root.getChild( 0 ) as Element;
const body = root.getChild( 1 );
if ( selectedElement === body && selectionPosition.isAtStart ) {
writer.setSelection( title.getChild( 0 )!, 0 );
cancel();
}
} );
} );
}
}
/**
* A view-to-model converter for the h1 that appears at the beginning of the document (a title element).
*
* @see module:engine/conversion/upcastdispatcher~UpcastDispatcher#event:element
* @param evt An object containing information about the fired event.
* @param data An object containing conversion input, a placeholder for conversion output and possibly other values.
* @param conversionApi Conversion interface to be used by the callback.
*/
function dataViewModelH1Insertion( evt: unknown, data: UpcastConversionData<ViewElement>, conversionApi: UpcastConversionApi ) {
const modelCursor = data.modelCursor;
const viewItem = data.viewItem;
if ( !modelCursor.isAtStart || !modelCursor.parent.is( 'element', '$root' ) ) {
return;
}
if ( !conversionApi.consumable.consume( viewItem, { name: true } ) ) {
return;
}
const modelWriter = conversionApi.writer;
const title = modelWriter.createElement( 'title' );
const titleContent = modelWriter.createElement( 'title-content' );
modelWriter.append( titleContent, title );
modelWriter.insert( title, modelCursor );
conversionApi.convertChildren( viewItem, titleContent );
conversionApi.updateConversionResult( title, data );
}
/**
* Maps position from the beginning of the model `title` element to the beginning of the view `h1` element.
*
* ```html
* <title>^<title-content>Foo</title-content></title> -> <h1>^Foo</h1>
* ```
*/
function mapModelPositionToView( editingView: View ): GetCallback<MapperModelToViewPositionEvent> {
return ( evt, data ) => {
const positionParent = data.modelPosition.parent;
if ( !positionParent.is( 'element', 'title' ) ) {
return;
}
const modelTitleElement = positionParent.parent as Element;
const viewElement = data.mapper.toViewElement( modelTitleElement )!;
data.viewPosition = editingView.createPositionAt( viewElement, 0 );
evt.stop();
};
}
/**
* @returns Returns true when given element is a title. Returns false otherwise.
*/
function isTitle( element: Element ) {
return element.is( 'element', 'title' );
}
/**
* Changes the given element to the title element.
*/
function changeElementToTitle( element: Element, writer: Writer, model: Model ) {
const title = writer.createElement( 'title' );
writer.insert( title, element, 'before' );
writer.insert( element, title, 0 );
writer.rename( element, 'title-content' );
model.schema.removeDisallowedAttributes( [ element ], writer );
}
/**
* Loops over the list of title elements and fixes additional ones.
*
* @returns Returns true when there was any change. Returns false otherwise.
*/
function fixAdditionalTitleElements( titleElements: Array<Element>, writer: Writer, model: Model ) {
let hasChanged = false;
for ( const title of titleElements ) {
if ( title.index !== 0 ) {
fixTitleElement( title, writer, model );
hasChanged = true;
}
}
return hasChanged;
}
/**
* Changes given title element to a paragraph or removes it when it is empty.
*/
function fixTitleElement( title: Element, writer: Writer, model: Model ) {
const child = title.getChild( 0 ) as Element;
// Empty title should be removed.
// It is created as a result of pasting to the title element.
if ( child.isEmpty ) {
writer.remove( title );
return;
}
writer.move( writer.createRangeOn( child ), title, 'before' );
writer.rename( child, 'paragraph' );
writer.remove( title );
model.schema.removeDisallowedAttributes( [ child ], writer );
}
/**
* Returns true when the last paragraph in the document was created only for the placeholder
* purpose and it's not needed anymore. Returns false otherwise.
*/
function shouldRemoveLastParagraph( placeholder: Element, root: RootElement ) {
if ( !placeholder || !placeholder.is( 'element', 'paragraph' ) || placeholder.childCount ) {
return false;
}
if ( root.childCount <= 2 || root.getChild( root.childCount - 1 ) !== placeholder ) {
return false;
}
return true;
}
/**
* The configuration of the {@link module:heading/title~Title title feature}.
*
* ```ts
* ClassicEditor
* .create( document.querySelector( '#editor' ), {
* plugins: [ Title, ... ],
* title: {
* placeholder: 'My custom placeholder for the title'
* },
* placeholder: 'My custom placeholder for the body'
* } )
* .then( ... )
* .catch( ... );
* ```
*
* See {@link module:core/editor/editorconfig~EditorConfig all editor configuration options}.
*/
export interface TitleConfig {
/**
* Defines a custom value of the placeholder for the title field.
*
* Read more in {@link module:heading/title~TitleConfig}.
*/
placeholder?: string;
}