From daa02431dfb5ea64990c776860ff634c088cf522 Mon Sep 17 00:00:00 2001 From: Sam Date: Thu, 13 Sep 2018 14:53:19 +1000 Subject: [PATCH] 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. --- .../discourse/lib/autocomplete.js.es6 | 5 +- vendor/assets/javascripts/caret_position.js | 64 +++++++++++++++---- 2 files changed, 52 insertions(+), 17 deletions(-) diff --git a/app/assets/javascripts/discourse/lib/autocomplete.js.es6 b/app/assets/javascripts/discourse/lib/autocomplete.js.es6 index 5900968a5ef..c229d8f1768 100644 --- a/app/assets/javascripts/discourse/lib/autocomplete.js.es6 +++ b/app/assets/javascripts/discourse/lib/autocomplete.js.es6 @@ -285,11 +285,10 @@ export default function(options) { hOffset = 0; } else { pos = me.caretPosition({ - pos: completeStart, - key: options.key + pos: completeStart + 1 }); - hOffset = 27; + hOffset = 10; if (options.treatAsTextarea) vOffset = -32; } diff --git a/vendor/assets/javascripts/caret_position.js b/vendor/assets/javascripts/caret_position.js index 8a7b2e0616e..094d504d4b5 100644 --- a/vendor/assets/javascripts/caret_position.js +++ b/vendor/assets/javascripts/caret_position.js @@ -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 var clone, getCaret; getCaret = function(el) { @@ -19,7 +22,7 @@ getCaret = function(el) { clone = null; -$.fn.caret = function(){ +$.fn.caret = function() { return getCaret(this[0]); }; @@ -29,7 +32,22 @@ $.fn.caret = function(){ @module $.fn.caretPosition **/ $.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) { clone.remove(); } @@ -44,17 +62,15 @@ $.fn.caretPosition = function(options) { } }; - styles = getStyles(textarea[0]); - clone = $("

").appendTo("body"); - p = clone.find("p"); - clone.width(textarea.width()); - clone.height(textarea.height()); - important = function(prop) { return styles.getPropertyValue(prop); }; - const isRTL = $('html').hasClass('rtl'); + styles = getStyles(textarea[0]); + clone = $("

").appendTo("body"); + p = clone.find("p"); + + const isRTL = $("html").hasClass("rtl"); clone.css({ border: "1px solid black", padding: important("padding"), @@ -77,7 +93,14 @@ $.fn.caretPosition = function(options) { "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", ""); if (options && options.key) { val = val.substring(0, pos) + options.key + val.substring(pos); @@ -103,12 +126,23 @@ $.fn.caretPosition = function(options) { var l; l = val.substring(pos, pos + 1); if (l === "\n") return "
"; - return "" + guard(l) + ""; + return ( + "" + + guard(l) + + "" + ); }; html = ""; + 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) { html += makeCursor(0, "post-before", "#d0ffff"); } @@ -130,9 +164,11 @@ $.fn.caretPosition = function(options) { } pPos = p.offset(); - return { + let position = { left: pos.left - pPos.left, - top: (pos.top - pPos.top) - clone.scrollTop() + top: pos.top - pPos.top - clone.scrollTop() }; + clone.remove(); + return position; };