Com*_*are 45
既然我的Activity只是一个包含同样功能的Fragment的包装器,那么如果我用XML标记带有标签的XML片段,我如何将该包带到Fragment?
你不能.
不过,欢迎您拨打findFragmentById()您FragmentManager检索片段后的通货膨胀,然后调用该片段的一些方法将数据与相关联.虽然显然是不能setArguments(),您的片段可以安排守住数据本身过去通过一些其他方式的构造变化(onSaveInstanceState(),setRetainInstance(true),等).
Ode*_*ner 43
它不是封装的方式,但最终我从父活动中"拉出"了这个包:
Bundle bundle = getActivity().getIntent().getExtras();
Run Code Online (Sandbox Code Playgroud)
另一种选择是不在XML中声明片段.我知道这不是你想要做的.但是,您可以在视图中声明一个简单的布局,如下所示:
<LinearLayout
android:id="@+id/fragment_container"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical" />
Run Code Online (Sandbox Code Playgroud)
然后在你的Activity课堂上,你以编程方式用片段夸大布局.这样您就可以使用args传递参数.
FragmentManager fragmentManager = getSupportFragmentManager();
FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();
MyFragment fragment = MyFragment.newInstance();
Bundle args = new Bundle();
args.putInt(Global.INTENT_INT_ROLE, 1);
fragment.setArguments(args);
fragmentTransaction.add(R.id.fragment_container, fragment, "MyActivity");
fragmentTransaction.commit();
Run Code Online (Sandbox Code Playgroud)
这种方法并不像在xml中声明它那样干净简单,但我已经转移到它,因为它可以让你对片段有更多的控制.
您不能传递捆绑包(除非您以编程方式而不是通过XML来对片段进行膨胀),但是可以通过XML将参数(或属性)传递给片段。
该过程类似于您定义“视图”自定义属性的方式。除了AndroidStudio(当前)之外,该过程无法为您提供帮助。
假设这是使用参数的片段(我将使用kotlin,但它也完全适用于Java):
class MyFragment: Fragment() {
// your fragment parameter, a string
private var screenName: String? = null
override fun onAttach(context: Context?) {
super.onAttach(context)
if (screenName == null) {
screenName = arguments?.getString("screen_name")
}
}
}
Run Code Online (Sandbox Code Playgroud)
而您想做这样的事情:
<fragment
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/myFragment"
android:name="com.example.MyFragment"
app:screen_name="@string/screen_a"
android:layout_width="match_parent"
android:layout_height="wrap_content"/>
Run Code Online (Sandbox Code Playgroud)
注意 app:screen_name="@string/screen_a"
要使其工作,只需将其添加到值文件中(fragment_attrs.xml或选择所需的任何名称):
<!-- define your attribute name and type -->
<attr name="screen_name" format="string|reference"/>
<!-- define a bunch of constants you wanna use -->
<string name="screen_a" translatable="false">ScreenA</string>
<string name="screen_b" translatable="false">ScreeenB</string>
<!-- now define which arguments your fragment is gonna have (can be more then one) -->
<!-- the convention is "FragmentClassName_MembersInjector" -->
<declare-styleable name="MyFragment_MembersInjector">
<attr name="screen_name"/>
</declare-styleable>
Run Code Online (Sandbox Code Playgroud)
差不多完成了,您只需要在片段中读取它,因此添加方法:
override fun onInflate(context: Context?, attrs: AttributeSet?, savedInstanceState: Bundle?) {
super.onInflate(context, attrs, savedInstanceState)
if (context != null && attrs != null && screenName == null) {
val ta = context.obtainStyledAttributes(attrs, R.styleable.MyFragment_MembersInjector)
if (ta.hasValue(R.styleable.MyFragment_MembersInjector_screen_name)) {
screenName = ta.getString(R.styleable.MyFragment_MembersInjector_screen_name)
}
ta.recycle()
}
}
Run Code Online (Sandbox Code Playgroud)
等,片段中的XML属性:)
局限性:
Parcelable,只能传递可以定义为Android属性的内容我知道答案为时已晚,但我认为有人需要:)
只是在活动覆盖 onAttachFragment()
@Override
public void onAttachFragment(Fragment fragment)
{
super.onAttachFragment(fragment);
if (fragment.getId() == R.id.frgBlank)
{
Bundle b = new Bundle();
b.putString("msg", "Message");
fragment.setArguments(b);
}
}
Run Code Online (Sandbox Code Playgroud)
并在片段 onCreateView 方法中
Bundle b = getArguments();
if (b != null)
{
Toast.makeText(getBaseContext(), b.getString("msg"), Toast.LENGTH_SHORT).show();
}
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
22840 次 |
| 最近记录: |