AI模特换装的前端实现

news2025/1/17 13:56:45

本文作者为 360 奇舞团前端开发工程师

随着AI的火热发展,涌现了一些AI模特换装的前端工具(比如weshop网站),他们是怎么实现的呢?使用了什么技术呢?下文我们就来探索一下其实现原理。

总体的实现流程如下:我们将下图中的这个模特的图片,使用Segment Anything Model在后端分割图层,然后将分割后的图层mask信息返回给前端处理。在前端中选择需要保留的图层信息(如下图中的模特的衣服图层),然后将选中的图层信息交给后端中的Stable Diffusion处理。后端使用原始图片结合选中的图层蒙版图片结合图生图的功能,可以实现weshop等网站的模特换衣等功能。

dbaf4d8dd8694d2e1908ee85b4e3f674.jpeg

本文先简单介绍一下使用SAM智能图层分割,然后主要介绍一下在前端中怎么对分割后的图层进行选择的处理流程。

使用SAM识别图层

首先我们需要对图层进行分割,在SAM出来之前,我们需要使用PS将模特的衣服选取出来,然后倒出衣服的模板,然后再使用其他工具进行替换。但是现在有了SAM后,我们可以对图片中的事物进去只能区分,获取各种物品的图层。

Segment Anything Model(SAM)是一种尖端的图像分割模型,可以进行快速分割,为图像分析任务提供无与伦比的多功能性。SAM 的先进设计使其能够在无需先验知识的情况下适应新的图像分布和任务,这一功能称为零样本传输。SAM 使任何人都可以在不依赖标记数据的情况下为其数据创建分段掩码。

要深入了解 Segment Anything 模型和 SA-1B 数据集,请访问Segment Anything 网站(https://segment-anything.com/)并查看研究论文Segment Anything(https://arxiv.org/abs/2304.02643)。

我们使用SAM进行图像分割,将一个图片中的物体分割成不同的部分。

def mask2rle(img):
    '''
    img: numpy array, 1 - mask, 0 - background
    Returns run length as string formated
    '''
    pixels = img.T.flatten()
    pixels = np.concatenate([[0], pixels, [0]])
    runs = np.where(pixels[1:] != pixels[:-1])[0] + 1
    runs[1::2] -= runs[::2]
    return ' '.join(str(x) for x in runs)


def trans_anns(anns):
    if len(anns) == 0:
        return
    sorted_anns = sorted(anns, key=(lambda x: x['area']), reverse=False)
    list = []
    index = 0
    # 对每个注释进行处理
    for ann in sorted_anns:
        bool_array = ann['segmentation']
        # 将boolean类型的数组转换为int类型
        int_array = bool_array.astype(int)
        # 转化为RLE格式
        rle = mask2rle(int_array)
        list.append({"index": index, "mask": rle})
        index += 1
    return list

image = cv2.imread('<your image path>')

import sys
sys.path.append('<your segment-anything link path>')
from segment_anything import sam_model_registry, SamAutomaticMaskGenerator, SamPredictor


# sam 模型路径
sam_checkpoint = '<your sam model path>'
# 根据下载的模型,设置对应的类型
model_type = "vit_h"

# device = "cuda"
sam = sam_model_registry[model_type](checkpoint=sam_checkpoint)
# sam.to(device=device)
mask_generator = SamAutomaticMaskGenerator(sam)
masks = mask_generator.generate(image)
# 处理sam返回的图层信息
mask_list = trans_anns(masks)

mask_obj = {
    "height": image.shape[0],
    "width": image.shape[1],
    "mask_list": mask_list
}

import json
print(json.dumps(mask_obj))

运行以上python代码之前,需要配置sam的python环境,具体的配置描述请查看sam的官方描述。

我们通过以上代码,将我们提供的图片,通过SAM处理后,返回图层分割数据。在trans_anns方法中,将图层按照area从小到大的顺序排序。遍历各个图层,将boolean类型的数组转换为 0 1 int类型,然后对二维numpy array类型的0 1二进制mask图像转换为RLE格式。

RLE是一种简单的无损数据压缩算法,通常用于表示连续的相同值的序列。RLE编码的字符串通常用于在图像分割等任务中存储和传输二进制掩码信息,以便更有效地表示图像中的目标区域。并且方便数据压缩和传输。我们参照的这种编解码方式。也可以使用coco RLE的编解码方式。

将编码后的各图层信息存储到list中,就可以通过接口传输给前端处理了。

前端选择图层

下面这些是本文的重点,在前端将刚才解析后的mask_list信息展示,并可以通过交互选取需要保留的模版,并生成最终合并选取的mask生成一个需要保留的服装模版。

body中的基本组件为

<div id="layer-box" style=" width: 500px; height: 500px;position: relative">
        <img style="width: 100%; height: 100%; position: absolute" src="https://p0.ssl.qhimg.com/t01989f0d446bed3e58.jpg" />
    </div>
    <div id="save" @click="save" style="margin-top: 20px;margin-right: 20px; margin-left: 20px;">保存</div>
    <canvas id="mergedCanvas" style="border:1px solid #000;"></canvas>

id为layer-box的div组件作为各个mask的父组件,用于查找和管理各个mask的隐藏和展示。其子组件中的第一个标签是展示原始的模特图片的。

id为save的组件在点击时可以处理保存选中的各个mask为一个新的mask图片,用于处理图片合成。

id为mergedCanvas的canvas是进行图片合成和展示合成后的图片的。

解析SAM处理后的mask_list信息
/**
         * rle格式图片信息转换为mask信息
         */
        function rle2mask(mask_rle, shape = [500, 500]) {
            /*
            mask_rle: run-length as string formatted (start length)
            shape: [width, height] of array to return
            Returns an array, 1 - mask, 0 - background
            */

            const s = mask_rle.split(" ");
            let starts = s.filter((_, index) => index % 2 === 0).map(Number);
            const lengths = s.filter((_, index) => index % 2 !== 0).map(Number);
            starts = starts.map(start => start - 1);
            const ends = starts.map((start, index) => start + lengths[index]);
            const img = new Array(shape[0] * shape[1]).fill(0);

            for (let i = 0; i < starts.length; i++) {
                for (let j = starts[i]; j < ends[i]; j++) {
                    img[j] = 1;
                }
            }

            // return transposeArray(img, shape);
            const transposed = new Array(shape[1]).fill(0).map(() => new Array(shape[0]).fill(0));
            for (let i = 0; i < shape[0]; i++) {
                for (let j = 0; j < shape[1]; j++) {
                    transposed[j][i] = img[i * shape[1] + j];
                }
            }
            return transposed;
        }


        /**
         * 转换mask图片信息,并设置mask的填充颜色
         */
        function transformMaskImage(item, _width, _height) {
            let canvas = document.createElement("canvas");
            let canvasContext = canvas.getContext("2d");
            canvas.width = _width;
            canvas.height = _height;
            let rgbaData = rle2mask(item.mask || '', [_width, _height])
            for (let y = 0; y < rgbaData.length; y++) {
                let row = rgbaData[y];
                for (let x = 0; x < row.length; x++) {
                    let dot = rgbaData[y][x];
                    if (1 === dot && canvasContext) {
                        // 值为1的点填充颜色
                        (canvasContext.fillStyle = "#4169eb"), canvasContext.fillRect(x, y, 1, 1);
                    }
                }
            }
            // canvas当前层的图片(base64格式)
            // matrix:上边生成的二维数组
            return { imageData: canvas.toDataURL("image/png"), matrix: rgbaData };
        }

        // 使用sam处理后的图层信息(rle编码后的,由于篇幅限制,已省略)
        const res = { "height": 500, "width": 500, "mask_list": [{ "index": 0, "mask": "109864 3 110361 7 110860 9 111359 10 111859 10 112359 10 112860 9 113360 10 113860 10 114360 10 114860 10 115360 10 115861 8" }, { "index": 1, "mask": "121910 2 122409 4 122908 6 123408 7 123907 8 124407 9 124907 9 125406 11 125905 12 126404 13 126905 12 127405 12 127906 12 128406 12 128907 11 129407 10 129908 8 130408 4" },......] }

        layers = res.mask_list.map((item) =>
            transformMaskImage(item, res.width, res.height)
        );

res是sam处理后返回的图层信息(由于篇幅限制,已省略,详情请看demo(https://github.com/yuhao1128/AI-model-mask-select-demo/blob/main/index.html)中的数据)。遍历mask_list,使用canvas保存各个mask的信息。由于前面sam处理后的mask_list是经过压缩编码的,所以在rle2mask方法中对rle编码后的数据解码为 0/1二维数组的格式。rle2mask中的解码方式请参考这种解码(https://www.kaggle.com/code/pestipeti/decoding-rle-masks)方式。

然后遍历二维数组,将值为1的点填充颜色,此处是填充的rgba为"#4169eb"的颜色,可以根据需要自己修改为其他的颜色。此处填充的颜色会在下文中鼠标移动到mask上面时,在mask展示的时候呈现此颜色。

最后在layers中存储各个mask的base64格式的图片信息和二维数组信息。

将各个mask添加到图层
const box = document.querySelector("#layer-box");
        const baseStyle = "width:100%;height:100%;position: absolute;";
        //将各个mask添加为layer-box的子组件,并隐藏mask的展示
        layers.forEach((ele) => {
            const image = document.createElement("img");
            image.src = ele.imageData;
            image.style = `${baseStyle}opacity:0`;
            image.className = "layer";
            box.append(image);
        });

将各个mask添加的图片添加为layer-box组件的子组件,并且设置opacity为0,先隐藏这些mask的展示,在下文会监听鼠标的位置,通过设置mask的opacity属性来展示mask。

监听鼠标的位置和点击
// 鼠标移入mask组件的区域时,展示mask
        box.addEventListener("mousemove", (e) => {
            const { clientX, clientY } = e;
            const X = box.getBoundingClientRect().left + document.body.scrollLeft;
            const Y = box.getBoundingClientRect().top + document.body.scrollTop;
            const x = parseInt(res.width * (clientX - X) / box.getBoundingClientRect().width)
            const y = parseInt(res.height * (clientY - Y) / box.getBoundingClientRect().height)
            const allLayers = box.querySelectorAll(".layer");
            const index = layers.findIndex((item) => item.matrix?.[y]?.[x]);
            allLayers.forEach((ele, i) => {
                if (i === index) {
                    ele.style = `${baseStyle}opacity:0.7`;
                } else {
                    // 已经选中的不需要隐藏
                    if (selectedIndexList.indexOf(i) === -1) {
                        ele.style = `${baseStyle}opacity:0`;
                    }
                }
            });
        });

        // 鼠标移出mask组件的区域时,隐藏mask
        box.addEventListener("mouseout", (e) => {
            console.log('mouseout selectedIndexList', selectedIndexList);
            const allLayers = box.querySelectorAll(".layer");
            allLayers.forEach((ele, i) => {
                // 只有选中的才会展示
                if (selectedIndexList.indexOf(i) > -1) {
                    ele.style = `${baseStyle}opacity:0.7`;
                } else {
                    ele.style = `${baseStyle}opacity:0`;
                }
            });
        });

        // 用户点击时,保存用户选中的mask的index
        box.addEventListener("mousedown", (e) => {
            const { clientX, clientY } = e;
            const X = box.getBoundingClientRect().left + document.body.scrollLeft;
            const Y = box.getBoundingClientRect().top + document.body.scrollTop;
            const x = parseInt(res.width * (clientX - X) / box.getBoundingClientRect().width)
            const y = parseInt(res.height * (clientY - Y) / box.getBoundingClientRect().height)
            const index = layers.findIndex((item) => item.matrix?.[y]?.[x]);
            if (selectedIndexList.indexOf(index) === -1) {
                //保存点击选中的元素index
                selectedIndexList.push(index)
            }
        });

box就是上文的layer-box,是各个mask的父组件。layer-box监听鼠标的move事件和click事件,当move到对应的mask上时,将mask展示,移除mask时,隐藏mask。mask在list中是从小到大的顺序,所以遍历匹配mask时,会优先匹配面积小的组件,方便灵活选择。当点击mask的位置时,保存mask在list中的index到selectedIndexList中,方便后续导出保存选择,并高亮展示选中的mask。

选中的mask合成图片
// 存储各个图层图片信息
        let layers = []
        // 选择layer的index
        const selectedIndexList = []


        // 点击保存
        document.getElementById('save').onclick = function () {
            const images = [];
            selectedIndexList.forEach(index => {
                images.push(layers[index].imageData)
            })
            drawing(images)
        }

        /**
           * 图片合成
           */
        function drawing(images) {
            const canvas = document.getElementById("mergedCanvas");
            canvas.width = 500;  // 设置canvas宽
            canvas.height = 500; // 设置canvas高
            const ctx = canvas.getContext("2d");
            let loadedImages = 0;
            images.forEach(function (src) {
                const img = new Image();
                img.src = src;
                img.onload = function () {
                    loadedImages++;
                    // 绘制每张图片到 canvas 上
                    ctx.drawImage(img, 0, 0);
                    // 如果所有图片都加载完成,保存合并后的图片
                    if (loadedImages === images.length) {
                         // 获取图片的像素数据
                        const imageData = ctx.getImageData(0, 0, img.width, img.height);
                        const data = imageData.data;
                        // 转换为黑白效果
                        for (let i = 0; i < data.length; i += 4) {
                          // 将 R、G、B 设置为0
                          data[i] = 0;
                          data[i + 1] = 0;
                          data[i + 2] = 0;
                        }
                        // 将修改后的数据放回 canvas
                        ctx.putImageData(imageData, 0, 0);
                        // 导出为 base64 图片
                        const mergedImageBase64 = canvas.toDataURL("image/png");
                        // 如果需要,你可以将mergedImageBase64图片用于其他操作,比如发送到服务器
                    }
                };
            });
        }

当选择完成后,可以点击“保存”按钮,将选择的mask使用canvas生成一个合并后的图片。此处已将合成后的图片转换为黑白蒙版照片,之后可以使用这个合并后的图片进行后续的处理。

根据选中的图层,点击保存后,生成的模板如下图所示。

389768c6055569702819cda63bb1dc49.jpeg

预览效果(https://yuhao1128.github.io/AI-model-mask-select-demo/)、代码详情(https://github.com/yuhao1128/AI-model-mask-select-demo/blob/main/index.html)

使用Stable Diffusion进行后续的处理

由于篇幅的限制,并且这部分网络上以及有很多的介绍资料,就不再本文中进行介绍了,可以参考这篇文章(https://www.uisdc.com/stable-diffusion-24)的介绍尝试体验一下在本地中使用Stable Diffusion的图生图的「重绘蒙版」来进行模特的重新绘制。

也可以在后端部署Stable Diffusion服务中处理模特换装。将前面的模特原图以及生成的蒙版图片,以及其他的SD的图生图功能的参数传给后端的SD服务处理。

除了模特换装的功能,上面的流程还可以应用到物品换背景的功能中。其他的一些智能抠图,智能替换的功能都可以扩展上面的处理流程来实现。

参考链接:

https://github.com/facebookresearch/segment-anything

https://juejin.cn/post/7248903246970503223#heading-2

https://www.uisdc.com/stable-diffusion-24

- END -

关于奇舞团

奇舞团是 360 集团最大的大前端团队,代表集团参与 W3C 和 ECMA 会员(TC39)工作。奇舞团非常重视人才培养,有工程师、讲师、翻译官、业务接口人、团队 Leader 等多种发展方向供员工选择,并辅以提供相应的技术力、专业力、通用力、领导力等培训课程。奇舞团以开放和求贤的心态欢迎各种优秀人才关注和加入奇舞团。

fe6c7c7319f195cbcc2ca1909b9dfe06.jpeg

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

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

相关文章

HarmonyOS将程序下载并运行到真机上 (华为手机为例)

前面的文章 我们讲到过一些关于这个预览器的操作 可以在上面看到我们代码的一个整体效果 但其实 这边可以真实的运行在我们自己的手机上 因为你这个预览器再好 还是和实际的手机环境有所偏差 首先 我们要设置一下手机 我们在设置中 找到 关于手机 然后 这下面 有一个 Harmo…

如何使用阿里云虚拟主机和域名设置网站?

本文档将向您展示如何使用阿里云虚拟主机来设置一个新网站&#xff0c;并完成一个域名。如果您按照此处的步骤操作&#xff0c;您将启动并运行一个新网站&#xff0c;可以使用您选择的名称在全球范围内访问&#xff0c;并托管在阿里云平台上。 本文档假设您已经拥有有效的阿里…

uView ui 1x uniapp 表格table行内容长度不一导致高度不统一而出现的不对齐问题

问题 因为td单元格内空长度不定导致行单元格未对齐 解决&#xff1a; 重置td的高度&#xff1a;height:100% 改为height:auto !import <u-table><u-tr v-for"(item,index) in Lineinfo.Cust_Name" ><u-td style"height: auto !important;back…

ABAP2XLSX 的安装和demo

ABAP2XLSX 是一个git上面的很好用的工具&#xff0c;它可以帮助abaper们更方便&#xff0c;更简单的生成各种各样复杂的自定义的excel&#xff0c;以满足各企业的信息化建设 在安装这个之前&#xff0c;请先查看之前的博客&#xff0c;去安装abapgit abap2xlsx地址&#xff1…

vue3通过el-dropdown实现动态菜单切换页面

这是效果图 首先是主页index.vue <template><el-row><el-col :span"20"><!-- 顶部菜单 --><div v-if"showTop"><topmenu /></div><!-- 右侧下方区域动态切换的内容 --><div style"flex: 1;&quo…

Python GUI编程:dearpygui和tkinter的对比与选择详解

概要 随着Python在GUI(图形用户界面)编程中的不断发展&#xff0c;出现了许多优秀的库&#xff0c;如dearpygui和tkinter。 这两个库在许多方面都有所不同&#xff0c;不仅是在功能方面&#xff0c;还在设计哲学和用途上。 本文将对比这两个库&#xff0c;并使用Python代码举…

智能AI问答系统ChatGPT网站系统源码+Midjourney绘画+支持GPT-4-Turbo模型+支持GPT-4图片理解能力

一、AI创作系统 SparkAi创作系统是基于ChatGPT进行开发的Ai智能问答系统和Midjourney绘画系统&#xff0c;支持OpenAI-GPT全模型国内AI全模型。本期针对源码系统整体测试下来非常完美&#xff0c;可以说SparkAi是目前国内一款的ChatGPT对接OpenAI软件系统。那么如何搭建部署AI…

Audacity降噪消除视频中杂音

简介: CSDN博客专家,专注Android/Linux系统,分享多mic语音方案、音视频、编解码等技术,与大家一起成长! 优质专栏:Audio工程师进阶系列【原创干货持续更新中……】🚀 优质专栏:多媒体系统工程师系列【原创干货持续更新中……】🚀 人生格言: 人生从来没有捷径,只…

面试篇之微服务(二)

目录 服务容灾 21.什么是服务雪崩&#xff1f; 22.什么是服务熔断&#xff1f;什么是服务降级&#xff1f; 什么是服务熔断&#xff1f; 什么是服务降级&#xff1f; 有哪些熔断降级方案实现&#xff1f; 23.Hystrix怎么实现服务容错&#xff1f; 24.Sentinel怎么实现限…

【物联网与大数据应用】Hadoop数据处理

Hadoop是目前最成熟的大数据处理技术。Hadoop利用分而治之的思想为大数据提供了一整套解决方案&#xff0c;如分布式文件系统HDFS、分布式计算框架MapReduce、NoSQL数据库HBase、数据仓库工具Hive等。 Hadoop的两个核心解决了数据存储问题&#xff08;HDFS分布式文件系统&#…

基于YOLOv8深度学习的生活垃圾分类目标检测系统【python源码+Pyqt5界面+数据集+训练代码】目标检测

《博主简介》 小伙伴们好&#xff0c;我是阿旭。专注于人工智能、AIGC、python、计算机视觉相关分享研究。 ✌更多学习资源&#xff0c;可关注公-仲-hao:【阿旭算法与机器学习】&#xff0c;共同学习交流~ &#x1f44d;感谢小伙伴们点赞、关注&#xff01; 《------往期经典推…

SOLIDWORKS 2024新功能之SOLIDWORKS Manage篇

SOLIDWORKS 2024 新功能 SOLIDWORKS Manage篇目录概述 • 在文档预览中测量 • Plenary Web 客户端 CAD 文件预览 • 受影响项目的字段条件 • 任务自动化 • 工作任务燃尽图 • 时间表工作时间 • 材料明细表数量 • 替换 BOM 品项的流程输出 • 向 BOM 添加子条件 一…

为什么 SQL 日志文件很大,我应该如何处理?

SQL Server 日志文件是记录所有数据库事务和修改的事务日志文件。在 SQL 术语中&#xff0c;此日志文件记录对数据库执行的所有 INSERT、UPDATE 和 DELETE 查询操作。 如果数据库处于联机状态或处于恢复状态时日志已满&#xff0c;则 SQL Server 通常会发出 9002 错误。在这种…

脚本格式问题记录

服务器上的一些脚本迁移到其他服务上发生的小问题 问题&#xff1a;执行一个在win10系统编写好的shell脚本&#xff0c;放到Linux上执行报错如下&#xff1a; bash: ./xxx.sh: /bin/bash^M: bad interpreter: No such file or directory 原因&#xff1a;window系统写的脚本&a…

c++——string字符串____迭代器.范围for.修改遍历容量操作

在成为大人的路上喘口气. 目录 &#x1f393;标准库类型string &#x1f393;定义和初始化string对象 &#x1f4bb;string类对象的常见构造 &#x1f4bb;string类对象的不常见构造 &#x1f4bb;读写string对象 &#x1f393; string类对象的修改操作 &#x1f4…

大数据——一文详解数据仓库概念(数据仓库的分层概念和维度建模详解)

1、ods是什么&#xff1f; ods层最好理解&#xff0c;基本上就是数据从源表拉过来&#xff0c;进行etl&#xff0c;比如MySQL映射到Hive&#xff0c;那么到了Hive里面就是ods层。ods全称是 Operational Data Store&#xff0c;操作数据存储——“面向主题的”&#xff0c;数据…

某60物联网安全之IoT漏洞利用实操1学习记录

物联网安全 文章目录 物联网安全IoT漏洞利用实操1&#xff08;逻辑漏洞&#xff09;实验目的实验环境实验工具实验原理实验内容实验步骤 IoT漏洞利用实操1&#xff08;逻辑漏洞&#xff09; 实验目的 学会使用fat模拟IoT设备固件 学会使用IDA分析设备固件内服务程序的逻辑漏洞…

外骨骼运动控制方法的简单解读

Title: 外骨骼运动控制方法的简单解读 文章目录 I. 前言II. 关节运动控制 —— 运动轨迹/运动意图的跟踪III. 柔性交互控制 —— 提高外骨骼和人交互的 "透明性"IV. 能量成型控制 —— 借鉴双足机器人的无源步态控制V. 贝叶斯优化 ——控制参数的优化与学习VI. 小节个…

Windows11编译Hadoop3.3.6源码

由于https://github.com/kontext-tech/winutils还未发布3.3.6版本&#xff0c;因此尝试源码编译 目录 环境和安装包准备&#xff0c;见2zlib编译方法一&#xff1a;方法二&#xff1a; 配置文件更改1. maven阿里云镜像2. Node版本3. 越过Javadoc检查 编译HadoopError,其他报错…

ThermalLabel SDK for .NET 13.0.23.1113 Crack

ThermalLabel SDK for .NET 是一个 .NET 典型类库&#xff0c;它允许用户和开发人员创建非常创新的条码标签并将其发布在 zebra ZPL、EPL、EPSON ESC、POS 以及 Honeywell intermec 指纹中通过在 VB.NET 或 C# 上编写 .NET 纯代码来实现热敏打印机&#xff0c;以实现项目框架的…