将数据从一个设计选项卡传递到另一个选项卡

Posted

技术标签:

【中文标题】将数据从一个设计选项卡传递到另一个选项卡【英文标题】:Pass data from one design tab to another tab 【发布时间】:2016-10-22 22:42:05 【问题描述】:

我使用材料设计标签如下:

以下是我的 MainActivity.class :

 public class MainActivity extends AppCompatActivity implements ViewPager.OnPageChangeListener 

    private TabLayout tabLayout;
    private ViewPager viewPager;

    @Override
    protected void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        tabLayout = (TabLayout) findViewById(R.id.tabs);
        viewPager = (ViewPager) findViewById(R.id.viewpager);
        ViewPagerAdapter adapter = new ViewPagerAdapter(getSupportFragmentManager());
        viewPager.setAdapter(adapter);
        viewPager.addOnPageChangeListener(this);
        tabLayout.setupWithViewPager(viewPager);

    

    @Override
    public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) 

    

    @Override
    public void onPageSelected(int position) 
        System.out.println("onPageSelected Called");
    

    @Override
    public void onPageScrollStateChanged(int state) 

    

我的 ViewPagerAdapter 如下:

    public class ViewPagerAdapter extends FragmentPagerAdapter 

    private static int count = 2;

    public ViewPagerAdapter(FragmentManager fm) 
        super(fm);
    

    @Override
    public Fragment getItem(int position) 

        switch (position) 
            case 0:
                 return new FragmentOne();
            case 1:
                return new FragmentTwo();
        
        return null;
    

    @Override
    public int getCount() 
        return count;
    

    @Override
    public CharSequence getPageTitle(int position) 
        switch (position)
        
            case 0 :
                return "Tab One";
            case 1 :
                return "Tab Two";
        
        return null;
    

这是我的片段:

这是我的第一个片段名称是 FragmentOne :

    public class FragmentOne extends Fragment 

    private EditText editText;
    private Button btnSendData;

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) 
        System.out.println("IN FRAGMENT ONE");
        View view = inflater.inflate(R.layout.fragment_one,container,false);

        editText = (EditText) view.findViewById(R.id.et_name);
        btnSendData = (Button) view.findViewById(R.id.btn_send);

        btnSendData.setOnClickListener(new View.OnClickListener() 
            @Override
            public void onClick(View v) 

                FragmentTwo fragment = new FragmentTwo();
                Bundle bundle = new Bundle();
                bundle.putString("username",editText.getText().toString());
                fragment.setArguments(bundle);
                getFragmentManager.beginTransaction.replace(R.id.frag_second,fragment).commit();

            
        );

        return view;
    

这是另一个名为 FragmentTwo 的片段:

 public class FragmentTwo extends Fragment 

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) 
        System.out.println("IN FRAGMENT TWO");
        View view = inflater.inflate(R.layout.fragment_two,container,false);
        Bundle bundle = getArguments();
        if(bundle!= null)
        
            String value = getArguments().getString("username");
        
        return  view;
    

    @Override
    public void onResume() 
        super.onResume();
        System.out.println("onResume gets called");
    

在第二个片段中,我正在获取数据,但它在前一个视图中添加了另一个视图。 看图片:

所以我想在两种情况下将数据从 FragmentOne 传递到 FragmentTwo : 1. 我想在点击按钮时传递数据 2.当我滑动到Fragment时,应该传递两个数据

此外,当我尝试滑动到 FragmentTwo 时, FragmentTwo 中什么也没有调用?为什么会这样? 此外,当我单击第二个选项卡时,什么也没有被调用。 请帮助我如何在单击按钮时将数据传递给 FragmentTwo?

以下是布局文件: 这是fragment_one

<LinearLayout 
       xmlns:android="http://schemas.android.com/apk/res/android"
       android:layout_
       android:layout_
       android:gravity="center"
       android:orientation="vertical">
           <EditText
               android:layout_marginLeft="16dp"
               android:layout_marginRight="16dp"
               android:id="@+id/et_name"
               android:hint="Username"

               android:layout_
               android:layout_/>

           <Button
                android:padding="16dp"
                android:text="Send Data"
                android:layout_marginLeft="16dp"
                android:layout_marginRight="16dp"
                android:id="@+id/btn_send"
                android:layout_
                android:layout_/>

第二个片段,fragment_second.xml:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical" android:layout_
android:layout_
android:id="@+id/second_frag"
android:gravity="center">
<TextView
    android:textSize="24sp"
    android:layout_
    android:layout_
    android:textAppearance="?android:attr/textAppearanceLarge"
    android:text="Fragment Two"
    android:id="@+id/textView2" />

【问题讨论】:

【参考方案1】:

每个片段都与父活动相关联。所以你不能直接从一个片段到另一个片段进行通信。您将需要使用界面通过父活动。

查看此文档:https://developer.android.com/training/basics/fragments/communicating.html

单击按钮时,将值传递给自定义界面中的方法,并在第二个片段中访问这些方法。

当我尝试滑动到 FragmentTwo 时,FragmentTwo 中没有调用任何内容

为此,您需要实现片段生命周期 - https://developer.android.com/reference/android/app/Fragment.html

更新

对您的代码进行了一些修改,看看下面的内容。代码 -

Manifex.xml

<application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".MainActivity"
            android:theme="@style/Theme.AppCompat.NoActionBar">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

MainActivity.java

package com.app.onkar.tabdemo;

import android.support.v4.app.Fragment;
import android.support.design.widget.TabLayout;

import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentPagerAdapter;
import android.support.v4.view.ViewPager;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.support.v7.widget.Toolbar;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends AppCompatActivity implements ViewPager.OnPageChangeListener 

    private Toolbar toolbar;
    private TabLayout tabLayout;
    private ViewPager viewPager;

    @Override
    protected void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);

        viewPager = (ViewPager) findViewById(R.id.viewpager);
        tabLayout = (TabLayout) findViewById(R.id.tabs);


        ViewPagerAdapter adapter = new ViewPagerAdapter(getSupportFragmentManager());

        adapter.addFragment(new FragmentOne(), "ONE");
        adapter.addFragment(new FragmentTwo(), "TWO");
        viewPager.setAdapter(adapter);

        viewPager.addOnPageChangeListener(this);
        tabLayout.setupWithViewPager(viewPager);
    

    @Override
    public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) 

    

    @Override
    public void onPageSelected(int position) 
        System.out.println("onPageSelected Called");
    

    @Override
    public void onPageScrollStateChanged(int state) 

    

    public static class ViewPagerAdapter extends FragmentPagerAdapter 
        private final List<Fragment> mFragmentList = new ArrayList<>();
        private final List<String> mFragmentTitleList = new ArrayList<>();
        private static int count = 2;

        public void addFragment(Fragment fragment, String title) 
            mFragmentList.add(fragment);
            mFragmentTitleList.add(title);
        

        public ViewPagerAdapter(FragmentManager fm) 
            super(fm);
        

        @Override
        public Fragment getItem(int position) 

            switch (position) 
                case 0:
                    return new FragmentOne();
                case 1:
                    return new FragmentTwo();
            
            return null;
        

        @Override
        public int getCount() 
            return count;
        

        @Override
        public CharSequence getPageTitle(int position) 
            switch (position) 
                case 0:
                    return "Tab One";
                case 1:
                    return "Tab Two";
            
            return null;
        
    

FragmentOne.java

package com.app.onkar.tabdemo;

import android.content.Context;
import android.net.Uri;
import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.EditText;

public class FragmentOne extends Fragment 
    private EditText editText;
    private Button btnSendData;

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) 
        System.out.println("IN FRAGMENT ONE");
        View view = inflater.inflate(R.layout.fragment_one,container,false);

        editText = (EditText) view.findViewById(R.id.et_name);
        btnSendData = (Button) view.findViewById(R.id.btn_send);

        btnSendData.setOnClickListener(new View.OnClickListener() 
            @Override
            public void onClick(View v) 

                FragmentTwo fragment = new FragmentTwo();
                Bundle bundle = new Bundle();
                bundle.putString("username",editText.getText().toString());
                fragment.setArguments(bundle);
                getActivity().getSupportFragmentManager().beginTransaction().replace(R.id.second_frag,fragment).commit();
            
        );

        return view;
    

FragmentTwo.java

package com.app.onkar.tabdemo;


import android.support.v4.app.Fragment;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;


public class FragmentTwo extends Fragment 

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) 
        System.out.println("IN FRAGMENT TWO");
        View view = inflater.inflate(R.layout.fragment_two,container,false);
        TextView txt2 = (TextView) view.findViewById(R.id.textView2);
        Bundle bundle = getArguments();
        if(bundle!= null)
        
            String value = getArguments().getString("username");
            txt2.setText(value);
        
        return  view;
    

    @Override
    public void onResume() 
        super.onResume();
        System.out.println("onResume gets called");
    


布局文件没有变化。只需尝试上面的代码 - 它完全按照您的意愿工作。希望它会有所帮助!

屏幕

【讨论】:

您能否提供代码,因为我是 Android 编程新手,我阅读了链接,但我没有得到它。可以加进去代码吗?? 检查以下链接 - 1) ***.com/questions/7149802/… 2) ***.com/questions/16036572/… 我希望它现在可以工作,如果不只是让我知道的话。我会给你示例代码。 我已经编辑了我的代码,请查看我更新的代码并查看图片,以便您了解。 好的,现在您正在第二个片段中获取数据,但问题是 - 来自第一个片段的数据也添加到了第二个片段的视图中,对吗? 不,我在第二个片段中已经有 3 个文本视图,当我单击第一个片段中的按钮时,我得到 6 个文本视图。你能运行这段代码就知道到底发生了什么吗?【参考方案2】:

ViewPagerAdapter 类

public class ViewPagerAdapter extends FragmentPagerAdapter 

private static int count = 2;


public ViewPagerAdapter(FragmentManager fm) 
super(fm);


@Override
public Fragment getItem(int position) 

switch (position) 
    case 0:
         Fragment fragmentone = new FragmentOne();
         Bundle args = new Bundle();
         args.putInt("code", 1);
         fragmentone.setArguments(args);
         return fragmentone;
    break;
    case 1:
        Fragment fragmenttwo = new FragmentTwo();
         Bundle args = new Bundle();
         args.putInt("code", 2);
         fragmenttwo.setArguments(args);
         return fragmenttwo ;
      break;

return null;


@Override
public int getCount() 
return count;


@Override
public CharSequence getPageTitle(int position) 
switch (position)

    case 0 :
        return "Tab One";
    case 1 :
        return "Tab Two";

return null;


args是bundle对象,可以在Fragment的onCreateView方法下面的代码中放入String int等值来使用这些值。

int codeForthisFragment = getArguments().getInt("code");

对于按钮点击: FragmentTransaction ft = getFragmentManager().beginTransaction(); 片段 secondFragment = new FragmentTwo();

                    Bundle args = new Bundle();
                    args.putInt("code", 2);
                    secondFragment.setArguments(args);
                    ft.replace(R.id.content_frame, secondFragment);
                    ft.commit();

【讨论】:

但是如果我在 FragmentOne 中单击我的按钮,您在适配器中设置数据会起作用吗? 我已经完成了你所说的一切,但它在第二个片段中添加了双重视图。 @swanandvaidya 转到您的 XML 文件并在您的片段元素中将背景颜色设置为白色或您想要的任何颜色 @swanandvaidya 请对对您有帮助的答案投票,如果答案正确,请选择接受 我的问题还没有解决,所以我不会投票。【参考方案3】:

在第二个片段中,我正在获取数据,但它在前一个视图中添加了另一个视图。看图

参考@Roy Miller(https://***.com/users/5255021/roy-miller) 的回答, 您可以通过以下更改来解决它:

MainActivity.java中,替换下面的开关:

 @Override
    public Fragment getItem(int position) 

        switch (position) 
            case 0:
                return new FragmentOne();
            case 1:
                return new FragmentTwo();
        
        return null;
    

与:

@Override
public Fragment getItem(int position) 

    switch (position)
        case 0:
            return mFragmentList.get(0);
        case 1:
            return mFragmentList.get(1);
        case 2:
            return mFragmentList.get(2);
        default:
            return null;
    


然后,在 FragmentOne.java

替换这个:

btnSendData.setOnClickListener(new View.OnClickListener() 
        @Override
        public void onClick(View v) 

            FragmentTwo fragment = new FragmentTwo();
            Bundle bundle = new Bundle();
            bundle.putString("username",editText.getText().toString());
            fragment.setArguments(bundle);
            getActivity().getSupportFragmentManager().beginTransaction().replace(R.id.second_frag,fragment).commit();
        
    );

与:

btnSendData.setOnClickListener(new View.OnClickListener() 
        @Override
        public void onClick(View v) 

/*This one FragmentCollection is a Singleton class that contains list of fragments, set it during initialization itself(code is attached below this code)*/
FragmentCollection fragmentCollection=FragmentCollection.getInstance();

            FragmentTwo fragment = (FragmentTwo) FragmentCollection.getmFragmentList().get(2); //getting the second fragment

             fragment.setText(editText.getText().toString());  //add this method in fragment two             

        
    );

现在在 FragmentTwo.java 添加方法:

public void updateTextView(String data)
    Log.d("TabTwo","Update Text view");
    TextView txt = (TextView)view.findViewById(R.id.textView2);
    txt.setText(data);

最后,我上面提到的单例类:

public class FragmentCollection 
static FragmentCollection fragmentCollection=new FragmentCollection();
private final List<Fragment> mFragmentList = new ArrayList<>();    
private FragmentCollection()
public static FragmentCollection getInstance()
    if(fragmentCollection!=null) 
        return fragmentCollection;
    else 
        return new FragmentCollection();
    

public void setmFragmentList(List<Fragment> mFragmentList)
    this.mFragmentList.addAll(mFragmentList);

public List<Fragment> getmFragmentList()
    return mFragmentList;


让旧视图重叠的问题是因为一次又一次地初始化片段

like new FragmentTwo();不止一处。

所以这里我们把它放在一个列表中并访问它,所以只有一个片段被改变。

【讨论】:

【参考方案4】:

我无法从第二个片段转到第一个片段。我正在使用选项卡布局 这是我的代码 通过单击第二个片段中的按钮

            Bundle bundle = new Bundle();
            bundle.putString("uniqueList1", uniqueList.get(0));
            bundle.putString("uniqueList2", uniqueList.get(1));
            bundle.putString("kycUniquesNo", kycUniqueNum);
            CustBasicFormFragement fragObj = new CustBasicFormFragement();
            fragObj.setArguments(bundle);

在第一个片段中

   @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle 
     savedInstanceState) 
    super.onCreateView(inflater, container, savedInstanceState);
    binding = DataBindingUtil.inflate(inflater, R.layout.cust_basic_form_fragment, container, false);
    view = binding.getRoot();

    if (getArguments() != null) 
         string1 = getArguments().getString("uniqueList1");
         string2 = getArguments().getString("uniqueList2");
         kycUnique=getArguments().getString("kycUniquesNo");
        System.out.print(string1);
        System.out.print(string2);
        System.out.print(kycUnique);
    return view;


【讨论】:

请告诉我我必须做什么?任何帮助将不胜感激,谢谢

以上是关于将数据从一个设计选项卡传递到另一个选项卡的主要内容,如果未能解决你的问题,请参考以下文章

在特定条件下将数据从一个选项卡自动填充到另一个选项卡

将 UIView 从一个选项卡移动到另一个选项卡

将行从一个选项卡复制并删除到另一个选项卡

将应用程序 URL 从一个选项卡复制到另一个选项卡导致 404 错误

当我从一个选项卡单击到另一个选项卡时,如何使TabLayout中的选项卡不可滚动

如何使用选项卡导航器将道具传递给 Screen 组件?