Back To Articles

使用 vue-tribute 實作標記功能

🧑🏻‍💻 海豹人 Sealman 📅 July 19, 2020

Article Image

這篇文章主要說明如何透過 vue-tribute 來實作網頁上的標記功能(@mention),先備知識必須要已經基本會使用 Vue。

開始使用 vue-tribute

vue-tribute @ GitHub

vue-tribute 是將 ES6 Native 的 tribute 經過 Vue.js 的寫法包裝後的套件,方便支援 Vue.js 框架的寫法。

安裝 vue-tribute

我個人是習慣使用 npm 安裝套件。

npm install vue-tribute --save

使用 vue-tribute

我們可以在 vue-tribute 中使用 inputtextareacontenteditable 等 HTML 標籤,當我們輸入內容時,vue-tribute 會存取選項 tributeOptions 裡的屬性與方法,呈現出 @mention 後的畫面與功能。

<vue-tribute :options="tributeOptions">
  <input type="text" placeholder="@..." />
</vue-tribute>

如果想要對輸入框做進一步的處理,像是讓文字變色、轉為連結等功能,可以使用 contenteditable 來完成。

<vue-tribute :options="tributeOptions">
  <div contenteditable="true" id="myElement" placeholder="輸入留言"></div>
</vue-tribute>

參數選項設定

tributeOptions 是 vue-tribute 的參數選項,這個參數選項在 vue-tribute 與 tribute 套件中都是相同的,因為 vue-tribute 的參數設定就是沿用 tribute 的設定。

以下列出一些我自己常用的參數:

  • values:唯一的必填參數,就是你要用來搜尋的資料
  • trigger:觸發條件,可使用符號或字串
  • selectTemplate:選取後新增到輸入框裡面的內容,如果不是使用 contenteditable 元素,這邊就只能寫入純文字哩
  • menuItemTemplate:標記清單所呈現的組合樣板,以 Twitter 的 @mention 功能來說,就會呈現頭像、名稱、ID 等資料
  • lookup:通常放字串即可,但若要透過資料裡的多個屬性來搜尋,可以把 key 相加來做複合搜尋
  • requireLeadingSpace:觸發前要有空格
  • allowSpaces:@mention 的內容允許空格
  • menuItemLimit:清單最多呈現多少筆篩選結果
  • menuShowMinLength:觸發前至少要打幾個字
tributeOptions: {
  values: [],
  trigger: '@',
  selectTemplate(item) {
    return `<span><a href="http://twitter.com/${item.original.id}">@${item.original.name}</a></span>`;
  },
  menuItemTemplate(item) {
    return `<span>${item.original.name}</span>`;
  },
  lookup(item) {
    return item.friend_name + item.friend_suuid;
  },
  requireLeadingSpace: true,
  allowSpaces: false,
  menuItemLimit: 10,
  menuShowMinLength: 1,
},

這是一個簡易的參數設定範例,若有其他需求可以參考 所有參數 的說明。

處理標記結果

接著,我們要來處理 @mention 得到的結果。
由於 vue-tribute 沒有提供方法來綁定與監聽選取到的項目,所以我們只好從選取後的文字框當中,將標記到的對象給篩選出來。

篩選出所有的標記對象

要擷取使用者這則留言總共標記了哪些人,我們可以直接用正規表達式篩選出 @ 後面的文字,並將結果以陣列傳回後端。

// 計算留言 (content) 裡面標記了哪些人
const str = this.content;
const pattern = /\B@([a-z0-9_-]+)/gi; // 透過正規表達式查找符合規則的字段
const arr = str.match(pattern); // ['@sean', '@sealman']
let result = [];
if (arr) {
  result = arr.map((item) => item.substr(1)); // ['sean', 'sealman']
}

將文字轉換成連結

透過正規表達式,我們也可以將標記文字轉為連結,像是將 @vuejs 這個標記,轉換為 Twitter 使用者的個人頁面連結,像是 http://twitter.com/vuejs

// 將 @mention 轉為 twitter.com/mention 帳號的 Link
const replaceContent = content.replace(
  /\B@([a-z0-9_-]+)/gi, // 可位於開頭 or 左右有空格,可包含 _ 與 - 符號
  '<a href="http://twitter.com/$1">@$1</a>',
);

參考資料:Replace @mention with link - Stack Overflow

Troubleshooting

根據過往的開發經驗,如果需求比較複雜,或是想要實作出像是留言板等功能較完善的輸入框的話,輸入框可能就會選用 contenteditable 元素。不過 contenteditable 在使用時可能會遇到一些問題,以下就是我自己在實作中遇到過的幾個坑。

Contenteditable 滑鼠游標自動 Focus 至最後方

使用 contenteditable 時,如果有 Focus 游標的需求,無法直接透過一般的 focus() 完成,需要使用到 SelectionRange 屬性。

在 Stack Overflow 上 Nico Burnsjavascript - How to move cursor to end of contenteditable entity - Stack Overflow 這篇問題裡提到的解法,應該是目前可行的解法。

不過若 contenteditable 裡面有放其他元素,在 Node 節點上可能要再做判斷

function setEndOfContenteditable(contentEditableElement) {
  var range, selection;
  if (document.createRange) {
    range = document.createRange();
    range.selectNodeContents(contentEditableElement);
    range.collapse(false);
    selection = window.getSelection();
    selection.removeAllRanges();
    selection.addRange(range);
  } else if (document.selection) {
    range = document.body.createTextRange();
    range.moveToElementText(contentEditableElement);
    range.collapse(false);
    range.select();
  }
}

接著,我們透過 setEndOfContenteditable 觸發整個方法,完成將游標移動到最後面的功能。

const elem = document.querySelector(".input-area");
this.setEndOfContenteditable(elem); // 游標移到最後面

Contenteditable 複製貼上時限制為純文字

若要防止使用者貼上文字以外的內容,可以加上以下程式碼,防止使用者貼上任何內容。

Javascript trick for ‘paste as plain text` in execCommand - Stack Overflow 這個問題串的最佳解答,算是比起另一個網路上的熱門答案更簡單、更好理解,且不容易出現 Bug 的寫法。

editor.addEventListener("paste", function (e) {
  e.preventDefault();
  var text = (e.originalEvent || e).clipboardData.getData("text/plain");
  document.execCommand("insertHTML", false, text);
});

以上資源是我自己整理過後的筆記,若有錯誤歡迎隨時和我聯繫