Microsoft MVP성태의 닷넷 이야기
COM 개체 관련: 13. 비동기 Drag & Drop 구현 : IAsyncOperation [링크 복사], [링크+제목 복사],
조회: 21568
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

보통 Drag & Drop을 구현하면 동기방식으로 구현을 하게 되지요. 반면에, 탐색기를 살펴보게 되면 비동기로 파일 이동/복사가 되는 것을 확인할 수 있습니다.

해당 기능을 구현하기 위한 관련 토픽은 다음과 같습니다.

http://msdn.microsoft.com/library/default.asp?url=/library/en-us/shellcc/platform/shell/programmersguide/shell_basics/shell_basics_programming/transferring/datascenarios.asp

예제 코드만 없을 뿐... 위의 문서대로 하시면 어렵지 않게 구현하실 수 있습니다.
하지만, 역시 스레드 사용으로 인해 COM 개체와 연관되는 경우라면 마샬링을 해야 하는 불편함이 따르게 되는군요.

암튼.... 순서는 이렇습니다.

Source 측에서 먼저 시작.

1. 구현하신 IDataObject 인터페이스 클래스에 IAsyncOperation을 상속받으세요.
2. 물론, IDataObject의 QueryInterface에서 IAsyncOperation에 대한 처리도
추가하시고.
3. DoDragDrop 하기 전에, IDataObject 인스턴스의 SetAsyncMode(TRUE);를
호출.
4. SetAsyncMode는 다음과 같이 간단하게 구현.
    virtual HRESULT STDMETHODCALLTYPE SetAsyncMode(
        /* [in] */ BOOL fDoOpAsync)
 {
  m_bAsyncInOperation = FALSE;
  m_bAsyncMode = fDoOpAsync;
  return S_OK;
 }
5. DoDragDrop 호출.

이젠 제어권이 Drop 쪽으로 넘어갑니다.

6. DragEnter에서 pDataObject 인자를 보관.
7. OnDrop에서
 CComQIPtr<IAsyncOperation> pAsync = m_pDataObject;
 if ( pAsync != NULL )
 {
  BOOL bAsync = FALSE;
  pAsync->GetAsyncMode( &bAsync );

  if ( bAsync == TRUE )
  {
   pAsync->StartOperation( NULL ); // 이 부분에서 순간 긴장. 인자형식이
IBindCtx인데, 다행히 아직 미구현. 무조건 NULL
   ThreadStart(); // 스레드 시작.
   return TRUE;
  }
 }

비동기 지원함을 알게 된 Drop 쪽에서 스레드 시작하고 바로 리턴.

다시 제어권이 Source 쪽으로 넘어갑니다. 즉, DoDragDrop을 호출했던 바로 다음
라인으로 가겠죠.
동기 처리인 경우, DoDragDrop 호출 후 IDataObject와 IDragSource를 바로
Release 할 텐데.
비동기인 경우에는 그래선 안되죠.

8. 인터페이스 해제 보류
       BOOL bDoing;
       pDragObj->InOperation( &bDoing );
       if ( bDoing == TRUE )
       {
       } else {
        pDragSrc->Release();
        pDragObj->Release();
       }

위와 같은 형식으로 처리를 하고 리턴해야 합니다.
그럼, 언제 IDataObject와 IDragSouce를 해제해 주느냐?
다시 순서를 ThreadStart 쪽으로 옮겨 보겠습니다.

9. 스레드에서.
void Threadstart()
{
   ; 데이터 추출 호출 측에서 pDragSrc와 pDataObj에 대해서 아직 Release를
안했으므로,
   ; 여전히 DragEnter에서 받아들였던 IDataObject 인스턴스는 유효하게
됩니다.
   // 데이터 추출

   ; 마지막에 반드시 호출
    pAsync->EndOperation( S_OK, NULL, 0 );
}


다시... 호출 측의 IDataObject로 가서. IAsyncOperation::EndOperation이
호출되어져서.
10. EndOperation에서 IDataSource와 IDataObject를 Release해 줍니다.

비동기 처리가 항상 그렇듯이... 실제 구현하면서 처리해야 할 것들이 아마 제법
늘어날 것입니다.








[최초 등록일: ]
[최종 수정일: 7/5/2021]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 31  32  33  34  35  36  37  38  39  40  41  [42]  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
12891정성태12/23/202115007스크립트: 38. 파이썬 - uwsgi의 --master 옵션
12890정성태12/23/202115261VC++: 152. Golang - (문자가 아닌) 바이트 위치를 반환하는 strings.IndexRune 함수
12889정성태12/22/202118067.NET Framework: 1123. C# - (SharpDX + DXGI) 화면 캡처한 이미지를 빠르게 JPG로 변환하는 방법파일 다운로드1
12888정성태12/21/202115172.NET Framework: 1122. C# - ImageCodecInfo 사용 시 System.Drawing.Image와 System.Drawing.Bitmap에 따른 Save 성능 차이파일 다운로드1
12887정성태12/21/202118731오류 유형: 777. OpenCVSharp4를 사용한 프로그램 실행 시 "The type initializer for 'OpenCvSharp.Internal.NativeMethods' threw an exception." 예외 발생
12886정성태12/20/202114749스크립트: 37. 파이썬 - uwsgi의 --enable-threads 옵션 [2]
12885정성태12/20/202115824오류 유형: 776. uwsgi-plugin-python3 환경에서 MySQLdb 사용 환경
12884정성태12/20/202114738개발 환경 구성: 620. Windows 10+에서 WMI root/Microsoft/Windows/WindowsUpdate 네임스페이스 제거
12883정성태12/19/202115147오류 유형: 775. uwsgi-plugin-python3 환경에서 "ModuleNotFoundError: No module named 'django'" 오류 발생
12882정성태12/18/202114653개발 환경 구성: 619. Windows Server에서 WSL을 위한 리눅스 배포본을 설치하는 방법
12881정성태12/17/202114223개발 환경 구성: 618. WSL Ubuntu 20.04에서 파이썬을 위한 uwsgi 설치 방법 (2)
12880정성태12/16/202115283VS.NET IDE: 170. Visual Studio에서 .NET Core/5+ 역어셈블 소스코드 확인하는 방법
12879정성태12/16/202121772오류 유형: 774. Windows Server 2022 + docker desktop 설치 시 WSL 2로 선택한 경우 "Failed to deploy distro docker-desktop to ..." 오류 발생
12878정성태12/15/202116007개발 환경 구성: 617. 윈도우 WSL 환경에서 같은 종류의 리눅스를 다중으로 설치하는 방법
12877정성태12/15/202115450스크립트: 36. 파이썬 - pymysql 기본 예제 코드
12876정성태12/14/202115308개발 환경 구성: 616. Custom Sources를 이용한 Azure Monitor Metric 만들기
12875정성태12/13/202114099스크립트: 35. python - time.sleep(...) 호출 시 hang이 걸리는 듯한 문제
12874정성태12/13/202113924오류 유형: 773. shell script 실행 시 "$'\r': command not found" 오류
12873정성태12/12/202115318오류 유형: 772. 리눅스 - PATH에 등록했는데도 "command not found"가 나온다면?
12872정성태12/12/202115716개발 환경 구성: 615. GoLang과 Python 빌드가 모두 가능한 docker 이미지 만들기
12871정성태12/12/202114735오류 유형: 771. docker: Error response from daemon: OCI runtime create failed
12870정성태12/9/202113878개발 환경 구성: 614. 파이썬 - PyPI 패키지 만들기 (4) package_data 옵션
12869정성태12/8/202116544개발 환경 구성: 613. git clone 실행 시 fingerprint 묻는 단계를 생략하는 방법
12868정성태12/7/202114937오류 유형: 770. twine 업로드 시 "HTTPError: 400 Bad Request ..." 오류 [1]
12867정성태12/7/202114707개발 환경 구성: 612. 파이썬 - PyPI 패키지 만들기 (3) entry_points 옵션
12866정성태12/7/202121647오류 유형: 769. "docker build ..." 시 "failed to solve with frontend dockerfile.v0: failed to read dockerfile ..." 오류
... 31  32  33  34  35  36  37  38  39  40  41  [42]  43  44  45  ...