Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)
(시리즈 글이 13개 있습니다.)
Graphics: 2. Unity로 실습하는 Shader
; https://www.sysnet.pe.kr/2/0/11607

Graphics: 3. Unity로 실습하는 Shader (1) - 컬러 반전 및 상하/좌우 뒤집기
; https://www.sysnet.pe.kr/2/0/11608

Graphics: 4. Unity로 실습하는 Shader (2) - 고로 셰이딩(gouraud shading) + 퐁 모델(Phong model)
; https://www.sysnet.pe.kr/2/0/11609

Graphics: 5. Unity로 실습하는 Shader (3) - 고로 셰이딩(gouraud shading) + 퐁 모델(Phong model) + Texture
; https://www.sysnet.pe.kr/2/0/11610

Graphics: 6. Unity로 실습하는 Shader (4) - 퐁 셰이딩(phong shading)
; https://www.sysnet.pe.kr/2/0/11611

Graphics: 7. Unity로 실습하는 Shader (5) - Flat Shading
; https://www.sysnet.pe.kr/2/0/11613

Graphics: 8. Unity Shader - Texture의 UV 좌표에 대응하는 Pixel 좌표
; https://www.sysnet.pe.kr/2/0/11614

Graphics: 9. Unity Shader - 전역 변수의 초기화
; https://www.sysnet.pe.kr/2/0/11616

Graphics: 10. Unity로 실습하는 Shader (6) - Mosaic Shading
; https://www.sysnet.pe.kr/2/0/11619

Graphics: 11. Unity로 실습하는 Shader (7) - Blur (평균값, 가우스, 중간값) 필터
; https://www.sysnet.pe.kr/2/0/11620

Graphics: 12. Unity로 실습하는 Shader (8) - 다중 패스(Multi-Pass Shader)
; https://www.sysnet.pe.kr/2/0/11628

Graphics: 13. Unity로 실습하는 Shader (9) - 투명 배경이 있는 텍스처 입히기
; https://www.sysnet.pe.kr/2/0/11631

Graphics: 19. Unity로 실습하는 Shader (10) - 빌보드 구현
; https://www.sysnet.pe.kr/2/0/11641




Unity로 실습하는 Shader (9) - 투명 배경이 있는 텍스처 입히기

지난 글에서,

Unity로 실습하는 Shader (1) - 컬러 반전 및 상하/좌우 뒤집기
; https://www.sysnet.pe.kr/2/0/11608

실습한 나무 그림의,

Trees0091
; https://www.textures.com/download/trees0091/74537

texture를 3D GameObject 중 Plane에 입혀 보겠습니다. 그런데, 분명히 png 이미지로는 똑바로 선 나무 그림인데 기본 shader + material 상태에서 평면에 누워 있는 나무 이미지가 카메라가 바라보는 시점에 대해 거꾸로 되어 있습니다.

blend_shader_1.png

검색해 보면, 이미 다들 알고 있는 문제(?)입니다.

Why are my textures on planes flipped / reversed left-to-right?
; https://answers.unity.com/questions/125855/why-are-my-textures-on-planes-flippedreversed-left.html

그래서 material의 Texture 설정에서 Tiling 값의 Y를 -1로 바꿔야 합니다. 그럼 나무가 똑바로 서긴 하는데 자세히 보면 좌우가 바뀐 상태입니다. 따라서 X 값도 -1로 설정해 줘야 합니다.

Tiling X == -1   Y == -1

그러고 나면 다음과 같이 이미지가 png 파일 그대로 (카메라 방향에 맞게 의도한 대로) 나오게 됩니다.

blend_shader_2.png

자, 이제 나무를 입힌 평면을 카메라에 정면으로 세워야 하는데요, 이를 위해 x축을 기준으로 회전시켜야 합니다. 보통 생각으로는 90도 회전을 시키면 될 것 같은데, Unity의 경우 왼손 좌표계이므로 x축 회전은 양수(+) 값에 대해 다음과 같이 회전을 하게 됩니다.

[그림 출처: http://www.fundza.com/rib/example4/index.html]
blend_shader_3.png

따라서 Plane의 앞면이 카메라가 보이는 쪽으로 서 있으려면 -90도를 해야 합니다. 자, 그래서 결국 다음과 같이 겨우 나무 하나를 보이게 만들었습니다. ^^

blend_shader_4.png




여기서 문제(?)가 있는데요, 저 나무 그림(png)의 배경은 원래 투명(alpha == 0)했는데 Unity의 Plane 객체에 material로 입히면 저렇게 배경이 검은색으로 나온다는 점입니다. 여기서 shader의 기본 코드를 볼까요?

Shader "My/blendShader"
{
    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 _MainTex_ST;
            
            v2f vert (appdata v)
            {
                v2f o;
                o.vertex = UnityObjectToClipPos(v.vertex);
                o.uv = TRANSFORM_TEX(v.uv, _MainTex);
                return o;
            }
            
            fixed4 frag (v2f i) : SV_Target
            {
                fixed4 col = tex2D(_MainTex, i.uv);
                return col;
            }
            ENDCG
        }
    }
}

pixel shader를 보면, 정상적으로 uv 좌표에 따라 texture로부터 fixed4 형식으로 alpha 값까지 해서 추출한 것이 맞습니다.

fixed4 frag (v2f i) : SV_Target
{
    fixed4 col = tex2D(_MainTex, i.uv);
    return col;
}

실제로 나무의 배경은 fixed4(0, 0, 0, 0) 값들이 반환되고 있습니다. 만약 alpha 값이 먹혔다면 투명으로 나와야겠지만 그렇지 않았다면 (0,0,0) 색상이니 검은색으로 나오는 것이 맞습니다. 그리고 이렇게 불투명하게 그리게 된 데에는 shader의 RenderType이 "Opaque"로 설정되어 있기 때문입니다.

Tags { "RenderType"="Opaque" }

투명 처리를 위해 이 값을 "Transparent"로 바꾸고, 아울러 렌더링 우선순위를 "Transparent"로 바꿔줍니다.

Tags { "RenderType"="Transparent" "Queue" = "Transparent" }

마지막으로, 배경과 어떻게 Alpha 채널이 섞이게 될지를 알려주는 Blend 옵션을 주는데,

Blend SrcAlpha OneMinusSrcAlpha

합이 1이 되는 아핀 조합으로 전형적인 선형 결합의 Blend 옵션입니다. 따라서, 투명 처리가 된 곳의 alpha == 0이므로,

SrcAlpha                OneMinusSrcAlpha
트리 texture == 0,      배경 == OneMinusSrcAlpha == 1 - 0 == 1

다음과 같이 투명 처리가 근사하게 적용된 것을 볼 수 있습니다. ^^

blend_shader_5.png




shader에 사용된 TRANSFORM_TEX는 (C:\Program Files\Unity\Editor\Data\CGIncludes\)UnityCG.cginc 파일에 다음과 같은 매크로로 정의되어 있습니다.

// Transforms 2D UV by scale/bias property
#define TRANSFORM_TEX(tex,name) (tex.xy * name##_ST.xy + name##_ST.zw)

따라서 shader에 사용된 코드는 다음과 같이 확장됩니다.

// o.uv = TRANSFORM_TEX(v.uv, _MainTex);
   o.uv = v.uv * _MainTex_ST.xy + _MainTex_ST.zw;

_MainTex_ST는 Unity의 Material에 대한 Inspector 창의 "Tiling"과 "Offset" 값을 담고 있습니다.

blend_shader_0.png

그렇기 때문에 "o.uv = TRANSFORM_TEX(v.uv, _MainTex);" 코드를 쓰지 않고 그냥 "o.uv = v.uv"와 같이 대입하면 Unity 에디터에서 설정한 "Tiling"과 "Offset"값이 적용되지 않습니다.




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 3/29/2024]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2020-10-15 05시24분
[최강한화] 한참을 찾던 내용입니다. 덕분에 골썩이던 크로마키 배경 투명처리하고 갑니다 ㅠ
[guest]

... 91  92  93  94  95  96  97  98  99  100  101  102  103  [104]  105  ...
NoWriterDateCnt.TitleFile(s)
11319정성태10/11/201728123.NET Framework: 686. C# - string 배열을 담은 구조체를 직렬화하는 방법
11318정성태10/7/201720884VS.NET IDE: 122. 비주얼 스튜디오에서 관리자 권한을 요구하는 C# 콘솔 프로그램 제작 [1]
11317정성태10/4/201726034VC++: 120. std::copy 등의 함수 사용 시 _SCL_SECURE_NO_WARNINGS 에러 발생
11316정성태9/30/201724108디버깅 기술: 99. (닷넷) 프로세스(EXE)에 디버거가 연결되어 있는지 아는 방법 [4]
11315정성태9/29/201740197기타: 68. "시작하세요! C# 6.0 프로그래밍: 기본 문법부터 실전 예제까지" 구매하신 분들을 위한 C# 7.0/7.1 추가 문법 PDF [8]
11314정성태9/28/201721934디버깅 기술: 98. windbg - 덤프 파일로부터 닷넷 버전 확인하는 방법
11313정성태9/25/201719251디버깅 기술: 97. windbg - 메모리 덤프로부터 DateTime 형식의 값을 알아내는 방법파일 다운로드1
11312정성태9/25/201722255.NET Framework: 685. C# - 구조체(값 형식)의 필드를 리플렉션을 이용해 값을 바꾸는 방법파일 다운로드1
11311정성태9/20/201716814.NET Framework: 684. System.Diagnostics.Process 객체의 명시적인 해제 권장
11310정성태9/19/201720219.NET Framework: 683. WPF의 Window 객체를 생성했는데 GC 수집 대상이 안 되는 이유 [3]
11309정성태9/13/201718344개발 환경 구성: 335. Octave의 명령 창에서 실행한 결과를 복사하는 방법
11308정성태9/13/201719386VS.NET IDE: 121. 비주얼 스튜디오에서 일부 텍스트 파일을 무조건 메모장으로만 여는 문제파일 다운로드1
11307정성태9/13/201721888오류 유형: 421. System.Runtime.InteropServices.SEHException - 0x80004005
11306정성태9/12/201719932.NET Framework: 682. 아웃룩 사용자를 위한 중국어 스팸 필터 Add-in
11305정성태9/12/201721444개발 환경 구성: 334. 기존 프로젝트를 Visual Studio를 이용해 Github의 신규 생성된 repo에 올리는 방법 [1]
11304정성태9/11/201718585개발 환경 구성: 333. 3ds Max를 Hyper-V VM에서 실행하는 방법
11303정성태9/11/201721879개발 환경 구성: 332. Inno Setup 파일의 관리자 권한을 제거하는 방법
11302정성태9/11/201718101개발 환경 구성: 331. SQL Server Express를 위한 방화벽 설정
11301정성태9/11/201717020오류 유형: 420. SQL Server Express 연결 오류 - A network-related or instance-specific error occurred while establishing a connection to SQL Server.
11300정성태9/10/201720815.NET Framework: 681. dotnet.exe - run, exec, build, restore, publish 차이점 [3]
11299정성태9/9/201719583개발 환경 구성: 330. Hyper-V VM의 Internal Network를 Private 유형으로 만드는 방법
11298정성태9/8/201722851VC++: 119. EnumProcesses / EnumProcessModules API 사용 시 주의점 [1]
11297정성태9/8/201719513디버깅 기술: 96. windbg - 풀 덤프에 포함된 모든 닷넷 모듈을 파일로 저장하는 방법
11296정성태9/8/201722721웹: 36. Edge - "이 웹 사이트는 이전 기술에서 실행되며 Internet Explorer에서만 작동합니다." 끄는 방법
11295정성태9/7/201720073디버깅 기술: 95. Windbg - .foreach 사용법
11294정성태9/4/201719878개발 환경 구성: 329. 마이크로소프트의 CoreCLR 프로파일러 예제 빌드 방법 [1]
... 91  92  93  94  95  96  97  98  99  100  101  102  103  [104]  105  ...