성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] VT sequences to "CONOUT$" vs. STD_O...
[정성태] NetCoreDbg is a managed code debugg...
[정성태] Evaluating tail call elimination in...
[정성태] What’s new in System.Text.Json in ....
[정성태] What's new in .NET 9: Cryptography ...
[정성태] 아... 제시해 주신 "https://akrzemi1.wordp...
[정성태] 다시 질문을 정리할 필요가 있을 것 같습니다. 제가 본문에...
[이승준] 완전히 잘못 짚었습니다. 댓글 지우고 싶네요. 검색을 해보...
[정성태] 우선 답글 감사합니다. ^^ 그런데, 사실 저 예제는 (g...
[이승준] 수정이 안되어서... byteArray는 BYTE* 타입입니다...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>C# - NTFS 파일에 사용자 정의 속성값 추가하는 방법</h1> <p> 관련해서 질문이 올라왔습니다. ^^<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > EXE파일에 속성값 추가하는 방법이 있나요? ; <a target='tab' href='https://www.sysnet.pe.kr/3/0/5680'>https://www.sysnet.pe.kr/3/0/5680</a> </pre> <br /> 질문하신 분은 속성 창의 "Details" 패널에,<br /> <br /> <img alt='custom_prop_1.png' src='/SysWebRes/bbs/custom_prop_1.png' /><br /> <br /> 사용자 정의 칼럼을 추가하고 싶은 듯한데요, (아마도 방법이 있을 것 같은데) 아쉽게도 관련 문서를 찾지를 못하겠습니다. ^^; 예전에는 이와 관련해서 자료가 많았는데, 어느 순간부터 Azure와 닷넷 위주로 자료들이 개편되면서 과거 C/C++로 만들었던 Explorer 확장과 관련한 문서들을 쉽게 찾을 수 없는 상황이 되었습니다.<br /> <br /> 일단, "Details" 패널에 보이진 않더라도 사용자 정의 칼럼을 추가하는 것은 가능합니다. 이에 대해서는 다음의 글에서 자세하게 설명하고 있는데요,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > The Complete Idiot's Guide to Writing Shell Extensions - Part VIII - A tutorial on adding columns to Explorer's details view via a column handler shell extension. ; <a target='tab' href='https://www.codeproject.com/Articles/722/The-Complete-Idiots-Guide-to-Writing-Shell-Exten-8'>https://www.codeproject.com/Articles/722/The-Complete-Idiots-Guide-to-Writing-Shell-Exten-8</a> </pre> <br /> 이렇게 해서 추가한 칼럼은 아래 화면과 같이,<br /> <br /> <img alt='custom_prop_2.gif' src='/SysWebRes/bbs/custom_prop_2.gif' /><br /> <br /> Explorer의 칼럼을 표시하는 헤더 영역을 마우스 우 클릭해 "More ..." 메뉴를 선택하는 것으로, 파일 탐색기의 리스트 뷰에서 보이게 할 수는 있습니다. (하지만 속성 창의 "Details" 패널에는 보이지 않습니다.)<br /> <br /> 대신, 저것을 꼭 속성 창에서 보고 싶다면 "Details"가 아닌 별도의 탭 페이지를 추가하는 식으로 구현할 수는 있습니다. 역시 이에 대해서도 다음의 문서에서 자세하게 설명하고 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > The Complete Idiot's Guide to Writing Shell Extensions - Part V - A tutorial on writing a shell extension that adds pages to the properties dialog of files. ; <a target='tab' href='https://www.codeproject.com/Articles/463/The-Complete-Idiots-Guide-to-Writing-Shell-Exten-5'>https://www.codeproject.com/Articles/463/The-Complete-Idiots-Guide-to-Writing-Shell-Exten-5</a> </pre> <br /> 그럼, 자신이 추가한 키/값을 보여주는 탭을 아래와 같은 식으로 자유자재로 추가하는 것이 가능합니다.<br /> <br /> <img alt='custom_prop_3.png' src='/SysWebRes/bbs/custom_prop_3.png' /><br /> <br /> <hr style='width: 50%' /><br /> <br /> C# 코드에서 사용자 정의 칼럼을 추가하는 방법 중의 하나는 DSOFile 구성요소를 활용하는 것입니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > DSOFile ; <a target='tab' href='https://www.nuget.org/packages/DSOFile/'>https://www.nuget.org/packages/DSOFile/</a> 64 Bit Application Cannot Use DSOfile ; <a target='tab' href='https://www.codeproject.com/Tips/1118708/Bit-Application-Cannot-Use-DSOfile'>https://www.codeproject.com/Tips/1118708/Bit-Application-Cannot-Use-DSOfile</a> C# Add and Edit Office Document Custom Properties ; <a target='tab' href='https://scatteredcode.net/c-add-and-edit-ntfs-custom-properties'>https://scatteredcode.net/c-add-and-edit-ntfs-custom-properties</a> </pre> <br /> 아쉽게도 현재 dsofile 패키지는 deprecated 상태라서 "Install-Package" 명령어로 설치가 안 되므로, 직접 dsofilenupkg.1.0.0.nupkg 패키지를 다운로드해,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > dsofilenupkg.1.0.0.nupkg ; <a target='tab' href='https://www.nuget.org/api/v2/package/dsoFileNuPKG/1.0.0'>https://www.nuget.org/api/v2/package/dsoFileNuPKG/1.0.0</a> </pre> <br /> 압축을 풀어 나온 dsofile.dll을 regsvr32로 등록한 다음,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > C:\Lib> <span style='color: blue; font-weight: bold'>dir</span> Volume in drive C has no label. Volume Serial Number is C00B-0E7F Directory of C:\Lib 2022-06-13 오전 10:21 <DIR> . 2022-06-13 오전 09:56 242,176 <span style='color: blue; font-weight: bold'>dsofile.dll</span> 1 File(s) 242,176 bytes 1 Dir(s) 673,986,424,832 bytes free C:\Lib> <span style='color: blue; font-weight: bold'>regsvr32 dsofile.dll</span> </pre> <br /> 비주얼 스튜디오의 COM 참조 탭을 이용해 "DSO OLE Document Properties Reader 2.0" 구성 요소를 추가해야 합니다. (참고로 <a target='tab' href='https://www.sysnet.pe.kr/2/0/1167'>manifest 파일을 이용하면</a> 배포 시 등록 과정은 생략할 수 있습니다.)<br /> <br /> 이후, 다음과 같이 원하는 파일에 대해 사용자 정의 속성 값을 추가/설정할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > using System; using System.Linq; <span style='color: blue; font-weight: bold'>// 주의 사항: dsofile은 deprecated 패키지이므로 보안상 문제가 발생할 수 있습니다.</span> internal class Program { static void Main(string[] args) { DSOFile.OleDocumentPropertiesClass file = new DSOFile.OleDocumentPropertiesClass(); file.Open(@"test.txt", false, DSOFile.dsoFileOpenOptions.dsoOptionDefault); string key = "key1"; object value = "value3"; if (file.CustomProperties.Exists(key) == false) { // Adds new custom property. <span style='color: blue; font-weight: bold'>file.CustomProperties.Add(key, ref value);</span> } else { <span style='color: blue; font-weight: bold'>file.CustomProperties[key].set_Value(ref value);</span> } // Go through existing custom properties. <span style='color: blue; font-weight: bold'>foreach (DSOFile.CustomProperty p in file.CustomProperties) { Console.WriteLine("{0}:{1}", p.Name, p.get_Value().ToString()); }</span> file.Save(); file.Close(true); } } static class External { internal static bool Exists(this DSOFile.CustomProperties props, string key) { foreach (DSOFile.CustomProperty p in props) { if (p.Name == key) { return true; } } return false; } } </pre> <br /> 위의 코드는 단순히 <a target='tab' href='https://www.sysnet.pe.kr/2/0/10795#ads'>ADS(Alternate Data Stream)</a>에 독자적인 유형의 "key1" 데이터를 쓴 것에 불과합니다. 실제로 <a target='tab' href='https://docs.microsoft.com/en-us/sysinternals/downloads/streams'>sysinternals의 streams와 같은 도구</a>를 이용해 다음과 같이 확인할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > C:\temp\ConsoleApp2\bin\Debug> <span style='color: blue; font-weight: bold'>streams test.txt</span> streams v1.60 - Reveal NTFS alternate streams. Copyright (C) 2005-2016 Mark Russinovich Sysinternals - www.sysinternals.com C:\temp\ConsoleApp2\bin\Debug\test.txt: :DocumentSummaryInformation:$DATA 212 :{4c8cc155-6c1e-11d1-8e41-00c04fb9386d}:$DATA 0 </pre> <br /> 당연히, 이렇게 추가한 속성은 코드로만 접근할 수 있을 뿐 "<a target='tab' href='https://www.codeproject.com/Articles/722/The-Complete-Idiots-Guide-to-Writing-Shell-Exten-8'>The Complete Idiot's Guide to Writing Shell Extensions - Part VIII</a>" 글에서 다룬 것처럼 탐색기의 헤더 칼럼과는 연동되지 않습니다.<br /> <br /> 어찌 보면... 저렇게 ads로만 기록하는 용도라면 굳이 dsofile을 이용하기보다는 <a target='tab' href='https://stackoverflow.com/questions/604960/how-to-read-and-modify-ntfs-alternate-data-streams-using-net'>ADS 읽기/쓰기를 직접 수행</a>하는 것이 더 편리할 것입니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> 만약 임의의 파일이 아닌, OLE Document에 한해 속성을 쓰는 것이라면 Microsoft.WindowsAPICodePack 패키지를 활용하는 것도 좋습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Install-Package Microsoft.WindowsAPICodePack-Shell Install-Package Microsoft.WindowsAPICodePack-Core </pre> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > using Microsoft.WindowsAPICodePack.Shell; using Microsoft.WindowsAPICodePack.Shell.PropertySystem; // Read/Write 'Extended' file properties (C#) // <a target='tab' href='https://stackoverflow.com/questions/220097/read-write-extended-file-properties-c'>https://stackoverflow.com/questions/220097/read-write-extended-file-properties-c</a> namespace ConsoleApp3 { internal class Program { static void Main(string[] args) { string filePath = @"<span style='color: blue; font-weight: bold'>test.docx</span>"; var file = ShellFile.FromFilePath(filePath); // Read and Write: string[] oldAuthors = file.Properties.System.Author.Value; string oldTitle = file.Properties.System.Title.Value; file.Properties.System.Author.Value = new string[] { "Author #1", "Author #2" }; file.Properties.System.Title.Value = "Example Title"; file.Properties.System.Comment.Value = "test is good"; // Alternate way to Write: ShellPropertyWriter propertyWriter = file.Properties.GetPropertyWriter(); propertyWriter.WriteProperty(SystemProperties.System.FileVersion, new string[] { "1.0.0.5 " }); // propertyWriter.WriteProperty(SystemProperties.System.Author, new string[] { "Author" }); propertyWriter.Close(); } } } </pre> <br /> 또는, OpenXml에 해당하는 Office 문서로 한정한다면 DocumentFormat.OpenXml 패키지를 이용하는 것도 고려할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Install-Package DocumentFormat.OpenXml </pre> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > using System; using DocumentFormat.OpenXml.Packaging; namespace GetApplicationProperty { // Install-Package DocumentFormat.OpenXml class Program { private const string FILENAME = @"C:\temp\<span style='color: blue; font-weight: bold'>test.docx</span>"; static void Main(string[] args) { using (WordprocessingDocument document = WordprocessingDocument.Open(FILENAME, true)) { var props = document.ExtendedFilePropertiesPart.Properties; props.Company = new DocumentFormat.OpenXml.ExtendedProperties.Company("test_cmp"); props.Save(); } using (WordprocessingDocument document = WordprocessingDocument.Open(FILENAME, false)) { var props = document.ExtendedFilePropertiesPart.Properties; if (props.Company != null) Console.WriteLine("Company = " + props.Company.Text); if (props.Lines != null) Console.WriteLine("Lines = " + props.Lines.Text); if (props.Manager != null) Console.WriteLine("Manager = " + props.Manager.Text); } } } } </pre> <br /> (<a target='tab' href='https://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?fid=1941&boardid=331301885'>첨부 파일은 이 글의 예제 코드를 포함</a>합니다.)<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
1781
(왼쪽의 숫자를 입력해야 합니다.)