React实现拖拽特效

前言

最近,我看到一个工程师的个人网站上,采用了拖拽作品集的互动特效,既有趣又吸引眼球。经过一些研究,我发现其实借助一些现成的套件,就能轻松实现这样的效果。今天就带大家一起看看,如何通过 Framer Motion 来制作这个特效吧!

安装 React + Framer Motion

我将使用 ReactFramer Motion 来实现这个特效。首先,用 Vite 来快速搭建一个 React 开发环境:

npm create vite@latest my-react-app -- --template react

执行完成后就依序执行以下指令,分别是:

  1. cd my-react-app : 移动到 my-react-app 资料夹
  2. npm install : 安装相关依赖
  3. npm run dev : 执行
cd my-react-app
npm install
npm run dev

打开浏览器,访问 http://localhost:5173,你应该会看到一个基础的 React 应用如下图。

并且也可以看到你的资料夹结构,如果只是想练习这个特效,直接改 App.jsx 就好

下一步就是安装 Framer Motion,直接在终端机打上以下指令就好:

npm install framer-motion

引入图片 & 创建容器

我们将图片保存在 ./public/drag-img/ 文件夹中,并用数组来存储图片路径。通过 Array.map() 方法,我们可以轻松地渲染出所有的图片。

const images = [
  '/drag-img/image-1.png',
  '/drag-img/image-2.png',
  '/drag-img/image-3.png',
  '/drag-img/image-4.png',
  '/drag-img/image-5.png',
  '/drag-img/image-6.png',
  '/drag-img/image-7.png',
  '/drag-img/image-8.png',
  '/drag-img/image-9.png',
  '/drag-img/image-10.png',
  '/drag-img/image-11.png',
  '/drag-img/image-12.png',
  '/drag-img/image-13.png',
  '/drag-img/image-14.png',
];

然后,我们创建一个容器来存放这些图片。为了方便后续操作,我们使用 useRef 来引用容器,以便后面获取容器的宽高。

export default function DragImg() {
  const containerRef = useRef(null);

  return (
    <div
      ref={containerRef}
      className='drag-img__container'
    >
       {/* 图片渲染 */}
    </div>
  );
}

接下来,稍微修改一下style样式

.drag-img__container {
  position: relative;
  width: 100vw;
  height: 100vh;
  overflow: hidden;
  background: #f0f0f0;
}

 

渲染图片 & 随机位置

通过刚才定义的 images 数组来渲染所有图片。这里使用的是 motion.img 标签,这样才能使用 Framer Motion 提供的动画和交互功能。

export default function DragImg() {
  const containerRef = useRef(null);

  return (
    <div
      ref={containerRef}
      className='drag-img__container'
    >
      {images.map((src, index) => (
        <motion.img
          key={index}
          src={src}
          className='drag-img__img'
          alt={`Image ${index + 1}`}
        />
      ))}
    </div>
  );
}
.drag-img__img {
  width: 200px;
  aspect-ratio: 4/3;
  object-fit: contain;
  padding: 4px;
  position: absolute;
  background: rgba(255, 255, 255, 0.2);
  border-radius: 6px;
  box-shadow: 0 0 2px rgba(0, 0, 0, 0.1);
  cursor: grab;
}

 稍微调整图片的宽度、比例,并让他的 position 是 absolute,其他就是一些小装饰,例如 padding、shadow 等等,现在所有的图片都会在右上角,因为我们还没调整他们的位置

接著可以利用 JavaScript 来随机图片的位置,顺便随机旋转的角度,让他有种散落在整个 container 的感觉。 

{images.map((src, index) => (
  <motion.img
    key={index}
    src={src}
    className='drag-img__img'
    alt={`Image ${index + 1}`}
    style={
  
  {
      top: `${Math.random() * (window.innerHeight - 200)}px`,
      left: `${Math.random() * (window.innerWidth - 150)}px`,
      rotate: `${Math.random() * 40 - 20}deg`,
    }}
  />
))}

实现拖拽效果

接下来,重点来了——使用 Framer Motion 来实现拖拽效果。由于 Framer Motion 内置了 drag 属性,整个过程非常简单,只需要在 motion.img 上添加 drag 属性,并指定拖拽范围。

{images.map((src, index) => (
  <motion.img
    key={index}
    src={src}
    className="drag-img__img"
    alt={`Image ${index + 1}`}
    style={
  
  {
      top: `${Math.random() * (window.innerHeight - 200)}px`,
      left: `${Math.random() * (window.innerWidth - 150)}px`,
      rotate: `${Math.random() * 40 - 20}deg`,
    }}
    drag
    dragConstraints={containerRef}  // 限制拖拽范围
    whileDrag={
  
  { scale: 1.1, rotate: 0 }}  // 拖拽时的样式调整
  />
))}

这里,我们添加了 dragConstraints,使图片只能在容器内拖动,不会超出边界。同时,使用 whileDrag 来调整拖拽时图片的缩放和旋转效果。

完整代码示例

到此为止就完全搞定了,其实非常简单!以下附上全部的代码: 

import { useRef } from 'react';
import { motion } from 'framer-motion';

const images = [
  '/drag-img/image-1.png',
  '/drag-img/image-2.png',
  '/drag-img/image-3.png',
  '/drag-img/image-4.png',
  '/drag-img/image-5.png',
  '/drag-img/image-6.png',
  '/drag-img/image-7.png',
  '/drag-img/image-8.png',
  '/drag-img/image-9.png',
  '/drag-img/image-10.png',
  '/drag-img/image-11.png',
  '/drag-img/image-12.png',
  '/drag-img/image-13.png',
  '/drag-img/image-14.png',
];

export default function DragImg() {
  const containerRef = useRef(null);

  return (
    <div
      ref={containerRef}
      className='drag-img__container'
    >
      {images.map((src, index) => (
        <motion.img
          key={index}
          src={src}
          className='drag-img__img'
          alt={`Image ${index + 1}`}
          style={
  
  {
            top: `${Math.random() * (window.innerHeight - 200)}px`,
            left: `${Math.random() * (window.innerWidth - 150)}px`,
            rotate: `${Math.random() * 40 - 20}deg`,
          }}
          drag
          dragConstraints={containerRef}
          whileDrag={
  
  { scale: 1.1, rotate: 0 }}
        />
      ))}
    </div>
  );
}
.drag-img__container {
  position: relative;
  width: 100vw;
  height: 100vh;
  overflow: hidden;
  background: #f0f0f0;
}

.drag-img__img {
  width: 200px;
  aspect-ratio: 4/3;
  object-fit: contain;
  padding: 4px;
  position: absolute;
  background: rgba(255, 255, 255, 0.2);
  border-radius: 6px;
  box-shadow: 0 0 2px rgba(0, 0, 0, 0.1);
  cursor: grab;
}

 通过以上步骤就成功创建了一个可以拖拽的图片展示特效。操作非常简单,而且效果十分酷炫!快来试试看吧!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/956554.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

leetcode904-水果成篮

leetcode 904 时间复杂度&#xff1a;O(n) 空间复杂度&#xff1a;O(1) 之前发布了一个滑动窗口的题目解答思路&#xff0c;参考博文&#xff1a;长度最小的子数组 本题也是基于滑动窗口的一个扩展题&#xff0c;主要解决方法是利用滑动窗口哈希表 var totalFruit function…

线性代数概述

矩阵与线性代数的关系 矩阵是线性代数的研究对象之一&#xff1a; 矩阵&#xff08;Matrix&#xff09;是一个按照长方阵列排列的复数或实数集合&#xff0c;是线性代数中的核心概念之一。矩阵的定义和性质构成了线性代数中矩阵理论的基础&#xff0c;而矩阵运算则简洁地表示和…

李宏毅机器学习HW1: COVID-19 Cases Prediction

Kaggle数据集和提交链接 特征选择&#xff08;主要修改地方&#xff09; 在sample code的基础上主要修改了Select_feat选择特征函数。 首先&#xff0c;因为数据集中的第一列是id&#xff0c;先在raw_x_train&#xff0c;raw_x_valid&#xff0c;raw_x_test中都去掉这一列。其…

owasp SQL 注入-03 (原理)

1: 先看一下注入界面: 点submit 后&#xff0c;可以看到有语法报错&#xff0c;说明已经起作用了: 报如下的错误: You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near at line 1 2:…

VD:生成a2l文件

目录 前言Simulink合并地址 ASAP2 editor 前言 我之前的方法都是通过Simulink模型生成代码的过程中顺便就把a2l文件生成出来了&#xff0c;这时的a2l文件还没有地址&#xff0c;所以紧接着会去通过elf文件更新地址&#xff0c;一直以为这是固定的流程和方法&#xff0c;今天无…

Navicat Premium 数据可视化

工作区&#xff0c;数据源以及图表 数据可视化是使用可视化组件&#xff08;例如图表&#xff0c;图形和地图&#xff09;的信息和数据的图形表示。 数据可视化工具提供了一种可访问的方式&#xff0c;用于查看和理解数据中的趋势&#xff0c;异常值和其他模式。 在Navicat中&…

设置 Git 默认推送不需要输入账号和密码【Ubuntu、SSH】

如何设置 Git 默认推送不需要输入账号和密码 在使用 Git 管理代码时&#xff0c;许多开发者会遇到每次推送&#xff08;push&#xff09;或拉取&#xff08;fetch&#xff09;代码时都需要输入 GitHub 或 GitLab 等远程仓库的账号和密码的情况。虽然设置了用户名和电子邮件信息…

TCP Window Full是怎么来的

wireshark查看包时&#xff0c;会看到TCP Window Full&#xff0c;总结下它的特点&#xff1a; 1. Sender会显示 TCP Window Full 2. “Sender已发出&#xff0c;但&#xff0c;Receiver尚未ack的字节”&#xff0c;即Sender的 bytes in flights 3. Sender的 bytes in fligh…

PyTorch框架——基于WebUI:Gradio深度学习ShuffleNetv2神经网络蔬菜图像识别分类系统

第一步&#xff1a;准备数据 蔬菜数据集&#xff0c;英文为Vegetable。 train 目录下有15000 张图片。 共十五种植物的幼苗图片集&#xff0c;分别为classes [Bean, Bitter_Gourd, Bottle_Gourd, Brinjal, Broccoli, Cabbage, Capsicum, Carrot, Cauliflower, Cucumber, Pa…

WPS数据分析000001

目录 一、表格的新建、保存、协作和分享 新建 保存 协作 二、认识WPS表格界面 三、认识WPS表格选项卡 开始选项卡 插入选项卡 页面布局选项卡 公式选项卡 数据选项卡 审阅选项卡 视图选项卡 会员专享选项卡 一、表格的新建、保存、协作和分享 新建 ctrlN------…

网络安全 | 什么是正向代理和反向代理?

关注&#xff1a;CodingTechWork 引言 在现代网络架构中&#xff0c;代理服务器扮演着重要的角色。它们在客户端和服务器之间充当中介&#xff0c;帮助管理、保护和优化数据流。根据代理的工作方向和用途&#xff0c;代理服务器可分为正向代理和反向代理。本文将深入探讨这两种…

某讯一面,感觉问Redis的难度不是很大

前不久&#xff0c;有位朋友去某讯面试&#xff0c;他说被问到了很多关于 Redis 的问题&#xff0c;比如为什么用 Redis 作为 MySQL 的缓存&#xff1f;Redis 中大量 key 集中过期怎么办&#xff1f;如何保证缓存和数据库数据的一致性&#xff1f;我将它们整理出来&#xff0c;…

基于机器学习的用户健康风险分类及预测分析

完整源码项目包获取→点击文章末尾名片&#xff01; 背景描述 在这个日益注重健康与体能的时代&#xff0c;健身已成为许多人追求健康生活的重要组成部分。 本数据集包含若干健身房会员的详细信息&#xff0c;包括年龄、性别、体重、身高、心率、锻炼类型、身体脂肪比例等多项关…

TCP TIME-WAIT 状态为什么要坚持 2MSL

经常有人问这个问题&#xff0c;这种问题问我就对了。我准备了下面的一幅时序图来解释这个问题&#xff1a; 简单点说就是两个目的&#xff1a; 正常处理被动关闭方的重传 FIN&#xff1b;确保当前连接的所有报文全部消失。 也就是说&#xff0c;无论任何情况下&#xff0c;…

Ubuntu升级Linux内核教程

本文作者CVE-柠檬i: CVE-柠檬i-CSDN博客 本文使用的方法是dpkg安装&#xff0c;目前版本为5.4.0-204&#xff0c;要升级成5.8.5版本 下载 下载网站&#xff1a;https://kernel.ubuntu.com/mainline/ 在该网站下载deb包&#xff0c;选择自己想要升级的版本&#xff0c;这里是5…

Java算法 数据结构 栈 单调栈实战 模版题 [洛谷-P5788]

目录 题目地址 题目描述 输入输出样例 代码 题目地址 【模板】单调栈 - 洛谷 题目描述 输入输出样例 代码 static void solve() throws Exception {int nsc.nextInt();int[] arrnew int[n1];int[] result new int[n1];for(int i1;i<n1;i) {arr[i]sc.nextInt();}Stack …

web前端1--基础

&#xff08;时隔数月我又来写笔记啦~&#xff09; 1、下载vscode 1、官网下载&#xff1a;Visual Studio Code - Code Editing. Redefined 2、步骤&#xff1a; 1、点击同意 一直下一步 勾一个创建桌面快捷方式 在一直下一步 2、在桌面新建文件夹 拖到vscode图标上 打开v…

Api网关Zuul

网关分类与开放API 开放API (OpenAPI) 企业需要将自身数据、能力等作为开发平台向外开放&#xff0c;通常会以REST的方式向外提供&#xff0c;最好的例子就是淘宝开放平台、腾讯公司的QQ开发平台、微信开放平台。开放API平台必然涉及到客户应用的接入、API权限的管理、调用次数…

Flink(八):DataStream API (五) Join

1. Window Join Window join 作用在两个流中有相同 key 且处于相同窗口的元素上。这些窗口可以通过 window assigner 定义&#xff0c;并且两个流中的元素都会被用于计算窗口的结果。两个流中的元素在组合之后&#xff0c;会被传递给用户定义的 JoinFunction 或 FlatJoinFunct…

数据结构漫游记:队列的动态模拟实现(C语言)

嘿&#xff0c;各位技术潮人&#xff01;好久不见甚是想念。生活就像一场奇妙冒险&#xff0c;而编程就是那把超酷的万能钥匙。此刻&#xff0c;阳光洒在键盘上&#xff0c;灵感在指尖跳跃&#xff0c;让我们抛开一切束缚&#xff0c;给平淡日子加点料&#xff0c;注入满满的pa…