百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 文章教程 > 正文

使用Node.js和PNG.js处理PNG图片,提取精灵图并生成CSS

yund56 2025-02-25 00:35 19 浏览

在前端开发中,精灵图(Sprite)是一种常见的优化技术,通过将多个小图标合并为一张大图片,减少HTTP请求,从而提升页面加载速度。然而,手动处理精灵图往往耗时且容易出错。今天,我们将通过Node.js和pngjs库,实现自动提取精灵图并生成对应的CSS代码,大大提高开发效率。

一、什么是精灵图?

精灵图是一种将多个小图片合并为一张大图片的技术。通过CSS的background-imagebackground-position属性,可以在网页中显示大图片的某一部分,从而实现多个小图标的显示。精灵图的优点是可以减少HTTP请求次数,优化网页性能。

二、技术栈介绍

  1. Node.js:一个基于Chrome V8引擎的JavaScript运行环境,适用于服务器端开发。
  2. PNG.js:一个用于处理PNG图片的Node.js库,支持读取、解析和修改PNG图片。

三、代码实现

以下是完整的代码实现,它将读取一个PNG图片文件,提取其中的精灵图,并生成对应的CSS文件和分割后的图片。

1. 安装依赖

在开始之前,确保你已经安装了pngjs库。可以通过以下命令安装:

npm install pngjs

2. 完整代码

var fs = require('fs');
var PNG = require('pngjs').PNG;

// 获取命令行参数中的PNG文件路径
var pngFile = process.argv[2];
var outDir = pngFile.replace(".png", "");

// 读取PNG文件并解析
fs.createReadStream(pngFile)
    .pipe(new PNG({ filterType: 4 }))
    .on('parsed', function () {
        // 复制图片数据
        var tempData = new Buffer(4 * this.width * this.height);
        this.data.copy(tempData);

        // 创建输出目录
        if (!fs.existsSync('export')) {
            fs.mkdirSync('export');
        }
        if (!fs.existsSync('export/' + outDir)) {
            fs.mkdirSync('export/' + outDir);
        }

        // 获取精灵图数组
        var spritesArray = getSprites(tempData, this.height, this.width);

        // 生成CSS代码
        var css = getCss(spritesArray, pngFile);
        fs.writeFile(outDir + '.css', css, function (err) {
            if (err) throw err;
            console.log('CSS文件已保存!');
        });

        // 保存每个精灵图为单独的PNG文件
        for (var i = 0; i < spritesArray.length; i++) {
            var rect = spritesArray[i];
            var newData = {
                data: null,
                height: (rect.rb.y - rect.rt.y),
                width: (rect.rt.x - rect.lt.x)
            };
            var newPng = new PNG({
                filterType: 4,
                width: newData.width,
                height: newData.height
            });

            // 截取精灵图区域
            this.bitblt(newPng, rect.lt.x, rect.lt.y, newData.width, newData.height, 0, 0);

            // 保存为PNG文件
            var dst = fs.createWriteStream('export/' + outDir + '/' + outDir + '' + i + '.png');
            newPng.pack().pipe(dst);
        }
    });

// 生成CSS代码
function getCss(spritesArray, pngname) {
    var css = `.sprite {display:inline-block; overflow:hidden; background-repeat: no-repeat;background-image:url(${pngname});}`;
    for (var i = 0; i < spritesArray.length; i++) {
        var rect = spritesArray[i];
        css += getSpriteCss('sprite' + i, rect);
    }
    return css;
}

// 生成单个精灵图的CSS代码
function getSpriteCss(spritename, rect) {
    return `.${spritename} {width:${rect.rt.x - rect.lt.x}px; height:${rect.rb.y - rect.rt.y}px; background-position: ${0 - rect.lt.x}px ${0 - rect.lt.y}px;}`;
}

// 提取精灵图数组
function getSprites(data, height, width) {
    var spritesArray = [];
    var contourVector = marchingSquares(data, height, width);

    while (contourVector.length > 3) {
        var rect = getRect(contourVector);
        if ((rect.rt.x - rect.lt.x > 3) && (rect.lb.y - rect.lt.y > 3)) {
            spritesArray.push(rect);
        }

        // 清空已提取的区域
        for (var y = rect.rt.y; y < rect.rb.y; y++) {
            for (var x = rect.lb.x; x < rect.rb.x; x++) {
                var idx = (width * y + x) << 2;
                data[idx] = 0;
                data[idx + 1] = 0;
                data[idx + 2] = 0;
                data[idx + 3] = 0;
            }
        }

        contourVector = marchingSquares(data, height, width);
    }
    return spritesArray;
}

// 获取矩形边界
function getRect(squareArray) {
    var rectXY = {};
    rectXY.maxX = squareArray[0].x;
    rectXY.minX = squareArray[0].x;
    rectXY.maxY = squareArray[0].y;
    rectXY.minY = squareArray[0].y;

    for (var i = 0; i < squareArray.length; i++) {
        var p = squareArray[i];
        rectXY.maxX = p.x > rectXY.maxX ? p.x : rectXY.maxX;
        rectXY.maxY = p.y > rectXY.maxY ? p.y : rectXY.maxY;
        rectXY.minX = p.x < rectXY.minX ? p.x : rectXY.minX;
        rectXY.minY = p.y < rectXY.minY ? p.y : rectXY.minY;
    }

    return {
        lt: { x: rectXY.minX, y: rectXY.minY },
        lb: { x: rectXY.minX, y: rectXY.maxY },
        rt: { x: rectXY.maxX, y: rectXY.minY },
        rb: { x: rectXY.maxX, y: rectXY.maxY }
    };
}

// Marching Squares算法实现
function marchingSquares(data, height, width) {
    var contourVector = [];
    var startPoint = getStartingPixel(data, height, width);

    if (startPoint != null && startPoint.x >= 0) {
        var pX = startPoint.x;
        var pY = startPoint.y;
        var stepX = 0;
        var stepY = 0;
        var prevX = 0;
        var prevY = 0;
        var closedLoop = false;

        while (!closedLoop) {
            var squareValue = getSquareValue(data, pX, pY, width);

            switch (squareValue) {
                case 1:
                case 5:
                case 13:
                    stepX = 0;
                    stepY = -1;
                    break;
                case 8:
                case 10:
                case 11:
                    stepX = 0;
                    stepY = 1;
                    break;
                case 4:
                case 12:
                case 14:
                    stepX = -1;
                    stepY = 0;
                    break;
                case 2:
                case 3:
                case 7:
                    stepX = 1;
                    stepY = 0;
                    break;
                case 6:
                    if (prevX == 0 && prevY == -1) {
                        stepX = -1;
                        stepY = 0;
                    } else {
                        stepX = 1;
                        stepY = 0;
                    }
                    break;
                case 9:
                    if (prevX == 1 && prevY == 0) {
                        stepX = 0;
                        stepY = -1;
                    } else {
                        stepX = 0;
                        stepY = 1;
                    }
                    break;
            }

            pX += stepX;
            pY += stepY;
            contourVector.push(new Point(pX, pY));
            prevX = stepX;
            prevY = stepY;

            if (pX == startPoint.x && pY == startPoint.y) {
                closedLoop = true;
            }
        }
    }
    return contourVector;
}

// 获取2x2像素网格的值
function getSquareValue(data, pX, pY, width) {
    var squareValue = 0;
    if (!isAlpha(data, pX - 1, pY - 1, width)) {
        squareValue += 1;
    }
    if (!isAlpha(data, pX, pY - 1, width)) {
        squareValue += 2;
    }
    if (!isAlpha(data, pX - 1, pY, width)) {
        squareValue += 4;
    }
    if (!isAlpha(data, pX, pY, width)) {
        squareValue += 8;
    }
    return squareValue;
}

// 找到第一个非透明像素作为起始点
function getStartingPixel(data, height, width) {
    var offsetPoint = new Point(-1, -1);
    for (var y = 0; y < height; y++) {
        for (var x = 0; x < width; x++) {
            var idx = (width * y + x) << 2;
            var alpha = data[idx + 3];
            if (alpha > 0) {
                offsetPoint.x = x;
                offsetPoint.y = y;
                return offsetPoint;
            }
        }
    }
    return offsetPoint;
}

// 检查像素是否透明
function isAlpha(data, x, y, width) {
    if (x < 0 || y < 0) {
        return true;
    }
    var idx = (width * y + x) << 2;
    var alpha = data[idx + 3];
    return alpha == 0;
}

// 定义点类
var Point = function (_x, _y) {
    this.x = _x;
    this.y = _y;
};

3. 运行代码

将上述代码保存为sprite.js,然后通过以下命令运行:

node sprite.js your-image.png

运行完成后,你将在export目录中找到提取的精灵图和对应的CSS文件。

四、代码解析

  1. PNG.js:用于读取和解析PNG图片,支持像素级操作。
  2. Marching Squares算法:用于提取图片中的轮廓,从而确定每个精灵图的边界。
  3. CSS生成:根据提取的精灵图边界,生成对应的CSS代码,方便在网页中使用。
  4. 文件输出:将提取的精灵图保存为单独的PNG文件,并生成一个CSS文件用于样式定义。

相关推荐

今日起,办理游戏版号这么做就行了!真的太方便了

  在“大众创业,万众创新”的浪潮下,我国很多创业者也看到了游戏的前景,准备在游戏行业分一杯羹。  但根据国家新闻出版广电总局颁布的《关于移动游戏出版服务管理的通知》,游戏需要通过国家新闻出版广电总局...

给大家推荐些好的c语言代码的网站

C语言,那就来推荐几个吧,部分含有C++:1、TheLinuxKernelArchives(kernel.org)Linux内核源码,仅限于C,但内核庞大,不太适合新手;2、redis(redi...

手游平台没有源码的三大危害

搭建一款属于自己的手游平台可以直接和游戏研发商对接游戏,既减少中介的差价,还能根据自己需求去选择游戏。对于玩家而言,手游平台给予了玩家更多的选择机会,对于运营者而言,借助平台可以更好地服务玩家,通过对...

游戏源代码开发时需要什么,需要哪些团队成员?

游戏由于她轻松娱乐,对战刺激,寓教于乐等特点,吸引住了一大批不一样年龄阶段的用户,例如喜爱竞技游戏的年轻群体,需要益智游戏的儿童等。游戏源代码是游戏构建的基础,尽管将开发时分成开发软件和游戏开发2个概...

育碧经典游戏《孤岛惊魂1》源代码遭泄露,玩家表示可以运行

IT之家7月3日消息,一份名为“FarCry1.34Complete”的游戏源代码已经出现在了互联网档案网站“Archive.org”上,并且在Reddit论坛和各种社交媒体上得到...

神秘网站倒数结束 令人一头雾水

还记得那个疑似小岛秀夫作品的《黑色猎犬》倒计时网站吗?现在该网站已经停止倒计时,仅剩一段话“这里原来有一个倒计时,现在没了”……点击这句话会跳转到国外网站Funhaus的一个莫名其妙的视频,然而评论的...

LOL源代码娜美免费领取地址 LOL源代码娜美领取活动网址分享

[海峡网]在英雄联盟中近日国服的服务器一直不稳定,繁出现卡顿和功能错误等问题,官方现在正在努力维护,为表歉意将免费赠送给玩家一款“源代码·娜美”的皮肤,那么这个皮肤要怎么领取呢,小编相信小伙伴们一定都...

个人网站集成js小游戏《圈小猫》教程及源码

今天在某网站浏览帖子的时候,发现帖子被删除了,然后弹出了404页面,页面上集成了一个小游戏,小游戏长什么样子呢?看下面这个图!第一步查看小游戏源码,发现这个小游戏完全是由JavaScript编写的,因...

Scratch创意编程-数学问答游戏

项目名称:数学问答游戏目标年龄群体:8-12岁项目简介:在这个Scratch创意编程项目中,学生们将扮演数学家,通过解答数学题目来挑战自己的数学技能。游戏中包含了加法、减法、乘法和除法等基本算术题,以...

少时不努力长大程序猿 酷比魔方AI百变编程套件体验测评

本文产品为厂家送测,坚持独立的评价观点是笔者创作的基本底线,绝不会因商品来源不同而有所偏颇,请各位放心。写在开始讲讲今天男主的故事这篇体验到的目标群体是跟我一样,家中有个在上小学二年级的小学生。首先...

孩子的scratch作品只能演示?教你把它三步变为电脑软件

随着少儿编程的发展,越来越多的家长和孩子开始投身其中。对于初学者来说,最好的编程工具就是Scratch,它是麻省理工学院的“终身幼儿园团队”开发的图形化编程工具,主要面对青少年开放。这是对孩子最好的编...

打地鼠小游戏制作教程

打地鼠这个小游戏貌似比我的年龄都要大,这次我们使用scratch3.0图形化编程软件来制作一款我们自己的“打地鼠”。我们先准备4样角色,分别是:地鼠角色、锤子角色、地洞角色、草地角色。地鼠→使用猫...

Scratch2.0接苹果小游戏讲义整理

Scratch2.0接苹果小游戏概貌见动图:这又是一款经典的Scratch小游戏,是孩子们学习Scratch编程软件的良好载体,不容错过。(一)玩法说明接到慢速的红苹果一个加1分;接到中速的红苹果一个...

少儿编程太难?原来可以闯关玩游戏啊

随着编程学习全球化的趋势,国内编程学习热潮日盛,越来越多的家长开始让孩子接触学习编程。然而我们都不了解这个少儿编程是到底是什么,近年来,许多家长开始给小孩报编程学习班。最小的从幼儿园开始就在学习...

如何在Scratch中创建一个两人赛艇游戏

本分步指南将教您如何使用Scratch程序创建划船游戏。完成对这个简单游戏的编程后,两条船将使用按键命令一起竞赛。步骤1.打开Scratch。2.删除名为“Sprite1”的猫。您可以通过右键单击它...