Microsoft MVP성태의 닷넷 이야기
WPF에서 로딩중 이미지를 구현 [링크 복사], [링크+제목 복사]
조회: 10059
글쓴 사람
우코아
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

안녕하세요.
WPF 개발 중 일주일간 풀리지 않는 문제에 봉착하여 간곡히 질문을 드립니다.

아래 링크와 같이 선생님께서 올려주신 글을 읽고 많은 도움이 되었습니다.
[https://www.sysnet.pe.kr/2/0/747]
어떤 문제 때문에 제 프로그램이 원하는대로 동작하지 않는건지 대략적인 내용을 알 수 있었습니다.
하지만, 스레드에 대한 우선순위로는 해결이 되지 않기에 불가능한 것인지, 몰라서 구현을 못하는 것인지 모르겠습니다.

위 링크와 같이 예제를 그대로 따라서 작성해 보았습니다.
설명해 주신 것 과 같이 Button 객체의 Enable은 원하는대로 동작을 합니다.
하지만, 본인이 원하는 것은 GIF 이미지와 TextBlock으로 '로딩중...'을 구현하고 싶었습니다.

'LoadingProcess.xaml'라는 Page를 하나 생성한 뒤, MainWindow에서 객체를아래와 같이 호출했습니다.
위 링크의 선생님의 예제 중 Button1 객체를 Enable하는 위치에 loadingProcess 객체를 생성하는 부분만 바꾸었습니다.

private void Button_Click(object sender, RoutedEventArgs e)
{
    loadingProcess = new LoadingProcess();
    loadingProcess.Top = this.Top + (this.ActualHeight - loadingProcess.Height) / 2;
    loadingProcess.Left = this.Left + (this.ActualWidth - loadingProcess.Width) / 2;
    loadingProcess.Show();
            
    this.Dispatcher.BeginInvoke((ThreadStart)(() => { }), DispatcherPriority.SystemIdle);

    Thread.Sleep(5000);

    loadingProcess.Close();
}


하지만, loadingProcess.xaml 페이지에서 생성한 GIF 이미지가 멈춰 있습니다.
다시 설명 드리자면, GIF 이미지가 멈춰있다기 보다는 스레드가 Sleep 되는 동안 팝업창도 같이 Sleep에 빠져있습니다.
새로운 팝업창도 블로킹 상태에 빠지게되어 GIF 이미지가 멈춰있는것 같습니다.

Sleep되는 영역에서는 사용자가 선택한 이미지를 WrapPanel에 add하는 로직이 들어갈 예정입니다.
하나씩 코드를 제거하면서 찾아보니, 이미지 로드의 문제가 아닌 스레드 작업중 화면이 블로킹 되는 것이 문제라고 판단하게 되었습니다.

위와 같은 상황에서 팝업된 화면이 정상적으로 랜더링(?) 될 수 있는 방법이 있을지요?
선생님의 소중한 답변 기다리겠으니, 저의 미숙한 질문이 부족하시다면 언제든지 덧글 부탁 드립니다.
정중하게 선생님의 의견을 여쭙습니다.


Ps.
추가로, GIF 이미지는 아래의 링크에서 설명한 대로 구현했습니다.
스레드 작업이 종료되면(Sleep 또는 Image 로딩 작업), GIF 이미지는 정상적으로 표현됩니다.
https://m.blog.naver.com/PostView.nhn?blogId=doksajokyo&logNo=221151321969&proxyReferer=http%3A%2F%2Fwww.google.com%2Furl%3Fsa%3Dt%26rct%3Dj%26q%3D%26esrc%3Ds%26source%3Dweb%26cd%3D4%26ved%3D2ahUKEwjV3t_ylsffAhVLu7wKHZ9KBAkQFjADegQIBxAB%26url%3Dhttp%253A%252F%252Fm.blog.naver.com%252Fdoksajokyo%252F221151321969%26usg%3DAOvVaw208IoMPIug52DGpiT3Cq4X

다른 질문자분과 동일한 상황입니다.
https://www.sysnet.pe.kr/Default.aspx?mode=3&sub=0&pageno=0&detail=1&wid=4882
선생님 답글도 잘 읽어보았습니다. 하지만, WPF에서는 Application.Run 메서드가 없어서 UI 스레드로 변경을 실패했습니다.
https://www.sysnet.pe.kr/2/0/11287




[연관 글]






[최초 등록일: ]
[최종 수정일: 1/2/2019]


비밀번호

댓글 작성자
 



2019-01-02 09시21분
스레드가 좀 어렵긴 한데, 간단하게 사람이라고 생각해 보세요. BeginInvoke는 람다 함수를 책상위의 서류 더미 위에 올려 놓는 것과 같습니다. 그다음 처리는 서류 더미 위에 올려 놓은 것을 처리하는 것이 아니라 지금 당장 실행해야 하는 코드인 Thread.Sleep을 처리하는 것입니다. 그리고 그 코드는 처리하는 사람에게 지정된 시간 동안 아무 것도 하지 말라고 하는 것과 같습니다.

5초가 지난 후에도 서류 더미 위에 쌓아두었던 작업을 하지 않습니다. Thread.Sleep 이후의 loadingProcess.Close를 호출하는 작업을 합니다. 그리고 그 모든 작업(즉, Button_Click 이벤트 내의 코드)을 수행하고 나서 서류 더미 위에 있던 작업들을 하나씩 차례대로... (남는 시간이므로) 처리하는 것입니다.

따라서 원하는 동작을 구현하려면 Loading 중이라는 메시지가 화면에 나오게 한 다음(그 방법을 747 글에서 설명한 것입니다.), GIF 로딩을 처리해야 합니다. 만약 GIF 로딩하는 사이 사용자 입력을 허용해야 한다면 GIF 로딩은 별도의 스레드에서 처리해야 합니다.
정성태
2019-01-02 12시04분
[우코아] 답변 감사합니다!
말씀주신 스레드의 컨셉은 이해가 되었습니다만, 추가로 질문이 더 있어서 덧글 드립니다.
Thread.Sleep(5000) 호출 대신, 아래와 같은 이미지를 로드하는 소스를 넣어도 결과는 마찬가지 입니다.
혹시, 이미지를 로딩하는 우선순위가 Thread.Sleep과 동일한 우선순위로 처리가 되는건지 궁금합니다.


DirectoryInfo di = new DirectoryInfo(@path);
String fileFilters = "*.jpg|*.jpeg|*.png|*.gif|*.tiff|*.bmp";
String[] files = fileFilters.Split('|').SelectMany(searchPattern => Directory.GetFiles(@path, searchPattern)).ToArray();

for (int i = 0; i < files.Length; i++)
{
    BitmapImage bitmapImage = new BitmapImage();
    bitmapImage.BeginInit();
    bitmapImage.UriSource = new Uri(@files[i], UriKind.Absolute);
    bitmapImage.DecodePixelWidth = 120;
    bitmapImage.CacheOption = BitmapCacheOption.OnLoad;
    bitmapImage.EndInit();

    Image image = new Image();
    image.Stretch = Stretch.Uniform;
    image.Source = bitmapImage;


    Border border = new Border();
    border.BorderThickness = new Thickness(1);
    border.BorderBrush = new SolidColorBrush(Color.FromRgb(41, 57, 86));
    border.Margin = new Thickness(2, 2, 2, 2);
    border.Width = imageWidth;
    border.Height = imageWidth;
    border.Child = image;

    TextBlock imageName = new TextBlock();
    imageName.Text = Path.GetFileName(files[i]);
    imageName.VerticalAlignment = VerticalAlignment.Center;
    imageName.HorizontalAlignment = HorizontalAlignment.Center;
    imageName.Margin = new Thickness(0, 0, 0, 10);
    imageName.MaxWidth = imageWidth - 4;

    DockPanel dockPanel = new DockPanel();
    DockPanel.SetDock(border, Dock.Top);
    DockPanel.SetDock(imageName, Dock.Bottom);
    dockPanel.Children.Add(border);
    dockPanel.Children.Add(imageName);

    WrapPanel_Images.Children.Add(dockPanel);
}
[guest]
2019-01-03 02시30분
[우코아] 위의 소스만 수행하거나 또는 Thread로 수행해도 마찬가지로, 메인창 과 팝업된 창 모두 UI 블록 상태에 빠집니다.
어쩔수 없는 경우인지... 해결이 가능하긴 한건지 궁금합니다 ㅠㅠ..
[guest]
2019-01-03 09시50분
음... ^^ 여전히 스레드에 대해 이해가 안 된 것 같은데요. Thread.Sleep을 하면 해당 스레드는 지정한 시간 동안 아무런 처리를 하지 않습니다. 이건 우선 순위의 문제가 아닙니다. 반면, Sleep이 아닌 코드를 실행하도록 만들었다면 스레드는 그냥 그 작업을 처리하는 것뿐입니다. 지금 눈 앞에 있는 코드를 다 처리하고 난 이후에야 Dispatcher에 쌓여 있는 작업을 우선 순위에 따라 진행하는 것입니다.

도저히 모르시겠다면, 문제를 최대한 단순화시킨 예제 프로젝트와 함께 질문/답변 게시판에 다시 올려주시겠어요?
정성태

1  2  3  4  5  6  7  8  [9]  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
5726양승조9/22/20224753C# dll 과 C++ 간 배열 전달. SafeArray [10]파일 다운로드1
5725김기헌9/21/20223653안녕하세요 선생님 윈폼 컨트롤 Dispose 관련 질문드립니다 [2]
5724감사합니...9/19/20223682스레드와 스레드 안전한 객체 사용관련 문의드립니다. [5]
5723드리렁9/13/20223611Pinned Object에 대해서 질문이 있습니다. [2]
5722김인태9/8/20223773대화상자에서 alt + tab 후킹 작업 [1]
5721우종9/7/20223917C++ DLL 과 C# 연동 문의 [2]
5720한예지 donator9/6/20223648학습 방법 질문 있습니다. [7]
5719김경한9/6/20223979안녕하세요 질문하나만드리겠습니다...! [10]
5718김민아9/2/20224023안녕하세요 생성자 호출 시 초기화 순서 질문드립니다 [2]
5716iili...8/26/20224214WinDbg 커널 디버깅에서의 thread freeze [2]
5715에릭8/19/20224632WMI 쿼리 결과값이 Windows Service와 Console 출력에서 상이한 이유가 있을까요? [9]파일 다운로드1
5714허니빠8/18/20224664.net6 hint path 를 프로젝트 단위로 지정할 수 있는 방법을 알고싶습니다 [8]
5713김기헌8/17/20224590안녕하세요 rgb 계산 오차가 있는데 원인을 모르겠습니다.. [3]
5712하태8/17/20224034안녕하세요 background service에서 user32dll 접근 질문 드리겠습니다.! [2]
5711하태8/16/20223554안녕하세요! 윈도우즈 해상도 관련 질문 드립니다. [1]
5710장성욱8/12/20223771c# 시리얼 통신 관련 질문 [3]
5709초보8/12/20223582WPF 커맨드 관련 질문 [2]
5708민성8/11/20223844안녕하세요 c#에서 화면의 배율 및 레이아웃을 변경할려면 어떻게 해야 할까요? [2]파일 다운로드1
5707민성8/10/20223476WPF 엣지 컨트롤에서 화면이 안보이는 현상 [2]파일 다운로드1
5706종규8/7/20225872WPF 에서 SVG 아이콘 사용 방법 문의 [2]
5705김기헌8/6/20224595안녕하세요 선생님 싱글톤 패턴 간단 질문 [2]
5704따봉이8/4/20224469EventHandler 관련 [1]
5703조민준8/3/20223894안녕하세요 정적 멤버 초기화 관련 간단한 질문 [2]
5702석우8/3/20223822C# WPF Ribbon IsMinimized 프로퍼티 변경 감지 방법문의 [2]파일 다운로드1
5701김기헌8/1/20224033foreach 문으로 컬렉션을 열거할 때 궁금한 점 [2]
5699갓경섭7/29/20224441C# 프로그램 느려짐 현상 [2]
1  2  3  4  5  6  7  8  [9]  10  11  12  13  14  15  ...