webgl canvas系列——快速加背景、抠图、加水印并下载图片

文章目录

    • ⭐前言
    • ⭐canvas绘制图片
      • 💖绘制csdn图片
      • 💖给png图片加背景
      • 💖cavans下载图片
      • 💖cavans上传图片并抠图
      • 💖cavans添加文字水印
      • 💖inscode 完整代码块
    • ⭐结束

yma16-logo

⭐前言

大家好,我是yma16,本文分享webgl canvas系列——快速抠图、加水印。
该系列往期文章
web canvas系列——快速入门上手绘制二维空间点、线、面

⭐canvas绘制图片

方法作用
drawImage(image, x, y)image 是 image 或者 canvas 对象,x 和 y 是其在目标 canvas 里的起始坐标。
drawImage(image, x, y, width, height)width 和 height,这两个参数用来控制 当向 canvas 画入时应该缩放的大小

💖绘制csdn图片

绘制csdn图片
csdn-img
js使用drawImage绘制图片

function drawImg() {
    const canvas = document.getElementById("tutorial");
    if (canvas.getContext) {
        const ctx = canvas.getContext("2d");
        const img = new Image();
        img.onload = function () {
            ctx.drawImage(img, 0, 0);
        };
        img.src = "/src/assets/csdn.png";

    }
    else {
        console.log('不支持canvas')
    }
}



window.onload = () => {
    drawImg()
}

csdn-draw

💖给png图片加背景

js先填充cavans背景在画图

function drawImg() {
    const canvas = document.getElementById("tutorial");
    if (canvas.getContext) {
        const ctx = canvas.getContext("2d");
        // background
        ctx.fillStyle = 'rgba(0, 0, 200, 0.5)';
        ctx.globalAlpha = 1
        ctx.fillRect(0, 0, ctx.canvas.width, ctx.canvas.height)
        const img = new Image();
        img.onload = function () {
            ctx.drawImage(img, 35, 0);
        };
        img.src = "/src/assets/csdn.png";

    }
    else {
        console.log('不支持canvas')
    }
}



window.onload = () => {
    drawImg()
}

效果
png+background

💖cavans下载图片

方法作用
canvas.toDataURL(type, encoderOptions);HTMLCanvasElement.toDataURL() 方法返回一个包含图片展示的 data URI。可以使用 type 参数指定其类型,默认为 PNG 格式。图片的分辨率为 96dpi
  • 如果画布的高度或宽度是 0,那么会返回字符串“data:,”。
  • 如果传入的类型非“image/png”,但是返回的值以“data:image/png”开头,那么该传入的类型是不支持的。 Chrome
  • 支持“image/webp”类型。

使用toDataURL属性下载

function downloadImage(dataURL, filename) {
    // 创建一个a元素,用于触发下载
    var link = document.createElement('a');
    link.download = filename;
    link.href = dataURL;
    link.click();
}


window.onload = () => {
    document.getElementById('down-btn').addEventListener('click', function () {
        const imgData = document.getElementById('tutorial').toDataURL("image/png");
        downloadImage(imgData, 'canvas图片')
    })
}

下载成功的效果效果
down-btn

💖cavans上传图片并抠图

前期的准备
html原生上传图片 并使用canvas绘制图片
html结构

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <link href="style.css" rel="stylesheet" type="text/css" />
    <title>canvas_调试</title>
</head>

<body>
    <div class="title">
        canvas
    </div>
    <div>
        <label for="avatar">选择图片上传:</label>
        <br>
        <input type="file" id="avatar" name="avatar" accept="image/png, image/jpeg" />
        <div id="preview-img">

        </div>
    </div>
    <div class="container">

        <canvas id="tutorial" width="250" height="100">

            current stock price: $3.15 +0.15
        </canvas>
    </div>
    <div>
        <button id="clear-back">
            自动抠图
        </button>
    </div>
    <br>
    <div>
        <button id="down-btn">
            下载图片
        </button>
    </div>
    <script src="script.js"></script>
</body>

</html>

js逻辑

const config = {
    cavansDom: null,
    uploadImgData: null,
}

function InitCanvas() {
    const canvas = document.getElementById("tutorial");
    config.cavansDom = canvas
    if (canvas.getContext) {
        const ctx = canvas.getContext("2d");
        // background
        ctx.fillStyle = 'rgba(0, 0, 200, 0.5)';
        ctx.globalAlpha = 1
        ctx.fillRect(0, 0, ctx.canvas.width, ctx.canvas.height)
        const img = new Image();
        img.onload = function () {
            ctx.drawImage(img, 35, 0);
        };
        img.src = "/src/assets/csdn.png";

    }
    else {
        console.log('不支持canvas')
    }
}

function drawImg(){
    canvas=config.cavansDom 
    if (canvas.getContext) {
        const ctx = canvas.getContext("2d");
        const img = new Image();
        img.onload = function () {
            const width=img.width
            const height=img.height
            canvas.width=width
            canvas.height=height
            ctx.drawImage(img, 0, 0);
        };
        img.src = config.uploadImgData
        console.log('img.src',img.src)

    }
    else {
        console.log('不支持canvas')
    }
}


function downloadImage(dataURL, filename) {
    // 创建一个a元素,用于触发下载
    var link = document.createElement('a');
    link.download = filename;
    link.href = dataURL;
    link.click();
}

function initUploadImg() {
    const input = document.getElementById('avatar')
    input.addEventListener('change', function (e) {
        console.log('e')
        const curFiles = input.files;
        console.log('curFiles', curFiles)
        config.uploadImgData = URL.createObjectURL(curFiles[0]);
        drawImg()
    })
}

function initDown() {
    document.getElementById('down-btn').addEventListener('click', function () {
        const imgData = config.cavansDom.toDataURL("image/png");
        downloadImage(imgData, 'canvas图片')
    })
}
// 抠图
function clearColor(){
    
}



window.onload = () => {
    InitCanvas()
    initDown()
    initUploadImg()
}

效果:实现了图片的上传并使用canvas适配绘制图片
上传csdn的个人头像upload-canvas
抠图逻辑:
为了实现抠图效果,可以使用绘图上下文的globalCompositeOperation属性来控制绘制的方式。其中,source-in和source-out是实现抠图的两种常用方式。

  • 使用source-in时,绘图上下文会根据Canvas上已有的内容和新绘制的图像的像素进行合并,只保留二者重叠的部分。通过调整绘图上下文的globalAlpha属性,可以实现抠图的透明效果。

  • 使用source-out时,绘图上下文会根据Canvas上已有的内容和新绘制的图像的像素进行合并,只保留二者不重叠的部分。同样,可以通过调整绘图上下文的globalAlpha属性,实现抠图的透明效果。

source-in效果

// 抠图
function clearColor(){
    document.getElementById('clear-back').addEventListener('click',function(){
        const ctx = config.cavansDom.getContext("2d");
        ctx.globalCompositeOperation = "source-in";
        ctx.fillStyle = "rgba(0, 0, 0, 0.5)"; // 设置抠图的透明度
        ctx.fillRect(0, 0, canvas.width, canvas.height);
    })
}

效果对比
source-in
source-out效果

// 抠图
function clearColor(){
    document.getElementById('clear-back').addEventListener('click',function(){
        const ctx = config.cavansDom.getContext("2d");
        ctx.globalCompositeOperation = "source-out";
        ctx.fillStyle = "rgba(0, 0, 0, 0.5)"; // 设置抠图的透明度
        ctx.fillRect(0, 0, canvas.width, canvas.height);
    })
}

效果对比
source-out

💖cavans添加文字水印

要在JavaScript的canvas上添加文字水印,可以使用以下代码:

const canvas = document.getElementById("myCanvas");
const ctx = canvas.getContext("2d");
ctx.font = "30px Arial";
ctx.fillStyle = "rgba(0, 0, 0, 0.5)";
ctx.fillText("Watermark", 10, 50);

首先,通过getElementById函数获取到canvas元素,并通过getContext函数获取到2D绘图上下文对象ctx。

然后,设置字体样式和颜色,可以使用font属性和fillStyle属性,分别设置字体大小和颜色。

最后,使用fillText方法绘制文字水印,传入要显示的文字、x坐标和y坐标。
添加水印成功
watermask

💖inscode 完整代码块

⭐结束

本文分享到这结束,如有错误或者不足之处欢迎指出!
earth

👍 点赞,是我创作的动力!
⭐️ 收藏,是我努力的方向!
✏️ 评论,是我进步的财富!
💖 最后,感谢你的阅读!

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

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

相关文章

mysql性能调优

mysql性能调优 sysbench压测调优到百万级别qps sysbench压测调优到百万级别qps 这篇文章https://www.percona.com/blog/millions-queries-per-second-postgresql-and-mysql-peaceful-battle-at-modern-demanding-workloads/#:~:textWe%20contacted%20SysBench%20author%20Alex…

抖音,剪映,TikTok,竖屏短视频转场pr模板视频素材

120个叠加效果视频转场过渡素材&#xff0c;抖音,剪映,TikTok,短视频转场pr模板项目工程文件。 效果&#xff1a;VHS、光效、胶片、霓虹灯闪光、X射线、信号、老电影等。 适用软件&#xff1a;Adobe Premiere Pro 2018 12.0或更高版本。 视频素材与大多数应用程序兼容&#xff…

ES高可用

分布式搜索引擎ES 分布式搜索引擎ES1.数据聚合1.1.聚合的种类1.2.DSL实现聚合1.3.RestAPI实现聚合 2.自动补全2.1.拼音分词器2.2.自定义分词器2.3.自动补全查询2.4.实现酒店搜索框自动补全 3.数据同步思路分析 4.集群4.1 ES集群相关概念4.2.集群脑裂问题4.3.集群分布式存储4.4.…

Diff算法详解

简要了解 Diff 算法目的就是找出新旧虚拟dom差异&#xff0c;最小化更新视图&#xff1b;即本质就是比较两个JS对象的差异&#xff1b;并不是页面上所有的更新都需要Diff算法。 在了解Diff算法之前&#xff0c;我们首先需要了解一下什么是虚拟DOM。 虚拟DOM 虚拟DOM是表示真实…

iSAM2 部分状态更新算法 (I - 原理解读)

Title: iSAM2 部分状态更新算法 (I-原理解读) 文章目录 I. 前言II. 部分状态的更新 (Partial State Update)III. 因子图的线性化 (Linearization of Factor Grahps)1. 简单实例的设定2. 一个线性化计算3. 其他线性化计算4. 状态更新量说明 IV. 部分 QR 分解实现变量消元 (Elimi…

基于傅里叶描述子的手势动作识别,Matlab实现

博主简介&#xff1a; 专注、专一于Matlab图像处理学习、交流&#xff0c;matlab图像代码代做/项目合作可以联系&#xff08;QQ:3249726188&#xff09; 个人主页&#xff1a;Matlab_ImagePro-CSDN博客 原则&#xff1a;代码均由本人编写完成&#xff0c;非中介&#xff0c;提供…

什么是智慧公厕?智慧公厕打造公共厕所信息化应用基座

公共厕所一直以来都是城市管理的一项重要工作&#xff0c;而随着科技的发展&#xff0c;智慧公厕成为了城市管理的新方向。智慧公厕应用基座是利用物联网、互联网、大数据、云计算和自动化控制等技术&#xff0c;将公共厕所进行全方位的信息化、数字化和智慧化升级&#xff0c;…

训练YOLOv9-S

1. YOLOv9-S网络结构 1.1 改前改后的网络结构&#xff08;参数量、计算量&#xff09;对比 修改前调用的yolo.py测试的yolov9.yaml的打印网络情况&#xff0c;包含参数量、计算量 修改后调用的yolo.py测试的yolov9.yaml的打印网络情况&#xff0c;包含参数量、计算量 1.2 …

JAVA入门第一步

学习总结&#xff1a; 打开CMD常见的CMD命令 一、打开CMD CMD的概念 CMD是Windows操作系统中的命令提示符(Command Prompt)程序&#xff0c;它是一种命令行工具&#xff0c;可以让用户通过键入命令来与计算机进行交互。CMD是Windows中一个基本的系统组件&#xff0c;它提供了一…

Python学习:元组

Python 元组概念 Python 中的元组&#xff08;tuple&#xff09;是不可变的有序集合。它是一种数据类型&#xff0c;类似于列表&#xff08;list&#xff09;&#xff0c;但在创建后不能被修改。元组使用圆括号 () 来表示&#xff0c;其中的元素可以是任意类型&#xff0c;并且…

【C++ STL】string类最全解析(什么是string?string类的常用接口有哪些?)

目录 一、前言 二、什么是 string ? &#x1f4a6; string 类的基本概念 &#x1f4a6; string 类与 char * 的区别 &#x1f4a6; string 类的作用 &#x1f4a6; 总结 三、string 的常用接口详解 &#x1f4a6;string 类对象的默认成员函数 ① 构造函数(初始化) ② 赋值…

详解python中函数的参数传递

在这个用例中&#xff0c;我们要讨论的是关于函数的传参问题 我所使用的python版本为3.3.2 对于函数: def fun(arg):print(arg)def main():fun(hello,Hongten)if __name__ __main__:main() 当我们传递一个参数给fun()函数&#xff0c;即可打印出传递的参数值信息。 这里打印…

扫码签到效果如何制作?二维码签到表的制作技巧

一般参加活动或者会议时&#xff0c;都会需要在入口处签到登记之后才可进入&#xff0c;这种方式需要耗费大量的时间&#xff0c;而且带给参与者的体验也不好。面对这个问题&#xff0c;现在会通过签到二维码的方式来解决&#xff0c;只需要扫描二维码就可以在手机上登记信息&a…

c语言--字符转换函数(tolower、toupper.)

目录 一、前言二、使用举例 一、前言 C语⾔提供了2个字符转换函数&#xff1a; int tolower ( int c ); //将参数传进去的⼤写字⺟转⼩写 int toupper ( int c ); //将参数传进去的⼩写字⺟转⼤写二、使用举例 #include <ctype.h> #include<stdio.h> int main(…

go|sync系列:WaitGroup、Once、Cond

文章目录 sync.WaitGroup使用方式底层原理AddDoneWait总结 sync.Once存在的意义使用方式第一个例子&#xff0c;开启十个协程利用once运行同一个函数第二个例子&#xff0c;懒汉单例获取配置文件 底层原理存在的问题改进sync.Once解决问题 sync.Cond使用方式底层原理 参考文章 …

广西开放大学的电大搜题:为学子提供便捷高效的学习辅助

尊敬的读者朋友们&#xff0c;您了解过广西开放大学的电大搜题吗&#xff1f;作为一名现代学者&#xff0c;我有幸为您揭示这个令广大学子受益匪浅的学习利器。电大搜题是广西开放大学为学子们提供的一项便捷高效的学习辅助服务&#xff0c;旨在帮助学子们更好地应对学习难题&a…

一种震荡抑制电路

Hi uu们,好久没讲有意思的电路架构了,主要是做的要是有点价值都去申请专利了,刚好这个电路专利已经公开实质审查了,拉出来和大家分享下这是怎么一个玩法.图1展示了完整的电路图. 图1:积分器电路配合震荡抑制电路 其中框选部分为典型的积分器电路,右边这几个三极管构成了震荡抑…

GESP图形化编程二级认证真题 2024年3月

GESP 图形化二级试卷 &#xff08;满分&#xff1a;100 分 考试时间&#xff1a;120 分钟&#xff09; 一、单选题&#xff08;共 10 题&#xff0c;每题 3 分&#xff0c;共 30 分&#xff09; 1、小杨的父母最近刚刚给他买了一块华为手表&#xff0c;他说手表上跑的是鸿…

有趣的大模型之我见 | Mistral 7B 和 Mixtral 8x7B

开发者告诉我们&#xff0c;有一些因素阻碍了他们更好更广泛地使用基础模型。比如&#xff0c;在可预见的将来&#xff0c;随着技术的新进步&#xff0c;不断有新的模型加入&#xff0c;同时模型的升级和迭代也在不断加速。那么&#xff0c;对于特定的用例&#xff0c;如何选择…

24计算机考研调剂 | (研究所)北京微电子技术研究所

北京微电子技术研究所2024年考研调剂信息 调剂信息 一、招生专业 二、调剂对象 统考科目为思想政治理论、英语&#xff08;一&#xff09;、数学&#xff08;一&#xff09;&#xff1b;本科为电子科学与技术、微电子学、集成电路设计、电子信息工程、通信工程、计算机科学与…