質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.48%

  • C

    4641questions

    C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

  • Windows

    1730questions

    Windowsは、マイクロソフト社が開発したオペレーティングシステムです。当初は、MS-DOSに変わるOSとして開発されました。 GUIを採用し、主にインテル系のCPUを搭載したコンピューターで動作します。Windows系OSのシェアは、90%を超えるといわれています。 パソコン用以外に、POSシステムやスマートフォンなどの携帯端末用、サーバ用のOSもあります。

  • Win32 API

    297questions

    Win32 APIはMicrosoft Windowsの32bitプロセッサのOSで動作するAPIです。

  • DLL

    110questions

    DLL(Dynamic Link Library)とは、他のモジュールからも使用する事が出来る、関数とデータが格納されているモジュールのことです。

  • COM

    46questions

    COM(Component Object Model)はMicrosoftによるコンポーネントテクノロジーであり、 ソフトウェアの再利用を目的とした技術を指します。

Vista+のIFileDialogをXPで使えるようにしたい

解決済

回答 1

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 277

katahiromz

score 75

こんにちは、コンピュータ研究家の片山博文MZです。

今、vista2xpという下位互換性のためのツールを作っています。
https://github.com/katahiromz/vista2xp

これは、Vista用のアプリをXP用に変換するためのツールです。

API関数群についてはだいたい出来たのですが、VistaのIFileDialogをXPでハイジャックして再現する方法が分かりません。

大まかにやり方を教えて下さい。よろしくお願い致します。

(追記)
テストプログラムを書きました。

(追記×2)

  • IFileDialogがすでに実装されているか確認する方法
  • IFileDialogが実装されていなければ、システムに実装を追加する方法

の二つをお願いします。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

checkベストアンサー

+2

IFileDialogがすでに実装されているか確認する方法

CLSID_FileOpenDialog または CLSID_FileSaveDialog で実際に CoCreateInstance してみて REGDB_E_CLASSNOTREG が返されれば未実装と判断してしまえば良いかと思います。


IFileDialogが実装されていなければ、システムに実装を追加する方法

一般的に以下に挙げる 3 種類方法があります。

1.レジストリに登録する。

インストールされていない前提であれば上記の CLSID を持った DLL を通常の COM と同様にレジストリ登録するのが最も単純で良いかと思います。

2.クラスファクトリを登録する。

CoRegisterClassObject で独自に作成したコンポーネントのクラスファクトリをスレッド毎に登録することでも実現可能です。レジストリに登録されている設定よりも優先されるので、実は一番お勧めしたいのですが、アパートメント毎ではなくスレッド毎に CoRegisterClassObject を呼び出す必要があるので、意外と手間がかかります。DllMain 内では KERNEL32 の処理しか呼んではいけないことになっているので、DLL_THREAD_ATTACH で登録するというのは避けた方が良いかなとは思います。

ドキュメントには記載がありませんが、DLL_THREAD_ATTACH で CoRegisterInitializeSpy で IInitializeSpy をインストールして、IInitializeSpy::PostInitialize メソッドで CoRegisterClassObject するとうまくいくと思います。CoInitialize はウィンドウ作成したりするので、DLL_THREAD_ATTACH 内で使うとハングすることがままありますが、CoRegisterInitializeSpy で遅延させれば回避可能かと思います。
ちょうどスタックオーバーフローで同じ回答を見つけました。
https://stackoverflow.com/questions/24993611/define-a-com-class-that-is-only-createable-from-the-same-exe-where-it-is-registe
スタックオーバーフローの文面ではアパートメント毎にという記載がありますが、CoRegisterClassObject はスレッド毎に必要であったと記憶しています。ただし、今回は GUI を持ったコンポーネントを登録するので、MTA ではないので違いは出ません。

3.API をフックする。

CoGetClassObject、CoCreateInstance、CoCreateInstanceEx をフックして該当の CLSID が指定されている場合に自分で作成したコンポーネント(or クラスファクトリ)を返すようにしてみてください。これはレジストリ登録が不要なので次にお勧めです。 2番の実現性が高くなったのでお勧めとしては3番目にします。

なお、XP を前提としない場合、CoCreateInstanceFromApp もフック対象となります。また、ProgID からの変換が必要である場合、CLSIDFromProgID および CLSIDFromProgIDEx もフック対象となります。今回のファイルダイアログに関しては上記は不要です。

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2019/05/07 20:10

    インポートテーブルを用いたAPIフックにより解決しました。
    ありがとうございました。
    報酬アマゾンギフト券1万円でどうでしょうか。メールでご連絡下さい。
    katayama.hirofumi.mz@gmail.com

    キャンセル

  • 2019/05/08 02:47

    返答に困りましたが、私の回答に対して謝意を示していただけるのであれば、以下の質問を進めていただくことが私にとっては一番ありがたいことです。
    https://teratail.com/questions/184810
    質問への追記・修正の依頼に対して全く返答しないのは礼を失しています。そのために回答しにくくなっているのではないでしょうか?

    キャンセル

同じタグがついた質問を見る

  • C

    4641questions

    C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

  • Windows

    1730questions

    Windowsは、マイクロソフト社が開発したオペレーティングシステムです。当初は、MS-DOSに変わるOSとして開発されました。 GUIを採用し、主にインテル系のCPUを搭載したコンピューターで動作します。Windows系OSのシェアは、90%を超えるといわれています。 パソコン用以外に、POSシステムやスマートフォンなどの携帯端末用、サーバ用のOSもあります。

  • Win32 API

    297questions

    Win32 APIはMicrosoft Windowsの32bitプロセッサのOSで動作するAPIです。

  • DLL

    110questions

    DLL(Dynamic Link Library)とは、他のモジュールからも使用する事が出来る、関数とデータが格納されているモジュールのことです。

  • COM

    46questions

    COM(Component Object Model)はMicrosoftによるコンポーネントテクノロジーであり、 ソフトウェアの再利用を目的とした技術を指します。