Skip to main content

Làm thế nào để kiểm tra Auto Layout trên nhiều loại thiết bị iOS?

Chào các bạn, mấy bài trước tôi có hướng dẫn các bạn làm auto layout trên iOS, nếu bạn nào chưa đọc thì có thể xem lại tại đây. Vậy sau khi làm auto layout xong các bạn có tự tin rằng nó đã thực sự chạy tốt và không có bug xảy ra?

Các bạn có thể viết unit test cho phần logic rất dễ dàng nếu các bạn chia những code xử lý logic thành từng hàm có input và output rõ ràng, công việc này rất là tốt. Nếu sau này các bạn viết thêm những đoạn code khác hoặc fix bug khác làm logic đó sai, bạn có thể chạy unit test để phát hiện ra lỗi ngay. Một khi bạn làm unit test tốt và độ bao phủ cao thì số bug chắc chắn sẽ ít và thời gian bạn test bằng tay cũng giảm đi nhiều, và bạn sẽ rất tự tin khi nói với khách hàng của bạn rằng: code của mình sẽ không sinh những bug vặt.

Nhưng mặt hạn chế trước giờ  mình gặp phải là mình chỉ có thể viết unit test cho phần logic, còn phần layout thì mặc định công cụ XCode chưa hỗ trợ nhiều, thành ra số bug mình khó thấy và không test kỹ được là phần layout trên những loại thiết bị khác nhau. Những bug về layout tuy nhỏ nhưng cũng rất là khó chịu, đa số các công ty lớn cũng đau đầu vì chuyện này, thành ra công ty Linkedin đã hỗ trợ viết ra 1 thư viện giúp cho các bạn phát hiện ra những bug về layout 1 cách dễ dàng và nhanh chóng đó chính là thư viện LayoutTest-iOS. Tôi đáng giá cao thư viện này thành ra hôm nay tôi sẽ giới thiệu cho các bạn cách xài cũng như có cái nhìn tổng quát về như viện này.

Trong ví dụ của thư viện này họ đã viết bằng ngôn ngữ Swift với đối tượng Cell, vì thế tôi sẽ không dùng swift và cell để làm ví dụ nữa, mà sẽ hướng dẫn viết code bằng Objective-C. Đầu tiên tôi tạo 1 dự án tên là LayoutTest và khai báo trong pod file như sau để tải thư viện và tích hợp chúng vào dự án:
Lưu ý: Bạn phải mở comment 'use_frameworks!' để có thể xài được thư viện này trên Objective-C. Nếu không sẽ báo error khi sử dụng thư viện. Hiện tại thư viện mình tải về phiên bản 4.0.0.

I. Custom UIView:

Ví dụ thứ nhất tôi sẽ chỉ cho các bạn làm cách nào thể làm unit test cho 1 custom view, Giả sử tôi tạo 2 Label trong custom view như sau:

- Bước 1: Tạo 1 Category để khai báo những hàm để hỗ trợ chạy Unit Test, ví dụ mình đặt tên cho Category này là 'UITest', sau khi tạo xong nó sẽ sinh ra 2 tập tin 'CustomView+UITest.h' và 'CustomView+UITest.m'
Trong file .h: bạn khai báo để implement đối tượng 'LYTViewProvider'.
Sau đó trong file .m: 2 hàm bạn quan tâm bắt buộc phải thực thi là hàm 'dataSpecForTest' (dùng để khởi tạo dữ liệu test) và 'viewForData' (dùng để khởi tạo và sử dụng lại view).
Nếu bạn muốn hỗ trợ nhiều kích thước màn hình thì thêm hàm 'sizesForView' này vào.
Đối tượng 'LYTStringValues' chứa danh sách bộ test case mặc định cho kiểu NSString, nếu các bạn muốn tạo bộ dữ liệu cho riêng bạn thì nên kế thừa đối tượng này và override lại hàm trả danh sách 'values'.
Hàm 'setupWithJSON' dùng để truyền dữ liệu và cập nhật UI.
- Bước 2: Bạn tạo 1 file unit test và viết 1 hàm như sau:
Sau khi bạn viết hàm này xong thì chạy Test thử, thì sẽ trả về thành công.

Giả sử mình cho 2 label này chồng lên nhau để xem thư viện này có báo lỗi hay không như sau:

Sau đó mình chạy lại Unit Test thì sẽ báo như hình sau:
Bạn để ý trong console của XCode có báo dòng chữ "Snapshots of failing tests can be found in:", đây là dòng chữ chỉ cho bạn biết layout của bạn đang bị lỗi chồng lên nhau như thế nào, bạn copy đường dẫn này và past lên trên browser thì sẽ thấy màn hình của bạn đang bị lỗi gì:

Nếu việc chồng lên này là cố tình bạn muốn như vậy thì bạn có thể thiết lập để chúng bỏ qua lỗi này. Theo tôi tốt nhất khi thiết kế giao diện không nên để những đối tượng này chồng lên nhau.

II. Custom UIViewController:

Giả sử tôi tạo 1 ViewController trong storyboard như sau:
Tại đối tượng Top Right, mình cố tình tạo bug cho nó để có thể xuất lỗi như 'line of number' bằng 0, và không tạo constraint cho height. Nếu bộ dữ liệu test của bạn không đủ dài thì sẽ không thấy bug này.

Cách làm cho UIViewController hơi khác so với custom view, ở chỗ khai báo trong Category như sau:
Bạn thấy hàm 'viewForData' mình đã dùng đối tượng static để có thể reuse lại ViewController.
Sau khi chạy Unit Test xong thì sẽ báo lỗi như sau:
Bạn có thể thấy với 1 chuỗi dài do mình không giới hạn chiều cao nên nó sẽ bị kéo dài xuống và bị chồng đối tượng với nhau.

Ngoài việc kiểm tra có bị chồng lên nhau hay không thì các bạn có thể viết thêm những unit test trong file test để kiểm tra chính xác vị trí của đối tượng đó như sau:
Thế là xong những bước căn bản thiết lập thư viện và kiểm tra giao diện có bị chồng lên nhau hay không. Các bạn có thể xem toàn bộ source code của mình tại đây. Nếu thấy hay và hữu ích hãy vote +1 cho mình tại blog này hay nhấn nút star trên GitHub nhé. Cám ơn các bạn rất nhiều.

Các bạn có thể xem thông tin chi tiết tại trang chủ của thư viện LayoutTest-iOS tại đường dẫn sau. Để biết thêm chi tiết các bạn có thể xem thêm tài liệu hướng dẫn của Linkedin.

Comments

Popular posts from this blog

Phân biệt biến kiểu Property, Public, Protected, Private trong ngôn ngữ Objective C

- Theo kinh nghiệm làm việc của mình với các bạn trong nhóm khi lập trình Objective-C và cũng đọc qua code của những project cũ. Ít khi nào mọi người để ý và khai báo đúng với ý đồ của từng đối tượng, và vi phạm quy tắc tính đóng gói, tính bảo mật thông tin của đối tượng trong lập trình hướng đối tượng (Tham khảo lý thuyết Lập trình hướng đối tượng tại trang Wiki ). - Theo ngôn ngữ lập trình Java, người ta khuyến khích mỗi khi dùng biến kiểu public thì nên đặt 1 biến private và hỗ trợ những hàm getter/setter để truy suất biến private đó.     + Nguyên nhân họ nói là đảm bảo tính đóng gói, và nếu sau này có thay đổi gì trên biến đó bạn có thể sửa được dễ dàng, chi tiết về vấn đề này ở đây .     + Nói tóm tại thì nguyên nhân chính là có thể kiểm soát được truy xuất đến giá trị của 1 đối tượng từ bên ngoài, có thể dễ dàng mở rộng code bằng cách override lại những hàm getter/setter. - Các bạn có thể áp dụng nguyên tắc đó từ bên Java qua ngôn ngữ lập trình Object...

(Kinh nghiệm) Auto layout và Size classes trong iOS - Phần 1

Trước đây khi viết những ứng dụng chạy trên nhiều màn hình hoặc hỗ trợ màn hình xoay ngang, xoay dọc mọi người hay dùng code để có thể chỉnh được những vị trí cũng như kích thước của những đối tượng. Hoặc có thể dùng Autosizing để tự động canh chỉnh những đối tượng nhưng không tối ưu và tiện lợi cho lắm, hình minh hoạ ở dưới: Vì thế từ lúc Apple phát hành ra iPhone 5, iPhone 5s với kích thước màn hình là 4 inch, làm phát sinh thêm vấn đề " Làm thế nào ta có thể thiết kế giao diện có thể chạy được trên nhiều màn hình? " mà không làm thay đổi nhiều code để có thể dễ dàng bảo trì ứng dụng. Apple mới phát triển chức năng Auto Layout và Size Classes để thực hiện nhiệm vụ này. Nếu bạn đã quen dùng Autosizing để thiết kế giao diện thì bạn có thể vẫn sử dụng chúng. Nhưng tuỳ theo từng dự án mà khách hàng hay người PM hoặc leader của bạn muốn bạn dùng công nghệ mới Auto Layout và Size Classes   để làm layout trên iOS mà không cần dùng bất cứ đoạn code nào và chỉ viết trên 1 storyboard...

(Căn bản) Bài 1: Hướng dẫn tạo tài khoản Apple ID và iTune không cần thẻ Visa hoặc MasterCard

Nếu bạn muốn lập trình trên iOs hoặc trên MacOs thì bạn nên có tài khoản Apple ID và tài khoản iTune. Sẽ hữa ích cho bạn khi cài đặt và nân cấp chương trình XCode. Nhưng khó khăn ở chỗ nếu bạn không có thẻ Visa hoặc Master Card, mà vẫn muốn có tài khoản để có thể cài ứng dụng trên AppStore. Sau đây mình xin hướng dẫn cách tạo tài khoản mà không cần những thẻ đó và có thể tạo tài khoản trên những store ở các nước khác. Đầu tiên bạn mở chương trình iTune lên và vào tab "App Store", sau đó bạn kiếm chương trình nào Free và nhấn vào đó nó sẽ hiển thị ra màn hình như sau:    Khi đang ở màn hình này bạn nhấn vào chữ "Free" bên dưới hình sẽ hiển thị ô cho bạn nhập Apple ID và Password.Vì bạn chưa có nên hãy nhấn vào nút "Create Apple ID" để tạo tài khoản. Sau khi bạn nhấn vào nút "Create Apple ID" bạn sẽ qua màn hình như bên dưới. Tại đây bạn có thể chuyển AppStore của các nước (Vì có những chương trình chỉ cài được trên từng AppStore mỗi nước thôi) hi...