React Hooks + TypeScript 做個(gè)仿 MacOS 桌面(二):實(shí)現(xiàn) Docker 動(dòng)效

這是我的項(xiàng)目記錄系列文章第二篇,在上一篇我簡(jiǎn)單介紹了項(xiàng)目的初衷和流程等∈厣欤現(xiàn)在這個(gè)項(xiàng)目已經(jīng)做了一段時(shí)間滑肉,對(duì) Hooks 和 TypeScript 也有了一定認(rèn)識(shí)反肋,相信優(yōu)化和記錄能有更多理解,同時(shí)可能收獲大家的指導(dǎo)贷笛。

本篇文章我將梳理 Docker 效果的實(shí)現(xiàn)過程应又,你可以在我的項(xiàng)目 代碼(歡迎 watch 和 star)體驗(yàn),同時(shí)本文完整代碼均在 sandbox 供你把玩乏苦。

基礎(chǔ)結(jié)構(gòu)搭建

我們會(huì)在 app 文件下創(chuàng)建 footer丁频,在其內(nèi)部引入我們的 Docker 組件,我們找到幾張圖標(biāo) png 邑贴,以圖標(biāo)名組成 dockList 通過 require 引入在 Docker 內(nèi)部席里,它們就是本次主角,同時(shí)通過使用 useRef 鉤子給它們的父親 div 綁定一個(gè) ref拢驾,便于后續(xù)操作奖磁。

我們給每個(gè)圖標(biāo)一個(gè)默認(rèn)寬度 defaultWidth,由此可得到背景 div 的寬度和高度

import React, { useState, useRef } from "react";

export const Docker = () => {
  const [defaultWidth] = useState(76);
  const [dockList] = useState<string[]>([
    "Finder.png",
    "Launchpad.png",
    "PrefApp.png",
    "Chrome.png",
    "Terminal.png",
    "Calculator.png",
    "Drawing.png"
  ]);
  const [dockStyle, setDockStyle] = useState({
    width: defaultWidth * dockList.length,
    height: defaultWidth
  });
  const dockRef = useRef<HTMLDivElement>(null);

  return (
    <div ref={dockRef} style={dockStyle}>
      {dockList.map((item, index) => {
        return (
          <img src={require("./image/" + item)} alt={item} key={index + item} />
        );
      })}
    </div>
  );
};

初始化樣式

這里主要是設(shè)定每個(gè)圖片的默認(rèn)寬度和背景 div 的寬度和高度

同時(shí)這里用了 mouseleave繁疤,事實(shí)上也是鼠標(biāo)離開 dock 事件所需函數(shù)(看下面就懂了)咖为,此處的 setDockStyle 事實(shí)上是重置時(shí)候用的秕狰。

const mouseleave = useCallback(() => {
  if (!dockRef.current) {
    return;
  }
  setDockStyle({
    width: defaultWidth * dockList.length,
    height: defaultWidth
  });
  const imgList = dockRef.current.childNodes;
  for (let i = 0; i < imgList.length; i++) {
    const img = imgList[i] as HTMLImageElement;
    img.width = defaultWidth;
  }
}, [defaultWidth, setDockStyle, dockList]);

useEffect(() => {
  mouseleave();
}, [mouseleave]);

css 如下

.App {
  footer{
    position: fixed;
    bottom: 0;
    width: 100vw;
    display: flex;
    justify-content: center;
    div{
      display: flex;
      align-items: flex-end;
      background-color: rgba(222, 223, 227, 0.7);
      box-shadow: rgba(0, 0, 0, 0.31) 0px 0px 1px, rgba(0, 0, 0, 0.18) 0px 0px 5px,
        rgba(0, 0, 0, 0.3) 0px 8px 50px;
      border-top-left-radius: 0.4rem;
      border-top-right-radius: 0.4rem;
    }
  }
}

默認(rèn)效果已經(jīng)有了

事件邏輯

我們需要通過監(jiān)聽 Docker div 的鼠標(biāo)進(jìn)入和離開事件,一般我們使用 useCallback 緩存事件躁染,同時(shí)使用 useEffect 作事件變化監(jiān)聽處理鸣哀。mouseleave 我們已經(jīng)在上面展示,不作展開吞彤。

const mousemove = useCallback(e => {
  console.log(e);
}, []);
const mouseleave = useCallback(e => {
  console.log(e);
}, []);

useEffect(() => {
  if (!dockRef.current) {
    return;
  }
  const dockBackground: HTMLDivElement = dockRef.current;
  dockBackground.addEventListener("mousemove", mousemove);
  dockBackground.addEventListener("mouseleave", mouseleave);
  return () => {
    dockBackground.removeEventListener("mousemove", mousemove);
    dockBackground.removeEventListener("mouseleave", mouseleave);
  };
}, [mousemove, mouseleave]);

Docker 動(dòng)效思路

如圖箭頭我衬,我們根據(jù)鼠標(biāo)事件位置與各圖標(biāo)中心點(diǎn)距離來調(diào)整圖標(biāo)大小,通過該差值與設(shè)定寬度(這里我使用 dockStyle 初始寬度)比值作為圖標(biāo)放大參考饰恕,這里我使用的放大倍數(shù)為 2挠羔,直接看代碼:

const getOffset = useCallback(
  (el: HTMLElement, offset: "top" | "left"): number => {
    const elOffset = offset === "top" ? el.offsetTop : el.offsetLeft;
    if (el.offsetParent == null) {
      return elOffset;
    }
    return elOffset + getOffset(el.offsetParent as HTMLElement, offset);
  },
  []
);

const mousemove = useCallback(
  ({ clientX, clientY }) => {
    if (!dockRef.current) {
      return;
    }
    const imgList = dockRef.current.childNodes;
    let dockStyleWidth = 0;
    for (let i = 0; i < imgList.length; i++) {
      const img = imgList[i] as HTMLImageElement;
      const x = img.offsetLeft + defaultWidth / 2 - clientX;
      const y =
        img.offsetTop +
        getOffset(dockRef.current, "top") +
        img.offsetHeight / 2 -
        clientY;
      let imgScale =
        1 - Math.sqrt(x * x + y * y) / (imgList.length * defaultWidth);
      if (imgScale < 0.5) {
        imgScale = 0.5;
      }
      img.width = defaultWidth * 2 * imgScale;
      dockStyleWidth = dockStyleWidth + img.width;
    }
    setDockStyle({
      ...dockStyle,
      ...{ width: dockStyleWidth }
    });
  },
  [defaultWidth, getOffset, dockStyle]
);

至此,Docker 動(dòng)效就完成了埋嵌。我們還可以通過修改 dockStyleWidth 來調(diào)整圖標(biāo)大小破加,當(dāng)然還有動(dòng)效放大倍數(shù),甚至 Docker 位置雹嗦,做到與 Mac 桌面一樣范舀,這也正式我的項(xiàng)目在做的東西。

小結(jié)

在寫這篇文章的同時(shí)了罪,也對(duì)代碼和過程有了梳理尿背,目前該項(xiàng)目已完成部分功能,包括簡(jiǎn)單設(shè)置捶惜,基礎(chǔ)計(jì)算器田藐,基礎(chǔ)畫板等,即使是這些已有功能也有很多需要完善的地方吱七。

后續(xù)我會(huì)慢慢優(yōu)化汽久,并在相應(yīng)模塊代碼優(yōu)化到一定程度時(shí)不定時(shí)更新系列文章。

如果你喜歡這篇文章踊餐,不要忘了給我點(diǎn)贊景醇。??

本文參考
Mac Dock 效果及原理(勾股定理)

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市吝岭,隨后出現(xiàn)的幾起案子三痰,更是在濱河造成了極大的恐慌,老刑警劉巖窜管,帶你破解...
    沈念sama閱讀 217,734評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件散劫,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡幕帆,警方通過查閱死者的電腦和手機(jī)获搏,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來失乾,“玉大人常熙,你說我怎么就攤上這事纬乍。” “怎么了裸卫?”我有些...
    開封第一講書人閱讀 164,133評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵仿贬,是天一觀的道長。 經(jīng)常有香客問我墓贿,道長茧泪,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,532評(píng)論 1 293
  • 正文 為了忘掉前任募壕,我火速辦了婚禮调炬,結(jié)果婚禮上语盈,老公的妹妹穿的比我還像新娘舱馅。我一直安慰自己,他們只是感情好刀荒,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,585評(píng)論 6 392
  • 文/花漫 我一把揭開白布代嗤。 她就那樣靜靜地躺著,像睡著了一般缠借。 火紅的嫁衣襯著肌膚如雪干毅。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,462評(píng)論 1 302
  • 那天泼返,我揣著相機(jī)與錄音硝逢,去河邊找鬼。 笑死绅喉,一個(gè)胖子當(dāng)著我的面吹牛渠鸽,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播柴罐,決...
    沈念sama閱讀 40,262評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼徽缚,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了革屠?” 一聲冷哼從身側(cè)響起凿试,我...
    開封第一講書人閱讀 39,153評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎似芝,沒想到半個(gè)月后那婉,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,587評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡党瓮,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,792評(píng)論 3 336
  • 正文 我和宋清朗相戀三年吧恃,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片麻诀。...
    茶點(diǎn)故事閱讀 39,919評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡痕寓,死狀恐怖傲醉,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情呻率,我是刑警寧澤硬毕,帶...
    沈念sama閱讀 35,635評(píng)論 5 345
  • 正文 年R本政府宣布,位于F島的核電站礼仗,受9級(jí)特大地震影響吐咳,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜元践,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,237評(píng)論 3 329
  • 文/蒙蒙 一韭脊、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧单旁,春花似錦沪羔、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,855評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至愉豺,卻和暖如春篓吁,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背蚪拦。 一陣腳步聲響...
    開封第一講書人閱讀 32,983評(píng)論 1 269
  • 我被黑心中介騙來泰國打工杖剪, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人驰贷。 一個(gè)月前我還...
    沈念sama閱讀 48,048評(píng)論 3 370
  • 正文 我出身青樓盛嘿,卻偏偏與公主長得像,于是被迫代替她去往敵國和親饱苟。 傳聞我的和親對(duì)象是個(gè)殘疾皇子孩擂,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,864評(píng)論 2 354