Kotlin/Swift (iOS) Туда и Обратно
36 subscribers
129 photos
21 videos
9 files
86 links
Канал - журнал, рассказывающий об опыте изучения Swift & iOS backend-разработчиком на Java & Kotlin
Download Telegram
Как вам иконка приложения? Все опросы анонимные.
Anonymous Poll
0%
5
80%
4
20%
3
0%
2
0%
1
DerDieDas

Сегодня продолжал делать приложение для тренировки артиклей.
Думаю это хорошая альтернатива todo-list.

📄 Постараюсь на этой неделе написать отдельный пост с планами на это приложение, где опишу mvp (минимально жизнеспособный продукт) и, возможно, дальнейшие планы.

❗️ Пока главная цель зарелизить простое приложение (а может оно и совсем не простое?), понять цикл разработки, собрать фидбэк.
Please open Telegram to view this post
VIEW IN TELEGRAM
👍1
Пока в планах иметь CloudKit базу с словами. Но я сразу подумал о ситуации, когда пользователь установил приложение, но ни разу им не пользовался. Потом где-то в лесу 🌲🌲🌲 ему вдруг захотелось потренировать артикли, ну а где это делать?

Так вот в этом случае будет полезно иметь штук 200 слов, которые будут устанавливаться прям с приложением. Вариант JSON файлик выглядит очень привлекательно, поскольу я смогу легко его прочитать используя JSONDecoder.

Вы, кстати, знали, что это 👩‍💻 - логотип JSON? На мой взгляд невзрачный какой-то.

Тут возникает 2 сложности
🟢 model class для SwiftData плохо подходит под десериализацию - стоит отдельно сделать структуру (а на мой скромный взгляд, структура подходит сюда лучше)
🟢 этот файлик нужно загрузить в базу

С первой сложностью я справился (спасибо урокам Пола), а со второй сразу не получилось, т.к. мне было не очевидно, что после insert нужно вызвать save на объекте modelContext. И я не мог понять почему слова не отображаются.
Please open Telegram to view this post
VIEW IN TELEGRAM
👍1
На данный момент я нахожусь в ужасе от кода, который у меня получился. Это просто какая-то лапша…

Нужна насмотренность на готовые проекты, или нужно поскорее изучать MVVM (что-то другое тоже можно).

Но разделить бизнес логику и UI однозначно стоит.
Буду рад если кто-то подскажет в какую сторону можно посмотреть 👀
Please open Telegram to view this post
VIEW IN TELEGRAM
На данный момент получилось в таком виде.
Приделал ещё тулбар с 2 кнопками без обработчиков.

Из функционального - после ответа показывается предложение с примером использования.
🔥2
DerDieDas. Каким я вижу это приложение?

Самое важное - это должно быть очень простое приложение - с минимальным функционалом и лаконичным дизайном. Никакой перегрузки интерфейса: всё быстро, понятно и с фокусом на практику.

Хочется, чтобы приложением мог пользоваться любой человек - независимо от родного языка. Поэтому текста в интерфейсе должно быть как можно меньше. Все подсказки и элементы навигации - максимально визуальные: иконки, картинки, анимации.

Основные задачи:
🔸 тренировка артиклей (der, die, das);
🔸 ведение персональной статистики;
🔸 умный подбор слов, основанный на ошибках и повторениях, а не просто случайной выборке.

Что даст мне этот проект:
🔸 практику в работе с CloudKit 🌧 и синхронизацией данных между устройствами;
🔸 полноценное приложение, которое можно показать в портфолио или друзьям;
🔸 представление о полном цикле разработки: от идеи до релиза;
🔸 уверенность и мотивацию двигаться дальше после первого опубликованного проекта.

Дополнительно:
Я вижу в этом приложении не просто тренировку слов - а возможность выстроить полезную привычку. Каждый короткий сеанс - как шаг в сторону лучшего понимания языка. А ещё это отличная платформа для экспериментов: с UX, дизайном, локализацией и взаимодействием с пользователем (но для этого нужно научиться снимать метрики взаимодействия с приложением).
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥1
Эм.. изучаю CloudKit ☁️ уже несколько часов, только сейчас понял, что private база хранится в квоте пользователя
Но ведь у многих iCloud с дефолтной квотой в 5Gb, и туда ничего не влезает…

Я могу хранить основной набор слов в public базе, потом их синкать и сохранять локально.

❗️ А где сохранять статистику правильных/неправильных ответов пользователя?
Please open Telegram to view this post
VIEW IN TELEGRAM
Ещё раз про CloudKit

Кажется, я изначально неправильно понял, как работает эта технология.

Основываясь на уроках Hacking with Swift, я предположил, что механизм синхронизации “встроен”. И это действительно так - но только для приватной базы, связанной с конкретным пользователем.

Я думал, что если добавить записи в дашборде CloudKit, они автоматически подтянутся в приложение. И что в коде можно будет как-то отреагировать на приход новых данных, например, callbacks.

Оказалось, что данные из публичной базы нужно явно запрашивать. Никакой “магии” синхронизации тут нет.

Вот пример моей рабочей реализации, которая действительно получает записи, созданные через CloudKit Dashboard:

class CloudKitManager: ObservableObject {
private let database = CKContainer.default().publicCloudDatabase

@Published var words: [WordDto] = []

func fetchWords() {
print("Fetching words from NounWord record type...")
let query = CKQuery(recordType: "NounWord", predicate: NSPredicate(value: true))
let queryOperation = CKQueryOperation(query: query)

var fetchedWords: [WordDto] = []

queryOperation.recordMatchedBlock = { id, result in
switch result {
case .failure(let error):
print("Failed to load word with id \(id): \(error.localizedDescription)")
return
case .success(let record):
guard let word = record["word"] as? String else { return }
guard let articleString = record["article"] as? String else { return }
let dto = WordDto(id: record.recordID.recordName, word: word, article: NounArticle(rawValue: articleString)!, exampleSentences: [])
fetchedWords.append(dto)
}
}

queryOperation.queryResultBlock = { result in
print("Notifying UI about fetched words...")
DispatchQueue.main.async {
switch result {
case .success:
self.words = fetchedWords
case .failure(let error):
print("Error fetching words: \(error.localizedDescription)")
}
}
}

database.add(queryOperation)
}
}


Ещё в придачу, CloudKit API постоянно устаревает, и код взятый из примеров подсвечивается как deprecated…

#cloudkit
🤔1
Что такое Приватная и Публичная базы?

Приватная база привязана к конкретному Apple ID пользователя - данные из неё доступны только этому пользователю. Как раз она синхронизируется между устройствами.

Публичная база, наоборот, общая для всех пользователей приложения - но с ней нужно работать вручную. Т.е. явно отправлять и запрашивать записи, и автоматической синхронизации здесь нет. Она наполняется владельцем приложения.

Есть ещё Shared база, когда пользователь делится данными из приватной базы с другими пользователями. Не мой кейс, поэтому дальше не разбирался.

❗️ это моё текущее понимание
Please open Telegram to view this post
VIEW IN TELEGRAM
👍1
По итогу возвращаюсь к Core Data.
Раздумывал между следующими вариантами: Core Data, Realm, SQLite и обёрткой над SQLite - GRDB.

Поскольку с базами я работал за свою профессиональную деятельность много - хочется разобраться с Core Data 👍
Please open Telegram to view this post
VIEW IN TELEGRAM
👍1
Итоги этой недели

🟥 Первая версия приложения DerDieDas уже готова. Она позволяет тренировать слова и посмотреть статистику по ним. Залил в TestFlight.

🟥 Познакомился с основами CloudKit, и даже сделал запросы чтобы вытащить оттуда данные. Но, к сожалению, отказался от него. Хотя технология интересная.

🟥 Перевел с SwiftData на CoreData, т.к. не вижу особых преимуществ.
В CoreData сложнее работать с типами, т.к. фреймворк генерирует поля с базовыми типами, такими как NSSet. Причем, несмотря на отсутствие галочки на Optional чекбоксе, классы генерируются с опциональными полями.

🟥 Для поддержки большего количества устройств, откатил минимальную версию iOS до 15.6
Это повлекло за собой ещё и отказ от NavigationStack (он доступен для iOS 16+) и navigationDestination в пользу NavigationView и NavigationLink в таком виде

NavigationLink(
destination: WordsListView(),
isActive: $showWordsList,
label: { EmptyView() }
)


🟥 Внедрил MVVM для главного View (которое содержит артикль, слово и кнопки).
Результат очень даже ничего, код намного чище и логичнее, структурирован так, что его можно протестировать почти без проблем.
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥1
Сейчас структура проекта выглядит так. Всё по полочкам! И без тестов 🙈

Особенно порадовали имена с плюсами у сгенерированных файлов.
Впервые такой подход вижу 🙃
Please open Telegram to view this post
VIEW IN TELEGRAM
👏1
This media is not supported in your browser
VIEW IN TELEGRAM
Наконец дошли руки до Metal и шейдеров

Давно хотел разобраться с тем, что такое шейдеры. А тут ещё и на iOS.

Оказывается концепция простая, но очень мощьная. Если кратко, то шейдер это программа выполняемая на GPU, обрабатывающая пиксель или вершину. Обработка идёт в параллель, поэтому всё происходит так быстро.

Основные функции шейдера:
🔸 покрасить пиксель, зная только цвет текущего пикселя
🔸 изменить положение пикселя

Разбирался по статье: https://blog.jacobstechtavern.com/p/metal-in-swiftui-how-to-write-shaders. Доступно и с качественными примерами 👍

Эти примеры сделаны на iOS 17, используя iOS 15, всё выходит сильно сложнее, в плане заставить шейдеры отображаться (в статье не описано). Я попробовал самостоятельно, но у меня не получилось заставить MTKView работать. На 17 всё достаточно просто:

view.colorEffect(
ShaderLibrary.glowEdge(
.float(startDate.timeIntervalSinceNow),
.float2(viewSize)
)
)


#metal #shader
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥2
DerDieDas

Удалось сформировать список слов с примерами, но их корректность нужно проверять ⚠️

Для этого написал авторам одного прекрасного сайта, где есть правильная информация по словам, артиклям и т.д. Хочу либо API получить, либо разрешение закроулить эти слова и распарить их html.

Вот пока жду ответа, пробую разные штуки.
Please open Telegram to view this post
VIEW IN TELEGRAM
Спешу поделиться со всеми радостной новостью, мне удалось выпустить первое приложение в сторе.

https://apps.apple.com/de/app/derdiedas-de/id6745122637

Маленький, но огромный шаг 😎

Буду планировать шаги по реализации своего основного приложения
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥4👍1
Как я делал скриншоты

Одним из обязательных требований Apple является наличие скриншотов, соответствующих диагонали 6.9 дюйма.
Для этого я использовал симулятор iPhone 16 Pro Max и сделал снимки двух основных экранов приложения.

Где-то пишут, что требуется скриншот для iPad, но ревью пройденно без них.

После этого загрузил изображения на theapplaunchpad.com, чтобы добавить фон и надписи.

Существуют и другие подобные сервисы, но большинство из них платные. Этими изображениями удалось воспользоваться бесплатно.
👍1
Всем привет! 👋 Хочу поделиться забавной историей.

Недавно собрал фидбэк по приложению для артиклей у знакомых. Одно из предложений было подсвечивать слова в предложениях.

Для этого исходное предложение нужно разбить на слова, и что самое главное найти это слово вместе с артиклем. Осложняется это тем, что в зависимости от падежа и рода существительного, артикль может меняться.

Так вот пообщавшись с моим другом с утреца 📱 получил от него идею воспользоваться spaCy - Python библиотека для NLP (Обработка естественного языка). Инструмент действительно хороший, но не без ошибок.

А вечером я пошёл на мини-конференцию по Python 👩‍💻
Расписание я подробно посмотрел буквально перед входом - и был приятно удивлён, т.к. первый же докладчик была CEO из компании, которая занимается разработкой spaCy.

Я просто не мог уйти без их стикер 🔥
Please open Telegram to view this post
VIEW IN TELEGRAM
🔥1😁1
Сегодня смотрел как правильно вносить изменения в модели CoreData.

В текущей модели у меня предложения представлены массивом, который хранится в Core Data напрямую.
Переделал так, что каждое предложение представляет собой набор токенов (слово или знак препинания) и индексы токенов, которые нужно будет подсветить.

Наивным путём удалил одно поле и добавил другое 😮
В симуляторе и на телефоне отработало идеально. Но вот с preview всё сломалось. Ни перезагрузка, ни очистка рабочих директорий - ничего не помогало.

🚫 Ошибка была вида: Persistent store migration failed, missing mapping model

Ну и насколько я понял, нужно было создать новую версию модели. Сделал так. Получается в проекте теперь хранятся обе модели.

Сейчас всё снова работает как часы!
Please open Telegram to view this post
VIEW IN TELEGRAM
👍1
Мой микро апдейт висит уже 3 дня, и его еще не начинали ревьюить.

По кому они статистику собирают? 🤔
Please open Telegram to view this post
VIEW IN TELEGRAM