Excel VBA質問箱 IV

当質問箱は、有志のボランティア精神のおかげで成り立っています。
問題が解決したら、必ずお礼をしましょうね。
本サイトの基本方針をまとめました。こちら をご一読ください。

投稿種別の選択が必要です。ご注意ください。
迷惑投稿防止のため、URLの入力を制限しています。ご了承ください。


4480 / 13644 ツリー ←次へ | 前へ→

【56128】プリンタの電源確認について savex 08/6/3(火) 17:19 質問[未読]
【56179】Re:プリンタの電源確認について 熊谷隆史 08/6/7(土) 18:05 発言[未読]
【56266】Re:プリンタの電源確認について yasu 08/6/12(木) 11:21 発言[未読]
【56267】Re:プリンタの電源確認について savex 08/6/12(木) 11:59 お礼[未読]
【56302】Re:プリンタの電源確認について 熊谷隆史 08/6/12(木) 17:35 発言[未読]
【56345】Re:プリンタの電源確認について 熊谷隆史 08/6/14(土) 10:39 回答[未読]
【56362】Re:プリンタの電源確認について 熊谷隆史 08/6/15(日) 10:55 回答[未読]

【56128】プリンタの電源確認について
質問  savex  - 08/6/3(火) 17:19 -

引用なし
パスワード
   いつもお世話になっております。

プリンタの電源のON/OFF(又はUSBが外れている、ネットワークにつながっていないなど)を確認したいのですがどうしても方法がわかりません。

接続して動作しているかは、APIを使えば取得できますが、
ON/OFF自体は確認できませんでした。
反応がないかでしか判断できないのでしょうか。

もし、ご存知の方がおりましたら教えて頂きたいです。
よろしくお願い致します。

【56179】Re:プリンタの電源確認について
発言  熊谷隆史  - 08/6/7(土) 18:05 -

引用なし
パスワード
   ▼savex さん:
要するにプリンタのオンライン/オフラインを
検出したいと言うことなのでしょうけど、
GetPrinter APIで取れる/取れないとGoogle検索しても
イマイチはっきりしませんね。

他にはユーザーモードデバッガ(OllyDbg)で
プリンタドライバの動作を追いかけて
スタック領域にプッシュ(push)される
DeviceIoControl APIの引数を調査するとか
簡単でない方法もあるかなと。
(私は試したことがないので、よく分かりませんが)

【56266】Re:プリンタの電源確認について
発言  yasu  - 08/6/12(木) 11:21 -

引用なし
パスワード
   savex さん

質問はプリンタステータスが分かればよろしいのでしょうか?
VBAとは違いますが、参考になれば…。

ネットワークプリンタということなら、ping打てばreplyが返ってきますので、
on/offの確認が取れると思います。
また機種にもよりますが、IEのアドレスにプリンタのIPを入力すればGUIが
開けるので、そこでは設定やon/offが確認できます。

ローカルPCでUSBなどが外れてるとかはネットワークプリンタではないので、
遠くに設置しているわけではないと思うので、直接見た方が早そうな気がします。

【56267】Re:プリンタの電源確認について
お礼  savex  - 08/6/12(木) 11:59 -

引用なし
パスワード
   返事が遅くなりまして、大変申し訳ございません。
熊谷隆史さん、yasu さんご回答ありがとうございます!

▼熊谷隆史 さん:
>要するにプリンタのオンライン/オフラインを
>検出したいと言うことなのでしょうけど、
>GetPrinter APIで取れる/取れないとGoogle検索しても
>イマイチはっきりしませんね。

>他にはユーザーモードデバッガ(OllyDbg)で
>プリンタドライバの動作を追いかけて
>スタック領域にプッシュ(push)される
>DeviceIoControl APIの引数を調査するとか
>簡単でない方法もあるかなと。
> (私は試したことがないので、よく分かりませんが)
う-ん。やはり簡単な問題ではないですよね。
まだまだ無知な私にはレベルが高すぎるようです;


▼yasu さん:
>質問はプリンタステータスが分かればよろしいのでしょうか?
>VBAとは違いますが、参考になれば…。
>
>ネットワークプリンタということなら、ping打てばreplyが返ってきますので、
>on/offの確認が取れると思います。
>また機種にもよりますが、IEのアドレスにプリンタのIPを入力すればGUIが
>開けるので、そこでは設定やon/offが確認できます。

>ローカルPCでUSBなどが外れてるとかはネットワークプリンタではないので、
>遠くに設置しているわけではないと思うので、直接見た方が早そうな気がします。
確かに直接見たほうが早いですよね^^;
通常、この問題は運用するユーザに注意してもらっていたので、
とりあえずはこれからもそうしようと思います。


お二人の助言を参考にいろいろ考えてみます。
貴重なご意見ありがとうございました!

【56302】Re:プリンタの電源確認について
発言  熊谷隆史  - 08/6/12(木) 17:35 -

引用なし
パスワード
   ▼savex さん:
> 確かに直接見たほうが早いですよね^^;
> 通常、この問題は運用するユーザに注意してもらっていたので、
> とりあえずはこれからもそうしようと思います。

解決なのか未解決なのかよく分かりませんが、
少しはあがいてみた方がよろしいのでは。

www.google.co.jp/search?q=cache:www2.moug.net/bbs/exvba/20080220000020.htm
vsug.jp/tabid/63/forumid/72/postid/5482/view/topic/Default.aspx
www.google.co.jp/search?q=cache:moug.net/faq/viewtopic.php?t=15228

【56345】Re:プリンタの電源確認について
回答  熊谷隆史  - 08/6/14(土) 10:39 -

引用なし
パスワード
   ▼savex さん:
# もうご覧になっていないかも知れませんが。

結局、GetPrinterにしても、WMIにしても、
こちらの環境でも上手く取れないので、
取り合えず、コントロールパネルの
プリンタウィンドウ(画面)から取得するのが
まあ、直感的かなと。

EnumWindows。EnumChildWindowsで
プリンタウィンドウの起動待ちを一応、兼ねています。

FAQ|Excel (VBA)
moug.net/faq/viewforum.php?f=2
でかなり前に載せられたshiraさんのコードを
お借りしてます(IAccessible関連)。

こちらも参考。
www.roy.hi-ho.ne.jp/mutaguchi/wsh/technic.htm


では。


Option Explicit
'標準モジュール
Private Declare Function EnumWindows Lib "user32" _
    (ByVal lpEnumFunc As Long, ByVal lParam As Long) As Long
Private Declare Function EnumChildWindows Lib "user32" _
    (ByVal hWndParent As Long, ByVal lpEnumFunc As Long, _
    ByVal lParam As Long) As Long
Private Declare Function GetWindowText Lib "user32" _
    Alias "GetWindowTextA" _
    (ByVal hwnd As Long, ByVal lpString As String, _
    ByVal nMaxCount As Long) As Long
Private Declare Function GetClassName Lib "user32" _
    Alias "GetClassNameA" _
    (ByVal hwnd As Long, ByVal lpClassName As String, _
    ByVal nMaxCount As Long) As Long
Private Declare Function AccessibleObjectFromWindow Lib "oleacc" _
    (ByVal hwnd As Long, ByVal dwId As Long, _
    riid As Any, ppvObject As Any) As Long
Const OBJID_CLIENT = &HFFFFFFFC
Private Declare Function IIDFromString Lib "ole32" _
    (lpsz As Any, lpiid As Any) As Long
Const IID_IAccessible = "{618736E0-3C3D-11CF-810C-00AA00389B71}"
Const WM_CLOSE = &H10
Private Declare Function PostMessage Lib "user32" _
    Alias "PostMessageA" _
    (ByVal hwnd As Long, ByVal Msg As Long, _
    ByVal wParam As Long, ByVal lParam As Long) As Long
Private Declare Function IsWindow Lib "user32" (ByVal hwnd As Long) As Long
Private h As Long
Private hSysListView32 As Long

Private Function EnumWindowsProc(ByVal hwnd As Long, _
ByVal lParam As Long) As Long
  
  Dim ClassName As String * 128
  Dim WindowText As String * 128
  GetClassName hwnd, ClassName, Len(ClassName)
  GetWindowText hwnd, WindowText, Len(WindowText)
  If WindowText Like "プリンタ*" Then
    If ClassName Like "CabinetWClass*" Then
      h = hwnd
      EnumChildWindows hwnd, AddressOf EnumChildProc, 0
      EnumWindowsProc = 0
    End If
  End If
  EnumWindowsProc = 1
End Function

Private Function EnumChildProc(ByVal hwnd As Long, _
ByVal lParam As Long) As Long

  Dim ClassName As String * 128
  Dim WindowText As String * 128
  GetClassName hwnd, ClassName, Len(ClassName)
  GetWindowText hwnd, WindowText, Len(WindowText)
  
  If ClassName Like "SysListView32*" Then
    hSysListView32 = hwnd
    EnumChildProc = 0
   End If
   EnumChildProc = 1
End Function

' こちらを実行
Sub test()
  Dim IID(0 To 3) As Long
  Dim acc As IAccessible
  Dim i As Long
  Dim obj As Object
  h = 0
  hSysListView32 = 0
  For Each obj In CreateObject("Shell.Application").Namespace(3).Items
    If obj.Name = "プリンタ" Then
      obj.InvokeVerb
    End If
  Next
  EnumWindows AddressOf EnumWindowsProc, 0

  ' ウィンドウからIAccessibleを取り出す
  IIDFromString ByVal StrPtr(IID_IAccessible), IID(0)
  If AccessibleObjectFromWindow( _
        hSysListView32, OBJID_CLIENT, IID(0), acc) < 0 Then
    Exit Sub  ' エラー時
  End If

  'IAccessibleベースのオブジェクト作成待ち(適当)
  'これに気付かなくてしばらくハマッた。
  'この時の内部動作を調べれば確実に
  'オンライン/オフラインを判断できるのではと。
  Application.Wait Now() + TimeValue("00:00:01")

  For i = 1 To acc.accChildCount - 1
    Debug.Print acc.accName(i) 'プリンタ名
    Debug.Print acc.accDescription(i) 'OnLine/OffLine確認
  Next
  Set acc = Nothing
  'プリンタウィンドウを閉じる
  PostMessage h, WM_CLOSE, 0, 0
  
  'プリンタウィンドウが閉じるのを待つ
  Do
    DoEvents
  Loop While IsWindow(h) = 0
  
End Sub

【56362】Re:プリンタの電源確認について
回答  熊谷隆史  - 08/6/15(日) 10:55 -

引用なし
パスワード
   >   'プリンタウィンドウが閉じるのを待つ
>   Do
>     DoEvents
>   Loop While IsWindow(h) = 0

        ↓
  'プリンタウィンドウが閉じるのを待つ
  Do
    DoEvents
  Loop Until IsWindow(h) = 0        


>   'IAccessibleベースのオブジェクト作成待ち(適当)
>   'これに気付かなくてしばらくハマッた。
>   'この時の内部動作を調べれば確実に
>   'オンライン/オフラインを判断できるのではと。
>   Application.Wait Now() + TimeValue("00:00:01")

            ↓
   Application.Wait Now() + TimeValue("00:00:02")

4480 / 13644 ツリー ←次へ | 前へ→
ページ:  ┃  記事番号:
2610219
(SS)C-BOARD v3.8 is Free