Skip to content

Commit ffe88cd

Browse files
committed
UI: support showing basic context menus
1 parent 30038cd commit ffe88cd

7 files changed

+124
-2
lines changed

Sources/SwiftWin32/CMakeLists.txt

+2
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,9 @@ target_sources(SwiftWin32 PRIVATE
2525
UI/ContentSizeCategoryAdjusting.swift
2626
UI/ContentSizeCategoryImageAdjusting.swift
2727
UI/ContextMenuConfiguration.swift
28+
UI/ContextMenuInteraction.swift
2829
UI/ContextMenuInteractionAnimating.swift
30+
UI/ContextMenuInteractionDelegate.swift
2931
UI/Control.swift
3032
UI/CoordinateSpace.swift
3133
UI/DatePicker.swift

Sources/SwiftWin32/Support/WindowsHandle+UIExtensions.swift

+11
Original file line numberDiff line numberDiff line change
@@ -38,3 +38,14 @@ extension HBITMAP__: HandleValue {
3838
}
3939
}
4040
internal typealias BitmapHandle = ManagedHandle<HBITMAP__>
41+
42+
extension HMENU: HandleValue {
43+
typealias HandleType = HMENU
44+
internal static func release(_ hMenu: HandleType?) {
45+
if let hMenu = hMenu {
46+
DestroyMenu(hMenu)
47+
}
48+
}
49+
}
50+
51+
internal typealias MenuHandle = ManagedHandle<HMENU>

Sources/SwiftWin32/UI/ContextMenuConfiguration.swift

+4
Original file line numberDiff line numberDiff line change
@@ -44,4 +44,8 @@ public class ContextMenuConfiguration {
4444

4545
private var previewProvider: ContextMenuContentPreviewProvider
4646
private var actionProvider: ContextMenuActionProvider?
47+
48+
internal func provideActions() -> Menu? {
49+
actionProvider?([]) // TODO suggest default actions
50+
}
4751
}

Sources/SwiftWin32/UI/ContextMenuInteraction.swift

+9-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ extension ContextMenuInteraction {
1919
}
2020
}
2121

22-
public class ContextMenuInteraction {
22+
public class ContextMenuInteraction: Interaction {
2323
// MARK - Creating a Context Menu Interaction Object
2424

2525
/// Creates a context menu interaction object with the specified delegate
@@ -34,6 +34,14 @@ public class ContextMenuInteraction {
3434
/// and responds to interaction-related events.
3535
public private(set) weak var delegate: ContextMenuInteractionDelegate?
3636

37+
public private(set) var view: View? = nil
38+
39+
public func willMove(to view: View?) { }
40+
41+
public func didMove(to view: View?) {
42+
self.view = view
43+
}
44+
3745
// MARK - Getting the Interaction's Location
3846

3947
/// Returns the location of the user interaction in the specified view's

Sources/SwiftWin32/UI/Menu.swift

+20
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,8 @@
55
* SPDX-License-Identifier: BSD-3-Clause
66
**/
77

8+
import WinSDK
9+
810
extension Menu {
911
/// Constants for identifying an application's standard menus.
1012
public struct Identifier: Equatable, Hashable, RawRepresentable {
@@ -275,12 +277,30 @@ extension Menu.Options {
275277
/// A container for grouping related menu elements in an application menu or
276278
/// contextual menu.
277279
public class Menu: MenuElement {
280+
internal let children: [MenuElement]
281+
278282
/// Creating a Menu Object
279283

280284
/// Creates a new menu with the specified values.
281285
public init(title: String = "", image: Image? = nil,
282286
identifier: Menu.Identifier? = nil, options: Menu.Options = [],
283287
children: [MenuElement] = []) {
288+
self.children = children
284289
super.init(title: title, image: image)
285290
}
286291
}
292+
293+
internal struct Win32Menu {
294+
internal let hMenu: MenuHandle
295+
private let children: [Win32MenuElement]
296+
297+
internal init(_ hMenu: MenuHandle, children: [MenuElement]) {
298+
self.hMenu = hMenu
299+
self.children = children.map { Win32MenuElement.of($0) }
300+
for (index, child) in self.children.enumerated() {
301+
InsertMenuItemW(hMenu.value,
302+
UINT(index), true,
303+
&(child.info))
304+
}
305+
}
306+
}

Sources/SwiftWin32/UI/MenuElement.swift

+48
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,8 @@
55
* SPDX-License-Identifier: BSD-3-Clause
66
**/
77

8+
import WinSDK
9+
810
extension MenuElement {
911
/// Attributes that determine the style of the menu element.
1012
public struct Attributes: OptionSet {
@@ -67,3 +69,49 @@ public class MenuElement {
6769
self.image = image
6870
}
6971
}
72+
73+
internal class Win32MenuElement {
74+
internal var info: MENUITEMINFOW
75+
private var titleW: [WCHAR]
76+
private let subMenu: Win32Menu?
77+
78+
private init(title: String, subMenu: Win32Menu?, fType: Int32) {
79+
self.titleW = title.LPCWSTR
80+
let titleSize = titleW.count
81+
82+
self.subMenu = subMenu
83+
self.info = titleW.withUnsafeMutableBufferPointer {
84+
MENUITEMINFOW(cbSize: UINT(MemoryLayout<MENUITEMINFOW>.size),
85+
fMask: UINT(MIIM_FTYPE | MIIM_STATE | MIIM_ID | MIIM_STRING | MIIM_SUBMENU | MIIM_DATA),
86+
fType: UINT(fType),
87+
fState: UINT(MFS_ENABLED),
88+
wID: UInt32.random(in: .min ... .max),
89+
hSubMenu: subMenu?.hMenu.value,
90+
hbmpChecked: nil,
91+
hbmpUnchecked: nil,
92+
dwItemData: 0,
93+
dwTypeData: $0.baseAddress,
94+
cch: UINT(titleSize),
95+
hbmpItem: nil)
96+
}
97+
}
98+
99+
internal static func of(_ element: MenuElement) -> Win32MenuElement {
100+
let subMenu: Win32Menu?
101+
if let menu = element as? Menu {
102+
subMenu = Win32Menu(MenuHandle(owning: CreatePopupMenu()),
103+
children: menu.children)
104+
} else {
105+
subMenu = nil
106+
}
107+
return Win32MenuElement(title: element.title, subMenu: subMenu, fType: MFT_STRING)
108+
}
109+
110+
internal static var separator: Win32MenuElement {
111+
Win32MenuElement(title: "", subMenu: nil, fType: MFT_SEPARATOR)
112+
}
113+
114+
internal var isSeparator: Bool {
115+
info.fType == MFT_SEPARATOR
116+
}
117+
}

Sources/SwiftWin32/UI/View.swift

+30-1
Original file line numberDiff line numberDiff line change
@@ -11,8 +11,27 @@ private let SwiftViewProc: SUBCLASSPROC = { (hWnd, uMsg, wParam, lParam, uIdSubc
1111
let view: View? = unsafeBitCast(dwRefData, to: AnyObject.self) as? View
1212
switch uMsg {
1313
case UINT(WM_CONTEXTMENU):
14-
// TODO handle popup menu events
14+
guard let view = view,
15+
let menuInteraction = view.interactions.first(where: { $0 is ContextMenuInteraction })
16+
as? ContextMenuInteraction else { break }
17+
let x = Int16(truncatingIfNeeded: lParam)
18+
let y = Int16(truncatingIfNeeded: lParam >> 16)
19+
let point = Point(x: Int(x), y: Int(y))
20+
let menuConfiguration = menuInteraction.delegate?.contextMenuInteraction(menuInteraction,
21+
configurationForMenuAtLocation: point)
22+
if let menu = menuConfiguration?.provideActions() {
23+
view.win32ContextMenu = Win32Menu(MenuHandle(owning: CreatePopupMenu()),
24+
children: menu.children)
25+
} else {
26+
view.win32ContextMenu = nil
27+
}
28+
let hMenu = view.win32ContextMenu?.hMenu.value
29+
TrackPopupMenu(hMenu, UINT(TPM_RIGHTBUTTON),
30+
Int32(x), Int32(y), 0, view.hWnd, nil)
1531
return 0
32+
case UINT(WM_COMMAND):
33+
// TODO handle menu actions
34+
break
1635
default:
1736
break
1837
}
@@ -202,6 +221,16 @@ public class View: Responder {
202221
set { _ = EnableWindow(self.hWnd, newValue) }
203222
}
204223

224+
public private(set) var interactions: [Interaction] = []
225+
internal var win32ContextMenu: Win32Menu? = nil
226+
227+
public func addInteraction(_ interaction: Interaction) {
228+
interaction.willMove(to: self)
229+
interaction.view?.interactions.removeAll(where: { $0 === interaction })
230+
interactions.append(interaction)
231+
interaction.didMove(to: self)
232+
}
233+
205234
// MARK - Managing the View Hierarchy
206235

207236
public private(set) var superview: View?

0 commit comments

Comments
 (0)