-
Notifications
You must be signed in to change notification settings - Fork 704
/
Copy pathOverview.bs
794 lines (654 loc) · 31.8 KB
/
Overview.bs
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
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
<!--
This document is formatted using Bikeshed.
Roughly speaking, it is a Markdown preprocessor,
with additional functionality for cross-spec autolinking,
automatic generation of indexes/ToC/etc,
and many other features.
See https://tabatkins.github.io/bikeshed/ for detailed documentation.
When making edits, please respect the following coding style:
- Tabs for indentation, spaces for alignment.
- Semantic line breaks: at phrases boundaries, each line < ~80ch
-> see https://rhodesmill.org/brandon/2012/one-sentence-per-line/
- Indent the entire spec one level except for headings.
- Line break after opening heading tag, so heading text
is easy to pick out when scanning the source.
- Empty lines between blocks.
- Indent contents of block-level HTML elements
(except <p>, which we usually imply via Markdown formatting
and otherwise leave inlined at the start of the paragraph).
Definitely leave a break and indent
after any block start tag with attributes, though.
- No optional end tags.
- Use manual IDs so that IDs remain stable as you adjust the heading text;
add old IDs (via empty elements with IDs, or e.g. Bikeshed's oldids attribute)
when removing or changing IDs so that links to your spec don't break.
-->
<pre class=metadata>
Title: CSS Custom Highlight API Module Level 1
Level: 1
Shortname: css-highlight-api
Status: ED
Work Status: exploring
Group: csswg
ED: https://drafts.csswg.org/css-highlight-api-1/
TR: https://www.w3.org/TR/css-highlight-api-1/
Previous Version: https://www.w3.org/TR/2020/WD-css-highlight-api-1-20201022/
Previous Version: https://www.w3.org/TR/2020/WD-css-highlight-api-1-20201208/
Editor: Florian Rivoal, On behalf of Bloomberg, https://florian.rivoal.net/, w3cid 43241
Editor: Sanket Joshi, Microsoft Corporation https://www.microsoft.com, https://github.com/sanketj, w3cid 115721
Editor: Megan Gardner, Apple Inc. https://apple.com/, w3cid 110930
Abstract:
This CSS module describes a mechanism
for styling arbitrary ranges of a document identified by script.
Complain About: accidental-2119 yes
</pre>
<pre class=link-defaults>
spec:css-color-4; type:property; text:color
spec:css-pseudo-4; type:dfn; text:highlight overlay
spec:dom; type:dfn; text:range
spec:css-values-4; type:dfn; text:identifier
spec:dom; type:dfn; for:Element; text:shadow root
spec:dom; type:dfn; text:event
spec:css21; type:dfn; text:viewport
</pre>
<style>
.sample-out {
background: white;
padding: 0.5ch;
border: solid 1px;
font-size: 2em;
}
</style>
<div class=non-normative>
<h2 id='intro'>
Introduction</h2>
<em>This section is non-normative</em>.
The Custom Highlight API extends the concept of [=highlight pseudo-elements=] (see [[css-pseudo-4#highlight-pseudos]])
by providing a way for web developers to style the text
of arbitrary Range objects,
rather than being limited to the user agent defined
''::selection'',
''::inactive-selection'',
''::spelling-error'',
and ''::grammar-error''.
This is useful in a variety of scenarios,
including editing frameworks that wish to implement their own selection,
find-on-page over virtualized documents,
multiple selection to represent online collaboration,
or spellchecking frameworks.
The Custom Highlight API provides a programmatic way of adding and removing highlights
that do not affect the underlying DOM structure,
but instead applies styles to text based on [=range=] objects,
accessed via the ''::highlight()'' pseudo element.
<div id=intro-ex class=example>
The following code uses the ''::highlight()'' pseudo-element
to apply a yellow background and blue foreground color to the text <q>One two</q>.
It does so by adding a {{Highlight}} to the {{HighlightRegistry}}
(both of these are new concepts introduced by this specification).
The {{Highlight}} will contain a {{Range}} whose boundary points surround the text <q>One two</q>.
<xmp highlight=html>
<style>
:root::highlight(example-highlight) {
background-color: yellow;
color: blue;
}
</style>
<body><span>One </span><span>two </span><span>three…</span>
<script>
let r = new Range();
r.setStart(document.body, 0);
r.setEnd(document.body, 2);
CSS.highlights.set("example-highlight", new Highlight(r));
</script>
</xmp>
The result would look like:
<div class=sample-out>
<span style="background-color:yellow;color:blue;">One Two </span>three…
</div>
</div>
</div>
<h2 id="interaction">
Module Interactions</h2>
This module depends on the Infra Standard [[!INFRA]]
and on WebIDL [[!WebIDL]].
It assumes general familiarity with CSS
and with the DOM Standard [[DOM]],
and specifically extends the mechanisms defined in CSS Pseudo-Elements Module Level 4 [[!css-pseudo-4]]
to handle [=highlight pseudo-elements=].
The Selectors Level 4 [[!selectors-4]] specification defines how [=pseudo-elements=] work in general.
See [[#references]] for a full list of dependencies.
Note: This draft is an early version.
As it matures, the CSS-WG could decide to keep it as an independent module,
or might prefer to fold it into [[css-pseudo-4]],
or a later version of that module.
<h2 id=highlights-set-up>
Setting up Custom Highlights</h2>
<h3 id=creation>
Creating Custom Highlights</h3>
A <dfn export>custom highlight</dfn> is a collection of [=ranges=]
representing portions of a document.
They do not necessarily fit into the element tree,
and can arbitrarily cross element boundaries without honoring its nesting structure.
They can be used to affect the appearance of these portions of the document
(see [[#styling-highlights]]),
or to handle to events associated with them
(see [[#events]]).
[=Custom highlights=] are represented by
<dfn interface>Highlight</dfn> objects,
[=setlike=] objects whose [=set entries=] are {{AbstractRange}} objects.
[=Ranges=] can be added to a [=custom highlight=]
either by passing them to its constructor,
or by using the usual API of [=setlike=] objects
to manipulate its [=set entries=].
Note: As the [=ranges=] in a [=custom highlight=] are {{AbstractRange}} objects,
authors can chose between using {{Range}} objects and {{StaticRange}} objects.
See [[#range-invalidation]] for more details about this choice and its implications.
Note: When creating {{Range}} objects for use in [=custom highlights=], it is
suggested that authors avoid placing [=range=] endpoints in the middle of a [=grapheme=],
such as when a visible unit of text is comprised of multiple [=Unicode code points=].
Doing so can create undesirable highlighting effects, such as highlighting only part of
an Indic syllable. In addition, care needs to be taken to avoid placing an endpoint
in the middle of a [=supplementary character=].
<xmp class="idl">
enum HighlightType {
"highlight",
"spelling-error",
"grammar-error"
};
[Exposed=Window]
interface Highlight {
constructor(AbstractRange... initialRanges);
setlike<AbstractRange>;
attribute long priority;
attribute HighlightType type;
};
</xmp>
See [[#priorities]] for more information on the {{Highlight/priority}} attribute.
See [[#highlight-types]] for more information on the {{Highlight/type}} attribute.
<div algorithm="to create a custom highlight">
When the <dfn for=Highlight constructor>Highlight(AbstractRange... initialRanges)</dfn> constructor is invoked,
run the following steps:
<ol>
<li>
Let |highlight| be the new {{Highlight}} object.
<li>
Set |highlight|'s {{Highlight/priority}} to <code>0</code>.
<li>
Set |highlight|'s {{Highlight/type}} to {{HighlightType/highlight}}.
<li>
For each |range| of {{initialRanges}},
let |rangeArg| be the result of [=converted to an ECMAScript value|converting=] |range| to an ECMAScript value,
then run [[webidl#js-set-add|the steps for a built-in setlike add function]],
with |highlight| as the <code>this</code> value,
and |rangeArg| as the argument.
<li>
Return |highlight|.
</ol>
</div>
<h3 id=registration>
Registering Custom Highlights</h3>
In order to have any effect,
[=custom highlights=] need to be
[=registered=] into the [=highlight registry=].
The <dfn>highlight registry</dfn> is accessed via the {{CSS/highlights}} attribute of the {{CSS}} namespace,
and represents all the [=custom highlights=] [=registered=] for the [=current global object=]’s [=associated Document=].
It is a [=maplike=], and can be updated using the usual methods.
It's [=map entries=] is initially empty.
A [=custom highlight=] is said to be <dfn>registered</dfn>
if it is in the [=highlight registry=].
It stops being [=registered=] if it is later removed.
<xmp class="idl">
partial namespace CSS {
readonly attribute HighlightRegistry highlights;
};
[Exposed=Window]
interface HighlightRegistry {
maplike<DOMString, Highlight>;
};
</xmp>
<div algorithm="to register a custom highlight">
To [=register=] a [=custom highlight=],
invoke the <code>set</code> method of the [=highlight registry=]
which will run [[webidl#js-map-set|the steps for a built-in maplike set function]],
with the [=highlight registry=] as the <code>this</code> value,
the passed-in [=custom highlight name=] as <var ignore>keyArg</var>,
and the passed-in highlight as <var ignore>valueArg</var>.
The <dfn export>custom highlight name</dfn> assigned to a [=custom highlight=] when it is [=registered=]
is used to identify the highlight during styling (see [[#styling-highlights]]).
Note: When registering a [=custom highlight=],
authors are advised to use a [=custom highlight name=] that is a valid CSS [=identifier=].
Using a name that is not a valid identifier can make the highlight hard,
and in some cases impossible, to style via CSS.
Note: It is possible to [=register=] a [=custom highlight=] with more than one [=custom highlight name=].
However, using more than one name to style a highlight
will assign the highlight multiple different sets of styles,
without a way to control the stacking order of conflicting styles
within these sets during [[#painting|painting]].
This could be limiting for authors and could cause confusing painting behavior
(see the <a href="#styling-problems-with-multiple-names-per-highlight">example</a> below for more context).
Therefore, <b>authors are advised to only use one name per highlight during styling</b>.
<div class=example id=styling-problems-with-multiple-names-per-highlight>
<xmp highlight=html>
<style>
div::highlight(bar) {
color: red;
}
div::highlight(foo) {
color: green;
}
</style>
<body><div>abc</div>
<script>
let div = document.body.firstChild;
let r = new Range();
r.setStart(div, 0);
r.setEnd(div, 1);
let h = new Highlight(r);
CSS.highlights.set('foo', h);
CSS.highlights.set('bar', h);
</script>
</xmp>
In the example above,
the same [=custom highlight=] object is [=registered=] under the names <code>foo</code> and <code>bar</code>.
Since each of the [=style rules=] target the same highlight and have the same [=specificity=],
authors might expect the last rule to win in cascading order
and the highlighted content to be green.
However, each highlight name gets an independent set of highlight styles,
and the highlight will be painted once per name.
In this case, because <code>foo</code> was registered before <code>bar</code>,
the highlight will be first painted with <code>foo</code>'s color (green)
and then with <code>bar</code>'s color (red).
As a result, the highlighted content will appear red.
</div>
<h2 id=styling-highlights>
Styling Custom Highlights</h2>
<h3 id=custom-highlight-pseudo>
The Custom Highlight Pseudo-element: ''::highlight()''</h3>
The ''::highlight()'' pseudo-element
(also known as the <dfn>custom highlight pseudo-element</dfn>)
represents the portion of a document that
is being [=contained=] or [=partially contained=]
in all the [=ranges=] of the [=registered=] [=custom highlight=]
with the [=custom highlight name=] specified as its argument.
<h3 id=processing-model>
Processing Model</h3>
<h4 id=applicable-properties>
Applicable Properties</h4>
[=Custom highlight pseudo-elements=],
like the built-in [=highlight pseudo-elements=],
can only be styled with a limited set of properties.
See [[css-pseudo-4#highlight-styling]] for the full list.
<h4 id=default-styles>
Default Styles</h4>
UAs must not define any default UA stylesheet rules
or <a spec=css-pseudo>paired default highlight colors</a>
for any [=custom highlight pseudo-elements=].
In other words,
when some content is highlighted by an unstyled custom highlight,
its presentation must not change.
<h4 id=c-and-h>
Cascading and Inheritance</h4>
The [=cascading=] and [=inheritance=] of [=custom highlight pseudo-elements=] is handled
identically to that of the built-in [=highlight pseudo-elements=],
as defined in [[css-pseudo-4#highlight-cascade]].
<h4 id=painting>
Painting</h4>
The painting of [=custom highlights=] is also handled
identically to that of the built-in [=highlight pseudo-elements=],
as specified in
[[css-pseudo-4#highlight-bounds]] and [[css-pseudo-4#highlight-painting]],
with the following clarifications:
<ul>
<li>
[=Collapsed=] [=ranges=] are not rendered.
<li>
Overlapping [=ranges=] within a single [=custom highlight=] are rendered
as if a single range representing the union of the overlapping ones
had been specified.
<div class=example id=overlap-ranges-ex>
The following example renders in a single highlight with semi-transparent blue background,
not two overlapping ones which can be seen through each other.
<xmp highlight=html>
<style>
::highlight(sample) { background-color: rgba(0, 0, 255, 0.3); }
</style>
<body>Lorem Ipsum.
<script>
let textNode = document.body.firstChild;
let r1 = new Range();
r1.setStart(textNode, 1);
r1.setEnd(textNode, 5);
let r2 = new Range();
r2.setStart(textNode, 3);
r2.setEnd(textNode, 7);
CSS.highlights.set("sample", new Highlight(r1, r2));
</script>
</xmp>
In other words, this rendering would be correct:
<div class=sample-out>
L<span style="background-color: rgba(0, 0, 255, 0.3)">orem I</span>psum.
</div>
However, this one would be incorrect:
<div class=sample-out>
L<span style="background-color: rgba(0, 0, 255, 0.3)">or<span style="background-color: rgba(0, 0, 255, 0.3)">em</span> I</span>psum.
</div>
</div>
<li>
The [=highlight overlays=] of the [=custom highlights=]
are below those of the built-in [=highlight pseudo-elements=]
in the stacking order described in [[css-pseudo-4#highlight-painting]].
<li>
The relative stacking order of the [=highlight overlays=]
of multiple [=custom highlights=]
is defined by their [=priority=]
(see [[#priorities]]).
</ul>
<h4 id=priorities>
Priority of Overlapping Highlights</h4>
A [=custom highlight=]'s {{Highlight/priority}} attribute
defines its <dfn>priority</dfn>.
This is used to determine the stacking order of the corresponding [=highlight overlay=]
during painting operations (see [[#painting]]).
A higher [=priority=] results in being above in the stacking order.
A custom highlight will have a default numerical priority of 0
if its {{Highlight/priority}} attribute has not been explicitly set.
When two or more [=custom highlights=] have the same numerical priority,
the one most recently [=registered=] has the higher effective [=priority=].
<div class=example id=overlap-highlight-ex>
<xmp highlight=html>
<style>
:root::highlight(foo) {
color:blue;
background-color:yellow;
}
:root::highlight(bar) {
background-color:orange;
}
</style>
<body>Some text
<script>
let textNode = document.body.firstChild;
let r1 = new Range();
r1.setStart(textNode, 0);
r1.setEnd(textNode, 6);
let r2 = new Range();
r2.setStart(textNode, 3);
r2.setEnd(textNode, 9);
let h1 = new Highlight(r1);
let h2 = new Highlight(r2);
CSS.highlights.set("foo", h1);
CSS.highlights.set("bar", h2);
</script>
</xmp>
As there are no priorities set
(i.e. there is a tie between <code>h1</code> and <code>h2</code>),
the custom highlights' styles are stacked
in order of insertion into the [=highlight registry=].
The rendered results will have "Som" with blue text on yellow background,
"e t" with blue text on orange background,
and "ext" with the default color on orange background.
<div class=sample-out>
<span style="background:yellow;color:blue;">Som</span><span style="background:orange;color:blue;">e t</span><span style="background:orange;">ext</span>
</div>
Setting <code highlight=javascript>h1.priority = 1;</code>
would cause <code>h1</code> to stack higher than <code>h2</code>,
which would result in "Some t" being blue on yellow,
and "ext" being default color on orange.
<div class=sample-out>
<span style="background:yellow;color:blue;">Some t</span><span style="background:orange;">ext</span>
</div>
</div>
<h4 id=highlight-types>
Highlight types</h4>
A [=custom highlight=]'s {{Highlight/type}} attribute is used by authors
to specify the semantic meaning of the highlight.
This allows assistive technologies to include this meaning
when exposing the highlight to users.
A custom highlight will have a default type of {{HighlightType/highlight}}
if its {{Highlight/type}} attribute has not been explicitly set.
Note: Authors are advised to set a [=custom highlight=]'s {{Highlight/type}} to {{HighlightType/spelling-error}}
when that [=custom highlight=] is being used to emphasize misspelled content.
Authors are advised to set a [=custom highlight=]'s {{Highlight/type}} to {{HighlightType/grammar-error}}
when that [=custom highlight=] is being used to emphasize content that is grammatically incorrect.
For all other use cases {{Highlight/type}} is best left as {{HighlightType/highlight}}.
UAs should make [=custom highlight=]s available to assistive technologies.
When exposing a highlight using a given platform accessibility API,
UAs should expose the semantic meaning of the highlight
as specified by its {{Highlight/type}} attribute
with as much specificity as possible for that accessibility API.
Note: For example,
if a platform accessibility API has the capability to express spelling errors and grammar errors specifically,
then UAs is expected to use these capabilities to convey the semantics for highlights
with {{HighlightType/spelling-error}} and {{HighlightType/spelling-error}}.
If an accessibility API only has the capability to express spelling errors,
then UAs would be expected to convey both highlights with {{HighlightType/spelling-error}}
and with {{HighlightType/grammar-error}} using spelling error semantics.
If an accessibility API has support for expressing neither spelling errors nor grammar errors,
then UAs would expose all highlights as generic {{HighlightType/highlight}} regardless of their actual {{Highlight/type}}.
Note: This initial set of types was chosen
because they are expected to be popular use cases for Highlight API
and there is some existing support for expressing their semantics in platform accessibility APIs today.
Accessibility APIs currently don't have any way to express the specific semantics
of other expected Highlight API use cases.
More types could later be added to {{HighlightType}}
as accessibility APIs gain support for expressing additional popular use cases of Highlight API.
<h2 id=responding-to-changes>
Responding to Changes</h2>
<h3 id=repaint>
Repaints</h3>
The addition or removal
of a [=custom highlight=] in the [=highlight registry=],
or of a [=range=] in a [=registered=] [=custom highlight=],
must cause the user agent to reevaluate the rendering,
and to repaint if appropriate.
The user agent must also repaint highlights as needed
in response to changes by the author
to the {{Highlight/priority}},
or to the [=boundary points=] of {{Range}}s
of a [=registered=] [=custom highlight=].
This repaint is asynchronous, and the APIs mentioned above must not block while waiting for the
repaint to happen.
<h3 id=range-invalidation>
Range Updating and Invalidation</h3>
Authors can build [=custom highlights=] using either {{Range}}s or {{StaticRange}}s.
The resulting [=custom highlight=] represents the same parts of the document,
and can be styled identically.
However, the behavior is different
in case the underlying document is modified.
{{Range}}s are [=live ranges=].
The user agent will adjust the [=boundary points=] of {{Range}}s
in response to DOM changes overlapping the range or at its boundary,
and [[#repaint|repaint]] accordingly.
[=Boundary points=] of [=live ranges=] can also be changed
by the author.
On the other hand,
the user agent must not adjust the [=boundary points=] of {{StaticRange}}s
in response to DOM changes,
nor can they be modified by the author after creation.
The user agent is expected to store the actual {{StaticRange}}s, rather than backing
them up with live {{Range}}s.
<div class=advisement>
Updating all {{Range}} objects as the DOM is modified
has a significant performance cost.
Authors who intend to observe DOM changes and react to them
by adjusting or recreating the ranges in their [=custom highlights=]
are strongly encouraged to use {{StaticRange}}s
in order to avoid this costly but unnecessary step.
Conversely, authors who use {{StaticRange}}s
should observe and react to DOM changes,
by discarding stale [=ranges=] or [=custom highlights=]
and recreating new ones.
</div>
When computing how to render a document,
if [=start node=] or [=end node=] of any [=range=]
in the [=highlight registry=] associated with that document's window
refer to a {{Node}} whose [=shadow-including root=] is not that document,
the user agent must ignore that [=range=].
If any {{StaticRange}} in the [=highlight registry=] associated with that document's window
is not <a spec=dom for="StaticRange">valid</a>,
the user agent must ignore that [=range=].
<h2 id="interactions">
Interacting with Custom Highlights</h2>
The {{highlightsFromPoint}}(<var>x</var>, <var>y</var>, <var>options</var>) method allows developers to build scenarios
involving user interaction with [=custom highlights=]. The method returns a [=sequence=] containing the [=custom highlights=]
at a given <var>x</var>, <var>y</var> coordinate. The [=custom highlights=] are listed in this [=sequence=] in
descending [=priority=] order.
By default, [=custom highlights=] in a [=shadow tree=] are not returned, but the developer has the possibility to pass in
an optional <var>options</var> [=dictionary=] with a <var>shadowRoots</var> property containing a [=sequence=] of {{ShadowRoot}}
objects. [=custom highlights|Highlights=] contained within a [=shadow tree=] provided in this way will be returned.
<div class=example id=highlights-from-point-ex>
The following example shows a way to use {{highlightsFromPoint}} to interact with mouse click [=event|events=].
<xmp highlight=html>
<style>
:root::highlight(foo) {
background-color:yellow;
}
:root::highlight(bar) {
color:red;
}
</style>
<body>abc
<script>
document.addEventListener('click', (event) => {
const mouseX = event.clientX;
const mouseY = event.clientY;
console.log(CSS.highlights.highlightsFromPoint(mouseX, mouseY));
});
let textNode = document.body.firstChild;
let r1 = new Range();
r1.setStart(textNode, 0);
r1.setEnd(textNode, 2);
let r2 = new Range();
r2.setStart(textNode, 1);
r2.setEnd(textNode, 2);
let h1 = new Highlight(r1);
let h2 = new Highlight(r2);
h1.priority = 1;
h2.priority = 2;
CSS.highlights.set("foo", h1);
CSS.highlights.set("bar", h2);
</script>
</xmp>
The code above will display the following styled text, note that "b" is affected by both [=custom highlight|highlights=]
<var>h1</var> and <var>h2</var>, whereas "a" is only affected by <var>h1</var>:
<div class=sample-out style="color:black">
<span style="background:yellow;">a</span><span style="background:yellow;color:red;">b</span><span>c</span>
</div>
In this example there's an [=event listener=] set on click [=event|events=] that logs the [=custom highlights=]
present at the point where the click was made.
The following [=sequence|sequences=] are some examples of what will be printed to console after a click:
* <code>[ <var>h1</var> ]</code>, if the user clicks on character "a".
* <code>[ <var>h2</var>, <var>h1</var> ]</code>, if the user clicks on character "b",
as <var>h2</var> has higher priority than <var>h1</var>.
* <code>[]</code>, if the user clicks on character "c".
</div>
The method {{highlightsFromPoint}} is defined as part of the {{HighlightRegistry}} interface as follows:
<pre class=idl>
partial interface HighlightRegistry {
sequence<Highlight> highlightsFromPoint(float x, float y, optional <span>HighlightsFromPointOptions</span> options = {});
};
dictionary <dfn dictionary>HighlightsFromPointOptions</dfn> {
sequence<ShadowRoot> <dfn dict-member for="HighlightsFromPointOptions">shadowRoots</dfn> = [];
};
</pre>
The <dfn method for="HighlightRegistry">highlightsFromPoint(<var>x</var>, <var>y</var>, <var>options</var>)</dfn>
method must return the result of running these steps:
1. If any of the following are true, return the empty [=sequence=]:
* <var>x</var> is negative
* <var>y</var> is negative
* <var>x</var> is greater than the <a>viewport</a> width excluding the size of a rendered scroll bar (if any)
* <var>y</var> is greater than the <a>viewport</a> height excluding the size of a rendered scroll bar (if any)
1. Otherwise, return a [=sequence=] of [=custom highlights=] given by ordering the highlights contained in this {{HighlightRegistry}} in descending order of [=priority=],
including only those highlights that contain at least one {{AbstractRange}} <var>abstractRange</var> that satisfies the following:
* Let <var>range</var> be a {{Range}} object whose [=start node=] and [=end node=] are set to <var>abstractRange</var>'s [=start node=] and [=end node=] respectively,
and [=start offset=] and [=end offset=] are set to <var>abstractRange</var>'s [=start offset=] and [=end offset=] respectively.
* The coordinates <var>x</var>,<var>y</var> fall inside at least one of the {{DOMRect}}s returned by calling {{Range/getClientRects()}} on <var>range</var>.
Note: The specifics of hit testing are out of scope of this
specification and therefore the exact details of
{{highlightsFromPoint()}} are therefore too. Hit testing
will hopefully be defined in a future revision of CSS or HTML.
* The <var>range</var>'s {{commonAncestorContainer}} is not in a [=shadow tree=] or is in a [=shadow tree=] whose
[=shadow root=] is [=list/contains|contained by=] by <var>options</var>.<var>shadowRoots</var>.
<h2 id=events>
Event Handling</h2>
Issue: Section on Events TBD, based on https://github.com/MicrosoftEdge/MSEdgeExplainers/blob/master/highlight/events-explainer.md
Issue: should custom highlights have a dedicated event handling mechanism,
or should that be added to pseudo-elements in general?
<div class=non-normative>
<h2 class="no-num" id=priv>
Appendix A. Privacy Considerations</h2>
<em>This section is non-normative</em>.
This specification is not thought to introduce any new privacy concern.
Anyone suspecting that this is not accurate is encouraged
to get in touch with the CSS Working Group or the co-editors.
<h2 class="no-num" id=sec>
Appendix B. Security Considerations</h2>
<em>This section is non-normative</em>.
This specification is not thought to introduce any new security concern.
Anyone suspecting that this is not accurate is encouraged
to get in touch with the CSS Working Group or the co-editors.
</div>
<div class=non-normative>
<h2 class="no-num" id="credits">
Appendix C. Acknowledgements</h2>
<em>This section is non-normative</em>.
Issue: Acknowledge people (other than editors) who deserve credit for this.
</div>
<div class=non-normative>
<h2 class="no-num" id="changes">
Appendix D. Changes</h2>
<em>This section is non-normative</em>.
<h3 class="no-num" id="changes-since-20211215">
Changes since the <a href="https://www.w3.org/TR/2020/WD-css-highlight-api-1-20211215/">15 December 2021 Working Draft</a></h3>
In addition to various editorial improvements and minor tweaks,
the main changes are:
* Added a {{HighlightRegistry/highlightsFromPoint}} method to
{{HighlightRegistry}}.
(See <a href="https://github.com/w3c/csswg-drafts/pull/7513">Issue 7513</a>)
* Specified that highlight repainting has to be done asynchronously.
(See <a href="https://github.com/w3c/csswg-drafts/pull/6987">Issue 6987</a>)
* Clarify that UAs cannot specify paired default highlight colors for custom highlights.
(See <a href="https://github.com/w3c/csswg-drafts/pull/6665">Issue 6665</a>)
* Clarify that there is no restriction on custom highlights crossing containment boundaries.
(See <a href="https://github.com/w3c/csswg-drafts/issues/4598">Issue 4598</a>)
* Added an I18N warning about placement of [=range=] endpoints in the middle of a
[=grapheme=] or [=supplementary character=].
<h3 class="no-num" id="changes-since-20201208">
Changes since the <a href="https://www.w3.org/TR/2020/WD-css-highlight-api-1-20201208/">8 December 2020 Working Draft</a></h3>
In addition to various editorial improvements and minor tweaks,
the main changes are:
* Renamed <code>HighlightsRegister</code> to {{HighlightRegistry}}
* Removed the redundant <code>add()</code> method from {{HighlightRegistry}}.
(See <a href="https://github.com/w3c/csswg-drafts/pull/6092">Issue 6092</a>)
* Make custom highlight overlays stack below native highlight overlays.
(See <a href="https://github.com/w3c/csswg-drafts/pull/4595">Issue 4595</a>)
* Handle highlight priority with integers rather than floats.
(See <a href="https://github.com/w3c/csswg-drafts/pull/4592">Issue 4592</a>)
* Define the default value for highlight priority to be 0.
(See <a href="https://github.com/w3c/csswg-drafts/pull/6136">Issue 6136</a>)
* Made HighlightRegistry maplike (rather than setlike)
and remove <code>name</code> property from Highlight.
(See <a href="https://github.com/w3c/csswg-drafts/pull/5910">Issue 5910</a>)
* Clarified that ranges from the wrong window are not painted.
(See <a href="https://github.com/w3c/csswg-drafts/pull/6417">Issue 6417</a>)
* Specify that custom highlights have no UA styles.
(See <a href="https://github.com/w3c/csswg-drafts/pull/6375">Issue 6375</a>)
* Deferred to the [[DOM]] specification for range invalidation
(See <a href="https://github.com/w3c/csswg-drafts/pull/4597">Issue 4597</a>)
* Added a {{Highlight/type}} attribute to {{Highlight}}
to give clearer semantics to different highlights,
in support of exposing highlights to accessibility tools.
(See <a href="https://github.com/w3c/csswg-drafts/pull/6498">Issue 6498</a>)
<h3 class="no-num" id="changes-since-20201022">
Changes since the <a href="https://www.w3.org/TR/2020/WD-css-highlight-api-1-20201022/">22 October 2020 Working Draft</a></h3>
There have been only editorial changes since the <a href="https://www.w3.org/TR/2020/WD-css-highlight-api-1-20201022/">22 October 2020 Working Draft</a>;
see <a href="https://services.w3.org/htmldiff?doc1=https%3A%2F%2Fwww.w3.org%2FTR%2F2020%2FWD-css-highlight-api-1-20201022%2F&doc2=https%3A%2F%2Fwww.w3.org%2FTR%2Fcss-highlight-api-1%2F">diffs</a>.
</div>
<!-- This text from the explainer seems useful, and should go somewhere. But where?
----
If there are DOM/CSS changes that result in a different cascaded highlight map for a given element,
and there exists one or more Range objects in the highlight registry for the cascaded identifiers,
the layout representation of that element should be notified that the painting of the element might have changed.
Ranges that are positioned inside of documents that are not in the view are ignored.
The HighlightRegistry is per-document — therefore, Ranges that are positioned inside of a different document than the HighlightRegistry it is a part of are ignored for rendering.
-->