Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

(시리즈 글이 3개 있습니다.)
.NET Framework: 584. C# - 파일 확장자에 연결된 프로그램을 등록하는 방법 (1) - 기본
; https://www.sysnet.pe.kr/2/0/10966

.NET Framework: 585. C# - 파일 확장자에 연결된 프로그램을 등록하는 방법 (2) - 웹 브라우저가 다운로드 후 자동 실행
; https://www.sysnet.pe.kr/2/0/10967

.NET Framework: 586. C# - 파일 확장자에 연결된 프로그램을 등록하는 방법 (3) - "Open with" 목록에 등록
; https://www.sysnet.pe.kr/2/0/10969




C# - 파일 확장자에 연결된 프로그램을 등록하는 방법 (3) - "Open with" 목록에 등록

지난번에 .1myext 확장자에 연결 프로그램을 등록했는데요.

C# - 파일 확장자에 연결된 프로그램을 등록하는 방법 (1) - 기본
; https://www.sysnet.pe.kr/2/0/10966

C# - 파일 확장자에 연결된 프로그램을 등록하는 방법 (2) - 웹 브라우저가 다운로드 후 자동 실행
; https://www.sysnet.pe.kr/2/0/10967

그런데, 사용자가 탐색기에서 특정 파일과 연결하고 싶어 마우스 우클릭 후 "Open with" / "Choose another app"을 선택해도 우리가 만든 예제 프로그램은 항목에 나오질 않습니다.

그 목록에 끼고 싶다면, "SOFTWARE\Classes\Applications" 경로에 command를 소유한 레지스트리 키를 하나 등록해 두어야 합니다.

Windows Registry Editor Version 5.00

[HKEY_CURRENT_USER\SOFTWARE\Classes\Applications]

[HKEY_CURRENT_USER\SOFTWARE\Classes\Applications\YourExt.exe]

[HKEY_CURRENT_USER\SOFTWARE\Classes\Applications\YourExt.exe\shell]

[HKEY_CURRENT_USER\SOFTWARE\Classes\Applications\YourExt.exe\shell\open]

[HKEY_CURRENT_USER\SOFTWARE\Classes\Applications\YourExt.exe\shell\open\command]
@="\"C:\\YourExt\\bin\\Debug\\YourExt.exe\" \"%1\""

코드로 표현하면 대략 다음과 같은 정도의 변화만 있으면 됩니다.

private static void ProcessFileExtReg(bool register)
{
    using (RegistryKey classesKey = Registry.CurrentUser.OpenSubKey(@"Software\Classes", true))
    {
        // ...[생략]...

        RegistApplication(classesKey, register);
    }
}

private static void RegistApplication(RegistryKey classesKey, bool register)
{
    using (RegistryKey appKey = classesKey.OpenSubKey("Applications", true))
    {
        if (register == true)
        {
            using (RegistryKey exeKey = appKey.CreateSubKey(assocExeFileName))
            {
                RegistShellOpenCommand(exeKey);
            }
        }
        else
        {
            // ...[생략]...
        }
    }
}

private static void RegistShellOpenCommand(RegistryKey baseKey)
{
    using (RegistryKey shellKey = baseKey.CreateSubKey("shell"))
    {
        using (RegistryKey openKey = shellKey.CreateSubKey("open"))
        {
            using (RegistryKey commandKey = openKey.CreateSubKey("command"))
            {
                string assocExePath = GetProcessPath();
                string assocCommand = string.Format("\"{0}\" \"%1\"", assocExePath);

                commandKey.SetValue(null, assocCommand);
            }
        }
    }
}

변경된 등록 프로그램을 실행한 후, 탐색기에서 아무 파일이나 우클릭을 해 "Open with" / "Choose another app" 메뉴를 선택하면 다음과 같이 프로그램 항목이 보이는 것을 확인할 수 있습니다.

yourext_openwith_1.png

그런데, 이렇게 사용자가 "Open with"로 연결 프로그램을 바꾸면 어떤 변화가 있을까요?

아마도, 가장 쉬운 방법으로 그냥 해당 확장자의 연결 항목을 바꿔버리는 걸로 구현하는 것을 생각해 볼 수 있습니다. 즉, 다음과 같이 연결된 프로그램이 등록되었다면,

Windows Registry Editor Version 5.00

[HKEY_CURRENT_USER\Software\Classes\yourext.1myext.v1\shell\open\command]
@="c:\path\to\yourext.exe \"%1\""

[HKEY_CURRENT_USER\Software\Classes\.1myext]
@="yourext.1myext.v1"

이를 notepad.exe로 바꾼다면 다음과 같이 변경하면 그만입니다.

Windows Registry Editor Version 5.00

[HKEY_CURRENT_USER\Software\Classes\.1myext]
@="txtfile"

그런데, 마이크로소프트의 '탐색기' 개발자는 이런 방식을 취하지 않았습니다. 실제로, "test.1myext" 파일을 "Open with" 기능으로 연결 프로그램을 메모장으로 바꾸면 다음의 레지스트리 키가 생성되고,

HKEY_CURRENT_USER\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\FileExts\.1myext

그 하위에 "OpenWithList" 키를 통해 해당 확장자로 연결된 적이 있는 프로그램들의 "SOFTWARE\Classes\Applications\[...exe...]" 식별자가 등록됩니다. 가령, 방금 전에 ".1myext" 확장자를 "Open with"로 메모장과 연결했다면 다음과 같은 설정을 볼 수 있습니다.

yourext_openwith_2.png

OpenWithList가 연결된 적이 있는 프로그램들의 목록을 보여준다면, 실제로 현재 사용자가 연결한 프로그램 설정은 "UserChoice"에 있습니다.

yourext_openwith_3.png

"ProgId" 값이 "Applications\NOTEPAD.EXE"로 나오는데, 이는 "SOFTWARE\Classes\Applications\NOTEPAD.EXE" 항목을 의미합니다. 여기서 재미있는 것은 "Hash" 값입니다. 현재 "Dx/cvUgkjM8="로 되어 있는데 이는 "ProgId"의 유효성 검사에서 오는 듯 합니다. 즉, 외부에서 프로그래밍적인 방법을 이용해 임의로 이 값을 바꾸는 것이 허용되지 않습니다. 가령 제가 "ProgId" 값을 "Applications\test.exe"로 바꾸고 싶어도 Hash 값 생성 규칙을 알 수 없기 때문에 유효하지 않게 되는 것입니다. (참고로, 같은 NOTEPAD.EXE로 등록해도 다시 하면 Hash값이 바뀝니다.)

또 한가지 재미있는 점은, 연결 프로그램의 설정 값에 대한 우선 순위가 "Software\Classes\.1myext"보다 "SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\FileExts\.1myext"의 것이 더 높다는 점입니다. 이 때문에 사용자가 설정한 값이 이후 설치되는 프로그램의 확장자 연결에 상관없이 독립적으로 유지가 가능합니다. 사용자 입장에서의 배려가 돋보이는 부분입니다.




그렇다면, 자신의 프로그램으로 특정 확장자 연결을 강제로 설정하고 싶다면 어떻게 해야 할까요? 공식적인 방법은 없는 듯 하지만, 어쨌든 레지스트리 등록 정보를 기반으로 밀어버리는 것을 해볼 수 있습니다. 제 경우에 다음과 같은 레지스트리 값을 모두 새로운 프로그램으로 연결해 보았는데요.

HKEY_CURRENT_USER\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\FileExts\UserChoice 삭제
HKEY_CURRENT_USER\SOFTWARE\Microsoft\Windows\Roaming\OpenWith\FileExts\.1myext 삭제
HKEY_CURRENT_USER\Software\Classes\.1myext 의 ProgId 교체

위의 역할을 하는 프로그램을 첨부 파일의 ReplaceAssoc 프로젝트로 구현했습니다. 가령, ".1myext" 확장자를 교체하고 싶다면 다음과 같이 실행해 주면 됩니다.

c:\temp>ReplaceAssoc .1myext

테스트 해보면 windows 8에서는 ".1myext" 확장자 교체는 잘 되었지만 알집과 연결된 ".zip"의 경우에는 뭔가 설정이 더 있어서 그런지 연결 프로그램이 해제된 상태로만 바뀌고 교체까지는 안되었습니다. 반면, 윈도우 10의 경우 모두 연결 프로그램이 잘 교체되었습니다. (물론, 비공식적인 방법이므로 향후 얼마든지 동작은 바뀔 수 있습니다.)

(첨부 파일은 이 글의 예제 코드를 포함합니다.)





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







[최초 등록일: ]
[최종 수정일: 7/10/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)
12596정성태4/12/20219645개발 환경 구성: 568. Windows의 80 포트 점유를 해제하는 방법
12595정성태4/12/20219055.NET Framework: 1036. SQL 서버 - varbinary 타입에 대한 문자열의 CAST, CONVERT 변환을 C# 코드로 구현
12594정성태4/11/20218495.NET Framework: 1035. C# - kubectl 명령어 또는 REST API 대신 Kubernetes 클라이언트 라이브러리를 통해 프로그래밍으로 접근 [1]파일 다운로드1
12593정성태4/10/20219691개발 환경 구성: 567. Docker Desktop for Windows - kubectl proxy 없이 k8s 대시보드 접근 방법
12592정성태4/10/20219536개발 환경 구성: 566. Docker Desktop for Windows - k8s dashboard의 Kubeconfig 로그인 및 Skip 방법
12591정성태4/9/202112812.NET Framework: 1034. C# - byte 배열을 Hex(16진수) 문자열로 고속 변환하는 방법 [2]파일 다운로드1
12590정성태4/9/20219298.NET Framework: 1033. C# - .NET 4.0 이하에서 Console.IsInputRedirected 구현 [1]
12589정성태4/8/202110665.NET Framework: 1032. C# - Environment.OSVersion의 문제점 및 윈도우 운영체제의 버전을 구하는 다양한 방법 [1]
12588정성태4/7/202111187개발 환경 구성: 565. PowerShell - New-SelfSignedCertificate를 사용해 CA 인증서 생성 및 인증서 서명 방법
12587정성태4/6/202112044개발 환경 구성: 564. Windows 10 - ClickOnce 배포처럼 사용할 수 있는 MSIX 설치 파일 [1]
12586정성태4/5/20219706오류 유형: 710. Windows - Restart-Computer / shutdown 명령어 수행 시 Access is denied(E_ACCESSDENIED)
12585정성태4/5/20219419개발 환경 구성: 563. 기본 생성된 kubeconfig 파일의 내용을 새롭게 생성한 인증서로 구성하는 방법
12584정성태4/1/202110136개발 환경 구성: 562. kubeconfig 파일 없이 kubectl 옵션만으로 실행하는 방법
12583정성태3/29/202111612개발 환경 구성: 561. kubectl 수행 시 다른 k8s 클러스터로 접속하는 방법
12582정성태3/29/202110324오류 유형: 709. Visual C++ - 컴파일 에러 error C2059: syntax error: '__stdcall'
12581정성태3/28/202110241.NET Framework: 1031. WinForm/WPF에서 Console 창을 띄워 출력하는 방법 (2) - Output 디버깅 출력을 AllocConsole로 우회 [2]
12580정성태3/28/20218926오류 유형: 708. SQL Server Management Studio - Execution Timeout Expired.
12579정성태3/28/20219062오류 유형: 707. 중첩 가상화(Nested Virtualization) - The virtual machine could not be started because this platform does not support nested virtualization.
12578정성태3/27/20219312개발 환경 구성: 560. Docker Desktop for Windows 기반의 Kubernetes 구성 (2) - WSL 2 인스턴스에 kind가 구성한 k8s 서비스 위치
12577정성태3/26/202111354개발 환경 구성: 559. Docker Desktop for Windows 기반의 Kubernetes 구성 - WSL 2 인스턴스에 kind 도구로 k8s 클러스터 구성
12576정성태3/25/20219175개발 환경 구성: 558. Docker Desktop for Windows에서 DockerDesktopVM 기반의 Kubernetes 구성 (2) - k8s 서비스 위치
12575정성태3/24/20218185개발 환경 구성: 557. Docker Desktop for Windows에서 DockerDesktopVM 기반의 Kubernetes 구성
12574정성태3/23/202112303.NET Framework: 1030. C# Socket의 Close/Shutdown 동작 (동기 모드)
12573정성태3/22/202110085개발 환경 구성: 556. WSL 인스턴스 초기 설정 명령어 [1]
12572정성태3/22/20219618.NET Framework: 1029. C# - GC 호출로 인한 메모리 압축(Compaction)을 확인하는 방법파일 다운로드1
12571정성태3/21/20218773오류 유형: 706. WSL 2 기반으로 "Enable Kubernetes" 활성화 시 초기화 실패 [1]
... 31  32  33  34  35  36  37  38  39  40  41  [42]  43  44  45  ...