FIX: remove div used to measure textarea position

Also corrects the positioning of autocomplete (when typing @ or emoji)

Previously there were edge conditions where autocomplete would be hundreds
of pixels away due to a bug measuring.


This correct an issue where Firefox ends up having an enormous blank space
at the bottom of topics after editing.
This commit is contained in:
Sam 2018-09-13 14:53:19 +10:00
parent a3b3b0810d
commit daa02431df
2 changed files with 52 additions and 17 deletions

View File

@ -285,11 +285,10 @@ export default function(options) {
hOffset = 0; hOffset = 0;
} else { } else {
pos = me.caretPosition({ pos = me.caretPosition({
pos: completeStart, pos: completeStart + 1
key: options.key
}); });
hOffset = 27; hOffset = 10;
if (options.treatAsTextarea) vOffset = -32; if (options.treatAsTextarea) vOffset = -32;
} }

View File

@ -1,3 +1,6 @@
// TODO: This code should be moved to lib, it was heavily modified by us over the years, and mostly written by us
// except for the little snippet from StackOverflow
//
// http://stackoverflow.com/questions/263743/how-to-get-caret-position-in-textarea // http://stackoverflow.com/questions/263743/how-to-get-caret-position-in-textarea
var clone, getCaret; var clone, getCaret;
getCaret = function(el) { getCaret = function(el) {
@ -29,7 +32,22 @@ $.fn.caret = function(){
@module $.fn.caretPosition @module $.fn.caretPosition
**/ **/
$.fn.caretPosition = function(options) { $.fn.caretPosition = function(options) {
var after, before, getStyles, guard, html, important, insertSpaceAfterBefore, letter, makeCursor, p, pPos, pos, span, styles, textarea, val; var after,
before,
getStyles,
guard,
html,
important,
insertSpaceAfterBefore,
letter,
makeCursor,
p,
pPos,
pos,
span,
styles,
textarea,
val;
if (clone) { if (clone) {
clone.remove(); clone.remove();
} }
@ -44,17 +62,15 @@ $.fn.caretPosition = function(options) {
} }
}; };
styles = getStyles(textarea[0]);
clone = $("<div><p></p></div>").appendTo("body");
p = clone.find("p");
clone.width(textarea.width());
clone.height(textarea.height());
important = function(prop) { important = function(prop) {
return styles.getPropertyValue(prop); return styles.getPropertyValue(prop);
}; };
const isRTL = $('html').hasClass('rtl'); styles = getStyles(textarea[0]);
clone = $("<div><p></p></div>").appendTo("body");
p = clone.find("p");
const isRTL = $("html").hasClass("rtl");
clone.css({ clone.css({
border: "1px solid black", border: "1px solid black",
padding: important("padding"), padding: important("padding"),
@ -77,7 +93,14 @@ $.fn.caretPosition = function(options) {
"line-height": important("line-height") "line-height": important("line-height")
}); });
pos = options && (options.pos || options.pos === 0) ? options.pos : getCaret(textarea[0]); clone.width(textarea.width());
clone.height(textarea.height());
pos =
options && (options.pos || options.pos === 0)
? options.pos
: getCaret(textarea[0]);
val = textarea.val().replace("\r", ""); val = textarea.val().replace("\r", "");
if (options && options.key) { if (options && options.key) {
val = val.substring(0, pos) + options.key + val.substring(pos); val = val.substring(0, pos) + options.key + val.substring(pos);
@ -103,12 +126,23 @@ $.fn.caretPosition = function(options) {
var l; var l;
l = val.substring(pos, pos + 1); l = val.substring(pos, pos + 1);
if (l === "\n") return "<br>"; if (l === "\n") return "<br>";
return "<span class='" + klass + "' style='background-color:" + color + "; margin:0; padding: 0'>" + guard(l) + "</span>"; return (
"<span class='" +
klass +
"' style='background-color:" +
color +
"; margin:0; padding: 0'>" +
guard(l) +
"</span>"
);
}; };
html = ""; html = "";
if (before >= 0) { if (before >= 0) {
html += guard(val.substring(0, pos - 1)) + makeCursor(before, "before", "#d0ffff"); html +=
guard(val.substring(0, pos - 1)) +
makeCursor(before, "before", "#d0ffff");
if (insertSpaceAfterBefore) { if (insertSpaceAfterBefore) {
html += makeCursor(0, "post-before", "#d0ffff"); html += makeCursor(0, "post-before", "#d0ffff");
} }
@ -130,9 +164,11 @@ $.fn.caretPosition = function(options) {
} }
pPos = p.offset(); pPos = p.offset();
return { let position = {
left: pos.left - pPos.left, left: pos.left - pPos.left,
top: (pos.top - pPos.top) - clone.scrollTop() top: pos.top - pPos.top - clone.scrollTop()
}; };
clone.remove();
return position;
}; };