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

(시리즈 글이 3개 있습니다.)
.NET Framework: 287. Excel Sheet를 WinForm에서 사용하는 방법
; https://www.sysnet.pe.kr/2/0/1208

.NET Framework: 453. C# - 오피스 파워포인트(Powerpoint) 파일을 WinForm에서 보는 방법
; https://www.sysnet.pe.kr/2/0/1728

.NET Framework: 853. Excel Sheet를 WinForm에서 사용하는 방법 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/12002




Excel Sheet를 WinForm에서 사용하는 방법 - 두 번째 이야기

예전에 쓴,

Excel Sheet를 WinForm에서 사용하는 방법
; https://www.sysnet.pe.kr/2/0/1208

dsoframer.ocx가 아직도 잘 동작하는군요. ^^ 그래서 오래간만에 해당 프로젝트를 Visual Studio 2019로 마이그레이션 해봤습니다. 우선, C++ 프로젝트는 빌드하면 다음과 같은 식의 매크로 사용 오류가 발생합니다.

C3688 invalid literal suffix 'DSOFRAMERCTL_CLSIDSTR'; literal operator or literal operator template 'operator ""DSOFRAMERCTL_CLSIDSTR' not found

문제가 된 소스 코드를,

hr = RegRecursiveDeleteKey(HKEY_CLASSES_ROOT, "CLSID\\"DSOFRAMERCTL_CLSIDSTR);

매크로와 큰따옴표 간의 공백만 추가하는 것으로 바꾸면 해결됩니다.

hr = RegRecursiveDeleteKey(HKEY_CLASSES_ROOT, "CLSID\\" DSOFRAMERCTL_CLSIDSTR);

이 외에는 컴파일 오류는 발생하지 않으며, 링크 단계에서 다음의 오류가 발생합니다.

Error LNK2026 module unsafe for SAFESEH image.

이건 Q&A에 따라 "Linker / Advanced / Image Has Safe Exception Handlers" 옵션 값을 꺼주면(/SAFESEH:NO) 됩니다.

그 외에 개발자 PC에 설치한 Office 버전에 따라 #import 오류가 발생하는데 이 부분은 다음과 같은 식의 매크로를 정의하는 것으로 일단 우회해서 해결했습니다.

// 아래의 경로는 개발자의 환경에 맞게 수정해야 합니다.

#if OFFICEVER == 16
#import "C:\Program Files (x86)\Microsoft Office\root\VFS\ProgramFilesCommonX86\Microsoft Shared\OFFICE16\mso.dll" rename("ColorFormat", "ColorFormatEx"),rename_namespace("Office")
#import "C:\Program Files (x86)\Microsoft Office\root\VFS\ProgramFilesCommonX86\Microsoft Shared\VBA\VBA6\VBE6EXT.olb" rename_namespace("VBE6")
#import "C:\Program Files (x86)\Microsoft Office\root\Office16\MSWORD.olb" rename("ExitWindows","ExitWindowsEx"),rename_namespace("MSWord")
#import "C:\Program Files (x86)\Microsoft Office\root\Office16\EXCEL.EXE" rename("RGB","RGBEx"),rename("DialogBox","DialogBoxEx"),rename_namespace("MSExcel")
#import "C:\Program Files (x86)\Microsoft Office\root\Office16\MSPPT.OLB" named_guids,rename_namespace("MSPPT")

#else
    // ...[생략]...
#endif

변경한 소스 코드는 다음의 github에 올려 두었습니다.

DotNetSamples/Cpp/dsoframer_v2.2.1.2
; https://github.com/stjeong/DotNetSamples/tree/master/Cpp/dsoframer_v2.2.1.2

하는 김에, 예제 코드도 새롭게 reg-free를 적용해,

Registry 등록 과정 없이 COM 개체 사용 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/1167

ocx를 regsvr32.exe로 등록 없이 사용할 수 있도록 변경했습니다.

DotNetSamples/WinForms/EmbedExcelSample
; https://github.com/stjeong/DotNetSamples/tree/master/WinForms/EmbedExcelSample




참고로, dsoframer.ocx가 보여주는 Excel 화면을 프로그래밍으로 제어하고 싶을 수 있는데요. dsoframer 컨트롤에 보면 SetFieldValue 같은 메서드를 제공하고 있긴 하지만 소스 코드를 보면 그것은 Cell과는 전혀 무관한 동작을 수행하는 것을 볼 수 있습니다. 그럼 어떻게 해야 할까요?

간단합니다. dsoframer 컨트롤이 제공하는 ActiveDocument를 dynamic 변수로 받아 VBA 스크립트에서 사용하던 식으로 접근하면 됩니다. 예를 들어, 다음의 소스 코드는 A1, A2 셀에 대해 "test"라는 문자열을 설정합니다.

private void Button1_Click(object sender, EventArgs e)
{
    dynamic doc = this.axFramerControl1.ActiveDocument;
    doc.ActiveSheet.Range("A1:A2").Cells.value = "test";
}




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







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

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

비밀번호

댓글 작성자
 



2020-08-31 01시35분
[박세준] 글쓴이님 혹시 마지막에 문자열을 입력을 계속해서 밑으로 써내려갈수있는 방법이 있을까요?
A1:A2에 작성후 재입력시 B1:B2에 작성이 하고싶습니다
[guest]
2020-08-31 02시02분
@박세준 그런 경우라면 그냥 for 루프를 써서 cell의 위치를 직접 지정하면 될 텐데요.
정성태
2020-09-01 03시41분
[박세준] 답글 감사합니다!
한가지만더 조언을 구하고싶은데 엑셀시트를 오픈한상태로 이벤트를이용해 행을 삽입할수있는 방법이 있을까요..?
[guest]
2020-09-01 11시54분
이벤트를 이용한다는 것이 오픈된 엑셀 시트의 특정 이벤트를 구독하고 싶다는 건가요? 어쨌든, 예제 코드를 좀 더 추가했으니 확인해 보세요.

DotNetSamples/WinForms/EmbedExcelSample/Form1.cs
; https://github.com/stjeong/DotNetSamples/blob/master/WinForms/EmbedExcelSample/Form1.cs
정성태
2020-09-01 11시56분
* 이후, 더 이상 dsoFramer 컨트롤에 대한 질문은 받지 않겠습니다. (공식적인 컨트롤이 아닌데다 개인적으로 더 이상 흥미가 없는 자료입니다.)
정성태

1  2  3  4  [5]  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13505정성태12/27/20232754닷넷: 2189. C# - WebSocket 클라이언트를 닷넷으로 구현하는 예제 (System.Net.WebSockets)파일 다운로드1
13504정성태12/27/20232329닷넷: 2188. C# - ASP.NET Core SignalR로 구현하는 채팅 서비스 예제파일 다운로드1
13503정성태12/27/20232194Linux: 67. WSL 환경 + mlocate(locate) 도구의 /mnt 디렉터리 검색 문제
13502정성태12/26/20232307닷넷: 2187. C# - 다른 프로세스의 환경변수 읽는 예제파일 다운로드1
13501정성태12/25/20232102개발 환경 구성: 700. WSL + uwsgi - IPv6로 바인딩하는 방법
13500정성태12/24/20232186디버깅 기술: 194. Windbg - x64 가상 주소를 물리 주소로 변환
13498정성태12/23/20232868닷넷: 2186. 한국투자증권 KIS Developers OpenAPI의 C# 래퍼 버전 - eFriendOpenAPI NuGet 패키지
13497정성태12/22/20232299오류 유형: 885. Visual Studiio - error : Could not connect to the remote system. Please verify your connection settings, and that your machine is on the network and reachable.
13496정성태12/21/20232318Linux: 66. 리눅스 - 실행 중인 프로세스 내부의 환경변수 설정을 구하는 방법 (gdb)
13495정성태12/20/20232327Linux: 65. clang++로 공유 라이브러리의 -static 옵션 빌드가 가능할까요?
13494정성태12/20/20232509Linux: 64. Linux 응용 프로그램의 (C++) so 의존성 줄이기(ReleaseMinDependency) - 두 번째 이야기
13493정성태12/19/20232599닷넷: 2185. C# - object를 QueryString으로 직렬화하는 방법
13492정성태12/19/20232302개발 환경 구성: 699. WSL에 nopCommerce 예제 구성
13491정성태12/19/20232238Linux: 63. 리눅스 - 다중 그룹 또는 사용자를 리소스에 권한 부여
13490정성태12/19/20232358개발 환경 구성: 698. Golang - GLIBC 의존을 없애는 정적 빌드 방법
13489정성태12/19/20232143개발 환경 구성: 697. GoLand에서 ldflags 지정 방법
13488정성태12/18/20232074오류 유형: 884. HTTP 500.0 - 명령행에서 실행한 ASP.NET Core 응용 프로그램을 실행하는 방법
13487정성태12/16/20232392개발 환경 구성: 696. C# - 리눅스용 AOT 빌드를 docker에서 수행 [1]
13486정성태12/15/20232206개발 환경 구성: 695. Nuget config 파일에 값 설정/삭제 방법
13485정성태12/15/20232091오류 유형: 883. dotnet build/restore - error : Root element is missing
13484정성태12/14/20232168개발 환경 구성: 694. Windows 디렉터리 경로를 WSL의 /mnt 포맷으로 구하는 방법
13483정성태12/14/20232306닷넷: 2184. C# - 하나의 resource 파일을 여러 프로그램에서 (AOT 시에도) 사용하는 방법파일 다운로드1
13482정성태12/13/20232883닷넷: 2183. C# - eFriend Expert OCX 예제를 .NET Core/5+ Console App에서 사용하는 방법 [2]파일 다운로드1
13481정성태12/13/20232275개발 환경 구성: 693. msbuild - .NET Core/5+ 프로젝트에서 resgen을 이용한 리소스 파일 생성 방법파일 다운로드1
13480정성태12/12/20232626개발 환경 구성: 692. Windows WSL 2 + Chrome 웹 브라우저 설치
13479정성태12/11/20232320개발 환경 구성: 691. WSL 2 (Ubuntu) + nginx 환경 설정
1  2  3  4  [5]  6  7  8  9  10  11  12  13  14  15  ...