iOS 13 引入
UIMenu 在 iOS 13 中引入栅哀,可以很方便的創(chuàng)建程序菜單和上下文菜單不脯。
class ViewController: UIViewController {
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
navigationController?.isToolbarHidden = false
// 菜單綁定到UIBarButtonItem(iOS 14的構(gòu)造函數(shù))
let addNewItem = UIBarButtonItem(systemItem: .add, primaryAction: nil, menu: createMenuIOS13())
// 放到工具條
toolbarItems = [addNewItem]
}
func createMenuIOS13() -> UIMenu {
// 第一個(gè)菜單
let favorite = UIAction(title: "Favorite", image: UIImage(systemName: "heart.fill")) { _ in
print("favorite")
}
// 第二個(gè)菜單
let share = UIAction(title: "Share", image: UIImage(systemName: "square.and.arrow.up.fill")) { _ in
print("share")
}
// 第三個(gè)菜單
let delete = UIAction(title: "Delete", image: UIImage(systemName: "trash.fill"), attributes: [.destructive]) { _ in
print("delete")
}
let menuActions = [favorite, share, delete]
let addNewMenu = UIMenu(
title: "",
children: menuActions)
return addNewMenu
}
}
iOS 14 增強(qiáng)
iOS 14 中引入UIDeferredMenuElement
阿迈,允許異步地創(chuàng)建 UIMenu,也就是說(shuō)可以動(dòng)態(tài)在后臺(tái)配置菜單的內(nèi)容峰档。
class ViewController: UIViewController {
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
// 放到導(dǎo)航條
navigationItem.rightBarButtonItem = UIBarButtonItem(systemItem: .add, primaryAction: nil, menu: createMenuIOS14())
}
func createMenuIOS14() -> UIMenu {
// 應(yīng)該是是通過(guò)網(wǎng)絡(luò)獲取礼患,這里直接從Bundle加載
let menuItemsForUser = Bundle.main.decode([RemoteItem].self, from: "menu.json")
// 創(chuàng)建UIDeferredMenuElement
let dynamicElements = UIDeferredMenuElement { completion in
// 創(chuàng)建UIAction
let actions = menuItemsForUser.map { item in
UIAction(title: item.title, image: UIImage(systemName: item.icon)) { _ in
print("\(item.title) tapped")
}
}
// 一定要調(diào)用completion處理
completion(actions)
}
return UIMenu(
title: "",
children: [dynamicElements])
}
}
// 菜單Model
struct RemoteItem: Codable {
let title: String
let icon: String
}
// 加載文件并轉(zhuǎn)Model
extension Bundle {
func decode<T: Decodable>(_ type: T.Type, from file: String) -> T {
guard let url = self.url(forResource: file, withExtension: nil) else {
fatalError("Failed to find \(file) in bundle.")
}
guard let data = try? Data(contentsOf: url) else {
fatalError("Failed to load \(file) from bundle.")
}
let decoder = JSONDecoder()
guard let model = try? decoder.decode(T.self, from: data) else {
fatalError("Failed to decode \(file) from bundle.")
}
return model
}
}
JSON文件內(nèi)容如下:
[
{
"title": "Favorite",
"icon": "heart.fill"
},
{
"title": "Share",
"icon": "square.and.arrow.up.fill"
},
{
"title": "Delete",
"icon": "trash.fill"
}
]
效果
結(jié)果.gif