हालाँकि, मैंने डेटा बाइंडिंग पर भरोसा करते हुए कुछ अच्छे उत्तर दिए हैं, लेकिन मैंने एक्सएमएल में खंड-मुक्त लेआउट परिभाषाओं की अनुमति देते हुए टुकड़े के संकल्प को सक्षम करने के अर्थ में - उस दृष्टिकोण के साथ पूर्ण सीमा तक नहीं देखा।
बंधन मानते हुए डेटा तो है सक्षम है, तो यहाँ एक सामान्य समाधान मेरा प्रस्ताव कर सकता है; थोड़ा लंबा लेकिन यह निश्चित रूप से काम करता है (कुछ केवेट के साथ):
चरण 1: कस्टम ऑनक्लिक कार्यान्वयन
यह टेप-ऑन व्यू (उदाहरण के लिए बटन) से जुड़े संदर्भों के माध्यम से एक खंड-जागरूक खोज चलाएगा:
// CustomOnClick.kt
@file:JvmName("CustomOnClick")
package com.example
import android.app.Activity
import android.content.Context
import android.content.ContextWrapper
import android.view.View
import androidx.fragment.app.Fragment
import androidx.fragment.app.FragmentActivity
import java.lang.reflect.Method
fun onClick(view: View, methodName: String) {
resolveOnClickInvocation(view, methodName)?.invoke(view)
}
private data class OnClickInvocation(val obj: Any, val method: Method) {
fun invoke(view: View) {
method.invoke(obj, view)
}
}
private fun resolveOnClickInvocation(view: View, methodName: String): OnClickInvocation? =
searchContexts(view) { context ->
var invocation: OnClickInvocation? = null
if (context is Activity) {
val activity = context as? FragmentActivity
?: throw IllegalStateException("A non-FragmentActivity is not supported (looking up an onClick handler of $view)")
invocation = getTopFragment(activity)?.let { fragment ->
resolveInvocation(fragment, methodName)
}?: resolveInvocation(context, methodName)
}
invocation
}
private fun getTopFragment(activity: FragmentActivity): Fragment? {
val fragments = activity.supportFragmentManager.fragments
return if (fragments.isEmpty()) null else fragments.last()
}
private fun resolveInvocation(target: Any, methodName: String): OnClickInvocation? =
try {
val method = target.javaClass.getMethod(methodName, View::class.java)
OnClickInvocation(target, method)
} catch (e: NoSuchMethodException) {
null
}
private fun <T: Any> searchContexts(view: View, matcher: (context: Context) -> T?): T? {
var context = view.context
while (context != null && context is ContextWrapper) {
val result = matcher(context)
if (result == null) {
context = context.baseContext
} else {
return result
}
}
return null
}
नोट: शिथिल मूल एंड्रॉयड कार्यान्वयन पर आधारित है (देखें https://android.googlesource.com/platform/frameworks/base/+/a175a5b/core/java/android/view/View.java#3025 )
चरण 2: लेआउट फ़ाइलों में घोषणात्मक अनुप्रयोग
फिर, एक्सएमएल के डेटा बाध्यकारी बंधन में:
<layout>
<data>
<import type="com.example.CustomOnClick"/>
</data>
<Button
android:onClick='@{(v) -> CustomOnClick.onClick(v, "myClickMethod")}'
</Button>
</layout>
चेतावनियां
- 'आधुनिक'
FragmentActivity
आधारित कार्यान्वयन मानता है
- स्टैक में केवल "टॉप-मोस्ट" (यानी अंतिम ) टुकड़े की विधि खोज सकते हैं (हालांकि यह तय किया जा सकता है, अगर जरूरत हो तो)