1
0
Fork 0
mirror of https://github.com/ruby/ruby.git synced 2022-11-09 12:17:21 -05:00

WIN32OLE.codepage= accepts installed codepage.

git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@11550 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
suke 2007-01-20 12:12:36 +00:00
parent 75ce70d353
commit 3b5dbc4b06
3 changed files with 67 additions and 20 deletions

View file

@ -1,3 +1,10 @@
Sat Jan 20 21:05:18 2007 Masaki Suketa <masaki.suketa@nifty.ne.jp>
* ext/win32ole/win32ole.c (fole_s_set_code_page): WIN32OLE.codepage=
accepts installed codepage.
* test/win32ole/test_win32ole.rb (test_s_codepage_changed): ditto.
Sat Jan 20 11:18:49 2007 Masaki Suketa <masaki.suketa@nifty.ne.jp> Sat Jan 20 11:18:49 2007 Masaki Suketa <masaki.suketa@nifty.ne.jp>
* ext/win32ole/win32ole.c (ole_invoke, ole_invoke2, ole_propertyput): * ext/win32ole/win32ole.c (ole_invoke, ole_invoke2, ole_propertyput):

View file

@ -23,6 +23,7 @@
#include <ocidl.h> #include <ocidl.h>
#include <olectl.h> #include <olectl.h>
#include <ole2.h> #include <ole2.h>
#include <stdlib.h>
#ifdef HAVE_STDARG_PROTOTYPES #ifdef HAVE_STDARG_PROTOTYPES
#include <stdarg.h> #include <stdarg.h>
#define va_init_list(a,b) va_start(a,b) #define va_init_list(a,b) va_start(a,b)
@ -62,7 +63,7 @@
#define OLE_RELEASE_TYPEATTR(X, Y) ((X)->lpVtbl->ReleaseTypeAttr((X), (Y))) #define OLE_RELEASE_TYPEATTR(X, Y) ((X)->lpVtbl->ReleaseTypeAttr((X), (Y)))
#define OLE_FREE(x) {\ #define OLE_FREE(x) {\
if(gOLEInitialized == Qtrue) {\ if(gOLEInitialized == TRUE) {\
if(x) {\ if(x) {\
OLE_RELEASE(x);\ OLE_RELEASE(x);\
(x) = 0;\ (x) = 0;\
@ -79,13 +80,14 @@
#define WC2VSTR(x) ole_wc2vstr((x), TRUE) #define WC2VSTR(x) ole_wc2vstr((x), TRUE)
#define WIN32OLE_VERSION "0.8.6" #define WIN32OLE_VERSION "0.8.7"
typedef HRESULT (STDAPICALLTYPE FNCOCREATEINSTANCEEX) typedef HRESULT (STDAPICALLTYPE FNCOCREATEINSTANCEEX)
(REFCLSID, IUnknown*, DWORD, COSERVERINFO*, DWORD, MULTI_QI*); (REFCLSID, IUnknown*, DWORD, COSERVERINFO*, DWORD, MULTI_QI*);
typedef HWND (WINAPI FNHTMLHELP)(HWND hwndCaller, LPCSTR pszFile, typedef HWND (WINAPI FNHTMLHELP)(HWND hwndCaller, LPCSTR pszFile,
UINT uCommand, DWORD dwData); UINT uCommand, DWORD dwData);
typedef BOOL (FNENUMSYSEMCODEPAGES) (CODEPAGE_ENUMPROC, DWORD);
typedef struct { typedef struct {
struct IEventSinkVtbl * lpVtbl; struct IEventSinkVtbl * lpVtbl;
} IEventSink, *PEVENTSINK; } IEventSink, *PEVENTSINK;
@ -152,13 +154,15 @@ VALUE cWIN32OLE_PROPERTY;
static VALUE ary_ole_event; static VALUE ary_ole_event;
static ID id_events; static ID id_events;
static BOOL gOLEInitialized = Qfalse; static BOOL gOLEInitialized = FALSE;
static BOOL gCodePageInstalled = FALSE;
static HINSTANCE ghhctrl = NULL; static HINSTANCE ghhctrl = NULL;
static HINSTANCE gole32 = NULL; static HINSTANCE gole32 = NULL;
static FNCOCREATEINSTANCEEX *gCoCreateInstanceEx = NULL; static FNCOCREATEINSTANCEEX *gCoCreateInstanceEx = NULL;
static VALUE com_hash; static VALUE com_hash;
static IDispatchVtbl com_vtbl; static IDispatchVtbl com_vtbl;
static UINT cWIN32OLE_cp = CP_ACP; static UINT cWIN32OLE_cp = CP_ACP;
static UINT gTarget_cp = CP_ACP;
static VARTYPE g_nil_to = VT_ERROR; static VARTYPE g_nil_to = VT_ERROR;
struct oledata { struct oledata {
@ -583,7 +587,7 @@ void
ole_uninitialize() ole_uninitialize()
{ {
OleUninitialize(); OleUninitialize();
gOLEInitialized = Qfalse; gOLEInitialized = FALSE;
} }
static void static void
@ -591,12 +595,12 @@ ole_initialize()
{ {
HRESULT hr; HRESULT hr;
if(gOLEInitialized == Qfalse) { if(gOLEInitialized == FALSE) {
hr = OleInitialize(NULL); hr = OleInitialize(NULL);
if(FAILED(hr)) { if(FAILED(hr)) {
ole_raise(hr, rb_eRuntimeError, "fail: OLE initialize"); ole_raise(hr, rb_eRuntimeError, "fail: OLE initialize");
} }
gOLEInitialized = Qtrue; gOLEInitialized = TRUE;
/* /*
* In some situation, OleUninitialize does not work fine. ;-< * In some situation, OleUninitialize does not work fine. ;-<
*/ */
@ -2103,18 +2107,39 @@ fole_s_get_code_page(VALUE self)
return INT2FIX(cWIN32OLE_cp); return INT2FIX(cWIN32OLE_cp);
} }
static BOOL CALLBACK
installed_code_page_proc(LPTSTR str) {
if (atol(str) == gTarget_cp) {
gCodePageInstalled = TRUE;
return FALSE;
}
return TRUE;
}
static BOOL
code_page_installed(UINT cp) {
gCodePageInstalled = FALSE;
gTarget_cp = cp;
EnumSystemCodePages(installed_code_page_proc, CP_INSTALLED);
return gCodePageInstalled;
}
/* /*
* call-seq: * call-seq:
* WIN32OLE.codepage = CP * WIN32OLE.codepage = CP
* *
* Sets current codepage. * Sets current codepage.
* WIN32OLE.codepage = WIN32OLE::CP_UTF8 * WIN32OLE.codepage = WIN32OLE::CP_UTF8
* WIN32OLE.codepage = 20932
*/ */
static VALUE static VALUE
fole_s_set_code_page(VALUE self, VALUE vcp) fole_s_set_code_page(VALUE self, VALUE vcp)
{ {
UINT cp = FIX2INT(vcp); UINT cp = FIX2INT(vcp);
if (code_page_installed(cp)) {
cWIN32OLE_cp = cp;
} else {
switch(cp) { switch(cp) {
case CP_ACP: case CP_ACP:
case CP_OEMCP: case CP_OEMCP:
@ -2126,10 +2151,10 @@ fole_s_set_code_page(VALUE self, VALUE vcp)
cWIN32OLE_cp = cp; cWIN32OLE_cp = cp;
break; break;
default: default:
rb_raise(eWIN32OLE_RUNTIME_ERROR, "codepage should be WIN32OLE::CP_ACP, WIN32OLE::CP_OEMCP, WIN32OLE::CP_MACCP, WIN32OLE::CP_THREAD_ACP, WIN32OLE::CP_SYMBOL, WIN32OLE::CP_UTF7, WIN32OLE::CP_UTF8"); rb_raise(eWIN32OLE_RUNTIME_ERROR, "codepage should be WIN32OLE::CP_ACP, WIN32OLE::CP_OEMCP, WIN32OLE::CP_MACCP, WIN32OLE::CP_THREAD_ACP, WIN32OLE::CP_SYMBOL, WIN32OLE::CP_UTF7, WIN32OLE::CP_UTF8, or installed codepage.");
break; break;
} }
}
/* /*
* Should this method return old codepage? * Should this method return old codepage?
*/ */

View file

@ -1,5 +1,4 @@
# #
#
begin begin
require 'win32ole' require 'win32ole'
@ -273,6 +272,22 @@ if defined?(WIN32OLE)
str = ifs.read str = ifs.read
} }
assert_equal("\343\201", str) assert_equal("\343\201", str)
# This test fail if codepage 20932 (euc) is not installed.
begin
WIN32OLE.codepage = 20932
rescue WIN32OLERuntimeError
end
if (WIN32OLE.codepage == 20932)
file = fso.opentextfile(fname, 2, true)
file.write [164, 162].pack("c*")
file.close
open(fname) {|ifs|
str = ifs.read
}
assert_equal("\202\240", str)
end
ensure ensure
WIN32OLE.codepage = WIN32OLE::CP_ACP WIN32OLE.codepage = WIN32OLE::CP_ACP
if (File.exist?(fname)) if (File.exist?(fname))