Bermalas-malasan dengan argumen fragmen
Saya menikmati menghapus beberapa kode boilerplate saat saya bisa. Lebih sedikit kode berarti lebih sedikit rawan kesalahan.
Hari ini, saya akan berbicara tentang meneruskan argumen ke sebuah fragmen dan cara mendapatkannya dengan aman.
Catatan: Dunia "menulis" bukan bagian dari artikel ini. Selain itu, “Navigation compose” tidak diragukan lagi merupakan kelemahan dengan argumen yang tidak diketik , sebuah backtracking yang mengarah ke perilaku seperti javascript .
Jika seperti saya, Anda menggunakan Komponen Navigasi, yang merupakan alat yang hebat, untuk bernavigasi dari satu fragmen ke fragmen lain dengan aman, mungkin Anda harus membuat instance fragmen sendiri karena beberapa alasan. Misalnya, Anda dapat menggunakan ViewPager(2) dengan FragmentStateAdapter untuk membuat instance beberapa fragmen. Dalam hal ini, NavGraph dari Navigation Component tidak ada gunanya. Ini bukan tanggung jawab perpustakaan itu untuk mengelolanya.
Komponen Navigasi hadir dengan banyak ekstensi seperti “by navArgs()” yang memungkinkan kita mendapatkan argumen dari fragmen dengan cara yang aman dan malas diketik.
Di akhir artikel ini, dengan kekuatan Kotlin, kita akan memiliki fungsi ekstensi safety lazy “by args()” untuk mendapatkan argumen yang diketik dari fragmen kita sendiri.
Tapi pertama-tama, mari kita lihat cara lama. Di artikel ini, kami akan meningkatkan kode kami secara progresif untuk membuatnya lebih aman.
Bundel
Tidak mengherankan, meneruskan data itu terbatas, Anda harus meneruskan beberapa data ke dalam satu bundel. Tidak ada cara lain untuk meneruskan data ke sebuah fragmen.
Mari kita lihat sekilas, berikut adalah fungsi bundleOf dari paket androidx.core.os
public fun bundleOf(vararg pairs: Pair<String, Any?>): Bundle = Bundle(pairs.size).apply {
for ((key, value) in pairs) {
when (value) {
null -> putString(key, null) // Any nullable type will suffice.
// Scalars
is Boolean -> putBoolean(key, value)
is Byte -> putByte(key, value)
is Char -> putChar(key, value)
is Double -> putDouble(key, value)
is Float -> putFloat(key, value)
is Int -> putInt(key, value)
is Long -> putLong(key, value)
is Short -> putShort(key, value)
// References
is Bundle -> putBundle(key, value)
is CharSequence -> putCharSequence(key, value)
is Parcelable -> putParcelable(key, value)
// Scalar arrays
is BooleanArray -> putBooleanArray(key, value)
is ByteArray -> putByteArray(key, value)
is CharArray -> putCharArray(key, value)
is DoubleArray -> putDoubleArray(key, value)
is FloatArray -> putFloatArray(key, value)
is IntArray -> putIntArray(key, value)
is LongArray -> putLongArray(key, value)
is ShortArray -> putShortArray(key, value)
// Reference arrays
is Array<*> -> {
val componentType = value::class.java.componentType!!
@Suppress("UNCHECKED_CAST") // Checked by reflection.
when {
Parcelable::class.java.isAssignableFrom(componentType) -> {
putParcelableArray(key, value as Array<Parcelable>)
}
String::class.java.isAssignableFrom(componentType) -> {
putStringArray(key, value as Array<String>)
}
CharSequence::class.java.isAssignableFrom(componentType) -> {
putCharSequenceArray(key, value as Array<CharSequence>)
}
Serializable::class.java.isAssignableFrom(componentType) -> {
putSerializable(key, value)
}
else -> {
val valueType = componentType.canonicalName
throw IllegalArgumentException(
"Illegal value array type $valueType for key \"$key\""
)
}
}
}
// Last resort. Also we must check this after Array<*> as all arrays are serializable.
is Serializable -> putSerializable(key, value)
else -> {
if (Build.VERSION.SDK_INT >= 18 && value is IBinder) {
BundleApi18ImplKt.putBinder(this, key, value)
} else if (Build.VERSION.SDK_INT >= 21 && value is Size) {
BundleApi21ImplKt.putSize(this, key, value)
} else if (Build.VERSION.SDK_INT >= 21 && value is SizeF) {
BundleApi21ImplKt.putSizeF(this, key, value)
} else {
val valueType = value.javaClass.canonicalName
throw IllegalArgumentException("Illegal value type $valueType for key \"$key\"")
}
}
}
}
}
class MyFragment : Fragment() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
val contentId: Long = requireArguments().getLong(BUNDLE_ARGUMENT_KEY_CONTENT_ID)
val otherData: String = requireArguments().getString(BUNDLE_ARGUMENT_KEY_OTHER_DATA)!!
}
companion object {
private const val BUNDLE_ARGUMENT_KEY_CONTENT_ID = "content_id"
private const val BUNDLE_ARGUMENT_KEY_OTHER_DATA = "other_data"
fun newInstance(
contentId: Long,
otherData: String
): MyFragment {
return MyFragment().apply {
arguments = Bundle().apply {
putLong(BUNDLE_ARGUMENT_KEY_CONTENT_ID, contentId)
putString(BUNDLE_ARGUMENT_KEY_OTHER_DATA, otherData)
}
}
}
// Or
fun newInstance(
contentId: Long,
otherData: String
): MyFragment {
return MyFragment().apply {
arguments = bundleOf(
BUNDLE_ARGUMENT_KEY_CONTENT_ID to contentId,
BUNDLE_ARGUMENT_KEY_OTHER_DATA to otherData
)
}
}
}
}
1°) Setiap kali kami ingin menambahkan argumen baru, kami memiliki banyak perubahan untuk mendapatkan dan mengatur yang baru.
2°) Primitif di dunia java (selain String) tidak boleh nol dalam bundel karena pembatasan java.
3°) Karena kebiasaan, Kami dapat menyalin-tempel entri dan lupa menetapkan kunci bundel yang tepat ke argumen baru yang ditambahkan.
4°) dalam fungsi newInstance, suatu hari kita dapat memutuskan untuk mengubah parameter dari tidak dapat dibatalkan menjadi dapat dibatalkan. Dalam hal ini, tidak ada kesalahan pada IDE / waktu kompilasi yang akan dilakukan. Berpotensi menghasilkan saat runtime ke pengecualian penunjuk nol jika kami mencoba untuk mendapatkan bukan nol pada objek yang dapat dibatalkan.
Kesimpulan tentang cara lama ini :
Saya membuat kode semacam ini sejak lama. Kode ini rawan kesalahan, ada banyak hal yang perlu diingat untuk menghindari kesalahan bodoh saat menambahkan/menghapus/mengubah argumen. Saat ini, sebagai developer yang baik dan dengan kekuatan Kotlin, kami ingin menjadi lebih aman dari sebelumnya, menulis lebih sedikit kode, dan bermalas-malasan.
Enkapsulasi
Nah, mengirim banyak primitif secara langsung dalam satu bundel ke argumen fragmen mungkin merupakan pilihan yang baik sejak lama karena kinerja yang buruk. Membuat objek memiliki biaya.
Mengenkapsulasi primitif/objek menjadi objek argumen unik memiliki keuntungan besar: "Pertahankan kontrak model" dan izinkan "primitif yang dapat dibatalkan".
Catatan: Komponen navigasi memungkinkan kita menulis argumen terpisah untuk sebuah fragmen dalam file xml. Ini aman karena mereka menghasilkan kelas yang dibuat secara otomatis dengan akhiran "NavigationArgs" yang menyalin argumen xml ke dalamnya. Dengan "oleh navArgs", kami mendapatkan kelas yang dibuat secara otomatis ini.
@Parcelize
data class MyFragmentArgs(
val contentId: Long,
val otherData: String
) : Parcelable
class MyFragment : Fragment() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
val args: MyFragmentArgs = requireArguments().getParcelable(BUNDLE_ARGUMENT_KEY)!!
val (contentId: Long, otherData: String) = args
}
companion object {
private const val BUNDLE_ARGUMENT_KEY = "argument"
fun newInstance(
contentId: Long,
otherData: String
): MyFragment {
return MyFragment().apply {
arguments = Bundle().apply {
putParcelable(BUNDLE_ARGUMENT_KEY, MyFragmentArgs(contentId = contentId, otherData = otherData))
}
}
}
}
}
Agar berfungsi, model perlu memperluas kelas Parcelable dan dengan anotasi "@Parcelize", ia menghindari kode boilerplate.
Selain itu, jika dalam model MyFragmentArgs kita mengubah "val data lain: String" menjadi "val data lain: String?", Android Studio akan memperingatkan kita tentang "val (contentId: Long, data lain: String) = args"
Dapat digunakan kembali
Akan keren jika kita dapat mempertahankan kontrak set/mendapatkan argumen yang sama ke fragmen mana pun untuk menghindari duplikasi.
mari kita membuatnya
@Parcelize
data class MyFragmentArgs(
val contentId: Long,
val otherData: String
) : Parcelable
class MyFragment : Fragment() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
val args: MyFragmentArgs = args()
val (contentId: Long, otherData: String) = args
}
companion object {
private fun newInstance(
contentId: Long,
otherData: String
): MyFragment {
return MyFragment().setArgs(MyFragmentArgs(contentId = contentId, otherData = otherData))
}
}
}
private const val BUNDLE_ARGUMENTS_KEY = "arguments"
fun <FRAGMENT : Fragment> FRAGMENT.setArgs(args: Parcelable): FRAGMENT {
return apply {
arguments = Bundle().apply {
putParcelable(BUNDLE_ARGUMENTS_KEY, args)
}
}
}
fun <FRAGMENT : Fragment, ARGS : Parcelable> FRAGMENT.args(): ARGS {
return requireArguments().getParcelable(BUNDLE_ARGUMENTS_KEY)!!
}
Keamanan
Akan keren jika kita bisa menautkan penyetel dan pengambil pada argumen fragmen untuk menghindari kesalahan ketik kelas yang tepat.
Ayo lakukan !
@Parcelize
data class MyFragmentArgs(
val contentId: Long,
val otherData: String
) : IArgs
class MyFragment : Fragment(),
IFragmentArgs<MyFragmentArgs> {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
val (contentId: Long, otherData: String) = args()
}
companion object {
private fun newInstance(
contentId: Long,
otherData: String
): MyFragment {
return MyFragment().setArgs(MyFragmentArgs(contentId = contentId, otherData = otherData))
}
}
}
interface IArgs : Parcelable
interface IFragmentArgs<ARGS : IArgs>
private const val BUNDLE_ARGUMENTS_KEY = "arguments"
fun <FRAGMENT, ARGS : IArgs> FRAGMENT.setArgs(args: ARGS): FRAGMENT where FRAGMENT : Fragment, FRAGMENT : IFragmentArgs<ARGS> {
return apply {
arguments = Bundle().apply {
putParcelable(BUNDLE_ARGUMENTS_KEY, args)
}
}
}
fun <FRAGMENT, ARGS : IArgs> FRAGMENT.args(): ARGS where FRAGMENT : Fragment, FRAGMENT : IFragmentArgs<ARGS> {
return requireArguments().getParcelable(BUNDLE_ARGUMENTS_KEY)!!
}
1°) Model Argumen harus memperluas antarmuka IArgs
2°) Fragmen harus memperluas antarmuka IFragmentArgs
Jadi malas
Saya kembali ke "by navArgs()" dari "Navigation Component".
Mengapa ini berguna?
Catatan: Kita tidak boleh mengubah argumen setelah pertama kali kita menyetelnya. Jadi, saat mendapatkan kembali argumen, kami dapat, dengan pengoptimalan, menyimpannya saat kami membutuhkannya pertama kali. Ini adalah tujuan dari objek "Malas" dengan sistem cache-nya.
Sekarang, misinya adalah memiliki “by args()” kita sendiri!
@Parcelize
data class MyFragmentArgs(
val contentId: Long,
val otherData: String
) : IArgs
class MyFragment : Fragment(),
IFragmentArgs<MyFragmentArgs> {
val args by args()
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
val (contentId: Long, otherData: String) = args
}
companion object {
private fun newInstance(
contentId: Long,
otherData: String
): MyFragment {
return MyFragment().setArgs(MyFragmentArgs(contentId = contentId, otherData = otherData))
}
}
}
interface IArgs : Parcelable
interface IFragmentArgs<ARGS : IArgs>
private const val BUNDLE_ARGUMENTS_KEY = "argument"
fun <FRAGMENT, ARGS : IArgs> FRAGMENT.setArgs(args: ARGS): FRAGMENT where FRAGMENT : Fragment, FRAGMENT : IFragmentArgs<ARGS> {
return apply {
arguments = Bundle().apply {
putParcelable(BUNDLE_ARGUMENTS_KEY, args)
}
}
}
@MainThread
inline fun <FRAGMENT, reified ARGS : IArgs> FRAGMENT.args(): ArgsLazy<ARGS> where FRAGMENT : Fragment, FRAGMENT : IFragmentArgs<ARGS> = ArgsLazy {
arguments ?: throw IllegalStateException("Fragment $this has null arguments")
}
class ArgsLazy<ARGS : IArgs>(
private val argumentsProducer: () -> Bundle
) : Lazy<ARGS> {
private var cached: ARGS? = null
override val value: ARGS
get() {
var args = cached
if (args == null) {
val arguments = argumentsProducer()
args = arguments.getParcelable(BUNDLE_ARGUMENTS_KEY)!!
cached = args
}
return args
}
override fun isInitialized(): Boolean = cached != null
}
Gandakan MyFragmentArgs ke MyFragmentArgs2 Dan coba panggil setArgs() dengan model baru. Android Studio akan memperingatkan Anda tentang kesalahan yang akan Anda lakukan. Kontrak ini tidak dapat dilanggar
Catatan: dalam fungsi “args()”, anotasi “@MainThread” ditentukan karena kebuntuan. Ini mencegah mengakses yang ini dari beberapa utas latar belakang. Seharusnya hanya dapat diakses dari Thread Utama.
Bonus — Argumen fragmen dari ViewModel
class MyFragmentViewModel(override val savedStateHandle: SavedStateHandle) : ViewModel(),
IViewModelArgs<MyFragmentArgs> {
val args = args()
init {
val (contentId: Long, otherData: String) = args
}
}
interface IViewModelArgs<ARGS : IArgs> {
val savedStateHandle: SavedStateHandle
}
fun <VIEWMODEL, ARGS : IArgs> VIEWMODEL.args(): ARGS where VIEWMODEL : ViewModel, VIEWMODEL : IViewModelArgs<ARGS> {
return savedStateHandle.get(BUNDLE_ARGUMENTS_KEY) as? ARGS ?: throw IllegalStateException("ViewModel $this has null arguments")
}
@Parcelize
data class MyFragmentArgs(
val contentId: Long,
val otherData: String
) : IArgs
class MyFragment : Fragment(),
IFragmentArgs<MyFragmentArgs> {
val args by args()
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
val (contentId: Long, otherData: String) = args
}
companion object {
private fun newInstance(
contentId: Long,
otherData: String
): MyFragment {
return MyFragment().setArgs(MyFragmentArgs(contentId = contentId, otherData = otherData))
}
}
}
class MyFragmentViewModel(override val savedStateHandle: SavedStateHandle) : ViewModel(),
IViewModelArgs<MyFragmentArgs> {
val args = args()
init {
val (contentId: Long, otherData: String) = args
}
}
interface IArgs : Parcelable
interface IFragmentArgs<ARGS : IArgs>
interface IViewModelArgs<ARGS : IArgs> {
val savedStateHandle: SavedStateHandle
}
private const val BUNDLE_ARGUMENTS_KEY = "arguments"
fun <FRAGMENT, ARGS : IArgs> FRAGMENT.setArgs(args: ARGS): FRAGMENT where FRAGMENT : Fragment, FRAGMENT : IFragmentArgs<ARGS> {
return apply {
arguments = Bundle().apply {
putParcelable(BUNDLE_ARGUMENTS_KEY, args)
}
}
}
@MainThread
inline fun <FRAGMENT, reified ARGS : IArgs> FRAGMENT.args(): ArgsLazy<ARGS> where FRAGMENT : Fragment, FRAGMENT : IFragmentArgs<ARGS> = ArgsLazy {
arguments ?: throw IllegalStateException("Fragment $this has null arguments")
}
class ArgsLazy<ARGS : IArgs>(
private val argumentsProducer: () -> Bundle
) : Lazy<ARGS> {
private var cached: ARGS? = null
override val value: ARGS
get() {
var args = cached
if (args == null) {
val arguments = argumentsProducer()
args = arguments.getParcelable(BUNDLE_ARGUMENTS_KEY)!!
cached = args
}
return args
}
override fun isInitialized(): Boolean = cached != null
}
fun <VIEWMODEL, ARGS : IArgs> VIEWMODEL.args(): ARGS where VIEWMODEL : ViewModel, VIEWMODEL : IViewModelArgs<ARGS> {
return savedStateHandle.get(BUNDLE_ARGUMENTS_KEY) as? ARGS ?: throw IllegalStateException("ViewModel $this has null arguments")
}
Gunakan kecanggihan Kotlin untuk menulis kode yang aman dan menulis lebih sedikit kode jika memungkinkan :D
Ini adalah artikel kedua saya. Saya suka menulis tips yang berguna.
Jika Anda menyukai artikel semacam ini, jangan ragu untuk bertepuk tangan