From bd5e28568f73b3c3867776bf7f9a6e61fc599a89 Mon Sep 17 00:00:00 2001 From: Fritz Lin Date: Tue, 9 Aug 2022 19:17:59 +0800 Subject: [PATCH] fix: should scrollIntoView when insert in contentEditable --- src/At.vue | 1 + src/util.js | 33 +++++++++++++++++++++++++++++++-- 2 files changed, 32 insertions(+), 2 deletions(-) diff --git a/src/At.vue b/src/At.vue index 5263cb1..17cab89 100644 --- a/src/At.vue +++ b/src/At.vue @@ -491,6 +491,7 @@ export default { const t = itemName(curItem) + suffix this.insertText(t, r) } + scrollIntoView(window.getSelection()) this.$emit('insert', curItem) this.handleInput() diff --git a/src/util.js b/src/util.js index b633152..6e108ef 100644 --- a/src/util.js +++ b/src/util.js @@ -1,14 +1,43 @@ +// Scrolling The Selection Into View +// http://roysharon.com/blog/37 +export function scrollIntoView (t, scrollParent) { + if (typeof (t) !== 'object') return + + if (t.getRangeAt) { + // we have a Selection object + if (t.rangeCount === 0) return + t = t.getRangeAt(0) + } + + if (t.cloneRange) { + // we have a Range object + let r = t.cloneRange() // do not modify the source range + r.collapse(true) // collapse to start + t = r.startContainer + // if start is an element, then startOffset is the child number + // in which the range starts + if (t.nodeType === 1) t = t.childNodes[r.startOffset] + } + + // if t is not an element node, then we need to skip back until we find the + // previous element with which we can call scrollIntoView() + let o = t + while (o && o.nodeType !== 1) o = o.previousSibling + t = o || t.parentNode + if (t) scrollIntoViewElement(t, scrollParent) +} + // bug report: https://github.com/vuejs/awesome-vue/pull/1028 // https://developer.mozilla.org/en-US/docs/Web/API/Element/scrollIntoViewIfNeeded -export function scrollIntoView (el, scrollParent) { +export function scrollIntoViewElement (el, scrollParent) { if (el.scrollIntoViewIfNeeded) { el.scrollIntoViewIfNeeded(false) // alignToCenter=false } else { // should not use `el.scrollIntoView(false)` // alignToTop=false // bug report: https://stackoverflow.com/questions/11039885/scrollintoview-causing-the-whole-page-to-move + scrollParent = scrollParent || el.parentElement const diff = el.offsetTop - scrollParent.scrollTop if (diff < 0 || diff > scrollParent.offsetHeight - el.offsetHeight) { - scrollParent = scrollParent || el.parentElement scrollParent.scrollTop = el.offsetTop } }