React Hooks + TypeScript 做个仿 MacOS 桌面(二):实现 Docker 动效

这是我的项目记录系列文章第二篇,在上一篇我简单介绍了项目的初衷和流程等。现在这个项目已经做了一段时间,对 Hooks 和 TypeScript 也有了一定认识,相信优化和记录能有更多理解,同时可能收获大家的指导。

本篇文章我将梳理 Docker 效果的实现过程,你可以在我的项目 代码(欢迎 watch 和 star)体验,同时本文完整代码均在 sandbox 供你把玩。

基础结构搭建

我们会在 app 文件下创建 footer,在其内部引入我们的 Docker 组件,我们找到几张图标 png ,以图标名组成 dockList 通过 require 引入在 Docker 内部,它们就是本次主角,同时通过使用 useRef 钩子给它们的父亲 div 绑定一个 ref,便于后续操作。

我们给每个图标一个默认宽度 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>
  );
};

初始化样式

这里主要是设定每个图片的默认宽度和背景 div 的宽度和高度

同时这里用了 mouseleave,事实上也是鼠标离开 dock 事件所需函数(看下面就懂了),此处的 setDockStyle 事实上是重置时候用的。

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;
    }
  }
}

默认效果已经有了

事件逻辑

我们需要通过监听 Docker div 的鼠标进入和离开事件,一般我们使用 useCallback 缓存事件,同时使用 useEffect 作事件变化监听处理。mouseleave 我们已经在上面展示,不作展开。

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 动效思路

如图箭头,我们根据鼠标事件位置与各图标中心点距离来调整图标大小,通过该差值与设定宽度(这里我使用 dockStyle 初始宽度)比值作为图标放大参考,这里我使用的放大倍数为 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 动效就完成了。我们还可以通过修改 dockStyleWidth 来调整图标大小,当然还有动效放大倍数,甚至 Docker 位置,做到与 Mac 桌面一样,这也正式我的项目在做的东西。

小结

在写这篇文章的同时,也对代码和过程有了梳理,目前该项目已完成部分功能,包括简单设置,基础计算器,基础画板等,即使是这些已有功能也有很多需要完善的地方。

后续我会慢慢优化,并在相应模块代码优化到一定程度时不定时更新系列文章。

如果你喜欢这篇文章,不要忘了给我点赞。🍮

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

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,036评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,046评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,411评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,622评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,661评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,521评论 1 304
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,288评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,200评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,644评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,837评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,953评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,673评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,281评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,889评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,011评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,119评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,901评论 2 355