如何为 UIBarButtonItem 和 UIButton 修改 UIMenu

Jor*_*n H 9 uikit ios uimenu ios14

iOS 14 添加了在点击或长按 UIBarButtonItem 或 UIButton 时显示菜单的功能,如下所示:

let menu = UIMenu(children: [UIAction(title: "Action", image: nil) { action in
    //do something
}])
button.menu = menu
barButtonItem = UIBarButtonItem(title: "Show Menu", image: nil, primaryAction: nil, menu: menu)
Run Code Online (Sandbox Code Playgroud)

这通常会替换操作表(UIAlertControlleractionSheet样式)。有一个动态操作表是很常见的,其中基于用户点击按钮时的某些状态仅包含或可能禁用操作。但是使用此 API,菜单是在创建按钮时创建的。如何在菜单出现之前修改菜单或使其动态化以确保适当的操作可用并在出现时处于正确的状态?

Jor*_*n H 6

我发现的一种解决方案是存储对条形按钮项或按钮的引用,并在每次影响菜单中可用操作的状态发生变化时重新创建菜单。menu是可设置的属性,因此可以在创建按钮后随时更改。您还可以获取当前菜单并替换其子项,如下所示:button.menu = button.menu?.replacingChildren([])

对于在状态更改时未通知您的情况,您确实需要能够在菜单出现之前立即更新。对于UIButton的,你可以在用户通过目标/行动上的按钮向下接触这样修改的菜单button.addTarget(self, action: #selector(buttonTouchedDown(_:)), for: .touchDown)。我确认这至少在此时适用于 iOS 14 beta,但前提showsMenuAsPrimaryAction是为 false,因此他们必须长按才能打开菜单。我还没有找到 UIBarButtonItem 的解决方案,但您可以使用 UIButton 作为 UIBarButtonItem 的自定义视图。

这些感觉不是很好的解决方案,如果我找到更好的东西会更新。


Mic*_*nry 5

经过一番试验,我发现您可以通过将属性设置为first然后设置新的来修改UIButton's.menumenunullUIIMenu

这是我制作的示例代码

@IBOutlet weak var button: UIButton!

func generateMenu(max: Int, isRandom: Bool = false) -> UIMenu {
    let n = isRandom ? Int.random(in: 1...max) : max
    print("GENERATED MENU: \(n)")
    let items = (0..<n).compactMap { i -> UIAction in
        UIAction(
            title: "Menu \(i)",
            image: nil
        ) {[weak self] _ in
            guard let self = self else { return }

            self.button.menu = nil // HERE

            self.button.menu = self.generateMenu(max: 10, isRandom: true)
            print("Tap")
        }
    }
 
    let m = UIMenu(
        title: "Test", image: nil,
        identifier: UIMenu.Identifier(rawValue: "Hello.menu"),
        options: .displayInline, children: items)
    return m
}

override func viewDidLoad() {
    super.viewDidLoad()
    button.menu = generateMenu(max: 10)
    button.showsMenuAsPrimaryAction = true
}
Run Code Online (Sandbox Code Playgroud)