Microsoft MVP성태의 닷넷 이야기
.NET Framework: 126.3. CAG - 간단한 유형의 모듈 제작 [링크 복사], [링크+제목 복사],
조회: 19470
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
부모글 보이기/감추기
(연관된 글이 3개 있습니다.)

간단한 유형의 모듈 제작


이번 글에서는 간단하게 모듈 추가 절차를 알아볼 텐데, 전체적인 내용은 다음의 글을 정리한 것입니다.

Composite Application Guidance for WPF - June 2008
How to: Create a Module
; https://docs.microsoft.com/en-us/previous-versions/msp-n-p/ff921345(v=pandp.10)




우선, 모듈이란 뭘까요? 위의 링크에서 제공되는 글의 서두에서 그 정의를 찾아볼 수 있는데요.

A module encapsulates a set of related concerns. Modules are independently developed and deployed, and they interact with each other to create the application.



비록 모듈의 정의가 그러하지만, 현실적으로 CAG 프레임워크로 보자면 Unity Framework의 관리하에 놓인 어셈블리라고 보시면 됩니다. 따라서, 모듈로 정의된 어셈블리는 Unity에 의해서 로드되면서 Reflection 등의 과정을 거쳐서 IoC와 같은 헤택을 받게되는 것입니다.

실제로 추가를 해보면서 이해를 해보는 것도 좋겠지요.

"모듈"로 동작되는 어셈블리를 만들기 위해 "[그림 1]"과 같이 "StatusBar"라는 이름으로 프로젝트를 추가합니다.

[그림 1: StatusBar 프로젝트 추가]
impl_cag_app_step_by_step_chapter3_1.png

다음의 프로젝트 참조를 참조합니다.

  • PresentationCore.dll
  • PresentationFramework.dll
  • WindowsBase.dll
  • Microsoft.Practices.Composite.dll
  • Microsoft.Practices.Composite.Presentation.dll

{ModuleName}Module.cs와 같은 이름 규칙에 따라 새로운 Module 타입을 "StatusBarModule.cs"라는 이름으로 추가하고 코드는 다음과 같이 작성해 줍니다.

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using Microsoft.Practices.Composite.Modularity;
using Microsoft.Practices.Composite.Regions;

namespace StatusBar
{
    public class StatusBarModule : IModule
    {
        private readonly IRegionManager regionManager;

        public StatusBarModule(IRegionManager regionManager)
        {
            this.regionManager = regionManager;
        }

        public void Initialize()
        {
            this.regionManager.Regions["StatusRegion"].Add(new DefaultStatusBar());
        }
    }
}

우선, StatusBarModule 생성자를 보시면 IRegionManager 인터페이스가 넘어오는 것을 볼 수 있습니다. 실제 인스턴스 타입은 "Microsoft.Practices.Composite.Presentation.Regions.RegionManager"로 정의된 것인데, 런타임 시에 Unity에 의해서 기 등록된 IRegionManager의 인스턴스가 주입(Injection)되어 넘어옵니다.

즉, 만약 Unity에 의해서 위의 모듈이 로드되지 않고 단순히 "StatusBarModule test = new StatusBarModule();"과 같은 식으로 생성을 했다면 - 다시 말해서 Unity의 "관리"에 놓이지 않는다면 - IRegionManager 등의 값은 당연히 배정되지 않습니다.

그다음으로 Initialize 메서드를 살펴보면,

이와 같이 모듈 내에서는 해당 모듈이 제공하는 View / Service를 등록해 주면 되는데, 위의 코드에서는 Region 영역에 "DefaultStatusBar" 뷰를 하나 등록하고 있습니다.

본론에서 벗어나지 않기 위해 DefaultStatusBar.xaml은 다음과 같이 간단하게 구현하는 것으로 끝내겠습니다.

<UserControl x:Class="StatusBar.DefaultStatusBar"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
    <Grid>
        <Button MinWidth="80" MinHeight="25" Name="button1">Button</Button>
    </Grid>
</UserControl>

이것으로, 간단한 유형의 모듈 하나가 완성되었습니다. 이제 지금까지 새롭게 만들어진 모듈을 CAG로 하여금 로드를 하라고 시켜야 할 텐데요.

"이전의 글"에서 Bootstrapper.GetModuleCatalog 안에 DirectoryModuleCatalog를 사용했기 때문에 이렇게 만들어진 Module의 출력 경로를 "$(SolutionDir)\bin\Debug\Modules"로 설정해 주면 되는데, 이번 경우에는 코드로도 추가하는 방법과 폴더를 지정하는 방법을 모두 사용할 수 있도록 하는 예시를 보여주기 위해 StatusBar 모듈을 직접 참조해서 코드로 추가해 보겠습니다.

===== Bootstrapper.cs =====

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using Microsoft.Practices.Composite.UnityExtensions;
using System.Windows;
using Microsoft.Practices.Composite.Modularity;
using System.IO;
using StatusBar;

namespace DevToolManager
{
    public class Bootstrapper : UnityBootstrapper
    {
        protected override DependencyObject CreateShell()
        {
            Shell shell = Container.Resolve<Shell>();
            shell.Show();
            return shell;
        }

        protected override Microsoft.Practices.Composite.Modularity.IModuleCatalog GetModuleCatalog()
        {
            string modulePath = @".\Modules";
            if (Directory.Exists(modulePath) == false)
            {
                Directory.CreateDirectory(modulePath);
            }

            DirectoryModuleCatalog catalog = new DirectoryModuleCatalog() { ModulePath = modulePath };
            catalog.AddModule(typeof(StatusBarModule));
            return catalog;
        }
    }
}

간단하지요! 단순히 DirectoryModuleCatalog 인스턴스에서 정적 참조한 어셈블리에 정의된 모듈 타입을 AddModule로 전달해 주면 됩니다. 이렇게 해서 실행시키면 버튼 하나가 있는 WPF 응용 프로그램이 실행되는 것을 확인할 수 있습니다.

[그림 2: StatusBar 모듈이 반영된 실행 화면]
impl_cag_app_step_by_step_chapter3_2.png

다운로드: 예제 솔루션



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

[연관 글]






[최초 등록일: ]
[최종 수정일: 7/17/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)
12925정성태1/17/20227604개발 환경 구성: 627. AKS의 준비 단계 - ACR(Azure Container Registry)에 docker 이미지 배포
12924정성태1/15/20229138.NET Framework: 1134. C# - ffmpeg(FFmpeg.AutoGen)를 이용한 비디오 디코딩 예제(decode_video.c) [2]파일 다운로드1
12923정성태1/15/20228000개발 환경 구성: 626. ffmpeg.exe를 사용해 비디오 파일을 MPEG1 포맷으로 변경하는 방법
12922정성태1/14/20227092개발 환경 구성: 625. AKS - Azure Kubernetes Service 생성 및 SLO/SLA 변경 방법
12921정성태1/14/20226009개발 환경 구성: 624. Docker Desktop에서 별도 서버에 설치한 docker registry에 이미지 올리는 방법
12920정성태1/14/20226784오류 유형: 786. Camtasia - An error occurred with the camera: Failed to Add Video Sampler.
12919정성태1/13/20226643Windows: 199. Host Network Service (HNS)에 의해서 점유되는 포트
12918정성태1/13/20226866Linux: 47. WSL - shell script에서 설정한 환경 변수가 스크립트 실행 후 반영되지 않는 문제
12917정성태1/12/20225987오류 유형: 785. C# - The type or namespace name '...' could not be found (are you missing a using directive or an assembly reference?)
12916정성태1/12/20225797오류 유형: 784. TFS - One or more source control bindings for this solution are not valid and are listed below.
12915정성태1/11/20226073오류 유형: 783. Visual Studio - We didn't find any interpreters
12914정성태1/11/20228105VS.NET IDE: 172. 비주얼 스튜디오 2022의 파이선 개발 환경 지원
12913정성태1/11/20228591.NET Framework: 1133. C# - byte * (바이트 포인터)를 FileStream으로 쓰는 방법 [1]
12912정성태1/11/20229275개발 환경 구성: 623. ffmpeg.exe를 사용해 비디오 파일의 이미지를 PGM(Portable Gray Map) 파일 포맷으로 출력하는 방법 [1]
12911정성태1/11/20226454VS.NET IDE: 171. 비주얼 스튜디오 - 더 이상 만들 수 없는 "ASP.NET Core 3.1 Web Application (.NET Framework)" 프로젝트
12910정성태1/10/20226976제니퍼 .NET: 30. 제니퍼 닷넷 적용 사례 (8) - CPU high와 DB 쿼리 성능에 문제가 함께 있는 사이트
12909정성태1/10/20228349오류 유형: 782. Visual Studio 2022 설치 시 "Couldn't install Microsoft.VisualCpp.Redist.14.Latest"
12908정성태1/10/20226156.NET Framework: 1132. C# - ref/out 매개변수의 IL 코드 처리
12907정성태1/9/20226714오류 유형: 781. (youtube-dl.exe) 실행 시 "This app can't run on your PC" / "Access is denied." 오류 발생
12906정성태1/9/20227363.NET Framework: 1131. C# - 네임스페이스까지 동일한 타입을 2개의 DLL에서 제공하는 경우 충돌을 우회하는 방법 [1]파일 다운로드1
12905정성태1/8/20227020오류 유형: 780. Could not load file or assembly 'Microsoft.VisualStudio.TextTemplating.VSHost.15.0, Version=16.0.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a' or one of its dependencies.
12904정성태1/8/20229009개발 환경 구성: 623. Visual Studio 2022 빌드 환경을 위한 github Actions 설정 [1]
12903정성태1/7/20227621.NET Framework: 1130. C# - ELEMENT_TYPE_INTERNAL 유형의 사용 예
12902정성태1/7/20227655오류 유형: 779. SQL 서버 로그인 에러 - provider: Shared Memory Provider, error: 0 - No process is on the other end of the pipe.
12901정성태1/5/20227684오류 유형: 778. C# - .NET 5+에서 warning CA1416: This call site is reachable on all platforms. '...' is only supported on: 'windows' 경고 발생
12900정성태1/5/20229345개발 환경 구성: 622. vcpkg로 ffmpeg를 빌드하는 경우 생성될 구성 요소 제어하는 방법
... 16  17  18  19  20  21  22  23  24  25  26  27  28  [29]  30  ...