<?xml version="1.0" encoding="utf-8" ?><feed xmlns="http://www.w3.org/2005/Atom" xmlns:tt="http://teletype.in/" xmlns:opensearch="http://a9.com/-/spec/opensearch/1.1/"><title>@easy_dev991</title><subtitle>Делюсь интересными находками/лайфхаками в процессе разработки под iOS, и возможно ты найдешь что-то полезное для себя!</subtitle><author><name>@easy_dev991</name></author><id>https://teletype.in/atom/easy_dev991</id><link rel="self" type="application/atom+xml" href="https://teletype.in/atom/easy_dev991?offset=0"></link><link rel="alternate" type="text/html" href="https://teletype.in/@easy_dev991?utm_source=teletype&amp;utm_medium=feed_atom&amp;utm_campaign=easy_dev991"></link><link rel="next" type="application/rss+xml" href="https://teletype.in/atom/easy_dev991?offset=10"></link><link rel="search" type="application/opensearchdescription+xml" title="Teletype" href="https://teletype.in/opensearch.xml"></link><updated>2026-07-01T21:59:31.628Z</updated><entry><id>easy_dev991:TQsjWIzfHde</id><link rel="alternate" type="text/html" href="https://teletype.in/@easy_dev991/TQsjWIzfHde?utm_source=teletype&amp;utm_medium=feed_atom&amp;utm_campaign=easy_dev991"></link><title>openspec в работе</title><published>2026-06-27T12:09:21.394Z</published><updated>2026-06-27T12:09:48.063Z</updated><category term="agent" label="agent"></category><summary type="html">&lt;img src=&quot;https://img4.teletype.in/files/f1/5d/f15d7427-e310-4ff3-9992-d20d928d0596.png&quot;&gt;Примерно два месяца назад я узнал про openspec, и решил попробовать. Пет-проекты на тот момент были на паузе, поэтому пробовать начал сразу в рабочем проекте. В этой статье покажу как выглядит рабочий процесс с использованием openspec внутри opencode.</summary><content type="html">
  &lt;p id=&quot;vI0G&quot;&gt;Примерно два месяца назад я узнал про &lt;a href=&quot;https://github.com/Fission-AI/openspec&quot; target=&quot;_blank&quot;&gt;openspec&lt;/a&gt;, и решил попробовать. Пет-проекты на тот момент были на паузе, поэтому пробовать начал сразу в рабочем проекте. В этой статье покажу как выглядит рабочий процесс с использованием &lt;strong&gt;openspec&lt;/strong&gt; внутри &lt;strong&gt;opencode&lt;/strong&gt;.&lt;/p&gt;
  &lt;p id=&quot;7W5G&quot;&gt;Дублировать документацию из гитхаба не буду - ее можно прочитать по ссылке в первом абзаце, а расскажу именно про мой опыт использования &lt;em&gt;openspec&lt;/em&gt;.&lt;/p&gt;
  &lt;h2 id=&quot;ybcG&quot;&gt;Установка в проекте&lt;/h2&gt;
  &lt;p id=&quot;4mQl&quot;&gt;В терминале в папке с проектом выполнил команду &lt;code&gt;openspec init&lt;/code&gt; и отметил нужные инструменты (&lt;em&gt;opencode&lt;/em&gt; и т.д.)&lt;/p&gt;
  &lt;figure id=&quot;rfYQ&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/f1/5d/f15d7427-e310-4ff3-9992-d20d928d0596.png&quot; width=&quot;1028&quot; /&gt;
    &lt;figcaption&gt;Выбрал opencode&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;mjs4&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/36/6c/366c67f5-8518-4fd7-ac25-2da29805269c.png&quot; width=&quot;1026&quot; /&gt;
    &lt;figcaption&gt;После нажатия enter нужно перезагрузить IDE, в моем случае - просто завершить сеанс opencode в терминале и запустить его заново&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;TJmR&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/12/2b/122b0da8-e227-4526-b7d5-b10dde024185.png&quot; width=&quot;996&quot; /&gt;
    &lt;figcaption&gt;Теперь нужно закоммитить добавленные в проект навыки и команды, чтобы ими можно было пользоваться&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;h2 id=&quot;06kT&quot;&gt;Использование в работе&lt;/h2&gt;
  &lt;p id=&quot;cab8&quot;&gt;Запускаем &lt;em&gt;opencode&lt;/em&gt; и начинаем вводить команду &lt;code&gt;/o&lt;/code&gt;&lt;/p&gt;
  &lt;figure id=&quot;7Dkp&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img3.teletype.in/files/af/d9/afd95472-f7bc-4d89-a477-aa1adf90dcf0.png&quot; width=&quot;2042&quot; /&gt;
    &lt;figcaption&gt;Первые несколько секунд идет загрузка всех команд, а потом в списке появляются новые команды openspec&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;h3 id=&quot;y2NE&quot;&gt;Новая мелкая фича&lt;/h3&gt;
  &lt;p id=&quot;xgTV&quot;&gt;Если есть &lt;strong&gt;полное понимание, что нужно делать&lt;/strong&gt;, то можно сразу использовать &lt;code&gt;/opsx-ff&lt;/code&gt; - я так делал два раза для мелких задач, где все максимально очевидно и обсуждать нечего, например:&lt;/p&gt;
  &lt;pre id=&quot;hkWL&quot;&gt;/opsx-ff task1234-toggle&lt;/pre&gt;
  &lt;p id=&quot;9tC6&quot;&gt;Тут &lt;strong&gt;task1234-toggle&lt;/strong&gt; является кратким названием задачи - это важное название и лучше делать его уникальным, чтобы потом было проще ориентироваться в задачах. Потом для продолжения работы над задачей может потребоваться явно передать в промпте это самое название задачи, чтобы агент знал, над которой задачей вы сейчас работаете.&lt;/p&gt;
  &lt;p id=&quot;Kw3Z&quot;&gt;Отправляем промпт в работу, и получаем ответ типа такого: &lt;/p&gt;
  &lt;blockquote id=&quot;3xDA&quot;&gt;I found existing toggle-related code in the project (reminder settings toggle, filter dialog toggle), but &amp;quot;task-1234-toggle&amp;quot; doesn&amp;#x27;t tell me what you want to build or fix.&lt;br /&gt;What do you want this change to do? For example:&lt;br /&gt;- Add a new toggle for some feature&lt;br /&gt;- Fix a bug with an existing toggle&lt;br /&gt;- Refactor toggle logic&lt;br /&gt;Describe the feature/fix and I&amp;#x27;ll create all the OpenSpec artifacts.&lt;/blockquote&gt;
  &lt;p id=&quot;0UWn&quot;&gt;Дальше отправляю промпт с кратким описанием фичи (ведь она мелкая):&lt;/p&gt;
  &lt;blockquote id=&quot;XKrp&quot;&gt;нужно сделать фичефлаг для настройки возможности включения или отключения экрана с логами для разработчика на экране&lt;/blockquote&gt;
  &lt;p id=&quot;EqEq&quot;&gt;Агент какое-то время думает и создает все необходимые документы (артефакты) по задаче:&lt;/p&gt;
  &lt;figure id=&quot;lrtu&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img3.teletype.in/files/28/e7/28e70851-53e2-4d00-bf16-689adb0b306a.png&quot; width=&quot;1474&quot; /&gt;
    &lt;figcaption&gt;Окончание работы по созданию документов к задаче&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;5Pa5&quot;&gt;Теперь нужно проверить получившиеся документы самостоятельно (т.к. там действительно немного текста) и при обнаружении нестыковок/ошибок нужно внести правки в документы (вручную или опять же через агента).&lt;/p&gt;
  &lt;p id=&quot;uMFm&quot;&gt;После проверки/исправления всех документов можно сразу переходить к &lt;code&gt;/opsx-apply&lt;/code&gt; или сохранить прогресс (делаем коммит) и переключиться на что-то еще.&lt;/p&gt;
  &lt;h3 id=&quot;crQZ&quot;&gt;Новая стандартная фича&lt;/h3&gt;
  &lt;p id=&quot;eMVp&quot;&gt;Если хочется больше контроля над процессом создания документации, или еще не до конца известны все требования для задачи, то есть смысл начинать с команды &lt;code&gt;/opsx-new&lt;/code&gt;, например:&lt;/p&gt;
  &lt;pre id=&quot;iJJ6&quot;&gt;/opsx-new task1234-medium-feature&lt;/pre&gt;
  &lt;p id=&quot;U2kO&quot;&gt;Отправляем этот промпт и получаем ответ: &lt;/p&gt;
  &lt;blockquote id=&quot;rZbF&quot;&gt;Here&amp;#x27;s the summary:&lt;br /&gt;Change: task1234-medium-feature&lt;br /&gt;Location: openspec/changes/task1234-medium-feature/&lt;br /&gt;Schema: spec-driven (4 artifacts)&lt;br /&gt;Progress: 0/4 complete&lt;br /&gt;[ ] proposal          ← ready now&lt;br /&gt;[-] design             (blocked by: proposal)&lt;br /&gt;[-] specs              (blocked by: proposal)&lt;br /&gt;[-] tasks               (blocked by: design, specs)&lt;br /&gt;First artifact: proposal.md — establish why this change is needed.&lt;br /&gt;Template:&lt;br /&gt;## Why&lt;br /&gt;&amp;lt;!-- Problem/opportunity, 1-2 sentences --&amp;gt;&lt;/blockquote&gt;
  &lt;blockquote id=&quot;Jk78&quot;&gt;## What Changes&lt;br /&gt;&amp;lt;!-- Bullet list: new capabilities, modifications, removals --&amp;gt;&lt;/blockquote&gt;
  &lt;blockquote id=&quot;jhkE&quot;&gt;## Capabilities&lt;br /&gt;### New Capabilities&lt;br /&gt;- &amp;#x60;&amp;lt;name&amp;gt;&amp;#x60;: &amp;lt;description&amp;gt;&lt;/blockquote&gt;
  &lt;blockquote id=&quot;IzeZ&quot;&gt;### Modified Capabilities&lt;br /&gt;&amp;lt;!-- List only if spec-level behavior changes --&amp;gt;&lt;/blockquote&gt;
  &lt;blockquote id=&quot;nLhC&quot;&gt;## Impact&lt;br /&gt;&amp;lt;!— Affected code, APIs, dependencies --&amp;gt;&lt;br /&gt;Ready to create the first artifact? Run /opsx-continue or describe what this change is about and I&amp;#x27;ll draft it.&lt;/blockquote&gt;
  &lt;p id=&quot;RYdh&quot;&gt;В таком сценарии можно делать все документы по порядку, начиная с &lt;em&gt;proposal&lt;/em&gt;, который является кратким описанием задачи для последующей детализации в других документах, напишем промпт с использованием команды &lt;code&gt;/opsx-continue&lt;/code&gt;:&lt;/p&gt;
  &lt;blockquote id=&quot;5Hrr&quot;&gt;/opsx-continue task1234-medium-feature нужно сделать фичефлаг для настройки возможности включения или отключения экрана с логами для разработчика на экране&lt;/blockquote&gt;
  &lt;p id=&quot;o1So&quot;&gt;Отправляем этот промпт и получаем ответ: &lt;/p&gt;
  &lt;blockquote id=&quot;rvTT&quot;&gt;Created: proposal.md&lt;br /&gt;Schema: spec-driven | Progress: 1/4&lt;br /&gt;[x] proposal       ✓&lt;br /&gt;[ ] design          ← unlocked&lt;br /&gt;[ ] specs           ← unlocked&lt;br /&gt;[-] tasks           (blocked by: design, specs)&lt;br /&gt;What&amp;#x27;s in the proposal:&lt;br /&gt;- Why: Нет способа просмотреть логи внутри приложения, нужна фиче-флаг для скрытия от обычных пользователей&lt;br /&gt;- Capabilities: dev-log-screen (экран логов) + dev-log-feature-flag (флаг управления)&lt;br /&gt;- Impact: AppSettingsDataStore, Screen, MoreScreen, новый DevLogScreen + ViewModel&lt;br /&gt;Ready to create the next artifact? Run /opsx-continue — I&amp;#x27;ll draft either design.md or the spec files next.&lt;/blockquote&gt;
  &lt;p id=&quot;5jqU&quot;&gt;Теперь читаем &lt;em&gt;proposal&lt;/em&gt;, вносим правки при необходимости, и либо продолжаем дальше работу в том же формате с командой &lt;code&gt;/opsx-continue&lt;/code&gt;, либо ставим на паузу и переключаемся на что-то другое - главное сделать коммит, чтобы сохранить прогресс.&lt;/p&gt;
  &lt;p id=&quot;BIad&quot;&gt;В нашем случае для примера я &amp;quot;промотаю&amp;quot; создание документов командой &lt;code&gt;/opsx-ff&lt;/code&gt;, чтобы показать как работает &lt;code&gt;/opsx-apply&lt;/code&gt;:&lt;/p&gt;
  &lt;figure id=&quot;F9tI&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img1.teletype.in/files/82/a9/82a96c7a-a47a-4f30-a635-574989f86196.png&quot; width=&quot;956&quot; /&gt;
    &lt;figcaption&gt;Агент начал работу&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;1yVU&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/73/64/7364e7c0-773d-41dd-8cb8-e62bfb4e49bd.png&quot; width=&quot;1658&quot; /&gt;
    &lt;figcaption&gt;После окончания работы агент предлагает добавить фичу в архив как выполненную&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;oEUp&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img3.teletype.in/files/26/2b/262b15ba-3959-4f58-ae83-28809846a83c.png&quot; width=&quot;1260&quot; /&gt;
    &lt;figcaption&gt;Результат работы - тоггл &amp;quot;Developer logs&amp;quot;&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;LMey&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img1.teletype.in/files/c4/dd/c4ddd3fe-cddd-41ef-bace-31c43027b5ca.png&quot; width=&quot;1260&quot; /&gt;
    &lt;figcaption&gt;Если тоггл включить, появится новая кнопка Dev Log&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;CviC&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/92/4a/924ac31c-de1a-4556-a221-ae31446b59c2.png&quot; width=&quot;1260&quot; /&gt;
    &lt;figcaption&gt;А вот и сам экран логов&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;mHIn&quot;&gt;Представим, что положение тоггла было таким и задумано изначально.&lt;/p&gt;
  &lt;p id=&quot;4Afv&quot;&gt;И вот пришло новое требование (ну ничего себе) - нужно тоггл переместить в нижнюю часть экрана. Правим артефакты либо вручную, либо новым промптом (без использования команд &lt;code&gt;/opsx&lt;/code&gt;) и снова вызываем &lt;code&gt;/opsx-apply&lt;/code&gt; для применения правок в коде:&lt;/p&gt;
  &lt;figure id=&quot;FXjg&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/b7/37/b73750e2-8a67-4fb3-bad2-63c57fc55742.png&quot; width=&quot;1260&quot; /&gt;
    &lt;figcaption&gt;Успех! Положение тоггла исправлено&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;h3 id=&quot;ZL9p&quot;&gt;Верификация статуса&lt;/h3&gt;
  &lt;p id=&quot;RnoL&quot;&gt;Бывает, что задача откладывается на неопределенный срок, пока ее тестируют, обсуждают и т.д., и все это время код по задаче не пишется, но возможно меняются документы (вручную или нет, не важно).&lt;/p&gt;
  &lt;p id=&quot;ZqfY&quot;&gt;Когда к задаче нужно вернуться, то можно проверить статус с помощью команды &lt;code&gt;/opsx-verify&lt;/code&gt;, это я и сделал в нашем случае, вот итоги проверки:&lt;/p&gt;
  &lt;blockquote id=&quot;AM6H&quot;&gt;No critical issues found. &lt;br /&gt;3 warnings to consider — all minor. The implementation correctly follows all design decisions, covers all requirements and scenarios, and maintains code pattern consistency.&lt;br /&gt;Ready for archive. ✓&lt;/blockquote&gt;
  &lt;p id=&quot;d5Hl&quot;&gt;Там сначала идет отчет по шаблону, и в нем перечислены все детали проверок, в нашем случае их нет смысла смотреть, но в реальной задаче следует прочитать и убедиться, что все корректно.&lt;/p&gt;
  &lt;p id=&quot;4sHl&quot;&gt;Когда задачу &amp;quot;приняли&amp;quot; окончательно, ее нужно архивировать командой &lt;code&gt;/opsx-archive&lt;/code&gt;:&lt;/p&gt;
  &lt;blockquote id=&quot;XZmm&quot;&gt;Archive Complete&lt;br /&gt;Change: task1234-medium-feature&lt;br /&gt;Schema: spec-driven&lt;br /&gt;Archived to: openspec/changes/archive/2026-06-27-task1234-medium-feature/&lt;br /&gt;Specs: Delta specs preserved (new capabilities — no main specs to sync to)&lt;br /&gt;All 4 artifacts complete. All 20 tasks complete.&lt;/blockquote&gt;
  &lt;p id=&quot;10QY&quot;&gt;Вот и все! Работа над задачей завершена, теперь любые доработки нужно делать через отдельные задачи (&lt;code&gt;/opsx-new&lt;/code&gt; или &lt;code&gt;/opsx-ff&lt;/code&gt;).&lt;/p&gt;
  &lt;h2 id=&quot;oMdM&quot;&gt;Мой отзыв&lt;/h2&gt;
  &lt;p id=&quot;j5oY&quot;&gt;Мне нравится использовать &lt;em&gt;openspec&lt;/em&gt;, потому что:&lt;/p&gt;
  &lt;ol id=&quot;td2B&quot;&gt;
    &lt;li id=&quot;q6RM&quot;&gt;Стандартизируется рабочий процесс - сначала документация, потом реализация по четкому плану&lt;/li&gt;
    &lt;li id=&quot;2kwI&quot;&gt;Появляется документация в удобном формате прямо в папке с проектом, а не где-то далеко в отдельной системе&lt;/li&gt;
    &lt;li id=&quot;Dzqq&quot;&gt;Удобно отслеживать работу и возвращаться к задаче после длительного перерыва&lt;/li&gt;
  &lt;/ol&gt;
  &lt;p id=&quot;g4Iy&quot;&gt;Не нравится, что агент часто генерирует документацию на английском. Да что уж там, иногда иероглифы до сих пор вставляет то тут, то там 😅, но даже с учетом этого недостатка штука очень удобная.&lt;/p&gt;
  &lt;h3 id=&quot;gfRB&quot;&gt;Нюансы&lt;/h3&gt;
  &lt;ul id=&quot;Rj3x&quot;&gt;
    &lt;li id=&quot;C9Ld&quot;&gt;Многое зависит от используемой модели и настроек проекта (в том числе &lt;code&gt;AGENTS.md&lt;/code&gt;) - не нужно ожидать классную реализацию плана с нужным стилем/подходом, если агент не знает ничего этого&lt;/li&gt;
    &lt;li id=&quot;TsXh&quot;&gt;Даже если проект настроен правильно, нейросеть может ошибиться в составлении документации (например), и потом сделать работу по неправильному плану, поэтому нужно перепроверять документацию и реализацию (как это делать - другой вопрос, кто-то будет сам читать, а у кого-то будут автоматизированные проверки)&lt;/li&gt;
  &lt;/ul&gt;
  &lt;p id=&quot;F6RT&quot;&gt;Еще раз ссылка на репозиторий &lt;a href=&quot;https://github.com/Fission-AI/openspec&quot; target=&quot;_blank&quot;&gt;openspec&lt;/a&gt;, рекомендую попробовать самостоятельно 😉&lt;/p&gt;

</content></entry><entry><id>easy_dev991:Greg5st017e</id><link rel="alternate" type="text/html" href="https://teletype.in/@easy_dev991/Greg5st017e?utm_source=teletype&amp;utm_medium=feed_atom&amp;utm_campaign=easy_dev991"></link><title>Краткий обзор RTK</title><published>2026-06-08T16:46:02.849Z</published><updated>2026-06-08T16:48:49.996Z</updated><summary type="html">Недавно нашёл ещё один полезный инструмент для использования в opencode — RTK (Rust Token Killer).</summary><content type="html">
  &lt;p id=&quot;w7ME&quot;&gt;Недавно нашёл ещё один полезный инструмент для использования в &lt;code&gt;opencode&lt;/code&gt; — &lt;a href=&quot;https://github.com/rtk-ai/rtk&quot; target=&quot;_blank&quot;&gt;RTK&lt;/a&gt; (Rust Token Killer). &lt;/p&gt;
  &lt;p id=&quot;0Fj6&quot;&gt;Устанавливается одной командой, встраивается через &lt;code&gt;hook&lt;/code&gt; и автоматом сжимает вывод команд перед тем, как он попадёт в контекст нейросети.&lt;/p&gt;
  &lt;h2 id=&quot;md-heading-1&quot;&gt;Как это работает&lt;/h2&gt;
  &lt;p id=&quot;WxZl&quot;&gt;RTK перехватывает команды и применяет к ним четыре стратегии:&lt;/p&gt;
  &lt;ol id=&quot;s7Ip&quot;&gt;
    &lt;li id=&quot;vnBU&quot;&gt;фильтрацию шума&lt;/li&gt;
    &lt;li id=&quot;yzP2&quot;&gt;группировку похожих элементов&lt;/li&gt;
    &lt;li id=&quot;vIF7&quot;&gt;обрезку лишнего&lt;/li&gt;
    &lt;li id=&quot;OidT&quot;&gt;дедупликацию повторяющихся строк.&lt;/li&gt;
  &lt;/ol&gt;
  &lt;p id=&quot;GNyd&quot;&gt;В результате модель получает ровно ту же информацию, но в 3–10 раз меньше токенов.&lt;/p&gt;
  &lt;h2 id=&quot;n0jx&quot;&gt;Примеры&lt;/h2&gt;
  &lt;h3 id=&quot;NRgE&quot;&gt;git status&lt;/h3&gt;
  &lt;p id=&quot;QtR5&quot;&gt;&lt;strong&gt;Без RTK (~67 токенов):&lt;/strong&gt;&lt;/p&gt;
  &lt;pre id=&quot;RKpk&quot;&gt;On branch main
Your branch is up to date with &amp;#x27;origin/main&amp;#x27;.&lt;/pre&gt;
  &lt;pre id=&quot;F0jd&quot;&gt;Untracked files:
  (use &amp;quot;git add &amp;lt;file&amp;gt;...&amp;quot; to include in what will be committed)
	.serena/
	docs/&lt;/pre&gt;
  &lt;pre id=&quot;nzIL&quot;&gt;nothing added to commit but untracked files present&lt;/pre&gt;
  &lt;p id=&quot;jgZs&quot;&gt;&lt;strong&gt;С RTK (~12 токенов):&lt;/strong&gt;&lt;/p&gt;
  &lt;pre id=&quot;q1dT&quot;&gt;* main...origin/main
?? .serena/
?? docs/&lt;/pre&gt;
  &lt;p id=&quot;qrOY&quot;&gt;Шесть строк превратились в три — при этом полезная информация не потерялась.&lt;/p&gt;
  &lt;h3 id=&quot;md-heading-3&quot;&gt;git diff&lt;/h3&gt;
  &lt;p id=&quot;ZSau&quot;&gt;&lt;strong&gt;Без RTK (~200 токенов):&lt;/strong&gt;&lt;/p&gt;
  &lt;pre id=&quot;1jgr&quot;&gt;diff --git a/Shared SwiftUI Content/100_ScenePhaseExample.swift b/Shared SwiftUI Content/100_ScenePhaseExample.swift
index 3a4b5c6d..7e8f9a0b 100644
--- a/Shared SwiftUI Content/100_ScenePhaseExample.swift
+++ b/Shared SwiftUI Content/100_ScenePhaseExample.swift
@@ -10,6 +10,7 @@ struct ScenePhaseExample: View {
     @State private var isActive = false&lt;/pre&gt;
  &lt;pre id=&quot;Mvwc&quot;&gt;     var body: some View {
+        // MARK: - Main Content
         VStack(spacing: 20) {
             Text(isActive ? &amp;quot;Active&amp;quot; : &amp;quot;Inactive&amp;quot;)
                 .font(.largeTitle)
&lt;/pre&gt;
  &lt;p id=&quot;tc4y&quot;&gt;&lt;strong&gt;С RTK (~40 токенов):&lt;/strong&gt;&lt;/p&gt;
  &lt;pre id=&quot;Ycrx&quot;&gt;100_ScenePhaseExample.swift: +1
  + // MARK: - Main Content&lt;/pre&gt;
  &lt;p id=&quot;ojbe&quot;&gt;RTK показывает только имя файла, количество изменённых строк и сами добавленные/удалённые строки — без служебной информации &lt;code&gt;diff&lt;/code&gt; (индексы, заголовки патчей, контекстные строки).&lt;/p&gt;
  &lt;h2 id=&quot;md-heading-4&quot;&gt;Установка и настройка&lt;/h2&gt;
  &lt;p id=&quot;5H4A&quot;&gt;С помощью &lt;code&gt;homebrew&lt;/code&gt;:&lt;/p&gt;
  &lt;pre id=&quot;lXBE&quot;&gt;brew install rtk&lt;/pre&gt;
  &lt;p id=&quot;E2E9&quot;&gt;После установки нужно инициализировать &lt;code&gt;hook&lt;/code&gt; для вашего рабочего инструмента:&lt;/p&gt;
  &lt;pre id=&quot;5JrW&quot;&gt;rtk init -g                     # Claude Code / Copilot (по умолчанию)
rtk init -g --gemini            # Gemini CLI
rtk init -g --codex             # Codex (OpenAI)
rtk init -g --agent cursor      # Cursor
rtk init -g --agent windsurf    # Windsurf
rtk init -g --opencode          # opencode&lt;/pre&gt;
  &lt;p id=&quot;gMNc&quot;&gt;После настройки для &lt;code&gt;opencode&lt;/code&gt; в &lt;code&gt;~/.config/opencode/plugins/&lt;/code&gt; появится файл &lt;code&gt;rtk.ts&lt;/code&gt;, который автоматически будет перехватывать вызовы Bash-инструмента и переписывать команды через RTK.&lt;/p&gt;
  &lt;h2 id=&quot;6309&quot;&gt;Заключение&lt;/h2&gt;
  &lt;p id=&quot;mO1P&quot;&gt;RTK — это один из тех инструментов, которые ставишь один раз и сразу получаешь профит. Буквально за пару минут получаем экономию токенов, которая особенно заметна на длинных сессиях при работе с агентом 👍&lt;/p&gt;

</content></entry><entry><id>easy_dev991:orJ5fF-h3jo</id><link rel="alternate" type="text/html" href="https://teletype.in/@easy_dev991/orJ5fF-h3jo?utm_source=teletype&amp;utm_medium=feed_atom&amp;utm_campaign=easy_dev991"></link><title>Подключаем нейросеть к Xcode</title><published>2026-06-05T09:13:54.508Z</published><updated>2026-06-05T09:16:02.560Z</updated><media:thumbnail xmlns:media="http://search.yahoo.com/mrss/" url="https://img1.teletype.in/files/41/b0/41b08270-305a-4266-9621-30807738876d.png"></media:thumbnail><category term="i-os" label="iOS"></category><summary type="html">&lt;img src=&quot;https://img1.teletype.in/files/00/39/003964b1-9383-45db-95d7-d066878e3662.png&quot;&gt;В прошлом году презентовали Xcode 26 с возможностью добавления других провайдеров нейросетей кроме chatgpt/claude, но вот незадача - это не так и просто сделать, если не погуглить дополнительно.</summary><content type="html">
  &lt;figure id=&quot;litW&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img1.teletype.in/files/00/39/003964b1-9383-45db-95d7-d066878e3662.png&quot; width=&quot;1644&quot; /&gt;
    &lt;figcaption&gt;Xcode выдает ошибку при попытке подключить kimi&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;8cJO&quot;&gt;В прошлом году презентовали &lt;code&gt;Xcode 26&lt;/code&gt; с возможностью добавления других провайдеров нейросетей кроме &lt;code&gt;chatgpt&lt;/code&gt;/&lt;code&gt;claude&lt;/code&gt;, но вот незадача - это не так и просто сделать, если не погуглить дополнительно.&lt;/p&gt;
  &lt;p id=&quot;zeAm&quot;&gt;Для сравнения вот как выглядит настройка в &lt;code&gt;Android Studio&lt;/code&gt;:&lt;/p&gt;
  &lt;figure id=&quot;xXMO&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/55/92/55928ade-57ee-4450-9b14-900c6ccb68cd.png&quot; width=&quot;2188&quot; /&gt;
    &lt;figcaption&gt;Подключается с первого раза без ошибок, чудеса&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;9JLP&quot;&gt;Если просто ввести в настройках &lt;code&gt;Xcode&lt;/code&gt; те же данные, что в &lt;code&gt;Android Studio&lt;/code&gt;, то ничего не выйдет.&lt;/p&gt;
  &lt;p id=&quot;8Tpz&quot;&gt;Причина проблемы в поле URL: &lt;code&gt;Xcode&lt;/code&gt; автоматически добавляет &lt;code&gt;/v1/models&lt;/code&gt; или &lt;code&gt;/models&lt;/code&gt; к URL провайдера для валидации подключения.&lt;/p&gt;
  &lt;p id=&quot;A0QU&quot;&gt;Поэтому нужно удалить часть URL с версией... чтобы &lt;code&gt;Xcode&lt;/code&gt; сам подставил ее. &lt;/p&gt;
  &lt;p id=&quot;t0Tx&quot;&gt;Гениально, не так ли?&lt;/p&gt;
  &lt;figure id=&quot;3xPJ&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/56/14/56140c5d-9238-46f0-8dac-38a9c5c0f47c.png&quot; width=&quot;1644&quot; /&gt;
    &lt;figcaption&gt;Пример корректной настройки для kimi в Xcode&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;h6Ms&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img1.teletype.in/files/c2/6d/c26de0e7-40a0-470f-be71-6a91757bd641.png&quot; width=&quot;1644&quot; /&gt;
    &lt;figcaption&gt;Успешно подключили kimi&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;LJKL&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/d0/eb/d0ebadfc-fd5c-44f4-8601-b76bc07ecd4e.png&quot; width=&quot;1644&quot; /&gt;
    &lt;figcaption&gt;Детальный экран подключенного kimi&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;T3Zs&quot;&gt;И чтобы не было подозрений, что я рекламирую kimi для &lt;code&gt;Xcode&lt;/code&gt;:&lt;/p&gt;
  &lt;figure id=&quot;NdSr&quot; class=&quot;m_original&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/ba/2b/ba2b11f6-2724-415a-8023-7c81b2bde693.png&quot; width=&quot;780&quot; /&gt;
    &lt;figcaption&gt;kimi нифига не работает в Xcode&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;4JiK&quot;&gt;Всем удачных подключений!&lt;/p&gt;

</content></entry><entry><id>easy_dev991:P3qUJL1LTfs</id><link rel="alternate" type="text/html" href="https://teletype.in/@easy_dev991/P3qUJL1LTfs?utm_source=teletype&amp;utm_medium=feed_atom&amp;utm_campaign=easy_dev991"></link><title>Отказ от Icon Composer</title><published>2026-04-19T14:01:16.566Z</published><updated>2026-04-19T14:02:20.280Z</updated><media:thumbnail xmlns:media="http://search.yahoo.com/mrss/" url="https://img2.teletype.in/files/58/bc/58bc94a6-a0ca-4458-a234-7251c37302f2.png"></media:thumbnail><category term="i-os" label="iOS"></category><summary type="html">Если у вас в приложение несколько альтернативных иконок, и все они сделаны с использованием Icon Composer, эта статья для вас.</summary><content type="html">
  &lt;p id=&quot;YYRP&quot;&gt;Если у вас в приложение несколько альтернативных иконок, и все они сделаны с использованием &lt;code&gt;Icon Composer&lt;/code&gt;, эта статья для вас.&lt;/p&gt;
  &lt;p id=&quot;vTFC&quot;&gt;Или если у вас простая иконка, которая почему-то получает тень или стекло при установке приложения на девайс, эта статья для вас.&lt;/p&gt;
  &lt;h2 id=&quot;HgRL&quot;&gt;Предыстория&lt;/h2&gt;
  &lt;p id=&quot;g4ku&quot;&gt;В &lt;code&gt;Xcode 26.1&lt;/code&gt; добавили баг при создании сборки приложения - ассеты иконок, сделанных в &lt;code&gt;Icon Composer&lt;/code&gt; добавляются в итоговый архив несколько раз, что приводит к заметному увеличению веса приложения. Ссылки на форум: &lt;a href=&quot;https://developer.apple.com/forums/thread/804739&quot; target=&quot;_blank&quot;&gt;первая&lt;/a&gt;, &lt;a href=&quot;https://developer.apple.com/forums/thread/821826&quot; target=&quot;_blank&quot;&gt;вторая&lt;/a&gt;.&lt;/p&gt;
  &lt;p id=&quot;PvjP&quot;&gt;Если иконку создать в композере, то она будет выглядеть как и задумано, вот только в итоговой сборке эта иконка будет весить в несколько раз больше, чем нужно. И если в приложении есть альтернативные иконки, то общий вес легко может вырасти в 2 раза для легкого приложения (например, &lt;a href=&quot;https://github.com/easydev991/SwiftUI-Days&quot; target=&quot;_blank&quot;&gt;этого&lt;/a&gt;).&lt;/p&gt;
  &lt;p id=&quot;nJmG&quot;&gt;Обходное решение есть - не использовать композер, и добавлять иконки в приложение старым способом (через ассеты).&lt;/p&gt;
  &lt;p id=&quot;Ctmr&quot;&gt;Однако для простых иконок это приводит к некрасивому результату:&lt;/p&gt;
  &lt;figure id=&quot;OF4i&quot; class=&quot;m_original&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/b8/ee/b8ee6a3f-71f4-42ab-a3a8-39d359cc5c86.jpeg&quot; width=&quot;136&quot; /&gt;
    &lt;figcaption&gt;Иконка приложения с дефолтным градиентом&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;blockquote id=&quot;8LLt&quot;&gt;Простой я называю иконку, которая состоит из двух цветов без градиентов или прозрачности.&lt;/blockquote&gt;
  &lt;p id=&quot;EoRK&quot;&gt;Получается, что либо мы снижаем вес приложения и миримся с некрасивой тенью (или стеклом, в зависимости от иконки), либо соглашаемся с повышением веса приложения, и целиком управляем тенями и стеклом (привет, композер)?&lt;/p&gt;
  &lt;p id=&quot;HFqF&quot;&gt;&lt;u&gt;Так не пойдет.&lt;/u&gt;&lt;/p&gt;
  &lt;h2 id=&quot;xSmz&quot;&gt;Решение&lt;/h2&gt;
  &lt;figure id=&quot;vRDD&quot; class=&quot;m_original&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/ff/fe/fffe123e-29d3-46c1-baaa-56474e25c881.png&quot; width=&quot;472&quot; /&gt;
    &lt;figcaption&gt;Настройка градиента в фигме - добавляем градиент для иконки, чтобы система не считала картинку такой уж простой для накладывания своих лишних эффектов&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;E9Du&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img1.teletype.in/files/c0/cd/c0cd6390-2008-4cf8-bd99-0541f6e61f57.png&quot; width=&quot;1850&quot; /&gt;
    &lt;figcaption&gt;Слева старая иконка без градиентов, справа - новая с градиентом. Глазом разницу почти не видно, зато избавляемся от некрасивой тени на экране девайса&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;1PqV&quot; class=&quot;m_original&quot;&gt;
    &lt;img src=&quot;https://img1.teletype.in/files/00/7b/007b9eb9-19a1-424a-86bb-be3b37aa06ec.png&quot; width=&quot;154&quot; /&gt;
    &lt;figcaption&gt;Так выглядит новая иконка с незаметным градиентом на экране девайса - без тени и стекла&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;h2 id=&quot;AGCC&quot;&gt;Результат в мегабайтах&lt;/h2&gt;
  &lt;figure id=&quot;PFrQ&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/3b/55/3b55d6bf-f705-4c92-967b-7611f85b0508.png&quot; width=&quot;1408&quot; /&gt;
    &lt;figcaption&gt;Размер приложения с иконками, сделанными в Icon Composer&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;TVbJ&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/f9/72/f972833b-1bed-46c8-8b46-2d40580ca6f2.png&quot; width=&quot;1410&quot; /&gt;
    &lt;figcaption&gt;Размер приложения после обновления основной иконки и альтернативных&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;DLAY&quot;&gt;Вес приложения снизился почти на 4 мегабайта!&lt;/p&gt;
  &lt;p id=&quot;FF7Z&quot;&gt;И при этом новые градиентные иконки весят в 5-7 раз больше, чем простые однотонные. Но простые, как мы уже знаем, система покрывает своим стеклом или тенью, а выглядит это некрасиво.&lt;/p&gt;
  &lt;p id=&quot;nbJQ&quot;&gt;Интересный момент: в темной теме ни стекло, ни дополнительная тень на иконки не накладывается - по крайней мере в &lt;a href=&quot;https://github.com/easydev991/SwiftUI-Days&quot; target=&quot;_blank&quot;&gt;счётчике дней&lt;/a&gt; так.&lt;/p&gt;
  &lt;h2 id=&quot;JHCS&quot;&gt;Заключение&lt;/h2&gt;
  &lt;p id=&quot;ZbjO&quot;&gt;Если у вас в приложении много альтернативных иконок, или вы используете &lt;code&gt;Icon Composer&lt;/code&gt; для создания ваших иконок, то будет здорово вернуться к ассетам и подождать, пока композер починят (а починить его не могут уже больше 5 месяцев).&lt;/p&gt;
  &lt;p id=&quot;QSpa&quot;&gt;Зачем заставлять пользователя скачивать лишние мегабайты мусора вместе с нормальным приложением?&lt;/p&gt;

</content></entry><entry><id>easy_dev991:h91zeBj4yyW</id><link rel="alternate" type="text/html" href="https://teletype.in/@easy_dev991/h91zeBj4yyW?utm_source=teletype&amp;utm_medium=feed_atom&amp;utm_campaign=easy_dev991"></link><title>Чиним синхронизацию Swift Data</title><published>2026-04-17T19:28:53.706Z</published><updated>2026-04-17T19:29:09.532Z</updated><category term="i-os" label="iOS"></category><summary type="html">В моем iOS-приложении для подсчета дней используется Swift Data и в проекте все настроено для автоматической синхронизации данных между всеми девайсами пользователя в iCloud. Сегодня я починил эту синхронизацию (она давно не работала).</summary><content type="html">
  &lt;p id=&quot;4xrH&quot;&gt;В моем &lt;a href=&quot;https://github.com/easydev991/SwiftUI-Days&quot; target=&quot;_blank&quot;&gt;iOS-приложении для подсчета дней&lt;/a&gt; используется &lt;code&gt;Swift Data&lt;/code&gt; и в проекте все настроено для автоматической синхронизации данных между всеми девайсами пользователя в &lt;code&gt;iCloud&lt;/code&gt;. Сегодня я починил эту синхронизацию (она давно не работала).&lt;/p&gt;
  &lt;h2 id=&quot;1IlA&quot;&gt;Проблемный сценарий&lt;/h2&gt;
  &lt;p id=&quot;TAct&quot;&gt;На одном девайсе создаю запись в приложении, жду, сворачиваю/разворачиваю приложение, проверяю на другом девайсе через минуту - там эта запись не появляется.&lt;/p&gt;
  &lt;h2 id=&quot;cC8P&quot;&gt;Решение&lt;/h2&gt;
  &lt;p id=&quot;M3Vp&quot;&gt;Открываем &lt;a href=&quot;https://icloud.developer.apple.com/&quot; target=&quot;_blank&quot;&gt;https://icloud.developer.apple.com/&lt;/a&gt;&lt;/p&gt;
  &lt;figure id=&quot;rBMx&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/19/e7/19e78701-e72f-48f0-9809-a07c98bf2ece.png&quot; width=&quot;1864&quot; /&gt;
    &lt;figcaption&gt;Жмем CloudKit Database&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;oUtH&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img3.teletype.in/files/ae/3a/ae3af325-ccc8-4790-bbd5-aab8c031ea7f.png&quot; width=&quot;848&quot; /&gt;
    &lt;figcaption&gt;Выбираем справа сверху настроенный для приложения контейнер и вариант Production&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;3Vy9&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img3.teletype.in/files/af/61/af617173-007e-4dbd-83f7-a7c911f67040.png&quot; width=&quot;908&quot; /&gt;
    &lt;figcaption&gt;Открываем логи&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;MLqd&quot;&gt;Находим запись с ошибкой, например так:&lt;/p&gt;
  &lt;figure id=&quot;U2v2&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/d8/f9/d8f993b1-a68d-485d-b47d-df89835d91dd.png&quot; width=&quot;3168&quot; /&gt;
    &lt;figcaption&gt;Фильтруем по OverallStatus = USER_ERROR и жмем справа синюю кнопку Query&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;DEY9&quot;&gt;Проверяем схему (типы записей) в development и production&lt;/p&gt;
  &lt;figure id=&quot;cPuV&quot; class=&quot;m_original&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/93/5c/935cd717-10d2-4571-ba26-5ccf72210d31.png&quot; width=&quot;446&quot; /&gt;
    &lt;figcaption&gt;Жмем сюда для development и production по очереди, чтобы убедиться в наличии нестыковок&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;x0Vg&quot;&gt;Нестыковки найти очень легко: в моем случае в production-схеме буквально не было нужного типа записи с полями из iOS-приложения.&lt;/p&gt;
  &lt;figure id=&quot;OlPK&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img3.teletype.in/files/e5/78/e57896b4-472b-42a4-bd7f-c6b958543fa2.png&quot; width=&quot;3210&quot; /&gt;
    &lt;figcaption&gt;Вот как у меня было в прод-схеме - только один тип записи &amp;quot;Users&amp;quot;&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;Q3Nl&quot;&gt;Если найдены нестыковки, и development-схема содержит более актуальные данные, нужно ... опубликовать эти изменения в прод.&lt;/p&gt;
  &lt;figure id=&quot;Incq&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img1.teletype.in/files/06/ce/06cee1f4-1fc5-4c55-8ce9-18c079d8aa12.png&quot; width=&quot;3202&quot; /&gt;
    &lt;figcaption&gt;Вот как у меня было в дев-схеме (есть нужный мне тип записи &amp;quot;CD_Item&amp;quot;), и тут же выделил слева внизу кнопку для публикации изменений&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;C4MW&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/33/4a/334ae115-28fd-4fc0-8f26-d6ef91b1f1a8.png&quot; width=&quot;1292&quot; /&gt;
    &lt;figcaption&gt;Перед деплоем можно посмотреть изменения - тут в общем все поля моей модели данных, которая используется в &amp;quot;Счётчике дней&amp;quot; на iOS&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;Qit4&quot;&gt;После нажатия &amp;quot;Deploy&amp;quot; можно подождать пару минут и повторить тест на проблемном сценарии. У меня проблема была решена через 2 минуты после деплоя актуальной схемы в прод.&lt;/p&gt;
  &lt;h2 id=&quot;6EhW&quot;&gt;Заключение&lt;/h2&gt;
  &lt;p id=&quot;l4tE&quot;&gt;Когда я в первый раз настраивал &lt;code&gt;Swift Data&lt;/code&gt;, то вообще не заглядывал в консоль &lt;code&gt;icloud.developer&lt;/code&gt;, не проверял ничего связанного со схемами, и даже статей на эту тему не искал и не читал. Оказалось, настроить синхронизацию с облаком несложно, нужно только знать как это делается 😅&lt;/p&gt;
  &lt;p id=&quot;bws5&quot;&gt;Спасибо интернету! И нейросетям 😉&lt;/p&gt;

</content></entry><entry><id>easy_dev991:hlx_HA24Cgy</id><link rel="alternate" type="text/html" href="https://teletype.in/@easy_dev991/hlx_HA24Cgy?utm_source=teletype&amp;utm_medium=feed_atom&amp;utm_campaign=easy_dev991"></link><title>Какой макбук выбрать в 2026 году</title><published>2026-03-28T10:19:10.347Z</published><updated>2026-03-28T10:22:52.957Z</updated><media:thumbnail xmlns:media="http://search.yahoo.com/mrss/" url="https://img4.teletype.in/files/7c/ca/7cca7d4a-d86e-4dce-acee-14c2ddc59bca.png"></media:thumbnail><summary type="html">&lt;img src=&quot;https://img2.teletype.in/files/16/2f/162f3954-4a8e-41c5-9496-7cad859f3172.png&quot;&gt;Предыдущая статья про 2025 год доступна по ссылке - большинство рекомендаций из нее все еще актуальны, кроме цен и версии Xcode, конечно.</summary><content type="html">
  &lt;p id=&quot;GWNq&quot;&gt;Предыдущая статья про 2025 год доступна по &lt;a href=&quot;https://telegra.ph/115-Kakoj-makbuk-vybrat-v-2025-godu-04-16&quot; target=&quot;_blank&quot;&gt;ссылке&lt;/a&gt; - большинство рекомендаций из нее все еще актуальны, кроме цен и версии &lt;code&gt;Xcode&lt;/code&gt;, конечно.&lt;/p&gt;
  &lt;p id=&quot;ZOyD&quot;&gt;За год изменилось немного: появился чип &lt;strong&gt;m5 &lt;/strong&gt;и вышел &lt;strong&gt;Xcode 26...&lt;/strong&gt;&lt;/p&gt;
  &lt;h2 id=&quot;Макбук-для-работы&quot;&gt;Макбук для работы&lt;/h2&gt;
  &lt;p id=&quot;RFz4&quot;&gt;Рекомендации по поводу рабочего макбука двигаем на &lt;strong&gt;1&lt;/strong&gt; цифру вперед, то есть на работе заказываем &lt;strong&gt;MacBook Pro m5 pro &lt;/strong&gt;- как обычно, даже в базовой конфигурации это будет отличный вариант на несколько лет вперед.&lt;/p&gt;
  &lt;p id=&quot;Ulxm&quot;&gt;Если компания не выдает технику, то есть смысл взять то, на что хватает денег из линейки &lt;strong&gt;m3&lt;/strong&gt;-&lt;strong&gt;m5&lt;/strong&gt;. &lt;strong&gt;m1&lt;/strong&gt;-&lt;strong&gt;m2&lt;/strong&gt; не рассматриваем, потому что они уже прилично состарились, а &lt;strong&gt;m2&lt;/strong&gt; и вовсе проиграл в некоторых тестах предшественнику.&lt;/p&gt;
  &lt;h2 id=&quot;Макбук-не-для-работы&quot;&gt;Макбук не для работы&lt;/h2&gt;
  &lt;p id=&quot;tB2h&quot;&gt;В 2020 году я купил себе &lt;strong&gt;MacBook Pro 13&amp;#x27;&amp;#x27; m1 &lt;/strong&gt;за&lt;strong&gt; 150к &lt;/strong&gt;(на официальном сайте с доставкой на дом, чудеса вообще), и он отлично прослужил мне до второй половины 2025 года, когда я его продал за &lt;strong&gt;64к&lt;/strong&gt; - это была отличная продажа и я очень доволен таким успехом, и покупатель будет еще долго доволен, т.к. ему макбук нужен был не для программирования, а для повседневного использования.&lt;/p&gt;
  &lt;p id=&quot;wQLY&quot;&gt;В 2026 не могу рекомендовать макбуки из линейки &lt;strong&gt;m1-m2&lt;/strong&gt;, так что находим выгодные варианты &lt;strong&gt;m3&lt;/strong&gt;-&lt;strong&gt;m5&lt;/strong&gt; (как и для работы) на маркетплейсах и радуемся запасу производительности на ближайшие 3-5 лет.&lt;/p&gt;
  &lt;p id=&quot;rAkm&quot;&gt;Очевидно, &lt;strong&gt;m5&lt;/strong&gt; прослужит лучше в течение 5 лет, чем &lt;strong&gt;m3&lt;/strong&gt;, но для рядового пользователя, полагаю, разницы не будет вообще - не всем же нужно запускать &lt;code&gt;Xcode&lt;/code&gt;/&lt;code&gt;Android Studio&lt;/code&gt; каждый день по несколько раз и ждать выполнение unit/ui-тестов 😁&lt;/p&gt;
  &lt;h2 id=&quot;yaz9&quot;&gt;Macbook Neo&lt;/h2&gt;
  &lt;p id=&quot;Lriu&quot;&gt;Сделал отдельный раздел специально для этой модели, потому что в личку мне уже написали гениальную идею купить этот макбук для ios-разработки.&lt;/p&gt;
  &lt;p id=&quot;nfIW&quot;&gt;Этот раздел статьи не для слабонервных!&lt;/p&gt;
  &lt;p id=&quot;I6kO&quot;&gt;.&lt;/p&gt;
  &lt;p id=&quot;xc0g&quot;&gt;.&lt;/p&gt;
  &lt;p id=&quot;rM6S&quot;&gt;.&lt;/p&gt;
  &lt;p id=&quot;Dc1f&quot;&gt;.&lt;/p&gt;
  &lt;p id=&quot;ljUM&quot;&gt;.&lt;/p&gt;
  &lt;p id=&quot;ILCT&quot;&gt;.&lt;/p&gt;
  &lt;p id=&quot;uSq4&quot;&gt;Я серьезно, если вы фанат Apple или просто ярый поклонник всех макбуков, то лучше не читать дальше!&lt;/p&gt;
  &lt;p id=&quot;VjEX&quot;&gt;.&lt;/p&gt;
  &lt;p id=&quot;G5Hd&quot;&gt;.&lt;/p&gt;
  &lt;p id=&quot;AHif&quot;&gt;.&lt;/p&gt;
  &lt;p id=&quot;I9uX&quot;&gt;.&lt;/p&gt;
  &lt;p id=&quot;Xx1J&quot;&gt;.&lt;/p&gt;
  &lt;p id=&quot;ulIb&quot;&gt;.&lt;/p&gt;
  &lt;p id=&quot;7gTg&quot;&gt;Вы еще тут? Окей, поехали)&lt;/p&gt;
  &lt;p id=&quot;hWNE&quot;&gt;Специально для этой модели я покажу скриншоты цен (актуально на конец марта - начало апреля 2026) без рекламы конкретного маркетплейса, и потом мы вместе сравним с ценами на другие модели на том же маркетплейсе, опять же не реклама:&lt;/p&gt;
  &lt;figure id=&quot;pnBQ&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/16/2f/162f3954-4a8e-41c5-9496-7cad859f3172.png&quot; width=&quot;2268&quot; /&gt;
    &lt;figcaption&gt;Цены на безымянном (популярном) маркетплейсе на Macbook Neo - от 66к руб с учетом пошлины&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;E9Xj&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/b8/32/b832a1ec-d3a0-420d-a268-01144b1c17ea.png&quot; width=&quot;2318&quot; /&gt;
    &lt;figcaption&gt;Цены на том же маркетплейсе на Macbook Air M4 - от 89к руб с учетом пошлины&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;Ajp4&quot;&gt;&lt;strong&gt;Важная деталь&lt;/strong&gt;: это сравнение цен на новые ноутбуки, т.е. не б/у, не паленые, а прямо новые.&lt;/p&gt;
  &lt;p id=&quot;CFKc&quot;&gt;&lt;strong&gt;Air m4&lt;/strong&gt; дороже на &lt;strong&gt;23к&lt;/strong&gt; руб (на этом маркетплейсе), но и по большинству параметров он намного лучше.&lt;/p&gt;
  &lt;p id=&quot;9vhE&quot;&gt;У мелких перекупщиков на других популярных сайтах можно найти тот же &lt;strong&gt;air m4&lt;/strong&gt; за &lt;strong&gt;69к&lt;/strong&gt;.&lt;/p&gt;
  &lt;p id=&quot;hHov&quot;&gt;&lt;u&gt;Внимательный&lt;/u&gt; читатель может спросить: &amp;quot;&lt;em&gt;Зачем покупать дорогой air m4, если можно купить новый модный и молодежный neo дешевле?&lt;/em&gt;&amp;quot;.&lt;/p&gt;
  &lt;p id=&quot;jsVf&quot;&gt;На это я &lt;em&gt;отвечу&lt;/em&gt; внимательному читателю: &amp;quot;&lt;em&gt;Если не получается найти разницу при беглом сравнении параметров обеих моделей под свои потребности, то не нужно платить больше&lt;/em&gt;&amp;quot;.&lt;/p&gt;
  &lt;p id=&quot;n66a&quot;&gt;Потом &lt;u&gt;сообразительный&lt;/u&gt; читатель может спросить: &amp;quot;&lt;em&gt;Зачем вообще покупать макбук в 2026 году за такие деньги не для ios-разработки, когда можно купить ноутбук на виндовс под любые потребности и по любой цене, в том числе от 15к и до 60к?&lt;/em&gt;&amp;quot;&lt;/p&gt;
  &lt;p id=&quot;GInl&quot;&gt;На что я отвечу сообразительному читателю: &amp;quot;&lt;em&gt;Отличный вопрос, мой дорогой читатель! Нет никакого смысла тратить деньги на макбук, если нет конкретных задач, для которых нужен именно макбук, и с которыми не справятся другие ноутбуки&lt;/em&gt;&amp;quot;.&lt;/p&gt;
  &lt;p id=&quot;eLf5&quot;&gt;И пока другие блогеры и СМИ активно пиарят новый &lt;strong&gt;macbook neo&lt;/strong&gt; как убийцу доступных ноутбуков на винде, мы с вами как самые сообразительные понимаем, что это пустой трёп, который нацелен только на отработку рекламного ТЗ, или на сбор хайпа (или и то, и другое разом).&lt;/p&gt;
  &lt;figure id=&quot;fIlM&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/92/09/920960e0-5641-4551-bb55-18da1fbb2857.png&quot; width=&quot;2266&quot; /&gt;
    &lt;figcaption&gt;Вот пример реальных убийц macbook neo на том же маркетплейсе дешевле 25к руб. В карточках товара кринж, но это не так важно.&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;gY7l&quot;&gt;И прежде чем набегут желающие защитить божественный бренд Apple от сравнения с безымянными ноутбуками, сначала вспомним, что этот раздел был не для слабонервных и не для фанатов Apple!&lt;/p&gt;
  &lt;p id=&quot;srx3&quot;&gt;А теперь можно вместе подумать, что еще можно купить на оставшиеся 45к руб после покупки ноутбука на винде:&lt;/p&gt;
  &lt;ul id=&quot;0rEu&quot;&gt;
    &lt;li id=&quot;xx5Z&quot;&gt;Отличный новый андроид-смартфон&lt;/li&gt;
    &lt;li id=&quot;2qoL&quot;&gt;Два хороших новых андроид-смартфона&lt;/li&gt;
    &lt;li id=&quot;oxUm&quot;&gt;Хороший андроид-смартфон и еще один ноутбук на винде&lt;/li&gt;
    &lt;li id=&quot;nnbN&quot;&gt;Два дополнительных ноутбука на винде&lt;/li&gt;
    &lt;li id=&quot;1ABD&quot;&gt;45 пицц по 1000 рублей каждая&lt;/li&gt;
  &lt;/ul&gt;
  &lt;p id=&quot;z2uI&quot;&gt;Ну вы поняли, надеюсь 😉&lt;/p&gt;
  &lt;h2 id=&quot;oBfa&quot;&gt;Заключение&lt;/h2&gt;
  &lt;p id=&quot;AYFS&quot;&gt;Итоговый выбор как всегда зависит от ваших конкретных задач (но не покупайте, пожалуйста, &lt;strong&gt;macbook neo&lt;/strong&gt;), для которых выбирается макбук, и в большинстве случаев для разработки подойдет любая модель с чипом &lt;strong&gt;m4&lt;/strong&gt;-&lt;strong&gt;m5&lt;/strong&gt;.&lt;/p&gt;
  &lt;p id=&quot;gfix&quot;&gt;Удачных покупок!&lt;/p&gt;

</content></entry><entry><id>easy_dev991:baZfVYrGA5H</id><link rel="alternate" type="text/html" href="https://teletype.in/@easy_dev991/baZfVYrGA5H?utm_source=teletype&amp;utm_medium=feed_atom&amp;utm_campaign=easy_dev991"></link><title>Загружаем фото в андроид-эмулятор</title><published>2026-03-20T19:18:24.928Z</published><updated>2026-03-20T19:19:44.680Z</updated><media:thumbnail xmlns:media="http://search.yahoo.com/mrss/" url="https://img2.teletype.in/files/1e/19/1e199205-20aa-4772-9b17-0e91fcd114fa.png"></media:thumbnail><summary type="html">&lt;img src=&quot;https://img2.teletype.in/files/51/55/5155b125-26a8-44dc-8192-92da8d2af2a6.png&quot;&gt;В процессе разработки андроид-приложения бывает нужно протестировать функционал добавления фотографий из галереи в приложение.</summary><content type="html">
  &lt;p id=&quot;DZbY&quot;&gt;В процессе разработки андроид-приложения бывает нужно протестировать функционал добавления фотографий из галереи в приложение.&lt;/p&gt;
  &lt;p id=&quot;uvnt&quot;&gt;Есть разные способы загрузки фоток в эмулятор, например:&lt;/p&gt;
  &lt;ul id=&quot;qJML&quot;&gt;
    &lt;li id=&quot;iMIh&quot;&gt;перетащить файлы с фотками на экран эмулятора&lt;/li&gt;
    &lt;li id=&quot;mVuf&quot;&gt;перетащить файлы с фотками в открытое приложение &amp;quot;файлы&amp;quot; на экране эмулятора&lt;/li&gt;
    &lt;li id=&quot;xIBT&quot;&gt;открыть Device Explorer и добавить фотки в папку sdcard/DCIM&lt;/li&gt;
  &lt;/ul&gt;
  &lt;p id=&quot;LOpN&quot;&gt;Но что бы вы ни сделали, все эти способы могут не привести к желаемому результату...&lt;/p&gt;
  &lt;p id=&quot;SB53&quot;&gt;А вот кстати и желаемый результат:&lt;/p&gt;
  &lt;figure id=&quot;1xXu&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/51/55/5155b125-26a8-44dc-8192-92da8d2af2a6.png&quot; width=&quot;1080&quot; /&gt;
    &lt;figcaption&gt;В приложении можно открыть галерею фотографий&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;ddko&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img1.teletype.in/files/0e/a8/0ea83121-a09d-4551-ad5c-df2c3961292e.png&quot; width=&quot;1080&quot; /&gt;
    &lt;figcaption&gt;Фото из галереи можно выгрузить в приложение&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;h2 id=&quot;KUrD&quot;&gt;Проблема&lt;/h2&gt;
  &lt;p id=&quot;8TnF&quot;&gt;Все перечисленные ранее способы не сработали:&lt;/p&gt;
  &lt;figure id=&quot;pslV&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img1.teletype.in/files/0f/95/0f9536df-c57a-4c4c-bac2-35b650a96af1.png&quot; width=&quot;1080&quot; /&gt;
    &lt;figcaption&gt;Фотографии добавлены в эмулятор каждым из перечисленных ранее способов, но приложение их не видит (как и сам эмулятор)&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;h2 id=&quot;9Tk4&quot;&gt;Решение&lt;/h2&gt;
  &lt;p id=&quot;cRx5&quot;&gt;После копирования фотографий на эмулятор любым из ранее перечисленных способов выключаем эмулятор и жмем &lt;code&gt;Cold Boot&lt;/code&gt;:&lt;/p&gt;
  &lt;figure id=&quot;qK93&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/fb/e3/fbe3b5a9-9508-4f3a-bb0e-e78475f5c01c.png&quot; width=&quot;1080&quot; /&gt;
    &lt;figcaption&gt;Чудо-кнопка, решающая проблему с обнаружением фотографий в эмуляторе&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;h2 id=&quot;AMrX&quot;&gt;Заключение&lt;/h2&gt;
  &lt;p id=&quot;lQAa&quot;&gt;Перед написанием этой статьи я пробовал добавить фотографии в эмулятор всеми перечисленными способами, даже стирал данные эмулятора и перезагружал его - ничего не помогало.&lt;/p&gt;
  &lt;p id=&quot;VheN&quot;&gt;И только чудо-кнопка &lt;code&gt;Cold Boot&lt;/code&gt; помогла.&lt;/p&gt;
  &lt;p id=&quot;Ox3a&quot;&gt;Будет здорово, если кому-то эта статья поможет сэкономить время в подобной ситуации &lt;/p&gt;

</content></entry><entry><id>easy_dev991:BJryrCj0DMr</id><link rel="alternate" type="text/html" href="https://teletype.in/@easy_dev991/BJryrCj0DMr?utm_source=teletype&amp;utm_medium=feed_atom&amp;utm_campaign=easy_dev991"></link><title>Xcode 26.3 Coding Assistant</title><published>2026-02-28T09:19:07.158Z</published><updated>2026-02-28T09:19:07.158Z</updated><media:thumbnail xmlns:media="http://search.yahoo.com/mrss/" url="https://img2.teletype.in/files/db/6f/db6f9ffb-be0f-404b-96bb-c8b0f1216126.png"></media:thumbnail><category term="i-os" label="iOS"></category><summary type="html">&lt;img src=&quot;https://img2.teletype.in/files/53/eb/53ebfbf1-2440-4b77-ae0e-5e1cf7028697.png&quot;&gt;Вчера активно тестировал режим агента в Xcode 26.3, чтобы вам не пришлось</summary><content type="html">
  &lt;p id=&quot;pk0p&quot;&gt;Вчера активно тестировал режим агента в &lt;code&gt;Xcode 26.3&lt;/code&gt;, чтобы вам не пришлось  &lt;/p&gt;
  &lt;h2 id=&quot;O4du&quot;&gt;С чем имеем дело&lt;/h2&gt;
  &lt;p id=&quot;B51q&quot;&gt;В &lt;code&gt;Xcode 26.3&lt;/code&gt; нельзя подключить любую свою модель кроме предусмотренных из коробки&lt;/p&gt;
  &lt;figure id=&quot;Uxb2&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/53/eb/53ebfbf1-2440-4b77-ae0e-5e1cf7028697.png&quot; width=&quot;1644&quot; /&gt;
    &lt;figcaption&gt;Это обман, не ведитесь на него зря!&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;bZ2Y&quot;&gt;Да, есть настройка для добавления кастомного провайдера моделей, но она не совместима с &lt;code&gt;Z.AI&lt;/code&gt; (провайдер &lt;code&gt;GLM&lt;/code&gt;-моделей), например.&lt;/p&gt;
  &lt;p id=&quot;SYFt&quot;&gt;Чтобы подключить своего провайдера, нужно подключать специальный прокси (при желании можно даже найти бесплатные варианты), так я и сделал.&lt;/p&gt;
  &lt;p id=&quot;xmwv&quot;&gt;Я не использовал &lt;code&gt;ChatGPT&lt;/code&gt;/&lt;code&gt;Claude&lt;/code&gt; в &lt;code&gt;Xcode&lt;/code&gt;, потому что у меня куплены другие подписки (&lt;code&gt;cursor&lt;/code&gt; + &lt;code&gt;Z.AI&lt;/code&gt;).&lt;/p&gt;
  &lt;h2 id=&quot;mmgr&quot;&gt;Сразу к результатам&lt;/h2&gt;
  &lt;h3 id=&quot;5vCg&quot;&gt;Положительные моменты&lt;/h3&gt;
  &lt;ul id=&quot;K28j&quot;&gt;
    &lt;li id=&quot;a52i&quot;&gt;Чат работает? - чек&lt;/li&gt;
    &lt;li id=&quot;FphR&quot;&gt;Анимации в чате работают? - чек&lt;/li&gt;
    &lt;li id=&quot;hSAc&quot;&gt;Градиенты в &lt;code&gt;Xcode&lt;/code&gt; отображаются? - чек&lt;/li&gt;
    &lt;li id=&quot;kGZk&quot;&gt;Код генерируется? - чек&lt;/li&gt;
  &lt;/ul&gt;
  &lt;h3 id=&quot;xsXv&quot;&gt;Отрицательные моменты&lt;/h3&gt;
  &lt;ul id=&quot;PNE7&quot;&gt;
    &lt;li id=&quot;cBnr&quot;&gt;Модель может выполнять команды в терминале? - нет&lt;/li&gt;
    &lt;li id=&quot;YGTd&quot;&gt;Модель может писать код в разных файлах, которые не открыты в &lt;code&gt;Xcode&lt;/code&gt;? - нет&lt;/li&gt;
    &lt;li id=&quot;3GXZ&quot;&gt;При включении/выключении прокси &lt;code&gt;Xcode&lt;/code&gt; обновляет статус в окне с чатом? - нет&lt;/li&gt;
    &lt;li id=&quot;WCpn&quot;&gt;Модель может взаимодействовать с тестами или симулятором? - нет&lt;/li&gt;
    &lt;li id=&quot;rCx9&quot;&gt;При начале нового диалога &lt;code&gt;Xcode&lt;/code&gt; запоминает выбранную ранее модель? - нет&lt;/li&gt;
  &lt;/ul&gt;
  &lt;figure id=&quot;pgbc&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/77/a8/77a86ba0-ff2c-40fe-9c11-857754776a7c.png&quot; width=&quot;784&quot; /&gt;
    &lt;figcaption&gt;Стрём, который мы заслужили от Apple (1)&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;Mw6g&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/d4/72/d472a71a-aeba-43ac-abf2-34aee293323e.png&quot; width=&quot;758&quot; /&gt;
    &lt;figcaption&gt;Стрём, который мы заслужили от Apple (2)&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;sd1q&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img3.teletype.in/files/2a/07/2a078e47-98fc-4477-8899-7b2dfaa511da.png&quot; width=&quot;934&quot; /&gt;
    &lt;figcaption&gt;Пруф, что тесты в проекте есть&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;7bdC&quot;&gt;Больше всего меня удивил момент, когда модель написал код в одном из двух нужных файлов, а вместо изменения второго файла создала новый, что привело к ошибке компиляции, и эту ошибку комплияции она не смогла исправить, потому что не удалось, видите ли, сделать изменения в нужном файле.&lt;/p&gt;
  &lt;p id=&quot;mxAF&quot;&gt;Полагаю, дело в объеме второго файла - там больше тысячи строк кода, но в других инструментах (&lt;code&gt;qwen-code&lt;/code&gt;, &lt;code&gt;opencode&lt;/code&gt;, &lt;code&gt;cursor&lt;/code&gt;) такой проблемы ни разу не было.&lt;/p&gt;
  &lt;h2 id=&quot;hzOt&quot;&gt;Сравнение с Android Studio&lt;/h2&gt;
  &lt;figure id=&quot;4bSw&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img3.teletype.in/files/ab/03/ab0316ba-58f8-4cf6-b18b-9a8f54ee0594.png&quot; width=&quot;3402&quot; /&gt;
    &lt;figcaption&gt;Все отлично - любую модель можно открыть в терминале и все будет отображаться в одном окне, очень удобно&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;h2 id=&quot;rTAr&quot;&gt;Вывод&lt;/h2&gt;
  &lt;p id=&quot;uZUe&quot;&gt;То, что Apple нам рекламируют, по уже сложившейся традиции оказывается полной ерундой.&lt;/p&gt;
  &lt;p id=&quot;VfE8&quot;&gt;Нет никакого оправдания, почему в 2026 году &lt;code&gt;Xcode&lt;/code&gt; не может дать доступ к терминалу подключенным нейросетям, или хотя бы дать дефолтный &lt;code&gt;mcp&lt;/code&gt; для работы с тестами и симулятором.&lt;/p&gt;
  &lt;p id=&quot;wKMs&quot;&gt;Ну и отсутствие терминала уже ни в какие рамки не идет. Хоть я и не знаю почти никаких команд в терминале, и вообще не фанат модных движений типа TUI, но отсутствие терминала в &lt;code&gt;Xcode&lt;/code&gt; - это показатель их провала.&lt;/p&gt;
  &lt;figure id=&quot;Ene0&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/52/c1/52c1eb17-e02b-4297-8dd9-bd64a1690a34.png&quot; width=&quot;1022&quot; /&gt;
    &lt;figcaption&gt;Да, тоггл есть, но толку от него нет. Так держать, Apple.&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;19wc&quot;&gt;Но вдруг у меня слишком высокие требования к инструментам?&lt;/p&gt;
  &lt;p id=&quot;12XD&quot;&gt;Если вам нравится использовать &lt;code&gt;Coding Assistant&lt;/code&gt; в &lt;code&gt;Xcode&lt;/code&gt;, поделитесь обратной связью в комментариях, пожалуйста. Вдруг я чего-то не знаю, и там все отлично работает, если использовать это по-другому.&lt;/p&gt;

</content></entry><entry><id>easy_dev991:zlfw_fKJvh9</id><link rel="alternate" type="text/html" href="https://teletype.in/@easy_dev991/zlfw_fKJvh9?utm_source=teletype&amp;utm_medium=feed_atom&amp;utm_campaign=easy_dev991"></link><title>Приложение для Android с нуля до релиза за 3 недели</title><published>2026-01-25T15:06:37.639Z</published><updated>2026-01-25T15:06:37.639Z</updated><category term="android" label="Android"></category><summary type="html">&lt;img src=&quot;https://img3.teletype.in/files/64/51/6451d1a1-8c87-4c66-b17e-922425af20e2.png&quot;&gt;Все больше классных смартфонов на андроиде выпускают, все сильнее стирается граница между iOS/Android, в особенности из-за санкций и ограничений на важные и нужные для жизни сервисы, и я решил сделать андроид-версию своего счётчика дней в новогодние праздники (которые у меня объединились с отпуском).</summary><content type="html">
  &lt;p id=&quot;t5Ee&quot;&gt;Все больше классных смартфонов на андроиде выпускают, все сильнее стирается граница между iOS/Android, в особенности из-за санкций и ограничений на важные и нужные для жизни сервисы, и я решил сделать андроид-версию своего &lt;a href=&quot;https://github.com/easydev991/SwiftUI-Days&quot; target=&quot;_blank&quot;&gt;счётчика дней&lt;/a&gt; в новогодние праздники (которые у меня объединились с отпуском).&lt;/p&gt;
  &lt;h2 id=&quot;OIx6&quot;&gt;Предыстория и о чем статья&lt;/h2&gt;
  &lt;p id=&quot;lSZ1&quot;&gt;Чуть больше двух лет назад я начинал изучать верстку с использованием Jetpack Compose, потому что это похожая штука на SwiftUI, и даже успел за пару месяцев сверстать большинство компонентов дизайн-системы для приложения с площадками для тренировок (&lt;a href=&quot;https://github.com/easydev991/SwiftUI-WorkoutApp&quot; target=&quot;_blank&quot;&gt;вот iOS-версия&lt;/a&gt;), но потом все же забросил это дело, потому что:&lt;/p&gt;
  &lt;ul id=&quot;VOCw&quot;&gt;
    &lt;li id=&quot;zeMi&quot;&gt;хотелось развивать iOS-приложения, и я это успешно делал&lt;/li&gt;
    &lt;li id=&quot;N4e8&quot;&gt;не хотелось тратить время на изучение андроид-разработки с нуля, когда я уже умею делать приложения для iOS&lt;/li&gt;
    &lt;li id=&quot;cbKV&quot;&gt;в то время мне еще совсем не казалось, что есть смысл смотреть в сторону смены айфона на андроид (а теперь ситуация постепенно меняется)&lt;/li&gt;
    &lt;li id=&quot;glL7&quot;&gt;не пользовался нейросетями для разработки &lt;/li&gt;
  &lt;/ul&gt;
  &lt;p id=&quot;582s&quot;&gt;Насчёт счётчика дней - именно с идеи этого приложения я и начинал свою карьеру в iOS-разработке, при этом сам же пользуюсь этим приложением на айфоне, и поэтому решил именно с него начать обучение андроид-разработке в этот раз.&lt;/p&gt;
  &lt;p id=&quot;EaKB&quot;&gt;В этой статье расскажу самое интересное и главное на мой взгляд про андроид-разработку с нуля и до публикации в русторе. И пара слов будет про рустор.&lt;/p&gt;
  &lt;h2 id=&quot;rDWF&quot;&gt;Что нужно знать и уметь&lt;/h2&gt;
  &lt;h3 id=&quot;60ya&quot;&gt;Язык программирования&lt;/h3&gt;
  &lt;p id=&quot;V3Gk&quot;&gt;Как ни странно, не нужно знать язык программирования, чтобы сделать андроид-приложение. Т.е. ни Kotlin, ни Java мне не пригодились. Первый я могу очень поверхностно читать (он похож на Swift), а второй вообще не знаю и не хочу знать . В разработке мне это не понадобилось совсем.&lt;/p&gt;
  &lt;h3 id=&quot;FUzC&quot;&gt;Софты&lt;/h3&gt;
  &lt;p id=&quot;wude&quot;&gt;Нужно уметь формулировать свои мысли в слова, чтобы писать запросы в нейросеть.&lt;/p&gt;
  &lt;p id=&quot;fGFK&quot;&gt;Нужно уметь излагать свои мысли простым и понятным языком, и в достаточной степени детализировать постановку задачи, чтобы получить результат как можно ближе к желаемому.&lt;/p&gt;
  &lt;h3 id=&quot;8Lxh&quot;&gt;Планирование&lt;/h3&gt;
  &lt;p id=&quot;VWoM&quot;&gt;Для разработки андроид-приложения мне пригодились навыки планирования и деления задач на подзадачи (декомпозиция).&lt;/p&gt;
  &lt;p id=&quot;qe4V&quot;&gt;Абсолютно все задачи для нейросети я провожу через планирование. Нельзя просто сказать &amp;quot;Хочу такое приложение, чтобы в нем можно было запоминать события и считать дни&amp;quot; - получится полная фигня, хотя что-то может быть даже будет работать &lt;/p&gt;
  &lt;p id=&quot;AT8B&quot;&gt;Нужно уметь составить план по разработке приложения для начала у себя в голове, а потом перенести его в текстовый формат для последующей обработки нейросетью.&lt;/p&gt;
  &lt;h3 id=&quot;urFz&quot;&gt;Принципы разработки&lt;/h3&gt;
  &lt;p id=&quot;I9HT&quot;&gt;В процессе разработки iOS-приложений я привык обращать внимание на разные моменты, а отдельно хочу выделить эти:&lt;/p&gt;
  &lt;ol id=&quot;Z8Oi&quot;&gt;
    &lt;li id=&quot;wWHc&quot;&gt;Пользовательский опыт (&lt;strong&gt;UX&lt;/strong&gt;) и интерфейс (&lt;strong&gt;UI&lt;/strong&gt;): дизайн должен быть интуитивным, простым и удобным для использования одной рукой; важна консистентность элементов управления и предсказуемое поведение.&lt;/li&gt;
    &lt;li id=&quot;VRA6&quot;&gt;&lt;strong&gt;Адаптивность:&lt;/strong&gt; интерфейс обязан корректно отображаться на экранах разного размера и разрешения&lt;/li&gt;
    &lt;li id=&quot;iwZp&quot;&gt;&lt;strong&gt;Следование гайдлайнам:&lt;/strong&gt; использование принципов &lt;u&gt;Material Design&lt;/u&gt; для Android и &lt;u&gt;Human Interface Guidelines&lt;/u&gt; для iOS для обеспечения привычного пользователям опыта&lt;/li&gt;
  &lt;/ol&gt;
  &lt;p id=&quot;P4KI&quot;&gt;Формулировки взял из интернета, надеюсь что звучат понятно.&lt;/p&gt;
  &lt;p id=&quot;H3NL&quot;&gt;Это все не означает, что я знаю наизусть гайдлайны Android/iOS, или что я всегда планирую разработку для смартфонов и планшетов сразу, но я осознанно стремлюсь делать приложение так, чтобы пользователю было удобно и привычно, и чтобы пользователю не пришлось сталкиваться с ошибками, которые портят впечатление от использования приложения.&lt;/p&gt;
  &lt;h3 id=&quot;oXkP&quot;&gt;Система контроля версий (Git)&lt;/h3&gt;
  &lt;p id=&quot;qt4Q&quot;&gt;Про гит я писал в отдельной статье (&lt;a href=&quot;https://telegra.ph/Pro-rabotu-s-Git-02-23&quot; target=&quot;_blank&quot;&gt;ссылка&lt;/a&gt;), и если вам эта статья пригодилась, то призываю поставить лайк в &lt;a href=&quot;https://t.me/easy_dev991/79&quot; target=&quot;_blank&quot;&gt;телеграм&lt;/a&gt;.&lt;/p&gt;
  &lt;p id=&quot;sV5O&quot;&gt;Гит позволяет вести разработку с высокой степенью уверенности в сохранении результатов работы. Поскольку нейросеть регулярно ломает код, то при помощи гита можно вернуться на предыдущий этап, когда все работало.&lt;/p&gt;
  &lt;h2 id=&quot;py7l&quot;&gt;Инструменты&lt;/h2&gt;
  &lt;p id=&quot;QnvP&quot;&gt;Я заранее определил идею для приложения, у меня был готовый дизайн в iOS-приложении, поэтому переходим к выбору инструментов.&lt;/p&gt;
  &lt;h3 id=&quot;r9xL&quot;&gt;&lt;strong&gt;Android Studio&lt;/strong&gt;&lt;/h3&gt;
  &lt;p id=&quot;Ig07&quot;&gt;Тут запускаю приложение на эмуляторе и нажимаю иногда кнопку для форматирования кода, фича кстати классная, вот демо:&lt;/p&gt;
  &lt;figure id=&quot;PJVW&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img3.teletype.in/files/eb/41/eb4178a1-2e13-4d83-8d93-30bbc86cc3d3.png&quot; width=&quot;1364&quot; /&gt;
    &lt;figcaption&gt;Правык клик на папке с файлами -&amp;gt; Reformat Code&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;MGRu&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img2.teletype.in/files/18/75/18750d7f-de41-4a4b-991d-053dc700a5fd.png&quot; width=&quot;1516&quot; /&gt;
    &lt;figcaption&gt;Ставлю три галки и жму &amp;quot;Run&amp;quot;&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;h3 id=&quot;fwKm&quot;&gt;Cursor&lt;/h3&gt;
  &lt;p id=&quot;5Erf&quot;&gt;Тут происходит основная разработка.&lt;/p&gt;
  &lt;p id=&quot;8V4Q&quot;&gt;Часто после генерации кода в курсоре мне приходится вручную исправлять пропущенные импорты в Android Studio для каких-нибудь кнопок или отступов, но это не требует знания языка программирования:&lt;/p&gt;
  &lt;figure id=&quot;pF3w&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img3.teletype.in/files/28/6d/286dd40e-0707-4b0a-b9f8-db51c96091ea.png&quot; width=&quot;1524&quot; /&gt;
    &lt;figcaption&gt;Навожу курсор на слово выделенное красным, жду пока появится всплывающее окно со словами Unresolved reference, и жму на кнопку Import class ...&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;h3 id=&quot;Nom6&quot;&gt;Qwen Code CLI&lt;/h3&gt;
  &lt;p id=&quot;W7pr&quot;&gt;Крутой бесплатный (на данный момент) инструмент для разработки (и не только), всем рекомендую попробовать, &lt;a href=&quot;https://github.com/QwenLM/qwen-code&quot; target=&quot;_blank&quot;&gt;вот ссылка на гитхаб&lt;/a&gt;.&lt;/p&gt;
  &lt;p id=&quot;1tH7&quot;&gt;Использовал для составления планов, исправления мелких ошибок, больших (но простых) рутинных задач типа перемещения файлов в проекте и т.д. - может работать часами и не упирается в лимит, в отличие от курсора. Ну и отвечает менее качественно, так что компромисс.&lt;/p&gt;
  &lt;h3 id=&quot;u7Qr&quot;&gt;Perplexity&lt;/h3&gt;
  &lt;p id=&quot;2RJM&quot;&gt;Бывает, что ни курсор, ни квен не дают нормальный ответ и предлагают плохие решения, которые очевидно не работают (судя по поведению приложения в эмуляторе), и я закидываю вопрос в перплексити, где внезапно получаю рабочее решение - просто пересылаю это решение в курсор/квен и говорю применить. Иногда помогает сразу, иногда нужно поискать обходные решения, но это тоже крутой инструмент.&lt;/p&gt;
  &lt;h2 id=&quot;sENU&quot;&gt;Сроки&lt;/h2&gt;
  &lt;p id=&quot;DTWx&quot;&gt;Начал разработку 29 декабря 2025 года, релиз в русторе опубликовал через 19 дней - 17 января 2026 года. Код приложения в &lt;a href=&quot;https://github.com/easydev991/Jetpack-Days&quot; target=&quot;_blank&quot;&gt;гитхабе&lt;/a&gt; - там можно посмотреть историю коммитов (первый коммит и коммит с тегом 1.0).&lt;/p&gt;
  &lt;p id=&quot;m9L8&quot;&gt;19 дней я округлил до трех недель просто для красоты. Если считать прямо по 8 часов разработки, то выйдет около 12 дней. В отпуске я сделал 90% работы, а после отпуска за несколько вечеров все доделал и выпустил релиз.&lt;/p&gt;
  &lt;h2 id=&quot;P4sE&quot;&gt;Что мне понравилось&lt;/h2&gt;
  &lt;h3 id=&quot;VaND&quot;&gt;Работа с Android Studio&lt;/h3&gt;
  &lt;p id=&quot;A7o5&quot;&gt;После Xcode это показалось просто прорывом во многих аспектах, кроме разве что тестов. Запускать тесты в студии очень неудобно, ну или я еще не научился делать это удобно: в Xcode есть одно меню для запуска нужных тестов, а в студии я такое не нашел 🫣&lt;/p&gt;
  &lt;p id=&quot;1M3Y&quot;&gt;Больше всего понравилось:&lt;/p&gt;
  &lt;ul id=&quot;ntfN&quot;&gt;
    &lt;li id=&quot;CgBm&quot;&gt;форматирование кода, о котором раньше уже написал и показал скриншоты&lt;/li&gt;
    &lt;li id=&quot;F6Qt&quot;&gt;превью Jetpack Compose (запускается быстро, никаких крашей в фоне, удобнее чем в Xcode)&lt;/li&gt;
    &lt;li id=&quot;YHXh&quot;&gt;наличие терминала и плагинов, например, для работы с Makefile&lt;/li&gt;
  &lt;/ul&gt;
  &lt;p id=&quot;0avA&quot;&gt;В терминале я и квен запускал, и команды из Makefile (пока не нашел плагин для работы с Makefile) - этого очень не хватает в Xcode:&lt;/p&gt;
  &lt;figure id=&quot;Bjpp&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/fe/49/fe49045f-7ce0-4fce-ac48-01e9dc3fa0e6.png&quot; width=&quot;1344&quot; /&gt;
    &lt;figcaption&gt;Кнопки для терминала и плагина Makefile&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;dmTE&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/f2/0c/f20c1634-5edc-4125-8556-49fefecb8da6.png&quot; width=&quot;3504&quot; /&gt;
    &lt;figcaption&gt;Слева код, справа тесты в терминале, и все это в одном месте (Android Studio) - чудеса!&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;figure id=&quot;g5T3&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img4.teletype.in/files/30/4d/304dd16f-0ef5-4191-bb5d-2c3e445ce02f.png&quot; width=&quot;820&quot; /&gt;
    &lt;figcaption&gt;Плагин для Makefile в действии - все команды из Makefile можно запускать прямо по кнопке &amp;quot;play&amp;quot;&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;h3 id=&quot;dewW&quot;&gt;Вес приложения&lt;/h3&gt;
  &lt;p id=&quot;x7x6&quot;&gt;Андроид приложение значительно легче iOS-версии. Опубликованное в русторе приложение весит всего &lt;strong&gt;2.6 мб&lt;/strong&gt; (AAB-файл), APK в гитхабе весит всего &lt;strong&gt;3.85 мб&lt;/strong&gt;, тогда как iOS-версия весит &lt;strong&gt;5.9 мб&lt;/strong&gt;, причем около 40% веса добавилось из-за перехода с Xcode 16.4 на Xcode 26 &lt;/p&gt;
  &lt;h3 id=&quot;LE0T&quot;&gt;Минимальная версия Android&lt;/h3&gt;
  &lt;p id=&quot;0BKU&quot;&gt;Приложение работает на версии Android 8 и выше, релиз этой версии андроид был в 2017 году. Вся верстка на Jetpack Compose, стабильная версия которого вышла в 2021 году!&lt;/p&gt;
  &lt;p id=&quot;Mhhu&quot;&gt;iOS-приложение работает на версии iOS 17 и выше, релиз этой версии iOS был в 2023 году. Эту версию системы я выбрал, чтобы использовать упрощенный инструмент для хранения данных приложения (&lt;a href=&quot;https://developer.apple.com/documentation/swiftdata&quot; target=&quot;_blank&quot;&gt;Swift Data&lt;/a&gt;). Вся верстка на SwiftUI, который появился в 2019 году и был настолько дырявым, что его мало кто использовал в первые 2 года после релиза.&lt;/p&gt;
  &lt;p id=&quot;NGGh&quot;&gt;В общем, андроид ушел в большой отрыв в плане удобства верстки.&lt;/p&gt;
  &lt;h2 id=&quot;extO&quot;&gt;Что мне не понравилось&lt;/h2&gt;
  &lt;h3 id=&quot;wz1E&quot;&gt;Навигация в Jetpack Compose&lt;/h3&gt;
  &lt;p id=&quot;ncLg&quot;&gt;В iOS есть стандартные (нативные) инструменты для создания навигации любой сложности, но главное - есть простой и понятный способ как сделать простую навигацию по приложению.&lt;/p&gt;
  &lt;p id=&quot;JMMn&quot;&gt;В андроиде с навигацией намного сложнее, как и со многими другими штуками - для навигации надо подключать специальную зависимость (&lt;a href=&quot;https://developer.android.com/jetpack/androidx/releases/navigation&quot; target=&quot;_blank&quot;&gt;ссылка на документацию&lt;/a&gt;), и с ней пришлось немало повозиться, чтобы настроить безопасную зону (фронтальная камера, нижняя панель навигации) и передачу данных между экранами.&lt;/p&gt;
  &lt;h3 id=&quot;fzik&quot;&gt;Gradle&lt;/h3&gt;
  &lt;p id=&quot;gl3a&quot;&gt;Если бы это было моей работой, я бы разобрался в этом.&lt;/p&gt;
  &lt;p id=&quot;Bg8N&quot;&gt;Но поскольку это не работа (а хобби), то ну его нафиг, потому что выглядит это нифига не весело, и ни разу не очевидно 🤪&lt;/p&gt;
  &lt;p id=&quot;uF6S&quot;&gt;Нахожу решения проблем, включая обходные решения, и на этом заканчиваю разбор. &lt;/p&gt;
  &lt;h3 id=&quot;MO6m&quot;&gt;Обилие инструментов&lt;/h3&gt;
  &lt;p id=&quot;9I0O&quot;&gt;После Xcode, в котором объективно почти ничего лишнего нет (как и некоторых нужных вещей), в студии просто дофига всего, что мне не нужно, и непонятно. И это уже после их редизайна, в котором неплохо так изменили UI и спрятали многие кнопки.&lt;/p&gt;
  &lt;p id=&quot;Jlkx&quot;&gt;В том числе было непросто настроить линтер и форматирование кода - выбрал два инструмента (ktlint + detekt), которые конечно же конфликтуют в некоторых настройках &lt;/p&gt;
  &lt;p id=&quot;1lLw&quot;&gt;Выбирал не по своему опыту (т.к. у меня его нет), а по предложению гугла и нейросети, но без линтера и форматирования кода можно намесить знатную кашу, поэтому лучше уж с ними, особенно если код пишу не сам.&lt;/p&gt;
  &lt;h3 id=&quot;o92I&quot;&gt;Эмуляторы&lt;/h3&gt;
  &lt;p id=&quot;EoO1&quot;&gt;Они издают неприятные системные звуки при смене языка в настройках и вообще при запуске приложения (и еще в разных ситуациях). Похоже, пришло время погуглить как это отключить (если это вообще возможно) &lt;/p&gt;
  &lt;h2 id=&quot;GsMR&quot;&gt;Рустор&lt;/h2&gt;
  &lt;p id=&quot;vA9W&quot;&gt;Специально пишу на русском языке название этого магазина приложений, потому что, блин, это специальный магазин приложений конкретно для РФ. Нафига его называть английскими буквами?)&lt;/p&gt;
  &lt;h3 id=&quot;TWvX&quot;&gt;Требования к приложению&lt;/h3&gt;
  &lt;p id=&quot;COF4&quot;&gt;&lt;strong&gt;5 звезд&lt;/strong&gt; за требования к приложению перед релизом - минимум полей для настройки приложения перед отправкой на модерацию.&lt;/p&gt;
  &lt;p id=&quot;41kY&quot;&gt;В AppStore просто миллион полей, еще и разбросаны по разным местам на сайте.&lt;/p&gt;
  &lt;h3 id=&quot;tpV3&quot;&gt;Проверка приложения&lt;/h3&gt;
  &lt;p id=&quot;ALXg&quot;&gt;&lt;strong&gt;5 звезд&lt;/strong&gt; за скорость проверки приложения на релизе - меньше часа с момента &lt;strong&gt;первой&lt;/strong&gt; отправки приложения до одобрения и публикации.&lt;/p&gt;
  &lt;p id=&quot;bk8s&quot;&gt;В iOS у меня был рекорд по скорости публикации одного из релизов - &lt;a href=&quot;https://t.me/easy_dev991/204&quot; target=&quot;_blank&quot;&gt;меньше часа&lt;/a&gt;, но это был не первый релиз, а это важно. &lt;strong&gt;Первый&lt;/strong&gt; релиз в iOS у меня был долгим для каждого из трех приложений: модератор задает одни и те же бесполезные вопросы про монетизацию, целевую аудиторию и т.д., хотя в приложении не подключена монетизация, есть описание приложения на русском и английском языках, и заполнены все миллион обязательных полей в кабинете разработчика.&lt;/p&gt;
  &lt;h3 id=&quot;evjQ&quot;&gt;UX в русторе&lt;/h3&gt;
  &lt;p id=&quot;6ER4&quot;&gt;&lt;strong&gt;3 звезды&lt;/strong&gt; - слишком часто выдает бессмысленные ошибки в стиле &amp;quot;что-то пошло не так&amp;quot; и просит заново авторизоваться. Пишу в поддержку - они просят скриншоты. Присылаю скриншоты - они пишут, что нужны другие скриншоты, и что я что-то делаю не так.&lt;/p&gt;
  &lt;h3 id=&quot;iQbs&quot;&gt;Аналитика по приложению&lt;/h3&gt;
  &lt;figure id=&quot;JjCc&quot; class=&quot;m_column&quot;&gt;
    &lt;img src=&quot;https://img3.teletype.in/files/6c/b5/6cb5814c-0ad8-46ab-819f-e6d9669d1c8b.png&quot; width=&quot;2234&quot; /&gt;
    &lt;figcaption&gt;Вот и вся доступная аналитика&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;ayHM&quot;&gt;&lt;strong&gt;1 звезда&lt;/strong&gt; - почти никакой аналитики нет, и это плохо. Даже в AppStore есть стандартная аналитика, включая падения приложения.&lt;/p&gt;
  &lt;p id=&quot;ycQF&quot;&gt;Поддержка рустора сказала, что можно установить их инструмент для сбора аналитики в качестве отдельной зависимости, но я решил, что пока рановато, и подключил Firebase для отслеживания падений приложения. К слову, пока падений нет, как и пользователей &lt;/p&gt;
  &lt;h2 id=&quot;8ZAE&quot;&gt;Заключение&lt;/h2&gt;
  &lt;p id=&quot;2Hr7&quot;&gt;Вот такой врыв в андроид я совершил неожиданно для самого себя в конце 2025 года, и считаю это успешным результатом. Еще раз &lt;a href=&quot;https://github.com/easydev991/Jetpack-Days&quot; target=&quot;_blank&quot;&gt;ссылка на код&lt;/a&gt;, и &lt;a href=&quot;https://www.rustore.ru/catalog/app/com.dayscounter&quot; target=&quot;_blank&quot;&gt;ссылка на приложение в русторе&lt;/a&gt; - буду признателен за отзывы и тестирование &lt;/p&gt;
  &lt;p id=&quot;0VBG&quot;&gt;Приложение делал, чтобы проверить, насколько это вообще реально без знания языка, ну и всей специфики андроида, и чтобы набить руку.&lt;/p&gt;
  &lt;p id=&quot;s8OL&quot;&gt;Сейчас в свободное время делаю приложение с площадками для тренировок (аналог &lt;a href=&quot;https://github.com/easydev991/SwiftUI-WorkoutApp&quot; target=&quot;_blank&quot;&gt;iOS-версии&lt;/a&gt;), и там уже будет полный фарш с точки зрения логики и количества фичей, потому что в приложении больше 50 запросов к серверу и куча экранов, это будет интересно &lt;/p&gt;
  &lt;p id=&quot;hcrC&quot;&gt;Отдельно в платном канале напишу больше подробностей и практических примеров разработки в андроиде, включая планирование, тесты и правила/навыки ... для нейросети.&lt;/p&gt;
  &lt;p id=&quot;TsnU&quot;&gt;Спасибо за внимание!&lt;/p&gt;

</content></entry><entry><id>easy_dev991:0f3ypt50ZJT</id><link rel="alternate" type="text/html" href="https://teletype.in/@easy_dev991/0f3ypt50ZJT?utm_source=teletype&amp;utm_medium=feed_atom&amp;utm_campaign=easy_dev991"></link><title>135. Тестируем парсинг моделей</title><published>2025-12-14T09:55:02.735Z</published><updated>2025-12-14T10:28:45.064Z</updated><media:thumbnail xmlns:media="http://search.yahoo.com/mrss/" url="https://img3.teletype.in/files/a2/09/a209dad4-9415-4a2d-aa66-be82a023adb6.png"></media:thumbnail><summary type="html">&lt;img src=&quot;https://img3.teletype.in/files/62/6c/626ce370-cf46-4cd2-8e39-35978103aea9.png&quot;&gt;Недавно я выпустил в аппстор приложение для тренировок, предварительно провел много ручных тестов и написал больше тысячи автотестов. В этой статье расскажу про важность забытых мной тестов 😅</summary><content type="html">
  &lt;p id=&quot;LxAA&quot;&gt;Недавно я выпустил в аппстор &lt;a href=&quot;https://apps.apple.com/app/id6753644091&quot; target=&quot;_blank&quot;&gt;приложение для тренировок&lt;/a&gt;, предварительно провел много ручных тестов и написал больше тысячи автотестов. В этой статье расскажу про важность забытых мной тестов 😅&lt;/p&gt;
  &lt;h2 id=&quot;VPU3&quot;&gt;Тесты в первом релизе&lt;/h2&gt;
  &lt;figure id=&quot;VTP1&quot; class=&quot;m_original&quot;&gt;
    &lt;img src=&quot;https://img3.teletype.in/files/62/6c/626ce370-cf46-4cd2-8e39-35978103aea9.png&quot; width=&quot;1404&quot; /&gt;
    &lt;figcaption&gt;Автотесты на момент релиза первой версии приложения&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;7igo&quot;&gt;Вот обобщенный перечень тестов, которые были на момент первого релиза:&lt;/p&gt;
  &lt;ul id=&quot;kmEj&quot;&gt;
    &lt;li id=&quot;PACM&quot;&gt;Тесты для экрана выполнения тренировки (настройка, завершение шагов, получение результата, работа с таймером, турбо-дни и т.д.)&lt;/li&gt;
    &lt;li id=&quot;e0O7&quot;&gt;Тесты для экрана превью тренировки (построение активности дня, редактирование комментариев, обновление данных, сохранение тренировки и т.д.)&lt;/li&gt;
    &lt;li id=&quot;RH0k&quot;&gt;Тесты для управления статусом программы (получение статуса, запуск, сброс, синхронизация, журнал синхронизации, интеграционные тесты и т.д.)&lt;/li&gt;
    &lt;li id=&quot;Pgar&quot;&gt;Тесты для работы с дневником тренировок (базовые операции, комментарии, разрешение конфликтов, обработка ошибок, тренировки и т.д.)&lt;/li&gt;
    &lt;li id=&quot;z4UF&quot;&gt;Тесты для работы с инфопостами (синхронизация, парсинг, управление ресурсами, работа с YouTube, избранное и т.д.)&lt;/li&gt;
    &lt;li id=&quot;ersE&quot;&gt;Тесты для работы с прогрессом пользователя (базовые операции, синхронизация, работа с фотографиями, калькулятор прогресса и т.д.)&lt;/li&gt;
    &lt;li id=&quot;eDOK&quot;&gt;Тесты для работы с пользовательскими упражнениями&lt;/li&gt;
    &lt;li id=&quot;d5s5&quot;&gt;Тесты для настроек приложения&lt;/li&gt;
    &lt;li id=&quot;Le3o&quot;&gt;Тесты для воспроизведения звука&lt;/li&gt;
    &lt;li id=&quot;RR2L&quot;&gt;Тесты для обновления стран&lt;/li&gt;
    &lt;li id=&quot;qymy&quot;&gt;Тесты для обработки изображений&lt;/li&gt;
    &lt;li id=&quot;OzRF&quot;&gt;Тесты для управления ресурсами изображений&lt;/li&gt;
    &lt;li id=&quot;9zUI&quot;&gt;Тесты для работы с YouTube видео&lt;/li&gt;
    &lt;li id=&quot;F7bP&quot;&gt;Тесты для моделей тренировок&lt;/li&gt;
    &lt;li id=&quot;nwG3&quot;&gt;Тесты для моделей дней&lt;/li&gt;
    &lt;li id=&quot;MJUK&quot;&gt;Тесты для моделей прогресса&lt;/li&gt;
    &lt;li id=&quot;ZnOv&quot;&gt;Тесты для моделей пользователя&lt;/li&gt;
    &lt;li id=&quot;PcpC&quot;&gt;Тесты для моделей синхронизации&lt;/li&gt;
    &lt;li id=&quot;w7qA&quot;&gt;Тесты для других моделей (HomeScreenModel, ShareAppURL, TimerSound)&lt;/li&gt;
    &lt;li id=&quot;VgOS&quot;&gt;Тесты для создания программы тренировок (инициализация дня, работа с активностью дня, турбо-дни, пользовательские упражнения, типы выполнения и т.д.)&lt;/li&gt;
  &lt;/ul&gt;
  &lt;p id=&quot;mWFS&quot;&gt;У меня была цель &lt;strong&gt;протестировать всю возможную логику&lt;/strong&gt;, чтобы было проще поддерживать и развивать приложение в любое время: хоть сейчас, хоть через год, когда я скорее всего забуду большинство мелочей, влияющих на ту же синхронизацию данных с сервером и т.д.&lt;/p&gt;
  &lt;h2 id=&quot;XRO7&quot;&gt;Проблема первого релиза&lt;/h2&gt;
  &lt;p id=&quot;ODqc&quot;&gt;Про релиз сделали пост на сайте, с которым синхронизируется приложение, и в их телеграм-чате - там я случайно увидел пост одного из пользователей:&lt;/p&gt;
  &lt;figure id=&quot;t1Lj&quot; class=&quot;m_original&quot;&gt;
    &lt;img src=&quot;https://img3.teletype.in/files/e8/e2/e8e20d50-4743-47e1-a404-b0cb7d1d6cb2.jpeg&quot; width=&quot;828&quot; /&gt;
    &lt;figcaption&gt;Всем привет, была у кого такая проблема в приложении на IOS?&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;Gu1O&quot;&gt;Моя реакция, когда я увидел этот пост: 🤯🤦‍♂️&lt;/p&gt;
  &lt;p id=&quot;lYN4&quot;&gt;Для разработчика проблема очевидна: не получилось спарсить (обработать) какой-то из ответов сервера, а судя по скриншоту, это вообще происходит при первой авторизации, т.к. на главном экране ничего нет кроме алерта.&lt;/p&gt;
  &lt;p id=&quot;S7If&quot;&gt;Это был &lt;strong&gt;единственный момент, который я не протестировал&lt;/strong&gt;: зарегистрировать нового пользователя на сайте, а потом авторизоваться в iOS-приложении.&lt;/p&gt;
  &lt;p id=&quot;MEgF&quot;&gt;Т.е. я тестировал только существующих пользователей, и для них все работает корректно. Вот это поворот 🙂&lt;/p&gt;
  &lt;p id=&quot;YUin&quot;&gt;С пользователем мне повезло, он прислал мне в личку детали по воспроизведению проблемы и я сходу смог у себя это повторить.&lt;/p&gt;
  &lt;h3 id=&quot;lCfz&quot;&gt;Причина проблемы&lt;/h3&gt;
  &lt;p id=&quot;Luk6&quot;&gt;Проблема возникала при декодировании JSON ответа от сервера в структуру &lt;code&gt;CurrentRunResponse&lt;/code&gt;.&lt;/p&gt;
  &lt;p id=&quot;afKY&quot;&gt;Я использовал стандартные методы декодирования, которые не могли корректно обработать поле с датой: метод &lt;code&gt;decodeIfPresent&lt;/code&gt; не мог правильно обработать случай, когда сервер возвращает &lt;code&gt;null&lt;/code&gt; в JSON в поле с датой.&lt;/p&gt;
  &lt;p id=&quot;NT1a&quot;&gt;При этом использовалась стратегия декодирования даты &lt;code&gt;iso8601&lt;/code&gt;, которая не поддерживала формат с дробными секундами, который иногда возвращал сервер.&lt;/p&gt;
  &lt;h3 id=&quot;TkO4&quot;&gt;Почему только для новых пользователей&lt;/h3&gt;
  &lt;p id=&quot;Jayg&quot;&gt;Проблема возникала только для новых пользователей, которые еще не начинали тренироваться на сайте/в приложении, потому что сервер возвращал разные данные в зависимости от того, есть ли для пользователя запись на сервере о дате начала тренировок.&lt;/p&gt;
  &lt;p id=&quot;zGVw&quot;&gt;У существующих пользователей, которые уже успели начать тренироваться до релиза этого приложения (в старом приложении или на сайте), проблема не возникала, потому что сервер возвращал валидные данные (не &lt;code&gt;null&lt;/code&gt;) - тут стандартные методы декодирования работали корректно, потому что все поля имели ожидаемые типы данных.&lt;/p&gt;
  &lt;h3 id=&quot;Q05N&quot;&gt;Исправление проблемы&lt;/h3&gt;
  &lt;p id=&quot;SKwL&quot;&gt;Пришлось создать специальные методы декодирования, которые корректно обрабатывают все возможные случаи от сервера, и конечно на этот раз добавил тесты для этой логики, код можно посмотреть в &lt;a href=&quot;https://github.com/easydev991/SwiftUI-SotkaApp/blob/main/SwiftUI-SotkaApp/Libraries/SWNetwork/Tests/SWNetworkTests/JSONDecoderExtensionTests.swift&quot; target=&quot;_blank&quot;&gt;гитхабе&lt;/a&gt;, а тут просто перечислю сценарии:&lt;/p&gt;
  &lt;ul id=&quot;lEJ5&quot;&gt;
    &lt;li id=&quot;2j1s&quot;&gt;Должен декодировать стандартный ISO8601 формат без дробных секунд&lt;/li&gt;
    &lt;li id=&quot;IH5r&quot;&gt;Должен декодировать ISO8601 формат с таймзоной +03:00&lt;/li&gt;
    &lt;li id=&quot;jAdD&quot;&gt;Должен декодировать ISO8601 формат с дробными секундами (одна цифра)&lt;/li&gt;
    &lt;li id=&quot;sWg8&quot;&gt;Должен декодировать ISO8601 формат с дробными секундами (две цифры)&lt;/li&gt;
    &lt;li id=&quot;urXQ&quot;&gt;Должен декодировать ISO8601 формат с дробными секундами (три цифры)&lt;/li&gt;
    &lt;li id=&quot;7qdy&quot;&gt;Должен декодировать ISO8601 формат с дробными секундами (четыре цифры)&lt;/li&gt;
    &lt;li id=&quot;KbV3&quot;&gt;Должен выбрасывать ошибку для невалидной даты&lt;/li&gt;
    &lt;li id=&quot;qiG2&quot;&gt;Должен выбрасывать ошибку для пустой строки&lt;/li&gt;
    &lt;li id=&quot;FpqM&quot;&gt;Должен выбрасывать ошибку для неправильного формата даты&lt;/li&gt;
    &lt;li id=&quot;5fC8&quot;&gt;Должен декодировать опциональное поле Date? с валидной датой&lt;/li&gt;
    &lt;li id=&quot;9cCz&quot;&gt;Должен декодировать опциональное поле Date? с null значением&lt;/li&gt;
    &lt;li id=&quot;ZSzQ&quot;&gt;Должен декодировать опциональное поле Date? с отсутствующим ключом&lt;/li&gt;
    &lt;li id=&quot;jgKe&quot;&gt;Должен декодировать server date time формат без часового пояса&lt;/li&gt;
    &lt;li id=&quot;ecn2&quot;&gt;Должен декодировать опциональное поле Date? с server date time форматом&lt;/li&gt;
    &lt;li id=&quot;e8HJ&quot;&gt;Должен декодировать минимальную дату&lt;/li&gt;
    &lt;li id=&quot;Mn06&quot;&gt;Должен декодировать минимальную дату в server date time формате&lt;/li&gt;
    &lt;li id=&quot;udHu&quot;&gt;Должен декодировать дату с максимальными дробными секундами&lt;/li&gt;
    &lt;li id=&quot;AE9x&quot;&gt;Должен декодировать ISO short date формат&lt;/li&gt;
    &lt;li id=&quot;gwMg&quot;&gt;Должен декодировать опциональное поле Date? с ISO short date форматом&lt;/li&gt;
  &lt;/ul&gt;
  &lt;h2 id=&quot;be1h&quot;&gt;Тесты во втором релизе&lt;/h2&gt;
  &lt;figure id=&quot;AGT7&quot; class=&quot;m_original&quot;&gt;
    &lt;img src=&quot;https://img1.teletype.in/files/49/f9/49f92614-0687-42d2-89e6-540277ef5424.png&quot; width=&quot;1410&quot; /&gt;
    &lt;figcaption&gt;Автотесты на момент второго релиза приложения&lt;/figcaption&gt;
  &lt;/figure&gt;
  &lt;p id=&quot;cd1Z&quot;&gt;Все тесты генерировал с помощью нейросети с использованием нескольких простых правил, о чем напишу в другой раз, а на исправление проблемы вместе с написанием новых тестов потратил около часа.&lt;/p&gt;
  &lt;p id=&quot;PzyP&quot;&gt;После этого оформил запрос на ускоренную проверку релиза и ... два дня разбирался с модераторами и отвечал на их вопросы 😁&lt;/p&gt;
  &lt;h2 id=&quot;19QT&quot;&gt;Заключение&lt;/h2&gt;
  &lt;p id=&quot;Ap8L&quot;&gt;Тестируем не только основную логику приложения, но и декодирование моделей, особенно в пет-проектах ☝️&lt;/p&gt;
  &lt;h3 id=&quot;7EfF&quot;&gt;Бонус&lt;/h3&gt;
  &lt;p id=&quot;TOnO&quot;&gt;Год назад еще можно было сказать, что автотесты - это дорого, сложно, непонятно, непривычно и так далее. Потому что многие еще не умели пользоваться или не знали про режим агента у популярных IDE типа cursor/windsurf.&lt;/p&gt;
  &lt;p id=&quot;TEHu&quot;&gt;Сейчас нельзя сказать, что ты не умеешь писать тесты, или что это дорого и долго - подобные вещи будут просто наглым враньем.&lt;/p&gt;
  &lt;p id=&quot;YEHB&quot;&gt;Если не хочется разбираться в новых инструментах - нужно так и говорить прямо, что все, пора на пенсию, потому что слишком сложно сказать нейросети написать тесты для логики или для модели.&lt;/p&gt;
  &lt;p id=&quot;EBlm&quot;&gt;Предлагаю всем, кто еще не пробовал генерировать тесты с нейросетью - начните это делать наконец-то, чтобы не ковыряться во внезапных багах и не ломать уже существующий функционал наших классных приложений 😉&lt;/p&gt;
  &lt;p id=&quot;4Nmw&quot;&gt;Если кто-то переживает, что у нас станет меньше работы при снижении количества багов, то это заблуждение - работы меньше не станет, как и багов, зато стабильности будет больше 👍&lt;/p&gt;

</content></entry></feed>