如何用html css js 画出曲线 或者斜线;

news2024/10/6 4:07:49

效果图

在这里插入图片描述

解题思路

将图片全部定位至中心点,然后x轴就变动translateX ,y轴同理;

这里有两个问题

浏览器: 以左上角为原点0,0 越往下y越大
数学坐标系:以中心点为原点0,0 越往下y越小;
曲线函数:坐标确定了原点确定了,就需要对应的曲线函数来描述这条线段

我们一个个来解决

一、如何把数学坐标系运用到浏览器之中,转化y轴的上下大小反序;

translateX 变化多少就是多少,translateY 变化多少就是反向移动多少,所以重要的是这个反向我们如何来描述,其实这里也很简单,我们使用css变量,然后在变化的时候,setProperty对应的变量值,然后在css里我们做对应的描述;

<style>
 #container {
 	width: 100vw;
    height: 100vh;
    background-color: aqua;
    position: relative;
    overflow: auto;
  }
  .curveImg {
    --size: 30px;
    --dx: 1;
    --dy: 1;
    border-radius: calc(var(--size) / 2);
    width: var(--size);
    height: var(--size);
    position: absolute;
    left: 50%;
    top: 50%;
    margin-left: calc(var(--size) / 2);
    margin-top: calc(var(--size) / 2);
    object-fit: cover; /* 保持图片纵横比 */
    transition: all 0.8s ease-out ;
    transform: translate(calc(var(--dx) * 1px), calc(var(--dy) * -1px));
  }
 </style>
  <body>
    <div id="container"></div>
  </body>

此时我们已经完成了准备工作,接着完成反向排序的准备工作,dy越大则在html中排列就越小,超越原点即为负数;

二、将原点挪至中心点

原点在与你想让它在那个位置,它都可以排列到那个位置,这里我想让它在容器内渲染,所以我想设置在容器内的中心位置,那么这个中心位置,其实就是我给定的最大值与最小值和的一半就是我的中心点;
那么我只要让其对应的横坐标点,减去当前的原点,就是它当前需要移动的距离,也就是dx,y同理;

曲线函数

不管是曲线,还是抛物线,还是斜线,其实都是有一个方程式,或者叫做函数来表示;

那么如果是在数学中;
y=x; 那就表示一段45度的斜线;
y=-x ; 那就表示一段-45度的斜线;
y=sin(x); 那就表示连绵的曲线;
数学坐标系中是无限延伸的,但在我们的浏览器中,他是有一个范围呢;所以我们还需要给他一个范围

如果是在代码中,如果我们得到了曲线的表示函数,接下来就是将其用html、css、js来绘制到我们的电脑屏幕中;

我们先来写一个类来表示这个计算类
class Curve {
/**    
  * 创建一个计算类的实例    *    
  * @constructor    
  * @param {Function} curveFunc 要添加的任务实例 
  * @param {Array} xRange x取值范围
  * @param {Array} yRange y取值范围
  */ 
  constructor(curveFunc, xRange, yRange) {
    this.xRange = xRange
    this.yRange = yRange
    this.curveFunc = curveFunc
  }
  /**
   * 计算曲线函数
   * @param {Number} x x值
   * @returns {Number} y y值
  */
  getY(x) {
    let y = this.curveFunc(x)
    if (y < this.yRange[0]) {
      y = this.yRange[0]
    }
    if (y > this.yRange[1]) {
      y = this.yRange[1]
    }
    return y
  }
}

上面我们已经创建好一个计算类、具体释义有jsdoc注释;

那么我们如何来表示曲线或者直线呢

const line = new Curve(x=>x,[-1,1],[-1,1])
const wave = new Curve(
  x => Math.sin(x),
  [-1 * Math.PI, 3 * Math.PI],
  [-1, 1]
)

这样就表示出来了;
曲线:y = wave.getY(x);
直线:y = line.getY(x);
那么关于坐标关系我们已经用代码描述出来绘制就是最简单的事了;

  <div id="container"></div>

假设我们现在有这么一个div;
然后我们先在div里创建一些图片(当然这里也可以换成你喜欢的样式);

const box = document.querySelector("#container")
const initImg = className => {
  const frag = document.createDocumentFragment()
  for (let i = 0; i < 100; i++) {
    const img = document.createElement("img")
    img.classList.add(className)
    img.src = "./test.jpg" // 设置图片路径
    frag.appendChild(img)
  }
  box.appendChild(frag)
}
initImg("curveImg")

接下来就是渲染

const layout = (curve, doms, width, height) => {
  const [minX, maxX] = curve.xRange
  const [minY, maxY] = curve.yRange
  // 步长 
  const xStep = (maxX - minX) / (doms.length - 1)
  // 与实际坐标轴的比例
  const xScale = width / (maxX - minX)
  const yScale = height / (maxY - minY)
  const cx = (maxX + minX) / 2
  const cy = (maxY + minY) / 2
  for (let i = 0; i < doms.length; i++) {
    const dom = doms[i]
    const x = minX + i * xStep
    const y = curve.getY(x)
    const dx = (x - cx) * xScale
    const dy = (y - cy) * yScale
    dom.style.setProperty("--dx", dx)
    dom.style.setProperty("--dy", dy)
  }
}

接下来只要我们执行layout函数,那么图片就会进行对应的渲染;

上全部代码

<!DOCTYPE html>
<html lang="en">
 <head>
   <meta charset="UTF-8" />
   <meta name="viewport" content="width=device-width, initial-scale=1.0" />
   <title>Document</title>
   <style>
     #container {
       width: 100vw;
       height: 100vh;
       background-color: aqua;
       position: relative;
       overflow: auto;
     }
     .curveImg {
       --size: 30px;
       --dx: 1;
       --dy: 1;
       border-radius: calc(var(--size) / 2);
       width: var(--size);
       height: var(--size);
       position: absolute;
       left: 50%;
       top: 50%;
       margin-left: calc(var(--size) / 2);
       margin-top: calc(var(--size) / 2);
       object-fit: cover; /* 保持图片纵横比 */
       transition: all 0.8s ease-out ;
       transform: translate(calc(var(--dx) * 1px), calc(var(--dy) * -1px));
     }
    
   </style>
 </head>
 <body>
   <div>
     <button onclick="renderLine()">直线</button>
     <button onclick="renderWave()">曲线</button>
     <button onclick="renderLineX()">x线</button>
     <button onclick="renderWaveX()">交叉曲线线</button>
   </div>
   <div id="container"></div>
   <script>
     class Curve {
       /**
        * 创建一个计算类的实例    *
        * @constructor
        * @param {Function} curveFunc 要添加的任务实例
        * @param {Array} xRange x取值范围
        * @param {Array} yRange y取值范围
        */
       constructor(curveFunc, xRange, yRange) {
         this.xRange = xRange
         this.yRange = yRange
         this.curveFunc = curveFunc
       }
       /**
        * 计算曲线函数
        * @param {Number} x x值
        * @returns {Number} y y值
        */
       getY(x) {
         let y = this.curveFunc(x)
         if (y < this.yRange[0]) {
           y = this.yRange[0]
         }
         if (y > this.yRange[1]) {
           y = this.yRange[1]
         }
         return y
       }
     }

     const layout = (curve, doms, width, height) => {

       const [minX, maxX] = curve.xRange
       const [minY, maxY] = curve.yRange
       const xStep = (maxX - minX) / (doms.length - 1)
       const xScale = width / (maxX - minX)
       const yScale = height / (maxY - minY)
       const cx = (maxX + minX) / 2
       const cy = (maxY + minY) / 2
       for (let i = 0; i < doms.length; i++) {
         const dom = doms[i]
         const x = minX + i * xStep
         const y = curve.getY(x)
         const dx = (x - cx) * xScale
         const dy = (y - cy) * yScale
         dom.style.setProperty("--dx", dx)
         dom.style.setProperty("--dy", dy)
       }
     }
     const box = document.querySelector("#container")
     const initImg = className => {
       const frag = document.createDocumentFragment()
       for (let i = 0; i < 100; i++) {
         const img = document.createElement("img")
         img.classList.add(className)
         img.src = "./test.jpg" // 设置图片路径
         frag.appendChild(img)
       }
       box.appendChild(frag)
     }
     initImg("curveImg")
     const images = document.querySelectorAll(".curveImg")
     const wave = new Curve(
       x => Math.sin(x),
       [-1 * Math.PI, 3 * Math.PI],
       [-1, 1]
     )
     const wave2 = new Curve(
       x => Math.sin(x),
       [0 * Math.PI, 4 * Math.PI],
       [-1, 1]
     )
     const line = new Curve(x => x, [-1, 1], [-1, 1])
     const line2 = new Curve(x => -x, [-1, 1], [-1, 1])
     const middleIndex = Math.floor(images.length / 2)
     const dom1 = Array.from(images).slice(0,middleIndex);
     const dom2 = Array.from(images).slice(middleIndex);
     function renderLine() {
       layout(line, images, box.clientWidth - 200, box.clientHeight - 200)
     }
     function renderWave() {
       layout(wave, images, box.clientWidth - 200, box.clientHeight - 200)
     }
     function renderLineX() {
       layout(line, dom1, box.clientWidth - 200, box.clientHeight - 200)
       layout(line2, dom2, box.clientWidth - 200, box.clientHeight - 200)
     }
     function renderWaveX() {
       layout(wave, dom1, box.clientWidth - 200, box.clientHeight - 200)
       layout(wave2, dom2, box.clientWidth - 200, box.clientHeight - 200)
     }
     document.addEventListener("DOMContentLoaded", () => {
       renderLine()
     })
   </script>
 </body>
</html>

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

1334. 阈值距离内邻居最少的城市/Floyd 【leetcode】

1334. 阈值距离内邻居最少的城市 有 n 个城市&#xff0c;按从 0 到 n-1 编号。给你一个边数组 edges&#xff0c;其中 edges[i] [fromi, toi, weighti] 代表 fromi 和 toi 两个城市之间的双向加权边&#xff0c;距离阈值是一个整数 distanceThreshold。 返回能通过某些路径…

组件插槽,生命周期,轮播图组件的封装,自定义指令的封装等详解以及axios的卖座案例

3.组件插槽 3-1组件插槽 注意 插槽内容可以访问到父组件的数据作用域,因为插槽内容本身就是在父组件模版中定义的 插槽内容无法访问子组件的数据.vue模版中的表达式只能访问其定义时所处的作用域,这和JavaScript的词法作用域是一致的,换言之: 父组件模版的表达式只能访问父组…

代码示例:基于JAX-WS和JAXB,其中http请求和响应的报文体都是xml数据

说明 基于JAX-WS编写了RESTful的web服务端点。 http请求和响应的报文体都是xml数据&#xff0c;服务端分别对应了用JAXB注解的请求和响应类。 只实现了服务端的代码示例 客户端使用了Postman 示例 要实现的目标&#xff1a;http请求和响应报文体的xml数据 http请求报文体的…

Idea安装完成配置

目录&#xff1a; 环境配置Java配置Maven配置Git配置 基础设置编码级设置File Header自动生成序列化编号配置 插件安装MyBtisPlusRestfulTooklkit-fix 环境配置 Java配置 Idea右上方&#xff0c;找到Project Settings. 有些版本直接有&#xff0c;有些是在设置下的二级菜单下…

【Linux系统编程十九】:(进程通信)--匿名管道/模拟实现进程池

【Linux系统编程十九】&#xff1a;匿名管道原理/模拟实现进程池 一.进程通信理解二.通信实现原理三.系统接口四.五大特性与四种情况五.应用场景--进程池 一.进程通信理解 什么是通信&#xff1f; 通信其实就是一个进程想把数据给另一个进程&#xff0c;但因为进程具有独立性…

小程序授权获取头像

wxml <view class"header"><text>头像</text><button class"butt" plain"true" open-type"chooseAvatar" bind:chooseavatar"chooseAvatar"><image src"{{HeadUrl}}" mode"&quo…

Java - 位运算的基本原理和用途

Java - 位运算的基本原理和用途 前言一. Java 位运算基本操作1.1 按位与 &1.2 按位或 |1.3 按位异或 ^1.4 按位取反 ~1.5 位移运算1.5.1 左移运算符 <<1.5.2 右移运算符 >>1.5.3 无符号右移运算符 >>> 二. 位运算实际运用2.1 判断奇偶性&#xff08;&…

DB9串口引脚介绍

一、公头和母头 图片示意源于网络: 二、 每个引脚的功能定义 公头&#xff1a;所有排针式的接头&#xff08;5针朝上&#xff0c;从左到右序号依次是1~9&#xff09; 母头&#xff1a;所有插槽式的接孔&#xff08;5孔朝上&#xff0c;从右到左序号依次是1~9&#xff09; 针…

计算机毕业设计选题推荐-掌心办公微信小程序/安卓APP-项目实战

✨作者主页&#xff1a;IT毕设梦工厂✨ 个人简介&#xff1a;曾从事计算机专业培训教学&#xff0c;擅长Java、Python、微信小程序、Golang、安卓Android等项目实战。接项目定制开发、代码讲解、答辩教学、文档编写、降重等。 ☑文末获取源码☑ 精彩专栏推荐⬇⬇⬇ Java项目 Py…

Web与DNS综合实验

目录 题目&#xff1a; 步骤一&#xff1a;准备工作 步骤二&#xff1a;在server端搭建简单网页 步骤三&#xff1a;node1端的DNS解析配置 步骤五&#xff1a;node2端进行测试。 题目&#xff1a; 1.打开3个主机&#xff0c;server、node1、node2 2.server为web主机建立网…

typora使用PicGo自动上传图片到chevereto图床

typora使用PicGo自动上传图片到chevereto图床 近期发现&#xff0c;gitee图床不能用了。github又涉及科学上网。搜索了开源图床方案&#xff0c;找到了chevereto&#xff0c;使用起来还不错。分享给大家。 文章目录 typora使用PicGo自动上传图片到chevereto图床chevereto图床安…

轻量级 Java 日志组件

日志记录功能在开发中很常用&#xff0c;不仅可以记录程序运行的细节&#xff0c;方便调试&#xff0c;也可以记录用户的行为&#xff0c;是框架中不可或缺的组件。为最大程度复用现有的组件&#xff0c;我们就地取材使用了 JDK 自带的 JUL&#xff08;java.util.logging&#…

Linux--初识和几个简单的指令(1)

目录 前言 0.什么是操作系统 0.1 搭建 Linux 环境 0.2搭建 Linux 环境小结 1.使用 XShell 远程登录 Linux 1.1关于 Linux 桌面 1.2下载安装 XShell 1.3查看 Linux 主机 ip 1.4XShell 下的复制粘贴 2.Linux下基本指令 2.1 pwd命令 2.2 ls命令 2.3 mkdir指令 2.4 cd…

Linux上编译和安装SOFA23.06

前言 你可以直接使用编译安装好的SOFA版本Installing from all-included binaries (v23.06.00)&#xff1a; 如果你想自己编译&#xff0c;可以看我下面写的内容&#xff0c;不过绝大多数是从官网来的&#xff0c;如果和官网有出入&#xff0c;建议还是以官网为准。 在Linux下…

josef约瑟 时间继电器 ST3PA-A AC220V 带插座PF085A

ST3P系列时间继电器适用于交流50Hz或60Hz,额定电压380V及以下或直流24V的控制电路中作廷时元件,按预定的时间接通或分断电路。具有体积小,精度高,延时范围宽,可与JSZ3系列继电器等同互换使用。 系列型号 ST3PF-2Z(JSZ3F-2Z) 5s AC110V ST3PF(JSZ3F) 10s AC48V ST3PC-1(AH3-3)…

IJ中配置TortoiseSVN插件:

文章目录 一、报错情况&#xff1a;二、配置TortoiseSVN插件&#xff1a; 一、报错情况&#xff1a; 由于公司电脑加密&#xff0c;TortoiseSVN菜单没有提交和更新按钮&#xff0c;所以需要使用IJ的SVN进行代码相关操作 二、配置TortoiseSVN插件&#xff1a; 需要设置一个svn.…

Go 之 captcha 生成图像验证码

目前 chptcha 好像只可以生成纯数字的图像验证码&#xff0c;不过对于普通简单应用来说也足够了。captcha默认将store封装到内部&#xff0c;未提供对外操作的接口&#xff0c;因此使用自己显式生成的store&#xff0c;可以通过store自定义要生成的验证码。 package mainimpor…

LINUX入门篇【6】----第一个LINUX小程序---进度条及相关知识讲解

前言&#xff1a; 本篇我们将开始尝试构建我们的第一个LINUX的小程序----进度条作为一个十分常见的程序&#xff0c;在我们之后的工程实践中也是需要多次运用&#xff0c;但是介于我们目前还没有去学习网络等方面的知识&#xff0c;没法独立的去利用程序去下载一个真正的程序&…

【Proteus仿真】【Arduino单片机】LM35温度计

文章目录 一、功能简介二、软件设计三、实验现象联系作者 一、功能简介 本项目使用Proteus8仿真Arduino单片机控制器&#xff0c;使用PCF8574、LCD1602液晶、LM35传感器等。 主要功能&#xff1a; 系统运行后&#xff0c;LCD1602显示传感器检测温度。 二、软件设计 /* 作者&a…