Внедряем материальный дизайн

Настало время переходить на Lollipop, друзья. Как бы смешно это не звучало.image

Буквально вчера мы в Surfingbird обновили дизайн приложения и сегодня, по свежим следам, хотелось бы поделиться впечатлениями от перехода на material design.

Подготовительный этап.

Чтобы минимизировать количество проблем, лучше обновить все)

Устанавливаем образ lollipop на свой нексус телефон Обновляем Java до 7 версии, если еще нет Обновляем IDE, мы используем Intellij Idea Обновляем SDK, не забудьте обновить Tools, Platform-tools, Build-tools, Sdk и Support library Внедряем RecyclerView

RecyclerView это новый ViewGroup компонент, который пришел на замену List/GridView. Но он не является их потомком, скорее это альтернативная ветвь эволюции. С одной стороны, это гораздо более гибкий и более эффективно работающий компонент, с другой — в нем из коробки отсутствуют, либо делаются по другому некоторые вещи, к которым мы привыкли в List/GridView (разделители, быстрый скролл, селекторы, хидеры и т.п.).Во-первых, по субъективным ощущениям, скроллинг стал более плавным, чем при использовании listview+viewholder, во-вторых, появилось множество прекрасных штук, так что игра несомненно стоит свеч.

Перейти на этот компонент очень просто. Закидываем в библиотеки соответствующий sdk ▸ extras ▸ android ▸ support ▸ v7 ▸ recyclerview ▸ libs▸ android-support-v7-recyclerview.jar/подключаем в богомерзком gradle или чем вы пользуетесь.

1. Обновляем адаптер, если вы уже использовали view-holder паттерн, то все привычно

Заменяем BaseAdapter (или что там у вас было) на RecyclerView.Adapter

В onCreateViewHolder — парсим layout

@Override public ViewHolder onCreateViewHolder (ViewGroup viewGroup, int i) { View view = layoutInflater.inflate (R.layout.main_adapter_griditem, null); return new ViewHolder (view); } где, собственно ViewHolder — привычная заглушка

class ViewHolder extends RecyclerView.ViewHolder{ private ImageView stgvImageView;

public ViewHolder (View holderView) { super (holderView); stgvImageView = (ImageView) holderView.findViewById (R.id.stgvImageView); } } и переносим логику наполнения view из getView в onBindViewHolder (обращаясь к холдеру — holder.stgvImageView и т.п.)

Удаляем ставшие ненужными методы типа getItem

2. Заменяем ListView на RecyclerView

public RecyclerView gridView;//здесь был Grid/ListView public AdapterMain adapterMain; public ArrayList rows; //Это способ отображения recycleview. Кроме сетки с столбцами переменной высоты есть более канонические //GridLayoutManager (Grid) и LinearLayoutManager (List) public StaggeredGridLayoutManager mLayoutManager;

@Override public View onCreateView (LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { aq = new AQuery (getActivity ()); //Не пугайтесь это просто контейнер final LinearLayout linearLayout = new LinearLayout (getActivity ()); linearLayout.setOrientation (LinearLayout.VERTICAL); linearLayout.setGravity (Gravity.CENTER); gridView = new RecyclerView (getActivity ()); gridView.setHasFixedSize (true); mLayoutManager = new StaggeredGridLayoutManager (UtilsScreen.getDisplayColumns (getActivity ()), StaggeredGridLayoutManager.VERTICAL); //можно задать горизонтальную ориентацию. Будет свежо и необычно. Наверное gridView.setLayoutManager (mLayoutManager); gridView.setItemAnimator (new DefaultItemAnimator ()); //Это новый метод для задания divider //gridView.addItemDecoration (new DividerItemDecoration (getActivity ())); //Этих методов больше нет //gridView.setSmoothScrollbarEnabled (true); //gridView.setDivider (new ColorDrawable (this.getResources ().getColor (R.color.gray_divider))); //gridView.setDividerHeight (UtilsScreen.dpToPx (8)); rows = new ArrayList(); linearLayout.addView (gridView); return linearLayout; } 3. Продолжаем разговор.

Работа с адаптером практически не изменилась.

@Override public void onViewCreated (View view, Bundle savedInstanceState) { super.onViewCreated (view, savedInstanceState); adapterMain = new AdapterMain (getActivity (), rows); gridView.setAdapter (adapterMain); gridView.setOnScrollListener (onScroll); } Стал ненужным метод отключения адаптера на момент изменения (DataSetInvalidated), нотификация об изменении осталась без изменения

adapterMain.notifyDataSetChanged (); if (page == 1) gridView.scrollToPosition (0);//точно не помню как раньше назывался этот метод Изменился метод вычисления последнего видимого элемента (для автоматической подгрузки следующей порции). Предполагаю, что эту логику лучше перенести в адаптер, но, если очень некогда, то можно так, например:

gridView.setOnScrollListener (onScroll); //--- private RecyclerView.OnScrollListener onScroll = new RecyclerView.OnScrollListener () { @Override public void onScrolled (RecyclerView recyclerView, int dx, int dy) { super.onScrolled (recyclerView, dx, dy); int[] visibleItems = ((StaggeredGridLayoutManager) gridView.getLayoutManager ()).findLastVisibleItemPositions (null); int lastitem=0; for (int i: visibleItems) { lastitem = Math.max (lastitem, i); } if (lastitem>0 && lastitem>adapterMain.data.size ()-5 && ! isRunning) { if (! internetIsOver) { refresh (); } } } }; Вообще, скролинг стал более низкоуровневым, теперь можно прямо в этом методе получать информацию куда и насколько проскролено (простите мой английский)

На этом месте у вас все должно заработать. Если, например, нужно добавить разделители, то их можно добавить перекрыв класс DividerItemDecoration, например так: (вертикальные разделители)(Ахтунг, копипаста сами знаете с какого сайта)

public class DividerItemDecoration extends RecyclerView.ItemDecoration {

private static final int[] ATTRS = new int[]{ android.R.attr.listDivider };

private Drawable mDivider; private int offset = 0;

public DividerItemDecoration (Context context) { final TypedArray a = context.obtainStyledAttributes (ATTRS); mDivider = a.getDrawable (0); offset = UtilsScreen.dpToPx (16); a.recycle (); }

@Override public void onDraw (Canvas c, RecyclerView parent) { drawVertical (c, parent); }

public void drawVertical (Canvas c, RecyclerView parent) { final int left = parent.getPaddingLeft (); final int right = parent.getWidth () — parent.getPaddingRight ();

final int childCount = parent.getChildCount (); for (int i = 0; i < childCount; i++) { final View child = parent.getChildAt(i); final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child .getLayoutParams(); final int top = child.getBottom() + params.bottomMargin; final int bottom = top + offset;//mDivider.getIntrinsicHeight(); mDivider.setBounds(left, top, right, bottom); mDivider.draw(c); } }

@Override public void getItemOffsets (Rect outRect, int itemPosition, RecyclerView parent) { outRect.set (0, 0, 0, offset);//mDivider.getIntrinsicHeight ()); } } Но не спешите с этим! Потому что появились прекрасные Карточки!

Внедряем CardView

Помню, когда я был еще совсем молодым android-разработчиком, вышел пинтерест и все офигели. Мы часами разглядывали, как они реализовали карточки переменной высоты, плавающие кнопки (или это было в Path?), не суть важно. Сейчас можно получить неплохо выглядящие карточки (в том числе, переменной высоты и прямо как в пинтерест) буквально в пару строк кода.

Подключаем cardview как library project/прописываем магическую строку в систему сборки, закидываем jar, не забыв обновить версию саппорт лайбрари.

По сути, карточки — это фрейм вокруг вашего лейаута с тенюшками и скругляшками, поэтому просто обрамляем ими ваш лэйаут:

//--- Готово, Милорд!

Теперь, допустим, для планшетной версии задаем отображение в две колонки, а для телефонов в одну:(Ахтунг, копипаста сами знаете с какого сайта)

public static boolean isTablet (Context context) { boolean xlarge = ((context.getResources ().getConfiguration ().screenLayout & Configuration.SCREENLAYOUT_SIZE_MASK) == 4); boolean large = ((context.getResources ().getConfiguration ().screenLayout & Configuration.SCREENLAYOUT_SIZE_MASK) == Configuration.SCREENLAYOUT_SIZE_LARGE); return (xlarge || large); }

public static int getDisplayColumns (Activity activity) { int columnCount = 1; if (isTablet (activity)) { columnCount = 2; } return columnCount; } И задаем для разных устройств разный формат отображения:

mLayoutManager = new StaggeredGridLayoutManager (UtilsScreen.getDisplayColumns (getActivity ()), StaggeredGridLayoutManager.VERTICAL); Должно получиться примерно так: 52e0f92ef41a49eaa7bd2da4d2981b27.png

Некоторые нюансы:

После того, как мы выложили приложение в стор, на некоторых устройствах (почему-то на нексусах) и почему-то в том числе на 4.4.4 — приложение странным образом начало падать в районе саппорт лайбрари (причем на наших телефонах (включая нексусы) все работало). Пришлось отключить proguard, это помогло, но осадок остался. Нам не очень понравился цвет шрифта в дефолтной светлой теме. Он очееень нежен, учитывая то, что на всех андроид устройствах цветопередача нарушена разная, поэтму мы решили перекрыть цвет шрифта на чуть более темный. Отключить тень у акшенбара теперь можно, например, так: getSupportActionBar ().setElevation (0); Приложение не будет работать на бете лоллипоп так же, как не работают на ней и все остальные приложения в лоллипоп дизайне (gmail, пресса) Иконки акшенбара стали меньше. Мы просто перенесли их в папку (xxhdpi) Мы пока решили забить на анимации. Перед глазами гугл-пресса и все вроде дико красиво крутится/вертится/плавает/мигает, но мы еще не готовы к такой решительной анимации. Результат

Глаз разработчика «замылен», сложно сказать получилось хорошо или так себе. Я почему-то ожидал большего, если честно. Динамически падающих тенюшек при скролинге, например, больше магии. А в целом, все получилось чуть свежее. Хотя, конечно, мы еще не до конца ололлипопились. Посмотреть результат можно в маркете.

Я наверняка что-то забыл. Делитесь нюансами, рецептами и советами перехода на лоллипоп в комментариях. Тема актуальная, всем нам будет полезно и интересно.

© Habrahabr.ru