Изображение перехода общего элемента становится белым

Я использую переход общего элемента RecyclerView в ViewPager. Проблема заключается в том, что когда просмотрщик перенаправляется на другое изображение, после возврата к просмотру повторного использования изображение, которое сначала было анимировано, становится белым.

Изображение перехода общего элемента становится белым

Запуск вьюпейджера:

FragmentTransaction fragmentTransaction = getSupportFragmentManager()
            .beginTransaction()
            .setReorderingAllowed(true)
            .addSharedElement(view, name)
            .hide(recyclerFragment)
            .add(R.id.main_frameLayout, viewpagerFragment, tag)
            .commit();

Возвращаясь к фрагменту recyclerview:

FragmentTransaction fragmentTransaction = getSupportFragmentManager()
            .beginTransaction()
            .setReorderingAllowed(true)
            .addSharedElement(view, name)
            .remove(viewpagerFragment)
            .show(recyclerFragment)
            .commit();

Редактировать: Обратите внимание, что я знаю позиции и правильно устанавливаю имена переходов.

Вам удалось ее решить?

Elad Finish 10.10.2021 14:57

@EladFinish Да. Принятый ответ правильный. Однако моя проблема заключалась в том, что я понял, что мой код использует не RecyclerView, а 3 ImageViews.

Amir 11.10.2021 08:03
3
2
2 020
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Я создал для вас образец примера с «Recyclerview + ViewPager + SharedTransition».

Сначала позвольте мне рассказать вам о проблеме в вашем коде: вы добавляете и удаляете фрагмент вместо его замены. при запуске viewpager и возвращении из viewpager. Также вы должны взять целочисленную переменную, которая будет содержать текущую позицию recyclerview, а также viewpager.

Ниже приведен пример GIF, чтобы показать, чего я добился:

Сначала возьмите «currentPosition» как int в вашей MainActivity, как показано ниже:

public static int currentPosition;

Затем создайте абстрактный класс «ImageData», который будет содержать массив изображений, как показано ниже:

abstract class ImageData {
    static final int[] IMAGE_DRAWABLES = {
            R.drawable.android1,
            R.drawable.android2,
            R.drawable.android3,
            R.drawable.android4,
            R.drawable.android5,
    };
}

Теперь создайте «GridAdapter», который будет отображать список изображений в recyclerView:

/**
 * A fragment for displaying a grid of images.
 */
public class GridAdapter extends RecyclerView.Adapter<ImageViewHolder> {

    /**
     * A listener that is attached to all ViewHolders to handle image loading events and clicks.
     */
    private interface ViewHolderListener {

        void onLoadCompleted(ImageView view, int adapterPosition);

        void onItemClicked(View view, int adapterPosition);
    }

    private final RequestManager requestManager;
    private final ViewHolderListener viewHolderListener;

    /**
     * Constructs a new grid adapter for the given {@link Fragment}.
     */
    public GridAdapter(Fragment fragment) {
        this.requestManager = Glide.with(fragment);
        this.viewHolderListener = new ViewHolderListenerImpl(fragment);
    }

    @Override
    public ImageViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View view = LayoutInflater.from(parent.getContext())
                .inflate(R.layout.image_card, parent, false);
        return new ImageViewHolder(view, requestManager, viewHolderListener);
    }

    @Override
    public void onBindViewHolder(ImageViewHolder holder, int position) {
        holder.onBind();
    }

    @Override
    public int getItemCount() {
        return IMAGE_DRAWABLES.length;
    }

    private static class ViewHolderListenerImpl implements ViewHolderListener {

        private Fragment fragment;
        private AtomicBoolean enterTransitionStarted;

        ViewHolderListenerImpl(Fragment fragment) {
            this.fragment = fragment;
            this.enterTransitionStarted = new AtomicBoolean();
        }

        @Override
        public void onLoadCompleted(ImageView view, int position) {
            // Call startPostponedEnterTransition only when the 'selected' image loading is completed.
            if (MainActivity.currentPosition != position) {
                return;
            }
            if (enterTransitionStarted.getAndSet(true)) {
                return;
            }
            fragment.startPostponedEnterTransition();
        }

        @Override
        public void onItemClicked(View view, int position) {
            // Update the position.
            MainActivity.currentPosition = position;

            // Exclude the clicked card from the exit transition (e.g. the card will disappear immediately
            // instead of fading out with the rest to prevent an overlapping animation of fade and move).
            ((TransitionSet) fragment.getExitTransition()).excludeTarget(view, true);

            ImageView transitioningView = view.findViewById(R.id.card_image);
            fragment.getFragmentManager()
                    .beginTransaction()
                    .setReorderingAllowed(true) // Optimize for shared element transition
                    .addSharedElement(transitioningView, transitioningView.getTransitionName())
                    .replace(R.id.fragment_container, new ImagePagerFragment(), ImagePagerFragment.class
                            .getSimpleName())
                    .addToBackStack(null)
                    .commit();
        }
    }

    /**
     * ViewHolder for the grid's images.
     */
    static class ImageViewHolder extends RecyclerView.ViewHolder implements
            View.OnClickListener {

        private final ImageView image;
        private final RequestManager requestManager;
        private final ViewHolderListener viewHolderListener;

        ImageViewHolder(View itemView, RequestManager requestManager,
                        ViewHolderListener viewHolderListener) {
            super(itemView);
            this.image = itemView.findViewById(R.id.card_image);
            this.requestManager = requestManager;
            this.viewHolderListener = viewHolderListener;
            itemView.findViewById(R.id.card_view).setOnClickListener(this);
        }

        void onBind() {
            int adapterPosition = getAdapterPosition();
            setImage(adapterPosition);
            // Set the string value of the image resource as the unique transition name for the view.
            image.setTransitionName(String.valueOf(IMAGE_DRAWABLES[adapterPosition]));
        }

        void setImage(final int adapterPosition) {
            // Load the image with Glide to prevent OOM error when the image drawables are very large.
            requestManager
                    .load(IMAGE_DRAWABLES[adapterPosition])
                    .listener(new RequestListener<Drawable>() {
                        @Override
                        public boolean onLoadFailed(@Nullable GlideException e, Object model,
                                                    Target<Drawable> target, boolean isFirstResource) {
                            viewHolderListener.onLoadCompleted(image, adapterPosition);
                            return false;
                        }

                        @Override
                        public boolean onResourceReady(Drawable resource, Object model, Target<Drawable>
                                target, DataSource dataSource, boolean isFirstResource) {
                            viewHolderListener.onLoadCompleted(image, adapterPosition);
                            return false;
                        }
                    })
                    .into(image);
        }

        @Override
        public void onClick(View view) {
            // Let the listener start the ImagePagerFragment.
            viewHolderListener.onItemClicked(view, getAdapterPosition());
        }
    }

}

Теперь создайте адаптер для viewpager, который будет отображать изображения в viewpager после нажатия элемента из recyclerView:

public class ImagePagerAdapter extends FragmentStatePagerAdapter {

  public ImagePagerAdapter(Fragment fragment) {
    // Note: Initialize with the child fragment manager.
    super(fragment.getChildFragmentManager());
  }

  @Override
  public int getCount() {
    return IMAGE_DRAWABLES.length;
  }

  @Override
  public Fragment getItem(int position) {
    return ImageFragment.newInstance(IMAGE_DRAWABLES[position]);
  }
}

Сначала создайте xml и переходы для «ImagePagerFragment», как показано ниже:

fragment_image.xml

<ImageView
    xmlns:android = "http://schemas.android.com/apk/res/android"
    android:id = "@+id/image"
    android:layout_width = "match_parent"
    android:layout_height = "match_parent"
    android:contentDescription = "@string/image_description"
    android:scaleType = "fitCenter"/>

создайте папку tansition в каталоге res:

image_shared_element_transition.xml

<?xml version = "1.0" encoding = "utf-8"?>
<transitionSet xmlns:android = "http://schemas.android.com/apk/res/android"
    android:duration = "375"
    android:interpolator = "@android:interpolator/fast_out_slow_in"
    android:transitionOrdering = "together">
    <changeClipBounds />
    <changeTransform />
    <changeBounds />
</transitionSet>

grid_exit_transition.xml

<?xml version = "1.0" encoding = "utf-8"?>
<transitionSet xmlns:android = "http://schemas.android.com/apk/res/android"
    android:duration = "375"
    android:interpolator = "@android:interpolator/fast_out_slow_in"
    android:startDelay = "25">
    <fade>
        <targets android:targetId = "@id/card_view" />
    </fade>
</transitionSet>

Теперь создайте «ImagePagerFragment» для отображения изображений в viewPager.

/**
 * A fragment for displaying a pager of images.
 */
public class ImagePagerFragment extends Fragment {

    private ViewPager viewPager;

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container,
                             @Nullable Bundle savedInstanceState) {
        viewPager = (ViewPager) inflater.inflate(R.layout.fragment_pager, container, false);
        viewPager.setAdapter(new ImagePagerAdapter(this));
        // Set the current position and add a listener that will update the selection coordinator when
        // paging the images.
        viewPager.setCurrentItem(MainActivity.currentPosition);
        viewPager.addOnPageChangeListener(new ViewPager.SimpleOnPageChangeListener() {
            @Override
            public void onPageSelected(int position) {
                MainActivity.currentPosition = position;
            }
        });

        prepareSharedElementTransition();

        // Avoid a postponeEnterTransition on orientation change, and postpone only of first creation.
        if (savedInstanceState == null) {
            postponeEnterTransition();
        }

        return viewPager;
    }

    /**
     * Prepares the shared element transition from and back to the grid fragment.
     */
    private void prepareSharedElementTransition() {
        Transition transition =
                TransitionInflater.from(getContext())
                        .inflateTransition(R.transition.image_shared_element_transition);
        setSharedElementEnterTransition(transition);

        // A similar mapping is set at the GridFragment with a setExitSharedElementCallback.
        setEnterSharedElementCallback(
                new SharedElementCallback() {
                    @Override
                    public void onMapSharedElements(List<String> names, Map<String, View> sharedElements) {
                        // Locate the image view at the primary fragment (the ImageFragment that is currently
                        // visible). To locate the fragment, call instantiateItem with the selection position.
                        // At this stage, the method will simply return the fragment at the position and will
                        // not create a new one.
                        Fragment currentFragment = (Fragment) viewPager.getAdapter()
                                .instantiateItem(viewPager, MainActivity.currentPosition);
                        View view = currentFragment.getView();
                        if (view == null) {
                            return;
                        }

                        // Map the first shared element name to the child ImageView.
                        sharedElements.put(names.get(0), view.findViewById(R.id.image));
                    }
                });
    }
}

Теперь создайте «ImageFragment», который будет открываться как элемент viewpager:

public class ImageFragment extends Fragment {

    private static final String KEY_IMAGE_RES = "com.key.imageRes";

    public static ImageFragment newInstance(@DrawableRes int drawableRes) {
        ImageFragment fragment = new ImageFragment();
        Bundle argument = new Bundle();
        argument.putInt(KEY_IMAGE_RES, drawableRes);
        fragment.setArguments(argument);
        return fragment;
    }

    @Nullable
    @Override
    public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container,
                             @Nullable Bundle savedInstanceState) {
        final View view = inflater.inflate(R.layout.fragment_image, container, false);

        Bundle arguments = getArguments();
        @DrawableRes int imageRes = arguments.getInt(KEY_IMAGE_RES);

        // Just like we do when binding views at the grid, we set the transition name to be the string
        // value of the image res.
        view.findViewById(R.id.image).setTransitionName(String.valueOf(imageRes));

        // Load the image with Glide to prevent OOM error when the image drawables are very large.
        Glide.with(this)
                .load(imageRes)
                .listener(new RequestListener<Drawable>() {
                    @Override
                    public boolean onLoadFailed(@Nullable GlideException e, Object model, Target<Drawable>
                            target, boolean isFirstResource) {
                        // The postponeEnterTransition is called on the parent ImagePagerFragment, so the
                        // startPostponedEnterTransition() should also be called on it to get the transition
                        // going in case of a failure.
                        getParentFragment().startPostponedEnterTransition();
                        return false;
                    }

                    @Override
                    public boolean onResourceReady(Drawable resource, Object model, Target<Drawable>
                            target, DataSource dataSource, boolean isFirstResource) {
                        // The postponeEnterTransition is called on the parent ImagePagerFragment, so the
                        // startPostponedEnterTransition() should also be called on it to get the transition
                        // going when the image is ready.
                        getParentFragment().startPostponedEnterTransition();
                        return false;
                    }
                })
                .into((ImageView) view.findViewById(R.id.image));
        return view;
    }
}

Все сделано здесь! Вы можете запустить этот код и получить то, что я добился в приведенном выше gif. Здесь вы можете видеть, что я не добавляю и не удаляю фрагменты. Просто замените текущий фрагмент общим переходным элементом.

Note: In this example i have only take 5 items here. if you want to take extra items and load dynamic urls then you can also do with this.

Спасибо, что нашли время и полный и подробный ответ. К сожалению, я не могу использовать replace при запуске viewpager, потому что мой первый фрагмент представляет собой сложную страницу, и ее замена завершит ее жизненный цикл, поэтому возвращение назад и повторная инициализация с самого начала обходится дорого. Есть ли какое-нибудь решение с hide и show, чтобы избежать белого изображения?

Amir 08.04.2019 06:26

так что другой вариант будет таким, как если бы вы могли 1. отсоединить фрагмент, 2. попытаться сохранить изображение в кеше с помощью glide в recyclerview, 3. установить лимит просмотра страницы вне экрана на три

android 08.04.2019 06:59

Я получаю тот же результат, используя detach и attach. Если бы был способ обновить состояние изображения до нормального, проблема была бы решена. Я не знаю, как переход превращает изображение в белое. Я пробовал setVisibility, setAlpha и requestLayout, чтобы обновить вид.

Amir 08.04.2019 08:16

да. Я использую библиотеку Picasso и кэширую изображения. Я тоже пробовал setOffscreenPageLimit.

Amir 08.04.2019 08:22

можете ли вы поделиться всем файлом класса как для viewpager, так и для recyclerview? возможно, я могу помочь тебе

android 08.04.2019 08:42

Извините, код и классы слишком сложны, и мне не разрешено делиться всем проектом.

Amir 08.04.2019 09:12

Другие вопросы по теме