使用SwiftUI客製化Alert

閱讀時間約 14 分鐘

UIAlertController應該是大家都用過吧,最簡單的用法就是創造一個UIAlertController設定標題、內容跟按鈕

UIAlertConrtoller

UIAlertConrtoller



let alert = UIAlertController(title: "這是標題", message: "這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容這是內容", preferredStyle: .alert)
let action = UIAlertAction(title: "OK", style: .default)
alert.addAction(action)
self.present(alert, animated: true)


但是今天有一個需求要客製化字體大小、內嵌連接在裡面🫨 同時要保留原生Alert高度隨著公告字數增長的特性,增長到最大範圍後才提供scroll操作

原生Alert高度會隨字數增長

原生Alert高度會隨字數增長


使用SwiftUI客製

覺得客製化的需求實在很難改,官方也講了UIAlertController不接受繼承

The UIAlertController class is intended to be used as-is and doesn’t support subclassing. The view hierarchy for this class is private and must not be modified.

直接用SwiftUI重刻一個

基本UI

基本UI


這邊為了方便看Alert都先把背景設成粉紅色
先做了標題、內容、連結、分隔線跟按鈕,並且固定寬度,把高度設在一定範圍之間

struct BasicAlert: View {
    let title: String
    let message: String
    let link: String?
    let buttonTitle: String
let action: (()->Void)?
    var body: some View {
        VStack {
            VStack {
                // 標題
                Text(title)
                    .font(.system(size: 20, weight: .bold))
                // 內容
                Text(message)
                    .font(.system(size: 18))
                // 連結
                if let link, let url = URL(string: link) {
                    Link("點擊這裡前往網站", destination: url)
                        .foregroundColor(.blue)
                        .frame(maxWidth: .infinity, alignment: .leading)
                        .padding(.leading, 9)
                }
            }.padding()
            // 一條分隔線
            Divider()
            // 按鈕
            createButton(title: buttonTitle).frame(height: 40.0)
        }
        .background(Color.pink)
        .cornerRadius(15.0)
        .frame(width: 270.0) // 固定寬度
        .frame(minHeight: 161, maxHeight: 603) // 設定高度範圍
    }

}

但是測試在高度超出範圍後,文字並沒有變成可以捲動,加上ScrollView看看

VStack {
  ScrollView { // 新增scrollView
       VStack {
        // 標題
        Text(title)
         .font(.system(size: 20, weight: .bold))

// ....​


加上scrollView

加上scrollView


長的內容可以捲動了,但是scrollView的高度卻固定在最高限制🫨

去查SwiftUI的scrollView要如何設定捲動範圍,查到了fixedSize這個屬性

這邊官方範例就解釋得很清楚,會幫你把元件調整成適合的大小
在scrollView加上這個屬性,並固定最高高度為500

ScrollView {
   // ....
}
.frame(maxHeight: 500)
.fixedSize(horizontal: false, vertical: true)
//...


完美啦

完美啦

完整code

struct BasicAlert: View {
    let title: String
    let message: String
    let link: String?
    let buttonTitle: String
let action: (()->Void)?
    var body: some View {
        VStack {
            ScrollView {
                VStack {
                    // 標題
                    Text(title)
                        .font(.system(size: 20, weight: .bold))

                    // 內容
                    Text(message)
                        .font(.system(size: 18))
                    // 連結
                    if let link, let url = URL(string: link) {
                        Link("點擊這裡前往網站", destination: url)
                            .foregroundColor(.blue)
                            .frame(maxWidth: .infinity, alignment: .leading)
                            .padding(.leading, 9)

                    }
                }.padding()

            }
            .frame(maxHeight: 500)
            .fixedSize(horizontal: false, vertical: true)

        
            // 一條分隔線
            Divider()

            // 按鈕
            createButton(title: buttonTitle)

        }
        .background(Color.white)
        .cornerRadius(15.0)
        .frame(width: 270.0) // 固定寬度

    }

}

extension BasicAlert {
    func createButton(title: String) -> some View {
        Button {
action?()
        } label: {
            Text(title)
                .frame(height: 40.0)
                .frame(minWidth: 0, maxWidth: .infinity, alignment: .center)
        }

    }

}

背景透明黑底

這邊有很多種做法,因為我的專案大部分都是UIKit,所以我是用UIHostingController轉接再放到UIWindow上

class BasicAlertVC: UIViewController {
    let alertTitle: String
    let message: String
    let buttonTitle: String
    let link: String?
let action: (()->Void)?
    lazy var basicAlertView = makeBasicAlertView()

    init(title: String, message: String, link: String?, buttonTitle: String, action: (()->Void)?) {
        self.alertTitle = title
        self.message = message
        self.link = link
        self.buttonTitle = buttonTitle
self.action = action
        super.init(nibName: nil, bundle: nil)

    }

    required init?(coder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
    override func viewDidLoad() {
        super.viewDidLoad()
        addChild(basicAlertView)

        basicAlertView.view.backgroundColor = .clear
        basicAlertView.view.translatesAutoresizingMaskIntoConstraints = false
        view.addSubview(basicAlertView.view)
        basicAlertView.didMove(toParent: self)

        NSLayoutConstraint.activate([
            basicAlertView.view.widthAnchor.constraint(equalTo: view.widthAnchor),
            basicAlertView.view.topAnchor.constraint(equalTo: view.topAnchor),
            basicAlertView.view.centerXAnchor.constraint(equalTo: view.centerXAnchor),
            basicAlertView.view.bottomAnchor.constraint(equalTo: view.bottomAnchor)
        ])

    }

    private func makeBasicAlertView() -> UIHostingController<BasicAlert> {
        let basicAlert = BasicAlert(title: alertTitle, message: message, link: link, buttonTitle: buttonTitle, action: action)
        let controller = UIHostingController(rootView: basicAlert)
        return controller
    }

}

加到window上

let alertWindow = UIWindow(frame: UIScreen.main.bounds)
alertWindow.backgroundColor = UIColor(white: 0, alpha: 0.6)
alertWindow.windowLevel = .alert
alertWindow.rootViewController = BasicAlertVC(title: title, message: message, link: link, buttonTitle: buttonTitle, action: action)
alertWindow.makeKeyAndVisible()
alertWindow.isHidden = false
完整Alert

完整Alert




avatar-img
7會員
35內容數
紀錄iOS開發上遇到的問題或是一些流程筆記。主要都是Swift。
留言0
查看全部
avatar-img
發表第一個留言支持創作者!
Michelle Chen的沙龍 的其他內容
本文介紹瞭如何在SwiftUI中調整元件的對齊方式,包括置中、向左/向右/向上/向下對齊的方法。透過調整HStack、VStack以及frame的maxWidth、maxHeight和alignment屬性,可以達到想要的對齊效果。
瞭解如何在Xcode15及以上使用Logger進行更好的程式debug。Logger可以更好的組織Log,但也有一些缺點需要注意。本文將介紹Logger的基本使用方式,以及一些注意事項。
UIEdgeInsets 是一個結構體,用來表示矩形的邊距。它通常在視圖佈局中使用,用於修改視圖的框架(frame)。邊距值可以是正數或負數,這會對矩形的大小產生不同的影響。 UIEdgeInsets 結構體 UIEdgeInsets 結構體包含四個屬性: top:矩形頂部的邊距 left:
iOS15推出了新的按鈕外觀設定功能,讓設定變得更加直觀。使用不同的圓角設定和圖片位置可以輕鬆創建不同風格的按鈕。另外,新的選取效果變化功能也讓按鈕設計更加靈活。本文將介紹這些新功能的使用方法和效果。
本文介紹如何使用UINavigationBarAppearance調整四種場景下的UI外觀,並探討客製化返回鍵UI又保留返回手勢的做法,可以有效地客製化NavigationBar的外觀,並避免一些NG作法。
這篇文章介紹了 Swift 中字串的比較方法,並討論了使用日期字串進行比較的結果。同時也介紹了數字字串、符號字串和表情符號字串的比較原理。最後指出比較日期字串還是要轉成Date才是安全的做法。
本文介紹瞭如何在SwiftUI中調整元件的對齊方式,包括置中、向左/向右/向上/向下對齊的方法。透過調整HStack、VStack以及frame的maxWidth、maxHeight和alignment屬性,可以達到想要的對齊效果。
瞭解如何在Xcode15及以上使用Logger進行更好的程式debug。Logger可以更好的組織Log,但也有一些缺點需要注意。本文將介紹Logger的基本使用方式,以及一些注意事項。
UIEdgeInsets 是一個結構體,用來表示矩形的邊距。它通常在視圖佈局中使用,用於修改視圖的框架(frame)。邊距值可以是正數或負數,這會對矩形的大小產生不同的影響。 UIEdgeInsets 結構體 UIEdgeInsets 結構體包含四個屬性: top:矩形頂部的邊距 left:
iOS15推出了新的按鈕外觀設定功能,讓設定變得更加直觀。使用不同的圓角設定和圖片位置可以輕鬆創建不同風格的按鈕。另外,新的選取效果變化功能也讓按鈕設計更加靈活。本文將介紹這些新功能的使用方法和效果。
本文介紹如何使用UINavigationBarAppearance調整四種場景下的UI外觀,並探討客製化返回鍵UI又保留返回手勢的做法,可以有效地客製化NavigationBar的外觀,並避免一些NG作法。
這篇文章介紹了 Swift 中字串的比較方法,並討論了使用日期字串進行比較的結果。同時也介紹了數字字串、符號字串和表情符號字串的比較原理。最後指出比較日期字串還是要轉成Date才是安全的做法。
你可能也想看
Google News 追蹤
我自己是希望可以製作iOS app來更好存放我的文章, 更進階一點,可以變成直接錄音後, 照我設定的方式轉換成文檔,讓iPhone變成我更強的助手。 感覺有很多可以探索,用時間慢慢累積經驗。
Thumbnail
需求情境: 一般的看盤軟體,雖然都能針對一籃子自選股票,列出其即時行情和當天漲幅,但若要看「五日漲幅」呢?那就少見了,但這對我很重要。因為小部位的波段性價差交易是個好策略,這時候若能排序好一整排看下來,可以節省大量點來點去的成本,很有價值,所以就來自己刻。 解決方案: 從大處著眼,UI 最外層
Thumbnail
iPhone 最新作業系統 iOS 18 控制中心大更新,預設有常用功能、音樂播放器、連線控制三個頁面,並允許使用者自訂頁面,可以直接在控制中心管理所需的功能、調整位置和選擇尺寸。
Thumbnail
本章節旨在為讀者提供Swift程式語言的基礎知識,包括其基本語法、註解方法和變數使用方式,並通過具體的程式碼示例來說明這些概念。這將幫助讀者理解Swift的基本結構,並學會如何在Swift中定義變數並使用註解。
Thumbnail
這份文件的目的是介紹Swift語言,包括它的特性、應用範疇,以及誰在使用它。它也提供了一些學習Swift的資源和工具,以及一些常見的Swift庫和框架。
Thumbnail
需求情境: 在設計畫面時,資料來源是後台的 api,每一次畫面細節的修修改改,都會觸發 Xcode Preview 程序,導致不斷呼叫後台。此時若資料結構和大小都具有一定規模,就會導致效率低落,不斷等待,且消耗伺服器資源甚鉅。 解決方案: 將後台傳回的資料以檔案形式暫存在本地端,每次 pr
Thumbnail
Part.1 搞定基本的 UI 開始開發 iOS App。 首先準備一台 Mac,然後安裝 Xcode,新增專案,系統即刻生成基本的專案結構。coding 的起點在檔案 ContentView.swift: import SwiftUI struct ContentView: View {  
Thumbnail
UIUX 設計師們,是否時常尋找新的靈感來豐富你的網頁設計?以下介紹三個寶藏網站,將為你的創作之路帶來耳目一新的風景。
Thumbnail
前篇測試如何把提示詞生成的圖像細節提高,這篇要測試的工作流是把任意圖像載入後經由放大模型放大,同時測試放大後重繪看看效果如何。
Thumbnail
UI(使用者介面)設計對於用戶體驗至關重要。一個好的UI設計可以讓用戶輕鬆地與應用互動,而糟糕的設計則可能導致用戶感到困惑甚至沮喪。本文將探討五個UI 設計在App開發中常見的錯誤,並提供相應的解決策略。
我自己是希望可以製作iOS app來更好存放我的文章, 更進階一點,可以變成直接錄音後, 照我設定的方式轉換成文檔,讓iPhone變成我更強的助手。 感覺有很多可以探索,用時間慢慢累積經驗。
Thumbnail
需求情境: 一般的看盤軟體,雖然都能針對一籃子自選股票,列出其即時行情和當天漲幅,但若要看「五日漲幅」呢?那就少見了,但這對我很重要。因為小部位的波段性價差交易是個好策略,這時候若能排序好一整排看下來,可以節省大量點來點去的成本,很有價值,所以就來自己刻。 解決方案: 從大處著眼,UI 最外層
Thumbnail
iPhone 最新作業系統 iOS 18 控制中心大更新,預設有常用功能、音樂播放器、連線控制三個頁面,並允許使用者自訂頁面,可以直接在控制中心管理所需的功能、調整位置和選擇尺寸。
Thumbnail
本章節旨在為讀者提供Swift程式語言的基礎知識,包括其基本語法、註解方法和變數使用方式,並通過具體的程式碼示例來說明這些概念。這將幫助讀者理解Swift的基本結構,並學會如何在Swift中定義變數並使用註解。
Thumbnail
這份文件的目的是介紹Swift語言,包括它的特性、應用範疇,以及誰在使用它。它也提供了一些學習Swift的資源和工具,以及一些常見的Swift庫和框架。
Thumbnail
需求情境: 在設計畫面時,資料來源是後台的 api,每一次畫面細節的修修改改,都會觸發 Xcode Preview 程序,導致不斷呼叫後台。此時若資料結構和大小都具有一定規模,就會導致效率低落,不斷等待,且消耗伺服器資源甚鉅。 解決方案: 將後台傳回的資料以檔案形式暫存在本地端,每次 pr
Thumbnail
Part.1 搞定基本的 UI 開始開發 iOS App。 首先準備一台 Mac,然後安裝 Xcode,新增專案,系統即刻生成基本的專案結構。coding 的起點在檔案 ContentView.swift: import SwiftUI struct ContentView: View {  
Thumbnail
UIUX 設計師們,是否時常尋找新的靈感來豐富你的網頁設計?以下介紹三個寶藏網站,將為你的創作之路帶來耳目一新的風景。
Thumbnail
前篇測試如何把提示詞生成的圖像細節提高,這篇要測試的工作流是把任意圖像載入後經由放大模型放大,同時測試放大後重繪看看效果如何。
Thumbnail
UI(使用者介面)設計對於用戶體驗至關重要。一個好的UI設計可以讓用戶輕鬆地與應用互動,而糟糕的設計則可能導致用戶感到困惑甚至沮喪。本文將探討五個UI 設計在App開發中常見的錯誤,並提供相應的解決策略。