#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.