mirror of
https://github.com/livebook-dev/livebook.git
synced 2025-01-11 01:18:12 +08:00
65 lines
1.7 KiB
JavaScript
65 lines
1.7 KiB
JavaScript
import { getAttributeOrThrow } from "../lib/attribute";
|
|
|
|
/**
|
|
* A hook used on [contenteditable] elements to update the specified
|
|
* attribute with the element text.
|
|
*
|
|
* Configuration:
|
|
*
|
|
* * `data-update-attribute` - the name of the attribute to update when content changes
|
|
*/
|
|
const ContentEditable = {
|
|
mounted() {
|
|
this.props = getProps(this);
|
|
|
|
this.__updateAttribute();
|
|
|
|
// Set the specified attribute on every content change
|
|
this.el.addEventListener("input", (event) => {
|
|
this.__updateAttribute();
|
|
});
|
|
|
|
// Make sure only plain text is pasted
|
|
this.el.addEventListener("paste", (event) => {
|
|
event.preventDefault();
|
|
const text = event.clipboardData.getData("text/plain");
|
|
document.execCommand("insertText", false, text);
|
|
});
|
|
|
|
// Unfocus the element on Enter or Escape
|
|
this.el.addEventListener("keydown", (event) => {
|
|
if (event.key === "Enter" || event.key === "Escape") {
|
|
this.el.blur();
|
|
}
|
|
});
|
|
|
|
// While the element is focused, ignore the incoming changes
|
|
this.el.addEventListener("focus", (event) => {
|
|
this.el.setAttribute("phx-update", "ignore");
|
|
});
|
|
|
|
this.el.addEventListener("blur", (event) => {
|
|
this.el.removeAttribute("phx-update");
|
|
});
|
|
},
|
|
|
|
updated() {
|
|
this.props = getProps(this);
|
|
|
|
// The element has been re-rendered so we have to add the attribute back
|
|
this.__updateAttribute();
|
|
},
|
|
|
|
__updateAttribute() {
|
|
const value = this.el.innerText.trim();
|
|
this.el.setAttribute(this.props.attribute, value);
|
|
},
|
|
};
|
|
|
|
function getProps(hook) {
|
|
return {
|
|
attribute: getAttributeOrThrow(hook.el, "data-update-attribute"),
|
|
};
|
|
}
|
|
|
|
export default ContentEditable;
|