반응형

한동안 WinSCP의 업데이트가 크게되지 않아 번역할 일이 없었는데 간만에 martin이 메일을 보내서 로컬라이징을 해 달라고 하는군요.

홈페이지에 들어가보니 WinSCP 5.0.6 Beta 버전을 만들었다고 하는데, 추가된 문장들이 꽤 있더군요.

 

기존 4 버전에서와 틀려진 점은 눈에 띄게 크게 보이진 않지만, 4 버전대는 ANSI로 프로그램을 작성했다면, 5 버전대는 유니코드로 프로그램을 재 작성했더군요.

 

덕분에 기존 로컬라이징한 파일을 UTF-8로 올리는 수고를 해야 했지만, 좀 더 다국어를 잘 지원할 수 있게 되었군요.

 

하루만에 번역해서 올려놓는 바람에 바쁘긴 했지만, 이렇게 안 해놓으면 다음번 버전업 시에 설치 패키지에 한글이 안들어간답니다.

 

반응형
반응형

MS VS 2008부터 MFC Feature Pack을 지원합니다.


Doc-View 형태의 응용프로그램을 만들면 MFC Feature Pack에서 제공하는 OutlookBar 와 Docking Pane를 이용할 수 있습니다. OutlookBar와 Docking Pane의 레이아웃은 자동으로 저장되고 로드 됩니다.

매우 편리한 기능이긴 하지만 또한 매우 불편한 기능입니다. 물론 이 레이아웃 상태에 대해서 커스터마이징을 할 수 있는 여지를 두었군요.

MFC Feature Pack으로 Doc-VIew 응용프로그램을 생성할 시 기본 App 클래스는
CWinApp에서 CWinAppEx로 변경됩니다.
CWinAppEx에서 이 상태 저장 및 복구를 하는 군요.
CWinAppEx의 메소드 중 다음 메소드를 주의해서 보시면 됩니다.

  • CleanState(): 레이아웃 상태를 모두 초기화 한다.
  • LoadState(): 레이아웃 상태를 로드한다.
  • SaveState(): 레이아웃 상태를 저장한다.

위와 같은 메소드가 있지만 CleanState() 정도가 유용하게 쓰일 수 있을 것 같습니다.

 

CWinAppEx를 상속받은 클래스의 InitInstance()에서 SetRegistryKey() 구문 다음에 CleanState()를 호출하면 항상 레이아웃이 초기화한 상태로 표시되게 됩니다. 레이아웃을 로드하는 시점은 ProcessShellCommand()에서 프레임워크상 자동으로 로딩되게 되어 있습니다.

또한 이 레이아웃 자동 저장이 필요없다면 CWinAppEx의 다음 메소드를 오버라이딩 합니다.

virtual void OnClosingMainFrame(CFrameImpl* pFrameImpl);

기본 동작은 SaveState() 메소드를 호출합니다.

 

그러므로 레이아웃 자동저장이 되지 않게 하기 위해서는 OnClosingMainFrame()메소드를 오버라이딩하여 빈 구문으로 놓으면 레이아웃이 자동으로 저장되지 않습니다.

 

반응형

'개발 > Windows' 카테고리의 다른 글

Visual Studio 2013 로그인이 안될 경우 해결 방법  (0) 2022.08.29
반응형

ActiveX를 만들때 매개변수로 배열을 받아야 할 필요가 있다.

많은 일련의 데이터를 한번에 받기 위해서 보통 문자열에 분리자(Separator)로 분리해서 문자열로 입력하는것이 보통인데
배열을 받는 방법은 없을까 해서 알아보았다.
일단 먼저 웹 스크립트에서 배열을 사용하는 방법을 알아보자. 배열은 다음과 같이 쓸수 있다.
자바 스크립트라면,

또는 VB Script라면,

 

이 배열을 ActiveX의 매개변수에 어떻게 넣느냐가 문제이다.

 

일단 기본 데이터 형식은 아니기 때문에, VARIANT 형식으로 받아야 하겠다. 
ActiveX의 메소드는 다음과 같이 정의를 하도록 하자.
void ShowArray(VARIANT varArray)

 

나는 VS2003을 쓰는데, 휴.. MFC ActiveX에서 이렇게 메소드를 만들자마자 버그코드를 만들기 시작한다.
IDL의 선언은 void ShowArray(VARIANT varArray); 로 정의되고,
컨트롤 소스에는 void ShowArray(VARIANT varArray);로 정의되었다. ㅠㅠ
컨트롤 소스가 잘못 생성되어서 나왔다. 일단 void ShowArray(LPVARIANT varArray); 로 정의를 변경해야 한다.
이놈의 VS2003 버그..ㅠㅠ 너무 많다..

 

배열을 VARIANT값으로 입력을 받을때 Java Script(JScript)와 VB Script는 다른 형식으로 입력된다.
Java Script(이하 JScript)는 VT_DISPATCH 형식으로 들어오고,
VB Script는 VT_ARRAY | VT_BSTR | VT_BYREF 형식으로 들어온다.

 

VB Script 배열 형식을 받아서 처리할 수 있는 예제는 인터넷에 널려있을것이다.
문제는 JScript 배열 형식을 받아서 처리하는 예제는 찾기가 어렵다.(나만 못찾는건지.)

 

일단 JScript 배열을 처리하려면, JScript 배열의 길이를 알아야 하지 않는가? 다음은 배열의 값이 접근을 할 수 있어야 하고...
JScript의 배열의 길이와 값에 접근 하기위해서는 다음과 같이 코딩해야한다.

 

void TestOCX::ShowArray(LPVARIANT varArray)
{
	CComVariant vtSource;
	// VT_BYREF 형식으로 입력되면 REF 값을 VARIANT로 복사한다.
	// VB Script 배열 형식은 VT_BYREF를 포함하므로 이걸 없앤다.
	::VariantCopyInd(&vtSource, varArray);
	// Variant 형식이 VT_DISPATCH인지 확인한다.
	if(VT_DISPATCH == (VT_DISPATCH & V_VT(&vtSource)) )
	{
		COleDispatchDriver disp;
		// COleDispatchDriver에 Attach한다. MFC는 좋은 클래스들을 많이 갖고 있다.
		disp.AttachDispatch(vtSource->pdispVal, FALSE);
		
		try
		{
			DISPID diLength;
			LPOLESTR length = L"length";
			long nArrayLength = 0;
			// Dispatch에서 length의 DISPID를 구한다.
			hr = disp->m_lpDispatch->GetIDsOfNames(IID_NULL, &length, 1, LOCALE_USER_DEFAULT, &diLength);
			if(FAILED(hr))
			{
				FireError(E_INVALIDARG, "매개변수 입력이 잘못되었습니다.");
			}
			
			// length의 속성값을 획득한다.
			disp.GetProperty(diLength, VT_I4, &nArrayLength);
			
			// 배열 길이 만큼 for문 돌면서 배열 값 출력
			for(int i = 0; i < nArrayLength; i++)
			{
				DISPID diIndex;
				TCHAR szIndex[20];
				CComBSTR bstrIndex;
				CComVariant vtData;
				
				// 배열 인덱스 속성값은 문자열로 지정하면 얻을 수 있다.
				// L"0", L"1", L"2" 처럼.

				StringCchPrintf(szIndex, 20, "%d", i);
				bstrIndex = szIndex;

				// 배열 인덱스의 DISPID를 얻는다.
				disp.m_lpDispatch->GetIDsOfNames(IID_NULL, &bstrIndex, 1, LOCALE_USER_DEFAULT, &diIndex);
				
				// 배열 값을 VARIANT 형식으로 받는다.
				disp.GetProperty(diIndex, VT_VARIANT, &vtData);
				
				// 예제에서는 배열의 값이 항상 문자열이여야 한다.
				if( VT_BSTR != V_VT(&vtData) )
				{
					FireError(E_INVALIDARG, "문자열 배열만 입력 가능합니다.");
					break;
				}
				
				//배열의 값을 각각 메세지 박스로 출력한다.
				AfxMessageBox(CString(vtData.bstrVal));
			}
		}
		catch(CMemoryException *pEx)
		{
			// 메모리 예외 처리 : 예제에서는 그냥 ReportError 및 FireError로 처리
			pEx->ReportError();
			FireError(E_OUTOFMEMORY, "메모리 오류입니다.");
			pEx->Delete();
		}
		catch(COleException *pEx)
		{
			// OLE 예외 처리 : 예제에서는 그냥 ReportError 및 FireError로 처리
			pEx->ReportError();
			FireError(pEx->m_sc, "OLE 오류입니다.");
			pEx->Delete();
		}
		catch(COleDispatchException* pEx)
		{
			// OLE Dispatch 예외 처리 : 예제에서는 그냥 ReportError 및 FireError로 처리
			pEx->ReportError();
			FireError(pEx->m_scError, "OLE Dispatch 오류입니다.");
			pEx->Delete();
		}
	}
}

 

예제를 보면 JScript 배열이 어떻게 구성되어 있는지 볼 수 있다.

 

여기서 특이한 점은 속성값의 DISPID이다.

TRACE를 따라가다보면 알겠지만, 배열의 크기가 3이라고 할때, length의 DISPID는 4값이면, "0"~"2"까지의 DISPID는 1~3 이다.

 

그럼 DISPID 0값은 무엇일까?

 

DISPID 0값은 "1, 2, 3" 이런식으로 모든 배열의 데이터가 ","로 분리되어서 나온다.

 

DISPID 5,6 값이 VT_DISPATCH로 되어 있고, 각각 constructor, prototype이라는 속성이 지원되는데, javascript 객체의 생성자와 객체의 재정의에 관련된 내용이여서 배열을 받는데 크게 사용되진 않는다. 

 

위의 JScript 예제에서

 

alert(arrArray);

 

라고 하면 DISPID 0값이 나오는 것으로 확인할 수 있겠다.

 

실제 작성한 코드는 VARIANT 형식으로 배열(JScript와 VBScript 상관없이) 또는 한개의 문자열을 받는 메소드 이지만 그중에서 JScript 처리 부분만 간추려서 올린다.
반응형

+ Recent posts