Life Goes On

締め切りが近い仕事があるのに、今日、月イチの通院日で、パスしようかと思ったんですが・・・薬切れかけで不安なので雨の中午前中某病院へ。いつもは最低30分は待たされるんですが、今日は診察券出して2~3分で名前をコール。今日はラッキーでした(^^

で、これです。リマスタリング版!

The Beatles 新ボックスセット

The Beatles 新ボックスセット

と、いっても、コレ、僕のものじゃなくて・・・(^^;;; Amazonで買おうかどうか迷っているって話してたら、「もう買ったよ」という上司がいて、「貸したるわ」ってなことで、借りました(^^

これは、リマスタリングしたとのことですが・・・ショボイ、オーディオ機器ではやっぱり違いがわかりません(笑) しかし、このボックスセットは、欲しくなりました。古き良き?レコードを思いださせるような、CDレーベル面とか装丁とか、なかなか。

早速アマゾンで予約しよ。

残り5GB

昨日撮った画像データをHDDに転送したら、500GBのHDDの残りがたったの5GBに・・・。警告が出るまで気付かなかった(^^;;;

で、夕方に急遽神戸に・・・いったけど、欲しいものがなくて、結局大阪のヨドバシまで脚を伸ばしました。分かってれば昨日出たついでに買ったんだけどね・・・

どんどん外付けのハードディスクが増えていくな・・・。

昨日のん、もう一枚上げときます。カビ防止のため?かなりひさしぶりに70-300Gをもちだして、強引に使ってきました(^^;

新井涼子さん

新井涼子さん

サクサク軽快~!

「サクサク軽快~!パソコン操作も、サクサク軽快~ ○○○を選んだあなたは大正解~♪」

という、とあるセキュリティ対策ソフトウェアのCMが気になってしょうがないです(^^;;; そろそろなんか、セキュリティソフトを入れないといけないんだけど・・・。

どうでもいいんだけど・・・、昔、とある電器屋のソフト売り場で、ウイルス対策ソフトのことを、なんども「このウイルスソフトは検出率が高くて・・・~」と言って商品説明してた、店頭販売のお姉さんがいて、「え?」と苦笑した覚えがあります・・・。

で、

先日から、会社の何年かに一度の莫大な予算(といっても大企業に比べるとささやかなもんだけど・・・)をかけた、催し物みたいなものの、告知ページ(ホームページの)の作成をしているんですが・・・それより気がかりなのが、打ち合わせのときに何気に見せてもらった人員配置表みたいな用紙に、僕の名前が・・・「会場カメラマン」・・・って・・・(笑)  カメコ病でコンパニオンのお姉さんばっかり写してたりしてたら、さすがに怒られる・・・というか、次の日から、白い目でみられますね(^^;;;

かなり有名な人(スポーツ系)が講演にくる予定(まだ分かんないけど)なので、ちょっと楽しみです。

今年の夏も終わったね

朝晩が涼しくなってきました。今年はエルニーニョのせいか、日本特有のジメジメした暑さが少なかったような・・・?今年の「海の家」は繁盛してたんでしょうか?

そういえば、今年は、浴衣美女を撮ってないなぁ~、とちと後悔しつつ、「さよなら夏の日」を聴きながら作業してます(^^;;;

去年のさあやちゃん。

去年のさあやちゃん。

全然関係ないんですが・・・、先週、三ヶ月ぶりに髪を思いきっり短く刈って、いや、まぁ、毎回そうなんですけど、普段あまり表情が良くなくて短く刈ると余計いかつく見えるらしく、雑踏なんかでちらしとかティッシュ配りの人が避けてくれるんですよね(^^;;; 「受け取らないぞ」オーラが出てんでしょうか。いや、実際うざいので受け取りたくないんだが・・・。

「愛想の良さそうな顔(表情)」を自然にできる人がうらやましい。なんのこっちゃ。

COMサーバーにJScriptのイベントハンドラを登録する

VC++ 2008EE で COMサーバーをカンタンに作成」のつづきです。

C++属性を使ってC++で簡単なCOMサーバーを作成はできました。次に、ブラウザ上でやってるような、イベントを発生させて、それをjscriptの関数、たとえば、

function onExec()
{
  WScript.Echo("メソッドが実行されました");
}

というようなイベントハンドラを登録して、COMサーバーのあるメソッドを実行したときに、このonExec という関数を実行させるようにしたい! と思います。

要は、HTML+Javascriptでやってるような、window.onload = function(){…} というようなことを自分が作成したCOMサーバーにやらせよう、というわけです。実行環境は、Windows Scripting Hostです。

はじめはどうやればさっぱり分かんなかったんですが、ネットで検索していろいろ調べた結果の備忘録となります。

COMサーバーから、各種イベント通知を受け取るには、C++だとIConnectionPointとシンクオブジェクトを経由したものになるんですが、JScriptにはそういった接続ポイントをゴニョゴニョして・・・というやり方はできないようです。つまりCOMでシンクを経由したイベントを受け取る機構は通用しないということです。

ここから補足 (2012/2/10)

正確には、動的にインスタンスを生成した(new ActiveXObject()とかで作成した)オブジェクトでシンクを経由したイベントを受け取る機構がないのです。たとえばHTAとかで

<object id=’xxxx’ classid=”clsid:aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee”></object>

という風にOBJECTタグをHTMLに埋め込んで、function xxxx::関数名(引数){…} というイベントハンドラを定義すれば、可能になります。たぶん。

JScriptをホストしているのがWSHならば、以下のようにイベントハンドラを定義して、シンクを経由した通常のCOMイベントを受け取れます。

var obj = WScript.CreateObject("MyServer.Utility","myserver_");
function myserver_onXXXX()
{
   WScript.Echo("Fire!");
}
obj.XXXX();

しかし、Webブラウザ上で、たとえば・・・

var button = document.getElementById('button');
button.onclick = function()
{
  window.alert("クリック!") ;
}

というようにもしたいわけです。

var myserver = WScript.CreateObject('MyServer.Utility');
myserver.onExec = function()
{
  WScript.Echo("実行");
}

とこんな感じにもできたらなぁ・・・と思ったのが今回の動機です。

単純に考えれば、COMサーバー側にonExecというプロパティを設置して、COMサーバーのあるメソッドが実行されたら、このonExecを実行させる(イベントを発生)させればいいだけかな、と思いました。function(){}の関数オブジェクトの部分。関数オブジェクトはC++で、IDispatch*で受け取ればいいですね。

ということで、いろいろネットで調べた結果、次のような単純なCOMサーバーを作成しました。

  • CurrentDirectory() というメソッドを一つ定義。
  • 関数オブジェクトの保持用にプロパティを一つ定義
  • CurrentDirectory()がコールされたら、プロパティに格納している関数オブジェクトを実行する。
    (つまりイベントを発生させる)

具体的なコードは・・・

/*
 コンパイル・コマンドライン :
   cl /LD MyServer.cpp /link /TLBOUT:MyServer.tlb
*/

#define STRICT
#ifndef _WIN32_WINNT
#define _WIN32_WINNT 0x0400
#endif
#define _ATL_ATTRIBUTES
#define _ATL_APARTMENT_THREADED
#define _ATL_NO_AUTOMATIC_NAMESPACE
#include <atlbase.h>
#include <atlcom.h>
#include <atlwin.h>
#include <atltypes.h>
#include <atlctl.h>
#include <atlhost.h>
using namespace ATL;

//マクロ定義
#define PPVOID(X) (reinterpret_cast<LPVOID*>(X))
#define RELEASE(X) (X ? X->Release() : 0)

// DllMain,DllCanUnloadNow,DllRegisterServer and DllUnregisterServer
[ module(dll, name = "MyServer", helpstring = "MyServer 1.0 Type Library") ];
[ emitidl ];

/////////////////////////////////////////////////////////////////////////////
// インターフェイス定義  :   IMyServer 
[
   object,
   dual,
   oleautomation,
   helpstring("IMyServer Interface"),
   pointer_default(unique)
]
__interface IMyServer : IDispatch
{
  [id(1)] HRESULT CurrentDirectory([out,retval]BSTR *sPath);

  [id(2),propput] HRESULT onCurrent([in] LPDISPATCH pDisp);
  [id(2),propget] HRESULT onCurrent([out,retval] LPDISPATCH *ppDisp);
};

/////////////////////////////////////////////////////////////////////////////
// イベントインターフェイス定義 : IMyServerEvent
[
  dispinterface,
  hidden,
  helpstring("IMyServerEvents Interface")
]
__interface IMyServerEvent
{
  [id(1)] HRESULT onCurrent();
};

/////////////////////////////////////////////////////////////////////////////
// インターフェイス実装クラス  : CMyServer
[
   coclass,
   threading(apartment),
   source(IMyServerEvent),
   event_source(com),
   vi_progid("MyServer.Utility"),
   progid("MyServer.Utility.1"),
   version(1.0),
   helpstring("MyServer Class")
]
class ATL_NO_VTABLE CMyServer :
   public IMyServer
{
public:
   CMyServer()
     {

     }

  //COMイベントのためのおまじない。
  __event __interface IMyServerEvent;

  DECLARE_PROTECT_FINAL_CONSTRUCT()
  HRESULT FinalConstruct()
    {
      //プライベート・メンバ初期化
      m_pDispatch = NULL;
      return S_OK;
    }

  void FinalRelease()
    {
      //後処理
      RELEASE(m_pDispatch);
      return;
    }

  //単にカレントディレクトリを返す。
  HRESULT CurrentDirectory(BSTR *sPath)
    {
      WCHAR pBuffer[MAX_PATH+1] = {0};
      ::GetCurrentDirectoryW(MAX_PATH+1,pBuffer);

      *sPath = ::SysAllocString(pBuffer);

      Fire();
      return S_OK;
    }

  // IDispatch*(スクリプトから渡されるオブジェクト)の設定と取得
  // スクリプトでいう、obj.onCurrent = function(){...} の部分
  HRESULT put_onCurrent(LPDISPATCH pDispatch)
    {
      LPDISPATCH pDispatchNew = NULL;
      HRESULT hRes = S_OK;

      if(SUCCEEDED(hRes =
           pDispatch->QueryInterface(IID_IDispatch,PPVOID(&pDispatchNew))))
        {
          RELEASE(m_pDispatch);
          m_pDispatch = pDispatchNew;
        }

      return hRes;
    }

  //関数オブジェクトの取得
  HRESULT get_onCurrent(LPDISPATCH *ppDisp)
    {
      HRESULT hRes = E_NOTIMPL;

      if(m_pDispatch)
        hRes = m_pDispatch->QueryInterface(IID_IDispatch,PPVOID(ppDisp));

      return hRes;
    }

private:

  LPDISPATCH m_pDispatch;

  void Fire()
    {
      DISPPARAMS dispparams = {NULL,NULL,0,0};

      //COMイベント発火
      IMyServerEvent_onCurrent();

      //設定された関数オブジェクトを実行,デフォルトDISPID(0)
      m_pDispatch->Invoke((DISPID)DISPID_VALUE,
                          IID_NULL,
                          LOCALE_USER_DEFAULT,
                          DISPATCH_METHOD,
                          &dispparams,
                          NULL,
                          NULL,
                          NULL);
    }
};

一応、シンクオブジェクト経由でもイベントを受け取れるようにもしました。仕組みが分かれば意外と簡単です。以前にも書きましたが、ほとんどサンプルの流用で、ATLを使った部分や属性部分はサンプルそのまま。

ただ、このテストサンプルは複数のイベントハンドラを登録できません。複数のイベントハンドラを登録できるようにするには、複数のIDispatch*ポインタを管理しなければなりません。内部でstd::vectorなどのポインタ配列用のコンテナ(COMだったらsafe arrayあたり?)を用意して・・・というような処理が必要になってくるでしょう。

できたMyServer.dllを regsvr32 で登録して、以下のJScriptで動作確認。

//シンクを経由した、イベントハンドラ
//WSHホストで動作可能です。
function myserver_onCurrent()
{
  WScript.Echo("Sync Event Occured!");
}

try
{
  //IEがホストなら・・・
  // var obj = new ActiveXObject("MyServer.Utility");
  // WScript.Echo は window.alert になります。
  var obj = WScript.CreateObject("MyServer.Utility","myserver_");

  obj.onCurrent = function()
    {
      WScript.Echo("Fire Current!");
    }

  WScript.Echo(obj.CurrentDirectory());
}
catch(e)
{
	WScript.Echo(e.message);
}

というわけで、備忘録終了。