Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 3개 있습니다.)
웹: 34. Edge 브라우저도 지원하는 클립보드 복사를 위한 자바스크립트 코드
; https://www.sysnet.pe.kr/2/0/11093

.NET Framework: 841. Windows Forms/C# - 클립보드에 RTF 텍스트를 복사 및 확인하는 방법
; https://www.sysnet.pe.kr/2/0/11929

.NET Framework: 858. C#/Windows - Clipboard(Ctrl+C, Ctrl+V)가 동작하지 않는다면?
; https://www.sysnet.pe.kr/2/0/12009




C#/Windows - Clipboard(Ctrl+C, Ctrl+V)가 동작하지 않는다면?

ctrl+c, ctrl+v가 어느 순간부터 동작하지 않았을 때, 다음과 같이 명령행에서 clip을 사용해 보면 "ERROR: Access is denied"인 경우를 볼 수 있습니다.

c:\temp> dir | clip
ERROR: Access is denied.

저 오류 메시지는 권한이 부족해서 발생하는 것이 아니고 clipboard 자원이 현재 다른 프로세스에 의해 잠겨 있기 때문일 수 있습니다. 실제로 한번 재현을 해볼까요? ^^

C#으로 다음과 같이 간단하게 프로그램을 만들고,

[DllImport("user32.dll", SetLastError = true)]
static extern bool CloseClipboard();

[DllImport("user32.dll", SetLastError = true)]
static extern bool OpenClipboard(IntPtr hWndNewOwner);

bool _opened = false;

void Button1_Click(object sender, EventArgs e)
{
    _opened = OpenClipboard(this.Handle);
    this.Text = _opened.ToString();
}

void Button2_Click(object sender, EventArgs e)
{
    if (_opened == true)
    {
        CloseClipboard();
    }
}

Button1만 눌러 Clipboard 자원을 Open 한 다음, 명령행에서 clip을 실행하면 "ERROR: Access is denied." 오류가 그대로 재현되는 것을 볼 수 있습니다.




그래서, ctrl+c/ctrl+v가 동작하지 않는 경우 Clipboard 자원을 Open만 하고 Close를 못 시킨 응용 프로그램을 종료하면 다시 정상적으로 동작하게 됩니다. 문제는? 도대체 어떤 프로그램이 소유하고 있는지 알 수 없으므로 현재 실행 중인 프로세스들을 임의로 모두 종료하는 식으로 대응해야 합니다.

명색이 프로그래머인데 ^^ 그러면 너무 없어 보이므로 간단하게 코드로,

How to check which application has the clipboard hold?
; https://superuser.com/questions/770476/how-to-check-which-application-has-the-clipboard-hold

작성해 명확하게 알아내는 것도 가능합니다.

[DllImport("user32.dll")]
static extern IntPtr GetOpenClipboardWindow();

[DllImport("user32.dll", SetLastError = true)]
static extern uint GetWindowThreadProcessId(IntPtr hWnd, out uint lpdwProcessId);

...[생략]...
{
    IntPtr pClipboardOwner = GetOpenClipboardWindow();

    if (pClipboardOwner == IntPtr.Zero)
    {
        return;
    }

    GetWindowThreadProcessId(pClipboardOwner, out uint dwProcessId);
    Process owner = Process.GetProcessById((int)dwProcessId);
    string name = owner.ProcessName;
    string mainModule = owner.MainModule.FileName;
}

일단 GetWindowThreadProcessId Win32 API까지의 호출은 문제없는데, 이후의 Process 타입을 통해 정보를 알아오는 경우 호출 측과 대상 프로세스의 플랫폼 차이(x86/x64)가 발생하면 오류가 뜹니다. 이런 경우 해당 코드만 실행해 그 결과를 반환하는 간단한 EXE를 x86과 x64로 각각 빌드해 실행하는 것도 가능하겠지만, 그보다는 간단하게 WMI 쿼리를 사용하는 것도 나쁘지 않습니다.

{
    // ...[생략]...
    GetWindowThreadProcessId(pClipboardOwner, out uint dwProcessId);

    textBox1.Text = $"{dwProcessId}, " + GetProcessInfo(dwProcessId);
}

string GetProcessInfo(uint processId)
{
    string arg = $"process where \"ProcessID={processId}\" get ExecutablePath";
    return GetOutput("wmic", arg);
}

private static string GetOutput(string executable, string arguments)
{
    ProcessStartInfo psi = new ProcessStartInfo(); // wmic를 사용하지 않고 System.Management.dll을 이용해 결과를 반환하는 것도 가능
    psi.FileName = executable;
    psi.Arguments = arguments;
    psi.RedirectStandardOutput = true;
    psi.UseShellExecute = false;
    psi.CreateNoWindow = true;

    Process proc = Process.Start(psi);
    string txt = proc.StandardOutput.ReadToEnd();
    return txt;
}




참고로, OpenClipboard에 윈도우 핸들 값이 아닌 null을 주는 것도 가능합니다.

OpenClipboard(IntPtr.Zero);

위와 같은 방식으로 클립보드를 잠근 경우에는 다른 프로세스에서라도 CloseClipboard를 호출하면 풀리게 됩니다. 물론, 윈도우 핸들을 넘긴 경우에는 무조건 그 윈도우를 소유한 프로세스에서 CloseClipboard를 호출해야 합니다. (또는 종료하든가.)

첨부 파일은 이 글의 소스 코드와 실행 파일(WindowsFormsApp1.exe)을 담고 있습니다.




(2024-04-12 업데이트)
How can I find out which process has locked me out of the clipboard?
; https://devblogs.microsoft.com/oldnewthing/20240410-00/?p=109632

위의 글에 보면, Get­Clipboard­Owner와 Get­Open­Clipboard­Window의 차이점을 설명하고 있습니다. 간단하게 정리하면 Owner는 EmptyClipboard API를 호출하는 시점에 (그 전에 OpenClipboard를 호출하면서 전달한 Hwnd 값이) Owner로 결정됩니다. 사실 OpenClipboard에 NULL 전달도 가능하므로 owner가 없는 것으로 나오는 경우도 가능하지만, 그런 경우에는 SetClipboardData 호출에 실패하므로 현실적으로 빈번하게 발생하지는 않을 것입니다.

반면 Get­Open­Clipboard­Window는 OpenClipboard를 호출했을 때 전달한 Hwnd 값을 반환하고, 실제로 그 윈도우가 클립보드에 대한 잠금을 하게 됩니다. 재미있는 건, 이게 전역적인 데이터를 접근하는 것이기 때문에 race condition이 발생할 수 있다는 점입니다. 그래서 이미 점유된 클립보드에 대해 OpenClipboard를 호출해 실패하는 경우, 어떤 윈도우가 클립보드를 소유하고 있는지 알아내기 위해 Get­Open­Clipboard­Window를 호출했을 때 null이 나올 수 있습니다. 왜냐하면, 하필 클립보드를 잠갔던 곳에서 CloseClipboard를 하기 바로 전에 여러분의 코드에서 OpenClipboard를 했다면 그것을 알아내기 위해 호출한 Get­Open­Clipboard­Window 시점에는 null이 반환될 수 있기 때문입니다.




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 4/12/2024]

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

비밀번호

댓글 작성자
 




... 91  92  93  94  95  96  97  98  99  [100]  101  102  103  104  105  ...
NoWriterDateCnt.TitleFile(s)
11429정성태1/9/201818444개발 환경 구성: 348. ASP.NET Core 2.1 Preview 버전 적용 방법
11428정성태1/6/201821220개발 환경 구성: 347. WinForm 프로젝트를 WPF 프로젝트 유형으로 변경하는 방법파일 다운로드1
11427정성태1/5/201819236오류 유형: 445. vcpkg 빌드 오류 - Starting the CLR failed with HRESULT 80040153
11426정성태1/5/201828903오류 유형: 444. curl로 호출할 때 발생하는 오류 정리
11425정성태1/4/201819480개발 환경 구성: 346. ASP.NET Core Web Application을 IIS에서 호스팅하는 방법 (2)
11424정성태1/4/201819053개발 환경 구성: 345. ASP.NET Core 프로젝트를 명령행에서 빌드하는 방법
11423정성태1/3/201837312VC++: 123. 내가 만든 코드보다 OpenCV의 속도가 월등히 빠른 이유 [8]파일 다운로드2
11422정성태1/2/201827941.NET Framework: 723. C# - OpenCvSharp 사용 시 C/C++을 이용한 속도 향상 (for 루프 연산) [4]파일 다운로드1
11421정성태1/2/201819697오류 유형: 443. Visual Studio - nuget configuration is invalid
11420정성태12/30/201723814.NET Framework: 722. C# - Windows 10 운영체제의 데스크톱 앱에서 음성인식(SpeechRecognizer) 사용하는 방법 [3]파일 다운로드1
11419정성태12/23/201725966.NET Framework: 721. WebClient 타입의 ...Async 메서드 호출은 왜 await + 동기 호출 시 hang 현상이 발생할까요? [2]파일 다운로드1
11418정성태12/23/201735749.NET Framework: 720. 비동기 메서드 내에서 await 시 ConfigureAwait 호출 의미 [2]파일 다운로드1
11417정성태12/22/201721593.NET Framework: 719. Task를 포함하는 async 메서드의 동작 방식 [2]
11416정성태12/21/201719259.NET Framework: 718. AsyncTaskMethodBuilder.Create() 메서드 동작 방식 [2]
11415정성태12/21/201720976.NET Framework: 717. Task를 포함하지 않는 async 메서드의 동작 방식 [6]
11414정성태12/21/201728130.NET Framework: 716. async 메서드의 void 반환 타입 사용에 대하여파일 다운로드2
11413정성태12/20/201722441개발 환경 구성: 344. 윈도우 10 - TTS 및 음성 인식을 위한 환경 설정
11412정성태12/20/201725050.NET Framework: 715. C# - Windows 10 운영체제의 데스크톱 앱에서 TTS(SpeechSynthesizer) 사용하는 방법 [1]파일 다운로드1
11411정성태12/20/201723349사물인터넷: 15. 라즈베리 파이용 C++ 프로젝트에 SSL Socket 적용
11410정성태12/20/201735639.NET Framework: 714. SSL Socket 예제 - C/C++ 서버, C# 클라이언트 [1]파일 다운로드1
11409정성태12/18/201741603VC++: 122. 오픈 소스 라이브러리를 쉽게 빌드해 주는 "C++ Package Manager for Windows: vcpkg" [7]
11408정성태12/18/201721267.NET Framework: 713. C# - SharpDX + DXGI를 이용한 윈도우 화면 캡처 소스 코드 + Direct2D 출력 + OpenCV (2)파일 다운로드1
11407정성태12/18/201724154.NET Framework: 712. C# - SharpDX + DXGI를 이용한 윈도우 화면 캡처 소스 코드 + Direct2D 출력 + OpenCV [1]파일 다운로드1
11406정성태12/17/201746543.NET Framework: 711. C# - OpenCvSharp의 Mat 데이터 조작 방법 [5]파일 다운로드1
11405정성태12/17/201742491.NET Framework: 710. C# - OpenCvSharp을 이용한 Webcam 영상 처리 + Direct2D [1]파일 다운로드1
11404정성태12/16/201729823.NET Framework: 709. C# - OpenCvSharp을 이용한 동영상(avi, mp4, ...) 처리 + Direct2D [7]파일 다운로드1
... 91  92  93  94  95  96  97  98  99  [100]  101  102  103  104  105  ...