-
Notifications
You must be signed in to change notification settings - Fork 39
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Proposal: compositionborder attribute #414
Comments
The problem that you are describing here can also be fixed by using the EditContext API, so not sure if we need to invent another approach to handle composition events. |
Yeah, I was aware of that API. It has been 3+ years and EditContext hasn't been implemented in any browser still. So the intent here with |
Oh, really? This is terrible result... I guess that your IME selects the word and replace it. I mean that this is probably depends on IME. If the
It's a job of IME that which characters are treated as word separator. Browsers let IME know surrounding text as plaintext.
I guess that CodeMirror just touches the DOM tree after committing composition. That's the safest way. Updating the DOM tree will be notified IME. Then, IME may stop the composition, i.e., the behavior becomes depending on IME. That must be a nightmare for web developers.
I think that this should be fixed in each browser without any spec changes. Could be avoided with inserting empty text node there and collapse |
In Firefox, using For example:
Will suggest "text" for autocomplete, but switching to
In my tests, you cannot modify the DOM during composition. Chrome will allow it, but the IME (as tested in Android) does a janky refresh if you do a DOM modification. Firefox will silently proceed, but has some weird behavior where the remaining composition events are all fired incorrectly on orphaned nodes. I suppose that was one of the stipulations of the Input Events Lvl 2 spec: all events could be prevented (e.g. you can manually handle DOM edits), except for composition events, which you must wait to complete before doing any DOM repairs. (If you could do preventDefault on composition events then that would solve everything, no need for In any case, modifying the DOM during composition without preventDefault would be problematic, since the browser might restyle/add/delete nodes based on whatever internal DOM editing logic it has. You'd be constantly fighting to repair the DOM. Which is sort of the idea behind
Tested now and using a text node seems to work for Firefox, but Chrome still retargets to the |
2023-02-09 call:
|
The idea is to allow a
compositionborder=true
attribute to elements within a content editable container. This signals to the browser that composition should not cross the element's boundary. When IME composition begins, the initial composition text is bounded bycompositionborder
. Input event Ranges will necessarily be within those bounds. An input event Range outside the range would necessarily trigger acompositionend
event first.Purpose
Composition events cannot be handled manually (specifically
preventDefault
is a not allowed in the InputEvents spec). During composition editing, browsers may add/delete nodes, or move textContent to adjacent nodes. This can cause a "style jank" where a style is modified and we must wait until composition has ended to repair the styles. This is only a problem with styled text, as with plain text we can repair the DOM afterwards without any visible jank. This also only affects compositions, since regular events we can cancel and manually edit the DOM.Here are some examples of unpredictable browser behavior around compositions. Each of these result in style jank, but could be solved by adding
compositionborder=true
to one or more elements.(Vertical bar indicates caret position)
1)
Autocomplete "picturesque" in Chrome, or typing "e" in Firefox Android will result in:
ruining our intended styling.
2)
In a code editor, the same might occur with:
The browser does not know that we're dealing with programming code, and so thinks
shor-ter
is a single word/phrase that should be composed together with the IME.This particular example can be observed in CodeMirror when editing on Android; when typing
this-some_var
, syntax highlighting is not properly applied until after composition finishes.3)
Typing "here" will result in
The browser treats the contents as one style-agnostic text phrase, ignoring that the cursor is inside
<u>
and retargeting it to<b>
instead. While normal input events we can cancel and retarget it back to<u>
, with composition events we can't edit the DOM until after composition has completed; and by that point, it is too late, since the text has already been styled bold.4)
(Assuming custom-element has a shadow root given by the template). Here, browsers interpret "Doeis" as the phrase for composition, even though the actual text being displayed is "~ John Doe ~ is...". A similar example can be crafted for before/after CSS pseudo elements.
Usage
Add
compositionborder=true
to an element to restrict IME composition. For example, given a caret given by vertical bar, the brackets indicate the composition bounds:In this example, the browser would insert text inside
<u>
instead of crossing to another element:Using the syntax of insertAdjacentElement, the composition boundary is the first position preceding/following the anchor whose element compositionborder is true or isContentEditable is false.
Using the node-boundary package, code for the range would look like:
Implementation
The idea here is a simple change to encapsulate composition events so they are more controllable. While browser devs have made clear full composition handling cannot be supported easily, this should be a simple addition which I believe solves the majority of complaints surrounding composition events. Browsers already seem to implement some logic surrounding this, although it is not exposed; for example, Firefox appears to exhibit
compositionborder=true
behavior for<article>
.The text was updated successfully, but these errors were encountered: