ActionBarSherlock и FragmentTabsPager

Как и многие люди, я внедряю FragmentTabsPager в свое приложение, которое использует ActionBarSherlock 4.0. Однако я потерян.

Фрагменты и все маленькие идеи, планы и методы Google, окружающие его, сбивают меня с толку. Если бы кто-нибудь мог взглянуть на мой код и провести меня через это, помогая заставить его работать, я бы поблагодарил их тысячу раз :D.

У меня есть еще один проект с своего рода началом для ViewPager, но вкладки просто лучше сочетаются, особенно когда они находятся в панели действий в альбомной ориентации и на планшетах.

Мой код заархивирован и готов к отправке сюда: http://dl.dropbox.com/u/21807195/Antonius%20College%202.zip

Заранее спасибо!


person DatBassie    schedule 20.03.2012    source источник
comment
github.com/ Джейк Уортон/ActionBarSherlock/blob/master/samples/   -  person Thomas Dignan    schedule 14.11.2012


Ответы (2)


Я покажу вам свой код, который имеет ViewPager, TabListener и систему фрагментов, связанных с каждой вкладкой. Он реализует ABS, но пока все еще падает на Gingerbread и ниже (хотя прекрасно работает на ICS):

import java.util.ArrayList;
import library.DatabaseHandler;
import org.json.JSONObject;
import com.actionbarsherlock.R;
import com.actionbarsherlock.app.ActionBar;
import com.actionbarsherlock.app.ActionBar.Tab;
import com.actionbarsherlock.app.SherlockFragmentActivity;
import android.app.AlertDialog;
import android.app.Dialog;
import android.app.ProgressDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.util.Log;
import android.view.Menu;
import android.view.MenuInflater;
import android.view.MenuItem;
import android.view.MenuItem.OnMenuItemClickListener;
import android.view.View.OnClickListener;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Switch;
import android.widget.TextView;
import android.widget.Toast;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentActivity;
import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentPagerAdapter;
import android.support.v4.app.FragmentTransaction;
import android.support.v4.view.ViewPager;
import android.support.v4.view.ViewPager.OnPageChangeListener;

public class Polling extends SherlockFragmentActivity {
    private ViewPager mViewPager;
    private TabsAdapter mTabsAdapter;
    private final static String TAG = "21st Polling:";

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

        mViewPager = new ViewPager(this);
        mViewPager.setId(R.id.pager);
        setContentView(mViewPager);
        ActionBar bar = getSupportActionBar();
        bar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
        bar.setDisplayShowTitleEnabled(false);
        bar.setDisplayShowHomeEnabled(false);

        mTabsAdapter = new TabsAdapter(this, mViewPager);
        mTabsAdapter.addTab(bar.newTab().setText(R.string.login),
                LoginFragment.class, null);
        mTabsAdapter.addTab(bar.newTab().setText(R.string.economics),
                EconFragment.class, null);
        mTabsAdapter.addTab(bar.newTab().setText(R.string.elections),
                ElectionsFragment.class, null);
        mTabsAdapter.addTab(bar.newTab().setText(R.string.politics),
                PoliticsFragment.class, null);
        mTabsAdapter.addTab(bar.newTab().setText(R.string.science),
                ScienceFragment.class, null);
        mTabsAdapter.addTab(bar.newTab().setText(R.string.finance),
                FinanceFragment.class, null);
        mTabsAdapter.addTab(bar.newTab().setText(R.string.religion),
                ReligionFragment.class, null);
        mTabsAdapter.addTab(bar.newTab().setText(R.string.military),
                MilitaryFragment.class, null);
        mTabsAdapter.addTab(bar.newTab().setText(R.string.international),
                InternationalFragment.class, null); 
        Log.v(TAG, (String)bar.getTabAt(0).getText());

    }
public static class TabsAdapter extends FragmentPagerAdapter
    implements ActionBar.TabListener, ViewPager.OnPageChangeListener {
        private final Context mContext;
        private final ActionBar mActionBar;
        private final ViewPager mViewPager;
        private final ArrayList<TabInfo> mTabs = new ArrayList<TabInfo>();

        static final class TabInfo {
            private final Class<?> clss;
            private final Bundle args;

            TabInfo(Class<?> _class, Bundle _args) {
                clss = _class;
                args = _args;
            }
        }

        public TabsAdapter(SherlockFragmentActivity activity, ViewPager pager) {
            super(activity.getSupportFragmentManager());
            mContext = activity;
            mActionBar = activity.getSupportActionBar();
            mViewPager = pager;
            mViewPager.setAdapter(this);
            mViewPager.setOnPageChangeListener(this);
        }

        public void addTab(ActionBar.Tab tab, Class<?> clss, Bundle args) {
            TabInfo info = new TabInfo(clss, args);
            tab.setTag(info);
            tab.setTabListener(this);
            mTabs.add(info);
            mActionBar.addTab(tab);
            notifyDataSetChanged();
        }


        public int getCount() {
            return mTabs.size();
        }

        public SherlockFragment getItem(int position) {
            TabInfo info = mTabs.get(position);
            return (SherlockFragment)Fragment.instantiate(mContext, info.clss.getName(), info.args);
        }


        public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
        }


        public void onPageSelected(int position) {
            mActionBar.setSelectedNavigationItem(position);
        }


        public void onPageScrollStateChanged(int state) {
        }


        public void onTabSelected(Tab tab, FragmentTransaction ft) {
            mViewPager.setCurrentItem(tab.getPosition());
            //Log.v(TAG, "clicked");
            Object tag = tab.getTag();
            for (int i=0; i<mTabs.size(); i++) {
                if (mTabs.get(i) == tag) {
                    mViewPager.setCurrentItem(i);
                }
            }
        }

        public void onTabUnselected(Tab tab, FragmentTransaction ft) {}

        public void onTabReselected(Tab tab, FragmentTransaction ft) {}

        public void onTabReselected(Tab tab, android.app.FragmentTransaction ft) {}

        public void onTabUnselected(Tab tab, android.app.FragmentTransaction ft) {}
    }

А вот как выглядит один фрагмент:

package com.davekelley.polling;

import com.actionbarsherlock.R;
import com.actionbarsherlock.app.SherlockFragment;

import android.support.v4.app.Fragment;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;

public class MilitaryFragment extends SherlockFragment {

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

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

Мое последнее замечание: в моем коде все еще есть еще одна проблема: отдельные фрагменты не всегда перезагружают свой интерфейс после того, как пользователь нажимает «Назад» (что приводит к удалению всего приложения с экрана, независимо от того, на какой вкладке/фрагменте они находятся, потому что у меня есть нет обратного стека). Вот над чем я сейчас работаю. Я думаю, что как только я разберусь с этим, я попытаюсь выяснить, почему у меня до сих пор не работает выполнение Gingerbread должным образом. В любом случае, я надеюсь, что просмотр этого кода поможет вам.

Вот фрагмент с некоторыми onClickListeners:

import com.actionbarsherlock.R;
import com.actionbarsherlock.app.SherlockFragment;
import android.app.Activity;
import android.app.ProgressDialog;
import android.content.Context;
import android.os.Bundle;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TableLayout;
import android.widget.TableRow;
import android.widget.TextView;
import android.widget.Toast;

public class LoginFragment extends SherlockFragment {

    Button loginButton;
    Button registerButton;
    Polling activity;

    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View v = inflater.inflate(R.layout.loginfragment, container, false);

        return v;
    }

    public void onResume() {
        super.onResume();
        Log.d("Econ", "onresume");

        loginButton = (Button) getActivity().findViewById(R.id.loginButton);
        loginButton.setOnClickListener(loginListener);
        registerButton = (Button) getActivity().findViewById(R.id.registerButton);
        registerButton.setOnClickListener(registerListener);

    }

    public OnClickListener loginListener = new OnClickListener() {
        @Override
        public void onClick(View v) {

            if(loginButton.getText().toString() == "Log Out") {

                activity.loginReport(2);
                loginButton.setText(R.string.login);
                //Remove user from dB sqllite when I know how
            }
            else {
            Log.v("LoginF", "onclick");
            ProgressDialog progressDialog = new ProgressDialog(getActivity());
            progressDialog.setMessage("Logging in...");
            LoginTask loginTask = new LoginTask((Polling) getActivity(), progressDialog);
            loginTask.execute();
            }
        }
    };


    public OnClickListener registerListener = new OnClickListener() {
        @Override
        public void onClick(View v) {


            Log.v("Register", "onclick");
            ProgressDialog progressDialog = new ProgressDialog(getActivity());
            progressDialog.setMessage("Registering new user...");
            RegisterTask registerTask = new RegisterTask((Polling) getActivity(), progressDialog);
            registerTask.execute();
            }

    };

    public void onAttach(Activity activity) {
        super.onAttach(activity);
        this.activity = (Polling) activity;
    }

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


    }

    public void onActivityCreated(Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
    }

    public void onStart() {
        super.onStart();
        Log.d("Econ", "onstart");
    }


    public void onPause() {
        super.onPause();Log.d("Econ", "onpause");
    }

    public void onStop() {
        super.onStop();
        Log.d("Econ", "onstop");
    }

    public void onDestroyView() {
        super.onDestroyView();
        Log.d("Econ", "ondestroyview");
    }

    public void onDestroy() {
        super.onDestroy();
        Log.d("Econ", "ondestroy");

    }

    public void onDetach() {
        super.onDetach();
        Log.d("Econ", "ondetach");
    }

}

Объекты loginTask, которые вы видите, на самом деле являются классами, которые расширяют ASyncTask — они обрабатывают подключение к моему серверу и регистрацию/вход в систему.

Я подумал, что было бы полезно добавить еще немного кода. Это еще один из моих фрагментов, как и LoginFragment, но он немного по-другому расширяет пользовательский интерфейс. В конце концов, то, что вы видите в цикле while ниже, будет направлено в ASyncTask для получения каждого вопроса с сервера, а не в фиктивный массив строк, который вы видите:

public class EconFragment extends SherlockFragment {

    private TableLayout questionContainer;
    int pos = 0;
    private String[] titles = {"The first title ", "hallo1","hallo2", "hallo3",
            "hallo4", "hallo5","hallo6", "hallo7","hallo8", "hallo9"};

    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        Log.v("Econ", "onCreateView");
        View v = inflater.inflate(R.layout.econfragment, container, false);
        questionContainer = (TableLayout) v.findViewById(R.id.questionContainer);
        //bs
        int leftMargin=5;
        int topMargin=5;
        int rightMargin=5;
        int bottomMargin=5;
        while (pos < 10) {
        View question = inflater.inflate(R.layout.question, null);
        question.setId(pos);
        TextView title = (TextView) question.findViewById(R.id.questionTextView);
        title.setText(titles[pos]);
        Button charts = (Button) question.findViewById(R.id.chartsButton);
        charts.setId(pos);
        charts.setOnClickListener(chartsListener);
        TableRow tr = (TableRow) question;
        TableLayout.LayoutParams trParams = new TableLayout.LayoutParams(
                TableLayout.LayoutParams.MATCH_PARENT,
                TableLayout.LayoutParams.WRAP_CONTENT);
        trParams.setMargins(leftMargin, topMargin, rightMargin, bottomMargin);
        tr.setLayoutParams(trParams);
        Log.v("econ", "while loop");
        questionContainer.addView(tr);
        pos++;
        }
        pos = 0;
        return v;
    }
person Davek804    schedule 31.03.2012
comment
Спасибо за ваш ответ. Я посмотрю на это на следующей неделе (если у меня будет время...) и отчитаюсь. Не могли бы вы также сказать мне, когда у вас все заработало, и что нужно было сделать? - person DatBassie; 01.04.2012
comment
Конечно. Причина сбоя в Gingerbread в том, что я использовал виджет Switch, который был представлен либо в 3.0, либо в 4.0, поэтому Gingerbread не мог запустить код. Я изменил этот переключатель на флажок, запустил Project->Clean, и все заработало нормально. Я забыл включить свой код TabsAdapter в свой исходный пост, поэтому я отредактирую его прямо сейчас. - person Davek804; 01.04.2012
comment
Спасибо! Но как вы обрабатываете все клики и т. д. во Фрагменте? (Да, Фрагменты меня чертовски смущают...) - person DatBassie; 02.04.2012
comment
@Bassie, через две минуты я опубликую пару своих обработчиков кликов в своем исходном комментарии вверху. - person Davek804; 02.04.2012
comment
Конечно, нет проблем. Я обнаружил, что процесс перехода от основных вкладок к вкладкам ActionBar, фрагментам, а затем к ViewPager на самом деле довольно болезненный процесс — больше, чем должен быть. Так что, если вы хотите скопировать мой код из опроса выше и запустить его в новом проекте или использовать в своем, у меня нет проблем с этим: в любом случае, большая его часть взята из различных руководств Google. В любом случае, теперь, когда это работает, у меня есть вкладки/пейджер/панель действий на 2.1-4.0, - person Davek804; 02.04.2012
comment
Добавлен в другой фрагмент примера, чтобы вы могли лучше понять их возможности. - person Davek804; 02.04.2012
comment
Привет, я снова здесь. Я добавляю это в свое приложение, но оно просто не будет работать... Мне действительно нужно сделать это правильно один раз, прежде чем я узнаю, как это сделать позже, всегда имейте это. Я нашел это: thepseudocoder. wordpress.com/2011/10/13/ некоторое время назад, но он отличается от вашего кода. Есть ли другой способ помочь (скайп или что-то в этом роде)? - person DatBassie; 10.04.2012
comment
Присоединяйтесь к моему чату в Stack Overflow. Вам может понадобиться 20 повторений, чтобы присоединиться, и в этом случае вы можете просто проголосовать за что-то, чтобы получить 2 повторения (я думаю). Вот ссылка: chat.stackoverflow.com/rooms/9922/viewpagerhelp Думаю, я посижу там звуковое уведомление воспроизводится, когда кто-то присоединяется. - person Davek804; 10.04.2012

Я перенес действие FragmentTabsPager и связанные с ним фрагменты из «Support4Demos» (образец библиотеки поддержки Android), чтобы использовать ActionBarSherlock и настоящие вкладки ActionBar. . Образец включает действие, которое использует как ViewPager, так и вкладки для переключения между фрагментами. Фрагменты содержат три вида ListView. Я протестировал его от ICS до Eclair (2.1). Вы можете просмотреть или загрузить код по адресу http://code.google.com/p/sherlock-demo/.

person Sparky    schedule 07.05.2012
comment
Можете ли вы разместить код в другом месте, так как я не могу загрузить его оттуда. У этого проекта в настоящее время нет загрузок. - person Guillermo Varini; 08.05.2012
comment
Ознакомьтесь с исходным кодом здесь: code.google.com/p/sherlock-demo/source. /checkout Просмотрите источник здесь: code.google.com/p /sherlock-demo/source/browse - person Sparky; 08.05.2012
comment
Не могли бы вы также сделать это для других функций, таких как рабочий EditText и Button и ListDialog? Возникли проблемы со всем этим... Конечно, это тоже хорошо для дальнейшего использования :). - person DatBassie; 08.05.2012