첨부 실행 코드는 나눔고딕코딩 폰트를 사용합니다.
유용한 소스 코드가 있으면 icodebroker@naver.com으로 보내주시면 감사합니다.
블로그 자료는 자유롭게 사용하세요.

728x90
반응형

■ Image 클래스 : 모니터별 DPI 기능 사용하기

------------------------------------------------------------------------------------------------------------------------


TestProject.zip


ImageDPIHelper.cs

 

 

using System;

using System.Text.RegularExpressions;

using System.Windows;

using System.Windows.Controls;

using System.Windows.Media;

using System.Windows.Media.Imaging;

 

namespace TestProject

{

    /// <summary>

    /// 이미지 DPI 헬퍼

    /// </summary>

    public static class ImageDPIHelper

    {

        //////////////////////////////////////////////////////////////////////////////////////////////////// Method

        ////////////////////////////////////////////////////////////////////////////////////////// Static

        //////////////////////////////////////////////////////////////////////////////// Public

 

        #region 최적 스케일 구하기 - GetBestScale(pixelCountPerDIP)

 

        /// <summary>

        /// 최적 스케일 구하기

        /// </summary>

        /// <param name="pixelCountPerDIP">DIP당 픽셀 카운트</param>

        /// <returns>최적 스케일</returns>

        public static int GetBestScale(double pixelCountPerDIP)

        {

            int currentScale = (int)(pixelCountPerDIP * 100);

 

            int bestScale;

 

            if(currentScale > 200)

            {

                bestScale = 400;

            }

            else if(currentScale > 100)

            {

                bestScale = 200;

            }

            else

            {

                bestScale = 100;

            }

 

            return bestScale;

        }

 

        #endregion

        #region 희망 이미지 URL 구하기 - GetDesiredImageURL(image)

 

        /// <summary>

        /// 희망 이미지 URL 구하기

        /// </summary>

        /// <param name="image">이미지</param>

        /// <returns>희망 이미지 URL</returns>

        public static string GetDesiredImageURL(Image image)

        {            

            DpiScale dpiScale = VisualTreeHelper.GetDpi(image);

 

            int bestScale = ImageDPIHelper.GetBestScale(dpiScale.PixelsPerDip);

 

            string sourceURL = image.Source.ToString();

 

            string imagePattern = Regex.Replace(sourceURL, ".scale-[0-9]{3}.", ".scale-{0}.");

 

            string newImagePath = null;

 

            if(imagePattern != null)

            {

                newImagePath = string.Format(imagePattern, bestScale);

            }

 

            return newImagePath;

        }

 

        #endregion

        #region 이미지 업데이트하기 - UpdateImage(image, uriString)

 

        /// <summary>

        /// 이미지 업데이트하기

        /// </summary>

        /// <param name="image">이미지</param>

        /// <param name="uriString">URI 문자열</param>

        public static void UpdateImage(Image image, string uriString)

        {

            Uri uri = new Uri(uriString, UriKind.RelativeOrAbsolute);

 

            BitmapImage bitmapImage = new BitmapImage(uri);

 

            image.Source = bitmapImage;

        }

 

        #endregion

    }

}

 

 

DPIAwareImage.cs

 

 

using System;

using System.Windows;

using System.Windows.Controls;

using System.Windows.Media;

using System.Windows.Media.Imaging;

 

namespace TestProject

{

    /// <summary>

    /// DPI 인식 이미지

    /// </summary>

    public class DPIAwareImage : Image

    {

        //////////////////////////////////////////////////////////////////////////////////////////////////// Field

        ////////////////////////////////////////////////////////////////////////////////////////// Private

 

        #region Field

 

        /// <summary>

        /// 최적 스케일

        /// </summary>

        private double bestScale;

 

        #endregion

 

        //////////////////////////////////////////////////////////////////////////////////////////////////// Constructor

        ////////////////////////////////////////////////////////////////////////////////////////// Public

 

        #region 생성자 - DPIAwareImage()

 

        /// <summary>

        /// 생성자

        /// </summary>

        public DPIAwareImage() : base()

        {

            Initialized += Image_Initialized;

        }

 

        #endregion

 

        //////////////////////////////////////////////////////////////////////////////////////////////////// Method

        ////////////////////////////////////////////////////////////////////////////////////////// Protected

        //////////////////////////////////////////////////////////////////////////////// Function

 

        #region DPI 변경시 처리하기 - OnDpiChanged(oldDPI, newDPI)

 

        /// <summary>

        /// DPI 변경시 처리하기

        /// </summary>

        /// <param name="oldDPI">구 DPI</param>

        /// <param name="newDPI">신 DPI</param>

        protected override void OnDpiChanged(DpiScale oldDPI, DpiScale newDPI)

        {

            ScaleImage(newDPI);

        }

 

        #endregion

 

        ////////////////////////////////////////////////////////////////////////////////////////// Private

        //////////////////////////////////////////////////////////////////////////////// Event

 

        #region 이미지 초기화시 처리하기 - Image_Initialized(sender, e)

 

        /// <summary>

        /// 이미지 초기화시 처리하기

        /// </summary>

        /// <param name="sender">이벤트 발생자</param>

        /// <param name="e">이벤트 인자</param>

        private void Image_Initialized(object sender, EventArgs e)

        {

            DpiScale dpiScale = VisualTreeHelper.GetDpi(sender as Visual);

 

            ScaleImage(dpiScale);

        }

 

        #endregion

        #region 비트맵 이미지 다운로드 완료시 처리하기 - bitmapImage_DownloadCompleted(sender, e)

 

        /// <summary>

        /// 비트맵 이미지 다운로드 완료시 처리하기

        /// </summary>

        /// <param name="sender">이벤트 발생자</param>

        /// <param name="e">이벤트 인자</param>

        private void bitmapImage_DownloadCompleted(object sender, EventArgs e)

        {

            BitmapImage bitmapImage = sender as BitmapImage;

 

            SetImageSize(bitmapImage);

 

            bitmapImage.DownloadCompleted -= bitmapImage_DownloadCompleted;

        }

 

        #endregion

 

        //////////////////////////////////////////////////////////////////////////////// Function

 

        #region 이미지 크기 설정하기 - SetImageSize(bitmapImage)

 

        /// <summary>

        /// 이미지 크기 설정하기

        /// </summary>

        /// <param name="bitmapImage">비트맵 이미지</param>

        private void SetImageSize(BitmapImage bitmapImage)

        {

            SetCurrentValue(Image.WidthProperty , bitmapImage.PixelWidth  * 100 / this.bestScale);

            SetCurrentValue(Image.HeightProperty, bitmapImage.PixelHeight * 100 / this.bestScale);

        }

 

        #endregion

        #region 이미지 업데이트하기 - UpdateImage(image, uriString)

 

        /// <summary>

        /// 이미지 업데이트하기

        /// </summary>

        /// <param name="image">이미지</param>

        /// <param name="uriString">URI 문자열</param>

        private void UpdateImage(Image image, string uriString)

        {

            Uri uri = new Uri(uriString, UriKind.RelativeOrAbsolute);

 

            BitmapImage bitmapImage = new BitmapImage();

            

            bitmapImage.BeginInit();

 

            bitmapImage.UriSource = uri;

 

            bitmapImage.EndInit();

 

            image.Source = bitmapImage;

 

            if(!bitmapImage.IsDownloading)

            {

                SetImageSize(bitmapImage);

            }

            else

            {

                bitmapImage.DownloadCompleted += bitmapImage_DownloadCompleted;

            }

        }

 

        #endregion

        #region 이미지 스케일 적용하기 - ScaleImage(dpiScale)

 

        /// <summary>

        /// 이미지 스케일 적용하기

        /// </summary>

        /// <param name="dpiScale">DPI 스케일</param>

        private void ScaleImage(DpiScale dpiScale)

        {

            this.bestScale = ImageDPIHelper.GetBestScale(dpiScale.PixelsPerDip);

 

            string imageURL = ImageDPIHelper.GetDesiredImageURL(this);

 

            UpdateImage(this, imageURL);

        }

 

        #endregion

    }

}

 

 

MainWindow.xaml

 

 

<Window x:Class="TestProject.MainWindow"

    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"

    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"

    xmlns:local="clr-namespace:TestProject"

    Width="800"

    Height="600"

    Title="Image 클래스 : 모니터별 DPI 기능 사용하기"

    FontFamily="나눔고딕코딩"

    FontSize="16">

    <Grid Margin="10">

        <Grid.RowDefinitions>

            <RowDefinition Height="Auto" />

            <RowDefinition Height="*"    />

            <RowDefinition Height="Auto" />

            <RowDefinition Height="*"    />

            <RowDefinition Height="Auto" />

        </Grid.RowDefinitions>

        <TextBlock Grid.Row="0"

            FontSize="18"

            FontWeight="Bold"

            Text="이미지" />

        <Image Name="image" Grid.Row="1"

            Source="IMAGE\logo.scale-100.png"

            Width="524"

            Height="193" />

        <TextBlock Grid.Row="2"

            FontSize="18"

            FontWeight="Bold"

            Text="DPI 인식 이미지" />

        <local:DPIAwareImage Grid.Row="3"

            Source="IMAGE\logo.scale-100.png" />

        <TextBlock Name="imageTextBlock" Grid.Row="4"

            Text="IMAGE\logo.scale-100.png" />

    </Grid>

</Window>

 

 

MainWindow.xaml.cs

 

 

using System.Windows;

using System.Windows.Controls;

using System.Windows.Media;

 

namespace TestProject

{

    /// <summary>

    /// 메인 윈도우

    /// </summary>

    public partial class MainWindow : Window

    {

        //////////////////////////////////////////////////////////////////////////////////////////////////// Constructor

        ////////////////////////////////////////////////////////////////////////////////////////// Public

 

        #region 생성자 - MainWindow()

 

        /// <summary>

        /// 생성자

        /// </summary>

        public MainWindow()

        {

            InitializeComponent();

 

            Loaded                += Window_Loaded;

            this.image.DpiChanged += image_DpiChanged;

        }

 

        #endregion

 

        //////////////////////////////////////////////////////////////////////////////////////////////////// Method

        ////////////////////////////////////////////////////////////////////////////////////////// Private

        //////////////////////////////////////////////////////////////////////////////// Event

 

        #region 윈도우 로드시 처리하기 - Window_Loaded(sender, e)

 

        /// <summary>

        /// 윈도우 로드시 처리하기

        /// </summary>

        /// <param name="sender">이벤트 발생자</param>

        /// <param name="e">이벤트 인자</param>

        private void Window_Loaded(object sender, RoutedEventArgs e)

        {

            UpdateImage(this.image);

        }

 

        #endregion

        #region 이미지 DPI 변경시 처리하기 - image_DpiChanged(sender, e)

 

        /// <summary>

        /// 이미지 DPI 변경시 처리하기

        /// </summary>

        /// <param name="sender">이벤트 발생자</param>

        /// <param name="e">이벤트 인자</param>

        private void image_DpiChanged(object sender, RoutedEventArgs e)

        {

            Image image = sender as Image;

 

            UpdateImage(image);

        }

 

        #endregion

 

        //////////////////////////////////////////////////////////////////////////////// Function

 

        #region 이미지 업데이트하기 - UpdateImage(image)

 

        /// <summary>

        /// 이미지 업데이트하기

        /// </summary>

        /// <param name="image">이미지</param>

        private void UpdateImage(Image image)

        {

            string newImageURL = ImageDPIHelper.GetDesiredImageURL(image);

 

            ImageDPIHelper.UpdateImage(image, newImageURL);

 

            this.imageTextBlock.Text = $"이미지 : {newImageURL}, 화면 DPI : {(VisualTreeHelper.GetDpi(image).PixelsPerDip * 100)}";

        }

 

        #endregion

    }

}

 

------------------------------------------------------------------------------------------------------------------------

728x90
반응형
Posted by 사용자 icodebroker
TAG , , , , ,

댓글을 달아 주세요