Tuesday, May 28, 2013

FragmentPagerAdapter và lỗi NullPointerException khi xoay màn hình

Hôm qua phát hiện ở phần mềm RC4W client của mình bị lỗi kì lạ: khi người sử dụng xoay màn hình, chương trình bị force close.

Nguyên nhân sau khi tìm hiểu thì do hàm getActivity dùng cho Fragment trả về giá trị null sau khi người sử dụng xoay màn hình, điều này phát sinh NullPointerException. Tìm hiểu rất nhiều nơi thì tìm được một câu trả lời hợp lý ở đây: http://stackoverflow.com/questions/11631408/android-fragment-getactivity-sometime-returns-null

Nguyên nhân lỗi

Ngắn gọn: mình đã làm "không theo sách".

Dài dòng hơn:
1) Lỗi thứ nhất: tạo fragment mới mỗi lần onCreate() được gọi.
myfragment = new MyFragment();

Lệnh này gọi trong mỗi lần onCreate() của Activity. Theo hướng dẫn (không biết nguồn chính thống ở đâu, nhưng có 1 ở  nguồn không chính thống ở đây ), KHÔNG nên tạo fragment mới mỗi lần onCreate() của Activity. Theo đó, người ta hướng dẫn là dùng kiểu như sau:
    CustomFragment fragment;
    if (savedInstanceState != null) {
        fragment = (CustomFragment) getSupportFragmentManager().findFragmentByTag("customtag");
    } else {
        fragment = new CustomFragment();
        getSupportFragmentManager().beginTransaction().add(R.id.container, fragment, "customtag").commit();
    }

Tức là mỗi lần onCreate() được gọi, kiểm tra xem fragment có đã được tạo ra và đang được FragmentManager quản lý hay không. Nếu có thì lấy ra (bằng tag), không thì tạo mới. Tag này là String do mình tùy chọn.

2) Nhưng vẫn chưa hiểu tạo sao fragment mới tạo ra mỗi lần onCreate() lại làm cho getActivity() của nó trả về null.
Nguyên nhân nằm ở FragmentPagerAdapter. FragmentPagerAdapter có cơ chế tự gán tag, từ tìm lại fragment đã tạo. Nhờ đó mình không phải quan tâm đến đoạn mã kiểm tra tag và gán tag như ở mục trên nữa. Xem mã nguồn của FragmentPagerAdapter để biết thêm chi tiết. Khi mình tạo mới một fragment khi onCreate(), FragmentPagerAdapter này nó lại không quan tâm, nó tìm xem fragment cũ có không, và nhiều khi nó còn tồn tại (do FragmentManager quản lý), thế là nó lấy fragment cũ ra sử dụng, fragment mới bị bỏ qua và không được gắn với Activity nào. Kết quả là hàm gọi getActivity() thực hiện với Fragment mới trả về null.
Nguyên nhân nữa nằm ở việc mình truy cập tới fragment thông qua tên biến fragment này. Tức là lúc nào cũng là Fragment mới.

Tóm lại, để có lỗi này cần làm những việc sau:
Khi coding:
+ Sử dụng Fragment, FragmentPagerAdapter
+ onCreate(): tạo new Fragment (có thể là CustomFragment của bạn)
+ trong mã lệnh của class CustomFragment có gọi getActivity()
+ truy cập đến 1 biến CustomFragment trực tiếp

Khi sử dụng:
+ Activity mở ra -> onCreate() -> new Fragment (1)
+ Khi cần hiển thị Fragment, FragmentPagerAdapter tìm xem có fragment này trong FragmentManager không. Nếu có thì lấy ra dùng luôn, nếu không sẽ gọi hàm getItem() và sẽ add fragment này vào FragmentManager với Tag nó tự tạo ra. Sau đó Attach fragment này vào Activity.
+ Xoay màn hình -> recreate lại Activity -> gọi onCreate() -> new fragment (2).
+ Khi cần hiển thị Fragment, FragmentPagerAdapter sẽ làm như trên và tìm lấy ra được fragment (1) và hiển thị bình thường
+ Ở đâu đó bạn truy cập vào fragment(2) một cách trực tiếp, trong đó lại gọi getActivity. ---> NullPointerException. Vì fragment(2) không được FragmentPagerAdapter sử dụng và attach cho Activity nào cả.

Cách khắc phục lỗi này: 

Lỗi này mình khắc phục bằng cách là mỗi lần onCreate(), không gọi tạo new fragment mới ngay mà kiểm tra findFragmentByTag trong FragmentManager trước. Đầu vào Tag được tạo từ hàm makeFragmentName() ở trong  mã nguồn của FragmentPagerAdapter

Dù cách này dùng được nhưng cho thấy một lỗi rất quái đản và sự thiếu logic trong SDK của android. Trước giờ mình đều nghĩ là ở onCreate() thì phải create (new) chứ sao lại phải tìm lại xem có fragment đã tạo trước rồi hay không. Hay mình đã sai từ đầu? Tất cả widget khi recreate đều có thể sử dụng lại, không cần tạo mới? Sẽ tìm hiểu thêm về vấn đề này.



Thursday, May 16, 2013

Android Studio mới 15/05/2013

Hôm nay 15/5/2013, Google đã ra mắt bộ IDE lập trình cho android: android studio. Android studio được phát triển dựa trên Intellij IDEA Communitity Edition.

Mình đã chuyển sang Intellij IDEA từ eclipse khi phát triển chương trình RC4W. Mình từng nói một số cái hay của IDE này trong một bài viết kêu gào về cái khó trong lập trình android. Trong bài đó nói rằng công cụ mặt định mà google giới thiệu trên trang developer android của mình là eclipse. Việc sử dụng eclipse cồng kềnh, nặng nề, lắm lỗi cho những người mới lập trình android là đúng là như đi vào bụi rậm. Giờ đây Google ra Android Studio dựa trên Intellij IDEA thật sự quá tốt.

Mình đã tải về dùng thử, cảm nhận ban đầu là tốt, nói chung là giống với Intellij IDEA. Google có bổ sung thêm một số thứ linh tinh khác nữa nhưng chưa thử.

Với người dùng Windows, có lẽ nên đặt biến môi trường (Enviroment variables) trước khi chạy: JDK_HOME = <đường dẫn đến JDK> (ví dụ: c:\Program Files\Java\jdk1.7.0_2)

+1 cho công cụ lập trình tốt.

Sunday, May 12, 2013

Debug phần mềm android sử dụng Bluestacks

Nếu như bạn chưa biết thì bluestacks là phần mềm giả lập android chạy trên Windows và Mac OS. Nó giả lập rất tốt! Cảm giác là phải chạy nhanh gấp 10 lần giả lập mặc định của Google. Cài đặt rất đơn giản, chỉ cần tải file cài đặt ở trang chủ www.bluestacks.com và cài đặt.

Sau khi cài đặt xong bluestacks và chạy chương trình, bạn có thể debug chương trình android của mình ngay lập tức vì bluestack hiện lên trong danh sách thiết bị:

Bạn có thể xem logcat đầy đủ:


Và đây là màn hình giả lập của bluestacks: Đây là phần mềm Remote controller for walaoke của mình giả lập trên bluestacks.

Khi bạn cài đặt bluestacks, có thể bạn sẽ thắc mắc là kích thước màn hình nó không như hình ảnh ở trên. Bởi vì mình dùng một phần mềm tweak kích thước màn hình để thay đổi kích thước cho nó giống như ở trên điện thoại (màn hình dọc). Bạn hoàn toàn có thể thay đổi để thử các loại kích thước màn hình khác nhau để kiểm tra khả năng tương thích của phần mềm của bạn.

Quảng cáo: nếu bạn thích hát karaoke, mời bạn dùng thử Walaoke để biến máy tính của bạn thành đầu karaoke cùng với phần mềm điều khiển từ xa mình mới viết để chọn bài trực quan thuận tiện. (http://rc4w.blogspot.com - địa chỉ blog hướng dẫn; nên xem hướng dẫn cách cài đặt cả server nữa vì trên android chỉ là client)

Saturday, May 11, 2013

Cài android ra bộ nhớ ngoài cho điện thoại mất bộ nhớ trong

Đây là vấn đề không liên quan đến lập trình android, mà nó liên quan đến sửa điện thoại. Tuy nhiên mình vừa mắc phải sự cố này: con Samsung captivate (I897) đột nhiên chết bộ nhớ trong, bootloop, không thể khởi động lên được. Mình mò mẫm các nơi tìm cách xử lý và tìm được ở xdadeveloper (cái này dùng cho galaxy tab, nhưng cũng tương tự) hay ở youtube (dùng cho I9000M, nhưng cũng tương tự)

CẢNH BÁO: ai có đọc bài này mà cố làm theo thì tự chịu trách nhiệm về hành động của mình! Bạn cần hiểu mình đang làm gì, đọc đi đọc lại cách làm nhiều lần trước khi tiến hành.

Hiện tượng:
+ Bootloop
+ Vào recovery thấy báo lỗi /dev/block/mmcblk0p1
+ Cố gắng format mọi thứ liên quan đến bộ nhớ trong (data, datadata) đều vô dụng

Cách xử lý (ngắn gọn): cài OS ra bộ nhớ ngoài

Cách xử lý (đầy đủ)

Chuẩn bị:
+ Thẻ nhớ càng to càng nhanh càng tốt: như mình 16GB class 10.

+ Phần mềm minitool partition dùng để phân vùng cho thẻ nhớ cho giống với bộ nhớ trong được phân vùng khi xuất xưởng
+ ROM và công cụ để flash ROM (samsung là odin; hãng khác thì không rõ)
+ Công cụ để root và adb.

Tiến hành:

+ Bước 1: dùng minitool partition tạo phân vùng cho thẻ nhớ (cắm vào đầu đọc thẻ USB) giống với bộ nhớ trong khi mà nó được xuất xưởng. Cái này thấy bảo tùy vào điện thoại. Ở con Galaxy S của Samsung (và các biến thể) thì: một Primary + FAT32 (để làm bộ nhớ ngoài) và Primary +Ext4 (để cài os - 2GB-4GB). Cách dùng thì các bạn mò sẽ biết. Chú ý các phân vùng phải là Primary.
+ Bước 2: cắm thẻ nhớ đã phân vùng vào điện thoại và dùng odin (cho samsung) hoặc cái gì tương tự (cho hãng khác) để flash ROM stock vào cho máy. Bạn phải biết cái này rồi. Đừng hỏi cách flash ở đây.
+ Bước 3: chắc là máy đã có thể khởi động. Nhưng vấn đề là nó không nhận phân vùng FAT32 kia làm bộ nhớ ngoài. Phải tiến hành đổi chỗ internal và external  (đây chỉ là ảo thôi, internal chết rồi còn đâu). Nếu ai đã quen thuộc với android: adb, root, thì bước này không quá khó, với ai chưa quen thì mình cũng chịu chả biết phải chỉ dẫn từ đâu.

EDIT:
Bước 3 khá là mơ hồ nên mình sẽ nói thêm, nhưng nó áp dụng cho máy I897 của mình, và không chắc nó sẽ đúng với máy của bạn. Tuy nhiên chắc nó cũng sẽ gần giống.
Ở bước này máy đã có thể khởi động và nhưng nó không nhận vùng nhớ FAT32 làm bộ nhớ ngoài. Bạn cần tiến hành ROOT máy.
Sau đó tải Es explorer về (https://play.google.com/store/apps/details?id=com.estrongs.android.pop). Dùng phần mềm này vì nó có root explorer và text edit khá tiện.
Dùng Es explorer (đã bật chế độ root explorer) tìm đến /system/etc/ 
Mở file vold.fstab
Nội dung file đại khái như sau:
FIXME: Swap again?
# internal sdcard
dev_mount emmc /storage/sdcard1 auto /devices/platform/s3c-sdhci.2/­­­mmc_host/mmc2
# external sdcard
dev_mount sdcard /storage/sdcard0 1 /devices/platform/s3c-sdhci.0/­­­mmc_host/mmc0

Ở phiên bản android khác nhau thì nội dung file nhìn khác nhau. Nhưng dù thế nào thì cũng cứ đổi chỗ 2 cái cho nhau: đỏ đổi với đỏ, xanh đổi với xanh. Nó thành như sau:
FIXME: Swap again?
# internal sdcard
dev_mount sdcard /storage/sdcard0 auto /devices/platform/s3c-sdhci.2/­­­mmc_host/mmc2
# external sdcard
dev_mount emmc /storage/sdcard1 1 /devices/platform/s3c-sdhci.0/­­­mmc_host/mmc0

Save lại.
Restart lại điện thoại là sẽ thấy thẻ nhớ ngoài (chính là vùng FAT32)


Tuesday, May 7, 2013

Remote controller for walaoke

Phần mềm mới đã làm hòm hòm tên là Remote Controller For Walaoke (RC4W - client)

Phần mềm này dùng để điều khiển từ xa Walaoke (một chương trình phát karaoke trên PC)
Trong phần mềm RC4W có màn hình login liên quan đến hướng dẫn về cách làm layout của android.

Blog của phần mềm này ở đây: http://rc4w.blogspot.com/

Blog này không giới thiệu nhiều, chỉ nói qua qua thế thôi.