unity3D 涂涂乐使用shader实现上色效果
unity3D 涂涂乐使用shader实现上色效果
之前我博文里面发过一个简单的通过截图方式来实现的模型上色方法,但是那个方法不合适商用,因为你需要对的很准确才可以把贴图完美截取下来,只要你手抖了一下就会发现贴歪了。那么有没有更好的方法来实现这个效果呢,这就需要使用Shader的方式来实现这个效果。
刚好看到了一篇有关于涂涂乐原理的实现方法,EsayAR的官方案例里面就是使用的这个方法,EsayAR的官方案例其实基本完成这个涂涂乐的效果在看过涂涂乐原理,结合案例的是用Vuforia实现的一个效果,发现利用Shader的方式来实现涂涂乐UV图片的修正这个方式比利用对准框的方法要好,而且也避免了UV在位置偏移的问题。(PS:点击打开链接 Vuforia的案例,注意PC上是没有问题的,但是移动端UV会出现破面的问题)
这里要注意PC上实现的效果并不代表手机上能实现涂涂了得效果,就如Vuforia的案例一样PC上是没有任何问题,但是在移动端就会出现一些问题。(PS:这里我使用的EsayAR的来实现Vuforia的方法)
注意这里提供的源码是使用EsayAR来写的(PS:移动端是有问题)
using System;
using UnityEngine;
using System.Collections;
using EasyAR; public class Coloring3D : MonoBehaviour
{ public MeshRenderer realityPlane;
public Camera rendercamera;
private Vector3[] marks;
private Material cubeMat;
private Vector2 sizeVector2;
private Vector2 webcamTex2ScreenScale;
private ImageTargetBehaviour imageTargetBehaviour;
private Texture2D scanTexture2D;
private bool isSetupScale = false; // Use this for initialization
void Start ()
{
Init();
} public void Init()
{
marks = new Vector3[4];
sizeVector2 = new Vector2();
imageTargetBehaviour = transform.GetComponentInParent<ImageTargetBehaviour>();
sizeVector2 = imageTargetBehaviour.Size; //计算识别Plane的4个角的位置
for (int i = 0; i < marks.Length; i++)
{
float x = -1, y = -1;
if ((i + 1) % 2 == 0)
x = 1;
else
x = -1;
if (i >= 2)
y = 1;
marks[i] = new Vector3(sizeVector2.x / 2 * x, 0, sizeVector2.y / 2 * y);
} } void SetupScale()
{
//if (!imageTargetBehaviour.GetIsShow())//判断模型是不是显示出来,可以注释掉
//return;
if (!isSetupScale)
{
cubeMat = this.transform.GetComponent<MeshRenderer>().material; var texture = realityPlane.material.mainTexture;//获取摄像头照到的纹理图片
cubeMat.mainTexture = texture;//将纹理赋值到材质球上 //计算屏幕和整张图片的比例
var maxScale = Mathf.Max((float)Screen.width / texture.width, (float)Screen.height / texture.height); var t2sWidth = texture.width * maxScale;
var t2sHeight = texture.height * maxScale; webcamTex2ScreenScale.x = Screen.width / t2sWidth;
webcamTex2ScreenScale.y = Screen.height / t2sHeight;
isSetupScale = true;
} for (int i = 0; i < marks.Length; i++)
{
marks[i] = marks[i] + this.transform.parent.position;
var pos =
AdjustUV(rendercamera.WorldToViewportPoint(marks[i]));
string posName = String.Format("p{0}", i);
cubeMat.SetVector(posName, pos);//传入到shader进行纹理的拉伸
} } Vector4 AdjustUV(Vector3 v)
{
//识别图在屏幕的位置
var webcamX = FixWebcamTextureToScreenUV(v.x, webcamTex2ScreenScale.x);
var webcamY = FixWebcamTextureToScreenUV(v.y, webcamTex2ScreenScale.y);
return new Vector4(webcamX*v.z, (1 - webcamY)*v.z, v.z, 1.0f);
} float FixWebcamTextureToScreenUV(float value, float scale)
{
return (1.0f - scale) * 0.5f + value * scale;
} // Update is called once per frame
void OnWillRenderObject()
{
SetupScale();
}
}
接下来是shader:
Shader "Unlit/PaintingShader"
{
Properties
{
_MainTex ("Texture", 2D) = "white" {}
}
SubShader
{
Tags { "RenderType"="Opaque" }
LOD 100 Pass
{
CGPROGRAM
#pragma vertex vert
#pragma fragment frag #include "UnityCG.cginc"
struct appdata
{
float4 vertex : POSITION;
float2 uv : TEXCOORD0;
}; struct v2f
{
float2 uv : TEXCOORD0;
float4 vertex : SV_POSITION;
}; sampler2D _MainTex;
float4 p0;
float4 p1;
float4 p2;
float4 p3; v2f vert (appdata v)
{
v2f o;
o.vertex = mul(UNITY_MATRIX_MVP, v.vertex);
o.uv.x = v.uv.x;
o.uv.y = v.uv.y;
return o;
} fixed4 frag (v2f i) : SV_Target
{
float2 mid;
float3 newUV; //计算纹理的位置
if (i.uv.x + i.uv.y >= 1) {
mid = i.uv;
newUV = p0 * (1 - mid.x - mid.y) + p1 * mid.x + p2 * mid.y;
} else {
mid = float2(1 - i.uv.y, i.uv.x + i.uv.y - 1);
newUV = p2 * (1 - mid.x - mid.y) + p1 * mid.x + p3 * mid.y;
}
//返回计算好的纹理位置
return tex2D(_MainTex, newUV.xy / newUV.zz);
} ENDCG
}
}
}
这里看看效果PC的效果图片吧!
(PS:这里要注意移动端是会出现问题的,但是PC端是没有问题的)
最后附上一张移动端的涂涂乐效果图出来
功能有实时渲染,点击屏幕后就取消实时渲染然后上色到模型上。(PS:移动端的和PC端的代码编写差异挺大的,但是还是离不上面用到的源码)
Vuforia的移动端问题有待解决,如果解决这个问题我会第一时间写博客告诉大家。
(如果想交流一下涂涂乐的问题,可以加我QQ:245076259)
unity3D 涂涂乐使用shader实现上色效果的更多相关文章
- unity3D AR涂涂乐制作浅谈
unity3D AR涂涂乐制作浅谈 AR为现在是虚拟现实较为火爆的一个技术,其中有个比较炫酷的就是AR涂涂乐的玩法,这个技术可以把扫描到的图片上的纹理 粘贴到模型上实现为模型上色的功能,但是我们需要怎 ...
- 手把手教你做个AR涂涂乐
前段时间公司有一个AR涂涂乐的项目,虽然之前接触过AR也写过小Demo,但是没有完整开发过AR项目.不过经过1个多星期的学习,现在已经把项目相关的技术都学会了,在此向互联网上那些乐于分享的程序员前辈们 ...
- AR涂涂乐
<1> 涂涂乐着色 https://blog.csdn.net/begonia__z/article/details/51282932 http://www.manew.com/blog- ...
- 涂涂影院APP-免费VIP电影观看「安卓APP」
最新下载链接:https://www.lanzous.com/u/niceyoo 2019年基本就没推广过这款APP,很失败,从第一版发布到现在涂涂影院已经做了2年了, 由于没有官网,所以基本百度能搜 ...
- Unity Shader实现描边效果
http://gad.qq.com/article/detail/28346 描边效果是游戏里面非常常用的一种效果,一般是为了凸显游戏中的某个对象,会给对象增加一个描边效果.本篇文章和大家介绍下利用S ...
- pixijs shader 制作百叶窗效果
pixijs shader 制作百叶窗效果 直接贴代码了 const app = new PIXI.Application({ transparent: true }); document.body. ...
- Unity3d 镜面反射 vertex and frag Shader源代码
Unity3d 镜面反射 网上能找到的基本上是固定管道或表面渲染的shader. 特此翻译为顶点.片段渲染的Shader, 本源代码仅仅涉及shader与cs部分. Editor部分使用NGUI绘制的 ...
- Unity3D 屏幕空间雪场景Shader渲染
笔者介绍:姜雪伟,IT公司技术合伙人,IT高级讲师,CSDN社区专家,特邀编辑,畅销书作者,已出版书籍:<手把手教你架构3D游戏引擎>电子工业出版社和<Unity3D实战核心技术详解 ...
- Unity3D之高级渲染-Shader Forge增强版
笔者介绍:姜雪伟,IT公司技术合伙人,IT高级讲师,CSDN社区专家.特邀编辑.畅销书作者,国家专利发明人;已出版书籍:<手把手教你架构3D游戏引擎>电子工业出版社和<Unity3D ...
随机推荐
- 学习HTML 第一节.小试牛刀
此贴并非教学,主要是自学笔记,所述内容只是些许个人学习心得的记录和备查积累,难以保证观点正确,也不一定能坚持完成. 如不幸到访,可能耽误您的时间,也难及时回复,贴主先此致歉.如偶有所得,相逢有缘,幸甚 ...
- Codeforces Round #502 (in memory of Leopoldo Taravilse, Div. 1 + Div. 2) E. The Supersonic Rocket
这道题比赛之后被重新加了几个case,很多人现在都过不了了 算法就是先求凸包,然后判断两个凸包相等 我们可以吧凸包序列化为两点距离和角度 角度如果直接拿向量的叉积是不对的,,因为钝角和锐角的叉积有可能 ...
- 《Learning scikit-learn Machine Learning in Python》chapter1
前言 由于实验原因,准备入坑 python 机器学习,而 python 机器学习常用的包就是 scikit-learn ,准备先了解一下这个工具.在这里搜了有 scikit-learn 关键字的书,找 ...
- 微软职位内部推荐-Principal Development Lead - SharePoint
微软近期Open的职位: SharePoint is a multi-billion dollar enterprise business that has grown from an on-prem ...
- charles 在mac下 抓取 https包
1. 打开charles --> help --> SSL proxying --> install charles root certificate 2. 在弹出的添加证书窗口中 ...
- 如何理解IPD+CMMI+Scrum一体化研发管理解决方案之CMMI篇
如何快速响应市场的变化,如何推出更有竞争力的产品,如何在竞争中脱颖而出,是国内研发企业普遍面临的核心问题,为了解决这些问题,越来越多的企业开始重视创新与研发管理,加强研发过程的规范化,集成产品开发(I ...
- php json 转换
在PHP语言中使用JSON 作者: 阮一峰 日期: 2011年1月14日 目前,JSON已经成为最流行的数据交换格式之一,各大网站的API几乎都支持它. 我写过一篇<数据类型和JSON格式& ...
- 20172305 2018-2019-1 《Java软件结构与数据结构》第六周学习总结
20172305 2018-2019-1 <Java软件结构与数据结构>第六周学习总结 教材学习内容总结 本周内容主要为书第十章内容: 树(一种非线性结构,其中的元素被组织成一个层次结构) ...
- tomcat启动时,报java.io.EOFException
在启动Tomcat的时候突然报出IO异常,具体异常如下图 在网上找了解决方法,测试了好几种,都不行,到最后看了一个博客解决了,在此记录一下. 百度了下,网上都是说去Tomcat目录下:将tomcat5 ...
- 【线段树求区间第一个不大于val的值】Lpl and Energy-saving Lamps
https://nanti.jisuanke.com/t/30996 线段树维护区间最小值,查询的时候优先向左走,如果左边已经找到了,就不用再往右了. 一个房间装满则把权值标记为INF,模拟一遍,注意 ...