Просмотр вкладок Android с динамическим содержимым FragmentStageAdapter

#java #android #android-fragments #tabs #x509certificate

#java #Android #android-фрагменты #вкладки #x509сертифицируйте

Вопрос:

Я пытаюсь реализовать средство просмотра с вкладками с использованием FragmentStageAdapter, фрагмента и android.support.v4.view.Фреймворк ViewPager. На каждой странице вкладки должен отображаться сертификат X509 (динамическое содержимое), который был ранее загружен из Интернета.

Процедура, которую я имел в виду, заключается в следующем:

  • Загрузите сертификаты и сохраните их в массиве
  • Заполните действие вкладки вкладками, каждая вкладка содержит сертификат

Что работает:

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

Что не работает

После того, как я добавил все вкладки, я вызываю метод notifyDataSetChanged() моего FragmentStagePageAdapter, отладчик останавливается в методе onCreateView моего фрагмента, когда я пытаюсь изменить фрагмент

 View certView = inflater.inflate(R.layout.certificate_view, container, false);
  

После этого я возвращаюсь во время отладки обратно к вызову метода notifyDataSetChanged (), и мой отладчик eclipse отключается и выдает исключение:

 An internal error occurred during: "JDI Event Dispatch".
  

java.lang.Исключение NullPointerException

В ADB logcat я вижу следующее исключение, возникающее несколько раз:

 java.lang.RuntimeException: Unable to instantiate application android.app.Application: java.lang.IllegalStateException: Unable to get package info for com.mydomain.mypackage; is package not installed?
  

Я уже пытался очистить свой проект, удалить apk со своего устройства, но ошибка все еще остается.

Ниже я перечислил некоторые из моих источников, некоторые из них я сократил для простоты.

Вопросы

Правильный ли это способ заполнения фрагментов в представлении вкладок «динамическими» данными? Насколько я понимаю, fragmentstagepageadapter должен использоваться для неопределенного количества вкладок вместо fragmentadapter, который используется для «статического» содержимого.

Вот мой FragmentActivity

 public class MyFragmentActivity extends FragmentActivity {
public static final String ARGUMENT_SERVER_URL = "URL";
private String mHostUrl;

private X509Certificate[] mX509Certificates;

private MyFragmentStagePageAdapter mCertPageAdapter;
private ActionBar mActionBar;
private ViewPager mViewPager;

private OnPageChangeListener onPageChangeListener = 
new ViewPager.SimpleOnPageChangeListener(){
    
    @Override
    public void onPageSelected(int position) {
        mActionBar = getActionBar();
        mActionBar.setSelectedNavigationItem(position);
    };
};

@Override
protected void onCreate(Bundle arg0) {
    super.onCreate(arg0);
    setContentView(R.layout.certificate_view_pager);
    
    // get the host url from the activity's received intent
    getHostUrlFromIntent();
    
    // instantiate the FragmentStatePagerAdapter 
    mCertPageAdapter = 
    new MyFragmentStagePageAdapter(getSupportFragmentManager());
    
    // create the ViewPager
    mViewPager = (ViewPager) findViewById(R.id.certificate_view_pager);
    
    // assign a onPageChangeListener
    mViewPager.addOnPageChangeListener(onPageChangeListener);
    // set the FragmentStatePagerAdapter
    mViewPager.setAdapter(mCertPageAdapter);
    
    mActionBar = getActionBar();
    mActionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
    
    startCertificateDownloadAsync();
}

private void getHostUrlFromIntent() {
    Intent intent = getIntent();
    Bundle serverCredentialsBundle = intent.getExtras();
    mHostUrl = serverCredentialsBundle.getString(ARGUMENT_SERVER_URL);
    if (mHostUrl == null) {
        throw new InvalidParameterException("URL must not be empty");
    }
}

private TabListener tabListener = new TabListener() {
    
    @Override
    public void onTabUnselected(Tab tab, FragmentTransaction ft) {
        
    }
    
    @Override
    public void onTabSelected(Tab tab, FragmentTransaction ft) {
        mViewPager.setCurrentItem(tab.getPosition());
    }
    
    @Override
    public void onTabReselected(Tab tab, FragmentTransaction ft) {
        
    }
};

private void startCertificateDownloadAsync(){
    new AsyncTask<Void, Void, Boolean>() {

        @Override
        protected Boolean doInBackground(Void... params) {
            // retreive certificates here 
            // (code is removed for the sake of simplicity)

        }
        
        @Override
        protected void onPostExecute(Boolean result) {
            super.onPostExecute(result);
            assignCertificatesToActionBarOnUiThread();
        }
    }.execute();
}

private void assignCertificatesToActionBarOnUiThread() {
    runOnUiThread(new Runnable() {
        
        @Override
        public void run() {
            if (mX509Certificates != null amp;amp; 
            mX509Certificates.length >  0) {
                
                
                for (int i = 0; i < mX509Certificates.length; i  ) {
                    Tab newTab = mActionBar.newTab();
                    X509Certificate cert = mX509Certificates[i];
                    newTab.setText(cert.getIssuerDN().getName());
                    newTab.setTabListener(tabListener);
                    mActionBar.addTab(newTab);
                }
                mCertPageAdapter.setCertificateArray(mX509Certificates);
                // call is required, when data was changed externally 
                mCertPageAdapter.notifyDataSetChanged();
            }
        }
        
    });
}
  

Here is my Fragment

 public class MyCertificateFragment extends Fragment {
public static final String ARG_CERT = "CERTIFICATE";

private X509Certificate certificate;

private TextView commonName;


@Override
public View onCreateView(LayoutInflater inflater, 
        ViewGroup container, Bundle savedInstanceState) {
    // The last two arguments ensure LayoutParams are inflated properly.
    View certView = inflater.inflate(R.layout.certificate_view, 
container, false);
    
    Activity activity = getActivity();
    commonName = (TextView) 
activity.findViewById(R.id.certViewTextViewCnValue);
    
    byte[] certificateBytes = 
savedInstanceState.getByteArray(LoginActivity.INTENT_EXTRA_CERTIFICATE);
    certificate = CertUtils.toX509Certificate(certificateBytes);
    
    commonName.setText(certificate.getSubjectX500Principal().getName());
    return certView;
}
}
  

This is how my certificate_view.xml looks like

 <?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@ id/certificate_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >

<LinearLayout
    android:id="@ id/certViewLinearLayoutCn"
    android:layout_width="fill_parent"
    android:layout_height="0dp"
    android:layout_gravity="end"
    android:layout_weight="1"
    android:gravity="fill_vertical|fill_horizontal"
    android:orientation="horizontal" >

    <TextView
        android:id="@ id/certViewTextViewCnLabel"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="start"
        android:text="Common Name:" />

    <TextView
        android:id="@ id/certViewTextViewCnValue"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_gravity="end"
        android:text="" />

</LinearLayout>

</LinearLayout>
  

Активность пейджера

 <?xml version="1.0" encoding="utf-8"?>
<android.support.v4.view.ViewPager 
xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@ id/certificate_view_pager"
android:layout_width="match_parent"
android:layout_height="match_parent" >

<android.support.v4.view.PagerTitleStrip
    android:id="@ id/pager_title_strip"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_gravity="top"
    android:background="#33b5e5"
    android:paddingBottom="4dp"
    android:paddingTop="4dp"
    android:textColor="#fff" />

</android.support.v4.view.ViewPager>
  

FrameStagePageAdapter

 public class MyViewpageAdapter extends FragmentStatePagerAdapter {

private X509Certificate[] certificateArray;

public void setCertificateArray(X509Certificate[] array) {
    this.certificateArray = array;
}

public MyViewpageAdapter(FragmentManager fm) {
    super(fm);
}

@Override
public Fragment getItem(int i) {
    Fragment certFragment = null;
    if (i > -1 amp;amp; i < certificateArray.length) {
        certFragment = new MyCertificateFragment();
        byte[] encodedCertificate = null;
        try {
            encodedCertificate = certificateArray[i].getEncoded();
        } catch (CertificateEncodingException e) {
            e.printStackTrace();
        }
        Bundle arguments = new Bundle();
        arguments.putByteArray(LoginActivity.INTENT_EXTRA_CERTIFICATE,
        encodedCertificate);
        certFragment.setArguments(arguments);
    }
    return certFragment;
}

@Override
public int getCount() {
    int count = 0;
    if (certificateArray != null){
        count = certificateArray.length;
    }
    return count;
}
  

}

Ответ №1:

Я обнаружил проблему в моем certificate_view.xml . В макете произошла ошибка. У меня была пустая высота для линейного макета. Поэтому certificate_view.xml не удалось загрузить в метод onCreateView.