Microsoft MVP성태의 닷넷 이야기
WPF에서 로딩중 이미지를 구현 [링크 복사], [링크+제목 복사]
조회: 1531
글쓴 사람
우코아
홈페이지
첨부 파일
 
안녕하세요.
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);
}

                                         
[손님]
2019-01-03 02시30분
[우코아] 위의 소스만 수행하거나 또는 Thread로 수행해도 마찬가지로, 메인창 과 팝업된 창 모두 UI 블록 상태에 빠집니다.
어쩔수 없는 경우인지... 해결이 가능하긴 한건지 궁금합니다 ㅠㅠ..
[손님]
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)
4878heyhey8/25/20173195프로세스의 프로세스 찾기(?) [2]
4877강준8/24/20172666SQLite journal_mode=wal 관련하여 질문드립니다. [1]
4876heyhey8/23/20172970프로세스 초기화하기 [1]
4874ho8/22/20173112파일 확장자명을 이용해 파일의 실행 프로그램의 전체 경로를 얻어 올 수 있을까요? [1]
4875ho8/23/20173011    답변글 [답변]: 파일 확장자명을 이용해 파일의 실행 프로그램의 전체 경로를 얻어 올 수 있을까요? [1]
4873kmi8/21/20173523전역 변수를 쓰지 않고 여러 군데에서 같은 변수를 공용하는 방법이 궁금합니다. [4]
4872abcd8/18/20173203프로세스를 초기화 하는 명령어도 있나요? [1]
4871kmi8/17/20173746메모리 부족으로 종료되는 현상의 여러가지 이유가 무엇인지 궁금합니다. [2]
4870heyhey8/14/20173394프로그램 실행시 중복일 때 버튼 색깔 원래대로 돌리기 [1]
4868kmi8/4/20173020string[] 에 Reverse 적용방법 질문해봅니다 [3]
4867heyhey8/4/20172920EventHandler에 관한 [1]
486610년차8/3/20172902dsoframer axframer open시 기존 오픈되어있는 엑셀을 먹어버리는 현상 [1]
4865heyhey7/31/20173619클릭원스로 배포 한 프로젝트가 끝났는지 알 수 있는 방법 [8]
4864초보자7/28/20173215DllIImport질문 드립니다. [1]
4863다연아빠7/23/20172938전역 예외처리에 대해 질문있습니다. [3]
4861라르크7/17/20174426window form 예제 따라하는 중인데 12.3 서비스 응용 프로그램에서 진행이 안됩니다. [3]파일 다운로드1
4859heyhey7/10/20172849다른 환경에서 실행하기 [1]
4858heyhey7/10/20173482Clickonce update에 관한질문입니다. [1]
4857heyhey7/7/20173549제가 여태까지 작성한 보고서입니다. [2]파일 다운로드1
4856heyhey7/6/20173105성태님 다른질문입니다. [4]
4855JP7/6/20173182Dispose 패턴 구현시 Finalize 재정의에 대한 질문드립니다. [2]
4854heyhey7/6/20172768
4853heyhey7/5/20173358성태님이 작성한대로 해봤습니다. [1]파일 다운로드1
4852김레오7/4/20174061서드파티 dll 디버깅에 대해 질문드립니다. [2]
4851김현준7/3/20173791Datagridview VirtualMode 시 GC가 계속 호출되는 현상이 이해가 안갑니다. [2]
4850heyhey7/3/20173632성태님 밑에 질문드렸던 오류입니다. [1]파일 다운로드1
1  2  3  4  5  6  7  8  9  10  11  12  13  14  [15]  ...