Разработка Stories (исчезающий контент) в мобильном приложении
Stories — это UI-паттерн с нетривиальной механикой: прогресс-бары, переключение по тапу, пауза при удержании, swipe-вниз для закрытия, и всё это с медиаконтентом, который нужно предзагружать без задержек. Плюс таймер на стороне сервера — контент исчезает через 24 часа.
Прогресс-бары и навигация
Горизонтальные прогресс-бары вверху — UIView с анимацией frame.width через CADisplayLink (iOS) или ValueAnimator (Android). CADisplayLink даёт 60/120 FPS синхронизацию с экраном — анимация плавная. Таймер через Timer.scheduledTimer — дёргается, особенно при нагрузке.
Длительность одной stories — для фото обычно 5–7 секунд, для видео — длина клипа (но не более 15 секунд). Прогресс-бар для видео нужно синхронизировать с AVPlayer.currentTime() через addPeriodicTimeObserver, не с системным таймером.
Навигация: тап в правую половину экрана — следующий сторис, в левую — предыдущий. UITapGestureRecognizer с проверкой location.x > bounds.width / 2. Удержание — UILongPressGestureRecognizer с minimumPressDuration: 0.1: при .began паузируем анимацию и AVPlayer, при .ended — возобновляем.
Swipe-вниз для закрытия — интерактивный dismiss через UIPanGestureRecognizer. Масштабирование и перемещение экрана вслед за жестом (transform), при отпускании — либо полное закрытие (dismiss), либо возврат (UIViewPropertyAnimator).
Предзагрузка медиа
Задержка при переключении сторис — главный UX-баг. Пользователь тапает — ждёт 1–2 секунды, пока загружается фото или видео. Правильное решение: предзагружать следующие 2–3 сторис.
Для фото — SDWebImage.prefetchURLs() или Kingfisher ImagePrefetcher. Для видео — AVAsset.loadValuesAsynchronously(forKeys: ["playable"]) + AVPlayerItem создаём заранее, AVPlayer не запускаем до показа.
На Android — Coil ImageLoader.enqueue(ImageRequest) для фото-предзагрузки, ExoPlayer с ConcatenatingMediaSource для видео: добавляем следующий элемент в очередь, ExoPlayer предзагружает буфер.
Кэшируем предзагруженные ресурсы в памяти (не только на диске) — повторный просмотр сторис в рамках сессии должен быть мгновенным.
Создание сторис
Фото — PHPickerViewController + опциональный редактор (текст, стикеры). Текст поверх изображения: UITextView поверх UIImageView, при сохранении — объединяем в UIGraphicsImageRenderer. Стикеры — UIView с UIPanGestureRecognizer + UIPinchGestureRecognizer + UIRotationGestureRecognizer.
Видео — запись через AVCaptureSession прямо в stories-камере (как в Instagram/Snapchat) или выбор из галереи. Запись с ограничением 15 секунд — AVCaptureMovieFileOutput.maxRecordedDuration. Удержание кнопки для записи, отпускание — стоп.
После создания — upload в фоне. Stories появляется в ленте сразу с placeholder, заменяется на реальный контент после загрузки.
Таймер исчезновения
24-часовой таймер — на стороне сервера, не клиента. Клиент просто запрашивает список актуальных сторис. Сервер фильтрует по created_at + 24h < now.
На клиенте для уже загруженных сторис из кэша — проверяем expiresAt при открытии. Если истёк — не показываем кэш, запрашиваем сервер (или удаляем из локального хранилища).
Просмотры — отмечаем на сервере через отдельный API-запрос после показа сторис. Батчим отметки: не один запрос на каждый просмотр, а накапливаем за сессию и отправляем при сворачивании приложения или каждые 30 секунд.
Сроки
Просмотр сторис с прогресс-барами, предзагрузкой, навигацией и таймером исчезновения — 3–4 дня. Создание с камерой, текстом и стикерами + upload — ещё 2–3 дня. Стоимость рассчитывается индивидуально.







