/pygments/samples/stdlib/MI.ahk

http://github.com/tinku99/ahklexers · AutoHotKey · 559 lines · 343 code · 69 blank · 147 comment · 80 complexity · 947367cf406f57dd01f8351c4bb828ff MD5 · raw file

  1. ;
  2. ; Menu Icons v2.21
  3. ; by Lexikos
  4. ;
  5. ; Associates an icon with a menu item.
  6. ; NOTE: On versions of Windows other than Vista, the menu MUST be shown with
  7. ; MI_ShowMenu() for the icons to appear.
  8. ;
  9. ; MenuNameOrHandle
  10. ; The name or handle of a menu. When setting icons for multiple items,
  11. ; it is more efficient to use a handle returned by MI_GetMenuHandle("menuname").
  12. ; ItemPos
  13. ; The position of the menu item, where 1 is the first item.
  14. ; FilenameOrHICON
  15. ; The filename or handle of an icon.
  16. ; SUPPORTS EXECUTABLE FILES ONLY (EXE/DLL/ICL/CPL/etc.)
  17. ; IconNumber
  18. ; The icon group to use (if omitted, it defaults to 1.)
  19. ; This is not used if FilenameOrHICON specifies an icon handle.
  20. ; IconSize
  21. ; The desired width and height of the icon. If omitted, the system's small icon size is used.
  22. ; h_bitmap
  23. ; h_icon
  24. ; v2.2: These parameters are no longer used as MI now automatically deletes the
  25. ; icon/bitmap if a new icon/bitmap is being set.
  26. ; OBSOLETE:
  27. ; These are set to the bitmap or icon resources which are used.
  28. ; Bitmaps and icons can be deleted as follows:
  29. ; DllCall("DeleteObject", "uint", h_bitmap)
  30. ; DllCall("DestroyIcon", "uint", h_icon)
  31. ; This is only necessary if the menu item displaying these resources
  32. ; is manually removed.
  33. ; Usually only one of h_icon or h_bitmap will be used, and the other will be 0 (NULL).
  34. ;
  35. ; OPERATING SYSTEM NOTES:
  36. ;
  37. ; Windows 2000 and above:
  38. ; PrivateExtractIcons() is used to extract the icon.
  39. ;
  40. ; Older versions of Windows:
  41. ; PrivateExtractIcons() is not available, so ExtractIconEx() is used.
  42. ; As a result, a 16x16 or 32x32 icon will be loaded. If a size is specified,
  43. ; the icon may be stretched to fit. If no size is specified, 16x16 is used.
  44. ;
  45. MI_SetMenuItemIcon(MenuNameOrHandle, ItemPos, FilenameOrHICON, IconNumber=1, IconSize=0, ByRef unused1="", ByRef unused2="")
  46. {
  47. ; Set for compatibility with older scripts:
  48. unused1=0
  49. unused2=0
  50. if MenuNameOrHandle is integer
  51. h_menu := MenuNameOrHandle
  52. else
  53. h_menu := MI_GetMenuHandle(MenuNameOrHandle)
  54. if !h_menu
  55. return false
  56. if FilenameOrHICON is integer
  57. {
  58. ; May be 0 to remove icon.
  59. h_icon := FilenameOrHICON
  60. ; Copy and potentially resize the icon. Since the caller is probably "caching"
  61. ; icon handles or assigning them to multiple items, we don't want to delete
  62. ; it if/when a future call to this function re-sets this item's icon.
  63. if h_icon
  64. h_icon := DllCall("CopyImage","uint",h_icon,"uint",1
  65. ,"int",IconSize,"int",IconSize,"uint",0)
  66. ; else caller wants to remove and delete existing icon.
  67. }
  68. else
  69. {
  70. ; Load icon from file. Remember to clean up this icon if we end up using a bitmap.
  71. ; Resizing is not necessary in this case since MI_ExtractIcon already does that.
  72. if !(loaded_icon := h_icon := MI_ExtractIcon(FilenameOrHICON, IconNumber, IconSize))
  73. return false
  74. }
  75. ; Windows Vista supports 32-bit alpha-blended bitmaps in menus. Note that
  76. ; A_OSVersion does not report WIN_VISTA when running in compatibility mode.
  77. ; To get nice icons on other versions of Windows, we need to owner-draw.
  78. ; DON'T TOUCH UNLESS YOU KNOW WHAT YOU'RE DOING:
  79. ; use_bitmap MUST have the same value for each use on a given menu item.
  80. if A_OSVersion in WIN_VISTA,WIN_7
  81. use_bitmap := true
  82. ; Get the previous bitmap or icon handle.
  83. VarSetCapacity(mii,48,0), NumPut(48,mii), NumPut(0xA0,mii,4)
  84. if DllCall("GetMenuItemInfo","uint",h_menu,"uint",ItemPos-1,"uint",1,"uint",&mii)
  85. h_previous := use_bitmap ? NumGet(mii,44,"int") : NumGet(mii,32,"int")
  86. if use_bitmap
  87. {
  88. if h_icon
  89. {
  90. h_bitmap := MI_GetBitmapFromIcon32Bit(h_icon, IconSize, IconSize)
  91. if loaded_icon
  92. {
  93. ; The icon we loaded is no longer needed.
  94. DllCall("DestroyIcon","uint",loaded_icon)
  95. ; Don't try to destroy the now invalid handle again:
  96. loaded_icon := 0
  97. }
  98. if !h_bitmap
  99. return false
  100. }
  101. else
  102. ; Caller wants to remove and delete existing icon.
  103. h_bitmap := 0
  104. NumPut(0x80,mii,4) ; fMask: Set hbmpItem only, not dwItemData.
  105. , NumPut(h_bitmap,mii,44) ; hbmpItem = h_bitmap
  106. }
  107. else
  108. {
  109. ; Associate the icon with the menu item. Relies on the probable case that no other
  110. ; script or dll will use dwItemData. If other scripts need to associate data with
  111. ; an item, MI should be expanded to allow it.
  112. NumPut(h_icon,mii,32) ; dwItemData = h_icon
  113. , NumPut(-1,mii,44) ; hbmpItem = HBMMENU_CALLBACK
  114. }
  115. if DllCall("SetMenuItemInfo","uint",h_menu,"uint",ItemPos-1,"uint",1,"uint",&mii)
  116. {
  117. ; Only now that we know it's a success, delete the previous icon or bitmap.
  118. if use_bitmap
  119. { ; Exclude NULL and predefined HBMMENU_ values (-1, 1..11).
  120. if (h_previous < -1 || h_previous > 11)
  121. DllCall("DeleteObject","uint",h_previous)
  122. } else
  123. DllCall("DestroyIcon","uint",h_previous)
  124. return true
  125. }
  126. ; ELSE FAIL
  127. if loaded_icon
  128. DllCall("DestroyIcon","uint",loaded_icon)
  129. return false
  130. }
  131. ; v2.2: This should be used to remove and delete all icons in a menu before deleting the menu.
  132. MI_RemoveIcons(MenuNameOrHandle)
  133. {
  134. if MenuNameOrHandle is integer
  135. h_menu := MenuNameOrHandle
  136. else
  137. h_menu := MI_GetMenuHandle(MenuNameOrHandle)
  138. if !h_menu
  139. return
  140. Loop % DllCall("GetMenuItemCount","uint",h_menu)
  141. MI_SetMenuItemIcon(h_menu, A_Index, 0)
  142. }
  143. ; Set a menu item's associated bitmap.
  144. ; hBitmap can be a handle to a bitmap, or a HBMMENU value (see below.)
  145. MI_SetMenuItemBitmap(MenuNameOrHandle, ItemPos, hBitmap)
  146. {
  147. if MenuNameOrHandle is integer
  148. h_menu := MenuNameOrHandle
  149. else
  150. h_menu := MI_GetMenuHandle(MenuNameOrHandle)
  151. if !h_menu
  152. return false
  153. VarSetCapacity(mii,48,0), NumPut(48,mii), NumPut(0x80,mii,4), NumPut(hBitmap,mii,44)
  154. return DllCall("SetMenuItemInfo","uint",h_menu,"uint",ItemPos-1,"uint",1,"uint",&mii)
  155. }
  156. /*
  157. HBMMENU_SYSTEM = 1
  158. HBMMENU_MBAR_RESTORE = 2
  159. HBMMENU_MBAR_MINIMIZE = 3
  160. HBMMENU_MBAR_CLOSE = 5
  161. HBMMENU_MBAR_CLOSE_D = 6
  162. HBMMENU_MBAR_MINIMIZE_D = 7
  163. HBMMENU_POPUP_CLOSE = 8
  164. HBMMENU_POPUP_RESTORE = 9
  165. HBMMENU_POPUP_MAXIMIZE = 10
  166. HBMMENU_POPUP_MINIMIZE = 11
  167. */
  168. ;
  169. ; General Functions
  170. ;
  171. ; Gets a menu handle from a menu name.
  172. ; Adapted from Shimanov's Menu_AssignBitmap()
  173. ; http://www.autohotkey.com/forum/topic7526.html
  174. MI_GetMenuHandle(menu_name)
  175. {
  176. static h_menuDummy
  177. ; v2.2: Check for !h_menuDummy instead of h_menuDummy="" in case init failed last time.
  178. If !h_menuDummy
  179. {
  180. Menu, menuDummy, Add
  181. Menu, menuDummy, DeleteAll
  182. Gui, 99:Menu, menuDummy
  183. ; v2.2: Use LastFound method instead of window title. [Thanks animeaime.]
  184. Gui, 99:+LastFound
  185. h_menuDummy := DllCall("GetMenu", "uint", WinExist())
  186. Gui, 99:Menu
  187. Gui, 99:Destroy
  188. ; v2.2: Return only after cleaning up. [Thanks animeaime.]
  189. if !h_menuDummy
  190. return 0
  191. }
  192. Menu, menuDummy, Add, :%menu_name%
  193. h_menu := DllCall( "GetSubMenu", "uint", h_menuDummy, "int", 0 )
  194. DllCall( "RemoveMenu", "uint", h_menuDummy, "uint", 0, "uint", 0x400 )
  195. Menu, menuDummy, Delete, :%menu_name%
  196. return h_menu
  197. }
  198. ; Valid (and safe to use) styles:
  199. ; MNS_AUTODISMISS 0x10000000
  200. ; MNS_CHECKORBMP 0x04000000 The same space is reserved for the check mark and the bitmap.
  201. ; MNS_NOCHECK 0x80000000 No space is reserved to the left of an item for a check mark.
  202. MI_SetMenuStyle(MenuNameOrHandle, style)
  203. {
  204. if MenuNameOrHandle is integer
  205. h_menu := MenuNameOrHandle
  206. else
  207. h_menu := MI_GetMenuHandle(MenuNameOrHandle)
  208. if !h_menu
  209. return
  210. VarSetCapacity(mi,28,0), NumPut(28,mi)
  211. NumPut(0x10,mi,4) ; fMask=MIM_STYLE
  212. NumPut(style,mi,8)
  213. DllCall("SetMenuInfo","uint",h_menu,"uint",&mi)
  214. }
  215. ; Extract an icon from an executable, DLL or icon file.
  216. MI_ExtractIcon(Filename, IconNumber, IconSize)
  217. {
  218. static ExtractIconEx
  219. ; LoadImage is not used..
  220. ; ..with exe/dll files because:
  221. ; it only works with modules loaded by the current process,
  222. ; it needs the resource ordinal (which is not the same as an icon index), and
  223. ; ..with ico files because:
  224. ; it can only load the first icon (of size %IconSize%) from an .ico file.
  225. ; If possible, use PrivateExtractIcons, which supports any size of icon.
  226. if (IconSize != 16 && IconSize != 32)
  227. {
  228. if A_OSVersion in WIN_7,WIN_XP,WIN_VISTA,WIN_2003,WIN_2000
  229. {
  230. if DllCall("PrivateExtractIcons"
  231. ,"str",Filename,"int",IconNumber-1,"int",IconSize,"int",IconSize
  232. ,"uint*",h_icon,"uint*",0,"uint",1,"uint",0,"int")
  233. {
  234. return h_icon
  235. }
  236. }
  237. }
  238. if !ExtractIconEx
  239. ExtractIconEx := MI_DllProcAorW("shell32","ExtractIconEx")
  240. ; Use ExtractIconEx, which only returns 16x16 or 32x32 icons.
  241. if DllCall(ExtractIconEx,"str",Filename,"int",IconNumber-1
  242. ,"uint*",h_icon,"uint*",h_icon_small,"uint",1)
  243. {
  244. SysGet, SmallIconSize, 49
  245. ; Use the best-fit size; delete the other. Defaults to small icon.
  246. if (IconSize <= SmallIconSize) {
  247. DllCall("DestroyIcon","uint",h_icon)
  248. h_icon := h_icon_small
  249. } else
  250. DllCall("DestroyIcon","uint",h_icon_small)
  251. ; I think PrivateExtractIcons resizes icons automatically,
  252. ; so resize icons returned by ExtractIconEx for consistency.
  253. if (h_icon && IconSize)
  254. h_icon := DllCall("CopyImage","uint",h_icon,"uint",1,"int",IconSize
  255. ,"int",IconSize,"uint",4|8)
  256. }
  257. return h_icon ? h_icon : 0
  258. }
  259. ;
  260. ; Owner-Drawn Menu Functions
  261. ;
  262. ; Sub-classes a window from THIS process to owner-draw menu icons.
  263. ; This allows the menu to be shown by means other than MI_ShowMenu().
  264. MI_EnableOwnerDrawnMenus(hwnd="")
  265. {
  266. if (hwnd="") { ; Use the script's main window if hwnd was omitted.
  267. dhw := A_DetectHiddenWindows
  268. DetectHiddenWindows, On
  269. Process, Exist
  270. hwnd := WinExist("ahk_class AutoHotkey ahk_pid " ErrorLevel)
  271. DetectHiddenWindows, %dhw%
  272. }
  273. if !hwnd
  274. return
  275. wndProc := RegisterCallback("MI_OwnerDrawnMenuItemWndProc","",4
  276. ,DllCall("GetWindowLong","uint",hwnd,"uint",-4))
  277. return DllCall("SetWindowLong","uint",hwnd,"int",-4,"int",wndProc,"uint")
  278. }
  279. ; Shows a menu, allowing owner-drawn icons to be drawn.
  280. MI_ShowMenu(MenuNameOrHandle, x="", y="")
  281. {
  282. static hInstance, hwnd, ClassName := "OwnerDrawnMenuMsgWin"
  283. , CreateWindowEx
  284. if MenuNameOrHandle is integer
  285. h_menu := MenuNameOrHandle
  286. else
  287. h_menu := MI_GetMenuHandle(MenuNameOrHandle)
  288. if !h_menu
  289. return false
  290. if !hwnd
  291. { ; Create a message window to receive owner-draw messages from the menu.
  292. ; Only one window is created per instance of the script.
  293. if !hInstance
  294. hInstance := DllCall("GetModuleHandle", "UInt", 0)
  295. ; Register a window class to associate OwnerDrawnMenuItemWndProc()
  296. ; with the window we will create.
  297. wndProc := RegisterCallback("MI_OwnerDrawnMenuItemWndProc","",4,0)
  298. if !wndProc {
  299. ErrorLevel = RegisterCallback
  300. return false
  301. }
  302. ; Create a new window class.
  303. VarSetCapacity(wc, 40, 0) ; WNDCLASS wc
  304. NumPut(wndProc, wc, 4) ; lpfnWndProc
  305. NumPut(hInstance, wc,16) ; hInstance
  306. NumPut(&ClassName,wc,36) ; lpszClassname
  307. ; Register the class.
  308. if !DllCall("RegisterClass","uint",&wc)
  309. { ; failed, free the callback.
  310. DllCall("GlobalFree","uint",wndProc)
  311. ErrorLevel = RegisterClass
  312. return false
  313. }
  314. ;
  315. ; Create the message window.
  316. ;
  317. if A_OSVersion in WIN_XP,WIN_7,WIN_VISTA,WIN_2003
  318. hwndParent = -3 ; HWND_MESSAGE (message-only window)
  319. else
  320. hwndParent = 0 ; un-owned
  321. hwnd := DllCall("CreateWindowEx","uint",0,"str",ClassName,"str",ClassName
  322. ,"uint",0,"int",0,"int",0,"int",0,"int",0,"uint",hwndParent
  323. ,"uint",0,"uint",hInstance,"uint",0)
  324. if !hwnd {
  325. ErrorLevel = CreateWindowEx
  326. return false
  327. }
  328. }
  329. prev_hwnd := DllCall("GetForegroundWindow")
  330. ; Required for the menu to initially have focus.
  331. ;DllCall("SetForegroundWindow","uint",hwnd)
  332. dhw := A_DetectHiddenWindows
  333. DetectHiddenWindows, On
  334. WinActivate, ahk_id %hwnd%
  335. DetectHiddenWindows, %dhw%
  336. if (x="" or y="") {
  337. CoordMode, Mouse, Screen
  338. MouseGetPos, x, y
  339. }
  340. ; returns non-zero on success.
  341. ret := DllCall("TrackPopupMenu","uint",h_menu,"uint",0,"int",x,"int",y
  342. ,"int",0,"uint",hwnd,"uint",0)
  343. if WinExist("ahk_id " prev_hwnd)
  344. DllCall("SetForegroundWindow","uint",prev_hwnd)
  345. ; Required to let AutoHotkey process WM_COMMAND messages we may have
  346. ; sent as a result of clicking a menu item. (Without this, the item-click
  347. ; won't register if there is an 'ExitApp' after ShowOwnerDrawnMenu returns.)
  348. Sleep, 1
  349. return ret
  350. }
  351. MI_OwnerDrawnMenuItemWndProc(hwnd, Msg, wParam, lParam)
  352. {
  353. static WM_DRAWITEM = 0x002B, WM_MEASUREITEM = 0x002C, WM_COMMAND = 0x111
  354. static ScriptHwnd
  355. Critical 500
  356. if (Msg = WM_MEASUREITEM && wParam = 0)
  357. { ; MSDN: wParam - If the value is zero, the message was sent by a menu.
  358. h_icon := NumGet(lParam+20)
  359. if !h_icon
  360. return false
  361. ; Measure icon and put results into lParam.
  362. VarSetCapacity(buf,24)
  363. if DllCall("GetIconInfo","uint",h_icon,"uint",&buf)
  364. {
  365. hbmColor := NumGet(buf,16)
  366. hbmMask := NumGet(buf,12)
  367. x := DllCall("GetObject","uint",hbmColor,"int",24,"uint",&buf)
  368. DllCall("DeleteObject","uint",hbmColor)
  369. DllCall("DeleteObject","uint",hbmMask)
  370. if !x
  371. return false
  372. NumPut(NumGet(buf,4,"int")+2, lParam+12) ; width
  373. NumPut(NumGet(buf,8,"int") , lParam+16) ; height
  374. return true
  375. }
  376. return false
  377. }
  378. else if (Msg = WM_DRAWITEM && wParam = 0)
  379. {
  380. hdcDest := NumGet(lParam+24)
  381. x := NumGet(lParam+28)
  382. y := NumGet(lParam+32)
  383. h_icon := NumGet(lParam+44)
  384. if !(h_icon && hdcDest)
  385. return false
  386. return DllCall("DrawIconEx","uint",hdcDest,"int",x,"int",y,"uint",h_icon
  387. ,"uint",0,"uint",0,"uint",0,"uint",0,"uint",3)
  388. }
  389. else if (Msg = WM_COMMAND && !(wParam>>16)) ; (clicked a menu item)
  390. {
  391. DetectHiddenWindows, On
  392. WinGetClass, class, ahk_id %hwnd%
  393. if (class != "AutoHotkeyGUI" && class != "AutoHotkey") {
  394. if !ScriptHwnd {
  395. Process, Exist
  396. ScriptHwnd := WinExist("ahk_class AutoHotkey ahk_pid " ErrorLevel)
  397. }
  398. ; Forward this message to the AutoHotkey main window.
  399. PostMessage, Msg, wParam, lParam,, ahk_id %ScriptHwnd%
  400. return ErrorLevel
  401. }
  402. }
  403. if A_EventInfo ; Let the "super-class" window procedure handle all other messages.
  404. return DllCall("CallWindowProc","uint",A_EventInfo,"uint",hwnd,"uint",Msg,"uint",wParam,"uint",lParam)
  405. else ; Let the default window procedure handle all other messages.
  406. return DllCall("DefWindowProc","uint",hwnd,"uint",Msg,"uint",wParam,"uint",lParam)
  407. }
  408. ;
  409. ; Windows Vista Menu Icons
  410. ;
  411. ; Note: 32-bit alpha-blended menu item bitmaps are supported only on Windows Vista.
  412. ; Article on menu icons in Vista:
  413. ; http://shellrevealed.com/blogs/shellblog/archive/2007/02/06/Vista-Style-Menus_2C00_-Part-1-_2D00_-Adding-icons-to-standard-menus.aspx
  414. MI_GetBitmapFromIcon32Bit(h_icon, width=0, height=0)
  415. {
  416. VarSetCapacity(buf,40) ; used as ICONINFO (20), BITMAP (24), BITMAPINFO (40)
  417. if DllCall("GetIconInfo","uint",h_icon,"uint",&buf) {
  418. hbmColor := NumGet(buf,16) ; used to measure the icon
  419. hbmMask := NumGet(buf,12) ; used to generate alpha data (if necessary)
  420. }
  421. if !(width && height) {
  422. if !hbmColor or !DllCall("GetObject","uint",hbmColor,"int",24,"uint",&buf)
  423. return 0
  424. width := NumGet(buf,4,"int"), height := NumGet(buf,8,"int")
  425. }
  426. ; Create a device context compatible with the screen.
  427. if (hdcDest := DllCall("CreateCompatibleDC","uint",0))
  428. {
  429. ; Create a 32-bit bitmap to draw the icon onto.
  430. VarSetCapacity(buf,40,0), NumPut(40,buf), NumPut(1,buf,12,"ushort")
  431. NumPut(width,buf,4), NumPut(height,buf,8), NumPut(32,buf,14,"ushort")
  432. if (bm := DllCall("CreateDIBSection","uint",hdcDest,"uint",&buf,"uint",0
  433. ,"uint*",pBits,"uint",0,"uint",0))
  434. {
  435. ; SelectObject -- use hdcDest to draw onto bm
  436. if (bmOld := DllCall("SelectObject","uint",hdcDest,"uint",bm))
  437. {
  438. ; Draw the icon onto the 32-bit bitmap.
  439. DllCall("DrawIconEx","uint",hdcDest,"int",0,"int",0,"uint",h_icon
  440. ,"uint",width,"uint",height,"uint",0,"uint",0,"uint",3)
  441. DllCall("SelectObject","uint",hdcDest,"uint",bmOld)
  442. }
  443. ; Check for alpha data.
  444. has_alpha_data := false
  445. Loop, % height*width
  446. if NumGet(pBits+0,(A_Index-1)*4) & 0xFF000000 {
  447. has_alpha_data := true
  448. break
  449. }
  450. if !has_alpha_data
  451. {
  452. ; Ensure the mask is the right size.
  453. hbmMask := DllCall("CopyImage","uint",hbmMask,"uint",0
  454. ,"int",width,"int",height,"uint",4|8)
  455. VarSetCapacity(mask_bits, width*height*4, 0)
  456. if DllCall("GetDIBits","uint",hdcDest,"uint",hbmMask,"uint",0
  457. ,"uint",height,"uint",&mask_bits,"uint",&buf,"uint",0)
  458. { ; Use icon mask to generate alpha data.
  459. Loop, % height*width
  460. if (NumGet(mask_bits, (A_Index-1)*4))
  461. NumPut(0, pBits+(A_Index-1)*4)
  462. else
  463. NumPut(NumGet(pBits+(A_Index-1)*4) | 0xFF000000, pBits+(A_Index-1)*4)
  464. } else { ; Make the bitmap entirely opaque.
  465. Loop, % height*width
  466. NumPut(NumGet(pBits+(A_Index-1)*4) | 0xFF000000, pBits+(A_Index-1)*4)
  467. }
  468. }
  469. }
  470. ; Done using the device context.
  471. DllCall("DeleteDC","uint",hdcDest)
  472. }
  473. if hbmColor
  474. DllCall("DeleteObject","uint",hbmColor)
  475. if hbmMask
  476. DllCall("DeleteObject","uint",hbmMask)
  477. return bm
  478. }
  479. ;
  480. ; Utils
  481. ;
  482. MI_DllProcAorW(dll, func) {
  483. ; In AutoHotkey_L/AutoHotkeyU we can just use "dll\func" and "A" or "W"
  484. ; will be appended as appropriate if func is not found, but since that
  485. ; won't work in regular AutoHotkey (for some dlls), resolve it manually:
  486. return DllCall("GetProcAddress"
  487. ,"uint", DllCall("GetModuleHandle","str",dll)
  488. , A_IsUnicode ? "astr":"str" ; Always an ANSI string.
  489. , func . (A_IsUnicode ? "W":"A"))
  490. }