フラグメント間で値を渡す方法


109

Fragmentsを使うのは初めてです。

私はフラグメントを使用する簡単なサンプルアプリケーションを構築しようとしています。私のシナリオは、2つのアクティビティがあり、各アクティビティ内に1つのフラグメントがあります。最初のフラグメントにはエディットテキストとボタンがあります。2番目のフラグメントにはテキストビューがあります。edittextに名前を入力してボタンをクリックすると、2番目のフラグメントのテキストビューには、最初のフラグメントのedittextに入力された名前が表示されます。

最初のフラグメントからそのアクティビティに、次にそのアクティビティから2番目のアクティビティに値を送信することができました。次に、2番目のフラグメント内でこの値をどのように使用しますか?

これがJavaコードです:::

package com.example.fragmentexample;

import android.app.Activity;
import android.app.Fragment;
import android.app.FragmentManager;
import android.content.Intent;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;

public class Fragment_1 extends Fragment{

    OnFragmentChangedListener mCallback;

    // Container Activity must implement this interface
    public interface OnFragmentChangedListener {
        public void onButtonClicked(String name);
    }

    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);

        // This makes sure that the container activity has implemented
        // the callback interface. If not, it throws an exception
        try {
            mCallback = (OnFragmentChangedListener) activity;
        } catch (ClassCastException e) {
            throw new ClassCastException(activity.toString()
                    + " must implement OnHeadlineSelectedListener");
        }
    }


    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        // TODO Auto-generated method stub      

        View view = inflater.inflate(R.layout.fragment_fragment_1, container, false);

        final EditText edtxtPersonName_Fragment = (EditText) view.findViewById(R.id.edtxtPersonName);
        Button btnSayHi_Fragment = (Button) view.findViewById(R.id.btnSayHi);

        btnSayHi_Fragment.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View v) {
                // TODO Auto-generated method stub

                String name = edtxtPersonName_Fragment.getText().toString();

                FragmentManager fm = getFragmentManager();
                Fragment_2 f2 = (Fragment_2) fm.findFragmentById(R.id.fragment_content_2);

                Activity activity = getActivity();

                if(activity != null)
                {
                    Toast.makeText(activity, "Say&ing Hi in Progress...", Toast.LENGTH_LONG).show();
                }


                if(f2 != null && f2.isInLayout())
                {
                    f2.setName(name);
                }
                else
                {
                    mCallback.onButtonClicked(name);
                }
            }
        });

        return view;


    }

}

MainActivity.Java

package com.example.fragmentexample;

import android.os.Bundle;
import android.app.Activity;
import android.content.Intent;

import android.view.Choreographer.FrameCallback;
import android.view.Menu;

public class MainActivity extends Activity implements Fragment_1.OnFragmentChangedListener {

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

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.activity_main, menu);
        return true;
    }

    @Override
    public void onButtonClicked(String name) {
        // TODO Auto-generated method stub

        Intent i = new Intent(this, SecondActivity.class);
        i.putExtra("", name);
        startActivity(i);
    }

}

SecondActivity.Java

package com.example.fragmentexample;

import android.app.Activity;
import android.os.Bundle;

public class SecondActivity extends Activity{

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        // TODO Auto-generated method stub
        super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_second);

        Bundle b = getIntent().getExtras();

        Fragment_2 f2 = new Fragment_2();
        f2.setArguments(b);
    }
}

Fragment_2.Java

package com.example.fragmentexample;

import android.app.Activity;
import android.app.Fragment;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;

public class Fragment_2 extends Fragment{

    View view;
    TextView txtName;


    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        // TODO Auto-generated method stub

        view = inflater.inflate(R.layout.fragment_fragment_2, container, false);

            // Exception at this line
        String name = getArguments().getString("message");
        txtName = (TextView) view.findViewById(R.id.txtViewResult);
        txtName.setText(name);

        return view;
    }

    @Override
    public void onAttach(Activity activity) {
        // TODO Auto-generated method stub
        super.onAttach(activity);       
    }

    public void setName(String name)
    {   
        txtName.setText("Hi " + name);
    }

}

次の例外が発生します:::

04-16 18:10:24.573: E/AndroidRuntime(713): FATAL EXCEPTION: main
04-16 18:10:24.573: E/AndroidRuntime(713): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example.fragmentexample/com.example.fragmentexample.SecondActivity}: android.view.InflateException: Binary XML file line #8: Error inflating class fragment
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:1815)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:1831)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.app.ActivityThread.access$500(ActivityThread.java:122)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1024)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.os.Handler.dispatchMessage(Handler.java:99)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.os.Looper.loop(Looper.java:132)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.app.ActivityThread.main(ActivityThread.java:4123)
04-16 18:10:24.573: E/AndroidRuntime(713):  at java.lang.reflect.Method.invokeNative(Native Method)
04-16 18:10:24.573: E/AndroidRuntime(713):  at java.lang.reflect.Method.invoke(Method.java:491)
04-16 18:10:24.573: E/AndroidRuntime(713):  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:841)
04-16 18:10:24.573: E/AndroidRuntime(713):  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:599)
04-16 18:10:24.573: E/AndroidRuntime(713):  at dalvik.system.NativeStart.main(Native Method)
04-16 18:10:24.573: E/AndroidRuntime(713): Caused by: android.view.InflateException: Binary XML file line #8: Error inflating class fragment
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:688)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.view.LayoutInflater.rInflate(LayoutInflater.java:724)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.view.LayoutInflater.inflate(LayoutInflater.java:479)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.view.LayoutInflater.inflate(LayoutInflater.java:391)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.view.LayoutInflater.inflate(LayoutInflater.java:347)
04-16 18:10:24.573: E/AndroidRuntime(713):  at com.android.internal.policy.impl.PhoneWindow.setContentView(PhoneWindow.java:223)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.app.Activity.setContentView(Activity.java:1786)
04-16 18:10:24.573: E/AndroidRuntime(713):  at com.example.fragmentexample.SecondActivity.onCreate(SecondActivity.java:13)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.app.Activity.performCreate(Activity.java:4397)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1048)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:1779)
04-16 18:10:24.573: E/AndroidRuntime(713):  ... 11 more
04-16 18:10:24.573: E/AndroidRuntime(713): Caused by: java.lang.NullPointerException
04-16 18:10:24.573: E/AndroidRuntime(713):  at com.example.fragmentexample.Fragment_2.onCreateView(Fragment_2.java:24)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.app.FragmentManagerImpl.moveToState(FragmentManager.java:754)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.app.FragmentManagerImpl.moveToState(FragmentManager.java:956)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.app.FragmentManagerImpl.addFragment(FragmentManager.java:1035)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.app.Activity.onCreateView(Activity.java:4177)
04-16 18:10:24.573: E/AndroidRuntime(713):  at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:664)
04-16 18:10:24.573: E/AndroidRuntime(713):  ... 21 more

SecondActivity.javaのバンドルからFragment_2.Javaに値を取得するにはどうすればよいですか?


問題が発生している部分をハイライトしてください
Nikhil Agrawal 2013

簡単な方法で動作するソリューション: stackoverflow.com/a/59332751/10201722
Adil Siddiqui

回答:


203

ステップ1。フラグメントからアクティビティにデータを送信する

Intent intent = new Intent(getActivity().getBaseContext(),
                        TargetActivity.class);
                intent.putExtra("message", message);
                getActivity().startActivity(intent);

ステップ2。アクティビティでこのデータを受信するには:

Intent intent = getIntent();
String message = intent.getStringExtra("message");

ステップ3。アクティビティから別のアクティビティにデータを送信するには、通常のアプローチに従います

Intent intent = new Intent(MainActivity.this,
                        TargetActivity.class);
                intent.putExtra("message", message);
                startActivity(intent);

アクティビティでこのデータを受信するためのステップ4

     Intent intent = getIntent();
  String message = intent.getStringExtra("message");

ステップ5.アクティビティから、次のような目的でデータをフラグメントに送信できます。

Bundle bundle=new Bundle();
bundle.putString("message", "From Activity");
  //set Fragmentclass Arguments
Fragmentclass fragobj=new Fragmentclass();
fragobj.setArguments(bundle);

また、フラグメントonCreateViewメソッドでフラグメントを受け取るには、次のようにします。

@Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
    Bundle savedInstanceState) {
          String strtext=getArguments().getString("message");

    return inflater.inflate(R.layout.fragment, container, false);
    }

2
ステップ5を試しました。フラグメントからlistactivityを呼び出しています。そのアクティビティからアイテムを選択した後、選択したファイル名でフラグメントに戻りたいと思います。しかし、機能しません。onCreateViewでNullpointer Exceptionが発生します。これに関する解決策はありますか?これは私の質問ですstackoverflow.com/questions/18208771/...
OnkarDhane

ここにPOJOを配置するにはどうすればよいですか?bundle.putString("message", "From Activity");
jeet.chanchawat 2017年

31
誰がどのようにこのような何かをやって、回旋でうんざりされている
エド・リー

1
まさにエディ、これは正しい方法ではありません。コンテナアクティビティを介してフラグメント間で通信するには、インターフェースを使用する必要があります。
Kaveesh Kanwal 2017

断片間で通信するためにインタフェースを使用に関する:developer.android.com/training/basics/fragments/...
アンドレLトーレス

49

開発者サイトで述べたように

多くの場合、たとえばユーザーイベントに基づいてコンテンツを変更するために、1つのフラグメントが別のフラグメントと通信する必要があります。すべてのフラグメントからフラグメントへの通信は、関連付けられたアクティビティを介して行われます。2つのフラグメントが直接通信することはありません。

フラグメント間の通信は、関連付けられたアクティビティを介して行う必要があります。

次のコンポーネントを用意しましょう。

アクティビティはフラグメントをホストし、フラグメント通信を許可します

FragmentAデータを送信する最初のフラグメント

FragmentBからDATASを受信する第2のフラグメントFragmentA

FragmentAの実装は次のとおりです。

public class FragmentA extends Fragment 
{
    DataPassListener mCallback;
    
    public interface DataPassListener{
        public void passData(String data);
    }

    @Override
    public void onAttach(Context context) 
    {
        super.onAttach(context);
        // This makes sure that the host activity has implemented the callback interface
        // If not, it throws an exception
        try 
        {
            mCallback = (OnImageClickListener) context;
        }
        catch (ClassCastException e) 
        {
            throw new ClassCastException(context.toString()+ " must implement OnImageClickListener");
        }
    }
    
    public View onCreateView(LayoutInflater inflater, ViewGroup container,Bundle savedInstanceState) 
    {
        // Suppose that when a button clicked second FragmentB will be inflated
        // some data on FragmentA will pass FragmentB
        // Button passDataButton = (Button).........
        
        passDataButton.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                if (view.getId() == R.id.passDataButton) {
                    mCallback.passData("Text to pass FragmentB");
                }
            }
        });
    }
}

MainActivityの実装は次のとおりです。

public class MainActivity extends ActionBarActivity implements DataPassListener{
    
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        if (findViewById(R.id.container) != null) {
            if (savedInstanceState != null) {
                return;
            }
            getFragmentManager().beginTransaction()
                    .add(R.id.container, new FragmentA()).commit();
        }
    }
    
    @Override
    public void passData(String data) {
        FragmentB fragmentB = new FragmentB ();
        Bundle args = new Bundle();
        args.putString(FragmentB.DATA_RECEIVE, data);
        fragmentB .setArguments(args);
        getFragmentManager().beginTransaction()
            .replace(R.id.container, fragmentB )
            .commit();
    }
}

FragmentBの実装は次のとおりです。

public class FragmentB extends Fragment{
    final static String DATA_RECEIVE = "data_receive";
    TextView showReceivedData;
    
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.fragment_B, container, false);
        showReceivedData = (TextView) view.findViewById(R.id.showReceivedData);
    }
    
    @Override
    public void onStart() {
        super.onStart();
        Bundle args = getArguments();
        if (args != null) {
            showReceivedData.setText(args.getString(DATA_RECEIVE));
        }
    }
}

これがお役に立てば幸いです。


4
ここでOnImageClickListenerとは何ですか?そしてそれはどのようにDataPassListener変数にキャストされますか?
Harsha 2018

「DataPassListener」であることがわかりました。私はそれを動作させましたが、ボタンがクリックされるたびにデータが送信されるため、見つけるだけです。送信されると、フラグメントが置き換えられます。つまり、リストは毎回更新されます。したがって、アイテムのリストをメモリに保持しません。
ウェズリーフランク

44

// Fragment_1.java内

Bundle bundle = new Bundle();
bundle.putString("key","abc"); // Put anything what you want

Fragment_2 fragment2 = new Fragment_2();
fragment2.setArguments(bundle);

getFragmentManager()
      .beginTransaction()
      .replace(R.id.content, fragment2)
      .commit();

// Fragment_2.java内

Bundle bundle = this.getArguments();

if(bundle != null){
     // handle your code here.
}

これがお役に立てば幸いです。


1
重要なポイントを示すための明確な答え。putString(またはputInt)を使用することが答えです!
Mehdi Dehghani

22

開発者のウェブサイトから:

多くの場合、たとえばユーザーイベントに基づいてコンテンツを変更するために、1つのフラグメントが別のフラグメントと通信する必要があります。すべてのフラグメントからフラグメントへの通信は、関連付けられたアクティビティを介して行われます。2つのフラグメントが直接通信することはありません。

アクティビティを使用して、フラグメント間で通信できます。このアプローチを使用して、アクティビティとフラグメント間で通信できます。

こちらのリンクもご確認ください。


あなたが与えた2番目のリンクは、それらが同じアクティビティにあるときにフラグメント間で通信する方法を示しています。両方が異なるアクティビティにいる場合、どのように通信しますか?
Vamsi Challa

5

まず、すべての答えが正しいですIntent。カスタムオブジェクトを除くデータをを使用して渡すことができます。カスタムオブジェクトを渡す場合は、実装するSerialazableParcelable、カスタムオブジェクトクラスに渡す必要があります。複雑すぎると思いました...

したがって、プロジェクトが単純な場合は、を使用してみてくださいDataCache。それはデータを渡すための非常にシンプルな方法を提供します。 参照: GithubプロジェクトCachePot

1-データを送信するビューまたはアクティビティまたはフラグメントに設定します

DataCache.getInstance().push(obj);

2-以下のような場所でデータを取得します

public class MainFragment extends Fragment
{
    private YourObject obj;

    @Override
    public void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);

        obj = DataCache.getInstance().pop(YourObject.class);

    }//end onCreate()
}//end class MainFragment

内部に静的な値を持つ追加のクラスを使用する方が簡単ではないでしょうか?これらの値には両方のフラグメントからアクセスできるため、追加のライブラリを使用してカスタムオブジェクトをキャッシュする必要はありません。
Neph

4

フラグメント間でデータを渡すための最新のソリューションは、ViewModelやLiveDataなどのAndroidアーキテクチャコンポーネントを使用して実装できます。このソリューションを使用すると、通信用のインターフェースを定義する必要がなく、構成変更によるデータの生存などのビューモデルを使用する利点を得ることができます。

このソリューションでは、通信に関与するフラグメントは、アクティビティのライフサイクルに関連付けられている同じviewmodelオブジェクトを共有します。ビューモデルオブジェクトにはlivedataオブジェクトが含まれています。1つのフラグメントはlivedataオブジェクトに渡されるデータを設定し、2番目のフラグメントはlivedataの変更を監視してデータを受信します。

ここに完全な例がありますhttp://www.zoftino.com/passing-data-between-android-fragments-using-viewmodel


3

フラグメント間で引数を渡す。 この質問に答えるのはかなり遅いですが、誰かを助けることができます! Fragment_1.java

Bundle i = new Bundle(); 
            i.putString("name", "Emmanuel");

            Fragment_1 frag = new Fragment_1();
            frag.setArguments(i);
            FragmentManager fragmentManager = getFragmentManager();
            fragmentManager.beginTransaction()
                    .replace(R.id.content_frame
                            , new Fragment_2())
                    .commit();

その後、あなたの中でFragment_2.javaあなたは通常あなたのonActivityCreated eg 内でparamatersを得ることができます

 Intent intent = getActivity().getIntent();
    if (intent.getExtras() != null) {
        String name =intent.getStringExtra("name");
    }

3

この問題を解決する良い方法は、カスタムインターフェイスを使用することだと思います。

同じアクティビティ内にある2つのフラグメント(AとB)があり、AからBにデータを送信するとします。

インターフェース :

public interface OnDataSentListener{
    void onDataSent(Object data);
}

アクティビティ:

    public class MyActivity extends AppCompatActivity{

            private OnDataSentListener onDataSentListener;

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

                FragmentTransaction trans = getFragmentManager().beginTransaction();

                FragmentA fa = new FragmentA();
                FragmentB fb = new FragmentB();

               fa.setOnDataSentListener(new Listeners.OnDataSentListener() {
                    @Override
                    public void onDataSent(Object data) {
                         if(onDataSentListener != null)  onDataSentListener.onDataSent(data);
                    }
               });

                transaction.add(R.id.frame_a, fa);
                transaction.add(R.id.frame_b, fb);

                transaction.commit();
            }

            public void setOnDataSentListener(OnDataSentListener listener){
                this.onDataSentListener = listener;
            }
        }

フラグメントA:

public class FragmentA extends Fragment{

    private OnDataSentListener onDataSentListener;

    private void sendDataToFragmentB(Object data){
        if(onDataSentListener != null) onDataSentListener.onDataSent(data);
    }

    public void setOnDataSentListener(OnDataSentListener listener){
        this.onDataSentListener = listener;
    }
}

フラグメントB:

public class FragmentB extends Fragment{

    private void initReceiver(){
        ((MyActivity) getActivity()).setOnDataSentListener(new OnDataSentListener() {
            @Override
            public void onDataSent(Object data) {
                //Here you receive the data from fragment A
            }
        });
    }
}

2

フラグメント間の通信はかなり複雑です(リスナーの概念を実装するのは少し難しいと思います)。

これらの通信を抽象化するには、「イベントバス」を使用するのが一般的です。これは、この通信を処理するサードパーティのライブラリです。

「オットー」はこれを行うために頻繁に使用されるものであり、調べる価値があるかもしれません:http : //square.github.io/otto/


2

フラグメントから別のフラグメントにデータを渡す

  • 最初のフラグメントから

    // Set data to pass
    MyFragment fragment = new MyFragment(); //Your Fragment
    Bundle bundle = new Bundle();
    bundle.putInt("year", 2017)  // Key, value
    fragment.setArguments(bundle); 
    // Pass data to other Fragment
    getFragmentManager()
     .beginTransaction()
     .replace(R.id.content, fragment)
     .commit(); 
  • 2番目のフラグメントについて

    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
         super.onCreate(savedInstanceState);
    
         Bundle bundle = this.getArguments();
         if (bundle != null) {
             Int receivedYear = bundle.getInt("year", ""); // Key, default value
         } 
    }

1

この単純な実装により、フラグメント間でデータを簡単な方法で渡すことができます。「Frgment1」から「Fragment2」にデータを渡したいと思う

Fragment1(送信するデータを設定)

 Bundle bundle = new Bundle();
 bundle.putString("key","Jhon Doe"); // set your parameteres

 Fragment2 nextFragment = new Fragment2();
 nextFragment.setArguments(bundle);

 FragmentManager fragmentManager = getActivity().getSupportFragmentManager();
 fragmentManager.beginTransaction().replace(R.id.content_drawer, nextFragment).commit();

Fragment2のonCreateViewメソッド(パラメーターを取得)

String value = this.getArguments().getString("key");//get your parameters
Toast.makeText(getActivity(), value+" ", Toast.LENGTH_LONG).show();//show data in tost

私のアプリでは、これを機能させようとしていますが、値が渡されていますが、フラグメント2に表示されていませんが、R.id.content_drawerの設定に誤りがあると思います。誰かが.xmlの例でR.id.content_drawerの正しい識別子を教えてくれますか?
ファルコン

1

コトリンウェイ

SharedViewModel公式のViewModel ドキュメントで提案されたものを使用する

アクティビティ内の2つ以上のフラグメントが互いに通信する必要があることは非常に一般的です。ユーザーがリストからアイテムを選択するフラグメントと、選択したアイテムのコンテンツを表示する別のフラグメントがあるマスター/ディテールフラグメントの一般的なケースを想像してください。両方のフラグメントがいくつかのインターフェース記述を定義する必要があり、所有者アクティビティーが2つを一緒にバインドする必要があるため、このケースは決して些細なことではありません。さらに、両方のフラグメントは、他のフラグメントがまだ作成されていないか、表示されていないシナリオを処理する必要があります。

この一般的な問題点は、ViewModelオブジェクトを使用することで対処できます。これらのフラグメントは、アクティビティスコープを使用してViewModelを共有し、この通信を処理できます

まずフラグメント-ktxを実装して、ビューモデルをより簡単にインスタンス化します

dependencies {
    implementation "androidx.fragment:fragment-ktx:1.2.2"
} 

次に、他のフラグメントと共有するデータをビューモデル内に配置するだけです

class SharedViewModel : ViewModel() {
    val selected = MutableLiveData<Item>()

    fun select(item: Item) {
        selected.value = item
    }
}

次に、終了するには、各フラグメントでviewModelをインスタンス化しselected、データを設定するフラグメントからの値を設定します

フラグメントA

class MasterFragment : Fragment() {

    private val model: SharedViewModel by activityViewModels()

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        itemSelector.setOnClickListener { item ->
        model.select(item)
      }

    }
}

そして、あなたのフラグメントの目的地でこの値を聞いてください

フラグメントB

 class DetailFragment : Fragment() {

        private val model: SharedViewModel by activityViewModels()

        override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
            super.onViewCreated(view, savedInstanceState)
            model.selected.observe(viewLifecycleOwner, Observer<Item> { item ->
                // Update the UI
            })
        }
    }

逆の方法でもできます


0

100%実用的な解決策:(もしあなたが助けたら、賛成投票をすることを忘れないでください)

最初のフラグメントに次のコードを挿入します。

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

                    Fragment_editprofile Fragment = new Fragment_editprofile();
.
.
.
.
.


                Fragment.getintentdetails(tv_name.getText().toString(),
                            tv_mob.getText().toString(), tv_email.getText().toString(), tv_dob.getText().toString(),
                            tv_gender.getText().toString(), photointent);

                }
            });

2番目のフラグメントで、以下のようなメソッドを作成します。

public void getintentdetails(String name, String mobile, String email, String dob, String gender,
                                 String photointent) {

        this.name_str= name;
        this.mob_str= mobile;
        this.email_str= email;
        this.dob_str= dob;
        this.gender_str= gender;
        this.photo_str= photointent;

    }

次に、クラスレベルで変数を定義します。

String name_str, mob_str, dob_str, photo_str, email_str, gender_str;

次に、2番目のフラグメントに別のメソッドを作成して値を設定します。

  setexistingdetails();


private void setexistingdetails() {

        if(!name_str.equalsIgnoreCase(""))
            (et_name).setText(name_str);
        if(!mob_str.equalsIgnoreCase(""))
            et_mobile.setText(mob_str);
        if(!email_str.equalsIgnoreCase(""))
            email_et.setText(email_str);
        if(!dob_str.equalsIgnoreCase(""))
            dob_et.setText(dob_str);
        if(!gender_str.equalsIgnoreCase("")){
            if (gender_str.equalsIgnoreCase("m")){
                male_radio.setChecked(true);
            }else {
                female_radio.setChecked(true);
            }
        }
        if(!photo_str.equalsIgnoreCase("")){
            try {
                Picasso.get().load(Const.BASE_PATH+"include/sub-domain/GENIUS/"+photo_str).into(adminpropic_edit);
            } catch (Exception e) {            }
        }
    }

0

Arnav RaoによってクリアされるViewModelおよびLive Dataによって目標を達成できます。今、私はそれをよりきれいにクリアするために例を置きます。

まず、仮定ViewModelはと名付けられSharedViewModel.javaます。

public class SharedViewModel extends ViewModel {
    private final MutableLiveData<Item> selected = new MutableLiveData<Item>();

    public void select(Item item) {
        selected.setValue(item);
    }
    public LiveData<Item> getSelected() {
        return selected;
    }
}

次に、ソースフラグメントは、MasterFragment.javaデータを送信する送信元です。

public class MasterFragment extends Fragment {
    private SharedViewModel model;

    public void onViewCreated(@NonNull View view, Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);
        model = new ViewModelProvider(requireActivity()).get(SharedViewModel.class);
        itemSelector.setOnClickListener(item -> {

            // Data is sent

            model.select(item);
        });
    }
}

最後に、宛先フラグメントDetailFragment.java、データを受信する場所です。

public class DetailFragment extends Fragment {

    public void onViewCreated(@NonNull View view, Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);
        SharedViewModel model = new ViewModelProvider(requireActivity()).get(SharedViewModel.class);
        model.getSelected().observe(getViewLifecycleOwner(), { item ->

           // Data is received 

        });
    }
}

0

ここまで掘り下げて簡単な解決策を見つけた後、実際にはフラグメントからデータを渡して、常に悪いアイデアをフラグメント化するべきではありません。フラグメントAからアクティビティにデータを渡して、フラグメントBのアクティビティからデータを取得できます。

For Example
//fragment A
//your method that will be called in Main Activity for getting data from this fragment
public List<customDataModel> getlist(){
return mlist; //your custom list 
}
buttonopen.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            //before calling  your new fragment, send a Broadcast signal to your main activity
 mContext.sendBroadcast(new Intent("call.myfragment.action"));
//here you are now changing the fragment
Fragment fragment1 = new VolunteerListDetailsFragment();

            FragmentTransaction ft = getSupportFragmentManager().beginTransaction();
            ft.replace(R.id.content_frame, fragment1);
            ft.commit();
        }
    });

//メインアクティビティ内

this.registerReceiver(mBroadcastReceiver, new IntentFilter("call.myfragment.action"));//this line should be in your onCreate method
   BroadcastReceiver mBroadcastReceiver = new BroadcastReceiver() {

    @Override
    public void onReceive(Context context, Intent intent) {
//init the fragment A
FragmenA fragment =
                (FragmentA) getSupportFragmentManager().findFragmentById(R.id.content_frame);
        mlistdetails = fragment.getlist();//this is my method in fragmentA which is retuning me a custom list, init this list type in top of on create fragment


    }
};
//here you are creating method that is returning the list that you just recieved From fragmentA
public List<customDataModel> getcustomlist(){
return mlistdetails; //you are returning the same list that you recived from fragmentA
}

// FragmentBになりました

 public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {

    View rootView = (View) inflater.inflate(R.layout.vol_listdetails, container, false);

    mlistdetails=((MainActivity)getActivity()).getcustomlist();//here you are calling the method that you have created in Main Activity

    Toast.makeText(getActivity().getApplicationContext(), mlistdetails.get(0).getItemqty(), Toast.LENGTH_SHORT).show();
    return rootView;

}

-2

私は私のような初心者にとって本当に簡単なものを作りました。私のactivity_main.xmlにテキストビューを作り

id=index
visibility=invisible

次に、最初のフラグメントからこのテキストビューを取得します

index= (Textview) getActivity().findviewbyid(R.id.index)
index.setText("fill me with the value")

そして2番目のフラグメントで値を取得します

index= (Textview) getActivity().findviewbyid(R.id.index)
String get_the_value= index.getText().toString();

それが役に立てば幸い!


回答は、尋ねられた質問とは関係ありません
カナガリンガム
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.