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

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

ただいまの
回答率

89.21%

他のアプリで使用中のフォント名を正しく取得できない

解決済

回答 1

投稿

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

mery

score 22

前提・実現したいこと

windowsのメモ帳で使用中のフォントの名前をプログラムから取得しようとして試してみたのですがうまくいきません。
メモ帳で使っているフォントは「MSゴシック」なのですが、取得されるフォント名は「System」が返されます。
どうすれば正しいフォント名が返されるようにできますか?

該当のソースコード

※メモ帳のフォント名を取得して表示する関数部分
void test(HWND hWnd){
HWND hnotepad;//メモ帳のハンドル保存用
HWND hedit;//メモ帳のエディットボックスのハンドル保存用
TCHAR szfontname[256];
HDC hdc;
hnotepad=FindWindow(TEXT("notepad"),NULL);
if (hnotepad==NULL) {
MessageBox(hWnd,TEXT("Windowsのメモ帳は起動していません"),TEXT("ウィンドウハンドル取得失敗"),MB_OK);
return;
}

hedit=FindWindowEx(hnotepad,NULL,TEXT("EDIT"),NULL);
hdc=GetDC(hedit);//メモ帳のエディットボックスのデバイスコンテキスト取得

GetTextFace(hdc,sizeof (szfontname) / sizeof(TCHAR),szfontname);
MessageBox(hWnd,szfontname,TEXT("フォント名"),MB_OK);
DeleteDC(hdc);
}

補足情報(FW/ツールのバージョンなど)

Microsoft Visual C++ 2010 Express C言語
WIN32 ユニコードビルド Windows7

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

質問への追記・修正、ベストアンサー選択の依頼

  • wwbQzhMkhhgEmhU

    2019/03/02 15:09

    他のプロセス、つまりメモ帳で使っているデバイスコンテキストは見れないと思います。
    ここでGetDC()してるのは、当該プロセスの(単一スレッドで)エディットコントロールのウィンドウに直書きするための新しいデバイスコンテキストなので、デフォルトフォントのSystemになっているのではないかと思います。
    私の認識ってだけなので、確実な情報についてはMicrosoftさんのSubscriptionを購入してMicrosoftさんに聞いてください。

    キャンセル

  • mery

    2019/03/02 19:57

    回答ありがとうございます。メモ帳のデバイスコンテキストを見れないのであれば、フォント名の取得にデバイスコンテキストではなくコントロールのハンドルを渡す関数があればできると考えて探したのですが見つかりません。そういう関数は無いでしょうか?

    キャンセル

  • wwbQzhMkhhgEmhU

    2019/03/02 23:44

    お力になれずにすみません。でもいい回答が入ったようです。

    キャンセル

回答 1

checkベストアンサー

+2

GetDC が返すウィンドウに関連付けられたデバイスコンテキストには共通、クラス、プライベートの 3 種類があります。それぞれ何が返されるかは RegisterClass(Ex) で渡すクラススタイルの設定によって決まります。

CS_CLASSDC ・・・クラス
CS_OWNDC・・・プライベート
CS_PARENTDC・・・親ウィンドウの設定による
未指定・・・共通

このうち、WM_SETFONT で設定されたフォントがそのまま帰ってくる可能性があるのはクラスとプライベートのもの(ただし、これは以前のフォント設定が維持されるだけであり WM_SETFONT で指定したフォントとは限らない)だけであり、他のスタイルが指定されている場合、GetDC で取得したデバイスコンテキストにはフォントは未設定のままとなります。エディットボックス等の標準コントロールは CS_PARENTDC が指定されているため、GetDC で取得したデバイスコンテキストにはテキストボックスに WM_SETFONT で指定されたフォントが設定されません。


さらに別プロセスに関する問題もあり、フォントのハンドルを取得できたとしても別プロセスからはフォントの名前を取得することができません。そのため他のアプリで使用中のフォント名を調べるにはグローバルフックで WM_SETFONT メッセージを監視するか、 DLL インジェクション等の手法で該当プロセス内に処理を送り込み WM_GETFONT を直接対象のウィンドウに SendMessage してフォントを取得するかぐらいの方法しかとることはできません。


同一プロセス内であればフォントハンドルからフォント名を取得するのには GetObject(https://docs.microsoft.com/en-us/windows/desktop/api/wingdi/nf-wingdi-getobject)が使用できます。非常に残念ですが、別プロセスでは動作しません。


思い込みはよくありませんね。他プロセスのコントロールでも WM_GETFONT でフォントハンドルを取得し、GetObject(https://docs.microsoft.com/en-us/windows/desktop/api/wingdi/nf-wingdi-getobject)でフォント名を取得することができました。

void test(HWND hWnd) {
    HWND hnotepad;//メモ帳のハンドル保存用
    HWND hedit;//メモ帳のエディットボックスのハンドル保存用
    hnotepad = FindWindow(TEXT("notepad"), NULL);
    if (hnotepad == NULL) {
        MessageBox(hWnd, TEXT("Windowsのメモ帳は起動していません"), TEXT("ウィンドウハンドル取得失敗"), MB_OK);
        return;
    }

    hedit = FindWindowEx(hnotepad, NULL, TEXT("EDIT"), NULL);

    HANDLE h = (HANDLE)SendMessage(hedit, WM_GETFONT, 0, 0);
    LOGFONT lf;
    GetObject(h, sizeof(lf), &lf);
    MessageBox(hWnd, lf.lfFaceName, TEXT("フォント名"), MB_OK);
}


Windows 10 64ビット環境で上記コードが動作しました。普通に使えそうな気はしますね。

追加でいろいろ調査してみましたが、このフォントオブジェクトは元のプロセス(メモ帳)と関連づけられているような動作になっていますね。描画には使用可能(SelectObjectできる)でしたが、削除には失敗(DeleteObjectは失敗)しました。

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2019/03/02 23:43

    WM_GETFONTですか。使ったことなくて知らなかったです。確かにメッセージ経由なら取れそうですが、そんなメッセージが存在するとは考えもしなかったです。
    挙動も面白いし、勉強になりました。ありがとうございます。

    キャンセル

  • 2019/03/03 08:56

    回答ありがとうございました。 WM_GETFONTでフォントハンドルを取得するまでは私も試したのですが、その後がどうしてもうまくいかなくて半分諦めていたのですが、GetObjectを使うことでフォント名を正常に取得できました。フォント名はGetTextFace関数で取得するものだと思い込んでいたのでGetObject関数を使うことは全然思いつかなかったです。
    具体的なコードまで示していただき大変助かりました。

    キャンセル

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

  • ただいまの回答率 89.21%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる