Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 2개 있습니다.)
(시리즈 글이 3개 있습니다.)
.NET Framework: 281. Shader 강좌와 함께 배워보는 XNA Framework (1) - 기초 프로그램 구조
; https://www.sysnet.pe.kr/2/0/1196

.NET Framework: 282. Shader 강좌와 함께 배워보는 XNA Framework (2) - RenderMonkey의 Shader/Model 파일 연동
; https://www.sysnet.pe.kr/2/0/1197

.NET Framework: 285. Shader 강좌와 함께 배워보는 XNA Framework (3) - 텍스처 매핑 예제
; https://www.sysnet.pe.kr/2/0/1206




Shader 강좌와 함께 배워보는 XNA Framework (2) - RenderMonkey의 Shader/Model 파일 연동


지난번 글에 이어서.

Shader 강좌와 함께 배워보는 XNA Framework (1) - 기초 프로그램 구조
; https://www.sysnet.pe.kr/2/0/1196

이번에는 다음의 글에 실려 있는 Visual C++ 소스 코드를 XNA 프로젝트로 변환해 보겠습니다.

[포프의 쉐이더 입문강좌] 02. 진짜 쉬운 빨강쉐이더 Part 1
; http://kblog.popekim.com/2011/12/02-part-1.html

[포프의 쉐이더 입문강좌] 02. 진짜 쉬운 빨강쉐이더 Part 2
; http://kblog.popekim.com/2011/12/02-part-2.html

실습을 위해 저도 RenderMonkey 프로그램을 다운로드했는데요.

RenderMonkey™ Toolsuite 
; http://developer.amd.com/archive/gpu/rendermonkey/pages/default.aspx

RenderMonkey 1.82를 실행하자마자 프로그램이 비정상 종료되는 현상이 Vista/Win7에서 나타날 수 있습니다. 그런 경우에는 "C:\Program Files (x86)\AMD\RenderMonkey 1.82" 폴더에 "Error.txt" 파일을 추가하고 일반 사용자 권한으로 읽기/쓰기가 가능하도록 설정해 주시면 그런 현상은 사라집니다.

자, 이제 본격적으로 코드 변환을 해볼까요? ^^

우선, 본문에 나온 #define 상수 먼저 C#에서 다음과 같이 추가해 줍니다.

public const float PI = 3.14159265f;
public const float FOV = (PI / 4.0f);
public readonly float ASPECT_RATIO;
public const int NEAR_PLANE = 1;
public const int FAR_PLANE = 10000;


public Game1()
{
    graphics = new GraphicsDeviceManager(this);
    Content.RootDirectory = "Content";

    this.graphics.PreferredBackBufferWidth = 800;
    this.graphics.PreferredBackBufferHeight = 600;

    ASPECT_RATIO = (this.graphics.PreferredBackBufferWidth / (float)this.graphics.PreferredBackBufferHeight);
}

그다음은 Sphere.x와 ColorShader.fx 파일을 로드해야 하는데요. 이 부분은 도움을 받아야겠습니다. ^^

Working with Shaders in XNA 
; http://rbwhitaker.wikidot.com/shaders-in-xna

위의 글을 읽으니 대강 감이 오는 군요.

2개의 파일을 WindowsGame1Content 프로젝트에 추가시키면, Visual Studio가 파일 확장자에 따라 .fx인 경우 "Effect - XNA Framework"으로, .x인 경우 "Model - XNA Framework"으로 알아서 "Content Processor"에 대응시켜줍니다.

그다음은 코드에서 이를 활용해야 할 텐데요. 복잡한 Visual C++ 소스 코드와는 달리 XNA에서는 다음과 같이 간단하게 끝납니다.

Model _modelSphere;
Effect _colorShader;

protected override void LoadContent()
{
    spriteBatch = new SpriteBatch(GraphicsDevice);

    _spriteFont = Content.Load<SpriteFont>("Font");
    _modelSphere = Content.Load<Model>("Sphere");
    _colorShader = Content.Load<Effect>("ColorShader");
}

Shader를 로드했으니, 이제 Shader 내부의 전역 변수에 값을 설정해야 하는데 이 부분은 Visual C++과 어느 정도 대응되는 면이 있습니다.

=== Visual C++ ===
D3DXMATRIXA16 matView;
D3DXVECTOR3 vEyePt(     0.0f, 0.0f, -200.0f );
D3DXVECTOR3 vLookatPt(  0.0f, 0.0f, 0.0f);
D3DXVECTOR3 vUpVec(     0.0f, 1.0f, 0.0f);
D3DXMatrixLookAtLH(&matView, &vEyePt, &vLookatPt, &vUpVec);

D3DXMATRIXA16 matProjection;
D3DXMatrixPerspectiveFovLH(&matProjection, FOV, ASPECT_RATIO, NEAR_PLANE, FAR_PLANE);

D3DXMATRIXA16 matWorld;
D3DXMatrixIdentity(&matWorld);

gpColorShader->SetMatrix("gWorldMatrix", &matWorld);
gpColorShader->SetMatrix("gViewMatrix", &matView);
gpColorShader->SetMatrix("gProjectionMatrix", &matProjection);


=== Visual C# ===
_matView = Matrix.CreateLookAt(new Vector3(0, 0, -200), new Vector3(0, 0, 0), new Vector3(0, 1, 0));
_matProjection = Matrix.CreatePerspectiveFieldOfView(FOV, ASPECT_RATIO, NEAR_PLANE, FAR_PLANE);
_matWorld = Matrix.Identity;
            
_colorShader.Parameters["gWorldMatrix"].SetValue(_matWorld);
_colorShader.Parameters["gViewMatrix"].SetValue(_matView);
_colorShader.Parameters["gProjectionMatrix"].SetValue(_matProjection);

마지막으로 Shader를 그려 봐야 하는데요. 사실 이 부분의 작업에 애를 먹긴 했습니다. 처음엔, Effect 타입에서 Begin/End 메서드를 정의해 주는 식으로 Visual C++ 소스 코드와 거의 동일할 거라고 생각해서 나름대로 유사한 함수와 속성을 찾아 보았는데요. 결국 포기하고 ^^ 다음의 글을 검색해서 도움을 받았습니다.

Game Creation with XNA/3D Development/Shaders and Effects
; http://en.wikibooks.org/wiki/Game_Creation_with_XNA/3D_Development/Shaders_and_Effects

아래는 최종 반영된 코드!

foreach (ModelMesh mesh in _modelSphere.Meshes)
{
    foreach (ModelMeshPart part in mesh.MeshParts)
    {
        part.Effect = _colorShader;
    }
                
    mesh.Draw();
}

실행하니, 정말 Visual C++ 소스 코드처럼 동일하게 결과물이 나왔습니다. (신기하네요. ^^)

how_to_draw_shader_1.png

참고로, Shader가 XNA 내부적으로도 몇 가지 제공되는 것 같습니다.

가령, "Working with Shaders in XNA" 글에 보면 다음과 같이 BasicEffect를 사용하는 것을 볼 수 있습니다.

foreach (ModelMesh mesh in _modelSphere.Meshes)
{
    foreach (BasicEffect effect in mesh.Effects)
    {
        effect.EnableDefaultLighting();
        effect.PreferPerPixelLighting = true;
        effect.World = _matWorld * mesh.ParentBone.Transform;
        effect.View = _matView;
        effect.Projection = _matProjection;
    }
}

위와 같이 하면, Sphere.x 모델에다 BasicEffect를 적용하는 것이므로 다음과 같이 렌더링이 됩니다.

how_to_draw_shader_2.png

그나저나... 단순히 '구' 하나 그리는 것도 저렇게 힘든데... Lineage2 / WOW 같은 3D MMORPG를 만드는 게임 프로그래머들은... 3차원 공간에 대해 머릿속에서 수학적으로 바라볼 수 있는 회로가 만들어진 천재들이 아닌가... 하는 생각이 듭니다. ^^;

첨부 파일:

01_DxFramework.zip: 원 강좌에서 공개된 소스 코드의 Unicode 버전
WindowsGame1.zip: XNA로 변환된 C# 프로젝트




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 6/23/2021]

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

비밀번호

댓글 작성자
 




... 16  17  18  19  20  21  22  23  24  25  26  27  28  29  [30]  ...
NoWriterDateCnt.TitleFile(s)
11871정성태4/16/201922022.NET Framework: 818. (번역글) .NET Internals Cookbook Part 3 - Initialization tricks [3]파일 다운로드1
11870정성태4/16/201921299.NET Framework: 817. Process.Start로 실행한 콘솔 프로그램의 출력 결과를 얻는 방법파일 다운로드1
11869정성태4/15/201927925.NET Framework: 816. (번역글) .NET Internals Cookbook Part 2 - GC-related things [2]파일 다운로드2
11868정성태4/15/201923060.NET Framework: 815. CER(Constrained Execution Region)이란?파일 다운로드1
11867정성태4/15/201922103.NET Framework: 814. Critical Finalizer와 SafeHandle의 사용 의미파일 다운로드1
11850정성태3/18/201924505.NET Framework: 813. C# async 메서드에서 out/ref/in 유형의 인자를 사용하지 못하는 이유
11849정성태3/18/201923358.NET Framework: 812. pscp.exe 기능을 C#으로 제어하는 방법파일 다운로드1
11838정성태3/7/201928250.NET Framework: 811. (번역글) .NET Internals Cookbook Part 1 - Exceptions, filters and corrupted processes [1]파일 다운로드1
11835정성태3/5/201923333.NET Framework: 810. C# 8.0의 Index/Range 연산자를 .NET Framework에서 사용하는 방법 및 비동기 스트림의 컴파일 방법 [3]파일 다운로드1
11813정성태2/12/201919549.NET Framework: 809. C# - ("Save File Dialog" 등의) 대화 창에 확장 속성을 보이는 방법
11810정성태2/11/201919790.NET Framework: 808. .NET Profiler - GAC 모듈에서 GAC 비-등록 모듈을 참조하는 경우의 문제
11809정성태2/11/201922719.NET Framework: 807. ClrMD를 이용해 메모리 덤프 파일로부터 특정 인스턴스를 참조하고 있는 소유자 확인
11805정성태1/28/201923865.NET Framework: 806. C# - int []와 object []의 차이로 이해하는 제네릭의 필요성 [4]파일 다운로드1
11802정성태1/7/201921908.NET Framework: 805. 두 개의 윈도우를 각각 실행하는 방법(Windows Forms, WPF)파일 다운로드1
11799정성태12/19/201824535.NET Framework: 804. WPF(또는 WinForm)에서 UWP UI 구성 요소 사용하는 방법 [3]파일 다운로드1
11778정성태11/14/201821214.NET Framework: 803. UWP 앱에서 한 컴퓨터(localhost, 127.0.0.1) 내에서의 소켓 연결
11776정성태11/13/201820698.NET Framework: 802. Windows에 로그인한 계정이 마이크로소프트의 계정인지, 로컬 계정인지 알아내는 방법
11768정성태11/2/201821559.NET Framework: 801. SOIL(Simple OpenGL Image Library) - Native DLL 및 .NET DLL 제공
11758정성태10/23/201822923.NET Framework: 800. C# - Azure REST API 사용을 위한 인증 획득 [3]파일 다운로드1
11747정성태10/17/201822179.NET Framework: 799. C# - DLL에도 EXE처럼 Main 메서드를 넣어 실행할 수 있도록 만드는 방법파일 다운로드1
11722정성태10/4/201823394.NET Framework: 798. C# - Hyper-V 가상 머신의 직렬 포트와 연결된 Named Pipe 간의 통신파일 다운로드1
11721정성태10/4/201823862.NET Framework: 797. Linux 환경의 .NET Core 응용 프로그램에서 직렬 포트(Serial Port, COM Port) 사용 방법파일 다운로드1
11719정성태10/4/201825940.NET Framework: 796. C# - 인증서를 윈도우에 설치하는 방법
11712정성태10/2/201829010.NET Framework: 795. C# - 폰트 목록을 한글이 아닌 영문으로 구하는 방법 [3]
11710정성태10/2/201823564.NET Framework: 794. C# - 같은 모양, 다른 값의 한글 자음을 비교하는 호환 분해 [5]
11696정성태9/19/201822393.NET Framework: 793. C# - REST API를 이용해 NuGet 저장소 제어파일 다운로드1
... 16  17  18  19  20  21  22  23  24  25  26  27  28  29  [30]  ...