仿动森服装设计DIY功能

神奇的服装设计

神奇吗?

的确,如果反过来,把左边衣服上的图案像素化成右边画板上的图案,确实不神奇,但这里神奇的是反过来了,竟然可以把像素画转换成高质量的平滑贴图。怎么做到的呢?

资料线索

网上相关资料较少,经过一番努力,终于找到了算法Pixel-art scaling algorithms
Wiki网站有时候打不开,我把上面的贴图借一个过来,给大家提供个线索:

各种算法效果

pixel art scaling, Image rescaling, 英语不精通的也很难想到scaling这个搜索关键词,比如我。
竟然有这么多的算法,真是大开眼界。这些算法主要应对在模拟器上运行早期低端游戏机上的游戏,早期很多经典游戏,受限于当时硬件条件,美术资源的分辨率非常低,甚至是像素级别,从而节省内存和提高效率,但是模拟器一般运行在现代高性能设备上,如果还保持原有分辨率,就太浪费了,为了不改变资源的情况下,提高画质,这些算法应运而生。

这篇Wiki里面简单描述了各种算法的原理,基本都是采样原始像素周边若干像素,然后经过一定的算法,最终确定目标图像的像素值。Wiki最后的References中提到了具体实现在Github上的地址,我们美术看了上面的效果图,选择了XBR4x这个算法,shader链接是References中第18条:XBR4x shader.
由于Unity本身支持CG shader,所以应该很好改。

实现

好了现在核心算法以及Shader有了,下面就是实现绘图板以及根据绘制的像素画,生成平滑的贴图,贴到Mesh上进行预览功能了。首先创建一个画布,挂载ImageScale组件:


画布

ImageScale源码如下:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.Experimental.Rendering;
using UnityEngine.UI;

public class ImageScale : MonoBehaviour
{
    public GameObject anchor;   //把生成的图贴到谁身上(测试用)
    public int scale;   //生成的图,相对于原始图,放大倍数
    public Shader shader;   //从原始图到生成图,所用到的Shader
    public Vector2Int resolution;   //原始图的大小
    public Color penColor;  //画笔颜色
    private Material mat;
    private RenderTexture renderTexture;
    private Texture2D texture;
    private bool isDrawing;
    private RectTransform thisRect;
    private Vector2Int lastPoint;
    private Vector2 pixelSize;

    // Start is called before the first frame update
    void Start()
    {
        isDrawing = false;
        thisRect = GetComponent<RectTransform>();

        //计算一个像素占多大尺寸
        pixelSize = thisRect.rect.size / resolution;
        
        Texture2D tex = new Texture2D(resolution.x, resolution.y, TextureFormat.RGBA32, false,true);
        
        if (tex != null && shader != null)
        {
            tex.filterMode = FilterMode.Point;
            texture = tex;
            for(int i=0; i<tex.width; i++)
                for(int j=0; j<tex.height; j++)
                    tex.SetPixel(i, j, Color.white);
            tex.Apply();

            //更新本身的图片
            RawImage image = GetComponent<RawImage>();
            image.texture = tex;


            mat = new Material(shader);
            mat.SetVector("texture_size", new Vector2(tex.width, tex.height));
            mat.SetTexture("_MainTex", tex);
            
            renderTexture =  new RenderTexture(tex.width * scale, tex.height * scale, 0);

            Graphics.Blit(tex, renderTexture, mat);

            MeshRenderer renderer = anchor.GetComponent<MeshRenderer>();
            renderer.material.SetTexture("_BaseMap", renderTexture);
        }
    }

    
    private void DrawLine(Vector2Int newPoint)
    {
        //只处理在框内的
        if (newPoint.x < 0 || newPoint.x >= resolution.x || newPoint.y < 0 || newPoint.y >= resolution.y)
            return;
        if(!isDrawing)
        {
            isDrawing = true;
            texture.SetPixel(newPoint.x, newPoint.y, penColor);
            texture.Apply();

            Graphics.Blit(texture, renderTexture, mat);

            lastPoint = newPoint;
            return;
        }

        //如果移动一段距离了,就划线
        if(newPoint != lastPoint)
        {
            List<Vector2Int> points = GridHelper.GetTouchedPosBetweenTwoPoints(lastPoint, newPoint);
            if(points.Count > 0)
            {
                foreach(Vector2Int p in points)
                {
                    texture.SetPixel(p.x, p.y, penColor);
                }
                texture.Apply();

                Graphics.Blit(texture, renderTexture, mat);

                lastPoint = newPoint;
            }
        }
    }

    //坐标转换
    private Vector2Int PointerDataToRelativePos(Vector2 clickPosition)
    {
        Vector2 result;
        RectTransformUtility.ScreenPointToLocalPointInRectangle(thisRect, clickPosition, null, out result);
        result += thisRect.rect.size / 2;
        Vector2Int intResult = Vector2Int.zero;
        pixelSize = thisRect.rect.size / resolution;
        intResult.Set(Mathf.FloorToInt(result.x / pixelSize.x), Mathf.FloorToInt(result.y / pixelSize.y));
        return intResult;
    }
    
    // Update is called once per frame
    void Update()
    {
        if(Input.GetMouseButton(0))
        {
            DrawLine(PointerDataToRelativePos(Input.mousePosition));
        }
        else
        {
            isDrawing = false;
        }
    }
}

其中在Start函数中,创建一个Texture2D用来绘画,然后创建一个RenderTexture用来接受平滑过的效果,并贴到墙纸对象(用来展示效果的GameObject)上, Update函数中检测鼠标事件,做完坐标转换后,在Texture2D上划线,并用给定的平滑算法Shader,Blit到RenderTexture上,就可以在场景中看到效果了。最终效果如下:


效果

怎么样,结果是不是出乎意料?几个像素就能生成如此平滑的图案,你还有什么创意,一起来试试吧。
【转载请注明出处】

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,290评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,399评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,021评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,034评论 0 207
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,412评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,651评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,902评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,605评论 0 199
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,339评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,586评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,076评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,400评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,060评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,083评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,851评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,685评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,595评论 2 270

推荐阅读更多精彩内容