Android: обновить содержимое фрагмента ViewPager после ввода данных из активности

Я просто новичок в Android, и я исследовал это в течение нескольких недель, но я до сих пор не могу найти способ заставить его работать. :(

У меня есть 4 вкладки фрагментов (созданные динамически) в моей основной деятельности. Я ввожу сумму из подсказки в основном действии и сохраняю ее в базе данных. Я хочу обновить значение на первой вкладке после нажатия кнопки «ОК» в приглашении, чтобы получить сумму, сохраненную в базе данных, и отобразить ее на первом фрагменте вкладки.

Как обновить значения на первой вкладке?

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

public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
  final View rootview = inflater.inflate(R.layout.home_tab, container, false);
  super.onCreate(savedInstanceState);
  myDb = new DatabaseHelper(getContext());
  viewBudget(rootview);
  return rootview;
}
public View viewBudget(final View rootview) {
  //get values from database and display in dynamic views
}

Я очень ценю ваши ответы. Заранее спасибо.

РЕДАКТИРОВАТЬ: Вот мой полный код для viewBudget:

public View viewBudget(final View rootview) { //display funds dynamically in config layout | LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState
  //Cursor res = myDb.getAllData();
  //String[] textArray = {"One", "Two", "Three", "Four"};
  Double budget, totalpct, temp_pct;

  Cursor res2 = myDb.getConfigData();
  res2.moveToFirst();

  Cursor res3 = myDb.getLatestIncome();
  res3.moveToFirst();
  //showMessage("Number of rows",Integer.toString(res2.getCount()));

  Cursor res4 = myDb.getTotalExpenes();
  res4.moveToFirst();

  //view contents of config table
  if (res2.getCount() == 0) {
    // show message
    //showMessage("Error","Nothing found");
    //return;
  }

  final ScrollView scrollView = (ScrollView) rootview.findViewById(R.id.scroll_budget);
  LinearLayout linearLayout = (LinearLayout) rootview.findViewById(R.id.ll_budget);

  LinearLayout llh_texts = new LinearLayout(getActivity());
  llh_texts.setOrientation(LinearLayout.HORIZONTAL);
  LinearLayout llh_texts2 = new LinearLayout(getActivity());
  llh_texts2.setOrientation(LinearLayout.HORIZONTAL);
  LinearLayout llh_texts3 = new LinearLayout(getActivity());
  llh_texts3.setOrientation(LinearLayout.HORIZONTAL);
  LinearLayout llh_texts4 = new LinearLayout(getActivity());
  llh_texts4.setOrientation(LinearLayout.HORIZONTAL);

  LinearLayout.LayoutParams lp_llh_texts = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.MATCH_PARENT);
  llh_texts.setLayoutParams(lp_llh_texts);

  //final View linearLayout = getActivity().findViewById(R.id.ll_config);
  linearLayout.removeAllViews(); //clear layout first - LINE WITH ISSUE
  linearLayout.setGravity(Gravity.CENTER);

  //LinearLayout.LayoutParams lp2 = new LinearLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT,ViewGroup.LayoutParams.WRAP_CONTENT);
  //LinearLayout.LayoutParams lp3 = new LinearLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);

  //compute for value of budget
  budget = 0.0;
  totalpct = 0.0;

  linearLayout.addView(llh_texts); //add button h.linearlayout to parent linearlayout inside scrollview

  TextView textBudget = new TextView(getActivity());
  TextView labelBudget = new TextView(getActivity());
  labelBudget.setText("Budget: ");
  llh_texts.addView(labelBudget);
  llh_texts.addView(textBudget);

  TextView textIncome = new TextView(getActivity());
  TextView labelIncome = new TextView(getActivity());
  labelIncome.setText("Income: ");
  linearLayout.addView(llh_texts2);
  llh_texts2.addView(labelIncome);
  llh_texts2.addView(textIncome);

  TextView textTotalFunds = new TextView(getActivity());
  TextView labelFunds = new TextView(getActivity());
  labelFunds.setText("Total Savings Funds: ");
  linearLayout.addView(llh_texts3);
  llh_texts3.addView(labelFunds);
  llh_texts3.addView(textTotalFunds);

  TextView textTotalExpenses = new TextView(getActivity());
  TextView labelExpenses = new TextView(getActivity());
  labelExpenses.setText("Total Scheduled Expenses: ");
  linearLayout.addView(llh_texts4);
  llh_texts4.addView(labelExpenses);
  llh_texts4.addView(textTotalExpenses);

  //create dynamic objects inside scrollview and dynamic linear layout - horizontal
  for (int i = 0; i < res2.getCount(); i++) {
    LinearLayout llh = new LinearLayout(getActivity());
    llh.setOrientation(LinearLayout.HORIZONTAL);
    LinearLayout.LayoutParams lp_llh = new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, LinearLayout.LayoutParams.MATCH_PARENT);
    llh.setLayoutParams(lp_llh);
    //llh.setLayoutParams(lp4,llh.setOrientation(LinearLayout.HORIZONTAL); //for length and width

    linearLayout.addView(llh);

    //LinearLayout.LayoutParams lp_np = new LinearLayout.LayoutParams(70, LinearLayout.LayoutParams.WRAP_CONTENT);

    temp_pct = (Double.valueOf(res2.getString(2)) / 100) * Double.valueOf(res3.getString(3));

    EditText editText = new EditText(getActivity());
    editText.setText(String.valueOf(temp_pct));
    editText.setEnabled(false);

    EditText editText2 = new EditText(getActivity());
    editText2.setText(String.valueOf(res2.getString(2)) + "%");
    editText2.setEnabled(false);

    //get total of funds
    totalpct = totalpct + temp_pct;

    //showMessage("value",res2.getString(3));
    TextView textView = new TextView(getActivity());
    textView.setText(res2.getString(1));

    llh.addView(textView);

    llh.addView(editText);
    llh.addView(editText2);

    res2.moveToNext();
  }

  //return scrollView;
  //create budget text and content
  budget = Double.valueOf(res3.getString(3)) - (totalpct + res4.getDouble(0));
  textBudget.setText(String.valueOf(budget));
  textIncome.setText(String.valueOf(res3.getString(3)));
  textTotalFunds.setText(String.valueOf(totalpct));
  textTotalExpenses.setText(String.valueOf(res4.getDouble(0)));

  return rootview;
}

РЕДАКТИРОВАТЬ: добавлен код на вкладку 3-го фрагмента на основе ответа Бхуванеша:

@Override
public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setHasOptionsMenu(true); //Make sure you have this line of code.
  setUserVisibleHint(false);
}

//@Override /*<-- this returned  "method does not override method in its superclass" so I commented out*/
public void setUserVisibleHint(final View view, boolean isVisibleToUser) {
  super.setUserVisibleHint(isVisibleToUser);
  if (isVisibleToUser) {
    //get data from database and refresh view.
    viewFunds(view); //this method refreshes the rootview of the 3rd fragment 
  }
}

@Override
public void onViewCreated(final View view, @Nullable Bundle savedInstanceState) {
  super.onViewCreated(view, savedInstanceState);
  ((Main2Activity) getActivity()).setPopupListener(new Main2Activity.PopupListener() {
    @Override
    public void onDialogClick(String value) {
      //After clicking dialog ok button in Activity
      // you will get value here.
      //viewFunds(view);
      setUserVisibleHint(view, true);
    }
  });
}


person Glenn C    schedule 11.09.2017    source источник
comment
У вас есть viewpager вместе с вкладкой?   -  person Sandip Fichadiya    schedule 11.09.2017
comment
Ваша проблема - архитектурная проблема. Я предлагаю вам сделать паузу и потратить один или два дня на изучение шаблона MVP, например, прочитав это замечательное введение: antonioleiva.com/mvp-android ПОЧЕМУ ? Потому что в краткосрочной перспективе вам нужно понять и применить разделение интересов. Ваша активность и фрагменты должны говорить, и вы можете просто создать интерфейс и на этом закончить, но, в конце концов, этого разделения будет недостаточно, когда все усложнится, оно будет плохо масштабироваться и станет головной болью. Это был мой опыт.   -  person Martin Marconcini    schedule 11.09.2017
comment
getSupportFragmentManager().beginTransaction().detach(yourFragment).attach(yourFragment).commit(); когда вы хотите перезагрузить фрагмент.   -  person HB.    schedule 11.09.2017
comment
Привет Х.Брукс, я тестирую разные подходы. Должен ли я поместить ваш код в основное действие? Чем заменить значение yourFragment? Я попытался предоставить ему имя класса моего фрагмента, который является home_tab, но есть ошибка, которая говорит Ожидается выражение. Спасибо   -  person Glenn C    schedule 13.09.2017


Ответы (3)


Вы можете сделать это через интерфейс.

Создайте объект интерфейса в своей деятельности и установите объект из своего фрагмента.

В вашей активности создайте объект Lister следующим образом:

public class MyActivity extends AppCompatActivity {

    private PopupListener popupListener;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //Your dialog code.
         setPositiveButton("Ok", new DialogInterface.OnClickListener() {
              @Override
              public void onClick(DialogInterface dialog, int which) {
                 popupListener.onDialogClick(your_value);
              }
         })
    }

    public void setPopupListener(PopupListener popupListener) {
        this.popupListener = popupListener;
    }

    public interface PopupListener{
        void onDialogClick(String value);
    }
}

В вашем фрагменте вы должны установить прослушиватель следующим образом:

public class MyFragment extends Fragment{

    @Override
    public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);
        ((MyActivity)getActivity()).setPopupListener(new MyActivity.PopupListener() {
            @Override
            public void onDialogClick(String value) {
                //After clicking dialog ok button in Activity
                // you will get value here.
            }
        });
    }
}

Чтобы обновить содержимое другого фрагмента вашего viewPager, переопределите приведенный ниже метод во всех ваших viewPager фрагментах.

@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
    super.setUserVisibleHint(isVisibleToUser);
    if (isVisibleToUser) { 
      //get data from database and refresh view.
    }
}

Перед этим добавьте следующий код в свои фрагменты onCreate().

public void onCreate(@Nullable Bundle savedInstanceState) {
    setUserVisibleHint(false);
}

Надеюсь, поможет:)

person Bhuvanesh BS    schedule 11.09.2017
comment
Привет, Бхуванеш, кажется, я не могу обновить другие фрагменты. Мне нужно обновить текущий отображаемый фрагмент. Какие необходимые обновления необходимы для кода, который вы предоставили? Спасибо! :) - person Glenn C; 12.09.2017
comment
Ответ обновлен. Попробуйте и спросите меня, есть ли у вас какие-либо сомнения. - person Bhuvanesh BS; 12.09.2017
comment
Привет, Бхуванеш, пожалуйста, ознакомьтесь с обновлениями в исходном вопросе выше и, пожалуйста, проверьте, правильно ли я добавил ваш код в 3-й фрагмент. Содержимое 3-го фрагмента по-прежнему не обновляется. Спасибо - person Glenn C; 13.09.2017
comment
Здравствуйте, Бхуванеш, я хотел бы продолжить, если у вас есть время. Спасибо - person Glenn C; 14.09.2017
comment
Используя обновления, я заметил, что когда я нажимаю первую вкладку и снова возвращаюсь к 3-й вкладке, представление обновляется. Какие обновления следует сделать, чтобы обновить 3-ю вкладку сразу после ввода из действия и не нужно щелкать 1-ю вкладку и возвращаться к 3-й? Спасибо! - person Glenn C; 14.09.2017

Во-первых, определите общедоступный метод во фрагменте, который будет повторно запрашивать базу данных и обновлять пользовательский интерфейс:

public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
  final View rootview = inflater.inflate(R.layout.home_tab, container, false);
  super.onCreate(savedInstanceState);
  myDb = new DatabaseHelper(getContext());
  viewBudget(rootview);
  return rootview;
}

public View viewBudget(final View rootview) {
  getUpdatedDatabaseData();
}

public void getUpdatedDatabaseData() {
  //get values from database and display in dynamic views
}

Затем добавьте переопределение instantiateItem() в свой FragmentPagerAdapter и сохраните массив ссылок на фрагменты для всех фрагментов в ViewPager. Вот простой пример:

class PagerAdapter extends FragmentPagerAdapter {
        String tabTitles[] = new String[] { "One", "Two", "Three", "Four"};
        Context context;

        //This will contain your Fragment references:
        public Fragment[] fragments = new Fragment[tabTitles.length];

        public PagerAdapter(FragmentManager fm, Context context) {
            super(fm);
            this.context = context;
        }
        @Override
        public int getCount() {
            return tabTitles.length;
        }
        @Override
        public Fragment getItem(int position) {
            switch (position) {
            case 0:
                return new FragmentOne();
            case 1:
                return new FragmentTwo();
            case 2:
                return new FragmentThree();   
            case 3:
                return new FragmentFour();
            }
            return null;
        }

        //This populates your Fragment reference array:
        @Override
        public Object instantiateItem(ViewGroup container, int position) {
            Fragment createdFragment = (Fragment) super.instantiateItem(container, position);
            fragments[position]  = createdFragment;
            return createdFragment;
        }

        @Override
        public CharSequence getPageTitle(int position) {
            // Generate title based on item position
            return tabTitles[position];
        }         
 }

Затем, когда пользователь использовал диалоговое окно для добавления данных в базу данных, вы можете использовать этот код в действии для обновления фрагмента на первой вкладке (если это одна из активных в данный момент вкладок в ViewPager).

    Fragment frag = mAdapter.fragments[0];
    if (frag != null && frag instanceof FragmentOne) {
      ((FragmentOne)frag).getUpdatedDatabaseData();
    }
person Daniel Nugent    schedule 11.09.2017
comment
Здравствуйте, Даниил, спасибо за помощь. Я перенес код из viewBudget в getUpdatedDatabaseData, однако у меня возникла проблема с этим кодом после переноса: LinearLayout linearLayout =(LinearLayout)rootview.findViewById(R.id.ll_budget); Я использую это для добавления представлений на основе значений базы данных, например: linearLayout.addView(llh_texts); Теперь, когда я перенес это в getUpdatedDatabaseData, мне также нужно объявить представление как параметр, что может вызвать проблему в: ((FragmentOne)frag).getUpdatedDatabaseData(); Что мы можем сделать в качестве обходного пути для этого? - person Glenn C; 12.09.2017
comment
Я включил свой код для viewBudget для справки, см. обновления выше в вопросе. Спасибо. - person Glenn C; 12.09.2017

Вы можете использовать BroadReceiver в своем фрагменте и отправлять данные из Activity через намерение вещания.

При сохранении в базу данных в MainActivity отправить эту трансляцию

Intent intent = new Intent("key_to_identify_the_broadcast");
intent.putExtra("refresh", true);
context.sendBroadcast(intent);

а затем для вашего первого фрагмента создайте широковещательный приемник

private final BroadcastReceiver mHandleMessageReceiver = new 
BroadcastReceiver() {
    @Override
    public void onReceive(Context context, Intent intent) {
        Boolean refresh = 
            intent.getBooleanExtra("refresh",false);
        if(refresh){
           notifyDataSetChanged();
        }
        //you can call any of your methods for using this value for your use case
  }
};

Вам необходимо зарегистрировать эту трансляцию в onCreateView() фрагмента

IntentFilter filter = new 
      IntentFilter("key_to_identify_the_broadcast");
       getActivity().getApplicationContext().
              registerReceiver(mHandleMessageReceiver, filter);

Вам нужно отменить регистрацию в onDestroy() вашего фрагмента

@Override
public void onDestroy() {
    try {

     getActivity().getApplicationContext().
         unregisterReceiver(mHandleMessageReceiver);

   } catch (Exception e) {
      Log.e("UnRegister Error", "> " + e.getMessage());
   }
   super.onDestroy();
}

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

person Shriyansh Gautam    schedule 11.09.2017
comment
Я бы не рекомендовал этот способ для этой конкретной ситуации. В случае фрагментов в ViewPager они будут создаваться и уничтожаться очень быстро, когда пользователь пролистывает все вкладки, поэтому с этим решением получатель будет регистрироваться/отменять регистрацию очень быстро и много раз, когда пользователь пролистывает вкладки. . - person Daniel Nugent; 11.09.2017