Blog

  • Android Bluetooth Low Energy (BLE) – Part 2

    Android Bluetooth Low Energy (BLE) – Part 2

    Chào các bạn, tiếp tục loạt bài vè chủ đề BLE, hôm nay mình tiếp tục trình bày về Connection, Disconnect BLE

    1. Connection

    Sau khi bạn đã Scan thấy thiết bị của mình bằng cách quét tìm thiết bị, bạn phải kết nối với thiết bị đó bằng cách gọi connectGatt(). Nó trả về một đối tượng BluetoothGatt mà sau đó bạn sẽ sử dụng cho tất cả các hoạt động liên quan đến GATT như đọc ghi dữ liệu thông qua BLE (Read and Write). Các bạn chú ý, có 2 version của phương thức connectGatt()

    image

    Hay cụ thể hơn

    image

    Nào chúng ta cùng đi vào sâu hơn một số đối số trong hàm Connect này.

    1. “autoConnect” parameter: Đối số chỉ ra rằng bạn có muốn kết nối ngay lập tức hay không.
    • Trong trường hợp này mình set giá trị là “False” tức là ‘connect immediately’. Theo mình tìm hiểu, Android sẽ cố gắng connect trong 30s, nếu không thành công đẩy ra lỗi Timeout (Thường với mã lỗi là 133). Có một lưu ý là bạn chỉ có thể tạo một kết nối tại một thời điểm bằng cách sử dụng false, vì Android sẽ hủy mọi kết nối khác có giá trị false, nếu có.
    • Vậy nếu mình set là “True” thì sao nhỉ? Android sẽ kết nối bất cứ khi nào nó nhìn thấy thiết bị và cuộc gọi này sẽ không bao giờ Timeout. Theo mình tìm hiểu, bên trong stack sẽ tự quét và khi nhìn thấy thiết bị, nó sẽ kết nối với thiết bị đó. Bạn có thể cân nhắc đến bài toán Reconnect device trong trường hợp này. Bạn chỉ cần tạo một đối tượng BluetoothDevice và gọi connectGatt với value là “True” tương ứng với đối số autoConnect.
    1. “transport” parameter: Đưa ra các mode cho việc connect giữa Device và ứng dụng. Nếu trong trường hợp của việc kết nối BLE thì các bạn nên sử dụng value TRANSPORT_LE để tránh những lỗi không mong muốn trong quá trình connect. Có một giá trị mà các bạn cũng cần lưu ý là TRANSPORT_AUTO, hỗ trợ cho việc connection giữa điện thoại và thiết bị hỗ trợ kết nối cả BLE và Bluetooth classic.

    Sau khi mình gọi việc connect, System sẽ đẩy kết quả và các callback tương ứng thông qua BluetoothGattCallback

    image

    2. Disconnection

    Mình có tìm hiểu có rất nhiều source mẫu handle việc gọi disconnect BLE sai cách. Sau đây là các đúng mà mình tìm hiểu được nếu bạn muốn Disconnect BLE

    • Call disconnect()
    • Đợi Callback onConnectionStateChange với trạng thái Disconnect
    • Call close()
    • Dispose gatt object

    image

    image

    Lệnh disconnect() thực sự sẽ thực hiện ngắt kết nối và cũng sẽ cập nhật trạng thái kết nối nội bộ của ngăn xếp Bluetooth. Sau đó, nó sẽ kích hoạt gọi lại onConnectionStateChange để thông báo cho bạn rằng trạng thái mới hiện đã bị ‘ngắt kết nối’.

    Lệnh gọi close() sẽ hủy đăng ký BluetoothGattCallback của bạn và giải phóng ‘client interface’.

    Cuối cùng, việc xử lý đối tượng BluetoothGatt sẽ giải phóng các tài nguyên khác liên quan đến kết nối.

    Ở biết tiếp theo mình sẽ tiếp tục với việc Discovery Service BLE sau khi Connection thành công và Transfer BLE Data của BLE. Hẹn các bạn trong bài viết sắp tới.

  • MVP Architecture Pattern và biến thể MVP-C

    MVP Architecture Pattern và biến thể MVP-C

    Là một Developer, chắc hẳn các bạn đã trải qua nhiều dự án khác nhau. Thông thường khi bạn càng làm nhiều dự án bạn càng có nhiều cơ hội tiếp cận đến các loại Architecture pattern khác nhau như MVC, MVP, MVVM, VIPPER, … 

    Sau khi chinh chiến ở các dự án lớn nhỏ khác nhau mình cũng tích luỹ được một chút kiến thức về MVP Architecture pattern, vì vậy mình muốn viết một bài để chia sẻ một số kiến thức nho nhỏ mà mình đã học được về MVP cho những bạn chưa có cơ hội làm việc với MVP Architecture pattern.

    Lịch sử hình thành và phát triển

    MVP là viết tắt của Model View Presenter, nó bắt nguồn từ đầu những năm 1990 tại Talligent, một liên doanh của Apple, IBM và Hewlett-Packard. MVP là mô hình lập trình cơ bản để phát triển ứng dụng trong môi trường CommonPoint dựa trên C++ của Taligent. Sau này nó đã được Taligent chuyển sang Java.

    Đến năm 1998 thì Taligent giải thể, Andy Bower và Blair McFlashan của Dolphin Samlltalk đã điều chỉnh MVP để tạo cơ sở cho Smalltalk của họ.

    Đến năm 2006 thì Microsoft cũng bắt đầu kết hợp MVP vào tài liệu và ví dụ về lập trình giao diện người dùng trong .NET Framework.

    Đến nay thì MVP được sử dụng khá là rộng rãi vì những lợi ích mà nó đem lại cho các lập trình viên. Ngoài ra MVP cũng có rất nhiều biến thể để cải thiện những nhược điểm của nó.

    MVP là gì?

    MVP là một mẫu kiến trúc giao diện người dùng(user interface architecture pattern) được thiết kế để tạo điều kiện thuận lợi cho Automated Unit Testing(Chạy Unit Test tự động) và cải thiện việc phân tách các thành phần trong trình bày logic(presentation logic).

    MVP sinh ra dựa trên kiến trúc MVC, nó hướng tới mục tiêu cải thiện kiến trúc MVC.

    MVP được thể hiện băng hình ảnh sau:

    Model: là một interface xác định dữ liệu được hiển thị hoặc dữ liệu này được thực hiện trong giao diện người dùng.

    View: là một interface thụ động dùng để hiện thị dữ liệu của Model và định hướng các lệnh người dùng (events) tới Presenter để Presenter hành động dựa trên các dữ liệu đó.

    Presenter: hành động theo Model và View. Presenter lấy dữ liệu từ kho lưu trữ (Model), sau đó định dạng dữ liệu và hiển thị lên View.

    Ưu điểm của MVP

    Như đã nói ở trên do MVP được xây dựng dựa trên kiến trúc MVC nên nó sẽ có các ưu điểm tương tự như MVC. Các bạn có thể xem thêm về MVC ở bài viết sau: iOS Architecture Patterns: Cocoa MVC

    Mục đích cao cả của MVP sinh ra là để cải thiện những nhược điểm của kiến trúc MVC vì vậy nó giúp giảm tải lượng lớn logic nằm ở tầng Model so với mô hình MVC

    Kiến trúc MVP có tầng Presenter chuyên để xử lý các logic hiển thị, nó là thành phần trung gian tương tác với View và Model qua interface nên nó có thể viết Unit testing một cách dễ dàng.

    Nhược điểm của MVP

    Cũng như MVC, kiến trúc MVP cũng có những nhược điểm. Nhược điểm lớn nhất của MVP là càng về sau Presenter của MVP sẽ càng phình to nếu logic được thêm mới. Khí đó bạn sẽ rất khó để chia nhỏ khi presenter quá lớn.

    Biến thể MVP-C trong iOS

    Khái niệm Coordinator lần đầu tiên được đưa ra bởi Khanlou vào năm 2015, nó là một giải pháp để xử logic luồng cho View Controller.

    Dựa trên điều này kiến trúc MVP-C được ra đời với C là Coordinator làm nhiệm vụ xử lý luồng cho ứng dụng và các tầng cũ là Model, View và Presenter vẫn giống như MVP được mô tả ở trên.

    Ưu điểm của MVP-C

    View controller có thể tập trung vào mục tiêu chính của chúng. Giúp phân chia rõ ràng vai trò của View.

    Giúp giảm tải các logic trên các tầng khác, ta có thể đưa một số logic như phân luồng di chuyển màn hình từ presenter vào coordinator để giúp presenter đỡ trở nên cồng kềnh khi có quá nhiều logic. Nó đã cải thiện được nhược điểm của kiến trúc MVP truyền thống.

    Ngoài ra Coordinator cũng được ứng dụng vào các kiến trúc khác như MVC để tạo ra MVC-C và MVVM tạo ra MVVM-C.

    Tổng kết

    Đó là những kiến thức mà mình đã tích luỹ được khi làm việc với các dự án được thực hiện theo kiến trúc MVP. Mình hi vọng nó sẽ giúp ích cho các bạn khi cần thiết.

  • iOS Architecture Patterns: Cocoa MVC

    iOS Architecture Patterns: Cocoa MVC

    Là một iOS developer chắc hẳn các bạn không lạ gì với Cocoa MVC. Nó được coi là một trong những architecture pattern để phát triển ứng dụng iOS phổ biến nhất. Nó rất dễ sử dụng và được chính Apple khuyên dùng. iOS, MacOS và watchOS đều sử dụng cấu trúc này làm kiến ​​trúc mặc định để phát triển. Tuy được rất phổ biến và được Apple khuyên dùng nhưng nó cũng có những ưu điểm và nhược điểm, vì vậy bài viết này mình sẽ giới thiệu và giải thích cho các bạn về Cocoa MVC, ưu điểm, nhược điểm và khi nào nên chọn Cocoa MVC sử dụng cho ứng dụng của bạn.

    Giải thích về Cocoa MVC

    Cocoa MVC là viết tắt của Cocoa Model View Controller. Cocoa MVC gán các đối tượng trong ứng dụng iOS bằng một trong 3 vai trò sau: Model, View hoặc Controller. Mẫu kiến trúc này không chỉ xác định vai trò của các đối tượng trong ứng dụng mà nó còn xác định cả cách các đối tượng giao tiếp với nhau. Ba loại đối tượng này được phân tách khỏi các loại khác bằng các ranh giới trừu tượng và giao tiếp với các đối tượng thuộc các loại khác thông qua các ranh giới đó. Tập hợp các đối tượng của một loại MVC nhất định trong một ứng dụng đôi khi được gọi là một Layer, ví dụ: Layer model.

    Cocoa MVC

    Model

    Những đối tượng được gán với vai trò model trong mẫu kiến trúc Cocoa MVC sẽ làm nhiệm vụ đóng gói dữ liệu cụ thể cho một ứng dụng và xác định logic và tính toán để thao tác và xử lý dữ liệu đó. Những đối tượng này có thể có một hoặc nhiều mối quan hệ với các đối tượng mô hình khác và do đó, đôi khi lớp Model của một ứng dụng thực sự là một hoặc nhiều đối tượng. Phần lớn dữ liệu đều nằm ở Model sau khi nó được tải vào ứng dụng bằng các cách khác nhau(API, Files, …). Vì Model đại diện cho kiến thức và chuyên môn cho một vấn đề cụ thể nên nó có thể được tái sử dụng khi có các trường hợp tương tự. Các Model sẽ không có liên kết trực tiếp với View và View cũng không được trực tiếp sửa dữ liệu của Model mà nó sẽ phải thực hiện thông qua Controller.

    Giao tiếp: Các hành động của người dùng trên View sẽ gọi đến Controller, khi này Controller sẽ gọi đến Model tương ứng để thực hiện cập nhật dữ liệu cho Model. Khi Model thay đổi (ví dụ: dữ liệu mới được nhận qua kết nối mạng), nó sẽ thông báo cho Controller, lúc này Controller sẽ cập nhật các Views thích hợp.

    View

    View là một đối tượng trong ứng dụng mà người dùng có thể nhìn thấy. Một đối tượng View biết cách tự vẽ và có thể phản hồi các hành động của người dùng. Mục đích chính của View là hiển thị dữ liệu từ Model của ứng dụng và cho phép chỉnh sửa dữ liệu đó. Mặc dù vậy, View thường được tách rời khỏi Model trong ứng dụng MVC.

    Bởi vì bạn thường sử dụng lại và cấu hình lại chúng, nên View cung cấp tính nhất quán giữa các ứng dụng. Cả UIKit và AppKit framework đều cung cấp các bộ sưu tập View classes và Trình tạo giao diện(Interface Builder) cung cấp hàng tá view objects trong Thư viện của nó.

    Giao tiếp: View tìm hiểu về các thay đổi trong dữ liệu của Model thông qua Controller của ứng dụng và truyền đạt các thay đổi do người dùng. ví dụ: văn bản được nhập vào TextField thông qua Controller đến Model của ứng dụng.

    Controller

    Controller vai trò trung gian giữa một hoặc nhiều View của ứng dụng và một hoặc nhiều Model của nó. Do đó, cController là một đường dẫn mà qua đó View tìm hiểu về những thay đổi trong Model và ngược lại. Controller cũng có thể thực hiện các tác vụ thiết lập và điều phối cho một ứng dụng và quản lý vòng đời của các đối tượng khác.

    Giao tiếp: Controller diễn giải các hành động của người dùng được thực hiện trong View và truyền dữ liệu mới hoặc dữ liệu đã thay đổi tới Model. Khi Model thay đổi, Controller sẽ giao tiếp dữ liệu Model mới đó với View để chúng có thể hiển thị nó.

    Ưu điểm của Cocoa MVC

    Cocoa MVC có khá nhiều ưu điểm như sau:

    1. Dễ hiểu và dễ sử dụng, vì vậy ai cũng có thể làm việc với nó một cách dễ dàng kể cả người mới
    2. Được Apple khuyên dùng, vì vậy nó rất phổ biến khi gặp vấn đề sẽ dễ xử lí.
    3. Giúp developer tách source của họ ra làm các đối tượng khác nhau với 3 vai trò riêng biệt. Khi có lỗi xảy ra chúng ta sẽ khoanh vùng được nơi xảy ra lỗi.
    4. Tránh việc phải tạo một file quá dài, ảnh hưởng tới việc maintain ứng dụng
    5. Có thể tái sử dụng và mở rộng

    Nhược điểm của Cocoa MVC

    1. Phân chia nhiệm vụ giữa các vai trò không đồng đều, View chỉ làm nhiệm vụ hiển thị và nhận action từ người dùng, Controller thì chỉ là trung gian điều hướng giữa View-Model, trong khi đó Model phải làm quá nhiều việc từ lưu dữ liệu, xử lí dữ liệu, thực hiện Business Logic của ứng dụng, … Đó là lí do Model còn hay được gọi với cái tên khác là Massive
    2. Không hỗ trợ tốt cho UnitTest bởi View phải phụ thuộc vào cả Controller và Model. View sẽ không thể xử lý được vấn đề gì bởi View không thể nhận yêu cầu và cũng không có dữ liệu để hiển thị. Để tiến hành UnitTest trên View, chúng ta cần giả lập cả Controller và Model.
    3. Đối với các ứng dụng quy mô lớn, quy trình xử lý nghiệp vụ có tính phức tạp cao, lượng dữ liệu lớn thì mô hình MVC trở nên rất cồng kềnh và khó để thực hiện.

    Khi nào bạn nên sử dụng Cocoa MVC

    Như đã phân tích ở trên, các bạn cũng đã nhìn thấy cách vận hành của mô hình này, các ưu điểm và nhược điểm của nó. Vậy khi nào thì chúng ta nên sử dụng Cocoa MVC cho ứng dụng của mình. Theo mình thì sẽ các tiêu chí như sau:

    1. Khi bạn không biết về các architecture patterns khác tốt hơn, bạn chỉ hiểu rõ về MVC hoặc bạn và member trong team là người mới thì nên chọn Cocoa MVC cho dự án của mình.
    2. Dự án của bạn có kích thước vừa và nhỏ, số iOS developer có số lượng ít, có ít hiểu biết về các architecture patterns khác.

    Tổng kết

    Qua bài viết trên mình đã giới thiệu cho các bạn về một architecture pattern rất phổ biến trong lập trình ứng dụng iOS. Ngoài ra cũng giúp các bạn hiểu rõ về cách hoạt động cũng như ưu điểm và nhược điểm của Cocoa MVC. Mình hi vọng bài viết này sẽ giúp các bạn có thể hiểu rõ hơn về Cocoa MVC cũng như có những lựa chọn tốt nhất cho từng dự án mà sắp tới các bạn phát triển.

  • Mutating func trong Struct và enum – Swift

    Mutating func trong Struct và enum – Swift

    Như mọi người đã biết, struct và enum trong Swift là value types(kiểu giá trị), mặc định thì các thuộc tính của kiểu giá trị thì không thể được sửa đổi ở bên trong các phương thức thể hiện của nó(instance methods).

    Tuy nhiên nếu chúng ta cần phải chỉnh sửa các thuộc tính của struct hoặc enum trong một phương thức cụ thể, thì chúng ta sẽ đặt mutating trước các func, nó sẽ giúp các func của bạn có thể thay đổi được các thuộc tính bên trong func và khi kết thúc func giá trị sẽ được nghi lại vào các thuộc tính của struct ban đầu. Phương thức này cũng có thể gán lại một instance mới cho thuộc tính self của nó và nó sẽ được thay thế khi phương thức kết thúc.

    Mutating trong Struct

    Trong ví dụ này mình sẽ tạo một Struct có tên là Counter và tạo ra một func lấy giá trị của thuộc tính count trong Counter như sau:

    struct Counter {
        private var count: Int = 0
        
        func getCount() -> Int {
            count
        }
    }

    Đây là một ví dụ bình thường về struct, hàm getCount() ở đây không thực hiện thay đổi giá trị của struct Counter mà nó chỉ lấy giá trị của thuộc tính count theo cách thông thường.

    Vậy khi chúng ta muốn viết một hàm increase() để tăng giá trị count thông thường chúng ta sẽ viết như sau:

    Nếu là class thì sẽ không vấn đề gì vì class là reference type. Ở trường họp này do chúng ta đang viết một func chỉnh sửa thuộc tính count của struct Counter nên xCode sẽ báo lỗi rằng self ở đây là immutable(không thể thay đổi), như đã giải thích ở trên thì struct là value type nên mặc định sẽ không thể thay đổi được thuộc tính của nó trong các func của struct đó.

    func getCount() không bị báo lỗi vì func này không làm thay đổi thuộc tính trong struct.

    Để func increase() không bị báo lỗi chúng ta cần thêm mutating đằng trước func để xCode biết là func này có thể thay đổi được thuộc tính của struct:

    struct Counter {
        private var count: Int = 0
        
        func getCount() -> Int {
            count
        }
        
        mutating func increase() {
            count += 1
        }
    }
    
    var counter = Counter() // count = 0
    counter.increase() // count = 1

    Để có thể thay đổi được thuộc tính của instance counter thì chúng ta cần phải khai báo nó là var, vì func increase() sẽ thay đổi giá trị của counter vì vậy cần khai báo là var để mutating func có thể gán lại giá trị mới cho instance counter.

    Nếu chúng ta để là let Xcode sẽ thông báo lỗi không thể sử dụng mutating func trên giá trị không thể thay đổi, counter đang là một “let” constant. Do struct là value type nên nó là immutable có nghĩa là không thay đổi được, nếu chúng ta cố tình khai báo let count Xcode sẽ thông báo lỗi.

    Lỗi khi khai báo let counter để call mutating func

    NOTE: Để hiểu rõ hơn các bạn có thể xem thêm thông tin ở đây: Stored Properties of Constant Structure Instances

    Mutating func không chỉ thay đổi được thuộc tính của struct mà nó còn có thể thay đổi cả giá trị của chính instance (self)

    struct Counter {
        private var count: Int = 0
        
        func getCount() -> Int {
            count
        }
        
        mutating func increase() {
            count += 1
        }
        
        mutating func resetCounter() {
            self = Counter(count: 0)
        }
    }
    
    var counter = Counter() // count = 0
    counter.increase() // count = 1
    counter.resetCounter() // count = 0

    mutating func resetCounter() là một ví dụ, ở trong func này chúng ta thực hiện tạo ra một instance Counter mới với giá trị khởi tạo là 0 và gán lại cho chính instance gọi func này.

    Mutating func trong enum

    Tương tự như struct, enum cũng là value type và để thay đổi giá trị trong func chúng cũng cần phải sửa dụng mutating cho func đó.

    Để hiểu rõ hơn ta đi vào ví dụ sau:

    Chúng ta cần tạo ra một công tắc quạt với một tính năng là mỗi khi bấm nút thì sẽ làm thay đổi tốc độ quay của quạt một cách tuần tự và lặp đi lặp lại. Để làm theo yêu cầu chúng ta sẽ tạo enum như sau:

    enum FanStateSwitch {
        case off, low, high
        mutating func next() {
            switch self {
            case .off:
                self = .low
            case .low:
                self = .high
            case .high:
                self = .off
            }
        }
    }
    
    var fanSwitch = FanStateSwitch.off
    fanSwitch.next() // fanSwitch is low
    fanSwitch.next() // fanSwitch is high
    fanSwitch.next() // fanSwitch is off

    Tương tự như struct khi khởi tạo enum hãy nhớ khởi tạo nó với var thay vì let.

    Hi vọng bài viết sẽ giúp các bạn hiểu rõ hơn về mutating func và cách sử dụng, ứng dụng nó vào trong dự án.

    Chúc các bạn thành công!

  • Xcode 14 có gì mới? (Phần 3)

    Xcode 14 có gì mới? (Phần 3)

    Swift 5.7

    Các tính năng mới

    • Thư viện tiêu chuẩn có một Regex<Output> loại mới.Loại này đại diện cho một biểu thức chính quy mở rộng , cho phép các hoạt động xử lý chuỗi trôi chảy hơn. Bạn có thể tạo một Regex bằng cách khởi tạo từ một chuỗi :
    let pattern = "a[bc]+" // matches "a" followed by one or more instances
                           // of either "b" or "c"
    let regex = try! Regex(pattern)

    Hoặc thông qua một regex literal:

    let regex = #/a[bc]+/#

    Có các thuật toán xử lý chuỗi mới hỗ trợ các loại StringRegex và arbitrary Collection. ( SE-0350 , 93923512 )

    • Xcode 14 kích hoạt cú pháp “dấu gạch chéo” của Swift 5.7 cho các ký tự biểu thức chính quy mới được giới thiệu. Trong một số trường hợp, kết quả là mã hiện có sử dụng /làm toán tử không được biên dịch. Bạn có thể định hướng điều này bằng cách thêm dấu ngoặc đơn (/). Bạn cũng có thể vô hiệu hóa hỗ trợ cho cú pháp chữ này bằng cách bỏ chọn “Bật các chữ viết Regex Bare Slash” (SWIFT_ENABLE_BARE_SLASH_REGEX = NO) trong project’s build settings. (93460568)
    • Các thao tác chuỗi trong phiên bản 5.7 của Thư viện tiêu chuẩn Swift triển khai xác thực cải tiến cho các chỉ mục chuỗi, khắc phục một số trường hợp biên trước đây dẫn đến lỗi thời gian chạy giả. Tuy nhiên, Swift hiện chẩn đoán các nỗ lực sử dụng chỉ mục ngoài giới hạn một cách đáng tin cậy hơn và điều này có thể làm lộ các lỗi lập chỉ mục chưa được phát hiện trước đó khi bạn biên dịch lại mã bằng Xcode 14. Nếu bạn thấy lỗi “Chỉ mục chuỗi nằm ngoài giới hạn” mới sau khi xây dựng lại, mã, sau đó kiểm tra kỹ xem bạn có đang áp dụng chỉ mục cũ cho giá trị chuỗi bị thay đổi không. Trong các bản phát hành trước, các trường hợp như vậy có thể âm thầm dẫn đến các giá trị bị hỏng hoặc vô nghĩa, trong khi ở Swift 5.7, giờ đây chúng gây ra lỗi thời gian chạy một cách đáng tin cậy. (89482809)
    • Giờ đây, Swift có thể suy ra loại trình giữ chỗ được viết ở nhiều vị trí cấp cao nhất. Loại suy luận hiện được đề xuất trong bản sửa lỗi.
    // error: type placeholder may not appear in function return type
    func replaceMe() -> _ { // note: replace the placeholder with the inferred type ‘Array<Int>’
      [ 42 ]
    }
    • Xcode 13 đã cung cấp cài đặt bản dựng Swift có tên là “Tối ưu hóa thời gian tồn tại của đối tượng” không khả dụng trong Xcode 14. Nếu dự án của bạn đã tùy chỉnh cài đặt bản dựng này, thì giờ đây, nó sẽ trở thành cài đặt do người dùng xác định. Nó không có hiệu lực và bạn có thể loại bỏ nó. Xcode 14 hiện liên tục tối ưu hóa thời gian tồn tại của đối tượng. (91971848)
    • Xcode 14 giới thiệu một tính năng mới cho khả năng tương tác C: “SE-0324 Thư giãn chẩn đoán cho các đối số con trỏ đối với các hàm C.” Chẩn đoán thoải mái chỉ áp dụng cho các đối số kiểu con trỏ, không phải inout đối số. Giờ đây, chẩn đoán được nới lỏng đối với các inout đối số trong cùng điều kiện như đối số con trỏ. Ví dụ: chuyển đổi này từ một inout Int đối số sang loại tham số C const char * hiện được cho phép:
      // C declaration:
      // long read_long(const char *input);
    
      func test() -> Int {
        var x = 3
        return read_long(&x)
      }

    Trước đây Swift đã chẩn đoán chuyển đổi inout-to-pointer là lỗi:

      error: cannot convert value of type 'UnsafePointer<Int>' to expected argument type
        'UnsafePointer<CChar>' (aka 'UnsafePointer<Int8>')
        return read_long(&x)
                         ^
      note: arguments to generic parameter 'Pointee' ('Int' and 'CChar' (aka 'Int8'))
      are expected to be equal
        return read_long(&x)
                         ^

    Swift hiện hỗ trợ các tham chiếu đến optional các phương thức trên siêu dữ liệu giao thức, cũng như các tham chiếu được tra cứu động trên AnyObject siêu dữ liệu. Các tham chiếu này luôn có loại hàm chấp nhận một đối số và trả về một giá trị tùy chọn của loại hàm:

    class Object {
      @objc func getTag() -> Int
    }
    
    @objc protocol P {
      @objc optional func didUpdateObject(withTag tag: Int)
    }
    
    let getTag: (AnyObject) -> (() -> Int)? = AnyObject.getTag
    
    let didUpdateObject: (any P) -> ((Int) -> Void)? = P.didUpdateObject

    Tải dữ liệu từ bộ nhớ thô được biểu thị bằng UnsafeRawPointer, UnsafeRawBufferPointer và các đối tác có thể thay đổi của chúng hiện hỗ trợ truy cập không được phân bổ. Điều này trước đây yêu cầu một cách giải quyết liên quan đến một bản sao trung gian:

    let result = unalignedData.withUnsafeBytes { buffer -> UInt32 in
      var storage = UInt32.zero
      withUnsafeMutableBytes(of: &storage) {
        $0.copyBytes(from: buffer.prefix(MemoryLayout<UInt32>.size))
      }
      return storage
    }

    Hiện nay

    let result = unalignedData.withUnsafeBytes { $0.loadUnaligned(as: UInt32.self) }

    Ngoài ra, storeBytes(of:toByteOffset:as:) đã dỡ bỏ hạn chế căn chỉnh, do đó, giờ đây việc lưu trữ thành các phần bù tùy ý của bộ nhớ thô có thể thành công. ( SE-0349 , 93654008 )

    UnsafeRawPointer và UnsafeMutableRawPointer có chức năng mới cho số học con trỏ, thêm các chức năng để lấy con trỏ nâng cao tới ranh giới căn chỉnh tiếp theo hoặc trước đó:

    ```swift
    extension UnsafeRawPointer {
      public func alignedUp<T>(for: T.type) -> UnsafeRawPointer
      public func alignedDown<T>(for: T.type) -> UnsafeRawPointer
      public func alignedUp(toMultipleOf alignment: Int) -> UnsafeRawPointer
      public func alignedDown(toMultipleOf alignment: Int) -> UnsafeRawPointer
    }
    ```
    <!– /wp:code –> <!– wp:paragraph –> <p>Bây giờ bạn có thể sử dụng con trỏ tới <code>struct</code> để lấy con trỏ tới một trong các thuộc tính được lưu trữ của nó:</p> <!– /wp:paragraph –> <!– wp:code –> <pre class="wp-block-code"><code>“`swift withUnsafeMutablePointer(to: &myStruct) { let interiorPointer = $0.pointer(to: \.myProperty)! return myCFunction(interiorPointer) } “`

    Swift đã đơn giản hóa việc so sánh giữa các con trỏ. Vì con trỏ là biểu diễn của các vị trí bộ nhớ trong một nhóm bộ nhớ cơ bản, nên giờ đây Swift cho phép so sánh con trỏ mà không yêu cầu chuyển đổi kiểu với các toán tử ==!=<<=>và >=. ( SE-0334 , 93667321 )

    • Giờ đây, bạn có thể sử dụng phương thức withMemoryRebound<T>() này trên bộ nhớ thô, bao gồm UnsafeRawPointer, UnsafeRawBufferPointer và các đối tượng có thể thay đổi của chúng. Ngoài ra, Swift đã làm rõ ngữ nghĩa withMemoryRebound<T>() khi được sử dụng trên bộ nhớ đã nhập (UnsafePointer<Pointee>, UnsafeBufferPointer<Pointee> và các đối tác có thể thay đổi của chúng). Trong khi Swift trước đây yêu cầu Pointee và T có cùng bước tiến, giờ đây bạn có thể liên kết Pointee và T lại trong các trường hợp là tổng hợp  hoặc ngược lại. Ví dụ: được cung cấp một UnsafeMutableBufferPointer<CGPoint>, bây giờ bạn có thể sử dụng withMemoryRebound để hoạt động tạm thời trên một UnsafeMutableBufferPointer<CGFloat>, bởi vì CGPoint là một tập hợp của CGFloat. ( SE- 0333 , 93668889 )
    • Giờ đây, bạn có thể gọi một hàm chung có giá trị là loại giao thức ở những nơi trước đây không thành công vì any các loại không tuân theo giao thức của chúng. Ví dụ:
    protocol P {
      associatedtype A
      func getA() -> A
    }
    
    func takeP<T: P>(_ value: T) { }
    
    func test(p: any P) {
      takeP(p) // was an error "type 'any P' cannot conform to 'P'", now accepted
    }
    • Điều này hoạt động bằng cách mở giá trị của loại giao thức và chuyển loại cơ bản trực tiếp đến chức năng chung. ( SE-0352 , 93669112 )
    • Bây giờ, bạn có thể sử dụng một biểu thức giá trị mặc định với một loại tham số chung để làm mặc định đối số và loại của nó:
    func compute<C: Collection>(_ values: C = [0, 1, 2]) {
      ...
    }
    • Trình biên dịch hiện chấp nhận một cuộc gọi đến compute()và [Int] được suy ra C tại các trang web cuộc gọi không cung cấp đối số một cách rõ ràng. ( SE-0347 , 93669409 )
    • Giờ đây, bạn có thể sử dụng các giao thức với các loại và Self yêu cầu được liên kết dưới dạng các loại giá trị với từ khoá any. Bạn có thể gọi các phương thức giao thức trả về các loại được liên kết trên một kiểu any; kết quả là loại bị xóa đến giới hạn trên của loại được liên kết, là một kiểu any khác có cùng ràng buộc với loại được liên kết. Ví dụ:
    protocol Surface {...}
      
    protocol Solid {
      associatedtype SurfaceType: Surface
      func boundary() -> SurfaceType
    }
      
    let solid: any Solid = ...
      
    // Type of 'boundary' is 'any Surface'
    let boundary = solid.boundary()
    • Bạn không thể sử dụng các phương thức giao thức có loại được liên kết hoặc Self với any; tuy nhiên, cùng với SE-0352 , bạn có thể chuyển kiểu any cho một hàm có tham số chung bị ràng buộc với giao thức. Trong ngữ cảnh chung, các mối quan hệ kiểu là rõ ràng và bạn có thể sử dụng tất cả các phương thức giao thức. ( SE-0309 , 93928911)
    • Giờ đây, các giao thức có thể khai báo danh sách một hoặc nhiều loại được liên kết chính , cho phép viết các yêu cầu cùng loại trên các loại được liên kết đó bằng cách sử dụng cú pháp dấu ngoặc nhọn:
    protocol Graph<Vertex, Edge> {
      associatedtype Vertex
      associatedtype Edge
    }

    Giờ đây, bạn có thể viết tên giao thức theo sau là các đối số loại trong dấu ngoặc nhọn, chẳng hạn như Graph<Int, String>, ở bất kỳ đâu mà yêu cầu tuân thủ giao thức có thể xuất hiện:

      func shortestPath<V, E>(_: some Graph<V, E>, from: V, to: V) -> [E]
    
      extension Graph<Int, String> {...}
    
      func build() -> some Graph<Int, String> {}

    Tên giao thức theo sau bởi dấu ngoặc nhọn là cách viết tắt của yêu cầu tuân thủ, cùng với yêu cầu cùng loại đối với các loại liên kết chính của giao thức. Hai ví dụ đầu tiên ở trên tương đương như sau:

      func shortestPath<V, E, G>(_: G, from: V, to: V) -> [E]
        where G: Graph, G.Vertex == V, G.Edge == E
    
      extension Graph where Vertex == Int, Edge == String {...}
    • Hàm build()trả về some Graph<Int, String> không thể được viết bằng một mệnh đề where; đây là một ví dụ về loại kết quả không rõ ràng bị ràng buộc, đây là biểu thức mới trong Swift 5.7. ( SE-0346 , 93929372 )
    • Bạn có thể sử dụng các giao thức áp dụng các loại được liên kết chính với any để bật các loại tồn tại bị ràng buộc. Ví dụ:
    let strings: any Collection<String> = [ "Hello" ]

    Điều này làm cho việc viết các trình bao bọc xóa kiểu cho mã chung đơn giản hơn nhiều vì một loại trình bao bọc riêng biệt không còn cần thiết nữa:

    protocol Producer<T> {
      associatedtype T
      func produce() -> T
    }
    
    typealias AnyProducer<T> = any Producer<T>
    
    /*
    struct AnyProducer<T> {
      var wrappedProduce: () -> T
    }
    */

    Giờ đây, bạn có thể sử dụng các giao thức với các loại được liên kết chính trong các loại tồn tại, cho phép các ràng buộc cùng loại đối với các loại được liên kết đó.

    let strings: any Collection<String> = [ "Hello" ]
    • Lưu ý rằng các tính năng ngôn ngữ yêu cầu hỗ trợ thời gian chạy như diễn viên động ( isas?as!), cũng như cách sử dụng chung của các tồn tại được tham số hóa trong các loại chung (ví dụ: Array<any Collection<Int>>) liên quan đến việc kiểm tra tính khả dụng bổ sung để sử dụng. Giờ đây, bạn có thể triển khai lại các tập quán ở các vị trí chung với cấu trúc trình bao bọc xóa kiểu chung, giờ đây việc triển khai đơn giản hơn nhiều:
    struct AnyCollection<T> {
      var wrapped: any Collection<T>
    }
    
    let arrayOfCollections: [AnyCollection<T>] = [ /**/ ]

    Giờ đây, bạn có thể sử dụng các loại không rõ ràng trong tham số của hàm và chỉ số con khi chúng cung cấp cú pháp tốc ký để giới thiệu tham số chung. Ví dụ như sau:

    func horizontal(_ v1: some View, _ v2: some View) -> some View {
      HStack {
        v1
        v2
      }
    }

    tương đương với:

    func horizontal<V1: View, V2: View>(_ v1: V1, _ v2: V2) -> some View {
      HStack {
        v1
        v2
      }
    }

    Với điều này, some trong một loại tham số cung cấp một khái quát hóa trong đó người gọi chọn loại tham số cũng như giá trị của nó, trong khi some ở loại kết quả cung cấp một khái quát hóa trong đó người gọi chọn loại và giá trị kết quả. ( SE- 0341 , 93675336 )

    Giờ đây, bạn có thể suy ra các loại tham số và kết quả từ phần thân của một bao đóng đa câu lệnh. Không còn sự phân biệt giữa các lần đóng đơn và đa câu lệnh.

    Việc sử dụng các bao đóng trở nên ít cồng kềnh hơn bằng cách loại bỏ nhu cầu liên tục chỉ định các loại bao đóng rõ ràng, đôi khi có thể khá lớn (ví dụ: khi có nhiều tham số hoặc loại kết quả bộ dữ liệu phức tạp).

    Ví dụ:

    func map<T>(fn: (Int) -> T) -> T {
      return fn(42)
    }
    
    func computeResult<U: BinaryInteger>(_: U) -> U { /* processing */ }
    
    let _ = map {
      if let $0 < 0 {
         // Do some processing.
      }
    
      return computeResult($0)
    }
    • Giờ đây, bạn có thể suy ra loại kết quả map từ phần thân của dấu đóng được truyền dưới dạng đối số. ( SE-0326 , 93669647 )
    • Giờ đây, bạn có thể hủy bao bọc các biến tùy chọn bằng cú pháp tốc ký làm ẩn phần khai báo hiện có. Ví dụ như sau:
    let foo: String? = "hello world"
    
    if let foo {
      print(foo) // Prints "hello world”.
    }

    tương đương với:

    let foo: String? = "hello world"
    
    if let foo = foo {
      print(foo) // Prints "hello world”.
    }
    • Giờ đây, bạn có thể làm cho các khai báo không khả dụng để sử dụng trong ngữ cảnh không đồng bộ với thuộc tính @available(*, noasync).
      Điều này bảo vệ người dùng API chống lại hành vi không xác định có thể xảy ra khi API sử dụng hoặc khuyến khích sử dụng lưu trữ cục bộ theo luồng trên các điểm tạm ngưng. Nó cũng bảo vệ các nhà phát triển chống lại việc giữ khóa trên các điểm treo, điều này có thể dẫn đến hành vi không xác định, đảo ngược mức độ ưu tiên hoặc deadlocks. ( SE-0340 , 93673989 )
    • Tập lệnh cấp cao nhất hiện hỗ trợ cuộc gọi không đồng bộ. Sử dụng một await bằng cách gọi một hàm không đồng bộ hoặc truy cập một biến bị cô lập sẽ chuyển cấp cao nhất sang ngữ cảnh không đồng bộ. Là một bối cảnh không đồng bộ, các biến cấp cao nhất được cách ly và cấp cao nhất được chạy trên .@MainActor@MainActor Lưu ý rằng quá trình chuyển đổi ảnh hưởng đến độ phân giải quá tải chức năng và bắt đầu một vòng lặp chạy ẩn để điều khiển máy móc đồng thời. Các tập lệnh chưa sửa đổi không bị ảnh hưởng bởi thay đổi này trừ khi -warn-concurrency được chuyển đến lời gọi trình biên dịch. Với -warn-concurrency, các biến ở cấp cao nhất được tách biệt với tác nhân chính và bối cảnh cấp cao nhất được tách biệt với tác nhân chính nhưng không phải là ngữ cảnh không đồng bộ. ( SE-0343 , 93674157 )
    • Swift hiện hỗ trợ lập trình phân tán với việc giới thiệu các tác nhân phân tán. Để biết thêm thông tin, hãy xem SE-0336 , SE-0344 . (70840120)
    • Bây giờ bạn có thể khai báo distributed actor và distributed func bên trong tệp distributed actor. Các tác nhân phân tán cung cấp các đảm bảo cách ly mạnh hơn so với các tác nhân “cục bộ” và chúng cho phép thực hiện các kiểm tra bổ sung đối với các loại trả về và tham số của các phương thức phân tán, ví dụ: kiểm tra xem chúng có tuân thủ Codable. Bạn có thể gọi các phương thức phân tán trên các tham chiếu “từ xa” của các tác nhân phân tán, biến các lệnh gọi đó thành các lệnh gọi thủ tục từ xa, bằng cách triển khai hệ thống tác nhân phân tán có thể cắm và mở rộng cho người dùng. Bản thân Swift không cung cấp bất kỳ hệ thống tác nhân phân tán cụ thể nào; tuy nhiên, các gói trong hệ sinh thái hoàn thành vai trò cung cấp các triển khai đó.
    distributed actor Greeter { 
      var greetingsSent = 0
      
      distributed func greet(name: String) -> String {
        greetingsSent += 1
        return "Hello, \(name)!"
      }
    }
    
    func talkTo(greeter: Greeter) async throws {
      // The isolation of distributed actors is stronger. You can't refer to
      // any stored properties of distributed actors from outside of them.
      greeter.greetingsSent // You can't access the distributed actor-isolated property 'name' from a non-isolated context.
      
      // Remote calls are implicitly throwing and async, 
      // to account for the potential networking involved.
      let greeting = try await greeter.greet(name: "Alice")
      print(greeting) // Hello, Alice!
    }

    Trình khử khởi tạo, hầu hết các trình khởi tạo cho actor các loại và các loại bị hạn chế bởi một tác nhân toàn cầu như @MainActor các quy tắc đã sửa đổi về những biểu thức nào được phép trong phần thân của chúng. Là một phần của SE-327, mục tiêu của các bản sửa đổi này là cải thiện tính an toàn và biểu cảm của ngôn ngữ. Nhiều mẫu lập trình khác hiện được cho phép trong các bộ khởi tạo này.

    Ví dụ: trình khởi tạo không đồng bộ của phiên bản actor trước Swift 5.7 đã đưa ra chẩn đoán bất kỳ lúc nào self thoát khỏi trình khởi tạo trước khi quay lại. Mục đích của chẩn đoán đó là để bảo vệ chống lại một cuộc chạy đua dữ liệu có thể xảy ra khi truy cập các thuộc tính được lưu trữ riêng biệt, nhưng nó đã được phát ra ngay cả khi không có quyền truy cập nguy hiểm.

    Trong Swift 5.7, trình biên dịch hiện kiểm tra các trình khởi tạo này để tìm quyền truy cập nguy hiểm vào các thuộc tính được lưu trữ bị cô lập xảy ra sau khi thoát self:

    actor Database {
      // ... other properties ...
      var rows: Int = 0
    
      init(_ world: DataUser) {
        defer { 
          print("last = \(self.rows)") // ❌ This access to 'rows' is illegal.
        }
        
        print("before = \(self.rows)") // ✅ This access to 'rows' is OK.
        world.publishDatabase(self)    // ✅ Passing 'self' is OK in Swift 5.7+.
        print("after = \(self.rows)")  // ❌ This access to 'rows' is illegal. 
        
        Task { [weak self] in          // ✅ Capturing 'self' is OK in Swift 5.7+.
          while let db = self { await db.prune() }
        }
      }
    }

    Đây là kiểm tra nhạy cảm với luồng điều khiển, nghĩa là truy cập bất hợp pháp không nhất thiết phải xuất hiện trên dòng nguồn sau khi thoát self(trong ví dụ trên, xem xét thời điểm defer thực thi). Trình biên dịch luôn chỉ ra một trong những lối thoát self khiến quyền truy cập trở thành bất hợp pháp.

    Ngoài ra, các công cụ khởi tạo ủy quyền của một actor không còn luôn luôn không bị cô lập. Điều này có nghĩa là trình async khởi tạo ủy quyền có thể thực hiện những việc tương tự như trình khởi tạo không ủy quyền. (84476555)

    • Trình khởi tạo actor không còn yêu cầu viết từ khóa convenience để ủy quyền (SE-327). Trước Swift 5.7, việc thêm hoặc xóa convenience đối với public init của một diễn viên là một thay đổi không linh hoạt, đối với các thư viện được biên dịch có bật tiến hóa. Các thư viện được biên dịch cho Swift 5.7+ hiện có khả năng phục hồi trước những thay đổi trong quá trình triển khai các trình khởi tạo đó để ủy quyền hay không và các chương trình hiện có được biên dịch cho Swift 5.7+ sẽ không yêu cầu biên dịch lại. (87567878)
    • Các loại mới đại diện cho thời gian và đồng hồ hiện đã có sẵn. Điều này bao gồm một giao thức Clock để xác định đồng hồ, cho phép bạn xác định khái niệm về “hiện tại” và cách đánh thức sau một khoảng thời gian nhất định. Một giao thức InstantProtocol mới để xác định thời gian cũng có sẵn. Và một giao thức DurationProtocol mới để xác định khoảng thời gian đã trôi qua giữa hai loại InstantProtocol nhất định cũng có sẵn. Các loại để sử dụng chung là phổ biến nhất ClockSuspending và ClockContinuous, đại diện cho các đồng hồ cơ bản nhất cho hệ thống. Loại ClockSuspendingClock không tiến triển khi máy bị treo, ngược lại ContinuousClock tiến triển bất kể trạng thái của máy.
    func delayedHello() async throws {
      try await Task.sleep(until: .now + .milliseconds(123), clock: .continuous)
      print("hello delayed world")
    }

    Clock cũng có các phương pháp để đo thời gian thực hiện công việc đã trôi qua. Trong trường hợp của SuspendingClock và ContinuousClock điều này đo lường với độ phân giải cao và phù hợp với điểm chuẩn.

    let clock = ContinuousClock()
    let elapsed = clock.measure {
      someLongRunningWork()
    }

    Trình biên dịch hiện phát ra cảnh báo khi một lớp không phải là lớp cuối cùng tuân theo một giao thức áp đặt yêu cầu cùng loại giữa Selfvà một loại được liên kết. Loại yêu cầu này làm cho sự phù hợp không hợp lý cho các lớp con.

    Ví dụ: Swift 5.6 cho phép đoạn mã sau, trong thời gian chạy mã này sẽ xây dựng một phiên bản C và sub C không như mong đợi:

    protocol P {
      associatedtype A : Q where Self == Self.A.B
    }
    
    protocol Q {
      associatedtype B
    
      static func getB() -> B
    }
    
    class C : P {
      typealias A = D
    }
    
    class D : Q {
      typealias B = C
    
      static func getB() -> C { return C() }
    }
    
    extension P {
      static func getAB() -> Self {
        // This is well-typed because `Self.A.getB()` returns
        // `Self.A.B`, which is equivalent to `Self`.
        return Self.A.getB()
      }
    }
    
    class SubC : C {}
    
    // P.getAB() declares a return type of `Self`, so it should
    // return `SubC`, but it actually returns a `C`.
    print(SubC.getAB())
    • Để làm cho ví dụ trên đúng, class C cần phải trở thành final(trong trường hợp đó SubC không thể được khai báo) hoặc giao thức P cần được thiết kế lại để không bao gồm yêu cầu cùng loại Self == Self.A.B. (93675134)
    • Trình biên dịch hiện đưa ra chính xác các cảnh báo cho nhiều biểu thức hơn khi sử dụng tuân thủ giao thức và có thể không khả dụng khi chạy. Trước đây, các biểu thức tham chiếu thành viên và biểu thức xóa loại sử dụng các tuân thủ có khả năng không khả dụng chưa được chẩn đoán, dẫn đến các sự cố có thể xảy ra trong thời gian chạy.
    struct Pancake {}
    protocol Food {}
    
    extension Food {
      var isGlutenFree: Bool { false }
    }
    
    @available(macOS 12.0, *)
    extension Pancake: Food {}
    
    @available(macOS 11.0, *)
    func eatPancake(_ pancake: Pancake) {
      if (pancake.isGlutenFree) { // Warning: Conformance of 'Pancake' to 'Food' is only available in macOS 12.0 or newer.
        eatFood(pancake) // Warning: Conformance of 'Pancake' to 'Food' is only available in macOS 12.0 or newer.
      }
    }
    
    func eatFood(_ food: Food) {}

    Các loại mờ (được biểu thị bằng some) hiện có thể được sử dụng ở các vị trí cấu trúc trong một loại kết quả, bao gồm cả việc có nhiều loại mờ trong cùng một kết quả. Ví dụ:

    func getSomeDictionary() -> [some Hashable: some Codable] {
      return [ 1: "One", 2: "Two" ]
    }

    Các giao thức khác nhau trong thư viện chuẩn hiện khai báo các loại liên kết chính, ví dụ Sequencevà Collectionkhai báo một loại liên kết chính duy nhất Element. Ví dụ: điều này cho phép ghi lại các loại some Collection<Int>và any Collection<Int>. ( SE-0358 , 93929895 )

    Vấn đề đã giải quyết

    • Các thuộc tính được lưu trữ trong Swift không được có thông tin loại có khả năng không khả dụng trong thời gian chạy. Tuy nhiên, trước Swift 5.7, trình biên dịch đã chấp nhận lỗi thuộc tính @available trên thuộc tính được lưu trữ khi thuộc tính có công cụ lazy sửa đổi hoặc trình bao bọc thuộc tính được đính kèm. Điều này có thể dẫn đến sự cố cho các ứng dụng chạy trên hệ điều hành cũ hơn. Trình biên dịch Swift hiện từ chối @available trên tất cả các thuộc tính được lưu trữ. (82713248) (FB9594187)
    • Phiên bản không đồng bộ addTeardownBlock của phương pháp trong XCTestCase hiện đã có. (85453819) (FB9762503)
    • Chẩn đoán về các biểu thức giá trị mặc định không tách biệt được giới thiệu cho Swift 5.6 trong bản phát hành Xcode 13.3 không còn nữa. Quy tắc được đề xuất trong SE-0327 không đủ chính xác để tránh gắn cờ một mẫu phổ biến nhưng vô thưởng vô phạt trong mã SwiftUI liên quan đến các thuộc tính @StateObject và @MainActor. (88971160)
    • Quá trình tái cấu trúc “Tạo Trình khởi tạo Memberwise” thêm chính xác các tham số cho các biến được đánh dấu bằng trình bao bọc thuộc tính. (89057767) (FB9910083)
    • Các thao tác chuỗi trong phiên bản 5.7 của Swift Standard Library triển khai xác thực cải tiến cho các chỉ mục chuỗi, khắc phục một số trường hợp biên trước đây dẫn đến lỗi thời gian chạy giả. Tuy nhiên, Swift hiện chẩn đoán các nỗ lực sử dụng chỉ mục ngoài giới hạn một cách đáng tin cậy hơn và điều này có thể làm lộ các lỗi lập chỉ mục chưa được phát hiện trước đó khi bạn biên dịch lại mã bằng Xcode 14. Nếu bạn thấy lỗi “Chỉ mục chuỗi nằm ngoài giới hạn” mới sau khi xây dựng lại, mã, sau đó kiểm tra kỹ xem bạn có đang áp dụng chỉ mục cũ cho giá trị chuỗi bị thay đổi không. Trong các bản phát hành trước, các trường hợp như vậy có thể âm thầm dẫn đến các giá trị bị hỏng hoặc vô nghĩa, trong khi ở Swift 5.7, giờ đây chúng gây ra lỗi thời gian chạy một cách đáng tin cậy. (89482809)
    • Khi xây dựng các thể hiện Chuỗi từ các chuỗi C, giờ đây Swift thực thi nghiêm ngặt việc chấm dứt null của bộ đệm đầu vào nếu bạn chuyển một đối số bằng cách chuyển đổi con trỏ. Hơn nữa, inout-to-pointer chuyển đổi hiện không được dùng cho cấu trúc Chuỗi. Các chức năng bị ảnh hưởng là String.init(cString:), String.init?(validatingUTF8:), String.decodeCString(_:as:repairingInvalidCodeUnits:) và String.init(decodingCString:as:). (90336023)
    • Chỉnh sửa tất cả trong Phạm vi đổi tên chính xác tất cả các lần xuất hiện của các biến được ghi lại bằng cách sử dụng cú pháp chụp tốc ký bao đóng [capturedVariable] hoặc cú pháp tốc ký if let optionalVariable. (91311033)
    • Bạn có thể sử dụng các giao thức áp dụng các loại được liên kết chính với từ khóa any để bật các loại tồn tại bị ràng buộc.Ví dụ:
    let strings: any Collection<String> = [ "Hello" ]

    Điều này làm cho việc viết các trình bao bọc xóa kiểu cho mã chung đơn giản hơn nhiều vì một loại trình bao bọc riêng biệt không còn cần thiết nữa:

    protocol Producer<T> {
      associatedtype T
      func produce() -> T
    }
    
    typealias AnyProducer<T> = any Producer<T>
    
    /*
    struct AnyProducer<T> {
      var wrappedProduce: () -> T
    }
    */

    Trình biên dịch Swift không còn cảnh báo về các yêu cầu dư thừa trong các khai báo chung. Ví dụ: trong phiên bản beta 1, đoạn mã sau đã chẩn đoán một cảnh báo về T.Iterator : IteratorProtocol yêu cầu là dư thừa, bởi vì nó được ngụ ý bởi T : Sequence:

    func firstElement<T: Sequence>(_: T) -> T.Element where T.Iterator: IteratorProtocol {...}

    Một yêu cầu dư thừa không chỉ ra lỗi mã hóa và đôi khi nên đánh vần chúng cho mục đích tài liệu. Vì lý do này, những cảnh báo này hiện mặc định bị tắt.

    Bạn có thể nhận hành vi trước đó và bật lại các cảnh báo này bằng cách đặt OTHER_SWIFT_FLAGS-X tùy chọn bản dựng trong Xcode thành “ frontend -warn-redundant-requirements”. (92092635)

    • Các phép chuyển động ( isas!as?) đến và từ các loại tồn tại được tham số hóa hiện kiểm tra chính xác các ràng buộc đối với các loại này. (92197049)
    • Đã sửa lỗi: Một số câu lệnh switch phức tạp nhất định trong phần đóng nhiều câu lệnh có thể khiến trình biên dịch gặp sự cố nếu chúng chứa câu lệnh fallthrough yêu cầu suy luận kiểu tại đích của nó, ví dụ: đích case sử dụng khớp mẫu với let các liên kết như case (let ..., let ...). (93796211)
    • Swift không thực hiện kiểm tra Có thể gửi khi thoát khỏi một actor để gọi vào mã không đồng bộ, không bị cô lập. Ví dụ:
    func f(_: NS) async { }
    
    actor A {
      func g(_ ns: NS) async {
        await f(ns) // Warn about passing non-Sendable type 'NS' to a non-isolated async function.
      }
    }

    Vấn đề đã biết

    • #if canImport(AudioVideoBridging)không biên dịch trên Mac Catalyst. (89289575)
      Giải pháp thay thế : Sử dụng #if os(macOS) để hạn chế AudioVideoBridging code cho các nền tảng được hỗ trợ.
    • ~= và các trường hợp trong switches thành công nếu toàn bộ chuỗi khớp, thay vì nếu có một kết quả khớp bên trong chuỗi. Điều này có thể thay đổi tùy thuộc vào kết quả của Swift Evolution cho SE-0357 . (93918632)
    • Trong Swift, một số kiểu chữ Foundation cho các loại chức năng (chẳng hạn như NSItemProvider.CompletionHandler) sẽ thấy @Sendable các loại chức năng, điều này có thể dẫn đến các cảnh báo không mong muốn trong mã chưa áp dụng Đồng thời Swift. Để loại bỏ các cảnh báo, hãy thay thế một tham chiếu đến bí danh kiểu chữ trong mã nguồn Swift bằng kiểu bên dưới của nó mà không có phần mở rộng @Sendable. (98343624)

      Ví dụ:
      thay thế: var completion: NSItemProvider.CompletionHandler
      bằng: var completion: (NSSecureCoding?, Error?) -> Void

    Bài viết được dịch dựa trên nội dung release note xCode14 của Apple
    Refer: https://developer.apple.com/documentation/xcode-release-notes/xcode-14-release-notes

  • Android Bluetooth Low Energy (BLE)

    Android Bluetooth Low Energy (BLE)

    Chào các bạn, hiện tại mình đang làm một dự án liên quan đến BLE (Bluetooth Low Energy). Mình tìm trên mạng thì thấy khá nhiều tài liệu liên quan đến Bluetooth Classic, còn tài liệu về BLE thì còn hạn chế hơn rất nhiều. Nhưng mình lại để ý rằng các ứng dụng phục vụ kết nối với thiết bị thì hiện tại sử dụng BLE là một giải pháp kết nối là khá thông dụng. Vậy nên hôm nay mình sẽ chia sẻ một chút kiến thức cơ bản liên quan đến chủ đề này. Đâu đấy sẽ giúp các bạn có thể Overview qua về hướng tiếp cận, để từ đó dễ dàng investigate và triển khai theo yêu cầu.

    Bài của mình liên bao gồm những phần chính sau:

    1. Giới thiệu về BLE
    2. Find BLE Device
    3. Connection and Disconnect BLE Device
    4. Transfer BLE Data

    Let’s start

    1. Giới thiệu về BLE

    Đặt vấn đề, trong hầu hết các trường hợp, các nhà thiết kế thiết bị có thể đeo, ngoại vi, cũng như tất cả các mặt hàng khác cần mở rộng chức năng của chúng với điện thoại thông minh. Đều cần tìm một giải pháp để kết nối, điều khiển và chia sẻ dữ liệu. Và Bluetooth Classic và BLE chính là giải pháp. Trong phạm trù bài chia sẻ hôm nay mình chỉ tập trung vào nền tảng Android. Như các bạn đã biết, Bluetooth Classic cho phép thiết bị trao đổi dữ liệu không dây với các thiết bị Bluetooth khác. Khung ứng dụng cung cấp quyền truy cập vào chức năng Bluetooth thông qua API Bluetooth. Các API này cho phép các ứng dụng kết nối với các thiết bị Bluetooth khác, cho phép các tính năng không dây điểm-điểm và đa điểm.

    Android version 4.3 (API 18) and above BLE ra đời. BLE được thiết kế để tiêu thụ điện năng thấp hơn đáng kể. Điều này cho phép các ứng dụng giao tiếp với các thiết bị BLE có yêu cầu năng lượng nghiêm ngặt hơn. Cụ thể: Các trường hợp sử dụng phổ biến bao gồm:

    • Truyền một lượng nhỏ dữ liệu giữa các thiết bị lân cận.
    • Tương tác với các cảm biến tiệm cận để cung cấp cho người dùng trải nghiệm tùy chỉnh dựa trên vị trí hiện tại của họ.

    2. Find BLE Device

    Đây là bước đầu tiên trước khi sử dụng các tính năng Bluetooth Classic hoặc BLE. Các bạn cần đảm bảo rằng tất cả các quyền và tính năng cần thiết đều được áp dụng và cho phép.

    Permission:

    • android.permission.BLUETOOTH – các tính năng Bluetooth classic và BLE cơ bản
    • android.permission.BLUETOOTH_ADMIN – các thao tác BC và BLE nâng cao như bật/tắt module _ Bluetooth, discovery device, ….
    • android.permission.ACCESS_COARSE_LOCATION – cần thiết để quét BLE trên Android 5.0 (API 21) trở lên. Lưu ý: Đây là Runtime Permission từ API 23.

    Mình có đọc thêm chú ý về một quyền mà mình thấy hay được khai báo khi xử lý vs BLE. Các bạn chú ý cũng cần đảm bảo rằng điện thoại hay Tablet có bộ điều hợp Bluetooth tích hợp. Nếu muốn không khả dụng cho các thiết bị không có Bluetooth, chúng tôi chỉ cần thêm một khai báo:

    <uses-feature android:name="android.hardware.bluetooth"/>
    

    Tuy nhiên, theo Document mình tìm hiểu, không bắt buộc nếu khai báo quyền Bluetooth và đặt phiên bản Android 5 trở lên.

    image

    Để tìm các thiết bị BLE, bạn sử dụng startScan()stopScan(). Phương thức này lấy leScanCallback làm tham số. Bạn phải triển khai lệnh gọi lại này vì đó là cách trả về kết quả quét. Chú ý đến việc optimize khi thực thi, ví dụ stopScan() khi đã tìm thấy device mà các bạn muốn hoặc đặt một interval time scan nhất định …

    image

    Bạn cũng có thể cân nhắc đến mode khi Scan trong trường hợp xuống Background và lên ForceGround tương ứng để optimize năng lượng của thiết bị.

    image

    Ở biết tiếp theo mình sẽ tiếp tục với việc Connection, Disconnect BLE Device và Transfer BLE Data của BLE.

    Hẹn các bạn trong bài viết sắp tới.

  • XCode 14 có gì mới? (Phần 2)

    XCode 14 có gì mới? (Phần 2)

    Interface Builder

    Các tính năng mới

    • UISplitViewController hiện hỗ trợ sidebars trong các ứng dụng Mac được xây dựng bằng Mac Catalyst. Để bật sidebars, hãy đặt Primary Style trong trình kiểm tra thuộc tính của bộ điều khiển chế độ split view. (82004740)
    • Trình tạo giao diện hiện hỗ trợ các nhóm mục trung tâm mới trên UINavigationItem. (83252931)
    • Đã thêm một checkbox trong Attributes inspector để bật giao diện người dùng tìm và thay thế UI cho UITextView. (83726669)
    • Trình tạo giao diện hiện cập nhật các cảnh không đồng bộ. (83786577)
    • Trình tạo giao diện hiện hỗ trợ tác giả NSComboButton. (85583290)
    • Trình tạo giao diện hiện hỗ trợ tác giả với MKMapView, MKMapConfiguration including standard, imagery và hybrid. (85607049)
    • Chuyển bàn phím từ trình kiểm tra thuộc tính của UIViewController để hiểu cách bàn phím ảnh hưởng đến hướng dẫn bố cục trong canvas. (87975498)
    • Một hộp kiểm mới xuất hiện để bật giao diện người dùng tìm và thay thế tiêu chuẩn. (88049266)
    • Giờ đây, bạn có thể chỉnh sửa cấu hình mặc định của Biểu tượng SF (bao gồm phông chữ, tỷ lệ và trọng lượng) bằng NSButton và NSImageView bằng cách chọn biểu tượng thông qua trình kiểm tra hình ảnh của điều khiển. (88400241)
    • Hỗ trợ cho iOS để cho phép dán nội dung bằng một lần nhấn mà không cần thông báo hoặc cảnh báo dán. Điều khiển có thể nhắm mục tiêu bất kỳ đối tượng nào tuân theo UIPasteConfigurationSupporting (ví dụ: UIResponder) để nhận nội dung đã dán. (88648426)
    • Truy cập và tìm kiếm Ký hiệu SF thông qua tab thư viện ký hiệu. Mở thư viện (Xcode > View > Show Library) và nhấp vào tab Symbols. Bạn có thể kéo các biểu tượng vào source editor. (88726368)
    • Trình tạo giao diện bao gồm NSColorWell có các kiểu mặc định, tối thiểu và mở rộng mới trong macOS 13. (89051231)
    • Trình tạo giao diện hiện hỗ trợ tác giả MKPointOfInterestFilter. (89368386)
    • Trình tạo giao diện hiện hỗ trợ tác giả MKLookAroundViewController. (90994596)
    • Trình tạo giao diện hiện hỗ trợ tác giả RoomCaptureView. (91640003)

    Vấn đề đã giải quyết

    • UIBarButtonItemGroup đã được thêm vào thư viện đối tượng Interface builder. Nó có thể được kéo vào một UINavigationItem để cung cấp các Center Items. (19160962)
    • UIColorWell hiện có sẵn trong thư viện đối tượng. Khi chạm vào, điều khiển sẽ hiển thị bộ chọn màu. (67016855)
    • Giờ đây, bạn có thể bật hướng dẫn bố cục bàn phím trên cảnh UIView thông qua trình kiểm tra kích thước. Hạn chế chế độ xem đối với hướng dẫn bố cục để chúng điều chỉnh trong quá trình bố trí khi bàn phím hiển thị trên màn hình. (81959069) (FB9514618)
    • Tạo khả năng hiển thị phác thảo cho các tài liệu mới phù hợp với trạng thái được chuyển đổi của người dùng cuối cùng. (82857926) (FB9607879)
    • Đã khắc phục sự cố với kết nối ổ cắm và hành động với AppDelegates dựa trên AppleScript. (83373726) (FB9643535)
    • Đã khắc phục sự cố khi lưu tài liệu có nhiều Trình điều khiển Chế độ xem Nhìn xung quanh sẽ làm Xcode bị lỗi. (92304543)
    • Bảng phân cảnh WatchKit không được dùng trong watchOS 7.0 trở lên. Vui lòng chuyển sang SwiftUI và Vòng đời SwiftUI. (94058186)
    • Chế độ xem phác thảo của Interface builder hiện lưu và khôi phục trạng thái hiển thị/độ rộng trên toàn cầu thay vì trên mỗi tài liệu. (97084370)

    Vấn đề đã biết

    • Kích thước phông chữ không thay đổi để phù hợp với kích thước kiểm soát cho NSComboButton. (94610724) (FB10094813)

    Linking

    Vấn đề đã biết

    • Các ứng dụng Swift được xây dựng bằng Xcode 14 có thể không liên kết được với các tệp libswiftFoundation.dylib. Điều này có thể khiến ứng dụng hoạt động sai khi chạy trên các hệ điều hành trước macOS 13 Ventura và iOS 16, bao gồm cả việc in các chuỗi không chính xác và đưa ra các ngoại lệ đối với các phương thức bị thiếu trên các loại dữ liệu của Tổ chức. (99457165)
      Giải pháp thay thế : Tham chiếu rõ ràng một ký hiệu từ trong mã libswiftFoundation, ví dụ bằng cách thêm vào một hàm _ = JSONDecoder()

    Localization

    Các tính năng mới

    • Bây giờ bạn có thể xuất Swift Packages cục bộ để bản địa hóa. Xcode tạo một danh mục bản địa hóa duy nhất cho tất cả các dự án và Swift Packages có trong một project hoặc workspace. Bạn cũng có thể sử dụng xcodebuild -importLocalizations và xcodebuild -exportLocalizations để export hoặc import gói Swift. (56355281)

    Vấn đề đã giải quyết

    • Các chuỗi được bản địa hóa chỉ định rõ ràng một bảng tham số không phải là chuỗi ký tự không còn được trích xuất khi chạy genstrings hoặc xuất để bản địa hóa. Trước đây, genstrings mặc định là một bảng có tên Localizable.strings. (65063595)
    • Đã khắc phục sự cố trong đó Xcode đôi khi không phát sinh lỗi khi xuất .strings tệp không đúng định dạng để bản địa hóa. (85278818)
    • Đã khắc phục sự cố trong đó Xcode âm thầm bỏ qua XLIFF không đúng định dạng khi nhập bản địa hóa. (86849358) (FB9819403)
    • xcodebuild -importLocalizations và xcodebuild -exportLocalizations hiện bao gồm dấu thời gian cho các hoạt động bản địa hóa. (89373526)
    • Xcode tự động trích xuất NSHumanReadableDescription từ ​​các tệp Info.plist khi xuất để bản địa hóa. (89591666) (FB9935770)
    • Các khóa trong .strings tệp tồn tại trong một .stringsdict với NSStringDeviceSpecificRuleType không còn được đánh dấu là translate=”no” trong XLIFF đã xuất, bởi vì những khóa này cũng rơi trở lại giá trị tệp .strings khi chạy. (90785024)
    • Đã khắc phục sự cố trong đó Xcode đưa ra cảnh báo đối với các đơn vị chưa được dịch được đánh dấu là translate="no"khi nhập bản địa hóa. (91692843) (FB9982115)

    Metal

    Các tính năng mới

    • TextureConverter 2.0 thêm hỗ trợ giải nén kết cấu, số liệu lỗi kết cấu nâng cao và hỗ trợ đọc và ghi tệp KTX2.Thư viện AppleTextureConverter mới giúp TextureConverter có sẵn để tích hợp vào các công cụ và công cụ của bên thứ ba. (82244472)

    Vấn đề đã biết

    • Hồ sơ Chụp kim loại có chứa pipeline bị vô hiệu hóa. (93255574)

    Organizer

    Các tính năng mới

    • Phản hồi ảnh chụp màn hình TestFlight hiện có sẵn trong Xcode Organizer. Giờ đây, bạn có thể xem ảnh chụp màn hình và phản hồi bằng văn bản cho iOS và macOS trong Xcode bằng cách chọn mục “Phản hồi” trong phần báo cáo của thanh bên. Để bắt đầu, hãy đăng nhập bằng tài khoản Nhà phát triển Apple được liên kết với ứng dụng TestFlight của bạn và mở Trình tổ chức bằng cách chọn Cửa sổ > Trình tổ chức trong thanh menu. Ngoài việc xem phản hồi, bạn có thể liên hệ trực tiếp với người thử nghiệm và chia sẻ phản hồi với các thành viên trong nhóm phát triển của mình. (56519107)
    • TestFlight Screenshot Feedback hiện có thể được mở trong Xcode Organizer thông qua nút “Open in Xcode 14” trong App Store Connect. (83599827)

    Previews

    Các tính năng mới

    • Bản xem trước Xcode hiện tự động tiếp tục khi tạo dự án mới. (50474683)
    • Xem trước Xcode không còn tạm dừng sau khi chỉnh sửa tệp và chuyển về chế độ xem trước và tiếp tục hoặc khi các dự án sửa đổi thư mục nguồn trong quá trình xây dựng. (71593736)
    • Lỗi từ các bản xem trước tiện ích và phức tạp hiện được hiển thị trong khung vẽ. (76966327)
    • Bản xem trước Xcode hiện sử dụng dữ liệu từ các tệp Cấu hình StoreKit nếu một tệp được đặt trong các tùy chọn chạy của lược đồ. Bản xem trước chỉ hỗ trợ một tập hợp con API StoreKit. (82312384)
    • Chẩn đoán xem trước hiện có thể truy cập được bằng mục trình đơn Trình chỉnh sửa > Canvas > Chẩn đoán của Xcode. (92620156)

    Vấn đề đã giải quyết

    • Bản xem trước Xcode hiện hiển thị chính xác các chuỗi ký tự có chứa chuỗi thoát unicode Swift. (32722474)
    • Bản xem trước Xcode hiện có thể chạy trên các thiết bị vật lý mà không yêu cầu ứng dụng chứa, giúp dễ dàng xem trước trên thiết bị cho các khung và gói Swift. Xcode tự động chuẩn bị một ứng dụng đã ký phù hợp cho danh tính ký mặc định của bạn để lưu trữ bản xem trước. (50206641)
    • Khung Xcode Previews hiện hỗ trợ và mặc định là zoom-to-fit. (51146527)
    • Đã khắc phục một số sự cố khi sử dụng bản xem trước watchOS trong Xcode. Các bản xem trước hiện hoạt động chính xác khi điều hướng giữa các tệp trong ứng dụng iOS chứa hoặc ứng dụng watchOS được nhúng hoặc các khung và gói được bao gồm bởi một hoặc nhiều mục tiêu đó. (53183015)
    • Đã sửa một số trường hợp khi thực hiện chỉnh sửa theo một thứ tự cụ thể sẽ khiến các bản xem trước không thành công với lỗi thiếu biểu tượng (53740398)
    • Bản xem trước Xcode hiện có thể sử dụng chương trình có thể chạy được khi quyết định sử dụng ứng dụng nào để lưu trữ bản xem trước. Ví dụ: trong dự án có khung mà cả phiên bản đầy đủ và phiên bản beta của ứng dụng đều dùng chung, Xcode Previews tự động chọn ứng dụng để khởi chạy để xem trước dựa trên lựa chọn trong lược đồ. (60251198)
    • Đã khắc phục sự cố khi sử dụng nhiều bản xem trước sẽ gây ra lỗi hết thời gian chờ trong Bản xem trước Xcode. (68939151)
    • Đã khắc phục sự cố khiến Bản xem trước Xcode không thành công đối với các tệp có hai dấu ngoặc nhọn đóng trên cùng một dòng. (74035344) (FB8992136)
    • Xem trước Xcode không còn tạm dừng trên các chỉnh sửa lớn. Thay vào đó, Bản xem trước tiếp tục tự động xây dựng bằng cách phát hiện các loại thay đổi được thực hiện và tự động điều chỉnh tần suất cập nhật để cân bằng thời lượng pin và độ trễ. (77799105)
    • Đã khắc phục sự cố trong Bản xem trước Xcode gây ra tình trạng tạm dừng thường xuyên khi sử dụng các dự án Swift Playgrounds được mở trong Xcode. (79206975)
    • Đã khắc phục sự cố trong một số dự án mà lỗi tải gói không bao giờ biến mất trong Bản xem trước Xcode. (79207076)
    • Đã khắc phục sự cố trong đó Bản xem trước Xcode có thể không hiển thị chính xác sau khi đóng và mở canvas. (83789694)
    • Đã sửa các trường hợp Bản xem trước Xcode không phản ánh các thay đổi được thực hiện đối với các tệp trong Gói Swift. (84529522)
    • Đã khắc phục sự cố trong đó canvas Xcode Previews không hiển thị chính xác khi sử dụng @_implementationOnly nhập. (86214393)
    • Xcode hiện hiển thị từng bản xem trước trên trang chuyên dụng của riêng nó bao gồm các điều khiển mới cho phép bạn thay đổi các cài đặt chung như bảng màu, hướng hoặc kích thước loại động mà không cần viết bất kỳ mã nào. (87357785)
    • Bản xem trước Xcode hiện có tính tương tác theo mặc định. Bạn có thể sử dụng trình chuyển đổi chế độ ở cuối canvas để chuyển đổi giữa các chế độ tương tác, lựa chọn và biến thể. (87358985)
    • Bản xem trước Xcode hiện hỗ trợ các biến thể xem trước: bản xem trước được tạo tự động cho phép bạn xem chế độ xem của mình ở nhiều dạng, kích thước loại hoặc hướng cùng một lúc mà không cần viết bất kỳ mã cấu hình nào. (88937848)
    • Bản xem trước hiện hoạt động trong các mô-đun bằng cách sử dụng @_spi. (89653122)
    • Sự cố trong bản xem trước hiện được phản ánh chính xác cho các dự án Swift Playgrounds được mở trong Xcode. (91716026)
    • Việc thay đổi đích chạy đang hoạt động hiện cập nhật chính xác canvas Xcode Previews. (92152617)
    • Cải thiện thời lượng pin khi sử dụng Xcode Previews. (92179785)
    • Các thay đổi đối với giá trị bằng chữ trong trình khởi tạo hiện được phản ánh chính xác trong Bản xem trước Xcode. (92599456)
    • Mục menu “Tự động làm mới canvas” trong canvas Xcode Previews đã được thay đổi để tạm dừng hành vi tạo tự động mới. Điều này cho phép làm mới bản xem trước theo cách thủ công ở nhịp mong muốn. (93261715)
    • Đã khắc phục một số sự cố với Bản xem trước Xcode không phản ánh các thay đổi mã dự kiến ​​khi điều hướng đến các tệp không phải SwiftUI hoặc khi đóng canvas và mở lại. (93785410)
    • Bản xem trước Xcode hiện hỗ trợ các biến chứng watchOS bên trong các ứng dụng watchOS độc lập. (95311509)
    • Bản xem trước Xcode hiện hoạt động với các gói chứa tài nguyên không có trong ứng dụng. (96828503)

    Deprecations

    • Hỗ trợ xem trước các tiện ích được tạo cho ứng dụng macOS và ứng dụng được tạo bằng Mac Catalyst đã bị xóa. (92531529)
      Giải pháp thay thế : Sử dụng Trình mô phỏng WidgetKit macOS.

    Project Navigator

    Vấn đề đã giải quyết

    • Đã sửa lỗi: Khi bạn sử dụng Trình kiểm tra tệp để thay đổi loại tệp, tệp có thể được mở lại trong trình chỉnh sửa khác. (91784648)
    • Đã sửa lỗi: Trình điều hướng sự cố không liệt kê các sự cố được tạo khi xây dựng mã Swift trong giai đoạn mô-đun phát ra. (92430695)

    Vấn đề đã biết

    • Option+Clicking vào nút đóng cho tab trình chỉnh sửa không được chọn sẽ đóng tất cả các tab khác (như mong đợi) và cũng điều hướng đến tab được nhấp bằng điều hướng tùy chọn. Điều hướng này là không chủ ý. (96958005)
    • Việc mở gói kết quả có nhật ký bản dựng lớn có thể mất 30 giây trở lên. (97328772)
    • Option+Clicking vào tab trình chỉnh sửa không thực hiện “điều hướng tùy chọn” như được định cấu hình trong tùy chọn của Xcode. (97690500)

    Refactoring

    Các tính năng mới

    • Đã thêm một hành động tái cấu trúc để thêm một Codable triển khai rõ ràng. (87904700)

    Vấn đề đã biết

    • Chuyển đổi sang Trình tạo Regex không xử lý chính xác các mẫu có chuỗi thoát Unicode. Ví dụ: một ký tự regex có chứa \u{0} sẽ được chuyển đổi thành "" kết quả. (95465206) (FB10343998)

    Server

    Vấn đề đã biết

    • Xcode Server không còn được hỗ trợ. (73888675)

    Signing and Distribution

    Các tính năng mới

    • Ký tự động hiện được hỗ trợ cho trình điều khiển DriverKit đã ký phát triển. Việc phân phối vẫn yêu cầu sự chấp thuận của Apple và cấu hình thủ công các khả năng bổ sung trên trang web Nhà phát triển của Apple . (81215709)
    • Quyền Game Center hiện có sẵn cho các ứng dụng trên iOS, watchOS và tvOS. Nếu ứng dụng của bạn sử dụng Game Center, ứng dụng sẽ tự động nhận được quyền này khi bạn tạo lại hồ sơ cấp phép của mình. Nếu bạn sử dụng ký tự động, Xcode sẽ tự động tạo hồ sơ cung cấp mới cho bạn. Nếu sử dụng ký thủ công, bạn cần đăng nhập vào tài khoản của mình và tạo lại hồ sơ cấp phép của mình
      Nếu bạn dự định tiếp tục sử dụng Game Center trong ứng dụng của mình, hãy thêm quyền com.apple.developer.game-center vào tệp quyền của bạn trong Xcode. Nếu không, hãy xóa capability trong Xcode và tắt Trung tâm trò chơi trên ID ứng dụng của bạn trong tài khoản nhà phát triển của bạn. (90667072)

    Vấn đề đã biết

    • xcodebuild -exportArchivetrả về thông báo lỗi hết hạn phiên nếu bạn sử dụng khóa xác thực để tải ứng dụng của mình lên App Store Connect. (76036452)Giải pháp thay thế : Khóa xác thực chỉ được hỗ trợ để ký và cung cấp. Để tải ứng dụng lên từ xcodebuild, trước tiên hãy đăng nhập vào Xcode bằng ID Apple của bạn.
    • xcodebuildthỉnh thoảng có thể gặp sự cố trong DVTPortalEntitlementsManager. (98678163) (FB11267326)Cách giải quyết: Bạn có thể vô hiệu hóa logic lọc quyền lợi của Xcode để giải quyết vấn đề này, bằng cách chạy trong Terminal hoặc bằng cách thêm vào lời gọi dòng lệnh của bạn.defaults write com.apple.dt.Xcode DVTEnableMultiPlatformEntitlementFiltering -bool NO-DVTEnableMultiPlatformEntitlementFiltering=NOxcodebuild

    Vấn đề đã giải quyết

    • Đã sửa lỗi: Khi chỉnh sửa khả năng của iCloud, Nhóm ứng dụng, Apple Pay hoặc Wallet, Xcode có thể đề nghị chọn một nhóm nếu bạn không chọn một nhóm. Việc chọn một nhóm khiến Xcode gặp sự cố. (93914533)

    Simulator

    Các tính năng mới

    • Simulator hiện hỗ trợ remote notifications trong iOS 16 khi chạy trong macOS 13 trên máy tính Mac có bộ xử lý Apple silicon hoặc T2. Simulator hỗ trợ môi trường Sandbox push notification services của Apple. Máy chủ của bạn có thể gửi thông báo từ xa tới ứng dụng của bạn đang chạy trong simulator đó bằng cách kết nối với sandbox APNS (api.sandbox.push.apple.com). Mỗi Simulator tạo mã thông báo đăng ký duy nhất cho sự kết hợp giữa trình mô phỏng đó và phần cứng máy Mac mà nó đang chạy trên đó. Xem Thông báo người dùng để biết thêm thông tin.Thông báo từ xa hỗ trợ nhiều tính năng hơn (như Tiện ích mở rộng dịch vụ thông báo) so với thông báo được mô phỏng cục bộ bằng cách sử dụng tệp .apns trọng tải hoặc lệnh đẩy simctl. Mã thông báo đăng ký thiết bị có độ dài thay đổi. Mã thông báo trong Trình mô phỏng có thể lớn hơn mã thông báo trên thiết bị vật lý hiện tại. Không mã hóa cứng bất kỳ độ dài hoặc định dạng cụ thể nào cho các mã thông báo này. (60974170)
    • simctl hiện hỗ trợ kiểm soát vị trí mô phỏng, bao gồm các kịch bản đang chạy và nội suy giữa danh sách các điểm tham chiếu. Xem xcrun simctl location để biết thêm thông tin. (59422559) (FB7577924)
    • Simulator hiện hỗ trợ hình ảnh đĩa thời gian chạy ngoài định dạng gói thời gian chạy hiện có. Ảnh đĩa được thêm vào vị trí lưu trữ do hệ thống quản lý được Bảo vệ tính toàn vẹn của hệ thống bảo vệ và được gắn tại các điểm gắn do hệ thống quản lý. Xem xcrun simctl thời gian chạy để biết thêm thông tin. (84169585)
    • simctl addmedia đã được cập nhật để hỗ trợ nhiều định dạng hình ảnh bổ sung (bao gồm nhiều định dạng RAW phổ biến). (87103990) (FB9832655)
    • Giờ đây, bạn có thể khởi động các thiết bị giả lập bằng cách sử dụng thời gian chạy chung là x86_64 trên máy Mac với Apple silicon bằng cách sử dụng --arch đối số dòng lệnh mới cho simctl boot. (88278366) (FB9860747)

    Vấn đề đã biết

    • Nếu bạn ngắt kết nối hoặc tách hình ảnh đĩa thời gian chạy trình mô phỏng theo cách thủ công (chẳng hạn như bằng cách sử dụng diskutil eject hoặc umount), Trình mô phỏng và Xcode có thể không xác định được liệu thời gian chạy đã được cài đặt hay chưa. Nỗ lực tải xuống lại thời gian chạy dẫn đến lỗi với lỗi thời gian chạy trùng lặp. (89589210)
      Giải pháp thay thế : Việc khởi động lại khiến Trình mô phỏng gắn lại ảnh đĩa thời gian chạy. Ngoài ra, bạn có thể sử dụng xcrun simctl runtime để định vị ảnh đĩa thời gian chạy bị ảnh hưởng, xóa nó, sau đó sử dụng Xcode để tải xuống lại.
    • Hình ảnh đĩa thời gian chạy trình mô phỏng trong /tmp đó bạn thêm bằng cách sử dụng xcrun simctl runtime add sẽ bị xóa. (93858264)
      Giải pháp thay thế: Đặt hình ảnh đĩa thời gian chạy giả lập ở một nơi khác /tmp/trước khi chuyển nó vào xcrun simctl runtime add.

    Siri Intents

    Vấn đề đã biết

    • Mã Swift đã tạo mà trình biên dịch định nghĩa Ý định tạo ra sẽ phát ra cảnh báo khi được biên dịch:
    method 'handle(intent:)' with Objective-C selector 'handleIntent:completion:' conflicts with method 'handle(intent:completion:)' with the same Objective-C selector; this is an error in Swift 6

    (91852710)

    Source Control

    Vấn đề đã giải quyết

    • Nhiều lỗi ảnh hưởng đến độ chính xác và độ trễ của trạng thái tệp git trong bộ điều hướng Dự án và Thay đổi đã được giải quyết. Các thay đổi đối với tệp trong bản sao làm việc được phản ánh chính xác trong bộ điều hướng và trang cam kết mà không có độ trễ đáng kể. Ngoài ra, các tệp hiển thị các thanh thay đổi khi được trình bày trong trình chỉnh sửa sẽ nhận được trạng thái đã sửa đổi ngay sau khi thay đổi được thực hiện trong bộ nhớ, điều này sẽ khiến nó có trạng thái đã sửa đổi khi được lưu vào đĩa. (49909533)
    • Xcode hiện hỗ trợ tạo và sử dụng các khóa ED25519 và ECDSA được tạo bên ngoài để thực hiện gitcác thao tác SSH. (85009643)

    Vấn đề đã biết

    • Định cấu hình ứng dụng mới trong Xcode Cloud sẽ không thành công khi nhóm nhà phát triển không có ứng dụng hiện có trong App Store Connect. (94199091)Giải pháp thay thế : Tạo ứng dụng trong App Store Connect trước, sau đó tích hợp sản phẩm trong Xcode.

    Source Editor

    Các tính năng mới

    • Xcode hiện ghim các thành phần cấu trúc mã của bạn lên đầu trình chỉnh sửa khi bạn cuộn qua tài liệu. Để chuyển đổi hành vi này, hãy sử dụng “Hiển thị: Cấu trúc mã trong khi cuộn” trong tùy chọn Chỉnh sửa văn bản của Xcode. (10582250)
    • Các lỗi trong tệp Swift hiện cung cấp bản sửa lỗi để thêm các lần nhập bị thiếu. (21533417) (FB5562997)
    • Gói mã bằng câu lệnh if giờ đây sẽ tự động xác định lại khối. (29215201)
    • Trình khởi tạo hiện được trình bày dưới dạng hoàn thành mã toàn cục trong Swift. (60399329)
    • Giao diện người dùng để chuyển đến các định nghĩa ký hiệu hoặc trình gọi hiện cung cấp cho bạn một mẫu mã từ mỗi vị trí. (69467155)
    • Việc hoàn thành mã hiện thu gọn các chức năng quá tải thành một hàng. (81338102)
    • Đã thêm hỗ trợ chỉnh sửa và đánh dấu cú pháp cho Biểu thức chính quy Swift. Giờ đây, bạn có thể chuyển đổi các ký tự biểu thức chính quy sang trình tạo biểu thức chính quy tương đương bằng cách sử dụng Trình chỉnh sửa > Tái cấu trúc > Chuyển đổi sang Trình tạo Regex. Khi di chuyển điểm chèn bên trong một biểu thức chính quy, cấu trúc con kèm theo của biểu thức chính quy được tô sáng. (82540073)
    • Xcode hiện cung cấp mẫu tệp để chọn tham gia các lựa chọn thay thế cảm ứng cho ứng dụng iOS của bạn. Bạn có thể sử dụng các lựa chọn thay thế cảm ứng để tương tác với ứng dụng của mình trên máy Mac có silicon của Apple – ví dụ: nhấn và giữ phím Tùy chọn để sử dụng bàn di chuột làm màn hình cảm ứng ảo. Để bật, hãy chọn Tệp > Tệp mới > iOS > Tài nguyên > Các lựa chọn thay thế cảm ứng và định cấu hình tệp mới được thêm com.apple.uikit.inputalternatives.plist để chọn các lựa chọn thay thế cảm ứng cho ứng dụng của bạn. (84271952)
    • Hoàn thành mã trong Swift hiện cung cấp đoạn mã khởi tạo theo thành viên. (84348512)
    • Hoàn thành mã trong Swift hiện cung cấp các đoạn trích cho các if case câu lệnh. (84381718)
    • Giờ đây, bạn có thể chọn bất kỳ tổ hợp tham số mặc định nào khi hoàn thành mã bằng cách nhập để khớp với tên tham số. (84906871)
    • Cải thiện độ chính xác của việc hoàn thành mã trong Swift. (85090778)
    • Khi chỉnh sửa mã, mục menu Chỉnh sửa > Nhân bản và phím tắt tương ứng của nó giờ đây sẽ sao chép văn bản đã chọn — hoặc dòng hiện chứa điểm chèn, nếu không có văn bản nào được chọn. (8614499) (FB5618491)
    • Hoàn thành mã trong Swift hiện cung cấp các đoạn mã để thêm Codable triển khai rõ ràng. (87904617)
    • Xcode hiện bản địa hóa các phím tắt cho tất cả các bàn phím phần cứng quốc tế để có khả năng truy cập tốt hơn. Bạn có thể tùy chỉnh thêm điều này trong cài đặt Key Bindings. (88397421)
    • Hoàn thành mã trong Swift hiện cung cấp các đoạn mã cho mapfilter và contains, dựa trên tên biến. (89717471)
    • Cấu trúc mã được ghim vào đầu trình chỉnh sửa với tùy chọn “Hiển thị: Cấu trúc mã trong khi cuộn” mới của Xcode 14 hiện bao gồm thông tin bổ sung cho các khai báo được ngắt dòng. (93591165)

    Vấn đề đã giải quyết

    • Đã sửa lỗi: Hoàn thành mã tại các trang web gọi hàm trong Swift hiện chèn các đối số bị thiếu. (9293666)
    • Một số cải tiến hiệu suất đã được thực hiện để xem và chỉnh sửa các tệp lớn. Các sự cố thường xuất hiện nhất trong tệp nguồn được tạo tự động và tệp thử nghiệm với số lượng lớn thử nghiệm và có thể tồi tệ hơn khi hiển thị dải băng gấp mã. (57789416)
    • Hiệu suất khi sử dụng Minimap trong các tệp HTML với các dòng rất dài đã được cải thiện. (58893150)
    • Đã khắc phục sự cố khiến hiệu suất bị giảm khi nhập tệp có nhiều lỗi hoặc cảnh báo. (59084580)
    • Trình kiểm tra Trợ giúp nhanh hiện hiển thị các mô tả về cài đặt bản dựng trong trình chỉnh sửa cài đặt bản dựng dưới dạng văn bản có định dạng với các liên kết có thể nhấp. (60067884)
    • Hoàn thành mã không còn tự động nhập các mô-đun. (78136559)
    • Xcode hiện ưu tiên các loại có thể được sử dụng làm thuộc tính khi gọi hoàn thành mã sau @. (78239501)
    • Đã khắc phục sự cố khiến khoảng trắng xuất hiện sau một số vùng mã được gấp lại. (78333320) (FB9114110)
    • Đã khắc phục sự cố trong đó Trình chỉnh sửa nguồn bị chèn nhầm *khi tạo một dòng mới sau nhận xét tài liệu kiểu khối. (79415983)
    • Đã thêm giao diện người dùng mới để chuyển đến các định nghĩa biểu tượng hoặc trình gọi tập trung vào thông tin phân biệt về từng vị trí. (81366453)
    • Cải thiện tốc độ và tính chính xác của việc hoàn thành mã trong các biểu thức phức tạp và SwiftUI. (83435550)
    • Xcode ưu tiên các loại Chế độ xem SwiftUI khi bạn nhập bên trong trình tạo chế độ xem SwiftUI. (83846531)
    • Đã khắc phục sự cố khi di chuột qua văn bản trong trình chỉnh sửa Đánh giá mã có thể làm Xcode bị lỗi. (85239396)
    • Đã khắc phục sự cố trong đó các cảnh báo cũ hoặc lỗi gạch chân không chính xác các phần của dòng mà chúng được đính kèm. (86225773)
    • Đoạn mã hoàn thành mã động liên quan đến vòng lặp for không còn đề xuất đặt tên các phần tử được lặp lại giống hệt với vùng chứa của chúng. (87167378)
    • Gấp một tệp Swift bằng cách sử dụng mục menu Phương pháp và chức năng gấp hoặc phím tắt được liên kết của nó giờ đây cũng gấp các thuộc tính và chỉ số được tính toán. (87692952) (FB9849362)
    • Hoàn thành mã trong SwiftUI hiện cung cấp đoạn mã cho List và ForEach. (87904499)
    • Đã khắc phục sự cố có thể dẫn đến giảm hiệu suất theo thời gian khi chỉnh sửa các tệp dài có bật Bản đồ thu nhỏ. (89916018)
    • Đã khắc phục sự cố khi sử dụng tính năng “Hiển thị: Cấu trúc mã trong khi cuộn” mới của Xcode 14 với con trỏ khối có thể dẫn đến tạo tác trực quan. (89973125)
    • Các khai báo Swift actor hiện xuất hiện trong Bản đồ nhỏ cùng với các khai báo lớp và cấu trúc. (90279950)
    • Đã khắc phục sự cố khi hoàn thành mã trong Swift hiển thị các ký hiệu không thể truy cập được. (90404828)
    • Đã khắc phục các sự cố khác nhau trong đó các biểu tượng hệ thống không được đánh dấu cú pháp trong các tệp Swift. (91654823)
    • Hoàn thành mã trong Swift hiện ưu tiên tốt hơn các API hệ thống phổ biến. (91977150)
    • Đã khắc phục sự cố có thể dẫn đến sự cố khi chuyển đến định nghĩa ký hiệu trong theo dõi GPU. (93434935)
    • Tô màu cú pháp của chữ regex đã được cập nhật để hỗ trợ đề xuất tiến hóa nhanh mới nhất SE-0354 Chữ regex . Đặc biệt, giờ đây nó xử lý chính xác những gì trông giống như các ký tự không được đóng dấu, theo sau là một nhận xét, các ký tự được sử dụng trong try và các await biểu thức, đồng thời phân định tốt hơn với các toán tử tiền tố có chứa phần mở rộng /. (93673226, 92355356, 94661164) (95146866)

    Vấn đề đã biết

    • Chuyển đổi sang Trình tạo Regex không xử lý chính xác các mẫu bằng tính năng tra cứu. Ví dụ: /foo(?=bar)/nên tạo ra Lookahead("bar")kết quả, nhưng chỉ tạo ra "bar". (97208700)

    StoreKit

    Vấn đề đã giải quyết

    • Xcode hiện có khả năng đồng bộ hóa các sản phẩm mua trong ứng dụng từ App Store Connect vào các tệp cấu hình StoreKit để thử nghiệm StoreKit nhanh hơn trong thiết lập Xcode. Ngoài ra còn có một trình quản lý giao dịch được cập nhật với tính năng lọc và trình kiểm tra giao dịch. (83863948)
    • Đã khắc phục sự cố khi gọi phương thức ‘clearTransactions()’ trong SKTestSession mà không xóa tất cả các giao dịch SKPaymentQueue khi thử nghiệm ứng dụng bằng API gốc để mua hàng trong ứng dụng. (86696132) (FB9814502)
    • Đã sửa lỗi: Trình quản lý giao dịch không còn hiển thị cảnh báo chưa hoàn thành đối với các giao dịch không thể hoàn thành khi sử dụng thử nghiệm StoreKit trong Xcode. (89419046) (FB9927448)

    Vấn đề đã biết

    • Việc sử dụng các thuộc tính và phương pháp StoreKit sau đây trên các ứng dụng có mục tiêu triển khai tối thiểu bên dưới iOS 16, macOS 13, watchOS 9 và tvOS 16 sẽ khiến ứng dụng gặp sự cố khi khởi chạy khi chạy trên các hệ thống cũ hơn iOS 16, macOS 13, watchOS 9 và tvOS 16:
      • priceFormatStylevà trên các giá trịsubscriptionPeriodFormatStyleProduct
      • environmentStringRepresentationvà trên các giá trịrecentSubscriptionStartDateProduct.SubscriptionInfo.RenewalInfo
      • environmentStringRepresentationvề Transactiongiá trị
      • dateRange(referenceDate:)và trên các giá trị (99962885) (FB11516463)formatted(_:referenceDate:)Product.SubscriptionPeriod
      Giải pháp thay thế : Đối với mỗi mục tiêu sử dụng API StoreKit được liệt kê ở trên, hãy điều hướng đến tab “Giai đoạn xây dựng” trong trình chỉnh sửa dự án với mục tiêu được chọn và thêm StoreKit.framework trong “Liên kết nhị phân với thư viện” nếu nó chưa có. Đặt cột “Trạng thái” thành “Tùy chọn”.

    Refer

    https://developer.apple.com/documentation/xcode-release-notes/xcode-14-release-notes

  • XCode 14 có gì mới? (Phần 1)

    XCode 14 có gì mới? (Phần 1)

    Tổng quan

    Xcode 14 bao gồm Swift 5.7 và SDK cho iOS 16, iPadOS 16, tvOS 16, watchOS 9 và macOS Monterey 12.3. Bản phát hành Xcode 14 hỗ trợ gỡ lỗi trên thiết bị trong iOS 11 trở lên, tvOS 11 trở lên và watchOS 4 trở lên. Xcode 14 yêu cầu máy Mac chạy macOS Monterey 12.5 trở lên.

    Thay đổi chung

    Các tính năng mới

    • Xcode 14 cho phép một target duy nhất hỗ trợ nhiều nền tảng và bao gồm có điều kiện các dependencies, code, resource và build settings cho các nền tảng cụ thể. (74664328)
    • Xcode 14 hỗ trợ phát triển trình điều khiển DriverKit cho iPadOS. (81117498)
    • Xcode 14 bao gồm một mẫu mặc định cho các ứng dụng watchOS kết hợp các mục tiêu Ứng dụng WatchKit và Tiện ích mở rộng ứng dụng WatchKit thành một mục tiêu Ứng dụng Watch duy nhất, đơn giản hóa mã, nội dung và quản lý bản địa hóa. Bạn có thể triển khai các ứng dụng watchOS một mục tiêu cho watchOS 7 trở lên. (83222217)

    Vấn đề đã giải quyết

    • Đã sửa lỗi: Khi storyboard đang mở và nội dung watchOS hoặc tvOS bị xóa khỏi Cài đặt Xcode, storyboard sẽ không đóng. (87471381)

    Vấn đề đã biết

    • Nếu một file Package.swift được thêm vào thư mục chứa trong khi thư mục đó đang mở trong Xcode, Package sẽ không được nhận dạng. (85075018)
      Cách giải quyết : Thoát và chạy lại Xcode.
    • CGFLOAT_EPSILON không còn luôn là Float trên watchOS và nó có thể gây ra sự cố biên dịch. (88698530)
      Cách giải quyết: Chuyển đổi nó trước sang CGFloat bằng cách sử dụng trình khởi tạo CGFloat(CGFLOAT_EPSILON). Hiện được hỗ trợ trên cả nền tảng 32 và 64 bit
    • Xcode 14 có thể không tìm thấy các công cụ bằng cách sử dụng xcodebuild -find(được sử dụng bởi xcrun và các trình bao bọc /usr/bin chẳng hạn như /usr/bin/clang) nếu nội dung khởi chạy lần đầu chưa được cài đặt. (98008921)
      Cách giải quyết : Chạy hoặc khởi chạy Xcode.app trước .xcodebuild -runFirstLaunch
    • Xcode đôi khi có thể coi thời gian chạy sim của nền tảng bị thiếu khi thực hiện Đăng xuất và Đăng nhập của người dùng. (99200503)
      Cách giải quyết : Khởi động lại máy.
    • Khi chạy trong macOS 13 beta, mã AppIntents có thể không xây dựng được bằng Xcode 14. (99661742) (FB11470314)
      Cách giải quyết : Tạo mã AppIntents với Xcode 14 beta 6 hoặc trên máy Mac chạy macOS Monterey 12 với Xcode 14.
    • Không thể sử dụng Xcode 14 với iOS 15.7 để phát triển. (99847608)
      Cách giải quyết: Sử dụng Xcode 13.4.1 với iOS 15.7.

    Trình biên dịch Apple Clang

    Các tính năng mới

    • Các dự án C++ mới mà bạn tạo trong Xcode sử dụng phương ngữ ngôn ngữ mặc định là C++20. (93456065)
    • Một số C++20 và C++2b papers đã được triển khai:
      • Các C++20 papers đã triển khai:
        • P0692R1 – Kiểm tra quyền truy cập vào chuyên môn
        • P0388R4 – Cho phép chuyển đổi thành mảng có giới hạn không xác định
      • Các C++2b papers đã được triển khai:
        • P1938R3 –if consteval
        • P1401R5 – Thu hẹp chuyển đổi theo ngữ cảnh thành bool
        • P1949R7 – Cú pháp mã định danh C++ sử dụng Unicode Standard Annex 31
        • P2360R0 – Mở rộng init để cho phép khai báo bí danh (93898598)

    Vấn đề đã giải quyết

    • Đã sửa lỗi: Xcode không cung cấp tính năng làm nổi bật ngữ nghĩa và hỗ trợ chuyển sang định nghĩa cho các khai báo khái niệm C++20 và các mệnh đề yêu cầu trong các mẫu. (93046529)

    Tính năng không còn khả dụng trên XCode14

    • Bắt đầu với Xcode 14, bitcode không còn cần thiết cho các ứng dụng watchOS và tvOS và App Store không còn chấp nhận gửi bitcode từ Xcode 14. Xcode không còn xây dựng mã bit theo mặc định và tạo thông báo cảnh báo nếu một dự án kích hoạt rõ ràng mã bit: “Việc xây dựng bằng mã bit không được dùng nữa. Vui lòng cập nhật cài đặt dự án và/hoặc mục tiêu của bạn để tắt mã bit.” Khả năng xây dựng bằng mã bit sẽ bị xóa trong bản phát hành Xcode trong tương lai. IPA có chứa mã bit sẽ bị tước mã bit trước khi được gửi tới App Store. Chỉ có thể tải xuống các biểu tượng gỡ lỗi từ App Store Connect / TestFlight cho các lần gửi mã bit hiện có và không còn khả dụng cho các lần gửi được thực hiện bằng Xcode 14. (86118779)

    Danh mục tài sản

    Các tính năng mới

    • Đơn giản hóa biểu tượng ứng dụng bằng một hình ảnh 1024×1024 được tự động thay đổi kích thước cho mục tiêu của nó. Chọn tùy chọn Single Size trong tab Attributes inspector của mục app icon. Bạn vẫn có thể ghi đè các kích thước riêng lẻ bằng tùy chọn All Sizes. (18475136) (FB5503050)
    • Giờ đây, bạn có thể dán trực tiếp hình ảnh đã sao chép từ Finder vào asset catalog outline. (58980721)
    • Giờ đây, bạn có thể nhấp đúp vào một vị trí hình ảnh để hiển thị bảng điều khiển tệp đang mở và chọn nội dung thay thế. (81365822)
    • Bạn có thể chỉ định chế độ kết xuất mặc định cho các ký hiệu tùy chỉnh trong Asset Catalog. Đặt thuộc tính Kết xuất dưới dạng thành tự động, mẫu, nhiều màu hoặc phân cấp. Sau đó, hệ thống sẽ sử dụng chế độ kết xuất mặc định cho biểu tượng, trừ khi bạn ghi đè rõ ràng chế độ đó. Để biết thêm thông tin về biểu tượng tùy chỉnh, hãy xem Tạo hình ảnh biểu tượng tùy chỉnh cho ứng dụng của bạn . (84513859)

    Vấn đề đã giải quyết

    • Chế độ “Tất cả kích thước” cho các biểu tượng ứng dụng đã được làm linh hoạt hơn và bao gồm một số kích thước bổ sung chưa được sử dụng trước đây. Chế độ “Tất cả kích thước” không yêu cầu phải lấp đầy tất cả các vị trí, kích thước chỉ có thể được cung cấp khi cần. Các biểu tượng ứng dụng được tạo bằng Xcode 13 trở về trước có thể được chuyển đổi từ chế độ “Tất cả kích thước (Xcode 13)” thành “Tất cả các kích thước” hoặc “Một kích thước”. (93682080)

    Vấn đề đã biết

    • Các ứng dụng sử dụng biểu tượng ứng dụng có kích thước đơn lẻ có thể không xác thực được App Store nếu mục tiêu triển khai cũ hơn iOS 12 hoặc watchOS 4. (98471456)

    Build System

    Các tính năng mới

    • Xcode cung cấp một trình chỉnh sửa trợ lý mới cho nhật ký bản dựng tập trung vào tính song song để giúp xác định các vấn đề về hiệu suất bản dựng. Hình ảnh trực quan này hiển thị các sự kiện dưới dạng lưới các khối màu trong đó trục dọc biểu thị mức độ song song và trục ngang biểu thị thời gian. (47858322)
    • Xcode 14 hiện có thể biên dịch các mục tiêu song song với các phụ thuộc mục tiêu Swift của chúng. (57116972)
    • Trình điều khiển Swift, thành phần phối hợp các yêu cầu giao diện người dùng Swift, hiện được tích hợp vào hệ thống xây dựng của Xcode, cho phép phụ thuộc chi tiết hơn vào các tác vụ hệ thống xây dựng khác và lập lịch trình rõ ràng. (72440175)
    • Trong Build Phases, giờ đây bạn có thể chỉnh sửa hàng loạt tệp trong chế độ xem bảng nhiều lựa chọn. Khi bạn chỉnh sửa cột bộ lọc nền tảng của bảng đó, hệ thống sẽ áp dụng các thay đổi cho tất cả các tệp trong lựa chọn. (80683128) (FB9340886)
    • Các mục tiêu thư viện động và khung chỉ dành cho Swift có thể chọn tham gia tối ưu hóa hệ thống bản dựng mới bằng cách sử dụng cài đặt bản dựng EAGER_LINKING. Khi bạn bật tính năng này, Xcode sẽ phát ra các thành phần tạo tác bổ sung trong quá trình biên dịch Swift, cho phép Xcode bỏ chặn liên kết của các mục tiêu xuôi dòng trước đó, tăng tính song song trong các bản dựng. (82396635)
    • Hệ thống xây dựng chạy song song các tác vụ từ các giai đoạn xây dựng khác nhau khi các yếu tố phụ thuộc đầu vào và đầu ra không thực thi thứ tự của chúng. Bạn có thể chọn tham gia hành vi mới này cho các giai đoạn xây dựng tập lệnh chạy bằng cách sử dụng cài đặt bản dựng FUSE_BUILD_SCRIPT_PHASES. (82396977)
    • App Store hiện hỗ trợ làm mỏng ứng dụng cho các trình đổ bóng Metal được biên dịch sẵn. (82902821)
    • Các bản dựng Xcode cho thiết bị watchOS hiện bao gồm kiến ​​trúc arm64 theo mặc định. (83319300)
    • Đã thêm toán tử thay thế :relativeto=macro mới cho cài đặt bản dựng mà bạn có thể sử dụng để tính toán đường dẫn tương đối từ đường dẫn này sang đường dẫn khác; 
      Ví dụ: $(INSTALL_PATH:relativeto=/usr/lib)
      Chỗ INSTALL_PATH là “ /usr/bin..” và đánh giá là “../lib ”. Bạn có thể sử dụng quy tắc này trong quy tắc xây dựng để sao chép một loạt tham chiếu tệp trong khi vẫn duy trì phân cấp thư mục của chúng trong thư mục đích hoặc để tính toán mục tiêu dự kiến rpaths ​​bằng cách sử dụng đường dẫn tương đối giữa đường dẫn cài đặt của chính nó và đường dẫn cài đặt đã biết của các thành phần phụ thuộc. (88293015)
    • Xcode hiện cung cấp các cài đặt RECOMMENDED_MACOSX_DEPLOYMENT_TARGET, RECOMMENDED_IPHONEOS_DEPLOYMENT_TARGET, RECOMMENDED_TVOS_DEPLOYMENT_TARGET, RECOMMENDED_WATCHOS_DEPLOYMENT_TARGET, RECOMMENDED_DRIVERKIT_DEPLOYMENT_TARGET và xây dựng cho biết các phiên bản triển khai tối thiểu được đề xuất cho từng nền tảng Xcode được hỗ trợ. (90464341)
    • Giờ đây, bạn có thể bật Sandbox cho các giai đoạn xây dựng tập lệnh shell bằng cách sử dụng ENABLE_USER_SCRIPT_SANDBOXING trong build settings. Sandbox chặn quyền truy cập vào các tệp bên trong gốc nguồn của dự án cũng như thư mục Dữ liệu được tạo trừ khi bạn liệt kê các tệp đó dưới dạng đầu vào hoặc đầu ra. Khi được bật, quá trình xây dựng không thành công do vi phạm Sandbox nếu một giai đoạn tập lệnh cố gắng đọc hoặc ghi vào một phần phụ thuộc không được khai báo, ngăn chặn các quá trình xây dựng không chính xác. (90506067)

    Vấn đề đã giải quyết

    • Đã khắc phục sự cố trong đó nhiều dự án Xcode tham chiếu cùng một xcconfig (do đó bao gồm một xcconfig khác) đã tính toán không chính xác cùng một cài đặt bản dựng trên cả hai dự án. (84319288) (FB9707003)
    • Đã sửa lỗi: đích chạy của thiết bị watchOS không còn xuất hiện hai lần trong menu đích chạy. (85635959)
    • Cải thiện tốc độ tải của không gian làm việc lớn có nhiều mục tiêu chia sẻ .xcconfigtệp. (85985712)
    • Đã sửa lỗi: Khi bạn lưu trữ ứng dụng watchOS với kiến ​​trúc arm64e được bật, ứng dụng đó không thể xây dựng. (93550623)

    Vấn đề đã biết

    • Báo cáo sự cố từ các ứng dụng Mac Catalyst không được ký hiệu đầy đủ trong Xcode Organizer. (94820955)

    Tính năng bị loại bỏ

    • Vì mã bit hiện không được dùng nữa nên các bản dựng cho iOS, tvOS và watchOS không còn bao gồm mã bit theo mặc định. (87590506)
    • Đã thêm cài đặt bản dựng mới, SWIFT_TOOLS_DIR, thay thế SWIFT_EXEC làm cơ chế được đề xuất để sử dụng tệp thực thi giao diện người dùng nhanh tùy chỉnh. SWIFT_TOOLS_DIR phải được đặt thành đường dẫn của thư mục chứa giao diện người dùng nhanh và các công cụ liên quan. Lưu ý rằng vì trình điều khiển Swift hiện được tích hợp trong hệ thống xây dựng Xcode, không thể sử dụng SWIFT_TOOLS_DIR và SWIFT_EXEC để chỉ định nhị phân trình điều khiển tùy chỉnh trừ khi SWIFT_USE_INTEGRATED_DRIVER được đặt thành NO. (88967344)
    • Hệ thống xây dựng cũ đã bị xóa. (90801041)
    • Việc xây dựng các dự án iOS với các mục tiêu triển khai cho kiến ​​trúc armv7, armv7s và i386 không còn được hỗ trợ nữa. (92831716)
    • Bản phát hành xây dựng để triển khai cho hệ điều hành cũ hơn macOS 10.13, iOS 11, tvOS 11 và watchOS 4 không còn được hỗ trợ. (92834476)
    • Khi xóa một điều kiện khỏi trình chỉnh sửa biến thể, giá trị sẽ không được duy trì. (98149034)
      Giải pháp thay thế: Sử dụng Trình chỉnh sửa cài đặt bản dựng để xóa điều kiện.

    Debugging

    Các tính năng mới

    • Trình gỡ lỗi biểu đồ bộ nhớ hiện hiển thị tất cả các tham chiếu đến và đi trong biểu đồ bộ nhớ. Bạn có thể điều chỉnh tập hợp các nút hiển thị trong cửa sổ bật lên mới. (69454709)
    • LLDB hiện hiển thị các cập nhật tiến độ trong Xcode và dòng lệnh cho các hoạt động chạy dài. (73511008)
    • Giờ đây, bạn có thể gọi tập lệnh nhật ký sự cố của LLDB bằng xcrun crashlog <path/to/crash>. (79991815)
    • Trình kiểm tra hiệu suất luồng hiển thị các vấn đề về hiệu suất thời gian chạy trong Trình điều hướng sự cố và trình chỉnh sửa nguồn trong khi gỡ lỗi ứng dụng. Chọn hộp kiểm Trình kiểm tra hiệu suất luồng trong Lược đồ chạy của mục tiêu ứng dụng của bạn để bật tính năng này. (80048810)
    • Giờ đây, bạn có thể mở một .xcresult cho một hành động scheme khởi chạy. (88932007)
    • Xcode hiện hiển thị nhật ký khởi chạy mới trong Trình điều hướng báo cáo. Nhật ký cho biết các hành động mà Xcode thực hiện để cài đặt, khởi chạy và gỡ lỗi. (90859910)

    Vấn đề đã giải quyết

    • Cải thiện hiệu suất khi gỡ lỗi chương trình Swift trên thiết bị iOS. Thay vì sao chép siêu dữ liệu phản ánh từ thiết bị iOS, giờ đây Xcode có thể đọc siêu dữ liệu từ đĩa. (73179144)
    • Xcode hiện nhóm các mục nhật ký cho cùng một hành động sơ đồ khởi chạy trong trình điều hướng Báo cáo. (87216988)
    • Đã sửa lỗi: Thay vào đó, bước vào chức năng không đồng bộ Swift trong LLDB sẽ hoàn thành chức năng hiện tại. (88142757)

    Vấn đề đã biết

    • Không thể đính kèm trình gỡ lỗi vào Tiện ích mở rộng tiện ích màn hình khóa. (93941779)
      Giải pháp thay thế: Chỉnh sửa Extension scheme – trong tab Run action, tab Argument, hãy đặt biến môi trường ‘_XCWidgetKind’ thành một trong các tên class/struct của bạn khi triển khai widget.

    Tài liệu

    Các tính năng mới

    • Swift-DocC trong Xcode hiện hỗ trợ viết và xây dựng tài liệu cho API Objective-C và C. (58760015)
    • Các trang web tài liệu Swift-DocC mà Xcode 14 tạo ra bao gồm một thanh bên điều hướng mới để khám phá và lọc tài liệu. (89031049)
    • Theo mặc định, tài liệu Swift-DocC mà Xcode 14 tạo ra hiện tương thích với hầu hết các dịch vụ lưu trữ được quản lý, bao gồm Trang GitHub. (91173450)

    Vấn đề đã giải quyết

    • Đã sửa lỗi: Swift-DocC đưa ra chẩn đoán không chính xác về các lệnh không được hỗ trợ trong tài liệu nguồn biểu tượng khi xây dựng tài liệu cho các dự án Objective-C bao gồm các lệnh Doxygen. (90953916)
    • Đã sửa lỗi: Tài liệu Mục tiêu-C cho các API đa ngôn ngữ hiển thị các mối quan hệ ký hiệu Swift thay vì các mối quan hệ Mục tiêu-C. (91627374)
    • Đã sửa lỗi: Swift-DocC đưa ra chẩn đoán không chính xác về các liên kết ký hiệu không thể giải quyết đối với tài liệu kế thừa từ các ký hiệu trong các mô-đun khác. (92185538)
    • Đã sửa lỗi : Không thể xây dựng tài liệu cho các loại phù hợp với ChartContent protocol khi conform Swift Charts. (93610106)

    Instruments

    Các tính năng mới

    • Các hàm tổng hợp hoạt động trên các khoảng thời gian hiện tính toán chính xác hơn kết quả của chúng khi bộ lọc thời gian đang hoạt động, chỉ dựa trên phần của khoảng thời gian tổng thể giao với bộ lọc thời gian hiện tại. (32330648)
    • Bộ lọc chi tiết hiện cho phép áp dụng các bộ lọc cho một cột cụ thể khi xem Chế độ xem danh sách. Có thể thêm các bộ lọc đã nhập này bằng cách sử dụng menu ngữ cảnh trên các giá trị được hiển thị hoặc bằng cách nhập mã thông báo rồi chọn loại mã thông báo. (56080914)
    • Các rãnh Chủ đề chính hiện được sắp xếp đầu tiên khi hiển thị các rãnh trong dòng thời gian của Nhạc cụ. (59876638)
    • Công cụ có công cụ Theo dõi treo mới hiển thị khi chuỗi chính của ứng dụng không thể xử lý các sự kiện đến trong một khoảng thời gian dài, có khả năng khiến giao diện người dùng bị treo. Ngoài ra, các công cụ Time Profiler và CPU Profiler cũng hiển thị khả năng bị treo. (65694830)
    • Biểu đồ biểu đồ trong Công cụ hiện hiển thị thời lượng của phạm vi thời gian được sử dụng để tổng hợp dữ liệu trong chú giải công cụ di chuột qua. (65684291)
    • Chế độ xem danh sách của công cụ hiện hiển thị nhãn trạng thái ở dưới cùng, báo cáo về số lượng hàng hiện có và được chọn trong chế độ xem hiện tại. (82652407)
    • Mẫu Core ML mới có sẵn trong Công cụ. Mẫu này bao gồm các công cụ Core ML và Neural Engine mới cùng với các công cụ GPU và Time Profiler. Sử dụng mẫu này để giúp lập hồ sơ sử dụng Core ML và hiểu cách các mô hình của bạn đang chạy trên thiết bị. Việc kết hợp thông tin từ Core ML, Công cụ thần kinh và Công cụ GPU có thể giúp theo dõi những hoạt động nào được thực thi trên phần cứng được tăng tốc. Dữ liệu thời gian tổng hợp có sẵn cho từng sự kiện, mô hình và mô hình con. (83123510)
    • Bạn có thể mở một menu ngữ cảnh mới bằng các khoảng thời gian bấm Điều khiển trong dòng thời gian. Menu cung cấp các hành động để đặt bộ lọc thời gian theo khoảng thời gian đã chọn hoặc hiển thị thông tin tương ứng trong khu vực chi tiết. (86728567)
    • Chế độ xem Thông tin Chạy của Dụng cụ hiện bao gồm kiến ​​trúc của tệp nhị phân đích khi ghi một dấu vết quy trình. (89733709)
    • Instruments có một công cụ Runloop mới hiển thị việc sử dụng runloop và các lần lặp lại riêng lẻ, đồng thời phân biệt trực quan các chế độ ngủ của runloop và các khoảng thời gian bận cho tất cả các runloop trong một quy trình. (89746568)
    • Các giá trị trong chế độ xem chi tiết tổng hợp hoặc danh sách giờ đây sẽ được tô sáng khi bạn trỏ con trỏ qua chúng và bạn có thể giữ Control khi bấm vào một giá trị để xem menu ngữ cảnh. Menu này hiển thị các hành động để sao chép giá trị, đặt bộ lọc chi tiết và thêm hoặc ghim rãnh nếu giá trị có đại diện cho rãnh. (90817779)
    • Các công cụ hiện bao gồm một mẫu Đồng thời Swift mới để theo dõi cách sử dụng và hành vi của các nguyên mẫu đồng thời của Swift. Mẫu chứa:
      • Một công cụ Swift Tasks mới hiển thị các trạng thái nhiệm vụ theo thời gian, tóm tắt các trạng thái nhiệm vụ, cung cấp tường thuật chi tiết về nhiệm vụ, minh họa các mối quan hệ đồng thời có cấu trúc và xây dựng cây cuộc gọi của các ngăn xếp công việc tạo nhiệm vụ.
      • Một công cụ Swift Actors mới để theo dõi hành vi tác vụ giữa các tác nhân hiển thị hàng đợi tác vụ cho từng tác nhân và giúp chẩn đoán các sự cố với mã và tranh chấp được cô lập bởi tác nhân.Các công cụ này yêu cầu thiết bị trong thời gian chạy đồng thời Swift lần đầu tiên có sẵn trong macOS 13, iOS 16, tvOS 16 và watchOS 9 trở lên. (69852855)
    • heapleaks, and Instruments’ leaks tool hiện báo cáo ít tham chiếu sai hơn bắt nguồn từ phân bổ liên quan đến đồng thời Swift và xác định đây là phân bổ Swift AsyncTask. (72907112)
    • leaks --atExit -- <command> giờ đây sẽ tự động đặt trong môi trường MallocStackLogging=lite cho lệnh đã chỉ định để Công cụ có thể hiển thị dấu vết ngược ngăn xếp cho phân bổ bị rò rỉ. Để sử dụng một cài đặt khác của biến môi trường đó (chẳng hạn như CÓ hoặc KHÔNG), hãy đặt biến môi trường trước khi chạy leaks. (73779272)
    • xctrace bây giờ sẽ gửi thông báo Darwin khi bắt đầu theo dõi, thông báo này nên được ưu tiên thay vì đọc đầu ra tiêu chuẩn của lệnh. Để sử dụng nó, hãy chỉ định notify-tracing-started tùy chọn có tên thông báo mong muốn và sử dụng notify_register_dispatch để nhận thông báo không đồng bộ trong tập lệnh của bạn. (75745933)
    • Cải thiện hiệu suất phân tích của mẫu Dấu vết Hệ thống. Các công cụ hiện hiển thị các bản ghi cho người dùng nhanh hơn; tuy nhiên, khi theo dõi một quy trình đơn lẻ, bạn không thể thấy trạng thái luồng cho các quy trình khác. Thay vào đó, để xem trạng thái của các quy trình khác, hãy theo dõi “Tất cả các quy trình”. (79904471)
    • Đối với các quy trình đang chạy MallocStackLogging được bật, heap, leaks và trình gỡ rối bộ nhớ Xcode hiển thị các tên loại dành cho phân bổ không đối tượng ở dạng malloc in <function name> trong đó “function name” là khung ngăn xếp không phân bổ đầu tiên. Tính năng này giúp xác định rõ hơn mục đích của bộ nhớ phi đối tượng. (85598783)
    • heap -addresses=all hiện hiển thị các mô tả của các đối tượng CFData có chứa các .plisttệp nhị phân theo cách dễ đọc hơn đối với con người. (88466642)

    Vấn đề đã giải quyết

    • heap , leaks, và stringdups bây giờ bao gồm nội dung của một số loại chuỗi Swift được cấp phát theo đống trong phần mô tả về cấp phát của các chuỗi đó. (14902403)
    • Công cụ Hiệu suất GCD hiện diễn giải chính xác các sự kiện có mã “7” và “9”. (69721960)
    • Thiết kế trình xem nguồn trong Công cụ bao gồm các cải tiến để hiển thị dữ liệu hiệu suất tốt hơn:
      • Có một chế độ Xen kẽ mới để xem mã nguồn và tháo gỡ liên kết với nhau, giúp việc liên kết các hướng dẫn được tạo cho từng dòng mã nguồn trở nên dễ dàng hơn.
      • Trình xem nguồn hiện hiển thị các giá trị được lấy mẫu được gán cho một dòng mã hoặc phân tách trong một cột riêng biệt thay vì dưới dạng chú thích dòng.
      • Trình xem nguồn hiện hiển thị Bộ đếm CPU, sự kiện PMC và công thức động được định cấu hình trong các tùy chọn ghi bên cạnh nguồn và tháo gỡ. (80232392)
    • Đã khắc phục sự cố trong đó Chế độ xem nguồn trong Công cụ đôi khi không hiển thị chú thích hiệu suất khi được mở từ chế độ xem Sơ đồ cuộc gọi. (80501587)
    • Khi sử dụng Instruments Source Viewer ở chế độ xem tách, việc chọn tháo gỡ bây giờ cũng chọn nguồn tương ứng ở bên trái. (82642041)
    • Đã khắc phục sự cố xctrace export đôi khi xuất giá trị chuỗi UTF-16 thay vì UTF-8, dẫn đến lỗi kết xuất trong Terminal. (87594987)
    • Đã khắc phục sự cố trong đó một số sự kiện đã ghi sau khi khởi động lại sẽ bị mất khi nhập kho lưu trữ nhật ký trong quá trình khởi động lại thiết bị. (89003393)
    • Đã sửa lỗi xctrace list devices đôi khi không bao gồm tất cả các thiết bị được kết nối vật lý trong đầu ra của nó. (89142775) (FB9913864)
    • heapleaks, and Instruments hiện nhận dạng chính xác bảng phụ của các đối tượng Swift đã hủy phân bổ dưới dạng bảng phụ. (89459805)
    • Đã khắc phục sự cố trong đó Công cụ không cho phép ghi các thiết bị tvOS được kết nối qua WiFi. (89592418) (FB9935783)
    • Đã khắc phục sự cố trong đó bản dựng gói Công cụ tùy chỉnh có thể không thành công khi cùng một biến được sử dụng nhiều lần trong trường biểu thức. (89653878)
    • heapleaks, and Instruments hiện hiển thị tên dễ đọc hơn cho các loại thư viện tiêu chuẩn Foundation và Swift bổ sung. (90441836)
    • xctrace không còn trả về trạng thái lỗi khi truy tìm chỉ gây ra cảnh báo. Các vấn đề hiện được quy cho mức độ nghiêm trọng thích hợp. (90560207) (FB9963155)
    • Đã khắc phục sự cố trong xctrace export khi dữ liệu theo dõi VM và phân bổ đôi khi được xuất với các hàng dữ liệu bị thiếu. (90560947) (FB9963169)
    • Đã khắc phục sự cố trong đó Công cụ sẽ quên đối số dòng lệnh sau khi mở và đóng trình chỉnh sửa đích khi đã có đối số. (90666084)
    • Đã khắc phục sự cố khi di chuyển đầu kiểm tra trong thước theo dõi cũng sẽ cuộn trục Y của dòng thời gian. (90810506)
    • Đã sửa lỗi: Khoảng thời gian chạy vòng lặp cho thời gian chạy, lặp, chờ và bận có nhãn và độ dài không chính xác khi được hiển thị trong rãnh quy trình. (91130163)
    • Đã khắc phục sự cố Công cụ không liên tục khi ghi dấu vết với các mẫu Phân bổ hoặc Rò rỉ. (93916110)
    • Đã sửa lỗi: Tạo Báo cáo hiệu suất Core ML có thể không thành công sau khi bật Chế độ nhà phát triển trên thiết bị. (93923607)

    Vấn đề đã biết

    • Ký hiệu mã người dùng dSYM có thể không thành công đối với tệp tailspin được nhập vào Công cụ. Các ký hiệu cho thư viện hệ thống cũng không xuất hiện khi các tệp này được tải. (93261223)
      Giải pháp thay thế : Ký hiệu nhật ký treo trong Terminal bằng spindump -i <UIKit-runloop*.ips>.
    • Vô hiệu hóa Trình kiểm tra hiệu suất luồng thông qua tab Chẩn đoán trong Trình chỉnh sửa lược đồ nếu Ứng dụng iOS hoặc macOS của bạn sử dụng Fishhook để kết nối các cuộc gọi tới libSystem nhằm mục đích gỡ lỗi/theo dõi. (94724380) (FB10131267)

    Refer: https://developer.apple.com/documentation/xcode-release-notes/xcode-14-release-notes

  • Splash Screen

    Splash Screen

    Hôm nay mình quay lại với chủ đề khá quen thuộc liên quan đến Splash Screen. Như chúng ta đã biết splash screen (màn hình khởi động ứng dụng) là trải nghiệm và là thứ đầu tiên người dùng nhìn thấy đối với mỗi ứng dụng.

    Những chức năng cụ thể của nó là show lên hình ảnh của App, một kiểu nhận dạng của App đó. Nhưng có thể sâu xa hơn của việc tạo ra màn hình này là thực hiện một số việc trước khi vào giao diện chính của ứng dụng, có thể liệt kể ở đâu là lấy và đồng bộ dữ liệu ở đâu đó (có thể là server, database, … ), chuẩn bị data hiển thị ở màn hình chính, …

    Bài viết dưới đây mình xin giới thiệu qua một số cách tiếp cận và triển khai khi thực hiện việc implement một Splash screen. Và một số ưu, nhược điểm của nó. Mình có thể liệt kê 3 cách chính mà thông thường mình thấy trong thời điểm hiện tại.

    1. Tạo một Activity và 1 layout XML tương ứng của nó

    Cách này theo mình thấy là cách phổ biến nhất mà các app trước khi có Android 12 đang sử dụng. Thực hiện cách này, lập trình viên chỉ cần tạo một giao diện XML theo đúng Splash UI theo requirement và setContentView cho nó. Khi muốn kết thúc, chúng ta chỉ cần start Activity chính tương bằng Intent và nhớ finish() Splash Screen hiện tại

    2. Tạo một Activity và set them background cho chúng

    Nếu thật sự các bạn thực hiện giống ở cách một, nhiều khả năng các bạn sẽ gặp phải trường hợp khi bập bật màn hình nên bạn thường thấy màn hình trắng một lát trước khi ảnh được hiện lên là do layout của bạn chỉ được hiển thị sau khi ứng dụng đã khởi tạo xong. Vì vậy để tránh việc này thay vì bạn tạo một layout cho màn hình này thì bạn sẽ tạo theme cho riêng Splash Activity này. Background mà bạn thiết lập giống như layout của Splash Screen mà bạn muốn triển khai.

    Các bước thực hiện

    Để có thể làm splash screen bằng theme, đầu tiên bạn cần tạo một XML drawable splash_background.xml bên trong thư mục res/drawable

    image

    Bước tiếp theo ta set splash_background.xml cho hình nền của theme. Ta thêm SplashTheme cho splash activity như sau:

    image

    Sau đó ta cấu hình SplashTheme cho Activity Splash screen trong AndroidManifest.xml

    image

    Bước cuối cùng ta tạo một activity SplashActivity.java mà không cần setContentView cho nó

    image

    Bằng cách làm trên View sẽ hiển thị từ theme mà không cần khỏi tạo layout nên ứng dụng sẽ chạy nhanh hơn, và thỉnh thoảng sẽ không bị xuất hiện hiện tượng màn hình trắng. Đây là một cách hợp lý mà các bạn có thể cân nhắc nếu vẫn muốn khởi tạo một Activity riêng biệt cho Splash Screen.

    3. Sử dụng API Splash Screen Android 12

    Như tên gọi của nó, từ Android 12 ra mắt một cách triển khai để tạo ra Splash Screen mới. Trải nghiệm mới mang đến các yếu tố thiết kế tiêu chuẩn cho mọi lần khởi chạy ứng dụng, nhưng nó cũng có thể tùy chỉnh để ứng dụng của bạn có thể duy trì thương hiệu riêng của ứng dụng.

    Vậy Splash Screen hoạt động thế nào Khi một app khởi chạy thì có thể rơi vào 1 trong các giai đoạn sau:

    • Cold Start (app chạy lần đầu). Ví dụ như ứng dụng của bạn được khởi chạy lần đầu tiên kể từ khi thiết bị khởi động hoặc kể từ khi hệ thống kill app
    • Warm Start (Khởi chạy các tiến trình nhưng chưa tạo Activity). Ví dụ: Khi đưa app xuống background một thời gian dài, hệ thống tự loại bỏ ứng dụng ra khỏi bộ nhớ Stack hiện tại, khi đó các tác vụ và process cần khởi động lại.
    • Hot start Khi mới bắt đầu, tất cả những gì hệ thống làm là đưa Activity của chúng ta lên hàng đầu của Stack. Nếu tất cả các Activity của ứng dụng vẫn nằm trong bộ nhớ, thì ứng dụng có thể tránh phải lặp lại quá trình khởi tạo đối tượng, lạm phát bố cục và hiển thị. Màn Splash này sẽ không chạy khi ở giai đoạn Hot start.

    Bài tiếp theo mình sẽ giới thiệu tiếp liên quan đến các yếu tố, cơ chế và triển khai cụ thể kèm code của API Splash Screen nhé.

  • Hướng dẫn cách làm slider hai chiều trong Swift

    Hướng dẫn cách làm slider hai chiều trong Swift

    Là một iOS developer, chắc hẳn mọi người đã từng làm những tính năng mà các UIKit của Apple không thể đáp ứng được yêu cầu. Lúc này chúng ta cần phải thực hiện custom hoặc tạo một common UI với cơ chế mới thoả mãn yêu cầu của khách hàng. Và ở dự án mình cũng có những yêu cầu như vậy, nên bài viết này mình sẽ hướng dẫn các bạn làm slider hai chiều trong Swift.

    Mục đích của chúng ta là đạt được kết quả như sau:

    Slider hai chiều

    I. Yêu cầu

    Cần tạo một common slider thoả mãn các điều kiện như sau:

    1. Slider được chia thành các đoạn bằng nhau có thể chỉnh sửa, Slider có 2 điểm kéo, có thể kéo được từ giá trị min tới giá trị max, kéo chồng lên nhau, khi kéo tới các điểm sẽ thực hiện rung.
    2. Cho phép thay đổi giao diện

    II. Giải pháp

    1. Tạo một mảng string để lưu các điểm cho phép và dùng nó để tính toán và hiển thị dữ liệu cho Slider hai chiều
    2. Dựa vào số phần tử trong mảng để tính khoảng cách giữa các điểm và vị trí của nó.
    3. Đối với tính năng kéo mình sẽ dùng UIPanGestureRecognizer để xử lí. Khi kết thúc hành động kéo thì xác định vị trí của điểm được kéo so với các điểm trong mảng, nếu gần điểm nào thì tự di chuyển tới vị trí của điểm đó.

    III. Hướng dẫn chi tiết

    Để cho tất cả mọi người đều có thể thực hiện một các dễ dàng thì mình sẽ hướng dẫn chi tiết theo các bước như sau:

    1. Tạo mới file swift

    Chúng ta cần tạo mới file swift để làm common Slider hai chiều theo các bước sau:

    Trong project, bấm chuột phải vào thư mục muốn tạo file -> new file -> chọn swift -> đặt tên cho file, ở đây mình đặt là: TWSlider.swift

    Kết quả tạo file

    2. Thêm nội dung và logic cho file TWSlider.swift

    Trước tiên chúng ta cần thêm image vào asset của project để sử dụng cho slider, các bạn cũng có thể thêm các image tương ứng với yêu cầu của dự án đang làm.

    Và chúng ta cũng cần thêm Then vào để setup UI cho tiện như sau

    public protocol Then {}
    
    public extension Then {
        @discardableResult
        func then(_ block: (Self) -> Void) -> Self {
            block(self)
            return self
        }
    }
    
    extension NSObject: Then {}
    extension CGPoint: Then {}
    extension CGRect: Then {}
    extension CGSize: Then {}
    extension CGPath: Then {}

    1. Tạo protocol để bắn dữ liệu cần thiết ra màn hình cần xử lí logic

    protocol TWSliderDelegate: AnyObject {
        func sliderScrolled(_ slider: TWSlider?,
                            toMinIndex minIndex: Int,
                            andMaxIndex maxIndex: Int,
                            endDragDrop: Bool)
    }

    2. Tạo các biến cho phép thay đổi các giá trị của slider

    class TWSlider: UIView {
        
        // PUBLIC variable
        // The number of points in slider
        var numberOfSegments: Int = 0
        
        // This value should be set if slider button should overlap or not. Default to NO. ie., 1 segment space will be present between the sliders.
        var shouldSliderButtonOverlap: Bool = false
        
        // The color that is used for rangeSlider unselected range view. (i.e., the view that is not within the slider points). Default is #DFDEE4.
        var rangeSliderBackgroundColor: UIColor = .gray
        
        // The color that is used for rangeSlider selected range view. (i.e., the view that is between the slider points). Default is #FF9800.
        var rangeSliderForegroundColor: UIColor = .orange
        
        // The label color to be used for min range display. Default is white.
        var rangeDisplayLabelColor: UIColor = .white
        
        // The label color to be used for min range display. Default is black.
        var minMaxDisplayLabelColor: UIColor = .black
        
        // The color for segment button when it is within the selected range. Default is clear.
        var segmentSelectedColor: UIColor = .clear
        
        // The color for segment button when it is outside the selected range. Default is #F8F8FA.
        var segmentUnSelectedColor: UIColor = .white
        
        // The image used for displaying slider buttons. By default "ic_sliderButton". rangeSliderButtonColor will be used if not set
        var rangeSliderButtonImage: UIImage?
        
        // The image for segment button when it is within the selected range. If not set, segmentSelectedColor will be used.
        var segmentSelectedImage: UIImage?
        
        // The image for segment button when it is outside the selected range. If not set, segmentUnSelectedColor will be used.
        var segmentUnSelectedImage: UIImage?
        
        // The size of the slider button. If not set, defaults to (16, 16).
        var sliderSize: CGSize = CGSize(width: 16, height: 16)
        
        // The size of the segments. If not set, defaults to (4, 4).
        var segmentSize: CGSize = CGSize(width: 4, height: 4)
        
        // The min and max range label text to be set by caller
        var minRangeText: String?
        var maxRangeText: String?
        
        // The delegate property
        weak var delegate: TWSliderDelegate?
        
        // PRIVATE VAR
        // Slider button size
        private var SLIDER_BUTTON_WIDTH: CGFloat = 44.0
        
        // Slider frame
        private var DEFAULT_SLIDER_FRAME: CGRect!
        
        // Segment width
        private var segmentWidth: CGFloat!
        
        // The backgroundView represent unselected/outside range view
        private var sliderBackgroundView: UIView!
        
        // The foregroundView represent selected/inside range view
        private var sliderForegroundView: UIView!
        
        // The label placed below the min and max sliders
        private var minRangeView: UIView!
        private var maxRangeView: UIView!
        private var minRangeLabel: UILabel!
        private var maxRangeLabel: UILabel!
        
        // min and max value
        private var minLabel: UILabel!
        private var maxLabel: UILabel!
        
        // Represent the range slider on either side of the slider
        private var startSliderButton: UIButton!
        private var endSliderButton: UIButton!
        
        // The segment index or percent for initial slider position loading for segmented and unsegmented respectively
        private var minRangeInitialIndex: Int?
        private var maxRangeInitialIndex: Int?
        
        // Padding range view with slider button
        private let iconRangeMidView = "ic_rangeMidView"
        private let iconRangeView = "ic_rangeView"
        private let iconSliderButton = "ic_sliderButton"
        private let iconSegmentUnSelected = "ic_segmentUnSelected"
        private let paddingRangeView: CGFloat = 4
        
        private var sliderMidView: UIView!
        private var sliderMidLabel: UILabel!
        private var checkInit: Bool = false
        // MARK: init
        required init?(coder: NSCoder) {
            super.init(coder: coder)
            setUpUI()
        }
        
        override init(frame: CGRect) {
            super.init(frame: frame)
            setUpUI()
        }
        
        func setUpUI() {
            setDefaultValues()
            initSliderViews()
        }
        
        override func layoutSubviews() {
            super.layoutSubviews()
            updateFrame()
        }
    }

    3. Tạo các hàm private để setup UI cho slider

    extension TWSlider {
        
        // Default Initializaer
        private func setDefaultValues() {
            checkInit = false
            numberOfSegments = 2
            shouldSliderButtonOverlap = false
            
            minRangeLabel = UILabel(frame: CGRect(x: 0, y: 0, width: 24, height: 24))
            minRangeLabel.then {
                $0.textColor = rangeDisplayLabelColor
                $0.textAlignment = .center
                $0.font = UIFont.systemFont(ofSize: 12, weight: .medium)
            }
            
            minRangeView = UIView(frame: CGRect(x: 0, y: 0, width: 24, height: 29))
            minRangeView.then {
                $0.center = CGPoint(x: SLIDER_BUTTON_WIDTH / 2, y: bounds.midY - (SLIDER_BUTTON_WIDTH / 2) - (paddingRangeView / 2))
                if let image = UIImage(named: iconRangeView) {
                    $0.backgroundColor = UIColor(patternImage: image)
                }
                addSubview($0)
                $0.addSubview(minRangeLabel)
            }
            
            maxRangeLabel = UILabel(frame: CGRect(x: 0, y: 0, width: 24, height: 24))
            maxRangeLabel.then {
                $0.textColor = rangeDisplayLabelColor
                $0.textAlignment = .center
                $0.font = UIFont.systemFont(ofSize: 12, weight: .medium)
            }
            
            maxRangeView = UIView(frame: CGRect(x: 0, y: 0, width: 24, height: 29))
            maxRangeView.then {
                $0.center = CGPoint(x: SLIDER_BUTTON_WIDTH / 2, y: bounds.midY + (SLIDER_BUTTON_WIDTH / 2) + (paddingRangeView / 2))
                if let image = UIImage(named: iconRangeView) {
                    $0.backgroundColor = UIColor(patternImage: image)
                }
                addSubview($0)
                $0.addSubview(maxRangeLabel)
            }
            
            minLabel = UILabel(frame: CGRect(x: 0, y: 0, width: 24, height: 24))
            minLabel.then {
                $0.center = CGPoint(x: SLIDER_BUTTON_WIDTH / 2, y: bounds.midY + (SLIDER_BUTTON_WIDTH / 2))
                $0.textColor = minMaxDisplayLabelColor
                $0.textAlignment = .center
                $0.font = UIFont.systemFont(ofSize: 12, weight: .regular)
                addSubview($0)
            }
            
            maxLabel = UILabel(frame: CGRect(x: 0, y: 0, width: 24, height: 24))
            maxLabel.then {
                $0.center = CGPoint(x: bounds.maxX - SLIDER_BUTTON_WIDTH, y: bounds.midY + (SLIDER_BUTTON_WIDTH / 2))
                $0.textColor = minMaxDisplayLabelColor
                $0.textAlignment = .center
                $0.font = UIFont.systemFont(ofSize: 12, weight: .regular)
                addSubview($0)
            }
            
            // Mid View
            sliderMidLabel = UILabel(frame: CGRect(x: 0, y: 0, width: 49, height: 24))
            sliderMidLabel.then {
                $0.textColor = rangeDisplayLabelColor
                $0.textAlignment = .center
                $0.font = UIFont.systemFont(ofSize: 12, weight: .medium)
            }
            sliderMidView = UIView(frame: CGRect(x: 0, y: 0, width: 49, height: 29))
            sliderMidView.then {
                $0.center = CGPoint(x: SLIDER_BUTTON_WIDTH / 2, y: bounds.midY + (SLIDER_BUTTON_WIDTH / 2) + (paddingRangeView / 2))
                if let image = UIImage(named: iconRangeMidView) {
                    $0.backgroundColor = UIColor(patternImage: image)
                }
                addSubview($0)
                $0.addSubview(sliderMidLabel)
                $0.alpha = 0
            }
            segmentWidth = getSegmentWidth(forSegmentCount: numberOfSegments)
            rangeSliderButtonImage = UIImage(named: iconSliderButton)
            segmentUnSelectedImage = UIImage(named: iconSegmentUnSelected)
            
        }
        
        // Init the sliding representing views
        private func initSliderViews() {
            DEFAULT_SLIDER_FRAME = CGRect(x: SLIDER_BUTTON_WIDTH / 2, y: self.bounds.midY, width: self.bounds.width - SLIDER_BUTTON_WIDTH, height: 4)
            
            sliderBackgroundView = UIView(frame: DEFAULT_SLIDER_FRAME)
            sliderBackgroundView.then {
                $0.backgroundColor = rangeSliderBackgroundColor
                addSubview($0)
            }
            
            sliderForegroundView = UIView(frame: DEFAULT_SLIDER_FRAME)
            sliderForegroundView.then {
                $0.backgroundColor = rangeSliderForegroundColor
                addSubview($0)
            }
            startSliderButton = getSegmentButton(withSegmentIndex: 1, isSlider: true)
            addSubview(startSliderButton)
            
            endSliderButton = getSegmentButton(withSegmentIndex: numberOfSegments, isSlider: true)
            addSubview(endSliderButton)
            
            // Pan gesture for identifying the sliding (More accurate than touchesMoved).
            addPanGestureRecognizer()
        }
        
        // Update the frames
        private func updateFrame() {
            if self.numberOfSegments >= 2 {
                // If the range selectors are at the extreme points, then reset the frame. Else fo nothing
                if isRangeSlidersPlacedAtExtremePosition() {
                    DEFAULT_SLIDER_FRAME = CGRect(x: SLIDER_BUTTON_WIDTH / 2, y: self.bounds.midY, width: self.bounds.width - SLIDER_BUTTON_WIDTH, height: 4)
                    self.sliderBackgroundView.frame = DEFAULT_SLIDER_FRAME
                    self.sliderForegroundView.frame = DEFAULT_SLIDER_FRAME
                    
                    self.sliderBackgroundView.backgroundColor = self.rangeSliderBackgroundColor
                    self.sliderForegroundView.backgroundColor = self.rangeSliderForegroundColor
                    
                    segmentWidth = getSegmentWidth(forSegmentCount: numberOfSegments)
                    
                    startSliderButton.center = CGPoint(x: SLIDER_BUTTON_WIDTH / 2, y: sliderBackgroundView.frame.midY)
                    endSliderButton.center = getSegmentCenterPoint(forSegmentIndex: numberOfSegments)
                    
                    minRangeView.center = CGPoint(x: startSliderButton.frame.midX, y: bounds.midY - (SLIDER_BUTTON_WIDTH / 2) - (paddingRangeView / 2))
                    maxRangeView.center = CGPoint(x: endSliderButton.frame.midX, y: bounds.midY - (SLIDER_BUTTON_WIDTH / 2) - (paddingRangeView / 2))
                    
                    minLabel.center = CGPoint(x: startSliderButton.frame.midX, y: bounds.midY + (SLIDER_BUTTON_WIDTH / 2))
                    maxLabel.center = CGPoint(x: endSliderButton.frame.midX, y: bounds.midY + (SLIDER_BUTTON_WIDTH / 2))
                    
                    sliderMidView.center = CGPoint(x: startSliderButton.frame.midX, y: bounds.midY - (SLIDER_BUTTON_WIDTH / 2) - (paddingRangeView / 2))
                    
                    setImageForSegmentOrSliderButton(startSliderButton, isSlider: true)
                    setImageForSegmentOrSliderButton(endSliderButton, isSlider: true)
                    
                    // Reset the frame of all the intermediate buttons
                    for segmentIndex in 1...numberOfSegments {
                        let segmentButton = self.viewWithTag(segmentIndex) as? UIButton
                        segmentButton?.center = getSegmentCenterPoint(forSegmentIndex: segmentIndex)
                        if let button = segmentButton {
                            setImageForSegmentOrSliderButton(button, isSlider: false)
                        }
                    }
                    
                    // Slide the buttons if the initial position is needed
                    slideRangeSliderButtonsIfNeeded()
                    checkInit = true
                }
            }
        }
    }

    4. Tạo các hàm để tính toán vị trí và di chuyển slider

    extension TWSlider {
        private func isRangeSlidersPlacedAtExtremePosition() -> Bool {
            let sliderBackgroundViewMaxX = sliderBackgroundView.frame.maxX + (SLIDER_BUTTON_WIDTH / 2)
            return (startSliderButton.frame.minX == 0.0 && endSliderButton.frame.maxX == sliderBackgroundViewMaxX)
        }
        
        private func slideRangeSliderButtonsIfNeeded() {
            var startScrollPoint = CGPoint.zero
            var endScrollPoint = CGPoint(x: bounds.size.width, y: 0)
            if let min = minRangeInitialIndex, let max = maxRangeInitialIndex, min < max {
                let startX = getSegmentCenterPoint(forSegmentIndex: min).x
                startScrollPoint.x = startX > (SLIDER_BUTTON_WIDTH / 2) ? startX : SLIDER_BUTTON_WIDTH / 2
                let endX = getSegmentCenterPoint(forSegmentIndex: max).x
                endScrollPoint.x = endX > (SLIDER_BUTTON_WIDTH / 2) ? endX : SLIDER_BUTTON_WIDTH / 2
            }
            scrollStartAndEndSlider(for: startScrollPoint, andEndScroll: endScrollPoint)
            minRangeInitialIndex = 0
            maxRangeInitialIndex = 0
        }
        
        private func scrollStartAndEndSlider(for startScrollPoint: CGPoint, andEndScroll endScrollPoint: CGPoint) {
            startSliderButton.isSelected = true
            sliderDidSlide(for: startScrollPoint)
            startSliderButton.isSelected = false
            
            endSliderButton.isSelected = true
            sliderDidSlide(for: endScrollPoint)
            endSliderButton.isSelected = false
        }
    }

    5. Tạo các hàm tính toán frame của slider

    extension TWSlider {
        private func getSliderViewWidth() -> CGFloat {
            let startMinX = startSliderButton.frame.midX
            let endMinX = endSliderButton.frame.midX
            if startMinX > endMinX {
                return startMinX - endMinX
            } else {
                return endMinX - startMinX
            }
        }
        
        private func sliderMidPoint(forPoint point: CGFloat) -> CGFloat {
            let sliderMidPoint = point - (SLIDER_BUTTON_WIDTH / 2)
            return sliderMidPoint
        }
        
        private func getSegmentWidth(forSegmentCount segmentCount: Int) -> CGFloat {
            let segmentCount = CGFloat(segmentCount - 1)
            let sliderWidth = frame.width - SLIDER_BUTTON_WIDTH
            return sliderWidth / segmentCount
        }
        
        private func getSegmentButton(withSegmentIndex segmentIndex: Int, isSlider: Bool) -> UIButton {
            // Create rounded button for representing slider segments
            let segmentButton = UIButton(type: .custom)
            segmentButton.frame = CGRect(x: 0, y: 0, width: SLIDER_BUTTON_WIDTH, height: SLIDER_BUTTON_WIDTH)
            segmentButton.center = getSegmentCenterPoint(forSegmentIndex: segmentIndex)
            setImageForSegmentOrSliderButton(segmentButton, isSlider: isSlider)
            return segmentButton
        }
        
        // MARK: - Calculation for segment button frame
        private func getSegmentCenterPoint(forSegmentIndex segmentIndex: Int) -> CGPoint {
            let pointX = CGFloat((CGFloat(segmentIndex - 1) * segmentWidth) + (SLIDER_BUTTON_WIDTH / 2))
            return CGPoint(x: pointX, y: sliderBackgroundView.frame.midY)
        }
        
        private func setImageForSegmentOrSliderButton(_ button: UIButton, isSlider: Bool) {
            if let image = rangeSliderButtonImage, isSlider {
                button.setImage(image, for: .normal)
            } else if let iamge = segmentSelectedImage {
                button.setImage(iamge, for: .normal)
            } else {
                button.setImage(getImageWithSize(isSlider ? sliderSize : segmentSize, with: segmentSelectedColor), for: .normal)
            }
            
            button.imageView?.layer.masksToBounds = true
            let buttonWidth = button.imageView?.frame.size.width ?? button.frame.size.width
            button.imageView?.layer.cornerRadius = buttonWidth / 2
        }
        
        private func getImageWithSize(_ size: CGSize, with backgroundColor: UIColor?) -> UIImage? {
            let imageView = UIView(frame: CGRect(x: 0, y: 0, width: size.width, height: size.height))
            imageView.backgroundColor = backgroundColor
            
            UIGraphicsBeginImageContextWithOptions(CGSize(width: size.width, height: size.height), false, 1.0)
            if let context = UIGraphicsGetCurrentContext() {
                imageView.layer.render(in: context)
            }
            let image = UIGraphicsGetImageFromCurrentImageContext()
            UIGraphicsEndImageContext()
            return image
        }
    }

    6. Tạo các hàm để tính toán vị trí và xử lí delegate của Pangesture

    extension TWSlider {
        
        private func addPanGestureRecognizer() {
            let panGesture = UIPanGestureRecognizer(target: self, action: #selector(handlePanGesture))
            panGesture.maximumNumberOfTouches = 1
            addGestureRecognizer(panGesture)
        }
        
        // Pan Gesture selector method
        
        @objc
        func handlePanGesture(_ panGesture: UIPanGestureRecognizer) {
            let point = panGesture.location(in: self)
            
            if panGesture.state == .began {
                self.setSelectedStateForSlidingButton(point)
            } else if panGesture.state == .changed {
                sliderDidSlide(for: point)
            } else if panGesture.state == .ended || panGesture.state == .failed || panGesture.state == .cancelled {
                // Move the slider to nearest segment
                moveSliderToNearestSegment(withEnding: point)
                resetSelectedStateForSlidingButtons()
            }
        }
        
        // If sliding began, check if startSlider is moved or endSlider is moved
        private  func setSelectedStateForSlidingButton(_ point: CGPoint) {
            if startSliderButton.frame.contains(point) {
                startSliderButton.isSelected = true
                endSliderButton.isSelected = false
            } else if endSliderButton.frame.contains(point) {
                endSliderButton.isSelected = true
                startSliderButton.isSelected = false
            } else {
                startSliderButton.isSelected = false
                endSliderButton.isSelected = false
            }
        }
        
        private func sliderDidSlide(for point: CGPoint) {
            var newPoint = point
            // Check if startButton is moved or endButton is moved. Based on the moved button, set the frame of the slider button and foregroundSliderView
            newPoint = resetFrameOnBoundsCross(for: point)
            
            if startSliderButton.isSelected {
                if shouldStartButtonSlide(for: newPoint) {
                    UIView.animate(withDuration: 0.1, animations: {
                        // Change only the x value for startbutton
                        self.startSliderButton.frame = CGRect(x: self.sliderMidPoint(forPoint: newPoint.x), y: self.startSliderButton.frame.origin.y, width: self.startSliderButton.frame.size.width, height: self.startSliderButton.frame.size.height)
                        
                        // Change the x and width for slider foreground view
                        if self.shouldSliderButtonOverlap {
                            let startMidX = self.startSliderButton.frame.midX
                            let endMidX = self.endSliderButton.frame.midX
                            var originX = self.startSliderButton.frame.origin.x
                            if endMidX < startMidX {
                                originX = self.endSliderButton.frame.origin.x
                            }
                            self.sliderForegroundView.frame = CGRect(x: originX + self.SLIDER_BUTTON_WIDTH / 2, y: self.sliderForegroundView.frame.origin.y, width: self.getSliderViewWidth(), height: self.sliderForegroundView.frame.size.height)
                        } else {
                            let originX = self.startSliderButton.frame.origin.x + self.SLIDER_BUTTON_WIDTH / 2
                            self.sliderForegroundView.frame = CGRect(x: originX, y: self.sliderForegroundView.frame.origin.y, width: self.getSliderViewWidth(), height: self.sliderForegroundView.frame.size.height)
                        }
                        
                        // Change the x and width for slider foreground view
                        self.minRangeView.center = CGPoint(x: self.startSliderButton.frame.midX, y: self.startSliderButton.frame.minY - self.paddingRangeView)
                        
                        self.sliderMidView.center = CGPoint(x: self.endSliderButton.frame.midX, y: self.startSliderButton.frame.minY - self.paddingRangeView)
                        
                        // Update the intermediate segment colors
                        self.updateSegmentColor(for: newPoint)
                    }, completion: { _ in
                        self.callScrollDelegate(point: newPoint, isStartSliderButton: true)
                    })
                }
            } else if endSliderButton.isSelected {
                if shouldEndButtonSlide(for: newPoint) {
                    UIView.animate(withDuration: 0.1, animations: {
                        // Change only the x value for endbutton
                        self.endSliderButton.frame = CGRect(x: self.sliderMidPoint(forPoint: newPoint.x), y: self.endSliderButton.frame.origin.y, width: self.endSliderButton.frame.size.width, height: self.endSliderButton.frame.size.height)
                        
                        // Change the x and width for slider foreground view
                        if self.shouldSliderButtonOverlap {
                            let startMidX = self.startSliderButton.frame.midX
                            let endMidX = self.endSliderButton.frame.midX
                            var originX = self.startSliderButton.frame.origin.x
                            if endMidX < startMidX {
                                originX = self.endSliderButton.frame.origin.x
                            }
                            self.sliderForegroundView.frame = CGRect(x: originX + self.SLIDER_BUTTON_WIDTH / 2, y: self.sliderForegroundView.frame.origin.y, width: self.getSliderViewWidth(), height: self.sliderForegroundView.frame.size.height)
                        } else {
                            let originX = self.startSliderButton.frame.origin.x + self.SLIDER_BUTTON_WIDTH / 2
                            self.sliderForegroundView.frame = CGRect(x: originX, y: self.sliderForegroundView.frame.origin.y, width: self.getSliderViewWidth(), height: self.sliderForegroundView.frame.size.height)
                        }
                        
                        // Change the x and width for slider foreground view
                        self.maxRangeView.center = CGPoint(x: self.endSliderButton.frame.midX, y: self.endSliderButton.frame.minY - self.paddingRangeView)
                        
                        self.sliderMidView.center = CGPoint(x: self.startSliderButton.frame.midX, y: self.endSliderButton.frame.minY - self.paddingRangeView)
                        
                        // Update the intermediate segment colors
                        self.updateSegmentColor(for: newPoint)
                    }, completion: { _ in
                        self.callScrollDelegate(point: newPoint, isStartSliderButton: false)
                    })
                }
            }
        }
        
        // Method that handles if the sliders move out of range
        private func resetFrameOnBoundsCross(for point: CGPoint) -> CGPoint {
            var newPoint = point
            if shouldSliderButtonOverlap {
                if point.x < 0 {
                    newPoint.x = 0
                } else if sliderMidPoint(forPoint: point.x) >= sliderBackgroundView.bounds.maxX {
                    newPoint.x = sliderBackgroundView.bounds.maxX + SLIDER_BUTTON_WIDTH / 2
                }
            } else {
                if startSliderButton.isSelected {
                    if sliderMidPoint(forPoint: point.x) >= endSliderButton.frame.midX - segmentWidth {
                        newPoint.x = endSliderButton.frame.midX - segmentWidth
                    } else if point.x < 0 {
                        newPoint.x = 0
                    }
                } else if endSliderButton.isSelected {
                    if point.x <= startSliderButton.frame.midX + segmentWidth {
                        newPoint.x = startSliderButton.frame.midX + segmentWidth
                    } else if sliderMidPoint(forPoint: point.x) >= sliderBackgroundView.bounds.maxX {
                        newPoint.x = sliderBackgroundView.bounds.maxX + SLIDER_BUTTON_WIDTH / 2
                    }
                }
            }
            
            return newPoint
        }
        
        private func shouldStartButtonSlide(for point: CGPoint) -> Bool {
            if shouldSliderButtonOverlap {
                return (point.x >= (SLIDER_BUTTON_WIDTH / 2)) && (point.x <= (bounds.maxX - SLIDER_BUTTON_WIDTH / 2))
            } else {
                var endButtonMidPoint = endSliderButton.frame.midX
                endButtonMidPoint -= segmentWidth
                return round(point.x) <= round(endButtonMidPoint) && point.x >= SLIDER_BUTTON_WIDTH / 2
            }
        }
        
        private func shouldEndButtonSlide(for point: CGPoint) -> Bool {
            if shouldSliderButtonOverlap {
                return point.x >= SLIDER_BUTTON_WIDTH / 2
            } else {
                var startButtonMidPoint = startSliderButton.frame.midX
                startButtonMidPoint += shouldSliderButtonOverlap ? 0 : segmentWidth
                return round(point.x) >= round(startButtonMidPoint) && point.x <= sliderMidPoint(forPoint: frame.size.width)
            }
        }
        
        // Call the delegate to set the label for min range and max range
        private func callScrollDelegate(point: CGPoint, isStartSliderButton: Bool) {
            let nearestSegmentIndex = Int(round(sliderMidPoint(forPoint: point.x) / segmentWidth))
            
            var startIndex = Int(round(startSliderButton.frame.minX / segmentWidth))
            var endIndex = Int(round(endSliderButton.frame.minX / segmentWidth))
            
            if isStartSliderButton {
                startIndex = nearestSegmentIndex
            } else {
                endIndex = nearestSegmentIndex
            }
            updateData(startIndex: startIndex, endIndex: endIndex, endDragDrop: false)
        }
        
        private func updateData(startIndex: Int, endIndex: Int, endDragDrop: Bool) {
            if startIndex > endIndex {
                let min = endIndex < 0 ? 0 : endIndex
                let max = startIndex > numberOfSegments ? numberOfSegments : startIndex
                delegate?.sliderScrolled(self, toMinIndex: min, andMaxIndex: max, endDragDrop: endDragDrop)
                minRangeLabel.text = maxRangeText
                maxRangeLabel.text = minRangeText
            } else {
                let min = startIndex < 0 ? 0 : startIndex
                let max = endIndex > numberOfSegments ? numberOfSegments : endIndex
                delegate?.sliderScrolled(self, toMinIndex: min, andMaxIndex: max, endDragDrop: endDragDrop)
                minRangeLabel.text = minRangeText
                maxRangeLabel.text = maxRangeText
            }
            
            if startIndex != endIndex {
                UIView.animate(withDuration: 0.3, delay: 0, options: .curveEaseIn) {
                    self.sliderMidView.alpha = 0
                    self.minRangeView.alpha = 1
                    self.maxRangeView.alpha = 1
                } completion: { _ in
                    print("completion \(startIndex) = \(endIndex)")
                }
            } else {
                sliderMidLabel.text = "\(minRangeText ?? "") - \(maxRangeText ?? "")"
                UIView.animate(withDuration: 0.3, delay: 0, options: .curveEaseIn) {
                    self.sliderMidView.alpha = 1
                    self.minRangeView.alpha = 0
                    self.maxRangeView.alpha = 0
                } completion: { _ in
                    print("completion \(startIndex) = \(endIndex)")
                }
            }
        }
        
        private func updateSegmentColor(for point: CGPoint) {
            
            if shouldSliderButtonOverlap {
                let startMinX = startSliderButton.frame.midX
                let endMinX = endSliderButton.frame.midX
                
                var min: CGFloat = 0.0
                var max: CGFloat = 0.0
                if startMinX > endMinX {
                    min = (endSliderButton.frame.minX / segmentWidth).rounded(.up)
                    max = (startSliderButton.frame.minX / segmentWidth).rounded(.down)
                } else {
                    min = (startSliderButton.frame.minX / segmentWidth).rounded(.up)
                    max = (endSliderButton.frame.minX / segmentWidth).rounded(.down)
                }
                updateSegmentColor(withStart: Int(min) + 1, andEnd: Int(max) + 1)
            } else {
                if startSliderButton.isSelected {
                    let startMinX = (sliderMidPoint(forPoint: startSliderButton.frame.midX) / segmentWidth).rounded(.up)
                    let endMinX = (sliderMidPoint(forPoint: endSliderButton.frame.midX) / segmentWidth).rounded(.up)
                    updateSegmentColor(withStart: Int(startMinX) + 1, andEnd: Int(endMinX) + 1)
                } else if endSliderButton.isSelected {
                    let startMinX = (startSliderButton.frame.minX / segmentWidth).rounded(.up)
                    let endMinX = (endSliderButton.frame.minX / segmentWidth).rounded(.down)
                    updateSegmentColor(withStart: Int(startMinX) + 1, andEnd: Int(endMinX) + 1)
                }
            }
            let pointX: Int = Int(round(point.x - SLIDER_BUTTON_WIDTH / 2))
            if pointX % Int(round(segmentWidth)) == 0 {
                let generator = UIImpactFeedbackGenerator(style: .medium)
                generator.impactOccurred()
            }
        }
        
        private func updateSegmentColor(withStart startIndex: Int, andEnd endIndex: Int) {
            // Segments before startSegment slider
            
            if startIndex > 1 {
                for segmentIndex in 1..<startIndex {
                    if let segmentButton = viewWithTag(segmentIndex) as? UIButton {
                        if let image = segmentUnSelectedImage {
                            segmentButton.setImage(image, for: .normal)
                        } else {
                            let image = getImageWithSize(segmentSize, with: segmentUnSelectedColor)
                            segmentButton.setImage(image, for: .normal)
                        }
                    }
                }
            }
            
            // Segments between startSegment slider and endSegment slider
            if startIndex <= endIndex {
                for segmentIndex in startIndex...endIndex {
                    if let segmentButton = viewWithTag(segmentIndex) as? UIButton {
                        if let image = segmentSelectedImage {
                            segmentButton.setImage(image, for: .normal)
                        } else {
                            let image = getImageWithSize(segmentSize, with: segmentSelectedColor)
                            segmentButton.setImage(image, for: .normal)
                        }
                    }
                }
            }
            
            // Segments after endSegment slider
            if endIndex + 1 <= numberOfSegments {
                for segmentIndex in (endIndex + 1)...numberOfSegments {
                    if let segmentButton = viewWithTag(segmentIndex) as? UIButton {
                        if let image = segmentUnSelectedImage {
                            segmentButton.setImage(image, for: .normal)
                        } else {
                            let image = getImageWithSize(segmentSize, with: segmentUnSelectedColor)
                            segmentButton.setImage(image, for: .normal)
                        }
                    }
                }
            }
        }
        
        // Slide to nearest position
        private func moveSliderToNearestSegment(withEnding point: CGPoint) {
            var newPoint = point
            newPoint = resetFrameOnBoundsCross(for: point)
            
            let nearestSegmentIndex = Int(round(sliderMidPoint(forPoint: newPoint.x) / segmentWidth))
            sliderDidSlide(for: CGPoint(x: CGFloat(SLIDER_BUTTON_WIDTH / 2 + CGFloat(nearestSegmentIndex) * segmentWidth), y: newPoint.y))
            
            var startIndex = Int(round(startSliderButton.frame.minX / segmentWidth))
            var endIndex = Int(round(endSliderButton.frame.minX / segmentWidth))
            
            if startSliderButton.isSelected {
                startIndex = nearestSegmentIndex
            } else if endSliderButton.isSelected {
                endIndex = nearestSegmentIndex
            }
            updateData(startIndex: startIndex, endIndex: endIndex, endDragDrop: true)
        }
        
        // After ending, reset the selected state of both buttons
        private func resetSelectedStateForSlidingButtons() {
            startSliderButton.isSelected = false
            endSliderButton.isSelected = false
        }
    }

    5. Tạo các hàm public cho phép thay đổi giá trị của TWSlider

    extension TWSlider {
        
        // Scroll to desired location on loading
        func scrollStartSlider(to startIndex: Int, andEnd endIndex: Int) {
            minRangeInitialIndex = startIndex
            maxRangeInitialIndex = endIndex
            if checkInit {
                slideRangeSliderButtonsIfNeeded()
            }
        }
        
        // MARK: - Setter methods, setup value for slider
        func setNumberOfSegments(_ numberOfSegments: Int, minText: String?, maxText: String?) {
            self.numberOfSegments = numberOfSegments
            
            // After setting the numberOfSegments, set all the necessary views
            segmentWidth = getSegmentWidth(forSegmentCount: self.numberOfSegments)
            addSegmentButtons()
            addSubview(startSliderButton)
            addSubview(endSliderButton)
            minLabel.text = minText
            maxLabel.text = maxText
        }
        
        private func addSegmentButtons() {
            for segmentIndex in 1...numberOfSegments {
                let segmentButton = getSegmentButton(withSegmentIndex: segmentIndex, isSlider: false)
                segmentButton.then {
                    $0.tag = segmentIndex
                    $0.isUserInteractionEnabled = false
                    addSubview($0)
                }
            }
        }
    }

    3. Cách sử dụng

    Bước 1: mở file storyboard hoặc file .xib và thêm UIView và constraint, sửa lại tên class thành class custom slider mới tạo ở trên ở project này là TWSlider -> Kéo outlet

    Bước 2: mở viewcontroller vừa được kéo outlet viết hàm setup cho slider như sau

    class ViewController: UIViewController {
    
        @IBOutlet weak var customSlider: TWSlider!
        private let data: [String] = ["10", "11", "12", "13", "14", "15", "16", "17", "18", "19", "20"]
        override func viewDidLoad() {
            super.viewDidLoad()
            // Do any additional setup after loading the view.
            customSlider.then {
                $0.shouldSliderButtonOverlap = true
                $0.delegate = self
            }
            customSlider.then {
                $0.setNumberOfSegments(data.count, minText: data.first, maxText: data.last)
                $0.scrollStartSlider(to: 0 + 1, andEnd: 10 + 1)
            }
        }
    }
    
    extension ViewController: TWSliderDelegate {
        func sliderScrolled(_ slider: TWSlider?, toMinIndex minIndex: Int, andMaxIndex maxIndex: Int, endDragDrop: Bool) {
            customSlider.minRangeText = data[minIndex]
            customSlider.maxRangeText = data[maxIndex]
        }
    }

    Bạn có thể tuỳ chỉnh số hiển thị trên slider bằng các số tương ứng với project.

    Mình để project ở đây cho mọi người tham khảo nếu cần nhé!

    Vậy là chúng ta đã hoàn thành việc custom Slider hai chiều, mình hi vọng nó có thể giải quyết được bài toán của các bạn. Cảm ơn các bạn đã theo dõi bài viết!