Microsoft MVP성태의 닷넷 이야기
OCX 로드 관련 질문입니다. [링크 복사], [링크+제목 복사]
조회: 14859
글쓴 사람
링크의 전설
홈페이지
첨부 파일

안녕하세요.


현재 진행중인 프로젝트에서 ActiveX 컨트롤을 동적으로 로드하여 사용해야하는데
구글링(현재 COM 관련 지식이 전무한 상태)해봐도 적절한 해결책을 찾을 수 없어 질문드립니다.


최종 구현 목표
1. .Net 환경에서 동적(n개 ocx를 사용자 선택에 따라)으로 OCX를 로드하여 윈폼에 올림
2. 스크립트(현재는 ClearScript)에서 로드한 OCX 객체의 Property, Method, Event 접근하여 관련 작업을 수행
   ex) axHost.NextDay(); // 스크립트에서 이와 같이 접근하여 사용


step1
아래와 같이 AxHost 상속하여 OCX 로드

class AxHostEx : AxHost
{
     public AxHostEx(string strCLSID)
      : base(strCLSID)
      {
      }

     // 아무런 추가 구현 없음
}

결과 : 로드되고 폼에 올려지며 정상 동작하나 해당 OCX 고유의 Property, Method, Event에 .Net 인터페이스로 접근 불가
       (해보기 전까진 어떤 형태로든 '.Net 님이 다 알아서 해주실꺼야'라고 생각하고 있었습니다)

상태 : 이후 조사해 본바로는 AxHost를 상속받은 클래스에 COM객체와 직접 통신는 코드(기계적?)를 작성하여 해당 멤버들을 직접 구현해야 한다는 것을 알게되었습니다(제대로 이해한 것이 맞다면).
       COM 관련 지식이 전무한 상태라 관련 내용을 살펴봐도 잘 이해가되지 않고
       무엇보다 코드를 직접 구현하고 컴파일해야하는 정적인 방식이라 최종 구현 목표에 부합되지 않아 보류

step2
AxImp.exe 사용 COM Interop 어셈블리와 COM 프록시 어셈블리를 생성하여 어셈블리 로드

결과 : 완전하지 않지만 구현 목표에 부합하며 잘 동작함

상태 : 문제없이 잘 동작 하지만 동작(구현) 방식이 우아(?)하지도 안전(?)하지도 않는데다
       배포되어 사용되는 시점에 모양새가 안타까워 가능한 다른 방법으로 찾고 있음


질문 요약
.Net에서 AxImp.exe를 사용하지 않고 동적으로 로드한 OCX의 인터페이스(Property, Method, Event)를 .Net 인터페이스 형태로 제공하는 것이 가능한가?
(step1의 구현 방식에서 로드한 COM 개체를 분석해 동적으로 멤버를 생성/추가하면되겠다(리플렉션이든 뭐든?)라고 막연히 생각해보는데 이것이 기술적으로 가능한 것인지?
만약 그런 방식으로 구현이 가능하다면 AxImp.exe와 같은 도구를 별도로 제공할필요가 없었텐데하는 의문에 질문하게 되었습니다)

혹은 상기 최종 구현 목표를 달성을 위한 또 다른 방법이 있다면 조언 부탁합니다.



답변 부탁드립니다.

감사합니다.


※ 첨부한 테스트 프로젝트는 용량 관계상 스크립트 라이브러리 패키지(Nuget)를 삭제한 상태입니다.
   OCX 로드 테스트에 사용한 MSCAL.OCX도 첨부했습니다.
   개발환경 : Win7 32bit, VS2015








[최초 등록일: ]
[최종 수정일: 10/12/2015]


비밀번호

댓글 작성자
 



2015-10-12 12시44분
결국, AxImp가 작성해 주는 것은 모든 ActiveX에 대한 공통적인 코드일 뿐입니다. 따라서, clsid나 progid를 넘겨주는 부분을 설정할 수 있도록 하기만 하면 AxImp가 작성해 준 코드를 그대로 재사용할 수 있습니다. 즉, aximp.exe를 같이 배포하지 않아도 됩니다.
정성태
2015-10-13 07시26분
[링크의 전설] 답변 감사합니다.

답변해주신 내용중에 "AxImp가 작성해 준 코드를 그대로 재사용할 수 있습니다"가 구체적으로 어떤 내용인지 잘 이해되지 않습니다.


우선 제가 테스트 한 내용(step2)을 좀더 자세히 설명하면
1. 첨부 파일에 첨부한 MSCAL.OCX를 아래와 같은 명령으로 변환했습니다(예제는 MSCAL.OCX로 한정하여 작성했지만 실제로 대상 OCX는 런타임에 사용자가 결정).
aximp.exe mscal.ocx

변환 결과 아래 두 파일이 생성됩니다.

AxMSACAL.dll(윈폼 래퍼(프록시))
MSACAL.dll(RCW)

2. .Net 클라이언트에서는 AxMSACAL.dll 로드 후 clsid로 Type(AxCalendar)을 객체를 생성/사용했습니다.
   (.Net 클라이언트에서 macal.ocx의 고유의 모든 PME에 .Net 형식의 인터페이스로 접근)

코드를 그대로 재사용하라는 의미가 위의 MSACAL.dll 코드를 참조하여 AxHost를 상속받아 적절하게 구현하여 사용하라는 말씀인가요?
아니면 다른 의미인가요?

제가 질문드린 내용을 다시 한번 정리하면
프로그램 런타임에 특정 ocx가 주어졌을때 aximp.exe로 변환하여 생성되는 윈폼래퍼(위 예에서 AxMSACAL.dll)가 제공하는 수준의 타입(정보(PME))을
aximp.exe를 사용하지 않고
프로그래밍적인 방법으로 클라이언트에게 제공하는 것이 가능한가? 였습니다. 답변해주신 내용이 핀트가 어긋난것은 아닌지 확인차 댓글 달아봅니다.
[guest]
2015-10-13 08시29분
[spowner] 정태님 말은
로딩하고자 하는 n개의 ocx들이 모두 동일한 인터페이스(Property, Method, Event)를 가지고 있는 경우 AxImp를 이용해 한번만 레퍼 클레스를 생성한 후, clsid나 progid를 넘겨주는 식으로 설정만 할 수 있다면
"동일한 인터페이스"로 각각의 ocx에 aximp.exe 없이 사용가능하다는 얘기로 이해합니다.

어쩌면 AxHost클래스를 이용해 각각의 Property, Method, Event에 대한 레핑을 만든 후 사용한다라면 AxHostEx로도 사용 가능할 것 같은데요..
만약, 각각의 ocx에서 제공하는 Property, Method, Event가 동일한 인터페이스가 아니라면 역시 AxHost에 관련 호출 가능한 invoke 기능이 제공할 것도 같습니다.

한번 첨부파일을 받아 저도 한번 테스트를 해봐야겠습니다 하하
[guest]
2015-10-13 11시37분
아... 제가 실수했습니다. 부분적으로만 보고 답변을 드렸군요. 원하시는 것이 로드뿐만 아니라 해당 ocx에 포함된 멤버(Method, Event...)도 동적으로 알고 싶다고 하셨는데, 이것이 가능하려면 COM의 TypeLibrary를 해석해야 합니다. (TypeLibrary는 닷넷의 메타데이터라고 보시면 됩니다.) 다행히 이런 기능을 이미 mscorlib 어셈블리에 TypeLibConverter.ConvertTypeLibToAssembly라는 메서드로 제공하고 있는데요. 사실 AxImp.exe도 .NET Reflector로 보면 결국 저 메서드를 내부적으로 호출하고 있습니다.

자세한 것은 제 설명보다는 AxImp.exe를 .NET Reflector로 보시고 필요한 코드만 가져가 사용하시면 될 듯 합니다.

하지만, 차라리 그러느니 그냥 AxImp.exe를 사용하시는 것이 속편할지도 모릅니다. 첨부하신 소스코드에 보면 "간헐적으로 에러가 난다"고 했는데요. 그 부분의 코드를 File.Exists로 하지 말고 process.WaitForExit()로 바꿔서 처리하시면 좀 더 안정적으로 동작할 것입니다.
정성태
2015-10-14 12시56분
[링크의 전설]

어둠이 걷히고 빛이 보이네요!!

AxImp.exe를 리플렉터로 까보면되는군요(아...생각해보면 당연한건데 답변을 보기전까진 미처 생각지 못했습니다;;하하).

아무튼 이제 어떻게든 해볼 수 있을 것 같습니다(일단은 시간 관계상 AxImp.exe를 사용할 것 같습니다만).

답변해주신 두분 정말 감사합니다.
[guest]

... 16  17  18  19  20  21  22  [23]  24  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
5344하태6/2/20206916c++ .ilb에서 c# dll 호출 질문 드리겠습니다 (콜백함수 전달) [4]
5343하태6/2/20209863안녕하십니까! c# dll을 c++ .lib에서 호출 질문 드립니다. [6]
5342진우5/30/202010986c++ 에서 C# DLL 사용 문의 [2]
5341미나리5/28/20206765스레드 lock키워드 관련 질문드립니다 [3]
5339민성5/27/20206054WPF cmd을 실행을 할때 파라미터 넘기는 방법 [1]
5338서영준5/26/20207956.Net Core Blazor 서버에 Xing API를 이용한 통신 요청 [5]파일 다운로드1
5337ogos...5/26/20208073C# DB connection string 보호 방법에 대하여 [2]
5336saki5/21/20207981이벤트 뷰어 .NET Runtime 오류 [3]
5335민성5/21/20206250안녕하세요 C#으로 컴퓨터 시작프로그램 목록을 가져와서 사용안함으로 바꿀려면 [1]
5334민성5/19/20206596안녕하세요 WPF 콘솔창을 띠우면서 Ping이라는 명령어가 콘솔에 Write되게 할려면 어떻게 해야 하나요?? [1]
5333초보5/18/20206654공유 메모리 관련 문의 [2]
5332질문요5/15/20206530TcpListener TcpClient 문의 [4]
5331sdd5/13/20206353안녕하세요 Settings관련하여 질문드립니다. [2]파일 다운로드1
5330하태5/7/202010656안녕하세요! 질문 드리겠습니다! C# dll에서 c++ CLR프로젝트를 참조 추가 할 수 있나요? [2]
5329김태령4/21/20206270젠킨스에서 원격 머신에 있는 실행 파일을 실행하면 백그라운드로 뜹니다 [2]
5328crea...4/17/20206323안녕하세요! [3]파일 다운로드1
5327김동욱4/17/20206565HttpListener 사용시 HTTP/2 질문입니다. [2]
5323나그네4/15/20205843해결 115p 네임스페이스의 ConsoleApp1.exe는 netcoreapp3.1 폴더에 있었습니다. [1]파일 다운로드1
5322나그네4/14/20206804질문 115p 네임스페이스의 ConsoleApp1.exe 컴파일 또는 빌드방법 [2]파일 다운로드1
5321나그네4/14/20207753오타인가요? [2]
5320kiki...4/10/20209830C# Serial 통신 관련 질문 입니다. [5]
5319권대현4/10/20206110C++(UWP)Dll에서 C#(UWP)Dll 호출이 가능한가요? [1]
5318이승준4/9/20206967동영상 관련 업계 근황? 입니다. [1]
5317kskk...4/7/20208973OpenCV 이용 해상도 설정 질문 입니다.. [3]
5316윤현수4/7/20206174pipe 비동기방식 질문 [1]파일 다운로드1
53154/6/20206841dll파일 로드 원리? 부분에서 도움받고싶습니다. [3]
... 16  17  18  19  20  21  22  [23]  24  25  26  27  28  29  30  ...