1 // Vistaの効果を有効にするフラグ(不要の場合はコメントアウト)
\r
2 #define USE_VISTA_EFFECTS
\r
5 using System.Runtime.InteropServices;
\r
6 using System.Windows.Forms;
\r
7 using System.Drawing;
\r
9 namespace AppliStation.Util
\r
12 /// Win32ネイティブメソッドを叩いてGUI操作するための関数群のクラス
\r
14 public sealed class NativeMethods
\r
19 private NativeMethods()
\r
24 /// WindowsVista向け、プログレスバーステータス(色)を設定する
\r
26 /// <param name="progBar">対象のプログレスバー</param>
\r
27 /// <param name="state">状態。(1:Normal,2:Error,3:Paused)</param>
\r
28 public static void ProgressBar_SetState(ProgressBar progBar, uint state)
\r
30 #if USE_VISTA_EFFECTS
\r
32 // status := (PBST_NORMAL | PBST_ERROR | PBST_PAUSED)
\r
33 // SendMessage(progressBar.Handle, PBM_SETSTATE, state, 0);
\r
34 SendMessage(progBar.Handle, 0x410, new IntPtr(state), IntPtr.Zero);
\r
35 } catch (Exception) {
\r
40 #region タスクバーおよびタイトルバーのフラッシュ
\r
43 /// タスクバーおよびタイトルバーボタンのフラッシュの設定フラグ
\r
45 public enum FlashFlag : uint {
\r
59 /// タイトルバーとタスクバーボタンを点滅
\r
63 /// Stopが設定されるまで点滅する
\r
67 /// フォアグラウンドの状態になるまで点滅
\r
72 [StructLayout(LayoutKind.Sequential)]
\r
77 public FlashFlag dwFlags;
\r
79 public uint dwTimeout;
\r
83 /// タスクバーおよびタイトルバーボタンを点滅させる
\r
85 /// <param name="form">対象フォーム</param>
\r
86 /// <param name="flag">点滅パラメータフラグ</param>
\r
87 /// <param name="count">点滅回数</param>
\r
88 /// <param name="timeout">点滅の間隔(ミリ秒)</param>
\r
89 /// <returns></returns>
\r
90 public static bool Form_FlashWindow(Form form, FlashFlag flag, uint count, uint timeout)
\r
93 FLASHWINFO info = new FLASHWINFO();
\r
94 info.cbSize = Marshal.SizeOf(typeof(FLASHWINFO));
\r
95 info.hWnd = form.Handle;
\r
96 info.dwFlags = flag;
\r
97 info.uCount = count;
\r
98 info.dwTimeout = timeout;
\r
100 return FlashWindowEx(ref info) == 0;
\r
101 } catch (Exception) {
\r
106 [DllImport("user32.dll")]
\r
107 static extern Int32 FlashWindowEx(ref FLASHWINFO pwfi);
\r
114 /// リストビューにダブルバッファでの描画をするか否かを設定する
\r
116 /// <remarks>マウスでの選択に半透明ツールを採用するか否かもこの設定に依存</remarks>
\r
117 /// <param name="listView">対象のリストビュー</param>
\r
118 /// <param name="bEnable">ダブルバッファでの描画をするとき<code>true</code></param>
\r
119 public static void ListView_SetDoubleBuffer(ListView listView, bool bEnable)
\r
122 // SendMessage(listView.Handle, LVM_SETEXTENDEDLISTVIEWSTYLE, LVS_EX_DOUBLEBUFFER, bEnable? LVS_EX_DOUBLEBUFFER:0);
\r
123 SendMessage(listView.Handle, 0x1036, new IntPtr(0x00010000), new IntPtr((bEnable)? 0x00010000u:0x0u));
\r
124 } catch (Exception) {
\r
129 /// 選択されたアイテムの部分の背景にグラデーションがかかった感じになる、
\r
130 /// Vista以降でのエクスプローラの見た目をListViewに反映させる。
\r
132 /// <remarks>Vista未満のバージョンでは何もしない。</remarks>
\r
133 /// <param name="listView">対象のListView</param>
\r
134 public static void ListView_EnableVistaExplorerTheme(ListView listView)
\r
136 #if USE_VISTA_EFFECTS
\r
138 OperatingSystem os = Environment.OSVersion;
\r
139 if (os.Platform != PlatformID.Win32NT || os.Version.Major < 6) return;
\r
142 SetWindowTheme(listView.Handle, "explorer", null);
\r
143 } catch (Exception) {
\r
149 /// ヘッダに"すべて選択"に似たチェックボックスを作るか否かを指定する
\r
151 /// <remarks>このオプションを設定するとVistaエクスプローラでの「チェックボックスを使用して項目を選択する」と同様の動作になる</remarks>
\r
152 /// <param name="listView">対象のListBox</param>
\r
153 /// <param name="bAutoCheckSelect">チェックボックスを使用して項目を選択するとき<code>true</code></param>
\r
154 public static void ListView_SetAutoCheckSelect(ListView listView, bool bAutoCheckSelect)
\r
156 #if USE_VISTA_EFFECTS
\r
158 // SendMessage(listView.Handle, LVM_SETEXTENDEDLISTVIEWSTYLE, LVS_EX_AUTOCHECKSELECT, bAutoCheckSelect?LVS_EX_AUTOCHECKSELECT:0);
\r
159 SendMessage(listView.Handle, 0x1036, new IntPtr(0x08000000), new IntPtr((bAutoCheckSelect)?0x08000000u:0x0u));
\r
160 } catch (Exception) {
\r
165 #region ColumnHeaderのソートの三角印用
\r
167 [StructLayout(LayoutKind.Sequential, CharSet=CharSet.Auto)]
\r
168 private struct HD_ITEM
\r
172 [MarshalAs(UnmanagedType.LPTStr)]public string pszText;
\r
174 public int cchTextMax;
\r
176 [MarshalAs(UnmanagedType.LPTStr)]public string lParam;
\r
177 public int iImage; // index of bitmap in ImageList
\r
182 /// WinXP以降、ソートの矢印を表示
\r
184 /// <param name="listView">対象のListView</param>
\r
185 /// <param name="column">表示する矢印のヘッダ</param>
\r
186 /// <param name="order">ソートの昇順・降順</param>
\r
187 public static void ColumnHeader_SetSortState(ListView listView, int column, SortOrder order)
\r
190 // SendMessage(hWnd, LVM_GETHEADER, NULL, NULL);
\r
191 IntPtr hWnd = SendMessage(listView.Handle, 0x101F, IntPtr.Zero, IntPtr.Zero);
\r
193 HD_ITEM hdi = new HD_ITEM();
\r
194 hdi.mask = 0x0004; // HDI_FORMAT;
\r
195 for (int i = 0; i < listView.Columns.Count; i++) {
\r
196 // SendMessage(hWnd, HDM_GETITEMW, i, &hdi);
\r
197 SendMessage(hWnd, 0x120b, new IntPtr(i), ref hdi);
\r
199 const int HDF_SORTUP = 0x400;
\r
200 const int HDF_SORTDOWN = 0x200;
\r
202 if (i != column || order == SortOrder.None) {
\r
203 hdi.fmt = hdi.fmt & ~(HDF_SORTUP | HDF_SORTDOWN);
\r
204 } else if (order == SortOrder.Ascending) { // 昇順
\r
205 hdi.fmt = hdi.fmt & ~HDF_SORTDOWN | HDF_SORTUP;
\r
206 } else if (order == SortOrder.Descending) { // 降順
\r
207 hdi.fmt = hdi.fmt & ~HDF_SORTUP | HDF_SORTDOWN;
\r
210 // SendMessage(hWnd, HDM_SETITEMW, i, &hdi);
\r
211 SendMessage(hWnd, 0x120c, new IntPtr(i), ref hdi);
\r
213 } catch (Exception) {
\r
220 /// <param name="listView">ListView</param>
\r
221 /// <returns>ヘッダ部のクライアントサイズ</returns>
\r
222 public static Size ColumnHeader_GetSize(ListView listView)
\r
227 // SendMessage(hWnd, LVM_GETHEADER, NULL, NULL);
\r
228 IntPtr hWnd = SendMessage(listView.Handle, 0x101F, IntPtr.Zero, IntPtr.Zero);
\r
229 GetClientRect(hWnd, out lrect);
\r
231 lrect.Left = lrect.Right = lrect.Top = lrect.Bottom = 0;
\r
234 Size size = new Size(lrect.Right - lrect.Left,
\r
235 lrect.Bottom - lrect.Top);
\r
243 #region EnableWindow(コメントアウト)
\r
245 // /// 指定されたコントロール(ウィンドウ)への、
\r
246 // /// キーボード入力およびマウス入力を有効化または無効化
\r
248 // /// <param name="control">対象のコントロールのハンドラ</param>
\r
249 // /// <param name="bEnable">有効にするか無効にするかを指定</param>
\r
250 // /// <returns>直前にウィンドウが無効状態だった場合はtrueを返す</returns>
\r
251 // public static bool Control_EnableWindow(Control ctrl, bool bEnable)
\r
254 // return EnableWindow(ctrl.Handle, bEnable);
\r
256 // ctrl.Enabled = bEnable;
\r
262 // /// 指定されたコントロール(ウィンドウ)への、
\r
263 // /// キーボード入力およびマウス入力を有効化または無効化
\r
265 // /// <param name="control">対象のコントロールのハンドラ</param>
\r
266 // /// <param name="bEnable">有効にするか無効にするかを指定</param>
\r
267 // /// <returns>直前にウィンドウが無効状態だった場合はtrueを返す</returns>
\r
268 // [DllImport("user32.dll")]
\r
269 // public static extern bool EnableWindow(IntPtr hWnd, bool bEnable);
\r
272 #region RichTextBox関連
\r
274 [StructLayout(LayoutKind.Sequential)]
\r
275 private struct CHARFORMAT2
\r
278 public uint dwMask;
\r
279 public uint dwEffects;
\r
280 public int yHeight;
\r
281 public int yOffset;
\r
282 public int crTextColor;
\r
283 public byte bCharSet;
\r
284 public byte bPitchAndFamily;
\r
285 [MarshalAs(UnmanagedType.ByValArray, SizeConst=32)]
\r
286 public char[] szFaceName;
\r
287 public short wWeight;
\r
288 public short sSpacing;
\r
289 public int crBackColor;
\r
291 public uint dwReserved;
\r
292 public short sStyle;
\r
293 public short wKerning;
\r
294 public byte bUnderlineType;
\r
295 public byte bAnimation;
\r
296 public byte bRevAuthor;
\r
297 public byte bReserved1;
\r
301 /// リッチテキストの選択部分に文字フォーマット効果を与える
\r
303 /// <param name="richTextBox">対象のリッチテキストボックス</param>
\r
304 /// <param name="dwMask">有効な文字フォーマット効果</param>
\r
305 /// <param name="dwEffect">文字フォーマット効果のフラグ</param>
\r
306 public static void RichTextBox_SetSelectionFormat(RichTextBox richTextBox, uint dwMask, uint dwEffect)
\r
308 CHARFORMAT2 cfmt = new CHARFORMAT2();
\r
309 cfmt.cbSize = Marshal.SizeOf(typeof(CHARFORMAT2));
\r
310 cfmt.dwMask = dwMask;
\r
311 cfmt.dwEffects = dwEffect;
\r
313 //SendMessage(richTextBox.Handle, EM_SETCHARFORMAT, SCF_SELECTION, ref cfmt);
\r
314 SendMessage(richTextBox.Handle, 0x0444, new IntPtr(0x0001), ref cfmt);
\r
317 /// リッチテキストにテキストリンクを追加する
\r
319 /// <param name="richTextBox">対象のリッチテキストボックス</param>
\r
320 /// <param name="text">追加するテキスト</param>
\r
321 public static void RichTextBox_AddTextLink(RichTextBox richTextBox, string text)
\r
323 int pos = richTextBox.TextLength;
\r
324 richTextBox.Select(pos,0);
\r
325 richTextBox.SelectedText = text;
\r
326 richTextBox.Select(pos, text.Length);
\r
328 // RichTextBox_SetSelectionFormat(richTextBox, CFM_LINK, CFE_LINK);
\r
329 RichTextBox_SetSelectionFormat(richTextBox, 0x00000020, 0x0020);
\r
331 richTextBox.Select(richTextBox.TextLength, 0);
\r
338 [DllImport("shell32.dll")]
\r
339 private static extern IntPtr ExtractIcon(IntPtr hInst, string lpszExeFileName, int nIconIndex);
\r
341 [DllImport("user32.dll", SetLastError=true)]
\r
342 [return: MarshalAs(UnmanagedType.Bool)]
\r
343 private static extern bool DestroyIcon(IntPtr hIcon);
\r
345 [StructLayout(LayoutKind.Sequential, CharSet=CharSet.Unicode)]
\r
346 private struct SHSTOCKICONINFO
\r
348 public Int32 cbSize;
\r
349 public IntPtr hIcon;
\r
350 public Int32 iSysImageIndex;
\r
351 public Int32 iIcon;
\r
352 [MarshalAs(UnmanagedType.ByValTStr, SizeConst=260)] public string path;
\r
355 [DllImport("shell32.dll", CharSet=CharSet.Unicode, ExactSpelling=true, SetLastError=true)]
\r
356 private static extern void SHGetStockIconInfo(uint siid, uint uFlags, ref SHSTOCKICONINFO sii);
\r
359 /// アイコンファイル(実行ファイル・DLL)を開いてアイコンを作る
\r
361 /// <remarks>内部でコピーされるのでWin32APIのDestroyIconを使わないでいいが、やや遅い動作</remarks>
\r
362 /// <param name="form">ハンドラ</param>
\r
363 /// <param name="lpszExeFileName">対象ファイル</param>
\r
364 /// <param name="nIconIndex">アイコンインデックス</param>
\r
365 /// <returns>生成されたアイコン</returns>
\r
366 public static Icon ExtractIcon(Form form, string lpszExeFileName, uint nIconIndex)
\r
370 if (! System.IO.File.Exists(lpszExeFileName)) {
\r
371 ico = Icon.ExtractAssociatedIcon(lpszExeFileName); // ExtractAssociatedIconに例外を吐いてもらう
\r
373 IntPtr hInst = (form != null)? form.Handle : IntPtr.Zero;
\r
374 IntPtr hIcon = IntPtr.Zero;
\r
377 hIcon = ExtractIcon(hInst, lpszExeFileName, (int) nIconIndex);
\r
378 if ((hIcon != IntPtr.Zero) && (hIcon.ToInt32() != 2)) {
\r
379 ico = (Icon) Icon.FromHandle(hIcon).Clone();
\r
380 DestroyIcon(hIcon);
\r
382 } catch (System.Runtime.InteropServices.COMException) {
\r
383 // ExtraIconのP/Invoke失敗時用
\r
390 /// ユーザ昇格が必要か設定し、設定必要なときシールドアイコンを表示する。
\r
392 /// <param name="button">対象のボタン</param>
\r
393 /// <param name="required">ユーザ昇格が必要か否か、すなわちシールドアイコンを表示するか</param>
\r
394 public static void Button_SetElevationRequiredState(Button button, bool required)
\r
396 #if USE_VISTA_EFFECTS
\r
397 if (Environment.OSVersion.Version.Major >= 6) {
\r
398 button.FlatStyle = FlatStyle.System;
\r
399 // SendMessage(hWnd, BCM_SETSHIELD, 0, required);
\r
400 SendMessage(button.Handle, 0x160C, IntPtr.Zero, new IntPtr((required)? 1u : 0u));
\r
401 } else { // Legacy OS
\r
403 // FlatStyle.System に設定されている場合、Image プロパティに割り当てられているイメージは表示されない対策
\r
404 if (button.FlatStyle == FlatStyle.System) {
\r
405 button.FlatStyle = FlatStyle.Standard;
\r
408 button.TextImageRelation = TextImageRelation.ImageBeforeText;
\r
410 button.ImageAlign = ContentAlignment.MiddleLeft;
\r
411 using (Icon ico = new Icon(SystemIcons.Shield, 16, 16)) {
\r
412 button.Image = ico.ToBitmap();
\r
415 button.Image = null;
\r
417 button.AutoSize = true;
\r
418 #if USE_VISTA_EFFECTS
\r
424 /// シェルのソトックアイコンのハンドラを取得する。
\r
426 /// <param name="siid">ストックID</param>
\r
427 /// <param name="isSmall">アイコンサイズは小さいのであればtrueを指定</param>
\r
428 /// <returns>アイコンのハンドラ。存在しない場合、null</returns>
\r
429 private static IntPtr SHGetStockIconHandle(uint siid, bool isSmall)
\r
431 #if USE_VISTA_EFFECTS
\r
433 SHSTOCKICONINFO sii = new SHSTOCKICONINFO();
\r
434 sii.cbSize = Marshal.SizeOf(typeof(SHSTOCKICONINFO));
\r
435 sii.hIcon = IntPtr.Zero;
\r
437 //SHGetStockIconInfo(siid, SHGFI_ICON | ((isSmall)? SHGFI_SMALLICON : SHGFI_LARGEICON), ref sii);
\r
438 SHGetStockIconInfo(siid, 0x100u | ((isSmall)? 0x1u : 0x0u), ref sii);
\r
441 } catch (Exception) {
\r
442 return IntPtr.Zero;
\r
445 return IntPtr.Zero;
\r
450 /// ユーザ昇格が必要か設定し、設定必要なときシールドアイコンを表示する。
\r
452 /// <param name="label">対象のリンクラベル</param>
\r
453 /// <param name="required">ユーザ昇格が必要か否か、すなわちシールドアイコンを表示するか</param>
\r
454 public static void LinkLabel_SetElevationRequiredState(LinkLabel label, bool required)
\r
457 IntPtr iconHandle = IntPtr.Zero;
\r
459 //SHGetStockIconHandle(SIID_SHIELD, true);
\r
460 iconHandle = SHGetStockIconHandle(77, true);
\r
461 if (iconHandle != IntPtr.Zero) {
\r
462 label.Image = Bitmap.FromHicon(iconHandle);
\r
464 using (Icon ico = new Icon(SystemIcons.Shield, 16, 16)) {
\r
465 label.Image = ico.ToBitmap();
\r
469 label.ImageAlign = ContentAlignment.MiddleLeft;
\r
470 label.Padding = new Padding(label.Image.Width, label.Padding.Top, label.Padding.Right, label.Padding.Bottom);
\r
472 label.Image = null;
\r
473 label.Padding = new Padding(0);
\r
479 [DllImport("user32.dll", CharSet = CharSet.Auto)]
\r
480 internal static extern IntPtr SendMessage(IntPtr hWnd, UInt32 Msg, IntPtr wParam, IntPtr lParam);
\r
482 [DllImport("user32.dll", CharSet=CharSet.Auto)]
\r
483 private static extern IntPtr SendMessage(IntPtr hWnd, UInt32 Msg, IntPtr wParam, ref HD_ITEM lParam);
\r
485 [DllImport("user32.dll", CharSet=CharSet.Auto)]
\r
486 private static extern IntPtr SendMessage(IntPtr hWnd, Int32 Msg, IntPtr wParam, ref CHARFORMAT2 lParam);
\r
488 [DllImport("uxtheme.dll", ExactSpelling=true, CharSet=CharSet.Unicode)]
\r
489 internal static extern int SetWindowTheme(IntPtr hWnd, String pszSubAppName, String pszSubIdList);
\r
492 [StructLayout(LayoutKind.Sequential)]
\r
493 private struct LRECT {
\r
500 [DllImport("user32.dll")]
\r
501 private static extern bool GetClientRect(IntPtr hWnd, out LRECT lpRect);
\r