2024年4月1日 星期一

Swift 字串遮罩

 這邊直接提供一個共用只要是字串都可已的方法

struct SecureText {

    private var originalText: String

    var prefixRange: Int!

    var numberOfStars: Int!

    

    init(_ text: String, prefix: Int, numberOfStars: Int) {

        self.originalText = text

        self.prefixRange = prefix

        self.numberOfStars = numberOfStars

    }

    

    var secureText: String {

        let firstRange = originalText.prefix(prefixRange)

        let endRange = originalText.suffix(originalText.count - prefixRange - numberOfStars)

        return String(firstRange) + String(repeating: "*", count: numberOfStars) + String(endRange)

    }

    

    var text: String {

        return originalText

    }

}

ex: 

let securePhone = SecureText("0987654987", prefix: 2, numberOfStars: 4)

print(securePhone.secureText) //輸出結果 "09****4987"

print(securePhone.text)   //輸出結果 "0987654987"

簡單說明一下,prefix為要遮罩的第幾位字元,numberOfStars為要遮罩幾字元。

假設您有特殊需求譬如遇到某個字但不知道整串字串為幾個字元時,就是計算該字在第幾字元,並且定義一個顯示標準去決定從第幾字元開始遮罩並遮罩幾字元。

但是除了上述的方法外可以按照個別需求去做單一邏輯的做法,例如:姓名、信箱、手機、地址等。

上述的四種我都會提供範例於下方,這邊就不多做輸出結果只提供使用範例...

以下的寫法我都是寫在String extension裡面,你可以自己再進行調整

姓名遮罩,邏輯是兩個字以下遮前,三個字以上只顯示頭尾

func markName() -> String {

        var nameToMask = self

        var isKids = false

        var maskedName = ""

        

        if nameToMask.count < 3 { // 姓名為兩個字的情況

            let lastChar = "*" + String(nameToMask.suffix(1))

            maskedName = lastChar

        } else {  // 姓名為三個字以上的情況

            let firstChar = String(nameToMask.prefix(1))

            let lastChar = String(nameToMask.suffix(1))

            let middleMask = String(repeating: "*", count: nameToMask.count - 2)

            

            maskedName = firstChar + middleMask + lastChar

        }

        

        return maskedName

    }

信箱遮罩,邏輯是取出帳號,並且帳號如果小於四字元顯示最後一字元,大於的話則顯示最後三字元

 func markEmail() -> String {

        guard self != "", self.contains("@") else {

            return ""

        }

        let mails = self.components(separatedBy: "@")

        let acount = mails[0]

        let mailHost = mails[1]

        var maskMail = ""

        

        if acount.count <= 4 {

            maskMail = String(repeating: "*", count: acount.count - 1) + String(acount.suffix(1)) + "@" + mailHost

        } else {

            let suffix = String(acount.suffix(3))

            maskMail = String(repeating: "*", count: acount.count - 3) + suffix + "@" + mailHost

        }

        return maskMail

    }

手機遮罩,顯示前四碼及最後兩碼

func maskPhone() -> String {

        guard self != "" else {

            return ""

        }

        let first = String(self.prefix(4))

        let last = String(self.suffix(2))

        let maskPhone = String(repeating: "*", count: self.count - 6)

        return first + maskPhone + last

    }

地址遮罩,這邊做法是判斷阿拉伯數字(因應大家習慣基本門牌號碼或是巷弄之類皆款性輸入啊拉博數字的撞太下),然後將數字前所有字串遮罩,你也可以改寫成顯示縣市地區至數字之間遮罩

func maskAddress() -> String {

        var maskedAddress = ""

        var foundDigit = false

        

        for (i, char) in self.enumerated() {

            let predicate = NSPredicate(format: "SELF MATCHES %@", "[0-9]")

            if predicate.evaluate(with: String(char)) && !foundDigit {

                foundDigit = true

                let index = self.count - i

                let suffix = String(self.suffix(index))

                maskedAddress = String(repeating: "*", count: i)

                maskedAddress = maskedAddress + suffix

                break

            }

        }

        return maskedAddress

    }


以上的方法全部都是類似規則,要注意的主要就是當您使用

String(repeating: <#T##String#>, count: <#T##Int#>)

後面count數基本上就是你要遮罩替換的字數,所以你會看到我直接帶字串的count - Int,

這寫法是你清楚有幾個字沒遮罩,向地址就是帶入i,因為我知道第i為數字,所以i以前都是要遮罩的。

使用方法就是字串直接點運算你要的方法例如"0988666888".maskPhone()

以上分享希望對於剛好有需求的朋友有所幫助

2021年6月7日 星期一

TouchID 或 FaceID 的基本應用

 今天來玩玩裝置生物識別的功能,

也就是TouchID or FaceID

雖然現在主要大多數有應用的APP基本上都是金融業居多。

但也是有部分的APP有引入此功能,

假設您的APP非一次性登入加入驗證生物識別可大大提升使用者體驗...

下面直接就來寫重點拉!

從基本設定開始,第一步先開啟info.plist

加入Privacy - Face ID Usage Description 後面字串就自行發揮吧!

例如:使用TouchID/FaceID進行登入

此加入僅限於FaceID,TouchID是不用加入該授權的。

接下來在要引用的ViewController加入import LocalAuthentication

然後宣告一個變數 並init. var context = LAContext()

接下來就是實作了,網路上我看過的範例多數都是直接寫一個Action執行,

我則是另外寫了一個方法進行,你可以依照你習慣的方式去進行不同的改變。

首先先檢查裝置是否可使用生物識別

 if context.canEvaluatePolicy(.deviceOwnerAuthentication, error: &error) { 

//省略

        } else {

            if let error = error {

               errorCode(error)

            }

            

        }

如果error直接跳錯誤訊息,成功則進行生物辨識,如下:

            //此文字會顯示於辨識時訊息

            let reason = "使用TouchID或FaceID進行驗證"

            // 使用哪種方法進行生物驗證

            context.evaluatePolicy(.deviceOwnerAuthentication, localizedReason: reason) { (success, error) in

                if success {

                    DispatchQueue.main.async { [unowned self] in

                        //成功直接執行login

                        showAlert(title: "成功", "您已驗證成功")

                    }

                } else {

                    DispatchQueue.main.async { [unowned self] in

                        if let error = error {

                            errorCode(error)

                        }

                        

                    }

                }

            }

reason所使用文字建議你們使用一個通用的文字..

順帶一提官方提供的生物辨識有兩種方法,deviceOwnerAuthenticationWithBiometrics

deviceOwnerAuthentication  而我引用後者,金融業都是使用前者。兩種方法玩起來不一樣,我引用的當您失敗過多次後仍然可透過裝置所設定密碼執行驗證,前者則無法...這邊再由各位自己去玩看看吧!

而驗證可直接寫在檢查裝置是否可使用生物識別的判斷內,也可跟我一樣拆開寫。看當下案子需求能做不同的變化。

在此再附上我的errorCode(錯誤有很多種可以自己完)基本上我的error因使用的是deviceOwnerAuthentication的方法照理說都用不太到,

要研究error建議使用deviceOwnerAuthenticationWithBiometrics

 func errorCode(_ error:Error) {

        switch error {

        case LAError.authenticationFailed:

            print("Authentication failed")

        case LAError.passcodeNotSet:

            showAlert(title: "錯誤", "您未設定密碼")

        case LAError.systemCancel:

            showAlert(title: "錯誤", "授權過程中被系統取消")

        case LAError.userCancel:

            showAlert(title: "錯誤", "授權被用戶取消")

        case LAError.userFallback:

            showAlert(title: "提示", "用戶取消生物辨識使用密碼驗證")

        case LAError.biometryNotAvailable:

            if context.biometryType == .faceID {

                showAlert(title: "提示", "您已關閉FaceID授權,請至系統設定中開啟。")

            } else if context.biometryType == .touchID {

                showAlert(title: "提示", "您已關閉TouchID授權,請至系統設定中開啟。")

            } else if context.biometryType == .none {

                showAlert(title: "提示", "您未授權TouchID或FaceID於APP中,請至系統設定中開啟。")

            }

        case LAError.biometryNotEnrolled:

            showAlert(title: "提示", "沒有登入生物識別,無法使用")

        case LAError.biometryLockout:

            showAlert(title: "錯誤", " 您已多次登入失敗,請使用一般登入。")

        case LAError.appCancel:

            print("APP取消驗證")

        default:

            print(error.localizedDescription)

        }

    }

錯誤訊息內你有注意到有個biometryType嗎?此方法是用來判斷它屬於哪種生物辨識驗證,至於none我個人覺得現在應該是不太會發生,因為iPhone 5s後的機種不是touchID就是FaceID所以照理說不會執行到none

以上很廢的分享文,提供您參考...希望能幫助到剛好有要使用沒玩過的朋友。


   

2020年12月17日 星期四

TextField判斷是否輸入Emoji

 最近剛好遇到一個需求就是禁止在TextField內輸入Emoji的需求,

當Google後發現方式滿多的,可是都不是那麼的完美(包含原生Emoji判斷),

說不完美是因為可能需求在於地址,可是iPhone的User可能習慣連動打字,

不完美的原因在於如果使用編碼判斷的方式會造成輸入時只能一個字一個字輸入...

這樣對於使用者體驗個人是覺得很差的...

所以最後使用了原生的方法 請參考:官方文件

裡面總共有以下四個判斷Emoji的方法

而我一開始使用isEmoji發現他連聲符都擋掉了...可是他所以的Emoji都可以判斷出來

最後使用isEmojiPresentation,聲符沒擋了,但是部分的Emoji是可以輸入的...

所以上方才會提說原生判斷也不是那麼完美。

以下提供我所寫的方法

func containsEmoji() -> Bool {

        for scalar in unicodeScalars {

            if !scalar.properties.isEmojiPresentation {

                continue

            }

            return true

        }

        return false

    }

而我們會在text的delegate去判斷

func textField(_ textField: UITextField, shouldChangeCharactersIn range: NSRange, replacementString string: String) -> Bool {

        

        if textField == txtAddressDetail {

            if string.containsEmoji() {

                return false

            }

        }

}

當判斷是Emoji直接return false就不會進入輸入欄位了

另外附上正則式的寫法如下:

func hasEmoji() -> Bool {

        let pattern = "[^\\u0020-\\u007E\\u00A0-\\u00BE\\u2E80-\\uA4CF\\uF900-\\uFAFF\\uFE30-\\uFE4F\\uFF00-\\uFFEF\\u0080-\\u009F\\u2000-\\u201f\\u2026\\u2022\\u20ac\r\n]"

        let pred = NSPredicate(format: "SELF MATCHES %@",pattern)

        return pred.evaluate(with: self)

    }

基本上用法是一樣的,但是不完美的就是上方所提的輸入問題,只能一個字一個字輸入。

再來就是每年可能都會有新的Emoji,Unicode要每年都去更新...

以上跟大家方享我目前所用的方法跟想法,如果您有更好的方法可以完美的呈現歡迎交流唷^^