Всем привет! Я Максим Земляникин, пишу на Flutter 5 лет. И должен вам признаться, уже пол года как использую Mix вместо стандартных виджетов. А в этой статье хочу рассказать чем он мне так понравился
Начнем с небольшой базы. Mix - UI фреймворк, построенный поверх Flutter, значительно упрощающий вёрстку. В его основе лежат 3 концепции:
Атрибуты: они задают цвет, размер, отступы и другие параметры UI
Стиль: объединяет в себе список атрибутов
Box: принимает стиль и отрисовывает UI по его атрибутам
Можно представить, что стиль это BoxDecoration
, атрибут это его параметр, например цвет или радиус, а Box
это DecoratedBox
. По началу Mix и был обёрткой вокруг контейнера
А вообще можно еще короче!
Атрибуты не ограничиваются возможностью Container
-а. Есть ещё Opacity
, TextStyle
, Clip
, Visibility
, все виды Transform
и многое другое!
А теперь соединим все сказанное выше и добавим ещё один важный факт. Что будет, если два раза указать один атрибут, например цвет? Тогда применится последний. А это приводит к очень удобным последствиям:
-
Можно сделать базовый стиль с нужными цветами, отступами, тенями. И по надобности переопределять его, добавляя атрибут с другим цветом, отступами и чем угодно ещё
-
Можно удобно, очень удобно, просто невероятно удобно делать условия в стилях. К черту слова, просто посмотрите на код
-
А ещё эти изменения можно проанимировать!
Но вы можете мне возразить, что также просто можно воспользоваться AnimatedContainer-ом. А я напомню, что он не проанимирует Scale, TextStyle, Clip и многое другое! Вообще Mix и flutter_animate делают 90% анимаций донельзя простыми
Лично мне этого было бы уже достаточно. Но Mix состоит не из одного только Box виджета. В нем есть StyledText
, StyledIcon
, StyledImage
, HBox
, VBox
, ZBox
и другие виджеты. Все они умеют работать со стандартными атрибутами и имеют свои собственные
HBox
, VBox
и ZBox
виджеты - аналоги Row
, Column
и Stack
соответственно. Только они принимают параметры в стиле Mix, через атрибуты
Обратите внимание на еще один сахарок от Mix. Чтобы задать атрибуты, но не писать каждый раз $flex
или $stack
, я использовал chain
Styled*
виджеты рассмотрим на примере StyledText
. Его можно настроить атрибутами точно также как и виджет Text
через DefaulTextStyle
. Причем я не зря упомянул именно DefaulTextStyle
, ведь StyledText
тоже ищет свои атрибуты через контекст
Ладно, с основами разобрались. Теперь расскажу про его фишки, которые мне больше всего нравятся
Сахар
Как вы уже могли заметить, Mix подсластили не меньше чем Kotlin в сравнении с Java. Он использует много фишек Dart, типо возможности вызвать класс как функцию, опциональные параметры и chain вызовы методов через ..
. Добавили к этому огромную кучу helper методов, Inherited виджеты, хорошую документацию и покрытие тестами. Поэтому, хоть Mix и позиционирует себя в первую очередь как инструмент для дизайн систем, я нашел его наиболее удобным для повседневной верстки. Ну правда, зачем писать padding: EdgeInsets.symmetric(vertical: 10, horizontal: 8)
, когда можно обойтись $box.padding(10, 8)
? Да и читать такой код тоже проще, меньше шума
Директивы - изменяют содержимое виджета. К примеру $text.upperCase
сделает все буквы в StyledText
виджете заглавными! Или $box.color.withOpacity(0.5)
сделает любой цвет, который вы укажете в атрибуте $box.color
, полупрозрачным
Вложенность становится меньше. Так как в Mix свойства задаются списком атрибутов, элементы UI можно описать парой-тройкой виджетов. К примеру нам нужно задать стиль текста и обернуть в карточку с фоном, тенью и закруглением
Во первых Mix в этом коде выразительнее и короче. Но есть ещё одна важная особенность малой вложенности. Что если понадобится изменить этот виджет? К примеру добавить прозрачность всей карточке. В Mix в список атрибутов добавится $with.opacity(...)
. А вот без него придется обернуть всё в виджет Opacity
и на ревь diff будет не одна строка, а 15! Считаю это не очевидной, но очень важной особенностью
Дизайн системы
Для них Mix и был создан. Сначала расскажу до боли знакомую многим разработчикам историю. На проекте дизайнер нарисовал 3 типа кнопок: outlined, filled и text. В этих кнопках могли быть текст с иконками по бокам и loader. Плюс не активное состояние. Потом добавились маленькие версии этих кнопок: меньше отступы, иконки, шрифт и не растягиваются на ширину экрана. Дизайнера было уже не остановить, он добавил icon button - те же маленькие кнопки, но только с иконкой. И отполировал это постоянными изменениями. Привело это к каше в коде и багам в вёрстке
Как поможет Mix? Во первых стили можно вынести в отдельное место. Во вторых их можно разбить на базовый, filled, outlined и text. В эти стили можно добавить variants: маленький и большой. Это позволит один раз описать decoration, стили текста и иконок, и переиспользовать их в 3 виджетах, которые в свою очередь будут заниматься только layout-ом
Минусы
Дополнительный слой абстракции поверх Flutter. Во первых нужно учить ещё один UI framework, знать в какие виджеты он маппит атрибуты. Во вторых в нем есть свои скрытые концепции. Например он сортирует некоторые атрибуты. Хочу я задать ширину виджету, и поместить его в угол, пишу Style($with.sizedBox(), $with.align())
. А Mix меняет их местами, помещает виджет в углу SizedBox
, а не наоборот. И узнаю я об этом только запустив приложение. А почему так происходит выясню через пол часа, изучив все issues и код пакета
Некоторые вещи можно сделать несколькими способами. Отступ можно задать через $box.padding
или $with.padding
. Размер через $box.height
и $box.width
или $with.sizedBox
. причем второй вариант более гибкий, например в него можно передать Size. Если задаешь через $box
, то в итоге оно превращается в Container
, соответственно и разницы в порядке $box
атрибутов нет. А в случае $with
атрибутов порядок важен, но не со всеми, ведь некоторые он отсортирует. Иногда думаешь, какой бы атрибут использовать, разумеется выбираешь неправильный, переписываешь
Вместо заключения
Вот вы и дочитали до конца, а значит что-то вас зацепило. Дальше советую углубиться в документацию или посмотреть выпуск Observable Flutter про Mix (на английском).
Комментарии (6)
ChessMax
23.12.2024 10:53В пункте о дизайн системе было бы неплохо прямо код показать как эти три вида кнопок в двух размерах будут выглядеть. Знак доллара очень неприятно смотрится (привет PHP). Фактически это шум. В дарт такое не принято. Есть ли в этом какой-то скрытый смысл?
Задумка интересная, но конкретную реализацию, я бы в свой проект не потащил.Maksimka101 Автор
23.12.2024 10:53Хотел бы про дизайн систему рассказать подробнее. Но пришлось бы еще больше новых концепций вводить, кода добавлять. В общем тяжеловато читать и писать)
Знак доллара тоже по началу смущал. Но в принципе он хорошо отделяет Mix от остального кода. Пишешь $ и получаешь лишь нужные подсказки. И в коде всякие box, with, on и прочие не мешаются. Да и никто не мешает сделать свои top level переменные без доллара, типоfinal box = $box;
Интересно, почему конкретно Mix не стал бы использовать? Он развивается 2 года. Апи хорошо продуман. У создателей есть виденье проекта. Что попало не дают вливать, я пробовал :) А еще у них 1325 тестов!
lil_master
23.12.2024 10:53Знак доллара очень неприятно смотрится (привет PHP). Фактически это шум.
Очередной господин, несостоявшийся в PHP. PHP - отличный язык и знак доллара там очень кстати! Знак доллара есть много где, кроме PHP: perl, jQuery и в дарте он тоже есть и ничем не бесит.
Фактически бессмысленный шум - это ваш комментарий.
Автору: отличная статья, благодарю, пишите ещё такое интересненькое! Кто не имеет достаточно опыта в написании разметки на flutter - естественно не поймет плюсы Mix.
PackRuble
23.12.2024 10:53Есть бенчмарки производительности стандартной вёрстки и mix-верстки?
Думаю, есть один существенный минус: обычно flutter-разработчики привыкли верстать в flutter-стиле. Поэтому новым разработчикам придётся учиться верстать с помощью mix, на что потребуется время и их большая мотивация ломать себе мозг новыми приколами.
Ещё: иногда чтобы сверстать что-то сложное (и может даже с вычурной анимацией) нужно немало времени и знаний. Mix действительно настолько классный, что позволит это сделать, или придётся от него отказываться в таких случаях? (в итоге получаем код вёрстки, в котором вот здесь mix, а здесь ваниль)
Спасибо, что рассказали о нём, впервые слышу. На первый взгляд он выглядит чужеродно.
Maksimka101 Автор
23.12.2024 10:53Бенчмарков не видел. Но Flutter приложения (почти никогда) не лагают из-за виджетов. Это же лишь описание UI, считай Data class. Они лагают из-за использования неоптимальных виджетов: жесткого блюра, или прозрачности. А Mix - это лишь другой способ описать дерево виджетов. В нем прямолинейная логика: читаем параметры из $box и передаем их в Container. Получается к цепочке Widget -> Element -> RenderObject добавляется Mix -> Widget. Так что в производительности проблем точно не ожидаю)
Второй ваш тезис на 100% верный. Новых разработчиков наверняка нужно будетмучить. И что? В любом случае они будут что-то учить. Все проекты разные, везде есть особенности, не все используют встроенный ChangeNotifier
Все ли можно сделать лишь с Mix? Нет конечно. Scaffold, AppBar, и многие другие виджеты все равно придется использовать. И что? Mix не стремится заменить все виджеты. Зато он упрощает 80% связанного с верской кода. Мне достаточно)
Рад что статья оказалась полезна :)
VailStrawb
Очень интересно, спасибо за статью : )