Реализация поддержки масштабирования шрифтов в Android
Android позволяет пользователю изменить масштаб шрифта в Settings → Display → Font Size. Диапазон: от 0.85x до 2.0x на большинстве стоков (Samsung One UI позволяет до 1.6x через стандартный UI, но fontScale в Configuration может достигать 2.0x). sp единицы — масштабируются; dp — нет.
Основные сломы
sp vs dp для текста
android:textSize="16sp" — правильно, шрифт масштабируется. android:textSize="16dp" — не масштабируется. Хардкод в dp — одна из самых частых ошибок, которую Lint не всегда ловит (он предупреждает, но не как Error).
В Compose: fontSize = 16.sp — масштабируется. fontSize = 16.dp — Lint выдаст ошибку компиляции начиная с Compose 1.3. Но TextUnit.Unspecified в кастомных компонентах — снова проблема.
Фиксированная высота контейнеров
Самая распространённая проблема: android:layout_height="48dp" на TextView или контейнере с текстом. При fontScale = 2.0 16sp текст занимает ~64dp высоты. Контейнер в 48dp — обрезка.
Правильно: wrap_content для высоты всех текстовых элементов. Для ConstraintLayout — убрать фиксированную высоту у цепочек с текстом.
minHeight можно оставить для touch target (минимум 48dp по Material Design) — но только через minHeight, не height.
Строки с вставками
String.format("Вы заработали %d очков", points) — при длинном числе строка удлиняется. Но проблема не в числе, а в том, что вокруг него фиксированный layout. ConstraintLayout с wrap_content справляется; вложенные LinearLayout с gravity — нет.
Compose: LocalDensity и масштаб
В Jetpack Compose LocalDensity содержит и density (DPI), и fontScale. Если вручную конвертировать размеры через density.density без учёта fontScale — кастомные компоненты с TextUnit-расчётами не масштабируются.
with(LocalDensity.current) { 16.sp.toPx() } — возвращает пиксели уже с учётом fontScale. Если используете это значение как высоту Canvas — всё хорошо. Если игнорируете результат и задаёте фиксированный height в Modifier — нет.
Нетипичный кейс: отключение масштабирования для отдельных элементов
Иногда дизайн требует, чтобы конкретный элемент (логотип с текстом, декоративная надпись) не масштабировался. В Compose: CompositionLocalProvider(LocalDensity provides Density(density = LocalDensity.current.density, fontScale = 1f)) — переопределяет fontScale внутри поддерева на 1.0. Это допустимо для декоративных элементов, но не для функциональных текстов.
В XML: android:textSize="16dp" — технически работает, но Lint ругается. Лучше оберните в стиль с суффиксом NonScalable чтобы это было явно.
Тестирование
adb shell settings put system font_scale 2.0 — мгновенно меняет масштаб без перезапуска устройства. После проверки: adb shell settings put system font_scale 1.0.
Android Studio Device Emulator: Pixel 6 с fontScale 2.0 — первый стенд для проверки layout. Реальный Samsung с One UI — второй стенд, там поведение отличается.
Espresso snapshot-тесты с InstrumentationRegistry.getInstrumentation().targetContext.resources.configuration.fontScale = 2f — для регрессионного контроля.
Срок: 2-3 дня. Стоимость рассчитывается индивидуально.







