with viewpager studio fragments example android view pager android-viewpager viewpage

viewpager - tabs android



Concéntrate Vista desde ViewPager (10)

¿Me estoy perdiendo de algo? Solo hay 3 niños dentro del ViewGroup, por lo que se reduce a:

int current = viewPager.getCurrentItem(); int childCount = viewPager.getChildCount(); // If there''s a single page or we''re at the beginning, return the first view if (childCount == 1 || current == 0) { return viewPager.getChildAt(0); } else { //For any other case, we want the second child. This is either the last page or the page in the middle for any other case. return viewPager.getChildAt(1); }

utilizo ViewPager para cambiar de vista con deslizamiento hacia la izquierda / derecha.

El ViewPager necesita un adaptador, así que lo he construido:

public class ListViewPagerAdapter extends PagerAdapter { protected static final String TAG = "ListViewPagerAdapter"; protected static final int NUM_VIEWS = 3; protected final Activity mActivity; public ListViewPagerAdapter(Activity activity) { mActivity = activity; } @Override public int getCount() { return NUM_VIEWS; } @Override public void startUpdate(View container) {} @Override public Object instantiateItem(View container, int position) { // ViewPager ViewPager viewPager = (ViewPager) container; // Wird verwendet, um die Views aufzurufen LayoutInflater layoutInflater = mActivity.getLayoutInflater(); // Standardmäßig ist news eingeblendet View view = layoutInflater.inflate(R.layout.news_fragment, null); // Falls sich die Position verändert, so verändert sich auch die View if (position == 0) { view = layoutInflater.inflate(R.layout.favorite_fragment, null); } else if (position == 2) { view = layoutInflater.inflate(R.layout.videos_fragment, null); } // View einblenden viewPager.addView(view, 0); return view; } @Override public void destroyItem(View container, int position, Object object) { // ViewPager ViewPager viewPager = (ViewPager) container; // View View view = (View) object; // View löschen viewPager.removeView(view); } @Override public void finishUpdate(View container) {} @Override public boolean isViewFromObject(View view, Object object) { View _view = (View) object; return view == _view; } @Override public Parcelable saveState() { return null; } @Override public void restoreState(Parcelable state, ClassLoader loader) {} }

Ahora, quiero obtener la vista enfocada actual del viewpager. Intenté getChildAt (x), pero no funciona.

¿Hay algún ejemplo o tienes alguna idea de cómo obtener la vista actual?

Gracias


Debes registrar un oyente en tu ViewPager :

pager.setOnPageChangeListener(new MyPageChangeListener());

Debe personalizar el oyente ampliando un oyente de código auxiliar:

private int focusedPage = 0; private class MyPageChangeListener extends ViewPager.SimpleOnPageChangeListener { @Override public void onPageSelected(int position) { focusedPage = position; } }

Encontré esto mirando el código fuente de ViewPager.java en la biblioteca de compatibilidad . Leí que podemos hacer más, por ejemplo, coger en onPageScrollStateChanged .

Para construir el adaptador, utilicé el código en esta publicación de blog . Es posible que desee echar un vistazo.


Dentro de tu FragmentStatePagerAdapter :

private View mCurrentView; @Override public void setPrimaryItem(ViewGroup container, int position, Object object) { Fragment f = (Fragment) object; mCurrentView = f.getView(); }



Para obtener la vista enfocada, uso de esta manera:

Cuando infla mi vista antes de agregarla a ViewPager, le configuro la etiqueta. Luego verifico esta etiqueta.

private View getCurrentView() { for (int i = 0; i < pager.getChildCount(); i++) { View v = pager.getChildAt(i); if (v != null) { if (v.getTag().equals(pageList.get(pager.getCurrentItem()).getTag())) return v; // pageList is a list of pages that I pass to page adapter } } return null; }


Prueba esto:

public View getCurrentView(ViewPager pager) { for (int i = 0; i < pager.getChildCount(); i++) { View child = pager.getChildAt(i); if (child.getX() <= pager.getScrollX() + pager.getWidth() && child.getX() + child.getWidth() >= pager.getScrollX() + pager.getWidth()) { return child; } } return getChildAt(0); }


Puede agregar una etiqueta a la vista creada en el método instantiateItem:

view.setTag(position);

Más tarde puede acceder a la vista seleccionada actualmente por:

mPager.findViewWithTag(mPager.getCurrentItem());


Recientemente necesité implementar esta solución exacta. Así es como lo hice:

Map<Integer, Object> views = Maps.newHashMap(); @Override public Object instantiateItem(View container, int position) { /* Create and add your View here */ Object result = ??? views.put(position, result); return result; } @Override public void destroyItem(View container, int position, Object object) { /** Remove your View here */ views.remove(position); } protected View findViewForPosition(int position) { Object object = views.get(position); if (object != null) { for (int i = 0; i < getChildCount(); i++) { View view = getChildAt(i); if (isViewFromObject(view, object)) { return view; } } } return null; }


Si lo examina detenidamente, hay como máximo 3 visualizaciones guardadas por ViewPager. Puede obtener fácilmente la vista actual

view = MyActivity.mViewPager.getChildAt(1);


las respuestas dadas no eran realmente adecuadas para mí, todas tienen efectos secundarios no deseados. O tengo que agregar variables innecesarias (me gusta el código limpio) o tengo que trabajar con el framework de Android.

Mi solución se basa en la reflexión, que accede a la lista de matriz de todos los objetos dentro de ViewPager y devuelve el objeto seleccionado actual en ViewPager. Casi no tiene efectos secundarios (la reflexión es lenta, subclases de una clase existente) y mantiene el código limpio.

public class ViewPagerEx extends ViewPager { private static final String TAG = ViewPagerEx.class.getSimpleName(); public ViewPagerEx(Context context, AttributeSet attrs) { super(context, attrs); } public ViewPagerEx(Context context) { super(context); } public Object getCurrentObject() { try { final Field itemsField = ViewPager.class.getDeclaredField("mItems"); itemsField.setAccessible(true); final ArrayList<Object> items = (ArrayList<Object>) itemsField.get(this); final int currentItemIndex = getCurrentItem(); if (currentItemIndex < 0 || currentItemIndex >= items.size()) { return null; } final Object infoItem = items.get(getCurrentItem()); final Class<?> itemInfoClass = findItemInfoClass(ViewPager.class.getDeclaredClasses()); final Field objectField = itemInfoClass.getDeclaredField("object"); objectField.setAccessible(true); return objectField.get(infoItem); } catch (NoSuchFieldException e) { Log.e(TAG, e.toString()); } catch (IllegalArgumentException e) { Log.e(TAG, e.toString()); } catch (IllegalAccessException e) { Log.e(TAG, e.toString()); } return null; } private Class<?> findItemInfoClass(final Class<?>[] classes) throws IllegalArgumentException { for (int i = 0; i < classes.length; i++) { if (classes[i].getSimpleName().equals("ItemInfo")) { return classes[i]; } } throw new IllegalArgumentException("cannot find class ItemInfo"); } }