Skip to content

Commit

Permalink
Add firePhxChangeWhileComposing opt to live_socket
Browse files Browse the repository at this point in the history
  • Loading branch information
nallwhy committed Dec 18, 2024
1 parent ea40066 commit e0fd81b
Show file tree
Hide file tree
Showing 7 changed files with 26 additions and 13 deletions.
20 changes: 15 additions & 5 deletions assets/js/phoenix_live_view/live_socket.js
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,18 @@
* @param {Object} [opts.localStorage] - An optional Storage compatible object
* Useful for when LiveView won't have access to `localStorage`.
* See `opts.sessionStorage` for examples.
* @param {boolean} [opts.firePhxChangeWhileComposing] - If set to `true`, ensures that
* `phx-change` events are triggered even while the user is composing input using IME
* (Input Method Editor). This is determined by the `e.isComposing` property on keyboard events,
* which is `true` when the user is in the process of entering composed characters (e.g., typing
* in languages like Japanese, Chinese, etc.). By default, `phx-change` will not fire during a
* composition session to prevent potential issues, especially on Safari, where input updates
* during composition can cause unexpected behavior. For more information, see:
*
* - https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/isComposing
* - https://github.com/phoenixframework/phoenix_live_view/issues/3322
*
* Defaults to `false`.
*/

import {
Expand Down Expand Up @@ -159,6 +171,7 @@ export default class LiveSocket {
this.failsafeJitter = opts.failsafeJitter || FAILSAFE_JITTER
this.localStorage = opts.localStorage || window.localStorage
this.sessionStorage = opts.sessionStorage || window.sessionStorage
this.firePhxChangeWhileComposing = opts.firePhxChangeWhileComposing || false
this.boundTopLevelEvents = false
this.boundEventNames = new Set()
this.serverCloseRef = null
Expand Down Expand Up @@ -901,11 +914,8 @@ export default class LiveSocket {
}
let phxChange = this.binding("change")
let input = e.target
// do not fire phx-change if we are in the middle of a composition session
// https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/isComposing
// Safari has issues if the input is updated while composing
// see https://github.com/phoenixframework/phoenix_live_view/issues/3322
if(e.isComposing){

if(!this.firePhxChangeWhileComposing && e.isComposing){
const key = `composition-listener-${type}`
if(!DOM.private(input, key)){
DOM.putPrivate(input, key, true)
Expand Down
3 changes: 2 additions & 1 deletion priv/static/phoenix_live_view.cjs.js

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 2 additions & 2 deletions priv/static/phoenix_live_view.cjs.js.map

Large diffs are not rendered by default.

3 changes: 2 additions & 1 deletion priv/static/phoenix_live_view.esm.js

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 2 additions & 2 deletions priv/static/phoenix_live_view.esm.js.map

Large diffs are not rendered by default.

3 changes: 2 additions & 1 deletion priv/static/phoenix_live_view.js
Original file line number Diff line number Diff line change
Expand Up @@ -4734,6 +4734,7 @@ removing illegal node: "${(childNode.outerHTML || childNode.nodeValue).trim()}"
this.failsafeJitter = opts.failsafeJitter || FAILSAFE_JITTER;
this.localStorage = opts.localStorage || window.localStorage;
this.sessionStorage = opts.sessionStorage || window.sessionStorage;
this.firePhxChangeWhileComposing = opts.firePhxChangeWhileComposing || false;
this.boundTopLevelEvents = false;
this.boundEventNames = /* @__PURE__ */ new Set();
this.serverCloseRef = null;
Expand Down Expand Up @@ -5459,7 +5460,7 @@ removing illegal node: "${(childNode.outerHTML || childNode.nodeValue).trim()}"
}
let phxChange = this.binding("change");
let input = e.target;
if (e.isComposing) {
if (!this.firePhxChangeWhileComposing && e.isComposing) {
const key = `composition-listener-${type}`;
if (!dom_default.private(input, key)) {
dom_default.putPrivate(input, key, true);
Expand Down
2 changes: 1 addition & 1 deletion priv/static/phoenix_live_view.min.js

Large diffs are not rendered by default.

0 comments on commit e0fd81b

Please sign in to comment.