Разработка расширения iMessage для iOS-приложения
iMessage-расширение — это отдельный Extension target внутри iOS-приложения, который встраивается в приложение «Сообщения». Пользователь открывает ящик с приложениями под полем ввода и запускает ваш мини-интерфейс прямо в переписке. Стикеры, интерактивные карточки, совместные игры, передача данных через сообщения — всё это без выхода из чата.
Два режима: Sticker Pack vs MSMessagesAppViewController
Sticker Pack — простейший вариант. Создаётся без кода: добавляем Extension target типа «Sticker Pack Application», кидаем PNG/APNG/GIF в Stickers.xcassets. Apple сам генерирует UI. Размеры: Small (100×100 pt), Regular (136×136 pt), Large (206×206 pt). APNG для анимации — до 500 КБ, GIF — до 25 МБ, но GIF нежелательно из-за качества.
MSMessagesAppExtension — полноценный Extension с UIKit/SwiftUI. Здесь MSMessagesAppViewController — точка входа. Два режима отображения: Compact (снизу экрана, ≈225 pt высота) и Expanded (почти полный экран). Переключение через requestPresentationStyle(.expanded).
Самое важное — MSMessage и MSMessageLayout. Это объект сообщения, который вставляется в чат. Имеет URL (с кастомной схемой приложения), MSMessageTemplateLayout с изображением, заголовком, подзаголовком. Когда получатель нажимает на такое сообщение — у него открывается то же расширение (если установлено) или приглашение установить приложение.
Проблемы, которые находят не сразу
Shared state между двумя участниками. Если хочется передавать данные (ход в игре, выбор в опросе), нельзя использовать UserDefaults или Keychain напрямую — у получателя другой device. Данные кодируются в URL сообщения через query параметры или base64 в path. Размер URL ограничен — не злоупотребляйте.
Паттерн: MSMessage.url = URL(string: "yourapp://state?data=\(encodedState)"). Получатель парсит URL в willBecomeActive(with:).
App Group и shared container. Extension изолирован от основного приложения по умолчанию. Если нужно читать данные приложения в расширении (пользовательские стикеры, авторизация), настраиваем App Group: одинаковый group.com.yourapp.shared в Capabilities основного таргета и расширения. Затем UserDefaults(suiteName:) и FileManager.containerURL(forSecurityApplicationGroupIdentifier:).
MSMessagesAppViewController не поддерживает все UIKit-компоненты. UIActivityViewController внутри iMessage Extension крашится — нельзя шарить контент через системный шер. UIImagePickerController — тоже нельзя. Для выбора изображений нужен PHPickerViewController.
Память. Extension работает в строгих лимитах памяти (~100 МБ). Загрузка тяжёлых ресурсов, кэшей изображений без ограничения — прямой путь к EXC_RESOURCE крашу. Используем NSCache с countLimit и totalCostLimit.
Что проверяем при разработке
- Расширение работает корректно в compact и expanded режиме
- Сообщения корректно декодируются у получателя (тест с двумя реальными устройствами)
- App Group настроен если нужны данные основного приложения
- Стикеры правильных размеров и форматов, анимации не превышают лимиты
- Extension не крашится при получении сообщения без установленного приложения
Сроки
Sticker Pack без кода: 1–3 дня (дизайн + сборка). iMessage Extension с интерактивными карточками: 3–5 недель. Мультиплеерная игра или сложный data exchange: 6–10 недель. Стоимость рассчитывается после анализа сценариев взаимодействия.







