передача аргумента DialogFragment

116

Мне нужно передать некоторые переменные DialogFragment, чтобы я мог выполнить действие. Eclipse предлагает использовать

Fragment#setArguments(Bundle)

Но я не знаю, как пользоваться этой функцией. Как я могу использовать его для передачи переменных в мой диалог?

giozh
источник
Посмотрите на пример кода: androidxref.com/4.4.2_r1/xref/development/samples/ApiDemos/src/…
Игорь Ганапольский

Ответы:

299

С помощью newInstance

public static MyDialogFragment newInstance(int num) {
    MyDialogFragment f = new MyDialogFragment();

    // Supply num input as an argument.
    Bundle args = new Bundle();
    args.putInt("num", num);
    f.setArguments(args);

    return f;
}

И получить такие аргументы

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    mNum = getArguments().getInt("num");
    ...
}

См. Полный пример здесь
http://developer.android.com/reference/android/app/DialogFragment.html

JafarKhQ
источник
Можете ли вы просто установить частные переменные в MyDialogFragment вместо использования пакета?
SIr Codealot
10
@SIrCodealot эффект будет таким же, как при установке переменных в Activity или Fragment. Если вы столкнетесь с чем-то, что разрушает и воссоздает DialogDragment, например, при изменении поворота, вы потеряете все переменные.
inmyth
2
Для всех тех, кто задается вопросом, почему в этом случае не используется перегруженный конструктор, см. Еще одно очень поучительное обсуждение этой темы: stackoverflow.com/questions/14011808/…
HondaGuy
Мне потребовалась минута, чтобы заметить, что saveInstanceState не используется.
Odys
25

Раньше я отправлял некоторые значения из моего списка

Как отправить

mListview.setOnItemLongClickListener(new AdapterView.OnItemLongClickListener() {
        @Override
        public boolean onItemLongClick(AdapterView<?> parent, View view, int position, long id) {
            Favorite clickedObj = (Favorite) parent.getItemAtPosition(position);

            Bundle args = new Bundle();
            args.putString("tar_name", clickedObj.getNameTarife());
            args.putString("fav_name", clickedObj.getName());

            FragmentManager fragmentManager = getSupportFragmentManager();
            TarifeDetayPopup userPopUp = new TarifeDetayPopup();
            userPopUp.setArguments(args);
            userPopUp.show(fragmentManager, "sam");

            return false;
        }
    });

Как получить внутри метода onCreate () DialogFragment

    Bundle mArgs = getArguments();
    String nameTrife = mArgs.getString("tar_name");
    String nameFav = mArgs.getString("fav_name");
    String name = "";

// Загрузка Котлина

 val fm = supportFragmentManager
        val dialogFragment = AddProgFargmentDialog() // my custom FargmentDialog
        var args: Bundle? = null
        args?.putString("title", model.title);
        dialogFragment.setArguments(args)
        dialogFragment.show(fm, "Sample Fragment")

// получить

 override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        if (getArguments() != null) {
            val mArgs = arguments
            var myDay= mArgs.getString("title")
        }
    }
Сэм
источник
1
Лучший ответ!
user55924
1
Спасибо ! Версия Kotlin помогла.
ArdenDev,
6

Итак, есть два способа передать значения из фрагмента / активности во фрагмент диалога: -

  1. Создайте объект фрагмента диалога с помощью метода make setter и передайте значение / аргумент.

  2. Передайте значение / аргумент через пакет.

Способ 1:

// Fragment or Activity 
@Override
public void onClick(View v) {
     DialogFragmentWithSetter dialog = new DialogFragmentWithSetter();
     dialog.setValue(header, body);
     dialog.show(getSupportFragmentManager(), "DialogFragmentWithSetter");         
}


//  your dialog fragment
public class MyDialogFragment extends DialogFragment {
    String header; 
    String body;
    public void setValue(String header, String body) {   
          this.header = header;
          this.body = body;
    }
    // use above variable into your dialog fragment
}

Примечание: - Это не лучший способ сделать

Способ 2:

// Fragment or Activity 
@Override
public void onClick(View v) {
     DialogFragmentWithSetter dialog = new DialogFragmentWithSetter();
     
     Bundle bundle = new Bundle();
     bundle.putString("header", "Header");
     bundle.putString("body", "Body");  
     dialog.setArguments(bundle);
     dialog.show(getSupportFragmentManager(), "DialogFragmentWithSetter");         
}


//  your dialog fragment
public class MyDialogFragment extends DialogFragment {
    String header; 
    String body;
    
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        if (getArguments() != null) {
             header = getArguments().getString("header","");
             body = getArguments().getString("body","");
        }
    }
    // use above variable into your dialog fragment
}

Примечание: - Это лучший способ сделать.

Дуггу
источник
Вы даже можете использовать библиотеку Gson для передачи объектов или ArrayList <Objects> как строк в Bundle.
Джеффри
@duggu, почему геттер и сеттер - не лучший способ?
sky91 06
5

в качестве общего способа работы с фрагментами, как заметил JafarKhQ, вы не должны передавать параметры в конструкторе, но с расширением Bundle.

встроенный метод для этого в Fragmentклассе - setArguments(Bundle)и getArguments().

По сути, вы создаете комплект со всеми своими Parcelableпредметами и отправляете их.
в свою очередь, ваш Фрагмент получит эти предметы onCreateи сотворит с ними магию.

способ, показанный в DialogFragmentссылке, был одним из способов сделать это в многократно появляющемся фрагменте с одним конкретным типом данных и большую часть времени работает нормально, но вы также можете сделать это вручную.

thepoosh
источник
0

В моем случае ни один из приведенных выше кодов не bundle-operateработает; Вот мое решение (не знаю, правильный это код или нет, но в моем случае он работает):

public class DialogMessageType extends DialogFragment {
    private static String bodyText;

    public static DialogMessageType addSomeString(String temp){
        DialogMessageType f = new DialogMessageType();
        bodyText = temp;
        return f;
    };

    @Override
    public Dialog onCreateDialog(Bundle savedInstanceState) {
        final String[] choiseArray = {"sms", "email"};
        String title = "Send text via:";
        final AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
        builder.setTitle(title).setItems(choiseArray, itemClickListener);
        builder.setCancelable(true);
        return builder.create();
    }

    DialogInterface.OnClickListener itemClickListener = new DialogInterface.OnClickListener() {
        @Override
        public void onClick(DialogInterface dialog, int which) {
            switch (which){
                case 0:
                    prepareToSendCoordsViaSMS(bodyText);
                    dialog.dismiss();
                    break;
                case 1:
                    prepareToSendCoordsViaEmail(bodyText);
                    dialog.dismiss();
                    break;
                default:
                    break;
            }
        }
    };
[...]
}

public class SendObjectActivity extends FragmentActivity {
[...]

DialogMessageType dialogMessageType = DialogMessageType.addSomeString(stringToSend);
dialogMessageType.show(getSupportFragmentManager(),"dialogMessageType");

[...]
}
Кирилл
источник
1) Сохраняя bodyText статически, вы фактически делаете невозможным одновременное использование двух экземпляров этого класса с разными основными текстами. Нет причин не хранить его как переменную экземпляра. 2) Вся цель отправки аргументов с помощью setArguments (Bundle) заключается в том, что ОС может затем воссоздать фрагмент в случае его потери в ситуации нехватки памяти и т. Д. С вашим решением фрагмент будет воссоздан, а основной текст будет быть тем, что использовалось в последний раз (поскольку он статичен). Правильное решение - установить основной текст как параметр пакета.
JHH
0

Просто я хочу показать, как сделать то, что сказал @JafarKhQ в Kotlin, для тех, кто использует kotlin, который может им помочь и сэкономить время темы:

поэтому вам нужно создать сопутствующий объект для создания новой функции newInstance

вы можете установить параметр функции, какой захотите. с помощью

 val args = Bundle()

вы можете установить свои аргументы.

Теперь вы можете использовать его args.putSomthingдля добавления аргументов, которые вы указываете в качестве параметра в вашей функции newInstance. putString(key:String,str:String)добавить строку, например, и т. д.

Теперь, чтобы получить аргумент, вы можете использовать arguments.getSomthing(Key:String)=> likearguments.getString("1")

вот полный пример

class IntervModifFragment : DialogFragment(), ModContract.View
{
    companion object {
        fun newInstance(  plom:String,type:String,position: Int):IntervModifFragment {
            val fragment =IntervModifFragment()
            val args = Bundle()
            args.putString( "1",plom)
            args.putString("2",type)
            args.putInt("3",position)
            fragment.arguments = args
            return fragment
        }
    }

...
    override fun onViewCreated(view: View?, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        fillSpinerPlom(view,arguments.getString("1"))
         fillSpinerType(view, arguments.getString("2"))
        confirmer_virme.setOnClickListener({on_confirmClick( arguments.getInt("3"))})


        val dateSetListener = object : DatePickerDialog.OnDateSetListener {
            override fun onDateSet(view: DatePicker, year: Int, monthOfYear: Int,
                                   dayOfMonth: Int) {
                val datep= DateT(year,monthOfYear,dayOfMonth)
                updateDateInView(datep.date)
            }
        }

    }
  ...
}

Теперь, как создать свой диалог, вы можете сделать что-то подобное в другом классе.

  val dialog = IntervModifFragment.newInstance(ListInter.list[position].plom,ListInter.list[position].type,position)

как это например

class InterListAdapter(private val context: Context, linkedList: LinkedList<InterItem> ) : RecyclerView.Adapter<InterListAdapter.ViewHolder>()
{
   ... 
    override fun onBindViewHolder(holder: ViewHolder, position: Int) {

        ...
        holder.btn_update!!.setOnClickListener {
           val dialog = IntervModifFragment.newInstance(ListInter.list[position].plom,ListInter.list[position].type,position)
           val ft = (context as AppCompatActivity).supportFragmentManager.beginTransaction()
            dialog.show(ft, ContentValues.TAG)
        }
        ...
    }
..

}
ДИНА ТАКЛИТ
источник