前端-熱鍵開發(fā)

最近開發(fā)熱鍵功能,有所感悟登舞。
從產(chǎn)品的角度思考,希望客戶可以操作簡單一點悬荣,通過上下左右方向鍵可以聚焦不同的元素菠秒,方便用戶操作。
從技術(shù)的角度思考

  • 問題1 鍵盤事件的監(jiān)聽
  • 問題2 這個是一個全局的行為
  • 問題3 我需要找到當(dāng)前聚焦元素距離上 下 左 右最近的元素氯迂,并且聚焦践叠。
首先問題1

因為鍵盤原生的左右是用來調(diào)整input 聚焦位置的,所以優(yōu)先考慮組合件比如option+上嚼蚀。
github上找到mousetrap 鍵盤快捷鍵庫禁灼。

問題2

全局行為的話,可以在main.ts里面注冊轿曙∨叮可以實現(xiàn)一個hooks。

問題3

這個也是最重要的拳芙。
邏輯: 觸發(fā)鍵盤按鈕找到所有的可以聚焦的元素察藐,算出每個元素距離上 左的距離。拿到當(dāng)前聚焦的元素舟扎。如果是左右鍵 直接在所有可以聚焦的list里面找到當(dāng)前聚焦的元素分飞,左鍵就是他的上一個,右鍵就是他的下一個睹限。上下鍵的邏輯比較復(fù)雜譬猫,如果是上 ,需要找到距離聚焦元素最近的所有元素羡疗,如果只有一個直接聚焦染服,如果是多個,繼續(xù)判斷l(xiāng)eft叨恨,距離最近的所有元素每個元素的left - 聚焦元素的left取絕對值柳刮。取到最小值的下標(biāo),選中這個下標(biāo)對應(yīng)el元素即可痒钝。下也同理秉颗。只是判斷的是 bottom 而已。
代碼如下

import Mousetrap from "mousetrap";
import "@/utils/mousetrap/index";

type EleArr = {
  el: HTMLElement;
  top: number;
  left: number;
};

const useHotKeys = async () => {
  let eleArr: EleArr[] = [];

  function setEleArr() {
    eleArr = [];
    let dom = document.body.classList.contains("el-popup-parent--hidden")
      ? Array.from(document.body.querySelectorAll("#app>.el-overlay")).at(-1)
      : document.querySelector(".el-main");
    if (!dom) {
      return;
    }
    const focusableElements = dom.querySelectorAll("a[href], input, select, textarea");
    focusableElements.forEach(element => {
      const rect = element.getBoundingClientRect();
      if (rect.top || rect.left) {
        eleArr.push({
          el: element as HTMLElement,
          top: rect.top + window.scrollY + (element.classList.contains("el-select__input") ? -7 : 0),
          left: rect.left + window.scrollX
        });
      }
    });
  }

  function getFocusedEle() {
    return document.activeElement as HTMLElement;
  }

  function operateKeys(type) {
    if (!eleArr.length) {
      return;
    }
    const el = getFocusedEle();
    if (el) {
      const eleIndex = eleArr.map(item => item.el).indexOf(el);
      let activeEle = eleArr.at(eleIndex);
      switch (type) {
        case "left":
          eleArr.at(eleIndex - 1)!.el.focus();
          break;
        case "right":
          let index = eleIndex + 1 >= eleArr.length ? 0 : eleIndex + 1;
          eleArr.at(index)!.el.focus();
          break;
        case "up":
          if (activeEle) {
            // 先找到在上它上面的所有元素
            const underList = eleArr.filter(item => item.top < activeEle!.top);
            if (underList.length) {
              let mapTop = underList.map(item => item.top);
              let maxTopItem = underList[mapTop.indexOf(Math.max(...mapTop))];
              const recentlyEleList = eleArr.filter(item => item.top == maxTopItem!.top);
              if (recentlyEleList.length == 1) {
                recentlyEleList[0].el.focus();
                break;
              }
              const numList = recentlyEleList.map(item => {
                return Math.abs(item.left - activeEle!.left);
              });
              const minIndex = numList.indexOf(Math.min(...numList));
              if (minIndex != -1) {
                recentlyEleList[minIndex].el.focus();
              }
            }
          }
          break;
        case "down":
          if (activeEle) {
            // 先找到在它下面的所有元素
            const underList = eleArr.filter(item => item.top > activeEle!.top);
            if (underList.length) {
              const recentlyEleList = eleArr.filter(item => item.top == underList.at(0)!.top);
              if (recentlyEleList.length == 1) {
                recentlyEleList[0].el.focus();
                break;
              }
              const numList = recentlyEleList.map(item => {
                return Math.abs(item.left - activeEle!.left);
              });
              const minIndex = numList.indexOf(Math.min(...numList));
              if (minIndex != -1) {
                recentlyEleList[minIndex].el.focus();
              }
            }
          }
          break;
        default:
          console.warn("無效的指令");
          break;
      }
      // let elLeft = el.getBoundingClientRect().left + window.scrollX;
      // let elTop = el.getBoundingClientRect().top + window.scrollX;
    } else {
      if (eleArr.length) {
        eleArr[0].el.focus();
      }
    }
  }

  const toNext = () => {
    setEleArr();
    operateKeys("down");
  };

  const toPrev = () => {
    setEleArr();
    operateKeys("up");
  };

  const toLeft = () => {
    setEleArr();
    operateKeys("left");
  };

  const toRight = () => {
    setEleArr();
    console.log(eleArr);
    operateKeys("right");
  };

  return { toNext, toPrev, toLeft, toRight };
};

// 添加鍵盤事件
export const addKeyBoard = async () => {
  const { toLeft, toRight, toNext, toPrev } = await useHotKeys();
  Mousetrap.bindGlobal("option+up", () => {
    toPrev();
    return false;
  });
  Mousetrap.bindGlobal("option+down", () => {
    toNext();
    return false;
  });
  Mousetrap.bindGlobal("option+left", () => {
    toLeft();
    return false;
  });
  Mousetrap.bindGlobal("option+right", () => {
    toRight();
    return false;
  });
};

以上js在main.ts 里面引入addKeyBoard方法調(diào)用即可
解釋

因為還有彈框所以判斷了
let dom = document.body.classList.contains("el-popup-parent--hidden")
      ? Array.from(document.body.querySelectorAll("#app>.el-overlay")).at(-1)
      : document.querySelector(".el-main");
因為mousetrap在input里面實現(xiàn)鍵盤事件送矩,需要安裝插件蚕甥。可以在官網(wǎng)找到栋荸。
import Mousetrap from "mousetrap";
import "@/utils/mousetrap/index";


// @/utils/mousetrap/index  內(nèi)容
(function (a) {
  let c = {},
    d = a.prototype.stopCallback;
  a.prototype.stopCallback = function (e, b, a, f) {
    return this.paused ? !0 : c[a] || c[f] ? !1 : d.call(this, e, b, a);
  };
  a.prototype.bindGlobal = function (a, b, d) {
    this.bind(a, b, d);
    if (a instanceof Array) for (b = 0; b < a.length; b++) c[a[b]] = !0;
    else c[a] = !0;
  };
  a.init();
})(Mousetrap);

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末菇怀,一起剝皮案震驚了整個濱河市凭舶,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌爱沟,老刑警劉巖帅霜,帶你破解...
    沈念sama閱讀 216,402評論 6 499
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異钥顽,居然都是意外死亡义屏,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,377評論 3 392
  • 文/潘曉璐 我一進(jìn)店門蜂大,熙熙樓的掌柜王于貴愁眉苦臉地迎上來闽铐,“玉大人,你說我怎么就攤上這事奶浦⌒质” “怎么了?”我有些...
    開封第一講書人閱讀 162,483評論 0 353
  • 文/不壞的土叔 我叫張陵澳叉,是天一觀的道長隙咸。 經(jīng)常有香客問我,道長成洗,這世上最難降的妖魔是什么五督? 我笑而不...
    開封第一講書人閱讀 58,165評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮瓶殃,結(jié)果婚禮上充包,老公的妹妹穿的比我還像新娘。我一直安慰自己遥椿,他們只是感情好基矮,可當(dāng)我...
    茶點故事閱讀 67,176評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著冠场,像睡著了一般家浇。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上碴裙,一...
    開封第一講書人閱讀 51,146評論 1 297
  • 那天钢悲,我揣著相機與錄音,去河邊找鬼舔株。 笑死莺琳,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的督笆。 我是一名探鬼主播芦昔,決...
    沈念sama閱讀 40,032評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼诱贿,長吁一口氣:“原來是場噩夢啊……” “哼娃肿!你這毒婦竟也來了咕缎?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,896評論 0 274
  • 序言:老撾萬榮一對情侶失蹤料扰,失蹤者是張志新(化名)和其女友劉穎凭豪,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體晒杈,經(jīng)...
    沈念sama閱讀 45,311評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡嫂伞,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,536評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了拯钻。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片帖努。...
    茶點故事閱讀 39,696評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖粪般,靈堂內(nèi)的尸體忽然破棺而出拼余,到底是詐尸還是另有隱情,我是刑警寧澤亩歹,帶...
    沈念sama閱讀 35,413評論 5 343
  • 正文 年R本政府宣布匙监,位于F島的核電站,受9級特大地震影響小作,放射性物質(zhì)發(fā)生泄漏亭姥。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,008評論 3 325
  • 文/蒙蒙 一顾稀、第九天 我趴在偏房一處隱蔽的房頂上張望达罗。 院中可真熱鬧,春花似錦础拨、人聲如沸氮块。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽滔蝉。三九已至,卻和暖如春塔沃,著一層夾襖步出監(jiān)牢的瞬間蝠引,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,815評論 1 269
  • 我被黑心中介騙來泰國打工蛀柴, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留螃概,地道東北人。 一個月前我還...
    沈念sama閱讀 47,698評論 2 368
  • 正文 我出身青樓鸽疾,卻偏偏與公主長得像吊洼,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子制肮,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,592評論 2 353

推薦閱讀更多精彩內(nèi)容