Prism 으로 프로젝트를 하고 있습니다.
스택오버플로우에서도 언급하기도 했는데 Prism은 대형 프로젝트에 적합하다고 합니다.
저는 이 의견에 대해서는 조금 더 생각 해 볼 필요가 있다고 생각이 됩니다.
WPF/Winform에서 가장 작은 단위의 프로젝트라고 하면 간단한 window application을 작업하는 형태라고 생각 됩니다. 하지만 이런 형태의 소규모 프로젝트에서도 Prism은 아주 유용하다고 생각 됩니다. 특히나 연차가 쌓여 갈 수록 사용하게 되는 Component들이나 구성하게 되는 아키텍처가 정형화 됩니다.
Prism은 여기서 아주 큰 힘을 발휘 합니다. 코드 재사용성에 있어서는 정말 미쳤다고 생각이 듭니다.
예를 들어보죠. 저는 visual studio 형태의 메뉴바를 선호 하는데 메뉴바 자체를 Prism의 Module로 만들어 버리고 나중에 새로운 프로젝트를 할 때 그냥 모듈만 불러옵니다. 고객의 요구사항에 따라서 해당 메뉴바의 기능만 조금 고칩니다. 더해서, 모듈을 불러오는 코드도 단 몇줄이면 import 됩니다.
Prism 이전의 프로젝트 방식은 다시 한번 코딩 하는 것이였다면, Prism 사용 이후로는 단지 모듈을 import 하는 것만으로도 프로젝트를 진행 할 수 있습니다. (미리 만들어둔 dll을 사용한다는 느낌과는 좀 다릅니다.)
결론은 Prism의 숙련도가 올라가면 올라갈수록 작은 프로젝트에서도 훨씬더 빠른 작업이 가능하다고 말하고 싶습니다. 프로젝트의 크기와 상관없이 유지보수와 생산성이 엄청나게 올라간다고 생각하고 있습니다.
다만, Prism의 단점은 배워야할게 너무 많다는 것 입니다. 러닝커브가 오기까지 정말 시간이 너무 오래걸립니다.
Dependency injection과 Inversion of Control 의 이해는 물론이고 이를 프로젝트에서 자유자재로 다룰 수 있어야 합니다. 또 WPF/Winform의 깊은 이해와 C#언어에 대한 이해도 같이 필요하구요. 언어 뿐만이 아니라 기본적인 OOP 지식과 디자인 패턴을 숙지하고 있어야 합니다. 거기다가 MVVM의 이해도까지… 여기서 WPF가 아니라 Xamarin으로 넘어간다면 지식의 요구사항이 더 올라갑니다. 더해서 Document가 생각보다 친절하지 않아서 정말 고생 고생하면서 삽질을 해야 한다는 거죠.
예를 들어볼까요?
버튼 클릭 → 화면 출력 형태의 구조를 만든다면
기본 WPF에서는 xaml에서 button을 만들어주고 code-behind나 ViewModel로 기능을 연결 해줍니다.
Prism은 xaml에서 region을 먼저 잡아주고 button이 region에 주입(dependency injection)되어 동작합니다. 기능은 ViewModel에서 동작합니다.
버튼의 기능 하나를 구현하는데 필요한 지식이 기존에 WPF를 이용해서 코딩하는 것과는 차원이 다릅니다.
아마 이런 문제 때문에 대규모 프로젝트에 어울리고 소규모에는 적합하지 않다는 말이 나오는 것 같습니다. 하지만 이건 숙련도의 문제지 프로젝트의 규모와는 상관이 없다고 생각이 듭니다.
저의 주장은 같은 숙련도라면 Prism으로 작업하는게 더 유지보수도 좋고 생산성도 높다는 의견입니다.
말로하면 잘 이해가 안가죠?
코드로 보도록 합시다.
Infragistics의 component들을 이용하여 간단하게 RibbonWindow를 만들어 보겠습니다.
- 버전: vs2022 pro
- .net: .net Core 6.0
MainWindow.xaml
<igRibbon:XamRibbonWindow x:Class="prismTesting.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:igRibbon="http://infragistics.com/Ribbon"
Width="600" Height="400">
<Grid>
<igRibbon:RibbonWindowContentHost>
<igRibbon:RibbonWindowContentHost.Ribbon>
<igRibbon:XamRibbon>
<igRibbon:RibbonTabItem Header="Hello">
<igRibbon:RibbonGroup>
<igRibbon:ButtonTool Caption="world" />
</igRibbon:RibbonGroup>
</igRibbon:RibbonTabItem>
</igRibbon:XamRibbon>
</igRibbon:RibbonWindowContentHost.Ribbon>
</igRibbon:RibbonWindowContentHost>
</Grid>
</igRibbon:XamRibbonWindow>
기존 코드라면 이렇게 xaml을 구성하게 됩니다. 물론 여기서 ribbontabitem을 다른 xaml 파일로 구성하여 import 할 수 있습니다. 하지만 여전히 mainwindow.xaml에 dependency가 있습니다. 또 Ribbon의 button은 code-behind나 ViewModel에서 처리 할 수 있습니다.
다음은 Prism code 입니다.
MainWindow.xaml
<igRibbon:XamRibbonWindow x:Class="prismTesting.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:igRibbon="http://infragistics.com/Ribbon"
Width="600" Height="400">
<Grid>
<igRibbon:RibbonWindowContentHost>
<igRibbon:RibbonWindowContentHost.Ribbon>
<igRibbon:XamRibbon prism:RegionManager.RegionName="{x:Static core:RegionNames.RibbonRegion}" />
</igRibbon:RibbonWindowContentHost.Ribbon>
</igRibbon:RibbonWindowContentHost>
</Grid>
</igRibbon:XamRibbonWindow>
XamRibbon이라는 component를 Prism은 모르기 때문에 해당 component를 region으로 인식 시켜주어야 합니다. 여기서 Region adapter라는 파일을 하나 더 작성하게 됩니다.
XamRibbonRegionAdapter.cs
using Infragistics.Windows.Ribbon;
using Prism.Regions;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace prismTesting.Core
{
public class XamRibbonRegionAdapter : RegionAdapterBase<XamRibbon>
{
public XamRibbonRegionAdapter(IRegionBehaviorFactory regionBehaviorFactory) : base(regionBehaviorFactory)
{
}
protected override void Adapt(IRegion region, XamRibbon regionTarget)
{
region.Views.CollectionChanged += (s, e) =>
{
if(e.Action == System.Collections.Specialized.NotifyCollectionChangedAction.Add)
{
foreach (var view in e.NewItems)
{
AddToRegion(view, regionTarget);
}
}
if(e.Action == System.Collections.Specialized.NotifyCollectionChangedAction.Remove)
{
foreach (var view in e.OldItems)
{
RemoveFromRegion(view, regionTarget);
}
}
};
}
private void RemoveFromRegion(object view, XamRibbon regionTarget)
{
if (view is RibbonTabItem ribbonTabItem)
regionTarget.Tabs.Remove(ribbonTabItem);
}
private void AddToRegion(object view, XamRibbon regionTarget)
{
if (view is RibbonTabItem ribbonTabItem)
regionTarget.Tabs.Add(ribbonTabItem);
if (view is ApplicationMenu applicationMenu)
regionTarget.ApplicationMenu = applicationMenu;
}
protected override IRegion CreateRegion()
{
return new SingleActiveRegion();
}
}
}
이후 App.xaml.cs에서 XamRibbon은 XamRibbonRegionAdapter를 이용해서 region을 이용한다고 알려주어야 합니다.
protected override void ConfigureRegionAdapterMappings(RegionAdapterMappings regionAdapterMappings)
{
base.ConfigureRegionAdapterMappings(regionAdapterMappings);
regionAdapterMappings.RegisterMapping<XamRibbon, XamRibbonRegionAdapter>();
}
벌써 기존 WPF의 보다 작성해야 할 코드가 2개나 더 늘었습니다. 하지만 해당 XamRibbon이라는 component를 모듈화 한다면
protected override void ConfigureModuleCatalog(IModuleCatalog moduleCatalog)
{
moduleCatalog.AddModule<RibbonModule>();
}
단 한줄만으로 만들어둔 ribbon을 불러오고 region만 넣어주면 됩니다.
어쩌다보니 글이 길어 졌는데 한국에서 Prism 정보가 많이 공유 되었으면 하는 바람에 글을 길게 쓰게 되네요.
특히나 Prism은 MVVM을 더 좋게 만들어 주기 때문에 unit testing 또한 편리해집니다. view의 기능에 대해서도 testing 할 수 있습니다. 여러가지 쓸말들이 많지만 글이 너무 길어지니 다른 포스팅으로 대체하는것이 좋을 것 같습니다.