[{"data":1,"prerenderedAt":10424},["ShallowReactive",2],{"navigation_docs":3,"-java-pr2-10a-code-quality":2949,"-java-pr2-10a-code-quality-surround":10419},[4,1640,1765,2219,2352,2559,2641,2691,2748,2782,2908,2945],{"title":5,"icon":6,"path":7,"stem":8,"children":9},"C#","i-devicon-csharp","/csharp","01.csharp",[10,13,60,90,120,202,219,253,379,404,457,650,1346,1636],{"title":11,"path":7,"stem":12},"C# Roadmap","01.csharp/index",{"title":14,"icon":15,"path":16,"stem":17,"children":18,"page":59},"Fundamentals","i-lucide-book-open","/csharp/fundamentals","01.csharp/01.fundamentals",[19,23,27,31,35,39,43,47,51,55],{"title":20,"path":21,"stem":22},"Вступ до екосистеми .NET","/csharp/fundamentals/introduction-to-ecosystem","01.csharp/01.fundamentals/01.introduction-to-ecosystem",{"title":24,"path":25,"stem":26},"Структура програми на C#","/csharp/fundamentals/program-structure","01.csharp/01.fundamentals/02.program-structure",{"title":28,"path":29,"stem":30},"Змінні та Типи Даних","/csharp/fundamentals/variables-data-types","01.csharp/01.fundamentals/03.variables-data-types",{"title":32,"path":33,"stem":34},"Масиви","/csharp/fundamentals/arrays","01.csharp/01.fundamentals/04.arrays",{"title":36,"path":37,"stem":38},"Strings & Text Handling","/csharp/fundamentals/strings-text-handling","01.csharp/01.fundamentals/05.strings-text-handling",{"title":40,"path":41,"stem":42},"Дати і Час","/csharp/fundamentals/dates-time-handling","01.csharp/01.fundamentals/06.dates-time-handling",{"title":44,"path":45,"stem":46},"Потік Керування","/csharp/fundamentals/control-flow","01.csharp/01.fundamentals/07.control-flow",{"title":48,"path":49,"stem":50},"Методи","/csharp/fundamentals/methods","01.csharp/01.fundamentals/08.methods",{"title":52,"path":53,"stem":54},"Основи Відлагодження","/csharp/fundamentals/debugging-basics","01.csharp/01.fundamentals/09.debugging-basics",{"title":56,"path":57,"stem":58},"Інтерактивна Консоль (Classic)","/csharp/fundamentals/interactive-console","01.csharp/01.fundamentals/10.interactive-console",false,{"title":61,"icon":62,"path":63,"stem":64,"children":65,"page":59},"OOP","i-lucide-box","/csharp/oop","01.csharp/02.oop",[66,70,74,78,82,86],{"title":67,"path":68,"stem":69},"Package Management (Управління Пакетами)","/csharp/oop/package-management","01.csharp/02.oop/01.package-management",{"title":71,"path":72,"stem":73},"Класи та Об'єкти","/csharp/oop/classes-objects","01.csharp/02.oop/02.classes-objects",{"title":75,"path":76,"stem":77},"Властивості та Поля","/csharp/oop/properties-fields","01.csharp/02.oop/03.properties-fields",{"title":79,"path":80,"stem":81},"Стовпи ООП","/csharp/oop/oop-pillars","01.csharp/02.oop/04.oop-pillars",{"title":83,"path":84,"stem":85},"Advanced Types","/csharp/oop/advanced-types","01.csharp/02.oop/05.advanced-types",{"title":87,"path":88,"stem":89},"Namespaces (Простори Імен)","/csharp/oop/namespaces","01.csharp/02.oop/06.namespaces",{"title":91,"icon":92,"path":93,"stem":94,"children":95,"page":59},"Advanced Core","i-lucide-zap","/csharp/advanced-core","01.csharp/03.advanced-core",[96,100,104,108,112,116],{"title":97,"path":98,"stem":99},"Generics (Узагальнення)","/csharp/advanced-core/generics","01.csharp/03.advanced-core/01.generics",{"title":101,"path":102,"stem":103},"Делегати, Події та Лямбда-вирази","/csharp/advanced-core/delegates-events-lambdas","01.csharp/03.advanced-core/02.delegates-events-lambdas",{"title":105,"path":106,"stem":107},"Interfaces Deep Dive (Інтерфейси: Поглиблений Розгляд)","/csharp/advanced-core/interfaces-deep-dive","01.csharp/03.advanced-core/03.interfaces-deep-dive",{"title":109,"path":110,"stem":111},"Обробка Винятків","/csharp/advanced-core/exception-handling","01.csharp/03.advanced-core/04.exception-handling",{"title":113,"path":114,"stem":115},"Pattern Matching","/csharp/advanced-core/pattern-matching","01.csharp/03.advanced-core/05.pattern-matching",{"title":117,"path":118,"stem":119},"Додаткові Можливості C#","/csharp/advanced-core/additional-features","01.csharp/03.advanced-core/06.additional-features",{"title":121,"icon":122,"path":123,"stem":124,"children":125,"page":59},"Architecture Best Practices","i-lucide-building-2","/csharp/architecture-best-practices","01.csharp/04.architecture-best-practices",[126,130,149,153,157,161,165,169],{"title":127,"path":128,"stem":129},"Software Design Principles (Частина 1)","/csharp/architecture-best-practices/software-design-principles","01.csharp/04.architecture-best-practices/01.software-design-principles",{"title":131,"icon":132,"path":133,"stem":134,"children":135,"page":59},"Design Patterns","i-lucide-folder","/csharp/architecture-best-practices/design-patterns","01.csharp/04.architecture-best-practices/02.design-patterns",[136],{"title":137,"icon":132,"path":138,"stem":139,"children":140,"page":59},"Creational","/csharp/architecture-best-practices/design-patterns/creational","01.csharp/04.architecture-best-practices/02.design-patterns/creational",[141,145],{"title":142,"path":143,"stem":144},"Singleton (Одинак)","/csharp/architecture-best-practices/design-patterns/creational/singleton","01.csharp/04.architecture-best-practices/02.design-patterns/creational/01.singleton",{"title":146,"path":147,"stem":148},"Builder (Будівельник)","/csharp/architecture-best-practices/design-patterns/creational/builder","01.csharp/04.architecture-best-practices/02.design-patterns/creational/02.builder",{"title":150,"path":151,"stem":152},"Building Professional CLIs","/csharp/architecture-best-practices/building-professional-clis","01.csharp/04.architecture-best-practices/03.building-professional-clis",{"title":154,"path":155,"stem":156},"Validation & Flow Control","/csharp/architecture-best-practices/validation-flow-control","01.csharp/04.architecture-best-practices/04.validation-flow-control",{"title":158,"path":159,"stem":160},"The Modern .NET Host (Microsoft.Extensions)","/csharp/architecture-best-practices/modern-dotnet-host","01.csharp/04.architecture-best-practices/05.modern-dotnet-host",{"title":162,"path":163,"stem":164},"Data Mapper: Repository та DAO патерни (Частина 1)","/csharp/architecture-best-practices/data-mapper-part1","01.csharp/04.architecture-best-practices/06.data-mapper-part1",{"title":166,"path":167,"stem":168},"Data Mapper: Repository та DAO патерни (Частина 2)","/csharp/architecture-best-practices/data-mapper-part2","01.csharp/04.architecture-best-practices/07.data-mapper-part2",{"title":170,"icon":132,"path":171,"stem":172,"children":173,"page":59},"Di Ioc","/csharp/architecture-best-practices/di-ioc","01.csharp/04.architecture-best-practices/08.di-ioc",[174,178,182,186,190,194,198],{"title":175,"path":176,"stem":177},"Проблема залежностей та Інверсія Контролю","/csharp/architecture-best-practices/di-ioc/the-dependency-problem","01.csharp/04.architecture-best-practices/08.di-ioc/01.the-dependency-problem",{"title":179,"path":180,"stem":181},"Будуємо власний Service Container","/csharp/architecture-best-practices/di-ioc/build-your-own-container","01.csharp/04.architecture-best-practices/08.di-ioc/02.build-your-own-container",{"title":183,"path":184,"stem":185},"Service Locator: Паттерн та Анти-паттерн","/csharp/architecture-best-practices/di-ioc/service-locator-pattern","01.csharp/04.architecture-best-practices/08.di-ioc/03.service-locator-pattern",{"title":187,"path":188,"stem":189},"Паттерни Dependency Injection","/csharp/architecture-best-practices/di-ioc/dependency-injection-patterns","01.csharp/04.architecture-best-practices/08.di-ioc/04.dependency-injection-patterns",{"title":191,"path":192,"stem":193},"Microsoft DI: IServiceCollection та IServiceProvider","/csharp/architecture-best-practices/di-ioc/microsoft-di-deep-dive","01.csharp/04.architecture-best-practices/08.di-ioc/05.microsoft-di-deep-dive",{"title":195,"path":196,"stem":197},"Service Lifetimes та Scopes","/csharp/architecture-best-practices/di-ioc/service-lifetimes-and-scopes","01.csharp/04.architecture-best-practices/08.di-ioc/06.service-lifetimes-and-scopes",{"title":199,"path":200,"stem":201},"DI Анти-паттерни та Найкращі Практики","/csharp/architecture-best-practices/di-ioc/di-anti-patterns-and-best-practices","01.csharp/04.architecture-best-practices/08.di-ioc/07.di-anti-patterns-and-best-practices",{"title":203,"icon":132,"path":204,"stem":205,"children":206,"page":59},"Standard Library","/csharp/standard-library","01.csharp/05.standard-library",[207,211,215],{"title":208,"path":209,"stem":210},"Collections (Колекції)","/csharp/standard-library/collections","01.csharp/05.standard-library/01.collections",{"title":212,"path":213,"stem":214},"High Performance Types (Високопродуктивні Типи)","/csharp/standard-library/high-performance-types","01.csharp/05.standard-library/02.high-performance-types",{"title":216,"path":217,"stem":218},"LINQ (Language Integrated Query)","/csharp/standard-library/linq","01.csharp/05.standard-library/03.linq",{"title":220,"icon":221,"path":222,"stem":223,"children":224,"page":59},"System Internals Concurrency","i-lucide-server","/csharp/system-internals-concurrency","01.csharp/06.system-internals-concurrency",[225,229,233,237,241,245,249],{"title":226,"path":227,"stem":228},"Memory Management","/csharp/system-internals-concurrency/memory-management","01.csharp/06.system-internals-concurrency/01.memory-management",{"title":230,"path":231,"stem":232},"Reflection API: System.Type та Метадані","/csharp/system-internals-concurrency/reflection-fundamentals","01.csharp/06.system-internals-concurrency/02.reflection-fundamentals",{"title":234,"path":235,"stem":236},"Attributes та Dynamic Language Runtime","/csharp/system-internals-concurrency/attributes-dynamic","01.csharp/06.system-internals-concurrency/03.attributes-dynamic",{"title":238,"path":239,"stem":240},"Expression Trees: Швидка Альтернатива Рефлексії","/csharp/system-internals-concurrency/expression-trees-compiled","01.csharp/06.system-internals-concurrency/04.expression-trees-compiled",{"title":242,"path":243,"stem":244},"Source Generators: Compile-Time Code Generation","/csharp/system-internals-concurrency/source-generators","01.csharp/06.system-internals-concurrency/05.source-generators",{"title":246,"path":247,"stem":248},"Multithreading Fundamentals","/csharp/system-internals-concurrency/multithreading-fundamentals","01.csharp/06.system-internals-concurrency/06.multithreading-fundamentals",{"title":250,"path":251,"stem":252},"Synchronization Primitives","/csharp/system-internals-concurrency/synchronization-primitives","01.csharp/06.system-internals-concurrency/07.synchronization-primitives",{"title":254,"icon":255,"path":256,"stem":257,"children":258,"page":59},"System Programming Windows","i-lucide-cpu","/csharp/system-programming-windows","01.csharp/07.system-programming-windows",[259,263,267,271,275,279,283,287,291,295,299,303,307,311,315,319,323,327,331,335,339,343,347,351,355,359,363,367,371,375],{"title":260,"path":261,"stem":262},"Як Працює Операційна Система","/csharp/system-programming-windows/how-os-works","01.csharp/07.system-programming-windows/01.how-os-works",{"title":264,"path":265,"stem":266},"Процеси в .NET — API та Запуск","/csharp/system-programming-windows/processes-in-dotnet","01.csharp/07.system-programming-windows/02.processes-in-dotnet",{"title":268,"path":269,"stem":270},"Процеси в .NET — IPC та Міжпроцесна Комунікація","/csharp/system-programming-windows/02a.processes-ipc","01.csharp/07.system-programming-windows/02a.processes-ipc",{"title":272,"path":273,"stem":274},"Application Domains та Збірки — AppDomain і AssemblyLoadContext","/csharp/system-programming-windows/appdomains-assemblies","01.csharp/07.system-programming-windows/03.appdomains-assemblies",{"title":276,"path":277,"stem":278},"Application Domains та Збірки — Plug-in Система з Hot-Reload","/csharp/system-programming-windows/03a.appdomains-plugin-system","01.csharp/07.system-programming-windows/03a.appdomains-plugin-system",{"title":280,"path":281,"stem":282},"Потоки — Основи та API Thread","/csharp/system-programming-windows/thread-fundamentals","01.csharp/07.system-programming-windows/04.thread-fundamentals",{"title":284,"path":285,"stem":286},"Потоки — Lifecycle, Пріоритети та Безпечне Завершення","/csharp/system-programming-windows/04a.thread-lifecycle-priorities","01.csharp/07.system-programming-windows/04a.thread-lifecycle-priorities",{"title":288,"path":289,"stem":290},"Проблеми Спільного Стану — Race Condition та Data Race","/csharp/system-programming-windows/shared-state-problems","01.csharp/07.system-programming-windows/05.shared-state-problems",{"title":292,"path":293,"stem":294},"Проблеми Спільного Стану — Memory Model та volatile","/csharp/system-programming-windows/05a.shared-state-memory-model","01.csharp/07.system-programming-windows/05a.shared-state-memory-model",{"title":296,"path":297,"stem":298},"Синхронізація — Monitor, lock та еволюція примітивів","/csharp/system-programming-windows/synchronization-fundamentals","01.csharp/07.system-programming-windows/06.synchronization-fundamentals",{"title":300,"path":301,"stem":302},"Синхронізація — Наскрізний Приклад та Deadlock Detection","/csharp/system-programming-windows/06a.synchronization-walkthrough","01.csharp/07.system-programming-windows/06a.synchronization-walkthrough",{"title":304,"path":305,"stem":306},"Синхронізація — Mutex, Semaphore та Event-Based Primitives","/csharp/system-programming-windows/synchronization-advanced","01.csharp/07.system-programming-windows/07.synchronization-advanced",{"title":308,"path":309,"stem":310},"Синхронізація — Interlocked, Volatile та Lock-Free Структури","/csharp/system-programming-windows/07a.synchronization-advanced-walkthrough","01.csharp/07.system-programming-windows/07a.synchronization-advanced-walkthrough",{"title":312,"path":313,"stem":314},"Interlocked, CAS та Lock-Free Структури","/csharp/system-programming-windows/interlocked-cas-lockfree","01.csharp/07.system-programming-windows/08.interlocked-cas-lockfree",{"title":316,"path":317,"stem":318},"Volatile, Memory Model та Spinning","/csharp/system-programming-windows/08a.volatile-memory-model","01.csharp/07.system-programming-windows/08a.volatile-memory-model",{"title":320,"path":321,"stem":322},"ThreadPool — Пул Потоків для Ефективного Виконання","/csharp/system-programming-windows/thread-pool","01.csharp/07.system-programming-windows/09.thread-pool",{"title":324,"path":325,"stem":326},"ThreadPool — Просунуті Сценарії та Внутрішня Будова","/csharp/system-programming-windows/09a.thread-pool-advanced","01.csharp/07.system-programming-windows/09a.thread-pool-advanced",{"title":328,"path":329,"stem":330},"Concurrent та Immutable Collections","/csharp/system-programming-windows/concurrent-collections","01.csharp/07.system-programming-windows/10.concurrent-collections",{"title":332,"path":333,"stem":334},"TPL, Task та Композиція — Від Thread до Task","/csharp/system-programming-windows/tpl-parallel-plinq","01.csharp/07.system-programming-windows/11.tpl-parallel-plinq",{"title":336,"path":337,"stem":338},"Parallel Class та PLINQ — Data Parallelism","/csharp/system-programming-windows/11a.tpl-parallel-plinq-advanced","01.csharp/07.system-programming-windows/11a.tpl-parallel-plinq-advanced",{"title":340,"path":341,"stem":342},"Async/Await — Фундамент Асинхронного Програмування","/csharp/system-programming-windows/async-fundamentals","01.csharp/07.system-programming-windows/12.async-fundamentals",{"title":344,"path":345,"stem":346},"SynchronizationContext та ConfigureAwait — Контекст Виконання","/csharp/system-programming-windows/async-context-configureawait","01.csharp/07.system-programming-windows/13.async-context-configureawait",{"title":348,"path":349,"stem":350},"Async — Просунуті Паттерни","/csharp/system-programming-windows/async-advanced","01.csharp/07.system-programming-windows/14.async-advanced",{"title":352,"path":353,"stem":354},"System.Threading.Channels — Async Producer-Consumer","/csharp/system-programming-windows/channels","01.csharp/07.system-programming-windows/15.channels",{"title":356,"path":357,"stem":358},"Асинхронна Синхронізація","/csharp/system-programming-windows/async-synchronization","01.csharp/07.system-programming-windows/16.async-synchronization",{"title":360,"path":361,"stem":362},"Unsafe Code та Вказівники","/csharp/system-programming-windows/unsafe-code","01.csharp/07.system-programming-windows/17.unsafe-code",{"title":364,"path":365,"stem":366},"P/Invoke та Windows API — Міст між .NET та Native Code","/csharp/system-programming-windows/pinvoke-winapi","01.csharp/07.system-programming-windows/18.pinvoke-winapi",{"title":368,"path":369,"stem":370},"Реєстр Windows — Центральна База Конфігурації Системи","/csharp/system-programming-windows/windows-registry","01.csharp/07.system-programming-windows/19.windows-registry",{"title":372,"path":373,"stem":374},"Windows Hooks, Hotkeys та Services — Глибока Інтеграція з ОС","/csharp/system-programming-windows/windows-hooks-services","01.csharp/07.system-programming-windows/20.windows-hooks-services",{"title":376,"path":377,"stem":378},"Системне Програмування C# (Windows) — 07.system-programming-windows","/csharp/system-programming-windows/implementation_plan","01.csharp/07.system-programming-windows/implementation_plan",{"title":380,"icon":132,"path":381,"stem":382,"children":383,"page":59},"Io","/csharp/io","01.csharp/08.io",[384,388,392,396,400],{"title":385,"path":386,"stem":387},"8.1.1. Основи роботи з файловою системою","/csharp/io/file-system-basics","01.csharp/08.io/01.file-system-basics",{"title":389,"path":390,"stem":391},"8.1.2. Потоки (Streams) та Серіалізація Даних","/csharp/io/streams-serialization","01.csharp/08.io/02.streams-serialization",{"title":393,"path":394,"stem":395},"8.2.1. JSON Serialization з System.Text.Json","/csharp/io/json-serialization","01.csharp/08.io/03.json-serialization",{"title":397,"path":398,"stem":399},"8.2.2. XML Serialization та LINQ to XML","/csharp/io/xml-serialization","01.csharp/08.io/04.xml-serialization",{"title":401,"path":402,"stem":403},"8.2.3. Binary Serialization: MessagePack та Protocol Buffers","/csharp/io/binary-serialization","01.csharp/08.io/05.binary-serialization",{"title":405,"icon":132,"path":406,"stem":407,"children":408,"page":59},"Ado Net","/csharp/ado-net","01.csharp/09.ado-net",[409,413,417,421,425,429,433,437,441,445,449,453],{"title":410,"path":411,"stem":412},"9.1. Введення в ADO.NET","/csharp/ado-net/introduction-to-adonet","01.csharp/09.ado-net/01.introduction-to-adonet",{"title":414,"path":415,"stem":416},"9.2. Клас DbConnection — з'єднання з базою даних","/csharp/ado-net/connection","01.csharp/09.ado-net/02.connection",{"title":418,"path":419,"stem":420},"9.3. Клас DbCommand — виконання SQL-запитів","/csharp/ado-net/command-and-queries","01.csharp/09.ado-net/03.command-and-queries",{"title":422,"path":423,"stem":424},"9.4. Клас DbDataReader — ефективне читання даних","/csharp/ado-net/datareader","01.csharp/09.ado-net/04.datareader",{"title":426,"path":427,"stem":428},"9.5. Параметризовані запити та захист від SQL Injection","/csharp/ado-net/parameters-and-sql-injection","01.csharp/09.ado-net/05.parameters-and-sql-injection",{"title":430,"path":431,"stem":432},"9.6. Транзакції в ADO.NET","/csharp/ado-net/transactions","01.csharp/09.ado-net/06.transactions",{"title":434,"path":435,"stem":436},"9.7. DbProviderFactory — провайдер-незалежний код","/csharp/ado-net/provider-factory","01.csharp/09.ado-net/07.provider-factory",{"title":438,"path":439,"stem":440},"9.8. Асинхронний доступ до даних","/csharp/ado-net/async-data-access","01.csharp/09.ado-net/08.async-data-access",{"title":442,"path":443,"stem":444},"9.9. Від'єднаний режим: DataSet, DataTable, DataRow","/csharp/ado-net/disconnected-mode-dataset","01.csharp/09.ado-net/09.disconnected-mode-dataset",{"title":446,"path":447,"stem":448},"9.10. DataAdapter — міст між DataSet та базою даних","/csharp/ado-net/data-adapter","01.csharp/09.ado-net/10.data-adapter",{"title":450,"path":451,"stem":452},"9.11. Data Mapper та Repository: Архітектура доступу до даних","/csharp/ado-net/data-mapper-repository","01.csharp/09.ado-net/11.data-mapper-repository",{"title":454,"path":455,"stem":456},"9.12. Identity Map, Unit of Work та Specification Pattern","/csharp/ado-net/advanced-patterns","01.csharp/09.ado-net/12.advanced-patterns",{"title":458,"icon":255,"path":459,"stem":460,"children":461,"page":59},"Ef Core","/csharp/ef-core","01.csharp/10.ef-core",[462,466,470,474,478,482,486,490,494,498,502,506,510,514,518,522,526,532,538,542,546,550,554,558,562,566,570,574,578,582,586,590,594,598,602,606,610,614,618,622,626,630,634,638,642,646],{"title":463,"path":464,"stem":465},"Що таке ORM? Від SQL до об'єктів","/csharp/ef-core/what-is-orm","01.csharp/10.ef-core/01.what-is-orm",{"title":467,"path":468,"stem":469},"Перший проєкт — від нуля до CRUD","/csharp/ef-core/first-project","01.csharp/10.ef-core/02.first-project",{"title":471,"path":472,"stem":473},"DbContext — Серце EF Core","/csharp/ef-core/dbcontext-deep-dive","01.csharp/10.ef-core/03.dbcontext-deep-dive",{"title":475,"path":476,"stem":477},"Провайдери баз даних — Архітектура та Вибір СУБД","/csharp/ef-core/database-providers","01.csharp/10.ef-core/04.database-providers",{"title":479,"path":480,"stem":481},"Конвенції EF Core — Магія без конфігурації","/csharp/ef-core/conventions","01.csharp/10.ef-core/05.conventions",{"title":483,"path":484,"stem":485},"Fluent API та Data Annotations — Явна конфігурація моделі","/csharp/ef-core/fluent-api-vs-annotations","01.csharp/10.ef-core/06.fluent-api-vs-annotations",{"title":487,"path":488,"stem":489},"Зв'язки — One-to-One та One-to-Many","/csharp/ef-core/relationships-basics","01.csharp/10.ef-core/07.relationships-basics",{"title":491,"path":492,"stem":493},"Зв'язки Advanced — Many-to-Many та Складні Сценарії","/csharp/ef-core/relationships-advanced","01.csharp/10.ef-core/08.relationships-advanced",{"title":495,"path":496,"stem":497},"Властивості — Типи, Конвертери, Компаратори (Частина 1)","/csharp/ef-core/property-configuration-part1","01.csharp/10.ef-core/09.property-configuration-part1",{"title":499,"path":500,"stem":501},"Властивості — Value Comparers, Generators, Shadow Properties (Частина 2)","/csharp/ef-core/property-configuration-part2","01.csharp/10.ef-core/09.property-configuration-part2",{"title":503,"path":504,"stem":505},"Складні типи — Owned Types та Complex Types (Частина 1)","/csharp/ef-core/complex-types-owned-part1","01.csharp/10.ef-core/10.complex-types-owned-part1",{"title":507,"path":508,"stem":509},"Складні типи — Complex Types, Keyless Entities, Порівняння (Частина 2)","/csharp/ef-core/complex-types-owned-part2","01.csharp/10.ef-core/10.complex-types-owned-part2",{"title":511,"path":512,"stem":513},"JSON Columns — Складні дані у JSON (Частина 1)","/csharp/ef-core/json-columns-part1","01.csharp/10.ef-core/11.json-columns-part1",{"title":515,"path":516,"stem":517},"JSON Columns — Value Comparers, Індекси, Провайдери (Частина 2)","/csharp/ef-core/json-columns-part2","01.csharp/10.ef-core/11.json-columns-part2",{"title":519,"path":520,"stem":521},"Успадкування — Абстрактні класи та TPH (Частина 1)","/csharp/ef-core/inheritance-part1","01.csharp/10.ef-core/12.inheritance-part1",{"title":523,"path":524,"stem":525},"Успадкування — TPT, TPC та Порівняння Стратегій (Частина 2)","/csharp/ef-core/inheritance-part2","01.csharp/10.ef-core/12.inheritance-part2",{"title":527,"path":528,"stem":529,"children":530},"Індекси, Обмеження та Схема (Частина 1)","/csharp/ef-core/indexes-constraints-part1","01.csharp/10.ef-core/13.indexes-constraints-part1",[531],{"title":527,"path":528,"stem":529},{"title":533,"path":534,"stem":535,"children":536},"Індекси, Обмеження та Схема (Частина 2)","/csharp/ef-core/indexes-constraints-part2","01.csharp/10.ef-core/13.indexes-constraints-part2",[537],{"title":533,"path":534,"stem":535},{"title":539,"path":540,"stem":541},"Seed Data — Початкові Дані (Частина 1)","/csharp/ef-core/seeding-part1","01.csharp/10.ef-core/14.seeding-part1",{"title":543,"path":544,"stem":545},"Seed Data — SQL-скрипти, Bogus та Стратегії (Частина 2)","/csharp/ef-core/seeding-part2","01.csharp/10.ef-core/14.seeding-part2",{"title":547,"path":548,"stem":549},"Global Query Filters — Глобальні Фільтри (Частина 1)","/csharp/ef-core/global-query-filters-part1","01.csharp/10.ef-core/15.global-query-filters-part1",{"title":551,"path":552,"stem":553},"Global Query Filters — Підводні камені та Інтеграція (Частина 2)","/csharp/ef-core/global-query-filters-part2","01.csharp/10.ef-core/15.global-query-filters-part2",{"title":555,"path":556,"stem":557},"LINQ-запити в EF Core (Частина 1)","/csharp/ef-core/linq-queries-part1","01.csharp/10.ef-core/16.linq-queries-part1",{"title":559,"path":560,"stem":561},"LINQ-запити в EF Core (Частина 2)","/csharp/ef-core/linq-queries-part2","01.csharp/10.ef-core/16.linq-queries-part2",{"title":563,"path":564,"stem":565},"Завантаження Пов'язаних Даних (Частина 1)","/csharp/ef-core/loading-related-data-part1","01.csharp/10.ef-core/17.loading-related-data-part1",{"title":567,"path":568,"stem":569},"Завантаження Пов'язаних Даних (Частина 2)","/csharp/ef-core/loading-related-data-part2","01.csharp/10.ef-core/17.loading-related-data-part2",{"title":571,"path":572,"stem":573},"Raw SQL, Views та Stored Procedures (Частина 1)","/csharp/ef-core/raw-sql-part1","01.csharp/10.ef-core/18.raw-sql-part1",{"title":575,"path":576,"stem":577},"Raw SQL — Stored Procedures, DbFunction та Bulk Operations (Частина 2)","/csharp/ef-core/raw-sql-part2","01.csharp/10.ef-core/18.raw-sql-part2",{"title":579,"path":580,"stem":581},"Продвинуті Запити — Compiled Queries, Bulk та Оптимізація (Частина 1)","/csharp/ef-core/advanced-queries-part1","01.csharp/10.ef-core/19.advanced-queries-part1",{"title":583,"path":584,"stem":585},"Продвинуті Запити — Query Tags, Bulk та Interceptors (Частина 2)","/csharp/ef-core/advanced-queries-part2","01.csharp/10.ef-core/19.advanced-queries-part2",{"title":587,"path":588,"stem":589},"Change Tracker — Відстеження Змін (Частина 1)","/csharp/ef-core/change-tracking-part1","01.csharp/10.ef-core/20.change-tracking-part1",{"title":591,"path":592,"stem":593},"Change Tracker — Графи Об'єктів та Disconnected (Частина 2)","/csharp/ef-core/change-tracking-part2","01.csharp/10.ef-core/20.change-tracking-part2",{"title":595,"path":596,"stem":597},"Збереження Даних та Транзакції (Частина 1)","/csharp/ef-core/saving-data-part1","01.csharp/10.ef-core/21.saving-data-part1",{"title":599,"path":600,"stem":601},"Збереження Даних — Concurrency та Outbox (Частина 2)","/csharp/ef-core/saving-data-part2","01.csharp/10.ef-core/21.saving-data-part2",{"title":603,"path":604,"stem":605},"Конкурентність та Блокування (Частина 1)","/csharp/ef-core/concurrency-part1","01.csharp/10.ef-core/22.concurrency-part1",{"title":607,"path":608,"stem":609},"Конкурентність — Дедлоки та Queue Processing (Частина 2)","/csharp/ef-core/concurrency-part2","01.csharp/10.ef-core/22.concurrency-part2",{"title":611,"path":612,"stem":613},"Міграції в EF Core — Основи (Частина 1)","/csharp/ef-core/migrations-basics-part1","01.csharp/10.ef-core/23.migrations-basics-part1",{"title":615,"path":616,"stem":617},"Міграції в EF Core — Основи (Частина 2)","/csharp/ef-core/migrations-basics-part2","01.csharp/10.ef-core/23.migrations-basics-part2",{"title":619,"path":620,"stem":621},"Міграції — Просунуті Сценарії (Частина 1)","/csharp/ef-core/migrations-advanced-part1","01.csharp/10.ef-core/24.migrations-advanced-part1",{"title":623,"path":624,"stem":625},"Міграції — Просунуті Сценарії (Частина 2)","/csharp/ef-core/migrations-advanced-part2","01.csharp/10.ef-core/24.migrations-advanced-part2",{"title":627,"path":628,"stem":629},"Управління Схемою та Database-First (Частина 1)","/csharp/ef-core/schema-management-part1","01.csharp/10.ef-core/25.schema-management-part1",{"title":631,"path":632,"stem":633},"Управління Схемою та Database-First (Частина 2)","/csharp/ef-core/schema-management-part2","01.csharp/10.ef-core/25.schema-management-part2",{"title":635,"path":636,"stem":637},"Продуктивність EF Core — Основи (Частина 1)","/csharp/ef-core/performance-fundamentals-part1","01.csharp/10.ef-core/26.performance-fundamentals-part1",{"title":639,"path":640,"stem":641},"Interceptors в EF Core (Частина 1)","/csharp/ef-core/interceptors-part1","01.csharp/10.ef-core/29.interceptors-part1",{"title":643,"path":644,"stem":645},"Interceptors в EF Core — Connection, Transaction та Materialization (Частина 2)","/csharp/ef-core/interceptors-part2","01.csharp/10.ef-core/29.interceptors-part2",{"title":647,"path":648,"stem":649},"План вивчення Entity Framework Core — Повний курс","/csharp/ef-core/implementation_plan","01.csharp/10.ef-core/implementation_plan",{"title":651,"icon":652,"path":653,"stem":654,"children":655,"page":59},"ASP.NET","i-devicon-dotnetcore","/csharp/aspnet","01.csharp/11.aspnet",[656,730,791,869,927,941,967,1057,1111,1182,1212,1289],{"title":657,"icon":658,"path":659,"stem":660,"children":661,"page":59},"Minimal API","i-lucide-network","/csharp/aspnet/minimal-api","01.csharp/11.aspnet/01.minimal-api",[662,666,670,674,678,682,686,690,694,698,702,706,710,714,718,722,726],{"title":663,"path":664,"stem":665},"Вступ до ASP.NET та еволюція фреймворку","/csharp/aspnet/minimal-api/introduction","01.csharp/11.aspnet/01.minimal-api/01.introduction",{"title":667,"path":668,"stem":669},"Перший додаток на ASP.NET Core","/csharp/aspnet/minimal-api/first-application","01.csharp/11.aspnet/01.minimal-api/02.first-application",{"title":671,"path":672,"stem":673},"WebApplication, Builder та Dependency Injection","/csharp/aspnet/minimal-api/webapplication-builder","01.csharp/11.aspnet/01.minimal-api/03.webapplication-builder",{"title":675,"path":676,"stem":677},"Конвеєр запитів та Middleware","/csharp/aspnet/minimal-api/request-pipeline-middleware","01.csharp/11.aspnet/01.minimal-api/04.request-pipeline-middleware",{"title":679,"path":680,"stem":681},"Маршрутизація в ASP.NET Core: Основи","/csharp/aspnet/minimal-api/routing-basics","01.csharp/11.aspnet/01.minimal-api/05.routing-basics",{"title":683,"path":684,"stem":685},"Маршрутизація в ASP.NET Core: Розширені можливості","/csharp/aspnet/minimal-api/routing-advanced","01.csharp/11.aspnet/01.minimal-api/06.routing-advanced",{"title":687,"path":688,"stem":689},"Статичні файли в ASP.NET Core","/csharp/aspnet/minimal-api/static-files","01.csharp/11.aspnet/01.minimal-api/07.static-files",{"title":691,"path":692,"stem":693},"Статичні Активи: MapStaticAssets (ASP.NET Core 9.0)","/csharp/aspnet/minimal-api/static-assets","01.csharp/11.aspnet/01.minimal-api/08.static-assets",{"title":695,"path":696,"stem":697},"Конфігурація в ASP.NET Core: Основи","/csharp/aspnet/minimal-api/configuration-fundamentals","01.csharp/11.aspnet/01.minimal-api/09.configuration-fundamentals",{"title":699,"path":700,"stem":701},"Конфігурація: Паттерн Options","/csharp/aspnet/minimal-api/configuration-options","01.csharp/11.aspnet/01.minimal-api/10.configuration-options",{"title":703,"path":704,"stem":705},"Логування в ASP.NET Core: Основи","/csharp/aspnet/minimal-api/logging-basics","01.csharp/11.aspnet/01.minimal-api/11.logging-basics",{"title":707,"path":708,"stem":709},"Логування: Serilog та Middleware","/csharp/aspnet/minimal-api/logging-advanced","01.csharp/11.aspnet/01.minimal-api/12.logging-advanced",{"title":711,"path":712,"stem":713},"Управління станом: HttpContext.Items та Cookies","/csharp/aspnet/minimal-api/state-management","01.csharp/11.aspnet/01.minimal-api/13.state-management",{"title":715,"path":716,"stem":717},"Стан сесії: Sessions","/csharp/aspnet/minimal-api/session-state","01.csharp/11.aspnet/01.minimal-api/14.session-state",{"title":719,"path":720,"stem":721},"Структура проєкту: від хаосу до архітектури","/csharp/aspnet/minimal-api/project-structure","01.csharp/11.aspnet/01.minimal-api/15.project-structure",{"title":723,"path":724,"stem":725},"Scalar у Minimal API: повний проєкт і Fluent OpenAPI","/csharp/aspnet/minimal-api/scalar-openapi-fluent","01.csharp/11.aspnet/01.minimal-api/16.scalar-openapi-fluent",{"title":727,"path":728,"stem":729},"Swagger / Swashbuckle у Minimal API: окремий класичний шлях","/csharp/aspnet/minimal-api/swagger-swashbuckle","01.csharp/11.aspnet/01.minimal-api/17.swagger-swashbuckle",{"title":731,"icon":658,"path":732,"stem":733,"children":734,"page":59},"API","/csharp/aspnet/api","01.csharp/11.aspnet/02.api",[735,739,743,747,751,755,759,763,767,771,775,779,783,787],{"title":736,"path":737,"stem":738},"Що таке API. Клієнт-серверна архітектура","/csharp/aspnet/api/what-is-api","01.csharp/11.aspnet/02.api/01.what-is-api",{"title":740,"path":741,"stem":742},"Формати даних: JSON, XML, TOML та бінарні формати","/csharp/aspnet/api/data-formats","01.csharp/11.aspnet/02.api/02.data-formats",{"title":744,"path":745,"stem":746},"Парадигми API та концепція REST","/csharp/aspnet/api/api-paradigms-rest","01.csharp/11.aspnet/02.api/03.api-paradigms-rest",{"title":748,"path":749,"stem":750},"HTTP-методи, статус-коди та заголовки","/csharp/aspnet/api/http-methods-status-codes","01.csharp/11.aspnet/02.api/04.http-methods-status-codes",{"title":752,"path":753,"stem":754},"Організація HTTP API за принципами REST","/csharp/aspnet/api/rest-organizing","01.csharp/11.aspnet/02.api/05.rest-organizing",{"title":756,"path":757,"stem":758},"Номенклатура URL та CRUD-операції","/csharp/aspnet/api/url-nomenclature-crud","01.csharp/11.aspnet/02.api/06.url-nomenclature-crud",{"title":760,"path":761,"stem":762},"Правила дизайну: іменування та стандарти","/csharp/aspnet/api/api-design-naming","01.csharp/11.aspnet/02.api/07.api-design-naming",{"title":764,"path":765,"stem":766},"Валідація, ліміти та обробка помилок","/csharp/aspnet/api/api-design-validation","01.csharp/11.aspnet/02.api/08.api-design-validation",{"title":768,"path":769,"stem":770},"Обробка помилок у Minimal API","/csharp/aspnet/api/error-handling-http","01.csharp/11.aspnet/02.api/09.error-handling-http",{"title":772,"path":773,"stem":774},"Ідемпотентність та синхронізація стану","/csharp/aspnet/api/idempotency-sync","01.csharp/11.aspnet/02.api/10.idempotency-sync",{"title":776,"path":777,"stem":778},"Пагінація та організація списків","/csharp/aspnet/api/pagination-lists","01.csharp/11.aspnet/02.api/11.pagination-lists",{"title":780,"path":781,"stem":782},"Безпека API, кешування та інтернаціоналізація","/csharp/aspnet/api/security-auth","01.csharp/11.aspnet/02.api/12.security-auth",{"title":784,"path":785,"stem":786},"Процес проєктування API та документування","/csharp/aspnet/api/api-design-process","01.csharp/11.aspnet/02.api/13.api-design-process",{"title":788,"path":789,"stem":790},"OpenAPI: контракт, специфікація та документація API","/csharp/aspnet/api/openapi","01.csharp/11.aspnet/02.api/14.openapi",{"title":792,"icon":793,"path":794,"stem":795,"children":796,"page":59},"Auth","i-lucide-shield-check","/csharp/aspnet/auth","01.csharp/11.aspnet/03.auth",[797,801,805,809,813,817,821,825,829,833,837,841,845,849,853,857,861,865],{"title":798,"path":799,"stem":800},"Основи аутентифікації та авторизації","/csharp/aspnet/auth/auth-fundamentals","01.csharp/11.aspnet/03.auth/01.auth-fundamentals",{"title":802,"path":803,"stem":804},"JWT-аутентифікація","/csharp/aspnet/auth/jwt-authentication","01.csharp/11.aspnet/03.auth/02.jwt-authentication",{"title":806,"path":807,"stem":808},"Авторизація: ролі, політики та resource-based доступ","/csharp/aspnet/auth/authorization-policies","01.csharp/11.aspnet/03.auth/03.authorization-policies",{"title":810,"path":811,"stem":812},"Cookie-аутентифікація та ASP.NET Core Identity","/csharp/aspnet/auth/cookie-auth-identity","01.csharp/11.aspnet/03.auth/04.cookie-auth-identity",{"title":814,"path":815,"stem":816},"JWT + Refresh Tokens (HttpOnly Cookie)","/csharp/aspnet/auth/04b.identity-auth-jwt","01.csharp/11.aspnet/03.auth/04b.identity-auth-jwt",{"title":818,"path":819,"stem":820},"Identity: Підтвердження Email та Скидання Пароля","/csharp/aspnet/auth/identity-email-confirmation","01.csharp/11.aspnet/03.auth/05.identity-email-confirmation",{"title":822,"path":823,"stem":824},"Identity: Двофакторна Аутентифікація (2FA)","/csharp/aspnet/auth/identity-two-factor","01.csharp/11.aspnet/03.auth/06.identity-two-factor",{"title":826,"path":827,"stem":828},"Identity: Внутрішня Архітектура та Кастомізація","/csharp/aspnet/auth/identity-internals","01.csharp/11.aspnet/03.auth/07.identity-internals",{"title":830,"path":831,"stem":832},"OAuth 2.0 та зовнішні провайдери","/csharp/aspnet/auth/oauth-external-providers","01.csharp/11.aspnet/03.auth/08.oauth-external-providers",{"title":834,"path":835,"stem":836},"Безпека на практиці: CORS, HTTPS та захист від атак","/csharp/aspnet/auth/security-hardening","01.csharp/11.aspnet/03.auth/09.security-hardening",{"title":838,"path":839,"stem":840},"Теорія OAuth 2.0: Поняття, Аналогії та Флоу","/csharp/aspnet/auth/oauth-theory","01.csharp/11.aspnet/03.auth/10.oauth-theory",{"title":842,"path":843,"stem":844},"OIDC, OAuth 2.0 та Keycloak в ASP.NET Core","/csharp/aspnet/auth/oidc-keycloak","01.csharp/11.aspnet/03.auth/10.oidc-keycloak",{"title":846,"path":847,"stem":848},"API Keys аутентифікація в ASP.NET Core","/csharp/aspnet/auth/api-keys","01.csharp/11.aspnet/03.auth/11.api-keys",{"title":850,"path":851,"stem":852},"Rate Limiting та Throttling в ASP.NET Core","/csharp/aspnet/auth/rate-limiting","01.csharp/11.aspnet/03.auth/12.rate-limiting",{"title":854,"path":855,"stem":856},"Refresh Token Rotation в ASP.NET Core","/csharp/aspnet/auth/refresh-token-rotation","01.csharp/11.aspnet/03.auth/13.refresh-token-rotation",{"title":858,"path":859,"stem":860},"Certificate Authentication та mTLS в ASP.NET Core","/csharp/aspnet/auth/certificate-auth","01.csharp/11.aspnet/03.auth/14.certificate-auth",{"title":862,"path":863,"stem":864},"RBAC, ABAC та ReBAC в ASP.NET Core","/csharp/aspnet/auth/rbac-abac-rebac","01.csharp/11.aspnet/03.auth/15.rbac-abac-rebac",{"title":866,"path":867,"stem":868},"Multi-tenancy та ізоляція даних в ASP.NET Core","/csharp/aspnet/auth/multi-tenancy","01.csharp/11.aspnet/03.auth/16.multi-tenancy",{"title":870,"icon":871,"path":872,"stem":873,"children":874,"page":59},"Нотифікації","i-lucide-bell","/csharp/aspnet/notifications","01.csharp/11.aspnet/04.notifications",[875,879,883,887,891,895,899,903,907,911,915,919,923],{"title":876,"path":877,"stem":878},"In-App нотифікації через базу даних","/csharp/aspnet/notifications/in-app-database-notifications","01.csharp/11.aspnet/04.notifications/01.in-app-database-notifications",{"title":880,"path":881,"stem":882},"Polling: Регулярний запит оновлень","/csharp/aspnet/notifications/polling","01.csharp/11.aspnet/04.notifications/02.polling",{"title":884,"path":885,"stem":886},"Server-Sent Events: Однострімовий push від сервера","/csharp/aspnet/notifications/server-sent-events","01.csharp/11.aspnet/04.notifications/03.server-sent-events",{"title":888,"path":889,"stem":890},"WebSockets: Двостороннє з'єднання в реальному часі","/csharp/aspnet/notifications/websockets","01.csharp/11.aspnet/04.notifications/04.websockets",{"title":892,"path":893,"stem":894},"SignalR: Абстракція над транспортами реального часу","/csharp/aspnet/notifications/signalr","01.csharp/11.aspnet/04.notifications/05.signalr",{"title":896,"path":897,"stem":898},"Background Services: Фонові задачі в ASP.NET Core","/csharp/aspnet/notifications/background-services","01.csharp/11.aspnet/04.notifications/06.background-services",{"title":900,"path":901,"stem":902},"Web Push нотифікації","/csharp/aspnet/notifications/web-push","01.csharp/11.aspnet/04.notifications/07.web-push",{"title":904,"path":905,"stem":906},"Email нотифікації","/csharp/aspnet/notifications/email-notifications","01.csharp/11.aspnet/04.notifications/08.email-notifications",{"title":908,"path":909,"stem":910},"Порівняння підходів: Як вибрати правильну технологію нотифікацій","/csharp/aspnet/notifications/choosing-the-right-approach","01.csharp/11.aspnet/04.notifications/09.choosing-the-right-approach",{"title":912,"path":913,"stem":914},"Hangfire: Надійне планування фонових задач","/csharp/aspnet/notifications/hangfire","01.csharp/11.aspnet/04.notifications/10.hangfire",{"title":916,"path":917,"stem":918},"Практика: Конвертація зображень у WebP через Hangfire","/csharp/aspnet/notifications/hangfire-image-webp","01.csharp/11.aspnet/04.notifications/11.hangfire-image-webp",{"title":920,"path":921,"stem":922},"Практика: Підготовка відео до HLS-стрімінгу через Hangfire","/csharp/aspnet/notifications/hangfire-video-hls","01.csharp/11.aspnet/04.notifications/12.hangfire-video-hls",{"title":924,"path":925,"stem":926},"Telegram-нотифікації: від одного повідомлення до масових розсилок і мульти-канального підходу","/csharp/aspnet/notifications/telegram-notifications","01.csharp/11.aspnet/04.notifications/13.telegram-notifications",{"title":928,"icon":929,"path":930,"stem":931,"children":932,"page":59},"Інтернаціоналізація","i-lucide-languages","/csharp/aspnet/i18n","01.csharp/11.aspnet/05.i18n",[933,937],{"title":934,"path":935,"stem":936},"Інтернаціоналізація (i18n) у Minimal API: від A до Я","/csharp/aspnet/i18n/internationalization","01.csharp/11.aspnet/05.i18n/01.internationalization",{"title":938,"path":939,"stem":940},"Humanizer: людиномовні рядки у .NET","/csharp/aspnet/i18n/humanizer","01.csharp/11.aspnet/05.i18n/02.humanizer",{"title":942,"icon":943,"path":944,"stem":945,"children":946,"page":59},"Кешування","i-lucide-layers","/csharp/aspnet/caching","01.csharp/11.aspnet/06.caching",[947,951,955,959,963],{"title":948,"path":949,"stem":950},"Огляд кешування: чотири рівні і коли що обирати","/csharp/aspnet/caching/caching","01.csharp/11.aspnet/06.caching/01.caching",{"title":952,"path":953,"stem":954},"IMemoryCache: кеш в оперативній пам'яті","/csharp/aspnet/caching/memory-cache","01.csharp/11.aspnet/06.caching/02.memory-cache",{"title":956,"path":957,"stem":958},"IDistributedCache і Redis: розподілений кеш","/csharp/aspnet/caching/distributed-cache","01.csharp/11.aspnet/06.caching/03.distributed-cache",{"title":960,"path":961,"stem":962},"Response Cache: HTTP-кешування через Cache-Control","/csharp/aspnet/caching/response-cache","01.csharp/11.aspnet/06.caching/04.response-cache",{"title":964,"path":965,"stem":966},"Output Cache: серверний кеш HTTP-відповідей (.NET 7+)","/csharp/aspnet/caching/output-cache","01.csharp/11.aspnet/06.caching/05.output-cache",{"title":968,"icon":969,"path":970,"stem":971,"children":972,"page":59},"Тестування","i-lucide-test-tube","/csharp/aspnet/testing","01.csharp/11.aspnet/07.testing",[973,977,981,985,989,993,997,1001,1005,1009,1013,1017,1021,1025,1029,1033,1037,1041,1045,1049,1053],{"title":974,"path":975,"stem":976},"Що таке тестування? Від інтуїції до науки","/csharp/aspnet/testing/what-is-testing","01.csharp/11.aspnet/07.testing/01.what-is-testing",{"title":978,"path":979,"stem":980},"Піраміда тестування — Стратегія, а не Догма","/csharp/aspnet/testing/testing-pyramid","01.csharp/11.aspnet/07.testing/02.testing-pyramid",{"title":982,"path":983,"stem":984},"Дві Школи Тестування — Лондон проти Детройту","/csharp/aspnet/testing/testing-schools","01.csharp/11.aspnet/07.testing/03.testing-schools",{"title":986,"path":987,"stem":988},"TDD та BDD — Тести як Дизайн-інструмент","/csharp/aspnet/testing/tdd-and-bdd","01.csharp/11.aspnet/07.testing/04.tdd-and-bdd",{"title":990,"path":991,"stem":992},"Що саме тестувати — Техніки аналізу та Циклomatична складність","/csharp/aspnet/testing/what-to-test","01.csharp/11.aspnet/07.testing/05.what-to-test",{"title":994,"path":995,"stem":996},"Тестові Фреймворки — Навіщо вони і що всередині","/csharp/aspnet/testing/test-frameworks","01.csharp/11.aspnet/07.testing/06.test-frameworks",{"title":998,"path":999,"stem":1000},"xUnit — Факти, Теорії та Lifecycle тестів","/csharp/aspnet/testing/xunit-basics","01.csharp/11.aspnet/07.testing/07.xunit-basics",{"title":1002,"path":1003,"stem":1004},"xUnit Advanced — Fixtures, Кастомізація та Розширення","/csharp/aspnet/testing/xunit-advanced","01.csharp/11.aspnet/07.testing/08.xunit-advanced",{"title":1006,"path":1007,"stem":1008},"Moq — Глибоке занурення в мокування","/csharp/aspnet/testing/mocking-with-moq","01.csharp/11.aspnet/07.testing/09.mocking-with-moq",{"title":1010,"path":1011,"stem":1012},"Тестування Баз Даних — EF Core, SQLite та Testcontainers","/csharp/aspnet/testing/database-testing","01.csharp/11.aspnet/07.testing/10.database-testing",{"title":1014,"path":1015,"stem":1016},"Integration Testing — Частина 1 [Теорія та WebApplicationFactory]","/csharp/aspnet/testing/integration-testing","01.csharp/11.aspnet/07.testing/11.integration-testing",{"title":1018,"path":1019,"stem":1020},"Інтеграційне тестування — Практика","/csharp/aspnet/testing/11a.integration-testing-practice","01.csharp/11.aspnet/07.testing/11a.integration-testing-practice",{"title":1022,"path":1023,"stem":1024},"Integration Testing — Частина 2 [Просунуті Сценарії та Testcontainers]","/csharp/aspnet/testing/integration-testing-advanced","01.csharp/11.aspnet/07.testing/12.integration-testing-advanced",{"title":1026,"path":1027,"stem":1028},"Професійний Postman: Колекції, Змінні та GitHub Інтеграція","/csharp/aspnet/testing/postman-professional","01.csharp/11.aspnet/07.testing/13.postman-professional",{"title":1030,"path":1031,"stem":1032},"HttpClient у Тестах Частина 1: Архітектура та MockHttpMessageHandler","/csharp/aspnet/testing/httpclient-testing","01.csharp/11.aspnet/07.testing/14.httpclient-testing",{"title":1034,"path":1035,"stem":1036},"HttpClient у Тестах Частина 2: WireMock.Net та Resilience","/csharp/aspnet/testing/wiremock-net","01.csharp/11.aspnet/07.testing/15.wiremock-net",{"title":1038,"path":1039,"stem":1040},"Патерни та Анти-патерни Тестування: Test Smells","/csharp/aspnet/testing/testing-patterns","01.csharp/11.aspnet/07.testing/16.testing-patterns",{"title":1042,"path":1043,"stem":1044},"Просунуті інструменти: Time, Snapshots та Властивості","/csharp/aspnet/testing/advanced-testing-tools","01.csharp/11.aspnet/07.testing/17.advanced-testing-tools",{"title":1046,"path":1047,"stem":1048},"Тестування Архітектури з NetArchTest","/csharp/aspnet/testing/architecture-testing","01.csharp/11.aspnet/07.testing/18.architecture-testing",{"title":1050,"path":1051,"stem":1052},"Тестування Продуктивності: BenchmarkDotNet, NBomber та k6","/csharp/aspnet/testing/performance-testing","01.csharp/11.aspnet/07.testing/19.performance-testing",{"title":1054,"path":1055,"stem":1056},"Залишок плану для курсу \"Тестування ASP.NET Minimal API\"","/csharp/aspnet/testing/remaining_plan","01.csharp/11.aspnet/07.testing/remaining_plan",{"title":1058,"icon":1059,"path":1060,"stem":1061,"children":1062,"page":59},"Платежі","i-lucide-credit-card","/csharp/aspnet/payments","01.csharp/11.aspnet/08.payments",[1063,1067,1071,1075,1079,1083,1087,1091,1095,1099,1103,1107],{"title":1064,"path":1065,"stem":1066},"Основи платіжної інфраструктури","/csharp/aspnet/payments/payment-fundamentals","01.csharp/11.aspnet/08.payments/01.payment-fundamentals",{"title":1068,"path":1069,"stem":1070},"Методи оплати в Україні","/csharp/aspnet/payments/payment-methods-ukraine","01.csharp/11.aspnet/08.payments/02.payment-methods-ukraine",{"title":1072,"path":1073,"stem":1074},"PCI DSS та безпека платежів","/csharp/aspnet/payments/pci-dss-security","01.csharp/11.aspnet/08.payments/03.pci-dss-security",{"title":1076,"path":1077,"stem":1078},"Архітектура платіжної підсистеми","/csharp/aspnet/payments/payment-architecture","01.csharp/11.aspnet/08.payments/04.payment-architecture",{"title":1080,"path":1081,"stem":1082},"Інтеграція LiqPay (ПриватБанк)","/csharp/aspnet/payments/liqpay-integration","01.csharp/11.aspnet/08.payments/05.liqpay-integration",{"title":1084,"path":1085,"stem":1086},"Інтеграція Monobank Acquiring API","/csharp/aspnet/payments/monobank-acquiring","01.csharp/11.aspnet/08.payments/06.monobank-acquiring",{"title":1088,"path":1089,"stem":1090},"Інтеграція Stripe","/csharp/aspnet/payments/stripe-integration","01.csharp/11.aspnet/08.payments/07.stripe-integration",{"title":1092,"path":1093,"stem":1094},"Webhooks — глибоке занурення","/csharp/aspnet/payments/webhooks-deep-dive","01.csharp/11.aspnet/08.payments/08.webhooks-deep-dive",{"title":1096,"path":1097,"stem":1098},"Підписки та рекурентні платежі","/csharp/aspnet/payments/subscriptions-recurring","01.csharp/11.aspnet/08.payments/09.subscriptions-recurring",{"title":1100,"path":1101,"stem":1102},"Повернення коштів та диспути","/csharp/aspnet/payments/refunds-disputes","01.csharp/11.aspnet/08.payments/10.refunds-disputes",{"title":1104,"path":1105,"stem":1106},"Тестування платіжних інтеграцій","/csharp/aspnet/payments/testing-payments","01.csharp/11.aspnet/08.payments/11.testing-payments",{"title":1108,"path":1109,"stem":1110},"Чекліст виходу в Production","/csharp/aspnet/payments/production-checklist","01.csharp/11.aspnet/08.payments/12.production-checklist",{"title":1112,"icon":1113,"items":1114,"path":1127,"stem":1128,"children":1129,"page":59},"Популярні бібліотеки","lucide:box",[1115,1116,1117,1118,1119,1120,1121,1122,1123,1124,1125,1126],"01.fluent-validation","02.mapster","03.erroror-result-pattern","04.serilog","05.mediatr","06.polly","07.health-checks","08.feature-management","09.fluent-email","10.quest-pdf","11.bogus","12.humanizer-guard","/csharp/aspnet/libraries","01.csharp/11.aspnet/09.libraries",[1130,1134,1138,1142,1146,1150,1154,1158,1162,1166,1170,1174,1178],{"title":1131,"path":1132,"stem":1133},"Валідація з FluentValidation в ASP.NET Core","/csharp/aspnet/libraries/fluent-validation","01.csharp/11.aspnet/09.libraries/01.fluent-validation",{"title":1135,"path":1136,"stem":1137},"Маппінг об","/csharp/aspnet/libraries/mapster","01.csharp/11.aspnet/09.libraries/02.mapster",{"title":1139,"path":1140,"stem":1141},"Обробка помилок з ErrorOr та Result Pattern в ASP.NET Core","/csharp/aspnet/libraries/erroror-result-pattern","01.csharp/11.aspnet/09.libraries/03.erroror-result-pattern",{"title":1143,"path":1144,"stem":1145},"Структуроване логування з Serilog в ASP.NET Core","/csharp/aspnet/libraries/serilog","01.csharp/11.aspnet/09.libraries/04.serilog",{"title":1147,"path":1148,"stem":1149},"CQRS та Mediator з MediatR в ASP.NET Core","/csharp/aspnet/libraries/mediatr","01.csharp/11.aspnet/09.libraries/05.mediatr",{"title":1151,"path":1152,"stem":1153},"Відмовостійкість з Polly в ASP.NET Core","/csharp/aspnet/libraries/polly","01.csharp/11.aspnet/09.libraries/06.polly",{"title":1155,"path":1156,"stem":1157},"Health Checks в ASP.NET Core","/csharp/aspnet/libraries/health-checks","01.csharp/11.aspnet/09.libraries/07.health-checks",{"title":1159,"path":1160,"stem":1161},"Feature Management та Feature Flags в ASP.NET Core","/csharp/aspnet/libraries/feature-management","01.csharp/11.aspnet/09.libraries/08.feature-management",{"title":1163,"path":1164,"stem":1165},"Відправка Email з FluentEmail в ASP.NET Core","/csharp/aspnet/libraries/fluent-email","01.csharp/11.aspnet/09.libraries/09.fluent-email",{"title":1167,"path":1168,"stem":1169},"Генерація PDF з QuestPDF в ASP.NET Core","/csharp/aspnet/libraries/quest-pdf","01.csharp/11.aspnet/09.libraries/10.quest-pdf",{"title":1171,"path":1172,"stem":1173},"Генерація тестових даних з Bogus в ASP.NET Core","/csharp/aspnet/libraries/bogus","01.csharp/11.aspnet/09.libraries/11.bogus",{"title":1175,"path":1176,"stem":1177},"Humanizer та Guard Clauses в ASP.NET Core","/csharp/aspnet/libraries/humanizer-guard","01.csharp/11.aspnet/09.libraries/12.humanizer-guard",{"title":1179,"path":1180,"stem":1181},"План модуля 10.libraries — Популярні бібліотеки ASP.NET","/csharp/aspnet/libraries/plan","01.csharp/11.aspnet/09.libraries/plan",{"title":1183,"icon":1184,"path":1185,"stem":1186,"children":1187,"page":59},"Razor Pages","i-lucide-layout-template","/csharp/aspnet/razor-pages","01.csharp/11.aspnet/10.razor-pages",[1188,1192,1196,1200,1204,1208],{"title":1189,"path":1190,"stem":1191},"Від Minimal API до Razor Pages: концептуальний перехід","/csharp/aspnet/razor-pages/from-minimal-api","01.csharp/11.aspnet/10.razor-pages/01.from-minimal-api",{"title":1193,"path":1194,"stem":1195},"PageModel: логіка сторінки Razor Pages","/csharp/aspnet/razor-pages/page-model","01.csharp/11.aspnet/10.razor-pages/02.page-model",{"title":1197,"path":1198,"stem":1199},"Razor синтаксис: шаблонізатор у .cshtml","/csharp/aspnet/razor-pages/razor-syntax","01.csharp/11.aspnet/10.razor-pages/03.razor-syntax",{"title":1201,"path":1202,"stem":1203},"Tag Helpers: типізований HTML","/csharp/aspnet/razor-pages/tag-helpers","01.csharp/11.aspnet/10.razor-pages/04.tag-helpers",{"title":1205,"path":1206,"stem":1207},"Форми і валідація: повний цикл обробки даних","/csharp/aspnet/razor-pages/forms-validation","01.csharp/11.aspnet/10.razor-pages/05.forms-validation",{"title":1209,"path":1210,"stem":1211},"Практичний проєкт: TaskManager на Razor Pages","/csharp/aspnet/razor-pages/project-task-manager","01.csharp/11.aspnet/10.razor-pages/06.project-task-manager",{"title":1213,"path":1214,"stem":1215,"children":1216,"page":59},"ASP.NET Core MVC","/csharp/aspnet/mvc","01.csharp/11.aspnet/11.mvc",[1217,1221,1225,1229,1233,1237,1241,1245,1249,1253,1257,1261,1265,1269,1273,1277,1281,1285],{"title":1218,"path":1219,"stem":1220},"Патерн MVC: архітектура, що змінила веб","/csharp/aspnet/mvc/mvc-pattern","01.csharp/11.aspnet/11.mvc/01.mvc-pattern",{"title":1222,"path":1223,"stem":1224},"Від Razor Pages до MVC: концептуальний перехід","/csharp/aspnet/mvc/from-razor-pages","01.csharp/11.aspnet/11.mvc/02.from-razor-pages",{"title":1226,"path":1227,"stem":1228},"Controllers та Actions: серце MVC","/csharp/aspnet/mvc/controllers-actions","01.csharp/11.aspnet/11.mvc/03.controllers-actions",{"title":1230,"path":1231,"stem":1232},"Маршрутизація в MVC: Convention vs Attribute Routing","/csharp/aspnet/mvc/routing-mvc","01.csharp/11.aspnet/11.mvc/04.routing-mvc",{"title":1234,"path":1235,"stem":1236},"Model Binding: від HTTP до C#","/csharp/aspnet/mvc/model-binding","01.csharp/11.aspnet/11.mvc/05.model-binding",{"title":1238,"path":1239,"stem":1240},"Views, ViewData, ViewBag, TempData і ViewModel","/csharp/aspnet/mvc/views-viewdata-tempdata","01.csharp/11.aspnet/11.mvc/06.views-viewdata-tempdata",{"title":1242,"path":1243,"stem":1244},"Filters: аспектно-орієнтоване програмування в MVC","/csharp/aspnet/mvc/filters","01.csharp/11.aspnet/11.mvc/07.filters",{"title":1246,"path":1247,"stem":1248},"Areas: структурування великих застосунків","/csharp/aspnet/mvc/areas","01.csharp/11.aspnet/11.mvc/08.areas",{"title":1250,"path":1251,"stem":1252},"View Components: повторювані незалежні блоки UI","/csharp/aspnet/mvc/view-components","01.csharp/11.aspnet/11.mvc/09.view-components",{"title":1254,"path":1255,"stem":1256},"Display та Editor Templates","/csharp/aspnet/mvc/display-editor-templates","01.csharp/11.aspnet/11.mvc/10.display-editor-templates",{"title":1258,"path":1259,"stem":1260},"Валідація: IValidatableObject та FluentValidation","/csharp/aspnet/mvc/validation-advanced","01.csharp/11.aspnet/11.mvc/11.validation-advanced",{"title":1262,"path":1263,"stem":1264},"HTMX: інтерактивність через HTML-атрибути","/csharp/aspnet/mvc/htmx","01.csharp/11.aspnet/11.mvc/12.htmx",{"title":1266,"path":1267,"stem":1268},"HTMX у ASP.NET Core MVC: серверна інтеграція","/csharp/aspnet/mvc/ajax-htmx-mvc","01.csharp/11.aspnet/11.mvc/13.ajax-htmx-mvc",{"title":1270,"path":1271,"stem":1272},"Практичний проєкт: Каталог товарів з HTMX","/csharp/aspnet/mvc/htmx-project","01.csharp/11.aspnet/11.mvc/14.htmx-project",{"title":1274,"path":1275,"stem":1276},"Завантаження та обробка файлів","/csharp/aspnet/mvc/file-upload","01.csharp/11.aspnet/11.mvc/15.file-upload",{"title":1278,"path":1279,"stem":1280},"Глобалізація та Локалізація MVC","/csharp/aspnet/mvc/globalization-localization","01.csharp/11.aspnet/11.mvc/16.globalization-localization",{"title":1282,"path":1283,"stem":1284},"Підсумковий проєкт: Блог-платформа","/csharp/aspnet/mvc/mvc-project","01.csharp/11.aspnet/11.mvc/17.mvc-project",{"title":1286,"path":1287,"stem":1288},"План курсу: ASP.NET Core MVC","/csharp/aspnet/mvc/plan","01.csharp/11.aspnet/11.mvc/plan",{"title":1290,"path":1291,"stem":1292,"children":1293,"page":59},"Web Api","/csharp/aspnet/web-api","01.csharp/11.aspnet/12.web-api",[1294,1298,1302,1306,1310,1314,1318,1322,1326,1330,1334,1338,1342],{"title":1295,"path":1296,"stem":1297},"Від Minimal API до Controller-based API","/csharp/aspnet/web-api/from-minimal-api-to-controllers","01.csharp/11.aspnet/12.web-api/01.from-minimal-api-to-controllers",{"title":1299,"path":1300,"stem":1301},"ControllerBase, ActionResult\u003CT> та Response Types","/csharp/aspnet/web-api/controller-base-actionresult","01.csharp/11.aspnet/12.web-api/02.controller-base-actionresult",{"title":1303,"path":1304,"stem":1305},"Content Negotiation - JSON, XML та власні форматери","/csharp/aspnet/web-api/content-negotiation","01.csharp/11.aspnet/12.web-api/03.content-negotiation",{"title":1307,"path":1308,"stem":1309},"Версіонування API","/csharp/aspnet/web-api/api-versioning","01.csharp/11.aspnet/12.web-api/04.api-versioning",{"title":1311,"path":1312,"stem":1313},"ProblemDetails та структурована обробка помилок","/csharp/aspnet/web-api/problemdetails-error-handling","01.csharp/11.aspnet/12.web-api/05.problemdetails-error-handling",{"title":1315,"path":1316,"stem":1317},"Фільтри у Web API контексті","/csharp/aspnet/web-api/filters-for-api","01.csharp/11.aspnet/12.web-api/06.filters-for-api",{"title":1319,"path":1320,"stem":1321},"Пагінація, фільтрація та сортування","/csharp/aspnet/web-api/pagination-filtering-sorting","01.csharp/11.aspnet/12.web-api/07.pagination-filtering-sorting",{"title":1323,"path":1324,"stem":1325},"HATEOAS та Resource Expansion","/csharp/aspnet/web-api/hateoas-resource-expansion","01.csharp/11.aspnet/12.web-api/08.hateoas-resource-expansion",{"title":1327,"path":1328,"stem":1329},"Гібридна архітектура - Minimal API + Controllers","/csharp/aspnet/web-api/minimal-api-vs-controllers-hybrid","01.csharp/11.aspnet/12.web-api/09.minimal-api-vs-controllers-hybrid",{"title":1331,"path":1332,"stem":1333},"Документація API - Swashbuckle, NSwag та генерація клієнтів","/csharp/aspnet/web-api/api-documentation-generation","01.csharp/11.aspnet/12.web-api/10.api-documentation-generation",{"title":1335,"path":1336,"stem":1337},"Health Checks та моніторинг API","/csharp/aspnet/web-api/health-checks-monitoring","01.csharp/11.aspnet/12.web-api/11.health-checks-monitoring",{"title":1339,"path":1340,"stem":1341},"Підсумковий проєкт - Production-Ready REST API","/csharp/aspnet/web-api/web-api-project","01.csharp/11.aspnet/12.web-api/12.web-api-project",{"title":1343,"path":1344,"stem":1345},"План курсу: ASP.NET Core Web API (Controllers)","/csharp/aspnet/web-api/plan","01.csharp/11.aspnet/12.web-api/plan",{"title":1347,"icon":1348,"path":1349,"stem":1350,"children":1351,"page":59},"Desktop UI","i-lucide-app-window","/csharp/desktop-ui","01.csharp/12.desktop-ui",[1352,1356,1360,1364,1368,1372,1376,1380,1384,1388,1392,1396,1400,1404,1408,1412,1416,1420,1424,1428,1432,1436,1440,1444,1448,1452,1456,1460,1464,1468,1472,1476,1480,1484,1488,1492,1496,1500,1504,1508,1512,1516,1520,1524,1528,1532,1536,1540,1544,1548,1552,1556,1560,1564,1568,1572,1576,1580,1584,1588,1592,1596,1600,1604,1608,1612,1616,1620,1624,1628,1632],{"title":1353,"path":1354,"stem":1355},"Що таке десктопна розробка?","/csharp/desktop-ui/what-is-desktop-dev","01.csharp/12.desktop-ui/01.what-is-desktop-dev",{"title":1357,"path":1358,"stem":1359},"Архітектура WPF — як влаштований графічний інтерфейс","/csharp/desktop-ui/wpf-architecture","01.csharp/12.desktop-ui/02.wpf-architecture",{"title":1361,"path":1362,"stem":1363},"Перший WPF-проєкт — від нуля до вікна","/csharp/desktop-ui/first-wpf-app","01.csharp/12.desktop-ui/03.first-wpf-app",{"title":1365,"path":1366,"stem":1367},"Перший Avalonia-проєкт: WPF для всіх платформ","/csharp/desktop-ui/03a.first-avalonia-app","01.csharp/12.desktop-ui/03a.first-avalonia-app",{"title":1369,"path":1370,"stem":1371},"XAML: декларативний інтерфейс","/csharp/desktop-ui/xaml-basics","01.csharp/12.desktop-ui/04.xaml-basics",{"title":1373,"path":1374,"stem":1375},"Fluent UI у WPF — сучасний дизайн Windows 11","/csharp/desktop-ui/04a.wpf-fluent-ui","01.csharp/12.desktop-ui/04a.wpf-fluent-ui",{"title":1377,"path":1378,"stem":1379},"WPF UI — сучасна бібліотека Fluent контролів","/csharp/desktop-ui/04b.wpf-ui-library","01.csharp/12.desktop-ui/04b.wpf-ui-library",{"title":1381,"path":1382,"stem":1383},"HandyControl — велика бібліотека UI контролів для WPF","/csharp/desktop-ui/04c.handycontrol-library","01.csharp/12.desktop-ui/04c.handycontrol-library",{"title":1385,"path":1386,"stem":1387},"Простори імен та ресурси XAML","/csharp/desktop-ui/xaml-namespaces-resources","01.csharp/12.desktop-ui/05.xaml-namespaces-resources",{"title":1389,"path":1390,"stem":1391},"XAML в Avalonia: ключові відмінності від WPF","/csharp/desktop-ui/05a.avalonia-xaml-differences","01.csharp/12.desktop-ui/05a.avalonia-xaml-differences",{"title":1393,"path":1394,"stem":1395},"Розширення розмітки XAML (Markup Extensions)","/csharp/desktop-ui/xaml-markup-extensions","01.csharp/12.desktop-ui/06.xaml-markup-extensions",{"title":1397,"path":1398,"stem":1399},"Панелі Layout: StackPanel, WrapPanel, DockPanel","/csharp/desktop-ui/layout-panels-part1","01.csharp/12.desktop-ui/07.layout-panels-part1",{"title":1401,"path":1402,"stem":1403},"Grid, Canvas, UniformGrid","/csharp/desktop-ui/layout-panels-part2","01.csharp/12.desktop-ui/07.layout-panels-part2",{"title":1405,"path":1406,"stem":1407},"Просунуті техніки Layout","/csharp/desktop-ui/layout-advanced","01.csharp/12.desktop-ui/08.layout-advanced",{"title":1409,"path":1410,"stem":1411},"Адаптивний Layout та найкращі практики","/csharp/desktop-ui/layout-responsive","01.csharp/12.desktop-ui/09.layout-responsive",{"title":1413,"path":1414,"stem":1415},"Layout в Avalonia: відмінності та нові можливості","/csharp/desktop-ui/09a.layout-avalonia","01.csharp/12.desktop-ui/09a.layout-avalonia",{"title":1417,"path":1418,"stem":1419},"Button, Image, ProgressBar та інші базові контроли","/csharp/desktop-ui/basic-controls","01.csharp/12.desktop-ui/10.basic-controls",{"title":1421,"path":1422,"stem":1423},"Контроли в Avalonia: відмінності від WPF","/csharp/desktop-ui/10a.controls-avalonia","01.csharp/12.desktop-ui/10a.controls-avalonia",{"title":1425,"path":1426,"stem":1427},"Текстові контроли — TextBlock, TextBox, RichTextBox","/csharp/desktop-ui/text-controls","01.csharp/12.desktop-ui/11.text-controls",{"title":1429,"path":1430,"stem":1431},"Контроли вибору — CheckBox, RadioButton, ComboBox, ListBox, DatePicker","/csharp/desktop-ui/selection-controls","01.csharp/12.desktop-ui/12.selection-controls",{"title":1433,"path":1434,"stem":1435},"Content Model — GroupBox, Expander, TabControl, StatusBar","/csharp/desktop-ui/content-controls","01.csharp/12.desktop-ui/13.content-controls",{"title":1437,"path":1438,"stem":1439},"UI/UX принципи десктопних застосунків","/csharp/desktop-ui/13a.ui-ux-principles","01.csharp/12.desktop-ui/13a.ui-ux-principles",{"title":1441,"path":1442,"stem":1443},"Dependency Properties — Концепція та Value Resolution","/csharp/desktop-ui/dependency-properties-part1","01.csharp/12.desktop-ui/14.dependency-properties-part1",{"title":1445,"path":1446,"stem":1447},"Avalonia Property System — StyledProperty та DirectProperty","/csharp/desktop-ui/14a.avalonia-property-system","01.csharp/12.desktop-ui/14a.avalonia-property-system",{"title":1449,"path":1450,"stem":1451},"Attached Properties — Властивості без меж","/csharp/desktop-ui/attached-properties","01.csharp/12.desktop-ui/15.attached-properties",{"title":1453,"path":1454,"stem":1455},"Routed Events — Маршрутизація подій у WPF","/csharp/desktop-ui/routed-events","01.csharp/12.desktop-ui/16.routed-events",{"title":1457,"path":1458,"stem":1459},"Data Binding — Від Code-Behind до Декларативності","/csharp/desktop-ui/data-binding-basics-part1","01.csharp/12.desktop-ui/17.data-binding-basics-part1",{"title":1461,"path":1462,"stem":1463},"INotifyPropertyChanged — Живе оновлення UI","/csharp/desktop-ui/data-binding-basics-part2","01.csharp/12.desktop-ui/17.data-binding-basics-part2",{"title":1465,"path":1466,"stem":1467},"Compiled Bindings в Avalonia — Безпека на етапі компіляції","/csharp/desktop-ui/17a.avalonia-compiled-bindings","01.csharp/12.desktop-ui/17a.avalonia-compiled-bindings",{"title":1469,"path":1470,"stem":1471},"Просунутий Data Binding — ElementName, RelativeSource, MultiBinding","/csharp/desktop-ui/data-binding-advanced","01.csharp/12.desktop-ui/18.data-binding-advanced",{"title":1473,"path":1474,"stem":1475},"Value Converters — Перетворення типів даних у Data Binding","/csharp/desktop-ui/value-converters","01.csharp/12.desktop-ui/19.value-converters",{"title":1477,"path":1478,"stem":1479},"Data Templates — Візуалізація об'єктів у WPF","/csharp/desktop-ui/data-templates","01.csharp/12.desktop-ui/20.data-templates",{"title":1481,"path":1482,"stem":1483},"Collections Binding Part 1 — ObservableCollection та ItemsControl","/csharp/desktop-ui/collections-binding-part1","01.csharp/12.desktop-ui/21.collections-binding-part1",{"title":1485,"path":1486,"stem":1487},"Collections Binding Part 2 — ICollectionView, Filtering, Sorting та Virtualization","/csharp/desktop-ui/collections-binding-part2","01.csharp/12.desktop-ui/21.collections-binding-part2",{"title":1489,"path":1490,"stem":1491},"MVVM Pattern — Від Spaghetti Code до архітектури","/csharp/desktop-ui/mvvm-pattern","01.csharp/12.desktop-ui/22.mvvm-pattern",{"title":1493,"path":1494,"stem":1495},"ViewModel Implementation — Від BaseViewModel до валідації","/csharp/desktop-ui/viewmodel-implementation","01.csharp/12.desktop-ui/23.viewmodel-implementation",{"title":1497,"path":1498,"stem":1499},"Commands — Від event handlers до декларативних команд","/csharp/desktop-ui/commands","01.csharp/12.desktop-ui/24.commands",{"title":1501,"path":1502,"stem":1503},"MVVM Toolkit — MVVM без boilerplate через Source Generators","/csharp/desktop-ui/mvvm-toolkit","01.csharp/12.desktop-ui/25.mvvm-toolkit",{"title":1505,"path":1506,"stem":1507},"Messenger Pattern — Комунікація між ViewModel без прямих посилань","/csharp/desktop-ui/messenger-pattern","01.csharp/12.desktop-ui/26.messenger-pattern",{"title":1509,"path":1510,"stem":1511},"Стилі WPF — CSS для десктопу","/csharp/desktop-ui/styles-basics","01.csharp/12.desktop-ui/27.styles-basics",{"title":1513,"path":1514,"stem":1515},"CSS-like стилі Avalonia","/csharp/desktop-ui/27a.avalonia-css-styling","01.csharp/12.desktop-ui/27a.avalonia-css-styling",{"title":1517,"path":1518,"stem":1519},"Control Templates — Частина 1. Концепція та TemplateBinding","/csharp/desktop-ui/control-templates-part1","01.csharp/12.desktop-ui/28.control-templates-part1",{"title":1521,"path":1522,"stem":1523},"Control Templates — Частина 2. Named Parts та ContentPresenter","/csharp/desktop-ui/control-templates-part2","01.csharp/12.desktop-ui/28.control-templates-part2",{"title":1525,"path":1526,"stem":1527},"Control Themes в Avalonia — нова ера стилізації","/csharp/desktop-ui/28a.avalonia-control-themes","01.csharp/12.desktop-ui/28a.avalonia-control-themes",{"title":1529,"path":1530,"stem":1531},"Triggers та Visual State Manager у WPF","/csharp/desktop-ui/triggers-visual-states","01.csharp/12.desktop-ui/29.triggers-visual-states",{"title":1533,"path":1534,"stem":1535},"Pseudo-classes в Avalonia — замість WPF Triggers","/csharp/desktop-ui/29a.avalonia-pseudo-classes","01.csharp/12.desktop-ui/29a.avalonia-pseudo-classes",{"title":1537,"path":1538,"stem":1539},"Теми та ресурсні словники у WPF","/csharp/desktop-ui/resources-themes","01.csharp/12.desktop-ui/30.resources-themes",{"title":1541,"path":1542,"stem":1543},"Avalonia Themes — Fluent Design та система тематизації","/csharp/desktop-ui/30a.avalonia-themes-fluent","01.csharp/12.desktop-ui/30a.avalonia-themes-fluent",{"title":1545,"path":1546,"stem":1547},"Контроли колекцій — глибоке занурення","/csharp/desktop-ui/collection-controls","01.csharp/12.desktop-ui/31.collection-controls",{"title":1549,"path":1550,"stem":1551},"DataGrid — колонки та базове відображення","/csharp/desktop-ui/datagrid-part1","01.csharp/12.desktop-ui/32.datagrid-part1",{"title":1553,"path":1554,"stem":1555},"DataGrid — сортування, фільтрація, редагування","/csharp/desktop-ui/datagrid-part2","01.csharp/12.desktop-ui/32.datagrid-part2",{"title":1557,"path":1558,"stem":1559},"TreeView та GridView","/csharp/desktop-ui/treeview-listview","01.csharp/12.desktop-ui/33.treeview-listview",{"title":1561,"path":1562,"stem":1563},"Меню, Toolbar, ContextMenu, StatusBar","/csharp/desktop-ui/menus-toolbars","01.csharp/12.desktop-ui/34.menus-toolbars",{"title":1565,"path":1566,"stem":1567},"Навігація та керування вікнами. Частина 1: вікна та сторінки","/csharp/desktop-ui/navigation-windows-part1","01.csharp/12.desktop-ui/35.navigation-windows-part1",{"title":1569,"path":1570,"stem":1571},"Навігація та керування вікнами. Частина 2: MVVM-навігація","/csharp/desktop-ui/navigation-windows-part2","01.csharp/12.desktop-ui/35.navigation-windows-part2",{"title":1573,"path":1574,"stem":1575},"Avalonia — Навігація та діалоги","/csharp/desktop-ui/35a.avalonia-navigation-dialogs","01.csharp/12.desktop-ui/35a.avalonia-navigation-dialogs",{"title":1577,"path":1578,"stem":1579},"Діалоги та File Pickers у WPF","/csharp/desktop-ui/dialogs-file-pickers","01.csharp/12.desktop-ui/36.dialogs-file-pickers",{"title":1581,"path":1582,"stem":1583},"UserControl: компонентний підхід у WPF","/csharp/desktop-ui/user-controls","01.csharp/12.desktop-ui/37.user-controls",{"title":1585,"path":1586,"stem":1587},"Custom Controls: Lookless Controls у WPF","/csharp/desktop-ui/custom-controls","01.csharp/12.desktop-ui/38.custom-controls",{"title":1589,"path":1590,"stem":1591},"Avalonia TemplatedControl — Lookless Controls","/csharp/desktop-ui/38a.avalonia-templated-controls","01.csharp/12.desktop-ui/38a.avalonia-templated-controls",{"title":1593,"path":1594,"stem":1595},"Анімації у WPF: Storyboard та Easing Functions","/csharp/desktop-ui/animations-transitions","01.csharp/12.desktop-ui/39.animations-transitions",{"title":1597,"path":1598,"stem":1599},"Анімації в Avalonia","/csharp/desktop-ui/39a.avalonia-animations","01.csharp/12.desktop-ui/39a.avalonia-animations",{"title":1601,"path":1602,"stem":1603},"2D Графіка та Мультимедіа у WPF","/csharp/desktop-ui/media-graphics","01.csharp/12.desktop-ui/40.media-graphics",{"title":1605,"path":1606,"stem":1607},"Dependency Injection у WPF та Avalonia","/csharp/desktop-ui/di-integration","01.csharp/12.desktop-ui/41.di-integration",{"title":1609,"path":1610,"stem":1611},"SQLite та EF Core у десктопних додатках","/csharp/desktop-ui/data-persistence-part1","01.csharp/12.desktop-ui/42.data-persistence-part1",{"title":1613,"path":1614,"stem":1615},"Repository Pattern та Unit of Work","/csharp/desktop-ui/data-persistence-part2","01.csharp/12.desktop-ui/43.data-persistence-part2",{"title":1617,"path":1618,"stem":1619},"Тестування ViewModels","/csharp/desktop-ui/viewmodel-testing","01.csharp/12.desktop-ui/44.viewmodel-testing",{"title":1621,"path":1622,"stem":1623},"Avalonia Headless Testing — тестування UI без вікон","/csharp/desktop-ui/44a.avalonia-headless-testing","01.csharp/12.desktop-ui/44a.avalonia-headless-testing",{"title":1625,"path":1626,"stem":1627},"Кросплатформна розробка з Avalonia","/csharp/desktop-ui/avalonia-cross-platform","01.csharp/12.desktop-ui/45.avalonia-cross-platform",{"title":1629,"path":1630,"stem":1631},"Пакування та розгортання Avalonia додатків","/csharp/desktop-ui/avalonia-packaging-deployment","01.csharp/12.desktop-ui/46.avalonia-packaging-deployment",{"title":1633,"path":1634,"stem":1635},"Розгортання WPF застосунків","/csharp/desktop-ui/wpf-packaging-deployment","01.csharp/12.desktop-ui/47.wpf-packaging-deployment",{"title":1637,"path":1638,"stem":1639},"C# & .NET: The Ultimate Roadmap","/csharp/roadmap","01.csharp/roadmap",{"title":1641,"icon":1642,"path":1643,"stem":1644,"children":1645,"page":59},"C++","i-devicon-cplusplus","/cpp","02.cpp",[1646,1650,1654,1658,1662,1666,1670,1674,1678,1681,1685,1689,1693,1697,1701,1705,1709,1713,1717,1721,1725,1729,1733,1737,1741,1745,1749,1753,1757,1761],{"title":1647,"path":1648,"stem":1649},"Вступ у програмування та алгоритми","/cpp/intro-algorithms","02.cpp/01.intro-algorithms",{"title":1651,"path":1652,"stem":1653},"Code Style: угоди про оформлення коду","/cpp/code-style","02.cpp/02.code-style",{"title":1655,"path":1656,"stem":1657},"Середовище розробки та перший проєкт","/cpp/ide-setup","02.cpp/03.ide-setup",{"title":1659,"path":1660,"stem":1661},"Вивід даних на екран","/cpp/data-output","02.cpp/04.data-output",{"title":1663,"path":1664,"stem":1665},"Типи даних, змінні та константи","/cpp/data-types-variables","02.cpp/05.data-types-variables",{"title":1667,"path":1668,"stem":1669},"Ввід даних з клавіатури","/cpp/data-input","02.cpp/06.data-input",{"title":1671,"path":1672,"stem":1673},"Оператори, перетворення типів та логічні операції","/cpp/operators-type-conversion","02.cpp/07.operators-type-conversion",{"title":1675,"path":1676,"stem":1677},"Цикли","/cpp/loops","02.cpp/08.loops",{"title":32,"path":1679,"stem":1680},"/cpp/arrays","02.cpp/09.arrays",{"title":1682,"path":1683,"stem":1684},"Алгоритми сортування та аналіз складності","/cpp/sorting","02.cpp/10.sorting",{"title":1686,"path":1687,"stem":1688},"Алгоритми пошуку","/cpp/searching","02.cpp/11.searching",{"title":1690,"path":1691,"stem":1692},"Функції: основи","/cpp/functions-basics","02.cpp/12.functions-basics",{"title":1694,"path":1695,"stem":1696},"Функції: прототипи, область видимості та додаткові можливості","/cpp/functions-scope","02.cpp/13.functions-scope",{"title":1698,"path":1699,"stem":1700},"Функції: перевантаження та шаблони","/cpp/functions-overloading-templates","02.cpp/14.functions-overloading-templates",{"title":1702,"path":1703,"stem":1704},"Вказівники: основи","/cpp/pointers-basics","02.cpp/15.pointers-basics",{"title":1706,"path":1707,"stem":1708},"Посилання (References)","/cpp/references","02.cpp/16.references",{"title":1710,"path":1711,"stem":1712},"Вказівники, const і масиви","/cpp/pointers-const-arrays","02.cpp/17.pointers-const-arrays",{"title":1714,"path":1715,"stem":1716},"Адресна арифметика","/cpp/pointer-arithmetic","02.cpp/18.pointer-arithmetic",{"title":1718,"path":1719,"stem":1720},"Динамічна пам'ять","/cpp/dynamic-memory","02.cpp/19.dynamic-memory",{"title":1722,"path":1723,"stem":1724},"Вказівники типу void","/cpp/void-pointers","02.cpp/20.void-pointers",{"title":1726,"path":1727,"stem":1728},"Вказівники на вказівники","/cpp/pointers-to-pointers","02.cpp/21.pointers-to-pointers",{"title":1730,"path":1731,"stem":1732},"Оператор доступу до членів через вказівник (->)","/cpp/member-access-operator","02.cpp/22.member-access-operator",{"title":1734,"path":1735,"stem":1736},"Цикл for-each (Range-based for)","/cpp/foreach-loop","02.cpp/23.foreach-loop",{"title":1738,"path":1739,"stem":1740},"Вказівники на функції","/cpp/function-pointers","02.cpp/24.function-pointers",{"title":1742,"path":1743,"stem":1744},"Лямбда-вирази","/cpp/lambdas","02.cpp/25.lambdas",{"title":1746,"path":1747,"stem":1748},"Лямбда-захоплення","/cpp/lambda-captures","02.cpp/26.lambda-captures",{"title":1750,"path":1751,"stem":1752},"Еліпсис","/cpp/ellipsis","02.cpp/27.ellipsis",{"title":1754,"path":1755,"stem":1756},"Аргументи командного рядка","/cpp/command-line-arguments","02.cpp/28.command-line-arguments",{"title":1758,"path":1759,"stem":1760},"Перерахування (enum)","/cpp/enum","02.cpp/29.enum",{"title":1762,"path":1763,"stem":1764},"План навчання: Курс C++ — Продовження (Статті 29–60+)","/cpp/curriculum-plan","02.cpp/curriculum-plan",{"title":1766,"icon":1767,"path":1768,"stem":1769,"children":1770,"page":59},"JavaScript","i-devicon-javascript","/javascript","03.javascript",[1771,1797,1851,1873,2177,2215],{"title":1772,"icon":1773,"path":1774,"stem":1775,"children":1776,"page":59},"Events","i-lucide-mouse-pointer-click","/javascript/events","03.javascript/01.events",[1777,1781,1785,1789,1793],{"title":1778,"path":1779,"stem":1780},"Вступ до подій браузера","/javascript/events/intro","03.javascript/01.events/01.intro",{"title":1782,"path":1783,"stem":1784},"Бульбашковий механізм (Bubbling) та занурення (Capturing)","/javascript/events/bubbling-capturing","03.javascript/01.events/02.bubbling-capturing",{"title":1786,"path":1787,"stem":1788},"Делегування подій (Event Delegation)","/javascript/events/delegate-events","03.javascript/01.events/03.delegate-events",{"title":1790,"path":1791,"stem":1792},"Типові дії браузера та preventDefault()","/javascript/events/prevent-default","03.javascript/01.events/04.prevent-default",{"title":1794,"path":1795,"stem":1796},"Запуск користувацьких подій (Custom Events)","/javascript/events/custom-events","03.javascript/01.events/05.custom-events",{"title":1798,"icon":1799,"path":1800,"stem":1801,"children":1802,"page":59},"Network","i-lucide-globe","/javascript/network","03.javascript/02.network",[1803,1807,1811,1815,1819,1823,1827,1831,1835,1839,1843,1847],{"title":1804,"path":1805,"stem":1806},"Fetch API - Сучасний підхід до HTTP-запитів","/javascript/network/01-fetch-api","03.javascript/02.network/01-fetch-api",{"title":1808,"path":1809,"stem":1810},"FormData - Робота з формами та файлами","/javascript/network/02-formdata","03.javascript/02.network/02-formdata",{"title":1812,"path":1813,"stem":1814},"Відстеження прогресу завантаження","/javascript/network/03-download-progress","03.javascript/02.network/03-download-progress",{"title":1816,"path":1817,"stem":1818},"Переривання fetch-запитів","/javascript/network/04-abort-requests","03.javascript/02.network/04-abort-requests",{"title":1820,"path":1821,"stem":1822},"CORS - Запити між різними джерелами","/javascript/network/05-cors","03.javascript/02.network/05-cors",{"title":1824,"path":1825,"stem":1826},"Fetch API - Повний довідник опцій","/javascript/network/06-fetch-options","03.javascript/02.network/06-fetch-options",{"title":1828,"path":1829,"stem":1830},"URL Objects - Робота з посиланнями","/javascript/network/07-url-objects","03.javascript/02.network/07-url-objects",{"title":1832,"path":1833,"stem":1834},"XMLHttpRequest - AJAX та низькорівневі запити","/javascript/network/08-xmlhttprequest","03.javascript/02.network/08-xmlhttprequest",{"title":1836,"path":1837,"stem":1838},"Відновлюване завантаження файлів","/javascript/network/09-resumable-upload","03.javascript/02.network/09-resumable-upload",{"title":1840,"path":1841,"stem":1842},"Cookies, document.cookie та світ після \"Cookiepocalypse\"","/javascript/network/10-cookies","03.javascript/02.network/10-cookies",{"title":1844,"path":1845,"stem":1846},"js-cookie: Керування Cookies без Болю","/javascript/network/11-js-cookie","03.javascript/02.network/11-js-cookie",{"title":1848,"path":1849,"stem":1850},"Axios: Потужний HTTP-клієнт для JavaScript","/javascript/network/12-axios","03.javascript/02.network/12-axios",{"title":1852,"icon":1853,"path":1854,"stem":1855,"children":1856,"page":59},"Bom","i-lucide-monitor","/javascript/bom","03.javascript/03.bom",[1857,1861,1865,1869],{"title":1858,"path":1859,"stem":1860},"LocalStorage, SessionStorage та patterns збереження даних","/javascript/bom/01-localstorage","03.javascript/03.bom/01-localstorage",{"title":1862,"path":1863,"stem":1864},"Location Object - Керування адресою сторінки","/javascript/bom/02-location-object","03.javascript/03.bom/02-location-object",{"title":1866,"path":1867,"stem":1868},"History API - Керування історією браузера","/javascript/bom/03-history-api","03.javascript/03.bom/03-history-api",{"title":1870,"path":1871,"stem":1872},"Navigator Object - Ідентифікація та Можливості Пристрою","/javascript/bom/04-navigator-object","03.javascript/03.bom/04-navigator-object",{"title":1874,"icon":1875,"path":1876,"stem":1877,"children":1878},"React","i-devicon-react","/javascript/react","03.javascript/04.react/index",[1879,1880,1884,1888,1892,1896,1959,1994,2146],{"title":1874,"path":1876,"stem":1877},{"title":1881,"path":1882,"stem":1883},"Робота з Формами в React","/javascript/react/react-forms","03.javascript/04.react/01.react-forms",{"title":1885,"path":1886,"stem":1887},"React Hook Form: Професійна Робота з Формами","/javascript/react/react-hook-form","03.javascript/04.react/02.react-hook-form",{"title":1889,"path":1890,"stem":1891},"React Hook Form: Глибоке Розуміння Архітектури та Оптимізації","/javascript/react/react-hook-form-new","03.javascript/04.react/02.react-hook-form-new",{"title":1893,"path":1894,"stem":1895},"Axios та React: Професійна Архітектура Запитів","/javascript/react/data-fetching-axios","03.javascript/04.react/03.data-fetching-axios",{"title":1897,"icon":132,"path":1898,"stem":1899,"children":1900},"Tanstack Query","/javascript/react/tanstack-query","03.javascript/04.react/04.tanstack-query/index",[1901,1903,1907,1911,1915,1919,1923,1927,1931,1935,1939,1943,1947,1951,1955],{"title":1902,"path":1898,"stem":1899},"TanStack Query: Майстерність Керування Станом Сервера",{"title":1904,"path":1905,"stem":1906},"Парадигма Server State: Чому useEffect недостатньо","/javascript/react/tanstack-query/server-state-paradigm","03.javascript/04.react/04.tanstack-query/01.server-state-paradigm",{"title":1908,"path":1909,"stem":1910},"Встановлення та Налаштування: Фундамент","/javascript/react/tanstack-query/installation-and-devtools","03.javascript/04.react/04.tanstack-query/02.installation-and-devtools",{"title":1912,"path":1913,"stem":1914},"Основи Запитів та Магія Ключів","/javascript/react/tanstack-query/query-basics-and-keys","03.javascript/04.react/04.tanstack-query/03.query-basics-and-keys",{"title":1916,"path":1917,"stem":1918},"Синхронізація Даних: Життєвий Цикл Запиту","/javascript/react/tanstack-query/data-synchronization","03.javascript/04.react/04.tanstack-query/04.data-synchronization",{"title":1920,"path":1921,"stem":1922},"Мутації та Інвалідація: Зміна Даних","/javascript/react/tanstack-query/mutations-and-invalidation","03.javascript/04.react/04.tanstack-query/05.mutations-and-invalidation",{"title":1924,"path":1925,"stem":1926},"Оптимістичні Оновлення: Швидше за Світло","/javascript/react/tanstack-query/optimistic-updates","03.javascript/04.react/04.tanstack-query/06.optimistic-updates",{"title":1928,"path":1929,"stem":1930},"Пагінація та Infinite Scroll","/javascript/react/tanstack-query/pagination-and-load-more","03.javascript/04.react/04.tanstack-query/07.pagination-and-load-more",{"title":1932,"path":1933,"stem":1934},"Просунуті Патерни та Оптимізація","/javascript/react/tanstack-query/advanced-patterns","03.javascript/04.react/04.tanstack-query/08.advanced-patterns",{"title":1936,"path":1937,"stem":1938},"Архітектура та Best Practices","/javascript/react/tanstack-query/architecture-and-best-practices","03.javascript/04.react/04.tanstack-query/09.architecture-and-best-practices",{"title":1940,"path":1941,"stem":1942},"Server-Side Rendering (SSR) та Гідратація","/javascript/react/tanstack-query/server-side-rendering","03.javascript/04.react/04.tanstack-query/10.server-side-rendering",{"title":1944,"path":1945,"stem":1946},"Стратегії Тестування","/javascript/react/tanstack-query/testing-strategies","03.javascript/04.react/04.tanstack-query/11.testing-strategies",{"title":1948,"path":1949,"stem":1950},"Аутентифікація та Обробка Помилок","/javascript/react/tanstack-query/authentication-and-errors","03.javascript/04.react/04.tanstack-query/12.authentication-and-errors",{"title":1952,"path":1953,"stem":1954},"React Suspense та Майбутнє","/javascript/react/tanstack-query/react-suspense","03.javascript/04.react/04.tanstack-query/13.react-suspense",{"title":1956,"path":1957,"stem":1958},"Глибоке Занурення в Продуктивність","/javascript/react/tanstack-query/performance-deep-dive","03.javascript/04.react/04.tanstack-query/14.performance-deep-dive",{"title":1960,"icon":1875,"path":1961,"stem":1962,"children":1963},"React Router","/javascript/react/react-router","03.javascript/04.react/05.react-router/index",[1964,1966,1970,1974,1978,1982,1986,1990],{"title":1965,"path":1961,"stem":1962},"React Router: Навігаційна система сучасного вебу",{"title":1967,"path":1968,"stem":1969},"Налаштування та Базовий Роутинг","/javascript/react/react-router/setup-and-basic-routing","03.javascript/04.react/05.react-router/01.setup-and-basic-routing",{"title":1971,"path":1972,"stem":1973},"Динамічна Навігація","/javascript/react/react-router/navigation-and-links","03.javascript/04.react/05.react-router/02.navigation-and-links",{"title":1975,"path":1976,"stem":1977},"Вкладені Маршрути та Макети","/javascript/react/react-router/nested-routes-and-layouts","03.javascript/04.react/05.react-router/03.nested-routes-and-layouts",{"title":1979,"path":1980,"stem":1981},"Динамічні Маршрути та Параметри","/javascript/react/react-router/dynamic-routing","03.javascript/04.react/05.react-router/04.dynamic-routing",{"title":1983,"path":1984,"stem":1985},"Data APIs: Loaders та Actions","/javascript/react/react-router/data-loading","03.javascript/04.react/05.react-router/05.data-loading",{"title":1987,"path":1988,"stem":1989},"Просунуті Патерни","/javascript/react/react-router/advanced-patterns","03.javascript/04.react/05.react-router/06.advanced-patterns",{"title":1991,"path":1992,"stem":1993},"Legacy Routing: Компонентний підхід","/javascript/react/react-router/legacy-routing","03.javascript/04.react/05.react-router/07.legacy-routing",{"title":1995,"icon":132,"path":1996,"stem":1997,"children":1998},"Redux","/javascript/react/redux","03.javascript/04.react/06.redux/index",[1999,2001,2017,2046,2055,2076,2092,2121],{"title":2000,"path":1996,"stem":1997},"Redux: Еволюція управління станом",{"title":14,"icon":15,"path":2002,"stem":2003,"children":2004,"page":59},"/javascript/react/redux/fundamentals","03.javascript/04.react/06.redux/01.fundamentals",[2005,2009,2013],{"title":2006,"path":2007,"stem":2008},"Вступ до State Management","/javascript/react/redux/fundamentals/intro-state-management","03.javascript/04.react/06.redux/01.fundamentals/01.intro-state-management",{"title":2010,"path":2011,"stem":2012},"Філософія Redux та Три Принципи","/javascript/react/redux/fundamentals/redux-philosophy","03.javascript/04.react/06.redux/01.fundamentals/02.redux-philosophy",{"title":2014,"path":2015,"stem":2016},"Чисті функції та Іммутабельність","/javascript/react/redux/fundamentals/pure-functions-immutability","03.javascript/04.react/06.redux/01.fundamentals/03.pure-functions-immutability",{"title":2018,"icon":132,"path":2019,"stem":2020,"children":2021,"page":59},"Classic Redux","/javascript/react/redux/classic-redux","03.javascript/04.react/06.redux/02.classic-redux",[2022,2026,2030,2034,2038,2042],{"title":2023,"path":2024,"stem":2025},"Створення Store (Classic Redux)","/javascript/react/redux/classic-redux/store-setup","03.javascript/04.react/06.redux/02.classic-redux/01.store-setup",{"title":2027,"path":2028,"stem":2029},"Actions, Constants та Action Creators","/javascript/react/redux/classic-redux/actions-constants","03.javascript/04.react/06.redux/02.classic-redux/02.actions-constants",{"title":2031,"path":2032,"stem":2033},"Логіка Reducers","/javascript/react/redux/classic-redux/reducers","03.javascript/04.react/06.redux/02.classic-redux/03.reducers",{"title":2035,"path":2036,"stem":2037},"Комбінування Reducers (Root Reducer)","/javascript/react/redux/classic-redux/data-flow","03.javascript/04.react/06.redux/02.classic-redux/04.data-flow",{"title":2039,"path":2040,"stem":2041},"Підключення до React (React-Redux)","/javascript/react/redux/classic-redux/react-redux-connection","03.javascript/04.react/06.redux/02.classic-redux/05.react-redux-connection",{"title":2043,"path":2044,"stem":2045},"Middleware та Асинхронність (Redux Thunk)","/javascript/react/redux/classic-redux/middleware-thunk","03.javascript/04.react/06.redux/02.classic-redux/06.middleware-thunk",{"title":2047,"icon":132,"path":2048,"stem":2049,"children":2050,"page":59},"Transition To Rtk","/javascript/react/redux/transition-to-rtk","03.javascript/04.react/06.redux/03.transition-to-rtk",[2051],{"title":2052,"path":2053,"stem":2054},"Проблеми класичного Redux","/javascript/react/redux/transition-to-rtk/problems-with-classic","03.javascript/04.react/06.redux/03.transition-to-rtk/01.problems-with-classic",{"title":2056,"icon":132,"path":2057,"stem":2058,"children":2059,"page":59},"Redux Toolkit","/javascript/react/redux/redux-toolkit","03.javascript/04.react/06.redux/04.redux-toolkit",[2060,2064,2068,2072],{"title":2061,"path":2062,"stem":2063},"Налаштування Store з configureStore","/javascript/react/redux/redux-toolkit/configure-store","03.javascript/04.react/06.redux/04.redux-toolkit/01.configure-store",{"title":2065,"path":2066,"stem":2067},"createSlice: Революція в Redux","/javascript/react/redux/redux-toolkit/create-slice","03.javascript/04.react/06.redux/04.redux-toolkit/02.create-slice",{"title":2069,"path":2070,"stem":2071},"Асинхронність з createAsyncThunk","/javascript/react/redux/redux-toolkit/async-thunks","03.javascript/04.react/06.redux/04.redux-toolkit/03.async-thunks",{"title":2073,"path":2074,"stem":2075},"04. Entity Adapter: Керування нормалізованим станом","/javascript/react/redux/redux-toolkit/entity-adapter","03.javascript/04.react/06.redux/04.redux-toolkit/04.entity-adapter",{"title":2077,"icon":92,"path":2078,"stem":2079,"children":2080,"page":59},"Advanced","/javascript/react/redux/advanced","03.javascript/04.react/06.redux/05.advanced",[2081,2085,2089],{"title":2082,"path":2083,"stem":2084},"Мемоізація та Селектори: Повний Гайд по Reselect","/javascript/react/redux/advanced/selectors-reselect","03.javascript/04.react/06.redux/05.advanced/01.selectors-reselect",{"title":2086,"path":2087,"stem":2088},"RTK Query: Архітектура Серверного Кешу","/javascript/react/redux/advanced/rtk-query-intro","03.javascript/04.react/06.redux/05.advanced/02.rtk-query-intro",{"title":1936,"path":2090,"stem":2091},"/javascript/react/redux/advanced/architecture-best-practices","03.javascript/04.react/06.redux/05.advanced/03.architecture-best-practices",{"title":2093,"icon":132,"path":2094,"stem":2095,"children":2096,"page":59},"Project Kanban","/javascript/react/redux/project-kanban","03.javascript/04.react/06.redux/06.project-kanban",[2097,2101,2105,2109,2113,2117],{"title":2098,"path":2099,"stem":2100},"Проєкт: Kanban Board (Trello Clone)","/javascript/react/redux/project-kanban/project-overview","03.javascript/04.react/06.redux/06.project-kanban/01.project-overview",{"title":2102,"path":2103,"stem":2104},"Налаштування та Типізація","/javascript/react/redux/project-kanban/setup-and-types","03.javascript/04.react/06.redux/06.project-kanban/02.setup-and-types",{"title":2106,"path":2107,"stem":2108},"Board Slice: Серце Дошки","/javascript/react/redux/project-kanban/board-slice","03.javascript/04.react/06.redux/06.project-kanban/03.board-slice",{"title":2110,"path":2111,"stem":2112},"Логіка Drag & Drop","/javascript/react/redux/project-kanban/drag-and-drop-logic","03.javascript/04.react/06.redux/06.project-kanban/04.drag-and-drop-logic",{"title":2114,"path":2115,"stem":2116},"Інтеграція з RTK Query","/javascript/react/redux/project-kanban/rtk-query-integration","03.javascript/04.react/06.redux/06.project-kanban/05.rtk-query-integration",{"title":2118,"path":2119,"stem":2120},"Optimistic Updates","/javascript/react/redux/project-kanban/optimistic-updates","03.javascript/04.react/06.redux/06.project-kanban/06.optimistic-updates",{"title":2122,"icon":132,"path":2123,"stem":2124,"children":2125,"page":59},"Testing","/javascript/react/redux/testing","03.javascript/04.react/06.redux/07.testing",[2126,2130,2134,2138,2142],{"title":2127,"path":2128,"stem":2129},"Тестування Redux","/javascript/react/redux/testing/intro-testing","03.javascript/04.react/06.redux/07.testing/01.intro-testing",{"title":2131,"path":2132,"stem":2133},"Тестування Reducers","/javascript/react/redux/testing/testing-reducers","03.javascript/04.react/06.redux/07.testing/02.testing-reducers",{"title":2135,"path":2136,"stem":2137},"Тестування Селекторів","/javascript/react/redux/testing/testing-selectors","03.javascript/04.react/06.redux/07.testing/03.testing-selectors",{"title":2139,"path":2140,"stem":2141},"Тестування Компонентів (Integration)","/javascript/react/redux/testing/testing-components","03.javascript/04.react/06.redux/07.testing/04.testing-components",{"title":2143,"path":2144,"stem":2145},"Тестування Async Thunks","/javascript/react/redux/testing/testing-thunks","03.javascript/04.react/06.redux/07.testing/05.testing-thunks",{"title":2147,"icon":132,"path":2148,"stem":2149,"children":2150},"Ui Libraries","/javascript/react/ui-libraries","03.javascript/04.react/07.ui-libraries/index",[2151,2153,2157,2161,2165,2169,2173],{"title":2152,"path":2148,"stem":2149},"UI Бібліотеки в React",{"title":2154,"path":2155,"stem":2156},"Вступ до UI Бібліотек: Навіщо Винаходити Велосипед Двічі?","/javascript/react/ui-libraries/introduction-to-ui-libraries","03.javascript/04.react/07.ui-libraries/01.introduction-to-ui-libraries",{"title":2158,"path":2159,"stem":2160},"Філософія shadcn/ui: \"Not a Component Library\"","/javascript/react/ui-libraries/shadcn-philosophy","03.javascript/04.react/07.ui-libraries/02.shadcn-philosophy",{"title":2162,"path":2163,"stem":2164},"Установка та Налаштування shadcn/ui","/javascript/react/ui-libraries/shadcn-installation","03.javascript/04.react/07.ui-libraries/03.shadcn-installation",{"title":2166,"path":2167,"stem":2168},"Базові Компоненти shadcn/ui: Фундамент Інтерфейсу","/javascript/react/ui-libraries/shadcn-components-basics","03.javascript/04.react/07.ui-libraries/04.shadcn-components-basics",{"title":2170,"path":2171,"stem":2172},"Компоненти Форм: Побудова Інтерактивних Form","/javascript/react/ui-libraries/shadcn-components-forms","03.javascript/04.react/07.ui-libraries/05.shadcn-components-forms",{"title":2174,"path":2175,"stem":2176},"Складні Компоненти: Dialog, Dropdown, Table та Command","/javascript/react/ui-libraries/shadcn-components-advanced","03.javascript/04.react/07.ui-libraries/06.shadcn-components-advanced",{"title":2178,"icon":2179,"path":2180,"stem":2181,"children":2182,"page":59},"TypeScript","i-devicon-typescript","/javascript/typescript","03.javascript/05.typescript",[2183,2187,2191,2195,2199,2203,2207,2211],{"title":2184,"path":2185,"stem":2186},"TypeScript: Броня для вашого коду","/javascript/typescript/intro-and-basic-types","03.javascript/05.typescript/01.intro-and-basic-types",{"title":2188,"path":2189,"stem":2190},"Майстерність Моделювання Даних: Інтерфейси та Просунуті Типи","/javascript/typescript/interfaces-and-advanced-types","03.javascript/05.typescript/02.interfaces-and-advanced-types",{"title":2192,"path":2193,"stem":2194},"Алхімія Типів: Generics та Utility Types","/javascript/typescript/generics-and-utilities","03.javascript/05.typescript/03.generics-and-utilities",{"title":2196,"path":2197,"stem":2198},"Архітектура та Шаблони: Класи в TypeScript","/javascript/typescript/classes-and-oop","03.javascript/05.typescript/04.classes-and-oop",{"title":2200,"path":2201,"stem":2202},"Продакшн та Екосистема: Advanced Config & Workflow","/javascript/typescript/advanced-patterns-and-config","03.javascript/05.typescript/05.advanced-patterns-and-config",{"title":2204,"path":2205,"stem":2206},"TypeScript у світі React","/javascript/typescript/react-basics","03.javascript/05.typescript/06.react-basics",{"title":2208,"path":2209,"stem":2210},"React + TypeScript: Продвинуті патерни","/javascript/typescript/react-advanced","03.javascript/05.typescript/07.react-advanced",{"title":2212,"path":2213,"stem":2214},"React + TypeScript: Екосистема та бібліотеки","/javascript/typescript/react-ecosystem","03.javascript/05.typescript/08.react-ecosystem",{"title":2216,"path":2217,"stem":2218},"Atomic Design","/javascript/atomic-design","03.javascript/2.atomic-design",{"title":2220,"icon":2221,"path":2222,"stem":2223,"children":2224,"page":59},"Java","i-devicon-java","/java","04.java",[2225,2228,2231,2235,2239,2243,2247],{"title":162,"path":2226,"stem":2227},"/java/data-mapper-part1","04.java/01.data-mapper-part1",{"title":166,"path":2229,"stem":2230},"/java/data-mapper-part2","04.java/02.data-mapper-part2",{"title":2232,"path":2233,"stem":2234},"Service Layer: Організація бізнес-логіки","/java/service-layer","04.java/03.service-layer",{"title":2236,"path":2237,"stem":2238},"Rich Domain Model та State Pattern","/java/rich-domain-model","04.java/04.rich-domain-model",{"title":2240,"path":2241,"stem":2242},"Патерни для складної бізнес-логіки","/java/business-logic-patterns","04.java/05.business-logic-patterns",{"title":2244,"path":2245,"stem":2246},"Обробка помилок та валідація","/java/error-handling-validation","04.java/06.error-handling-validation",{"title":2248,"path":2249,"stem":2250,"children":2251,"page":59},"Проектування баз даних","/java/pr2","04.java/pr2",[2252,2256,2260,2264,2268,2272,2276,2280,2284,2288,2292,2296,2300,2304,2308,2312,2316,2320,2324,2328,2332,2336,2340,2344,2348],{"title":2253,"path":2254,"stem":2255},"Концептуальне моделювання: Мистецтво розуміння предметної області","/java/pr2/conceptual-modeling","04.java/pr2/01.conceptual-modeling",{"title":2257,"path":2258,"stem":2259},"Логічне моделювання: Від бізнес-ідей до структур даних","/java/pr2/logical-modeling","04.java/pr2/02.logical-modeling",{"title":2261,"path":2262,"stem":2263},"Нормалізація: Гігієна даних та боротьба з аномаліями","/java/pr2/normalization","04.java/pr2/03.normalization",{"title":2265,"path":2266,"stem":2267},"Фізична схема: Від абстракції до DDL","/java/pr2/physical-schema","04.java/pr2/04.physical-schema",{"title":2269,"path":2270,"stem":2271},"Архітектурна класифікація таблиць","/java/pr2/table-classification","04.java/pr2/05.table-classification",{"title":2273,"path":2274,"stem":2275},"Database Migrations: Версіонування схеми з Flyway","/java/pr2/database-migrations","04.java/pr2/06.database-migrations",{"title":2277,"path":2278,"stem":2279},"А що, якби це була не реляційна БД?","/java/pr2/beyond-relational","04.java/pr2/07.beyond-relational",{"title":2281,"path":2282,"stem":2283},"Object-Relational Impedance Mismatch: Два світи, що не хочуть дружити","/java/pr2/impedance-mismatch","04.java/pr2/09.impedance-mismatch",{"title":2285,"path":2286,"stem":2287},"JDBC: Перший контакт із базою даних","/java/pr2/jdbc-fundamentals","04.java/pr2/10.jdbc-fundamentals",{"title":2289,"path":2290,"stem":2291},"Якість коду: Spotless, SpotBugs та SonarQube","/java/pr2/10a.code-quality","04.java/pr2/10a.code-quality",{"title":2293,"path":2294,"stem":2295},"Connection Pool: Патерн Object Pool для JDBC-з'єднань","/java/pr2/connection-pool","04.java/pr2/11.connection-pool",{"title":2297,"path":2298,"stem":2299},"Row Data Gateway: Об'єкт як обгортка рядка таблиці","/java/pr2/row-data-gateway","04.java/pr2/12.row-data-gateway",{"title":2301,"path":2302,"stem":2303},"Table Data Gateway: Фасад таблиці як архітектурний відступ","/java/pr2/table-data-gateway","04.java/pr2/13.table-data-gateway",{"title":2305,"path":2306,"stem":2307},"Repository + Data Mapper: Правильна шарова архітектура з JDBC","/java/pr2/repository-data-mapper","04.java/pr2/14.repository-data-mapper",{"title":2309,"path":2310,"stem":2311},"Identity Map: Кешування сутностей у рамках сесії","/java/pr2/identity-map","04.java/pr2/15.identity-map",{"title":2313,"path":2314,"stem":2315},"Unit of Work: Відстеження змін і координація JDBC-транзакцій","/java/pr2/unit-of-work","04.java/pr2/16.unit-of-work",{"title":2317,"path":2318,"stem":2319},"Strategy: Замінювані SQL-стратегії для підтримки різних СУБД","/java/pr2/strategy-sql","04.java/pr2/17.strategy-sql",{"title":2321,"path":2322,"stem":2323},"Proxy: Lazy Loading для One-To-Many колекцій","/java/pr2/proxy-lazy-loading","04.java/pr2/18.proxy-lazy-loading",{"title":2325,"path":2326,"stem":2327},"Generic Repository через Java Reflection: анотації та динамічний SQL","/java/pr2/generic-repository-reflection","04.java/pr2/19.generic-repository-reflection",{"title":2329,"path":2330,"stem":2331},"Specification Pattern: Композиція бізнес-правил для складних запитів","/java/pr2/specification-pattern","04.java/pr2/20.specification-pattern",{"title":2333,"path":2334,"stem":2335},"Розширені можливості Specification Pattern: підзапити, агрегації та гібридний підхід","/java/pr2/20a.advanced-specifications","04.java/pr2/20a.advanced-specifications",{"title":2337,"path":2338,"stem":2339},"Асинхронність у JDBC: Від блокуючих викликів до CompletableFuture","/java/pr2/asynchronous-jdbc","04.java/pr2/21.asynchronous-jdbc",{"title":2341,"path":2342,"stem":2343},"Інтеграційне тестування JDBC-репозиторіїв: Embedded H2 та патерн AAA","/java/pr2/integration-testing-h2","04.java/pr2/22.integration-testing-h2",{"title":2345,"path":2346,"stem":2347},"Testcontainers: Тестування з реальною PostgreSQL у Docker-контейнерах","/java/pr2/integration-testing-testcontainers","04.java/pr2/23.integration-testing-testcontainers",{"title":2349,"path":2350,"stem":2351},"Модуль \"Проектування реляційних баз даних\" для 04.java/pr2","/java/pr2/implementation_plan","04.java/pr2/implementation_plan",{"title":2353,"icon":2354,"path":2355,"stem":2356,"children":2357,"page":59},"Бази даних","i-lucide-database","/databases","06.databases",[2358,2388,2411,2448,2477,2495,2529,2541,2550],{"title":2359,"icon":2360,"path":2361,"stem":2362,"children":2363,"page":59},"Intro","i-lucide-play","/databases/intro","06.databases/01.intro",[2364,2368,2372,2376,2380,2384],{"title":2365,"path":2366,"stem":2367},"Введення в теорію баз даних","/databases/intro/introduction-to-databases","06.databases/01.intro/01.introduction-to-databases",{"title":2369,"path":2370,"stem":2371},"Реляційна модель даних","/databases/intro/relational-model-theory","06.databases/01.intro/02.relational-model-theory",{"title":2373,"path":2374,"stem":2375},"ER-моделювання","/databases/intro/er-modeling","06.databases/01.intro/03.er-modeling",{"title":2377,"path":2378,"stem":2379},"Логічне проектування БД","/databases/intro/logical-schema","06.databases/01.intro/04.logical-schema",{"title":2381,"path":2382,"stem":2383},"Класифікація таблиць","/databases/intro/table-classification","06.databases/01.intro/05.table-classification",{"title":2385,"path":2386,"stem":2387},"PlantUML для баз даних","/databases/intro/plantuml-diagrams","06.databases/01.intro/06.plantuml-diagrams",{"title":2389,"icon":2354,"path":2390,"stem":2391,"children":2392,"page":59},"MS SQL Server Start","/databases/ms-sql-server-start","06.databases/02.ms-sql-server-start",[2393,2397,2403,2407],{"title":2394,"path":2395,"stem":2396},"Типи даних у MS SQL Server","/databases/ms-sql-server-start/data-types","06.databases/02.ms-sql-server-start/01.data-types",{"title":2398,"path":2399,"stem":2400,"children":2401},"Індекси у MS SQL Server","/databases/ms-sql-server-start/sql-indexes","06.databases/02.ms-sql-server-start/02.sql-indexes",[2402],{"title":2398,"path":2399,"stem":2400},{"title":2404,"path":2405,"stem":2406},"Системні бази даних MS SQL Server","/databases/ms-sql-server-start/system-databases","06.databases/02.ms-sql-server-start/03.system-databases",{"title":2408,"path":2409,"stem":2410},"Огляд мови SQL та запитів","/databases/ms-sql-server-start/sql-queries-overview","06.databases/02.ms-sql-server-start/04.sql-queries-overview",{"title":2412,"icon":2354,"path":2413,"stem":2414,"children":2415,"page":59},"SQL","/databases/sql","06.databases/03.sql",[2416,2420,2424,2428,2432,2436,2440,2444],{"title":2417,"path":2418,"stem":2419},"Налаштування демонстраційної бази даних","/databases/sql/sample-database-setup","06.databases/03.sql/00.sample-database-setup",{"title":2421,"path":2422,"stem":2423},"DDL - Створення таблиць (CREATE TABLE)","/databases/sql/ddl-create-table","06.databases/03.sql/01.ddl-create-table",{"title":2425,"path":2426,"stem":2427},"DDL - Зміна та видалення таблиць (ALTER, DROP)","/databases/sql/ddl-alter-drop-table","06.databases/03.sql/02.ddl-alter-drop-table",{"title":2429,"path":2430,"stem":2431},"SELECT запити - Основи","/databases/sql/select-queries-fundamentals","06.databases/03.sql/03.select-queries-fundamentals",{"title":2433,"path":2434,"stem":2435},"SELECT запити - Розширені можливості","/databases/sql/select-queries-advanced","06.databases/03.sql/04.select-queries-advanced",{"title":2437,"path":2438,"stem":2439},"INSERT запити - Додавання даних","/databases/sql/insert-queries","06.databases/03.sql/05.insert-queries",{"title":2441,"path":2442,"stem":2443},"UPDATE та DELETE запити","/databases/sql/update-delete-queries","06.databases/03.sql/06.update-delete-queries",{"title":2445,"path":2446,"stem":2447},"Транзакції в SQL","/databases/sql/transactions","06.databases/03.sql/07.transactions",{"title":2449,"icon":2354,"path":2450,"stem":2451,"children":2452,"page":59},"Multi Table Databases","/databases/multi-table-databases","06.databases/04.multi-table-databases",[2453,2457,2461,2465,2469,2473],{"title":2454,"path":2455,"stem":2456},"Зв'язки та нормалізація БД","/databases/multi-table-databases/relationships-and-normalization","06.databases/04.multi-table-databases/00.relationships-and-normalization",{"title":2458,"path":2459,"stem":2460},"INNER JOIN - З'єднання таблиць","/databases/multi-table-databases/inner-join","06.databases/04.multi-table-databases/01.inner-join",{"title":2462,"path":2463,"stem":2464},"OUTER JOINs - LEFT, RIGHT, FULL","/databases/multi-table-databases/outer-joins","06.databases/04.multi-table-databases/02.outer-joins",{"title":2466,"path":2467,"stem":2468},"CROSS та SELF JOINs","/databases/multi-table-databases/cross-self-joins","06.databases/04.multi-table-databases/03.cross-self-joins",{"title":2470,"path":2471,"stem":2472},"Підзапити (Subqueries)","/databases/multi-table-databases/subqueries","06.databases/04.multi-table-databases/04.subqueries",{"title":2474,"path":2475,"stem":2476},"Агрегації з JOIN","/databases/multi-table-databases/aggregations-with-joins","06.databases/04.multi-table-databases/05.aggregations-with-joins",{"title":2478,"icon":2479,"path":2480,"stem":2481,"children":2482,"page":59},"Aggregate Functions","i-lucide-calculator","/databases/aggregate-functions","06.databases/05.aggregate-functions",[2483,2487,2491],{"title":2484,"path":2485,"stem":2486},"Функції агрегування в MS SQL Server","/databases/aggregate-functions/introduction-aggregate-functions","06.databases/05.aggregate-functions/01.introduction-aggregate-functions",{"title":2488,"path":2489,"stem":2490},"Групування даних в MS SQL Server","/databases/aggregate-functions/grouping-data","06.databases/05.aggregate-functions/02.grouping-data",{"title":2492,"path":2493,"stem":2494},"Підзапити з агрегатними функціями","/databases/aggregate-functions/subqueries-aggregates","06.databases/05.aggregate-functions/03.subqueries-aggregates",{"title":2496,"icon":2497,"path":2498,"stem":2499,"children":2500,"page":59},"Тригери та зберігаємі процедури","i-lucide-database-zap","/databases/triggers-stored-procedures","06.databases/07.triggers-stored-procedures",[2501,2505,2509,2513,2517,2521,2525],{"title":2502,"path":2503,"stem":2504},"DML-тригери","/databases/triggers-stored-procedures/dml-triggers","06.databases/07.triggers-stored-procedures/01.dml-triggers",{"title":2506,"path":2507,"stem":2508},"DDL-тригери","/databases/triggers-stored-procedures/ddl-triggers","06.databases/07.triggers-stored-procedures/02.ddl-triggers",{"title":2510,"path":2511,"stem":2512},"Transact-SQL розширення","/databases/triggers-stored-procedures/transact-sql-extensions","06.databases/07.triggers-stored-procedures/03.transact-sql-extensions",{"title":2514,"path":2515,"stem":2516},"Транзакції","/databases/triggers-stored-procedures/transactions","06.databases/07.triggers-stored-procedures/04.transactions",{"title":2518,"path":2519,"stem":2520},"Зберігаємі процедури","/databases/triggers-stored-procedures/stored-procedures","06.databases/07.triggers-stored-procedures/05.stored-procedures",{"title":2522,"path":2523,"stem":2524},"Користувацькі функції","/databases/triggers-stored-procedures/user-defined-functions","06.databases/07.triggers-stored-procedures/06.user-defined-functions",{"title":2526,"path":2527,"stem":2528},"Безпека баз даних","/databases/triggers-stored-procedures/security","06.databases/07.triggers-stored-procedures/08.security",{"title":2526,"icon":793,"path":2530,"stem":2531,"children":2532,"page":59},"/databases/security","06.databases/08.security",[2533,2537],{"title":2534,"path":2535,"stem":2536},"Вступ до безпеки баз даних","/databases/security/introduction","06.databases/08.security/01.introduction",{"title":2538,"path":2539,"stem":2540},"Системні представлення та метадані","/databases/security/system-views","06.databases/08.security/02.system-views",{"title":2542,"icon":2543,"path":2544,"stem":2545,"children":2546,"page":59},"Резервне копіювання та відновлення","i-lucide-database-backup","/databases/backup-recovery","06.databases/09.backup-recovery",[2547],{"title":2542,"path":2548,"stem":2549},"/databases/backup-recovery/backup-restore","06.databases/09.backup-recovery/01.backup-restore",{"title":2551,"icon":2552,"path":2553,"stem":2554,"children":2555,"page":59},"Повнотекстовий пошук","i-lucide-search","/databases/full-text-search","06.databases/10.full-text-search",[2556],{"title":2551,"path":2557,"stem":2558},"/databases/full-text-search/full-text-search","06.databases/10.full-text-search/01.full-text-search",{"title":2560,"icon":2561,"path":2562,"stem":2563,"children":2564,"page":59},"Tools","i-lucide-wrench","/tools","07.tools",[2565],{"title":2566,"icon":2567,"path":2568,"stem":2569,"children":2570},"Docker","i-simple-icons-docker","/tools/docker","07.tools/01.docker/index",[2571,2573,2577,2581,2585,2589,2593,2597,2601,2605,2609,2613,2617,2621,2625,2629,2633,2637],{"title":2572,"path":2568,"stem":2569},"Docker: від нуля до production",{"title":2574,"path":2575,"stem":2576},"Контейнеризація — від проблеми до рішення","/tools/docker/containerization-concept","07.tools/01.docker/01.containerization-concept",{"title":2578,"path":2579,"stem":2580},"Docker — що це і навіщо?","/tools/docker/docker-what-and-why","07.tools/01.docker/02.docker-what-and-why",{"title":2582,"path":2583,"stem":2584},"Архітектура Docker Engine","/tools/docker/docker-architecture","07.tools/01.docker/03.docker-architecture",{"title":2586,"path":2587,"stem":2588},"Встановлення Docker","/tools/docker/installation","07.tools/01.docker/04.installation",{"title":2590,"path":2591,"stem":2592},"Перший контейнер — docker run","/tools/docker/first-container","07.tools/01.docker/05.first-container",{"title":2594,"path":2595,"stem":2596},"Життєвий цикл контейнера","/tools/docker/container-lifecycle","07.tools/01.docker/06.container-lifecycle",{"title":2598,"path":2599,"stem":2600},"Docker Images — фундаментальні концепції","/tools/docker/docker-images-fundamentals","07.tools/01.docker/07.docker-images-fundamentals",{"title":2602,"path":2603,"stem":2604},"Dockerfile — основи","/tools/docker/dockerfile-basics","07.tools/01.docker/08.dockerfile-basics",{"title":2606,"path":2607,"stem":2608},"Dockerfile — просунуті техніки","/tools/docker/dockerfile-advanced","07.tools/01.docker/09.dockerfile-advanced",{"title":2610,"path":2611,"stem":2612},"Build Context та кешування шарів","/tools/docker/build-context-and-cache","07.tools/01.docker/10.build-context-and-cache",{"title":2614,"path":2615,"stem":2616},"Реєстри Docker-образів","/tools/docker/image-registries","07.tools/01.docker/11.image-registries",{"title":2618,"path":2619,"stem":2620},"Контейнеризація .NET додатків","/tools/docker/dotnet-containerization","07.tools/01.docker/12.dotnet-containerization",{"title":2622,"path":2623,"stem":2624},"Томи та збереження даних","/tools/docker/volumes-and-data","07.tools/01.docker/13.volumes-and-data",{"title":2626,"path":2627,"stem":2628},"Основи мережі в Docker","/tools/docker/networking-basics","07.tools/01.docker/14.networking-basics",{"title":2630,"path":2631,"stem":2632},"Змінні оточення та конфігурація","/tools/docker/environment-and-configuration","07.tools/01.docker/15.environment-and-configuration",{"title":2634,"path":2635,"stem":2636},"Docker Compose — оркестрація контейнерів","/tools/docker/docker-compose-basics","07.tools/01.docker/16.docker-compose-basics",{"title":2638,"path":2639,"stem":2640},"Docker Compose — Multi-Service застосунки","/tools/docker/compose-multi-service","07.tools/01.docker/17.compose-multi-service",{"title":2642,"icon":2643,"path":2644,"stem":2645,"children":2646,"page":59},"Software Engineering","i-lucide-code-2","/software-engineering","09.software-engineering",[2647,2651,2655,2659,2663,2667,2671,2675,2679,2683,2687],{"title":2648,"path":2649,"stem":2650},"1. Аналіз предметної області. Експертні знання та складність","/software-engineering/intro.subdomains","09.software-engineering/01.intro.subdomains",{"title":2652,"path":2653,"stem":2654},"2. Обмежені контексти. Інтеграція обмежених контекстів","/software-engineering/integrating-limited-contexts","09.software-engineering/02.integrating-limited-contexts",{"title":2656,"path":2657,"stem":2658},"3. Реалізація простої бізнес-логіки","/software-engineering/simple","09.software-engineering/03.simple",{"title":2660,"path":2661,"stem":2662},"4. Опрацювання складної бізнес-логіки","/software-engineering/complex-business-logic","09.software-engineering/04.complex-business-logic",{"title":2664,"path":2665,"stem":2666},"5. Моделювання фактора часу. Подієво-орієнтована архітектура.","/software-engineering/modelling-the-time-factor","09.software-engineering/05.modelling-the-time-factor",{"title":2668,"path":2669,"stem":2670},"6. Архітектурні патерни","/software-engineering/architectural-patterns","09.software-engineering/06.architectural-patterns",{"title":2672,"path":2673,"stem":2674},"Паттерни взаємодії","/software-engineering/patterns-of-interaction","09.software-engineering/07.patterns-of-interaction",{"title":2676,"path":2677,"stem":2678},"Евристика проєктування","/software-engineering/design-heuristics","09.software-engineering/08.design-heuristics",{"title":2680,"path":2681,"stem":2682},"Еволюція проєктних рішень","/software-engineering/evolution-of-design-solutions","09.software-engineering/09.evolution-of-design-solutions",{"title":2684,"path":2685,"stem":2686},"EventStorming","/software-engineering/eventstorming","09.software-engineering/10.eventstorming",{"title":2688,"path":2689,"stem":2690},"DDD на практиці","/software-engineering/ddd-in-practice","09.software-engineering/11.ddd-in-practice",{"title":2692,"icon":943,"path":2693,"stem":2694,"children":2695,"page":59},"DDD","/ddd","10.ddd",[2696,2700,2704,2708,2712,2716,2720,2724,2728,2732,2736,2740,2744],{"title":2697,"path":2698,"stem":2699},"Аналіз предметної області","/ddd/domain-analysis","10.ddd/01.domain-analysis",{"title":2701,"path":2702,"stem":2703},"Експертні знання про предметну область","/ddd/domain-expert-knowledge","10.ddd/02.domain-expert-knowledge",{"title":2705,"path":2706,"stem":2707},"Як осмислити складність предметної області","/ddd/managing-domain-complexity","10.ddd/03.managing-domain-complexity",{"title":2709,"path":2710,"stem":2711},"Інтеграція обмежених контекстів","/ddd/bounded-context-integration","10.ddd/04.bounded-context-integration",{"title":2713,"path":2714,"stem":2715},"Реалізація простої бізнес-логіки","/ddd/simple-business-logic","10.ddd/05.simple-business-logic",{"title":2717,"path":2718,"stem":2719},"Обробка складної бізнес-логіки","/ddd/complex-business-logic","10.ddd/06.complex-business-logic",{"title":2721,"path":2722,"stem":2723},"Моделювання фактора часу","/ddd/time-modeling","10.ddd/07.time-modeling",{"title":2725,"path":2726,"stem":2727},"Глава 8. Архітектурні Патерни","/ddd/architectural-patterns","10.ddd/08.architectural-patterns",{"title":2729,"path":2730,"stem":2731},"Глава 9. Патерни Взаємодії","/ddd/interaction-patterns","10.ddd/09.interaction-patterns",{"title":2733,"path":2734,"stem":2735},"Глава 10. Проектні Евристики","/ddd/design-heuristics","10.ddd/10.design-heuristics",{"title":2737,"path":2738,"stem":2739},"Глава 11. Еволюція Проектних Рішень","/ddd/evolution-of-design-decisions","10.ddd/11.evolution-of-design-decisions",{"title":2741,"path":2742,"stem":2743},"Глава 12. EventStorming","/ddd/event-storming","10.ddd/12.event-storming",{"title":2745,"path":2746,"stem":2747},"Глава 13. DDD на Практиці","/ddd/ddd-in-practice","10.ddd/13.ddd-in-practice",{"title":2749,"icon":2750,"path":2751,"stem":2752,"children":2753,"page":59},"Media Streaming","i-lucide-video","/media-streaming","11.media-streaming",[2754,2758,2762,2766,2770,2774,2778],{"title":2755,"path":2756,"stem":2757},"01. Магія Стрімінгу: Що відбувається, коли ви натискаєте \"Play\"","/media-streaming/introduction","11.media-streaming/01.introduction",{"title":2759,"path":2760,"stem":2761},"02. Анатомія Медіа: Кодеки, Контейнери та Стиснення","/media-streaming/audio-video-anatomy","11.media-streaming/02.audio-video-anatomy",{"title":2763,"path":2764,"stem":2765},"03. The Gym: FFmpeg Deep Dive","/media-streaming/ffmpeg-gym","11.media-streaming/03.ffmpeg-gym",{"title":2767,"path":2768,"stem":2769},"04. HLS Protocol: HTTP Live Streaming у Деталях","/media-streaming/hls-protocol","11.media-streaming/04.hls-protocol",{"title":2771,"path":2772,"stem":2773},"05. DASH Protocol: Відкритий Стандарт","/media-streaming/dash-protocol","11.media-streaming/05.dash-protocol",{"title":2775,"path":2776,"stem":2777},"06. Масштабування: CDN та Adaptive Bitrate","/media-streaming/cdn-and-adaptive-bitrate","11.media-streaming/06.cdn-and-adaptive-bitrate",{"title":2779,"path":2780,"stem":2781},"07. Війна із Затримкою (Latency)","/media-streaming/realtime-latency","11.media-streaming/07.realtime-latency",{"title":2783,"icon":2784,"path":2785,"stem":2786,"children":2787,"page":59},"HTML & CSS","i-devicon-html5","/html-css","12.html-css",[2788,2792,2796,2800,2804,2808,2812,2816,2820,2824,2828,2832,2836,2840,2844,2848,2852,2856,2860,2864,2868,2872,2876,2880,2884,2888,2892,2896,2900,2904],{"title":2789,"path":2790,"stem":2791},"Вступ до HTML. Структура документа","/html-css/intro-html-structure","12.html-css/01.intro-html-structure",{"title":2793,"path":2794,"stem":2795},"Форматування тексту в HTML","/html-css/html-text-formatting","12.html-css/02.html-text-formatting",{"title":2797,"path":2798,"stem":2799},"Посилання та зображення в HTML","/html-css/html-links-images","12.html-css/03.html-links-images",{"title":2801,"path":2802,"stem":2803},"Списки та таблиці в HTML","/html-css/html-lists-tables","12.html-css/04.html-lists-tables",{"title":2805,"path":2806,"stem":2807},"Форми в HTML","/html-css/html-forms","12.html-css/05.html-forms",{"title":2809,"path":2810,"stem":2811},"Семантичні елементи HTML5","/html-css/html-semantic-elements","12.html-css/06.html-semantic-elements",{"title":2813,"path":2814,"stem":2815},"Мультимедіа та розширені елементи HTML","/html-css/html-multimedia-advanced","12.html-css/07.html-multimedia-advanced",{"title":2817,"path":2818,"stem":2819},"Мікророзмітка та SEO в HTML","/html-css/html-microdata-seo","12.html-css/08.html-microdata-seo",{"title":2821,"path":2822,"stem":2823},"Вступ до CSS. Селектори та специфічність","/html-css/css-intro-selectors","12.html-css/09.css-intro-selectors",{"title":2825,"path":2826,"stem":2827},"Блокова модель CSS. Відступи. Box Sizing","/html-css/css-box-model","12.html-css/10.css-box-model",{"title":2829,"path":2830,"stem":2831},"Розміри у CSS: повний довідник одиниць і ключових слів","/html-css/10a.css-sizing","12.html-css/10a.css-sizing",{"title":2833,"path":2834,"stem":2835},"Типографіка в CSS. Шрифти та текст","/html-css/css-typography","12.html-css/11.css-typography",{"title":2837,"path":2838,"stem":2839},"Кольори та фони в CSS","/html-css/css-colors-backgrounds","12.html-css/12.css-colors-backgrounds",{"title":2841,"path":2842,"stem":2843},"Тіні та фільтри в CSS","/html-css/12b.css-shadows-filters","12.html-css/12b.css-shadows-filters",{"title":2845,"path":2846,"stem":2847},"CSS Flexbox: Фундамент гнучких макетів","/html-css/css-flexbox-fundamentals","12.html-css/13.css-flexbox-fundamentals",{"title":2849,"path":2850,"stem":2851},"CSS Flexbox: Вирівнювання та Позиціонування","/html-css/css-flexbox-alignment-sizing-and-patterns","12.html-css/14.css-flexbox-alignment-sizing-and-patterns",{"title":2853,"path":2854,"stem":2855},"CSS Grid. Двовимірний макет. Частина 1","/html-css/css-layout-grid","12.html-css/15.css-layout-grid",{"title":2857,"path":2858,"stem":2859},"CSS Grid. Двовимірний макет. Частина 2","/html-css/css-layout-grid-advanced","12.html-css/16.css-layout-grid-advanced",{"title":2861,"path":2862,"stem":2863},"Позиціонування в CSS. Z-index. Stacking Context","/html-css/css-positioning","12.html-css/17.css-positioning",{"title":2865,"path":2866,"stem":2867},"CSS Анімації та Переходи","/html-css/css-animations-transitions","12.html-css/18.css-animations-transitions",{"title":2869,"path":2870,"stem":2871},"Адаптивний дизайн. Media Queries. Частина 1","/html-css/css-responsive-media-queries","12.html-css/19.css-responsive-media-queries",{"title":2873,"path":2874,"stem":2875},"Адаптивний дизайн. Частина 2: clamp(), Container Queries, @layer","/html-css/css-responsive-advanced","12.html-css/20.css-responsive-advanced",{"title":2877,"path":2878,"stem":2879},"CSS Custom Properties. Методології. Сучасний CSS","/html-css/css-variables-methodologies","12.html-css/21.css-variables-methodologies",{"title":2881,"path":2882,"stem":2883},"Сучасний CSS 2023–2025: Нові можливості","/html-css/css-modern-features","12.html-css/22.css-modern-features",{"title":2885,"path":2886,"stem":2887},"CSS Nesting, @layer, @scope та @property: нативний препроцесор","/html-css/22a.css-nesting-modern-syntax","12.html-css/22a.css-nesting-modern-syntax",{"title":2889,"path":2890,"stem":2891},"CSS для форм та інтерактивних станів","/html-css/css-forms-interactive-states","12.html-css/23.css-forms-interactive-states",{"title":2893,"path":2894,"stem":2895},"Доступність у CSS (CSS Accessibility)","/html-css/css-accessibility","12.html-css/24.css-accessibility",{"title":2897,"path":2898,"stem":2899},"CSS-функції та сучасні sizing primitives","/html-css/css-functions-sizing","12.html-css/25.css-functions-sizing",{"title":2901,"path":2902,"stem":2903},"Rendering Pipeline і CSS Performance","/html-css/css-rendering-performance","12.html-css/26.css-rendering-performance",{"title":2905,"path":2906,"stem":2907},"CSS Best Practices: типові ситуації та правильні рішення","/html-css/css-best-practices","12.html-css/27.css-best-practices",{"title":2909,"path":2910,"stem":2911,"children":2912,"page":59},"Tailwind","/tailwind","21.tailwind",[2913,2917,2921,2925,2929,2933,2937,2941],{"title":2914,"path":2915,"stem":2916},"Що таке Tailwind CSS і навіщо він потрібен","/tailwind/tailwind-intro-philosophy","21.tailwind/01.tailwind-intro-philosophy",{"title":2918,"path":2919,"stem":2920},"Встановлення та налаштування Tailwind CSS v4","/tailwind/tailwind-installation-setup","21.tailwind/02.tailwind-installation-setup",{"title":2922,"path":2923,"stem":2924},"Utility-класи: основи та система Tailwind","/tailwind/tailwind-utility-classes-core","21.tailwind/03.tailwind-utility-classes-core",{"title":2926,"path":2927,"stem":2928},"Layout: Flexbox та Grid через Tailwind","/tailwind/tailwind-flexbox-grid","21.tailwind/04.tailwind-flexbox-grid",{"title":2930,"path":2931,"stem":2932},"Кастомізація теми через @theme у Tailwind v4","/tailwind/tailwind-theme-customization","21.tailwind/05.tailwind-theme-customization",{"title":2934,"path":2935,"stem":2936},"Варіанти: hover, focus, responsive, dark mode та нові v4","/tailwind/tailwind-variants-states","21.tailwind/06.tailwind-variants-states",{"title":2938,"path":2939,"stem":2940},"Типографіка та система кольорів у Tailwind v4","/tailwind/tailwind-typography-colors","21.tailwind/07.tailwind-typography-colors",{"title":2942,"path":2943,"stem":2944},"Компоненти та повторюваність: @apply, @utility та патерни","/tailwind/tailwind-components-patterns","21.tailwind/08.tailwind-components-patterns",{"title":2946,"path":2947,"stem":2948},"Showcase Компонентів kostyl.dev","/test-new-components","98.test-new-components",{"id":2950,"title":2289,"body":2951,"description":10413,"extension":10414,"links":10415,"meta":10416,"navigation":3394,"path":2290,"seo":10417,"stem":2291,"__hash__":10418},"docs/04.java/pr2/10a.code-quality.md",{"type":2952,"value":2953,"toc":10363},"minimark",[2954,2958,2963,2972,2975,2987,2994,2997,3039,3042,3069,3072,3076,3083,3088,3091,3101,3104,3173,3177,3180,3281,3301,3303,3307,3332,3336,3443,3450,3452,3456,3981,3986,4057,4059,4063,4794,4799,4867,4869,4873,4876,5214,5233,5235,5239,5251,5266,5288,5300,5302,5306,5309,5316,5322,5330,5366,5409,5418,5427,5439,5441,5448,5471,5513,5522,5524,5532,5543,5596,5704,5706,5710,5718,5725,5728,5845,5981,5986,6031,6033,6040,6043,6083,6144,6156,6158,6165,6201,6210,6212,6216,6223,6230,6340,6405,6418,6424,6426,6433,6486,6542,6554,6556,6563,6593,6611,6619,6621,6628,6659,6677,6689,6691,6698,6729,6747,6757,6759,6763,6770,6799,6816,6838,6843,6899,6901,6905,6908,6915,6941,6947,6949,6956,6982,6998,7000,7007,7050,7074,7076,7080,7087,7163,7267,7274,7280,7295,7297,7301,7308,7420,7542,7544,7551,7561,7620,7729,7742,7744,7751,8093,8095,8099,8307,8309,8313,8324,8331,8335,8446,8450,8648,8656,8819,8823,9074,9079,9158,9160,9164,9167,9171,9273,9276,9280,9368,9373,9400,9407,9410,9589,9594,9635,9638,10010,10014,10050,10053,10058,10109,10124,10126,10130,10271,10273,10277,10284,10344,10354,10360],[2955,2956,2289],"h1",{"id":2957},"якість-коду-spotless-spotbugs-та-sonarqube",[2959,2960,2962],"h2",{"id":2961},"вступ-чому-форматування-це-не-питання-смаку","Вступ: Чому форматування — це не питання смаку",[2964,2965,2966,2967,2971],"p",{},"Уявіть типову ситуацію: три розробники працюють над одним проєктом. Перший використовує відступ у 4 пробіли, другий — 2, третій — табуляцію. Перший додає порожній рядок перед ",[2968,2969,2970],"code",{},"return",", другий ні. Перший сортує imports вручну, другий покладається на IntelliJ, третій взагалі про це не думає.",[2964,2973,2974],{},"Результат? Кожен commit містить десятки змін форматування, що «ховають» справжні логічні зміни. Code review перетворюється на полювання за справжніми правками серед шуму. Git history втрачає цінність. Merge conflicts виникають там, де не повинні.",[2964,2976,2977,2978,2982,2983,2986],{},"Проблема фундаментальна: ",[2979,2980,2981],"strong",{},"домовленості між людьми ненадійні",". Навіть найдисциплінованіша команда забуде про правило під тиском дедлайну. Рішення — ",[2979,2984,2985],{},"автоматизація",": інструменти, що перевіряють і виправляють форматування механічно, без людської участі.",[2964,2988,2989,2990,2993],{},"У Java-світі де-факто стандартом форматування є ",[2979,2991,2992],{},"Google Java Style Guide"," — документ, що чітко визначає кожен аспект зовнішнього вигляду коду. Він використовується у Google, Android, Apache та більшості великих open-source проєктів.",[2964,2995,2996],{},"Існує три рівні контролю якості коду, і кожен вирішує свою задачу:",[2998,2999,3000,3014,3027],"card-group",{},[3001,3002,3005,3011],"card",{"icon":3003,"title":3004},"i-heroicons-paint-brush","Форматування",[2964,3006,3007,3010],{},[2979,3008,3009],{},"Інструменти:"," Spotless + google-java-format",[2964,3012,3013],{},"Гарантує: код виглядає однаково незалежно від того, хто і в якій IDE його писав. Перевіряється автоматично при кожному білді.",[3001,3015,3018,3024],{"icon":3016,"title":3017},"i-heroicons-bug-ant","Аналіз патернів",[2964,3019,3020,3023],{},[2979,3021,3022],{},"Інструмент:"," SpotBugs",[2964,3025,3026],{},"Знаходить: типові патерни помилок (null pointer, ігнорований результат, SQL-ін'єкція через конкатенацію). Без запуску коду.",[3001,3028,3031,3036],{"icon":3029,"title":3030},"i-heroicons-chart-bar","Архітектурний аналіз",[2964,3032,3033,3035],{},[2979,3034,3022],{}," SonarQube",[2964,3037,3038],{},"Вимірює: складність, дублювання, покриття тестами, технічний борг. Відстежує тренди у часі.",[2964,3040,3041],{},"У цій статті ми налаштуємо всі три рівні для нашого проєкту аудіоплатформи. Почнемо з найближчого до розробника — IDE — і підемо до повністю автоматизованого pipeline.",[3043,3044,3045,3048,3049,3052,3053,3056,3057,3060,3061,3064,3065,3068],"note",{},[2979,3046,3047],{},"Актуальні версії (квітень 2026):"," Spotless Gradle ",[2968,3050,3051],{},"8.4.0"," / Maven ",[2968,3054,3055],{},"3.4.0",", google-java-format ",[2968,3058,3059],{},"1.35.0",", SpotBugs Gradle ",[2968,3062,3063],{},"6.5.1",", SonarQube Gradle ",[2968,3066,3067],{},"7.2.3.7755",".",[3070,3071],"hr",{},[2959,3073,3075],{"id":3074},"intellij-idea-google-java-style-як-відправна-точка","IntelliJ IDEA: Google Java Style як відправна точка",[2964,3077,3078,3079,3082],{},"Перший крок — налаштувати IntelliJ IDEA так, щоб ",[2968,3080,3081],{},"Ctrl+Alt+L"," форматував код відповідно до Google Java Style Guide. Google надає готовий XML-файл конфігурації, сумісний з IntelliJ.",[3084,3085,3087],"h3",{"id":3086},"де-взяти-та-як-встановити","Де взяти та як встановити",[2964,3089,3090],{},"Файл конфігурації доступний за адресою:",[3092,3093,3098],"pre",{"className":3094,"code":3096,"language":3097},[3095],"language-text","https://google.github.io/styleguide/intellij-java-google-style.xml\n","text",[2968,3099,3096],{"__ignoreMap":3100},"",[2964,3102,3103],{},"Завантажте файл та імпортуйте його:",[3105,3106,3107,3111,3121,3125,3130,3134,3144,3148,3160,3164],"steps",{},[3084,3108,3110],{"id":3109},"відкрийте-налаштування","Відкрийте налаштування",[2964,3112,3113,3116,3117,3120],{},[2968,3114,3115],{},"File → Settings"," (Windows/Linux) або ",[2968,3118,3119],{},"IntelliJ IDEA → Settings"," (macOS)",[3084,3122,3124],{"id":3123},"перейдіть-до-code-style","Перейдіть до Code Style",[2964,3126,3127],{},[2968,3128,3129],{},"Editor → Code Style → Java",[3084,3131,3133],{"id":3132},"імпортуйте-схему","Імпортуйте схему",[2964,3135,3136,3137,3140,3141],{},"Натисніть іконку шестерні (⚙) праворуч від поля ",[2968,3138,3139],{},"Scheme"," → ",[2968,3142,3143],{},"Import Scheme → IntelliJ IDEA code style XML",[3084,3145,3147],{"id":3146},"оберіть-файл","Оберіть файл",[2964,3149,3150,3151,3140,3154,3140,3157],{},"Вкажіть завантажений ",[2968,3152,3153],{},"intellij-java-google-style.xml",[2968,3155,3156],{},"Apply",[2968,3158,3159],{},"OK",[3084,3161,3163],{"id":3162},"перевірте-активацію","Перевірте активацію",[2964,3165,3166,3167,3169,3170],{},"У полі ",[2968,3168,3139],{}," має відображатися ",[2968,3171,3172],{},"GoogleStyle",[3084,3174,3176],{"id":3175},"ключові-параметри-google-java-style","Ключові параметри Google Java Style",[2964,3178,3179],{},"Після імпорту IntelliJ знатиме такі правила форматування:",[3181,3182,3183,3199],"table",{},[3184,3185,3186],"thead",{},[3187,3188,3189,3193,3196],"tr",{},[3190,3191,3192],"th",{},"Параметр",[3190,3194,3195],{},"Значення",[3190,3197,3198],{},"Причина",[3200,3201,3202,3216,3229,3252,3265],"tbody",{},[3187,3203,3204,3208,3213],{},[3205,3206,3207],"td",{},"Відступ",[3205,3209,3210],{},[2979,3211,3212],{},"2 пробіли",[3205,3214,3215],{},"Менше горизонтального прокручування; код вміщується на split-screen",[3187,3217,3218,3221,3226],{},[3205,3219,3220],{},"Ширина рядка",[3205,3222,3223],{},[2979,3224,3225],{},"100 символів",[3205,3227,3228],{},"Компроміс між читабельністю та щільністю",[3187,3230,3231,3237,3242],{},[3205,3232,3233,3234],{},"Дужки в ",[2968,3235,3236],{},"if/for/while",[3205,3238,3239],{},[2979,3240,3241],{},"Обов'язкові",[3205,3243,3244,3245],{},"Запобігає класу помилок типу ",[3246,3247,3251],"a",{"href":3248,"rel":3249},"https://www.imperialviolet.org/2014/02/22/applebug.html",[3250],"nofollow","Apple SSL bug",[3187,3253,3254,3257,3262],{},[3205,3255,3256],{},"Порожні рядки між методами",[3205,3258,3259],{},[2979,3260,3261],{},"1",[3205,3263,3264],{},"Чіткі візуальні блоки без надлишкового пробілу",[3187,3266,3267,3270,3275],{},[3205,3268,3269],{},"Imports",[3205,3271,3272],{},[2979,3273,3274],{},"Без wildcard",[3205,3276,3277,3280],{},[2968,3278,3279],{},"import java.util.*"," ховає залежності",[3282,3283,3284,3285,3288,3289,3292,3293,3296,3297,3300],"warning",{},"IntelliJ-форматування вирішує лише частину проблеми. Воно спрацьовує ",[2979,3286,3287],{},"лише вручну"," (або при збереженні, якщо налаштовано). Воно ",[2979,3290,3291],{},"не перевіряє"," код у CI, ",[2979,3294,3295],{},"не блокує"," commit з неправильним форматуванням, і ",[2979,3298,3299],{},"залежить"," від того, чи у кожного члена команди однакові налаштування IDE. Наступний інструмент вирішує всі ці обмеження.",[3070,3302],{},[2959,3304,3306],{"id":3305},"spotless-форматування-як-частина-білда","Spotless: Форматування як частина білда",[2964,3308,3309,3312,3313,3316,3317,3320,3321,3324,3325,3328,3329,3331],{},[2979,3310,3311],{},"Spotless"," — це Gradle/Maven-плагін, що запускає форматування коду як частину процесу білду. Його ключова властивість: він не лише вміє ",[2979,3314,3315],{},"виправити"," форматування (",[2968,3318,3319],{},"apply","), а й ",[2979,3322,3323],{},"перевірити"," відповідність і провалити білд при невідповідності (",[2968,3326,3327],{},"check","). Саме ",[2968,3330,3327],{},"-режим використовується у CI.",[3084,3333,3335],{"id":3334},"як-працює-spotless","Як працює Spotless",[3337,3338,3339],"mermaid",{},[3092,3340,3343],{"className":3341,"code":3342,"language":3337,"meta":3100,"style":3100},"language-mermaid shiki shiki-themes light-plus dark-plus dark-plus","graph TD\n    SRC[\"*.java\u003Cbr/>вихідний код\"]\n    FMT[\"Spotless Formatter\u003Cbr/>google-java-format 1.35.0\"]\n    CMP{\"Відрізняється\u003Cbr/>від оригіналу?\"}\n    APPLY[\"spotlessApply\u003Cbr/>Перезаписує файл ✅\"]\n    FAIL[\"spotlessCheck\u003Cbr/>❌ Помилка білда\"]\n    OK[\"OK — код відформатований\"]\n\n    SRC --> FMT --> CMP\n    CMP -- \"так + apply\" --> APPLY\n    CMP -- \"так + check\" --> FAIL\n    CMP -- \"ні\" --> OK\n\n    style FAIL fill:#ef4444,stroke:#b91c1c,color:#ffffff\n    style APPLY fill:#22c55e,stroke:#15803d,color:#ffffff\n    style OK fill:#22c55e,stroke:#15803d,color:#ffffff\n",[2968,3344,3345,3353,3359,3365,3371,3377,3383,3389,3396,3402,3408,3414,3420,3425,3431,3437],{"__ignoreMap":3100},[3346,3347,3350],"span",{"class":3348,"line":3349},"line",1,[3346,3351,3352],{},"graph TD\n",[3346,3354,3356],{"class":3348,"line":3355},2,[3346,3357,3358],{},"    SRC[\"*.java\u003Cbr/>вихідний код\"]\n",[3346,3360,3362],{"class":3348,"line":3361},3,[3346,3363,3364],{},"    FMT[\"Spotless Formatter\u003Cbr/>google-java-format 1.35.0\"]\n",[3346,3366,3368],{"class":3348,"line":3367},4,[3346,3369,3370],{},"    CMP{\"Відрізняється\u003Cbr/>від оригіналу?\"}\n",[3346,3372,3374],{"class":3348,"line":3373},5,[3346,3375,3376],{},"    APPLY[\"spotlessApply\u003Cbr/>Перезаписує файл ✅\"]\n",[3346,3378,3380],{"class":3348,"line":3379},6,[3346,3381,3382],{},"    FAIL[\"spotlessCheck\u003Cbr/>❌ Помилка білда\"]\n",[3346,3384,3386],{"class":3348,"line":3385},7,[3346,3387,3388],{},"    OK[\"OK — код відформатований\"]\n",[3346,3390,3392],{"class":3348,"line":3391},8,[3346,3393,3395],{"emptyLinePlaceholder":3394},true,"\n",[3346,3397,3399],{"class":3348,"line":3398},9,[3346,3400,3401],{},"    SRC --> FMT --> CMP\n",[3346,3403,3405],{"class":3348,"line":3404},10,[3346,3406,3407],{},"    CMP -- \"так + apply\" --> APPLY\n",[3346,3409,3411],{"class":3348,"line":3410},11,[3346,3412,3413],{},"    CMP -- \"так + check\" --> FAIL\n",[3346,3415,3417],{"class":3348,"line":3416},12,[3346,3418,3419],{},"    CMP -- \"ні\" --> OK\n",[3346,3421,3423],{"class":3348,"line":3422},13,[3346,3424,3395],{"emptyLinePlaceholder":3394},[3346,3426,3428],{"class":3348,"line":3427},14,[3346,3429,3430],{},"    style FAIL fill:#ef4444,stroke:#b91c1c,color:#ffffff\n",[3346,3432,3434],{"class":3348,"line":3433},15,[3346,3435,3436],{},"    style APPLY fill:#22c55e,stroke:#15803d,color:#ffffff\n",[3346,3438,3440],{"class":3348,"line":3439},16,[3346,3441,3442],{},"    style OK fill:#22c55e,stroke:#15803d,color:#ffffff\n",[2964,3444,3445,3446,3449],{},"Spotless виступає ",[2979,3447,3448],{},"детермінованим арбітром",": для одного і того ж вихідного файлу він завжди дасть однаковий результат, незалежно від ОС, версії JDK чи IDE розробника. Саме це робить його надійнішим за ручне форматування.",[3070,3451],{},[3084,3453,3455],{"id":3454},"gradle-повна-конфігурація","Gradle — повна конфігурація",[3457,3458,3459,3813],"tabs",{},[3460,3461,3463],"tabs-item",{"label":3462},"Groovy DSL (build.gradle)",[3092,3464,3468],{"className":3465,"code":3466,"language":3467,"meta":3100,"style":3100},"language-groovy shiki shiki-themes light-plus dark-plus dark-plus","// build.gradle\nplugins {\n    id 'java'\n    id 'com.diffplug.spotless' version '8.4.0'\n}\n\nspotless {\n    // ratchetFrom 'origin/main'\n    // Якщо є legacy-код з поганим форматуванням — увімкніть цю опцію.\n    // Тоді Spotless перевірятиме лише файли, змінені відносно main-гілки.\n    // Дозволяє поступово вводити форматування у великих проєктах.\n\n    java {\n        // Цільові файли: всі *.java у src/\n        // За замовчуванням Spotless знаходить їх автоматично.\n        target 'src/**/*.java'\n\n        // ── Ядро форматування ─────────────────────────────────────────\n        // Google Java Format 1.35.0\n        // Дві стратегії відступу:\n        //   .style('GOOGLE') → 2 пробіли (за замовчуванням, рекомендовано)\n        //   .aosp()          → 4 пробіли (Android Open Source Project style)\n        googleJavaFormat('1.35.0')\n            .style('GOOGLE')\n            .reflowLongStrings()   // переносити довгі рядкові літерали\n            .formatJavadoc(true)   // форматувати Javadoc-коментарі\n            .reorderImports(false) // imports-порядок контролюємо через importOrder\n\n        // ── Imports ────────────────────────────────────────────────────\n        // Групи: java/javax → org → com → все інше → static imports\n        // '' означає \"всі не вказані явно\"\n        // '\\\\#' — префікс для static imports (import static ...)\n        importOrder('java', 'javax', 'org', 'com', '', '\\\\#')\n\n        // Автоматично видалити невикористані imports\n        removeUnusedImports()\n\n        // Заборонити wildcard imports (import java.util.*)\n        // Розкоментуйте за потреби:\n        // forbidWildcardImports()\n\n        // ── Type annotations ────────────────────────────────────────────\n        // Виправляє форматування @NonNull, @Nullable та інших анотацій типів\n        formatAnnotations()\n\n        // ── Базові правила ──────────────────────────────────────────────\n        trimTrailingWhitespace()  // видалити пробіли в кінці рядків\n        endWithNewline()          // файл має завершуватися переносом рядка\n    }\n\n    // Форматування misc-файлів (не Java)\n    format 'misc', {\n        target '*.gradle', '*.md', '.gitignore', '.gitattributes'\n        trimTrailingWhitespace()\n        endWithNewline()\n    }\n}\n\n// Автоматично запускати spotlessCheck при ./gradlew check або ./gradlew build\ntasks.named('check').configure {\n    dependsOn 'spotlessCheck'\n}\n","groovy",[2968,3469,3470,3475,3480,3485,3490,3495,3499,3504,3509,3514,3519,3524,3528,3533,3538,3543,3548,3553,3559,3565,3571,3577,3583,3589,3595,3601,3607,3613,3618,3624,3630,3636,3642,3648,3653,3659,3665,3670,3676,3682,3688,3693,3699,3705,3711,3716,3722,3728,3734,3740,3745,3751,3757,3763,3769,3775,3780,3785,3790,3796,3802,3808],{"__ignoreMap":3100},[3346,3471,3472],{"class":3348,"line":3349},[3346,3473,3474],{},"// build.gradle\n",[3346,3476,3477],{"class":3348,"line":3355},[3346,3478,3479],{},"plugins {\n",[3346,3481,3482],{"class":3348,"line":3361},[3346,3483,3484],{},"    id 'java'\n",[3346,3486,3487],{"class":3348,"line":3367},[3346,3488,3489],{},"    id 'com.diffplug.spotless' version '8.4.0'\n",[3346,3491,3492],{"class":3348,"line":3373},[3346,3493,3494],{},"}\n",[3346,3496,3497],{"class":3348,"line":3379},[3346,3498,3395],{"emptyLinePlaceholder":3394},[3346,3500,3501],{"class":3348,"line":3385},[3346,3502,3503],{},"spotless {\n",[3346,3505,3506],{"class":3348,"line":3391},[3346,3507,3508],{},"    // ratchetFrom 'origin/main'\n",[3346,3510,3511],{"class":3348,"line":3398},[3346,3512,3513],{},"    // Якщо є legacy-код з поганим форматуванням — увімкніть цю опцію.\n",[3346,3515,3516],{"class":3348,"line":3404},[3346,3517,3518],{},"    // Тоді Spotless перевірятиме лише файли, змінені відносно main-гілки.\n",[3346,3520,3521],{"class":3348,"line":3410},[3346,3522,3523],{},"    // Дозволяє поступово вводити форматування у великих проєктах.\n",[3346,3525,3526],{"class":3348,"line":3416},[3346,3527,3395],{"emptyLinePlaceholder":3394},[3346,3529,3530],{"class":3348,"line":3422},[3346,3531,3532],{},"    java {\n",[3346,3534,3535],{"class":3348,"line":3427},[3346,3536,3537],{},"        // Цільові файли: всі *.java у src/\n",[3346,3539,3540],{"class":3348,"line":3433},[3346,3541,3542],{},"        // За замовчуванням Spotless знаходить їх автоматично.\n",[3346,3544,3545],{"class":3348,"line":3439},[3346,3546,3547],{},"        target 'src/**/*.java'\n",[3346,3549,3551],{"class":3348,"line":3550},17,[3346,3552,3395],{"emptyLinePlaceholder":3394},[3346,3554,3556],{"class":3348,"line":3555},18,[3346,3557,3558],{},"        // ── Ядро форматування ─────────────────────────────────────────\n",[3346,3560,3562],{"class":3348,"line":3561},19,[3346,3563,3564],{},"        // Google Java Format 1.35.0\n",[3346,3566,3568],{"class":3348,"line":3567},20,[3346,3569,3570],{},"        // Дві стратегії відступу:\n",[3346,3572,3574],{"class":3348,"line":3573},21,[3346,3575,3576],{},"        //   .style('GOOGLE') → 2 пробіли (за замовчуванням, рекомендовано)\n",[3346,3578,3580],{"class":3348,"line":3579},22,[3346,3581,3582],{},"        //   .aosp()          → 4 пробіли (Android Open Source Project style)\n",[3346,3584,3586],{"class":3348,"line":3585},23,[3346,3587,3588],{},"        googleJavaFormat('1.35.0')\n",[3346,3590,3592],{"class":3348,"line":3591},24,[3346,3593,3594],{},"            .style('GOOGLE')\n",[3346,3596,3598],{"class":3348,"line":3597},25,[3346,3599,3600],{},"            .reflowLongStrings()   // переносити довгі рядкові літерали\n",[3346,3602,3604],{"class":3348,"line":3603},26,[3346,3605,3606],{},"            .formatJavadoc(true)   // форматувати Javadoc-коментарі\n",[3346,3608,3610],{"class":3348,"line":3609},27,[3346,3611,3612],{},"            .reorderImports(false) // imports-порядок контролюємо через importOrder\n",[3346,3614,3616],{"class":3348,"line":3615},28,[3346,3617,3395],{"emptyLinePlaceholder":3394},[3346,3619,3621],{"class":3348,"line":3620},29,[3346,3622,3623],{},"        // ── Imports ────────────────────────────────────────────────────\n",[3346,3625,3627],{"class":3348,"line":3626},30,[3346,3628,3629],{},"        // Групи: java/javax → org → com → все інше → static imports\n",[3346,3631,3633],{"class":3348,"line":3632},31,[3346,3634,3635],{},"        // '' означає \"всі не вказані явно\"\n",[3346,3637,3639],{"class":3348,"line":3638},32,[3346,3640,3641],{},"        // '\\\\#' — префікс для static imports (import static ...)\n",[3346,3643,3645],{"class":3348,"line":3644},33,[3346,3646,3647],{},"        importOrder('java', 'javax', 'org', 'com', '', '\\\\#')\n",[3346,3649,3651],{"class":3348,"line":3650},34,[3346,3652,3395],{"emptyLinePlaceholder":3394},[3346,3654,3656],{"class":3348,"line":3655},35,[3346,3657,3658],{},"        // Автоматично видалити невикористані imports\n",[3346,3660,3662],{"class":3348,"line":3661},36,[3346,3663,3664],{},"        removeUnusedImports()\n",[3346,3666,3668],{"class":3348,"line":3667},37,[3346,3669,3395],{"emptyLinePlaceholder":3394},[3346,3671,3673],{"class":3348,"line":3672},38,[3346,3674,3675],{},"        // Заборонити wildcard imports (import java.util.*)\n",[3346,3677,3679],{"class":3348,"line":3678},39,[3346,3680,3681],{},"        // Розкоментуйте за потреби:\n",[3346,3683,3685],{"class":3348,"line":3684},40,[3346,3686,3687],{},"        // forbidWildcardImports()\n",[3346,3689,3691],{"class":3348,"line":3690},41,[3346,3692,3395],{"emptyLinePlaceholder":3394},[3346,3694,3696],{"class":3348,"line":3695},42,[3346,3697,3698],{},"        // ── Type annotations ────────────────────────────────────────────\n",[3346,3700,3702],{"class":3348,"line":3701},43,[3346,3703,3704],{},"        // Виправляє форматування @NonNull, @Nullable та інших анотацій типів\n",[3346,3706,3708],{"class":3348,"line":3707},44,[3346,3709,3710],{},"        formatAnnotations()\n",[3346,3712,3714],{"class":3348,"line":3713},45,[3346,3715,3395],{"emptyLinePlaceholder":3394},[3346,3717,3719],{"class":3348,"line":3718},46,[3346,3720,3721],{},"        // ── Базові правила ──────────────────────────────────────────────\n",[3346,3723,3725],{"class":3348,"line":3724},47,[3346,3726,3727],{},"        trimTrailingWhitespace()  // видалити пробіли в кінці рядків\n",[3346,3729,3731],{"class":3348,"line":3730},48,[3346,3732,3733],{},"        endWithNewline()          // файл має завершуватися переносом рядка\n",[3346,3735,3737],{"class":3348,"line":3736},49,[3346,3738,3739],{},"    }\n",[3346,3741,3743],{"class":3348,"line":3742},50,[3346,3744,3395],{"emptyLinePlaceholder":3394},[3346,3746,3748],{"class":3348,"line":3747},51,[3346,3749,3750],{},"    // Форматування misc-файлів (не Java)\n",[3346,3752,3754],{"class":3348,"line":3753},52,[3346,3755,3756],{},"    format 'misc', {\n",[3346,3758,3760],{"class":3348,"line":3759},53,[3346,3761,3762],{},"        target '*.gradle', '*.md', '.gitignore', '.gitattributes'\n",[3346,3764,3766],{"class":3348,"line":3765},54,[3346,3767,3768],{},"        trimTrailingWhitespace()\n",[3346,3770,3772],{"class":3348,"line":3771},55,[3346,3773,3774],{},"        endWithNewline()\n",[3346,3776,3778],{"class":3348,"line":3777},56,[3346,3779,3739],{},[3346,3781,3783],{"class":3348,"line":3782},57,[3346,3784,3494],{},[3346,3786,3788],{"class":3348,"line":3787},58,[3346,3789,3395],{"emptyLinePlaceholder":3394},[3346,3791,3793],{"class":3348,"line":3792},59,[3346,3794,3795],{},"// Автоматично запускати spotlessCheck при ./gradlew check або ./gradlew build\n",[3346,3797,3799],{"class":3348,"line":3798},60,[3346,3800,3801],{},"tasks.named('check').configure {\n",[3346,3803,3805],{"class":3348,"line":3804},61,[3346,3806,3807],{},"    dependsOn 'spotlessCheck'\n",[3346,3809,3811],{"class":3348,"line":3810},62,[3346,3812,3494],{},[3460,3814,3816],{"label":3815},"Kotlin DSL (build.gradle.kts)",[3092,3817,3821],{"className":3818,"code":3819,"language":3820,"meta":3100,"style":3100},"language-kotlin shiki shiki-themes light-plus dark-plus dark-plus","// build.gradle.kts\nplugins {\n    java\n    id(\"com.diffplug.spotless\") version \"8.4.0\"\n}\n\nspotless {\n    // ratchetFrom(\"origin/main\") // для legacy-проєктів\n\n    java {\n        target(\"src/**/*.java\")\n\n        googleJavaFormat(\"1.35.0\")\n            .style(\"GOOGLE\")\n            .reflowLongStrings()\n            .formatJavadoc(true)\n            .reorderImports(false)\n\n        importOrder(\"java\", \"javax\", \"org\", \"com\", \"\", \"\\\\#\")\n        removeUnusedImports()\n        formatAnnotations()\n        trimTrailingWhitespace()\n        endWithNewline()\n    }\n\n    format(\"misc\") {\n        target(\"*.gradle.kts\", \"*.md\", \".gitignore\", \".gitattributes\")\n        trimTrailingWhitespace()\n        endWithNewline()\n    }\n}\n\ntasks.named(\"check\") {\n    dependsOn(\"spotlessCheck\")\n}\n","kotlin",[2968,3822,3823,3828,3832,3837,3842,3846,3850,3854,3862,3866,3870,3875,3879,3884,3889,3894,3899,3904,3908,3913,3917,3921,3925,3929,3933,3937,3942,3947,3951,3955,3959,3963,3967,3972,3977],{"__ignoreMap":3100},[3346,3824,3825],{"class":3348,"line":3349},[3346,3826,3827],{},"// build.gradle.kts\n",[3346,3829,3830],{"class":3348,"line":3355},[3346,3831,3479],{},[3346,3833,3834],{"class":3348,"line":3361},[3346,3835,3836],{},"    java\n",[3346,3838,3839],{"class":3348,"line":3367},[3346,3840,3841],{},"    id(\"com.diffplug.spotless\") version \"8.4.0\"\n",[3346,3843,3844],{"class":3348,"line":3373},[3346,3845,3494],{},[3346,3847,3848],{"class":3348,"line":3379},[3346,3849,3395],{"emptyLinePlaceholder":3394},[3346,3851,3852],{"class":3348,"line":3385},[3346,3853,3503],{},[3346,3855,3856,3859],{"class":3348,"line":3391},[3346,3857,3858],{},"    // ratchetFrom(\"origin/main\")",[3346,3860,3861],{}," // для legacy-проєктів\n",[3346,3863,3864],{"class":3348,"line":3398},[3346,3865,3395],{"emptyLinePlaceholder":3394},[3346,3867,3868],{"class":3348,"line":3404},[3346,3869,3532],{},[3346,3871,3872],{"class":3348,"line":3410},[3346,3873,3874],{},"        target(\"src/**/*.java\")\n",[3346,3876,3877],{"class":3348,"line":3416},[3346,3878,3395],{"emptyLinePlaceholder":3394},[3346,3880,3881],{"class":3348,"line":3422},[3346,3882,3883],{},"        googleJavaFormat(\"1.35.0\")\n",[3346,3885,3886],{"class":3348,"line":3427},[3346,3887,3888],{},"            .style(\"GOOGLE\")\n",[3346,3890,3891],{"class":3348,"line":3433},[3346,3892,3893],{},"            .reflowLongStrings()\n",[3346,3895,3896],{"class":3348,"line":3439},[3346,3897,3898],{},"            .formatJavadoc(true)\n",[3346,3900,3901],{"class":3348,"line":3550},[3346,3902,3903],{},"            .reorderImports(false)\n",[3346,3905,3906],{"class":3348,"line":3555},[3346,3907,3395],{"emptyLinePlaceholder":3394},[3346,3909,3910],{"class":3348,"line":3561},[3346,3911,3912],{},"        importOrder(\"java\", \"javax\", \"org\", \"com\", \"\", \"\\\\#\")\n",[3346,3914,3915],{"class":3348,"line":3567},[3346,3916,3664],{},[3346,3918,3919],{"class":3348,"line":3573},[3346,3920,3710],{},[3346,3922,3923],{"class":3348,"line":3579},[3346,3924,3768],{},[3346,3926,3927],{"class":3348,"line":3585},[3346,3928,3774],{},[3346,3930,3931],{"class":3348,"line":3591},[3346,3932,3739],{},[3346,3934,3935],{"class":3348,"line":3597},[3346,3936,3395],{"emptyLinePlaceholder":3394},[3346,3938,3939],{"class":3348,"line":3603},[3346,3940,3941],{},"    format(\"misc\") {\n",[3346,3943,3944],{"class":3348,"line":3609},[3346,3945,3946],{},"        target(\"*.gradle.kts\", \"*.md\", \".gitignore\", \".gitattributes\")\n",[3346,3948,3949],{"class":3348,"line":3615},[3346,3950,3768],{},[3346,3952,3953],{"class":3348,"line":3620},[3346,3954,3774],{},[3346,3956,3957],{"class":3348,"line":3626},[3346,3958,3739],{},[3346,3960,3961],{"class":3348,"line":3632},[3346,3962,3494],{},[3346,3964,3965],{"class":3348,"line":3638},[3346,3966,3395],{"emptyLinePlaceholder":3394},[3346,3968,3969],{"class":3348,"line":3644},[3346,3970,3971],{},"tasks.named(\"check\") {\n",[3346,3973,3974],{"class":3348,"line":3650},[3346,3975,3976],{},"    dependsOn(\"spotlessCheck\")\n",[3346,3978,3979],{"class":3348,"line":3655},[3346,3980,3494],{},[2964,3982,3983],{},[2979,3984,3985],{},"Команди Gradle:",[3092,3987,3991],{"className":3988,"code":3989,"language":3990,"meta":3100,"style":3100},"language-bash shiki shiki-themes light-plus dark-plus dark-plus","# Перевірити відповідність (не змінює файли — для CI)\n./gradlew spotlessCheck\n\n# Виправити форматування in-place (для локальної розробки)\n./gradlew spotlessApply\n\n# Перевірити тільки Java-файли\n./gradlew spotlessJavaCheck\n\n# Виправити тільки Java-файли\n./gradlew spotlessJavaApply\n","bash",[2968,3992,3993,3999,4009,4013,4018,4025,4029,4034,4041,4045,4050],{"__ignoreMap":3100},[3346,3994,3995],{"class":3348,"line":3349},[3346,3996,3998],{"class":3997},"spJ8K","# Перевірити відповідність (не змінює файли — для CI)\n",[3346,4000,4001,4005],{"class":3348,"line":3355},[3346,4002,4004],{"class":4003},"s8Opu","./gradlew",[3346,4006,4008],{"class":4007},"sbdoH"," spotlessCheck\n",[3346,4010,4011],{"class":3348,"line":3361},[3346,4012,3395],{"emptyLinePlaceholder":3394},[3346,4014,4015],{"class":3348,"line":3367},[3346,4016,4017],{"class":3997},"# Виправити форматування in-place (для локальної розробки)\n",[3346,4019,4020,4022],{"class":3348,"line":3373},[3346,4021,4004],{"class":4003},[3346,4023,4024],{"class":4007}," spotlessApply\n",[3346,4026,4027],{"class":3348,"line":3379},[3346,4028,3395],{"emptyLinePlaceholder":3394},[3346,4030,4031],{"class":3348,"line":3385},[3346,4032,4033],{"class":3997},"# Перевірити тільки Java-файли\n",[3346,4035,4036,4038],{"class":3348,"line":3391},[3346,4037,4004],{"class":4003},[3346,4039,4040],{"class":4007}," spotlessJavaCheck\n",[3346,4042,4043],{"class":3348,"line":3398},[3346,4044,3395],{"emptyLinePlaceholder":3394},[3346,4046,4047],{"class":3348,"line":3404},[3346,4048,4049],{"class":3997},"# Виправити тільки Java-файли\n",[3346,4051,4052,4054],{"class":3348,"line":3410},[3346,4053,4004],{"class":4003},[3346,4055,4056],{"class":4007}," spotlessJavaApply\n",[3070,4058],{},[3084,4060,4062],{"id":4061},"maven-повна-конфігурація","Maven — повна конфігурація",[3092,4064,4068],{"className":4065,"code":4066,"language":4067,"meta":3100,"style":3100},"language-xml shiki shiki-themes light-plus dark-plus dark-plus","\u003C!-- pom.xml → \u003Cbuild> → \u003Cplugins> -->\n\u003Cplugin>\n    \u003CgroupId>com.diffplug.spotless\u003C/groupId>\n    \u003CartifactId>spotless-maven-plugin\u003C/artifactId>\n    \u003Cversion>3.4.0\u003C/version>\n\n    \u003Cconfiguration>\n        \u003C!-- ratchetFrom: форматувати лише файли, змінені відносно origin/main -->\n        \u003C!-- \u003CratchetFrom>origin/main\u003C/ratchetFrom> -->\n\n        \u003Cjava>\n            \u003C!-- Цільові файли -->\n            \u003Cincludes>\n                \u003Cinclude>src/main/java/**/*.java\u003C/include>\n                \u003Cinclude>src/test/java/**/*.java\u003C/include>\n            \u003C/includes>\n\n            \u003C!-- ── Google Java Format 1.35.0 ──────────────────────── -->\n            \u003C!-- style: GOOGLE (2-пробільний відступ, рядки до 100 символів) -->\n            \u003C!-- style: AOSP   (4-пробільний відступ, для Android-проєктів)  -->\n            \u003CgoogleJavaFormat>\n                \u003Cversion>1.35.0\u003C/version>\n                \u003Cstyle>GOOGLE\u003C/style>\n                \u003CreflowLongStrings>true\u003C/reflowLongStrings>\n                \u003CformatJavadoc>true\u003C/formatJavadoc>\n                \u003CreorderImports>false\u003C/reorderImports>\n            \u003C/googleJavaFormat>\n\n            \u003C!-- ── Imports ────────────────────────────────────────── -->\n            \u003C!-- '|' об'єднує групи без порожнього рядка між ними    -->\n            \u003C!-- '' — всі imports, не вказані явно                    -->\n            \u003C!-- \\# — static imports                                  -->\n            \u003CimportOrder>\n                \u003Corder>java|javax,org,com,,\\#\u003C/order>\n            \u003C/importOrder>\n\n            \u003C!-- Видалити невикористані imports -->\n            \u003CremoveUnusedImports/>\n\n            \u003C!-- Заборонити wildcard imports -->\n            \u003C!-- \u003CforbidWildcardImports/> -->\n\n            \u003C!-- Виправити type annotations -->\n            \u003CformatAnnotations/>\n\n            \u003CtrimTrailingWhitespace/>\n            \u003CendWithNewline/>\n        \u003C/java>\n\n        \u003C!-- Форматування misc-файлів -->\n        \u003Cformats>\n            \u003Cformat>\n                \u003Cincludes>\n                    \u003Cinclude>pom.xml\u003C/include>\n                    \u003Cinclude>*.md\u003C/include>\n                    \u003Cinclude>.gitignore\u003C/include>\n                    \u003Cinclude>.gitattributes\u003C/include>\n                \u003C/includes>\n                \u003CtrimTrailingWhitespace/>\n                \u003CendWithNewline/>\n            \u003C/format>\n        \u003C/formats>\n    \u003C/configuration>\n\n    \u003C!-- Автоматично при mvn verify запускати spotless:check -->\n    \u003Cexecutions>\n        \u003Cexecution>\n            \u003Cid>spotless-check\u003C/id>\n            \u003Cphase>verify\u003C/phase>\n            \u003Cgoals>\n                \u003Cgoal>check\u003C/goal>\n            \u003C/goals>\n        \u003C/execution>\n    \u003C/executions>\n\u003C/plugin>\n","xml",[2968,4069,4070,4075,4088,4110,4128,4145,4149,4158,4163,4168,4172,4182,4187,4197,4216,4233,4242,4246,4251,4256,4261,4270,4286,4304,4322,4339,4357,4365,4369,4374,4379,4384,4389,4398,4416,4424,4428,4433,4443,4447,4452,4457,4461,4466,4475,4479,4488,4497,4506,4510,4515,4524,4533,4541,4559,4576,4593,4610,4619,4627,4635,4643,4651,4661,4666,4672,4682,4692,4711,4730,4740,4758,4767,4776,4785],{"__ignoreMap":3100},[3346,4071,4072],{"class":3348,"line":3349},[3346,4073,4074],{"class":3997},"\u003C!-- pom.xml → \u003Cbuild> → \u003Cplugins> -->\n",[3346,4076,4077,4081,4085],{"class":3348,"line":3355},[3346,4078,4080],{"class":4079},"s0P7L","\u003C",[3346,4082,4084],{"class":4083},"sKtos","plugin",[3346,4086,4087],{"class":4079},">\n",[3346,4089,4090,4093,4096,4099,4103,4106,4108],{"class":3348,"line":3361},[3346,4091,4092],{"class":4079},"    \u003C",[3346,4094,4095],{"class":4083},"groupId",[3346,4097,4098],{"class":4079},">",[3346,4100,4102],{"class":4101},"sHH4Y","com.diffplug.spotless",[3346,4104,4105],{"class":4079},"\u003C/",[3346,4107,4095],{"class":4083},[3346,4109,4087],{"class":4079},[3346,4111,4112,4114,4117,4119,4122,4124,4126],{"class":3348,"line":3367},[3346,4113,4092],{"class":4079},[3346,4115,4116],{"class":4083},"artifactId",[3346,4118,4098],{"class":4079},[3346,4120,4121],{"class":4101},"spotless-maven-plugin",[3346,4123,4105],{"class":4079},[3346,4125,4116],{"class":4083},[3346,4127,4087],{"class":4079},[3346,4129,4130,4132,4135,4137,4139,4141,4143],{"class":3348,"line":3373},[3346,4131,4092],{"class":4079},[3346,4133,4134],{"class":4083},"version",[3346,4136,4098],{"class":4079},[3346,4138,3055],{"class":4101},[3346,4140,4105],{"class":4079},[3346,4142,4134],{"class":4083},[3346,4144,4087],{"class":4079},[3346,4146,4147],{"class":3348,"line":3379},[3346,4148,3395],{"emptyLinePlaceholder":3394},[3346,4150,4151,4153,4156],{"class":3348,"line":3385},[3346,4152,4092],{"class":4079},[3346,4154,4155],{"class":4083},"configuration",[3346,4157,4087],{"class":4079},[3346,4159,4160],{"class":3348,"line":3391},[3346,4161,4162],{"class":3997},"        \u003C!-- ratchetFrom: форматувати лише файли, змінені відносно origin/main -->\n",[3346,4164,4165],{"class":3348,"line":3398},[3346,4166,4167],{"class":3997},"        \u003C!-- \u003CratchetFrom>origin/main\u003C/ratchetFrom> -->\n",[3346,4169,4170],{"class":3348,"line":3404},[3346,4171,3395],{"emptyLinePlaceholder":3394},[3346,4173,4174,4177,4180],{"class":3348,"line":3410},[3346,4175,4176],{"class":4079},"        \u003C",[3346,4178,4179],{"class":4083},"java",[3346,4181,4087],{"class":4079},[3346,4183,4184],{"class":3348,"line":3416},[3346,4185,4186],{"class":3997},"            \u003C!-- Цільові файли -->\n",[3346,4188,4189,4192,4195],{"class":3348,"line":3422},[3346,4190,4191],{"class":4079},"            \u003C",[3346,4193,4194],{"class":4083},"includes",[3346,4196,4087],{"class":4079},[3346,4198,4199,4202,4205,4207,4210,4212,4214],{"class":3348,"line":3427},[3346,4200,4201],{"class":4079},"                \u003C",[3346,4203,4204],{"class":4083},"include",[3346,4206,4098],{"class":4079},[3346,4208,4209],{"class":4101},"src/main/java/**/*.java",[3346,4211,4105],{"class":4079},[3346,4213,4204],{"class":4083},[3346,4215,4087],{"class":4079},[3346,4217,4218,4220,4222,4224,4227,4229,4231],{"class":3348,"line":3433},[3346,4219,4201],{"class":4079},[3346,4221,4204],{"class":4083},[3346,4223,4098],{"class":4079},[3346,4225,4226],{"class":4101},"src/test/java/**/*.java",[3346,4228,4105],{"class":4079},[3346,4230,4204],{"class":4083},[3346,4232,4087],{"class":4079},[3346,4234,4235,4238,4240],{"class":3348,"line":3439},[3346,4236,4237],{"class":4079},"            \u003C/",[3346,4239,4194],{"class":4083},[3346,4241,4087],{"class":4079},[3346,4243,4244],{"class":3348,"line":3550},[3346,4245,3395],{"emptyLinePlaceholder":3394},[3346,4247,4248],{"class":3348,"line":3555},[3346,4249,4250],{"class":3997},"            \u003C!-- ── Google Java Format 1.35.0 ──────────────────────── -->\n",[3346,4252,4253],{"class":3348,"line":3561},[3346,4254,4255],{"class":3997},"            \u003C!-- style: GOOGLE (2-пробільний відступ, рядки до 100 символів) -->\n",[3346,4257,4258],{"class":3348,"line":3567},[3346,4259,4260],{"class":3997},"            \u003C!-- style: AOSP   (4-пробільний відступ, для Android-проєктів)  -->\n",[3346,4262,4263,4265,4268],{"class":3348,"line":3573},[3346,4264,4191],{"class":4079},[3346,4266,4267],{"class":4083},"googleJavaFormat",[3346,4269,4087],{"class":4079},[3346,4271,4272,4274,4276,4278,4280,4282,4284],{"class":3348,"line":3579},[3346,4273,4201],{"class":4079},[3346,4275,4134],{"class":4083},[3346,4277,4098],{"class":4079},[3346,4279,3059],{"class":4101},[3346,4281,4105],{"class":4079},[3346,4283,4134],{"class":4083},[3346,4285,4087],{"class":4079},[3346,4287,4288,4290,4293,4295,4298,4300,4302],{"class":3348,"line":3585},[3346,4289,4201],{"class":4079},[3346,4291,4292],{"class":4083},"style",[3346,4294,4098],{"class":4079},[3346,4296,4297],{"class":4101},"GOOGLE",[3346,4299,4105],{"class":4079},[3346,4301,4292],{"class":4083},[3346,4303,4087],{"class":4079},[3346,4305,4306,4308,4311,4313,4316,4318,4320],{"class":3348,"line":3591},[3346,4307,4201],{"class":4079},[3346,4309,4310],{"class":4083},"reflowLongStrings",[3346,4312,4098],{"class":4079},[3346,4314,4315],{"class":4101},"true",[3346,4317,4105],{"class":4079},[3346,4319,4310],{"class":4083},[3346,4321,4087],{"class":4079},[3346,4323,4324,4326,4329,4331,4333,4335,4337],{"class":3348,"line":3597},[3346,4325,4201],{"class":4079},[3346,4327,4328],{"class":4083},"formatJavadoc",[3346,4330,4098],{"class":4079},[3346,4332,4315],{"class":4101},[3346,4334,4105],{"class":4079},[3346,4336,4328],{"class":4083},[3346,4338,4087],{"class":4079},[3346,4340,4341,4343,4346,4348,4351,4353,4355],{"class":3348,"line":3603},[3346,4342,4201],{"class":4079},[3346,4344,4345],{"class":4083},"reorderImports",[3346,4347,4098],{"class":4079},[3346,4349,4350],{"class":4101},"false",[3346,4352,4105],{"class":4079},[3346,4354,4345],{"class":4083},[3346,4356,4087],{"class":4079},[3346,4358,4359,4361,4363],{"class":3348,"line":3609},[3346,4360,4237],{"class":4079},[3346,4362,4267],{"class":4083},[3346,4364,4087],{"class":4079},[3346,4366,4367],{"class":3348,"line":3615},[3346,4368,3395],{"emptyLinePlaceholder":3394},[3346,4370,4371],{"class":3348,"line":3620},[3346,4372,4373],{"class":3997},"            \u003C!-- ── Imports ────────────────────────────────────────── -->\n",[3346,4375,4376],{"class":3348,"line":3626},[3346,4377,4378],{"class":3997},"            \u003C!-- '|' об'єднує групи без порожнього рядка між ними    -->\n",[3346,4380,4381],{"class":3348,"line":3632},[3346,4382,4383],{"class":3997},"            \u003C!-- '' — всі imports, не вказані явно                    -->\n",[3346,4385,4386],{"class":3348,"line":3638},[3346,4387,4388],{"class":3997},"            \u003C!-- \\# — static imports                                  -->\n",[3346,4390,4391,4393,4396],{"class":3348,"line":3644},[3346,4392,4191],{"class":4079},[3346,4394,4395],{"class":4083},"importOrder",[3346,4397,4087],{"class":4079},[3346,4399,4400,4402,4405,4407,4410,4412,4414],{"class":3348,"line":3650},[3346,4401,4201],{"class":4079},[3346,4403,4404],{"class":4083},"order",[3346,4406,4098],{"class":4079},[3346,4408,4409],{"class":4101},"java|javax,org,com,,\\#",[3346,4411,4105],{"class":4079},[3346,4413,4404],{"class":4083},[3346,4415,4087],{"class":4079},[3346,4417,4418,4420,4422],{"class":3348,"line":3655},[3346,4419,4237],{"class":4079},[3346,4421,4395],{"class":4083},[3346,4423,4087],{"class":4079},[3346,4425,4426],{"class":3348,"line":3661},[3346,4427,3395],{"emptyLinePlaceholder":3394},[3346,4429,4430],{"class":3348,"line":3667},[3346,4431,4432],{"class":3997},"            \u003C!-- Видалити невикористані imports -->\n",[3346,4434,4435,4437,4440],{"class":3348,"line":3672},[3346,4436,4191],{"class":4079},[3346,4438,4439],{"class":4083},"removeUnusedImports",[3346,4441,4442],{"class":4079},"/>\n",[3346,4444,4445],{"class":3348,"line":3678},[3346,4446,3395],{"emptyLinePlaceholder":3394},[3346,4448,4449],{"class":3348,"line":3684},[3346,4450,4451],{"class":3997},"            \u003C!-- Заборонити wildcard imports -->\n",[3346,4453,4454],{"class":3348,"line":3690},[3346,4455,4456],{"class":3997},"            \u003C!-- \u003CforbidWildcardImports/> -->\n",[3346,4458,4459],{"class":3348,"line":3695},[3346,4460,3395],{"emptyLinePlaceholder":3394},[3346,4462,4463],{"class":3348,"line":3701},[3346,4464,4465],{"class":3997},"            \u003C!-- Виправити type annotations -->\n",[3346,4467,4468,4470,4473],{"class":3348,"line":3707},[3346,4469,4191],{"class":4079},[3346,4471,4472],{"class":4083},"formatAnnotations",[3346,4474,4442],{"class":4079},[3346,4476,4477],{"class":3348,"line":3713},[3346,4478,3395],{"emptyLinePlaceholder":3394},[3346,4480,4481,4483,4486],{"class":3348,"line":3718},[3346,4482,4191],{"class":4079},[3346,4484,4485],{"class":4083},"trimTrailingWhitespace",[3346,4487,4442],{"class":4079},[3346,4489,4490,4492,4495],{"class":3348,"line":3724},[3346,4491,4191],{"class":4079},[3346,4493,4494],{"class":4083},"endWithNewline",[3346,4496,4442],{"class":4079},[3346,4498,4499,4502,4504],{"class":3348,"line":3730},[3346,4500,4501],{"class":4079},"        \u003C/",[3346,4503,4179],{"class":4083},[3346,4505,4087],{"class":4079},[3346,4507,4508],{"class":3348,"line":3736},[3346,4509,3395],{"emptyLinePlaceholder":3394},[3346,4511,4512],{"class":3348,"line":3742},[3346,4513,4514],{"class":3997},"        \u003C!-- Форматування misc-файлів -->\n",[3346,4516,4517,4519,4522],{"class":3348,"line":3747},[3346,4518,4176],{"class":4079},[3346,4520,4521],{"class":4083},"formats",[3346,4523,4087],{"class":4079},[3346,4525,4526,4528,4531],{"class":3348,"line":3753},[3346,4527,4191],{"class":4079},[3346,4529,4530],{"class":4083},"format",[3346,4532,4087],{"class":4079},[3346,4534,4535,4537,4539],{"class":3348,"line":3759},[3346,4536,4201],{"class":4079},[3346,4538,4194],{"class":4083},[3346,4540,4087],{"class":4079},[3346,4542,4543,4546,4548,4550,4553,4555,4557],{"class":3348,"line":3765},[3346,4544,4545],{"class":4079},"                    \u003C",[3346,4547,4204],{"class":4083},[3346,4549,4098],{"class":4079},[3346,4551,4552],{"class":4101},"pom.xml",[3346,4554,4105],{"class":4079},[3346,4556,4204],{"class":4083},[3346,4558,4087],{"class":4079},[3346,4560,4561,4563,4565,4567,4570,4572,4574],{"class":3348,"line":3771},[3346,4562,4545],{"class":4079},[3346,4564,4204],{"class":4083},[3346,4566,4098],{"class":4079},[3346,4568,4569],{"class":4101},"*.md",[3346,4571,4105],{"class":4079},[3346,4573,4204],{"class":4083},[3346,4575,4087],{"class":4079},[3346,4577,4578,4580,4582,4584,4587,4589,4591],{"class":3348,"line":3777},[3346,4579,4545],{"class":4079},[3346,4581,4204],{"class":4083},[3346,4583,4098],{"class":4079},[3346,4585,4586],{"class":4101},".gitignore",[3346,4588,4105],{"class":4079},[3346,4590,4204],{"class":4083},[3346,4592,4087],{"class":4079},[3346,4594,4595,4597,4599,4601,4604,4606,4608],{"class":3348,"line":3782},[3346,4596,4545],{"class":4079},[3346,4598,4204],{"class":4083},[3346,4600,4098],{"class":4079},[3346,4602,4603],{"class":4101},".gitattributes",[3346,4605,4105],{"class":4079},[3346,4607,4204],{"class":4083},[3346,4609,4087],{"class":4079},[3346,4611,4612,4615,4617],{"class":3348,"line":3787},[3346,4613,4614],{"class":4079},"                \u003C/",[3346,4616,4194],{"class":4083},[3346,4618,4087],{"class":4079},[3346,4620,4621,4623,4625],{"class":3348,"line":3792},[3346,4622,4201],{"class":4079},[3346,4624,4485],{"class":4083},[3346,4626,4442],{"class":4079},[3346,4628,4629,4631,4633],{"class":3348,"line":3798},[3346,4630,4201],{"class":4079},[3346,4632,4494],{"class":4083},[3346,4634,4442],{"class":4079},[3346,4636,4637,4639,4641],{"class":3348,"line":3804},[3346,4638,4237],{"class":4079},[3346,4640,4530],{"class":4083},[3346,4642,4087],{"class":4079},[3346,4644,4645,4647,4649],{"class":3348,"line":3810},[3346,4646,4501],{"class":4079},[3346,4648,4521],{"class":4083},[3346,4650,4087],{"class":4079},[3346,4652,4654,4657,4659],{"class":3348,"line":4653},63,[3346,4655,4656],{"class":4079},"    \u003C/",[3346,4658,4155],{"class":4083},[3346,4660,4087],{"class":4079},[3346,4662,4664],{"class":3348,"line":4663},64,[3346,4665,3395],{"emptyLinePlaceholder":3394},[3346,4667,4669],{"class":3348,"line":4668},65,[3346,4670,4671],{"class":3997},"    \u003C!-- Автоматично при mvn verify запускати spotless:check -->\n",[3346,4673,4675,4677,4680],{"class":3348,"line":4674},66,[3346,4676,4092],{"class":4079},[3346,4678,4679],{"class":4083},"executions",[3346,4681,4087],{"class":4079},[3346,4683,4685,4687,4690],{"class":3348,"line":4684},67,[3346,4686,4176],{"class":4079},[3346,4688,4689],{"class":4083},"execution",[3346,4691,4087],{"class":4079},[3346,4693,4695,4697,4700,4702,4705,4707,4709],{"class":3348,"line":4694},68,[3346,4696,4191],{"class":4079},[3346,4698,4699],{"class":4083},"id",[3346,4701,4098],{"class":4079},[3346,4703,4704],{"class":4101},"spotless-check",[3346,4706,4105],{"class":4079},[3346,4708,4699],{"class":4083},[3346,4710,4087],{"class":4079},[3346,4712,4714,4716,4719,4721,4724,4726,4728],{"class":3348,"line":4713},69,[3346,4715,4191],{"class":4079},[3346,4717,4718],{"class":4083},"phase",[3346,4720,4098],{"class":4079},[3346,4722,4723],{"class":4101},"verify",[3346,4725,4105],{"class":4079},[3346,4727,4718],{"class":4083},[3346,4729,4087],{"class":4079},[3346,4731,4733,4735,4738],{"class":3348,"line":4732},70,[3346,4734,4191],{"class":4079},[3346,4736,4737],{"class":4083},"goals",[3346,4739,4087],{"class":4079},[3346,4741,4743,4745,4748,4750,4752,4754,4756],{"class":3348,"line":4742},71,[3346,4744,4201],{"class":4079},[3346,4746,4747],{"class":4083},"goal",[3346,4749,4098],{"class":4079},[3346,4751,3327],{"class":4101},[3346,4753,4105],{"class":4079},[3346,4755,4747],{"class":4083},[3346,4757,4087],{"class":4079},[3346,4759,4761,4763,4765],{"class":3348,"line":4760},72,[3346,4762,4237],{"class":4079},[3346,4764,4737],{"class":4083},[3346,4766,4087],{"class":4079},[3346,4768,4770,4772,4774],{"class":3348,"line":4769},73,[3346,4771,4501],{"class":4079},[3346,4773,4689],{"class":4083},[3346,4775,4087],{"class":4079},[3346,4777,4779,4781,4783],{"class":3348,"line":4778},74,[3346,4780,4656],{"class":4079},[3346,4782,4679],{"class":4083},[3346,4784,4087],{"class":4079},[3346,4786,4788,4790,4792],{"class":3348,"line":4787},75,[3346,4789,4105],{"class":4079},[3346,4791,4084],{"class":4083},[3346,4793,4087],{"class":4079},[2964,4795,4796],{},[2979,4797,4798],{},"Команди Maven:",[3092,4800,4802],{"className":3988,"code":4801,"language":3990,"meta":3100,"style":3100},"# Перевірити (для CI — не змінює файли)\nmvn spotless:check\n\n# Виправити in-place\nmvn spotless:apply\n\n# Звичайний білд із автоматичною перевіркою (через фазу verify)\nmvn verify\n\n# Або одночасно з тестами і перевіркою форматування\nmvn clean verify\n",[2968,4803,4804,4809,4817,4821,4826,4833,4837,4842,4849,4853,4858],{"__ignoreMap":3100},[3346,4805,4806],{"class":3348,"line":3349},[3346,4807,4808],{"class":3997},"# Перевірити (для CI — не змінює файли)\n",[3346,4810,4811,4814],{"class":3348,"line":3355},[3346,4812,4813],{"class":4003},"mvn",[3346,4815,4816],{"class":4007}," spotless:check\n",[3346,4818,4819],{"class":3348,"line":3361},[3346,4820,3395],{"emptyLinePlaceholder":3394},[3346,4822,4823],{"class":3348,"line":3367},[3346,4824,4825],{"class":3997},"# Виправити in-place\n",[3346,4827,4828,4830],{"class":3348,"line":3373},[3346,4829,4813],{"class":4003},[3346,4831,4832],{"class":4007}," spotless:apply\n",[3346,4834,4835],{"class":3348,"line":3379},[3346,4836,3395],{"emptyLinePlaceholder":3394},[3346,4838,4839],{"class":3348,"line":3385},[3346,4840,4841],{"class":3997},"# Звичайний білд із автоматичною перевіркою (через фазу verify)\n",[3346,4843,4844,4846],{"class":3348,"line":3391},[3346,4845,4813],{"class":4003},[3346,4847,4848],{"class":4007}," verify\n",[3346,4850,4851],{"class":3348,"line":3398},[3346,4852,3395],{"emptyLinePlaceholder":3394},[3346,4854,4855],{"class":3348,"line":3404},[3346,4856,4857],{"class":3997},"# Або одночасно з тестами і перевіркою форматування\n",[3346,4859,4860,4862,4865],{"class":3348,"line":3410},[3346,4861,4813],{"class":4003},[3346,4863,4864],{"class":4007}," clean",[3346,4866,4848],{"class":4007},[3070,4868],{},[3084,4870,4872],{"id":4871},"інтеграція-з-ci-github-actions","Інтеграція з CI (GitHub Actions)",[2964,4874,4875],{},"Щоб PR з неформатованим кодом автоматично падав:",[3457,4877,4878,5062],{},[3460,4879,4881],{"label":4880},"Gradle",[3092,4882,4886],{"className":4883,"code":4884,"language":4885,"meta":3100,"style":3100},"language-yaml shiki shiki-themes light-plus dark-plus dark-plus","# .github/workflows/ci.yml\nname: CI\non: [push, pull_request]\n\njobs:\n  build:\n    runs-on: ubuntu-latest\n    steps:\n      - uses: actions/checkout@v4\n      - uses: actions/setup-java@v4\n        with:\n          java-version: '21'\n          distribution: 'temurin'\n\n      - name: Check code formatting (Spotless)\n        run: ./gradlew spotlessCheck\n\n      - name: Build and test\n        run: ./gradlew build\n","yaml",[2968,4887,4888,4893,4905,4926,4930,4938,4945,4955,4962,4975,4986,4993,5003,5013,5017,5028,5038,5042,5053],{"__ignoreMap":3100},[3346,4889,4890],{"class":3348,"line":3349},[3346,4891,4892],{"class":3997},"# .github/workflows/ci.yml\n",[3346,4894,4895,4898,4901],{"class":3348,"line":3355},[3346,4896,4897],{"class":4083},"name",[3346,4899,4900],{"class":4101},": ",[3346,4902,4904],{"class":4903},"su9tN","CI\n",[3346,4906,4907,4911,4914,4917,4920,4923],{"class":3348,"line":3361},[3346,4908,4910],{"class":4909},"su1O8","on",[3346,4912,4913],{"class":4101},": [",[3346,4915,4916],{"class":4903},"push",[3346,4918,4919],{"class":4101},", ",[3346,4921,4922],{"class":4903},"pull_request",[3346,4924,4925],{"class":4101},"]\n",[3346,4927,4928],{"class":3348,"line":3367},[3346,4929,3395],{"emptyLinePlaceholder":3394},[3346,4931,4932,4935],{"class":3348,"line":3373},[3346,4933,4934],{"class":4083},"jobs",[3346,4936,4937],{"class":4101},":\n",[3346,4939,4940,4943],{"class":3348,"line":3379},[3346,4941,4942],{"class":4083},"  build",[3346,4944,4937],{"class":4101},[3346,4946,4947,4950,4952],{"class":3348,"line":3385},[3346,4948,4949],{"class":4083},"    runs-on",[3346,4951,4900],{"class":4101},[3346,4953,4954],{"class":4903},"ubuntu-latest\n",[3346,4956,4957,4960],{"class":3348,"line":3391},[3346,4958,4959],{"class":4083},"    steps",[3346,4961,4937],{"class":4101},[3346,4963,4964,4967,4970,4972],{"class":3348,"line":3398},[3346,4965,4966],{"class":4101},"      - ",[3346,4968,4969],{"class":4083},"uses",[3346,4971,4900],{"class":4101},[3346,4973,4974],{"class":4903},"actions/checkout@v4\n",[3346,4976,4977,4979,4981,4983],{"class":3348,"line":3404},[3346,4978,4966],{"class":4101},[3346,4980,4969],{"class":4083},[3346,4982,4900],{"class":4101},[3346,4984,4985],{"class":4903},"actions/setup-java@v4\n",[3346,4987,4988,4991],{"class":3348,"line":3410},[3346,4989,4990],{"class":4083},"        with",[3346,4992,4937],{"class":4101},[3346,4994,4995,4998,5000],{"class":3348,"line":3416},[3346,4996,4997],{"class":4083},"          java-version",[3346,4999,4900],{"class":4101},[3346,5001,5002],{"class":4903},"'21'\n",[3346,5004,5005,5008,5010],{"class":3348,"line":3422},[3346,5006,5007],{"class":4083},"          distribution",[3346,5009,4900],{"class":4101},[3346,5011,5012],{"class":4903},"'temurin'\n",[3346,5014,5015],{"class":3348,"line":3427},[3346,5016,3395],{"emptyLinePlaceholder":3394},[3346,5018,5019,5021,5023,5025],{"class":3348,"line":3433},[3346,5020,4966],{"class":4101},[3346,5022,4897],{"class":4083},[3346,5024,4900],{"class":4101},[3346,5026,5027],{"class":4903},"Check code formatting (Spotless)\n",[3346,5029,5030,5033,5035],{"class":3348,"line":3439},[3346,5031,5032],{"class":4083},"        run",[3346,5034,4900],{"class":4101},[3346,5036,5037],{"class":4903},"./gradlew spotlessCheck\n",[3346,5039,5040],{"class":3348,"line":3550},[3346,5041,3395],{"emptyLinePlaceholder":3394},[3346,5043,5044,5046,5048,5050],{"class":3348,"line":3555},[3346,5045,4966],{"class":4101},[3346,5047,4897],{"class":4083},[3346,5049,4900],{"class":4101},[3346,5051,5052],{"class":4903},"Build and test\n",[3346,5054,5055,5057,5059],{"class":3348,"line":3561},[3346,5056,5032],{"class":4083},[3346,5058,4900],{"class":4101},[3346,5060,5061],{"class":4903},"./gradlew build\n",[3460,5063,5065],{"label":5064},"Maven",[3092,5066,5068],{"className":4883,"code":5067,"language":4885,"meta":3100,"style":3100},"# .github/workflows/ci.yml\nname: CI\non: [push, pull_request]\n\njobs:\n  build:\n    runs-on: ubuntu-latest\n    steps:\n      - uses: actions/checkout@v4\n      - uses: actions/setup-java@v4\n        with:\n          java-version: '21'\n          distribution: 'temurin'\n\n      - name: Check code formatting (Spotless)\n        run: mvn spotless:check\n\n      - name: Build and test\n        run: mvn verify\n",[2968,5069,5070,5074,5082,5096,5100,5106,5112,5120,5126,5136,5146,5152,5160,5168,5172,5182,5191,5195,5205],{"__ignoreMap":3100},[3346,5071,5072],{"class":3348,"line":3349},[3346,5073,4892],{"class":3997},[3346,5075,5076,5078,5080],{"class":3348,"line":3355},[3346,5077,4897],{"class":4083},[3346,5079,4900],{"class":4101},[3346,5081,4904],{"class":4903},[3346,5083,5084,5086,5088,5090,5092,5094],{"class":3348,"line":3361},[3346,5085,4910],{"class":4909},[3346,5087,4913],{"class":4101},[3346,5089,4916],{"class":4903},[3346,5091,4919],{"class":4101},[3346,5093,4922],{"class":4903},[3346,5095,4925],{"class":4101},[3346,5097,5098],{"class":3348,"line":3367},[3346,5099,3395],{"emptyLinePlaceholder":3394},[3346,5101,5102,5104],{"class":3348,"line":3373},[3346,5103,4934],{"class":4083},[3346,5105,4937],{"class":4101},[3346,5107,5108,5110],{"class":3348,"line":3379},[3346,5109,4942],{"class":4083},[3346,5111,4937],{"class":4101},[3346,5113,5114,5116,5118],{"class":3348,"line":3385},[3346,5115,4949],{"class":4083},[3346,5117,4900],{"class":4101},[3346,5119,4954],{"class":4903},[3346,5121,5122,5124],{"class":3348,"line":3391},[3346,5123,4959],{"class":4083},[3346,5125,4937],{"class":4101},[3346,5127,5128,5130,5132,5134],{"class":3348,"line":3398},[3346,5129,4966],{"class":4101},[3346,5131,4969],{"class":4083},[3346,5133,4900],{"class":4101},[3346,5135,4974],{"class":4903},[3346,5137,5138,5140,5142,5144],{"class":3348,"line":3404},[3346,5139,4966],{"class":4101},[3346,5141,4969],{"class":4083},[3346,5143,4900],{"class":4101},[3346,5145,4985],{"class":4903},[3346,5147,5148,5150],{"class":3348,"line":3410},[3346,5149,4990],{"class":4083},[3346,5151,4937],{"class":4101},[3346,5153,5154,5156,5158],{"class":3348,"line":3416},[3346,5155,4997],{"class":4083},[3346,5157,4900],{"class":4101},[3346,5159,5002],{"class":4903},[3346,5161,5162,5164,5166],{"class":3348,"line":3422},[3346,5163,5007],{"class":4083},[3346,5165,4900],{"class":4101},[3346,5167,5012],{"class":4903},[3346,5169,5170],{"class":3348,"line":3427},[3346,5171,3395],{"emptyLinePlaceholder":3394},[3346,5173,5174,5176,5178,5180],{"class":3348,"line":3433},[3346,5175,4966],{"class":4101},[3346,5177,4897],{"class":4083},[3346,5179,4900],{"class":4101},[3346,5181,5027],{"class":4903},[3346,5183,5184,5186,5188],{"class":3348,"line":3439},[3346,5185,5032],{"class":4083},[3346,5187,4900],{"class":4101},[3346,5189,5190],{"class":4903},"mvn spotless:check\n",[3346,5192,5193],{"class":3348,"line":3550},[3346,5194,3395],{"emptyLinePlaceholder":3394},[3346,5196,5197,5199,5201,5203],{"class":3348,"line":3555},[3346,5198,4966],{"class":4101},[3346,5200,4897],{"class":4083},[3346,5202,4900],{"class":4101},[3346,5204,5052],{"class":4903},[3346,5206,5207,5209,5211],{"class":3348,"line":3561},[3346,5208,5032],{"class":4083},[3346,5210,4900],{"class":4101},[3346,5212,5213],{"class":4903},"mvn verify\n",[5215,5216,5217,5220,5221,5224,5225,5228,5229,5232],"tip",{},[2979,5218,5219],{},"Workflow для розробника:"," перед кожним commit виконайте ",[2968,5222,5223],{},"./gradlew spotlessApply"," (або ",[2968,5226,5227],{},"mvn spotless:apply","), щоб автоматично виправити всі форматування. Після цього ",[2968,5230,5231],{},"spotlessCheck"," у CI завжди буде зеленим.",[3070,5234],{},[3084,5236,5238],{"id":5237},"анатомія-конфігурації-ключові-параметри","Анатомія конфігурації: ключові параметри",[2964,5240,5241,5246,5247,5250],{},[2979,5242,5243],{},[2968,5244,5245],{},"reflowLongStrings()"," — google-java-format за замовчуванням не розбиває довгі рядкові літерали (вони можуть виходити за межу 100 символів). Ця опція вмикає перенесення рядків і всередині ",[2968,5248,5249],{},"String","-констант.",[2964,5252,5253,5258,5259,4919,5262,5265],{},[2979,5254,5255],{},[2968,5256,5257],{},"formatJavadoc(true)"," — форматує Javadoc-коментарі: вирівнює теги ",[2968,5260,5261],{},"@param",[2968,5263,5264],{},"@return",", усуває зайві пробіли. Це важливо для збереження узгодженого вигляду документації.",[2964,5267,5268,5273,5274,5277,5278,3140,5281,3140,5284,5287],{},[2979,5269,5270],{},[2968,5271,5272],{},"reorderImports(false)"," + явний ",[2968,5275,5276],{},"importOrder(...)"," — google-java-format має вбудований механізм сортування imports, але він відрізняється від конфігурації, що склалася у нашому проєкті. Вимикаємо вбудований і налаштовуємо явно через Spotless: ",[2968,5279,5280],{},"java/javax",[2968,5282,5283],{},"org",[2968,5285,5286],{},"com"," → решта → static imports.",[2964,5289,5290,5295,5296,5299],{},[2979,5291,5292],{},[2968,5293,5294],{},"ratchetFrom 'origin/main'"," — «тріщина» для legacy-проєктів. Якщо ввімкнено — Spotless перевіряє форматування лише у файлах, що змінилися відносно ",[2968,5297,5298],{},"main",". Дозволяє поступово впроваджувати стандарт без обов'язкового переформатування всього коду одночасно.",[3070,5301],{},[2959,5303,5305],{"id":5304},"spotless-довідник-налаштувань","Spotless: Довідник налаштувань",[2964,5307,5308],{},"Цей розділ є повним довідником усіх параметрів Spotless для Java-проєктів. Він призначений для самостійного вивчення та використання як шпаргалка при налаштуванні нового проєкту.",[3084,5310,5312,5313],{"id":5311},"глобальні-параметри-блоку-spotless","Глобальні параметри блоку ",[2968,5314,5315],{},"spotless {}",[2964,5317,5318,5319,5321],{},"Ці параметри задаються на рівні кореневого блоку ",[2968,5320,5315],{}," і впливають на всі підблоки форматів.",[5323,5324,5326,5329],"h4",{"id":5325},"ratchetfrom-інкрементальна-перевірка",[2968,5327,5328],{},"ratchetFrom"," — інкрементальна перевірка",[3092,5331,5333],{"className":3465,"code":5332,"language":3467,"meta":3100,"style":3100},"// Gradle\nspotless {\n    ratchetFrom 'origin/main'  // перевіряти тільки файли, змінені відносно main\n    // ratchetFrom 'HEAD'       // тільки uncommitted changes\n    // ratchetFrom 'origin/develop'\n}\n",[2968,5334,5335,5340,5344,5349,5357,5362],{"__ignoreMap":3100},[3346,5336,5337],{"class":3348,"line":3349},[3346,5338,5339],{},"// Gradle\n",[3346,5341,5342],{"class":3348,"line":3355},[3346,5343,3503],{},[3346,5345,5346],{"class":3348,"line":3361},[3346,5347,5348],{},"    ratchetFrom 'origin/main'  // перевіряти тільки файли, змінені відносно main\n",[3346,5350,5351,5354],{"class":3348,"line":3367},[3346,5352,5353],{},"    // ratchetFrom 'HEAD'",[3346,5355,5356],{},"       // тільки uncommitted changes\n",[3346,5358,5359],{"class":3348,"line":3373},[3346,5360,5361],{},"    // ratchetFrom 'origin/develop'\n",[3346,5363,5364],{"class":3348,"line":3379},[3346,5365,3494],{},[3092,5367,5369],{"className":4065,"code":5368,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven -->\n\u003Cconfiguration>\n    \u003CratchetFrom>origin/main\u003C/ratchetFrom>\n\u003C/configuration>\n",[2968,5370,5371,5376,5384,5401],{"__ignoreMap":3100},[3346,5372,5373],{"class":3348,"line":3349},[3346,5374,5375],{"class":3997},"\u003C!-- Maven -->\n",[3346,5377,5378,5380,5382],{"class":3348,"line":3355},[3346,5379,4080],{"class":4079},[3346,5381,4155],{"class":4083},[3346,5383,4087],{"class":4079},[3346,5385,5386,5388,5390,5392,5395,5397,5399],{"class":3348,"line":3361},[3346,5387,4092],{"class":4079},[3346,5389,5328],{"class":4083},[3346,5391,4098],{"class":4079},[3346,5393,5394],{"class":4101},"origin/main",[3346,5396,4105],{"class":4079},[3346,5398,5328],{"class":4083},[3346,5400,4087],{"class":4079},[3346,5402,5403,5405,5407],{"class":3348,"line":3367},[3346,5404,4105],{"class":4079},[3346,5406,4155],{"class":4083},[3346,5408,4087],{"class":4079},[2964,5410,5411,5414,5415,5417],{},[2979,5412,5413],{},"Коли використовувати:"," при додаванні Spotless до існуючого проєкту з великою кодовою базою, де переформатування всього коду одразу — неприйнятне. З увімкненим ",[2968,5416,5328],{}," кожен новий commit поступово «підтягує» стандарт форматування.",[2964,5419,5420,5423,5424,5426],{},[2979,5421,5422],{},"Як це працює:"," Spotless визначає список файлів, що змінилися відносно вказаного git-ref (наприклад ",[2968,5425,5394],{},"), і перевіряє форматування лише для них.",[3282,5428,5429,5431,5432,5435,5436,3068],{},[2968,5430,5328],{}," вимагає, щоб репозиторій мав git-history (не shallow clone). У CI може знадобитися ",[2968,5433,5434],{},"fetch-depth: 0"," у кроці ",[2968,5437,5438],{},"actions/checkout",[3070,5440],{},[5323,5442,5444,5447],{"id":5443},"encoding-кодування-файлів",[2968,5445,5446],{},"encoding"," — кодування файлів",[3092,5449,5451],{"className":3465,"code":5450,"language":3467,"meta":3100,"style":3100},"// Gradle (за замовчуванням UTF-8)\nspotless {\n    encoding 'UTF-8'\n}\n",[2968,5452,5453,5458,5462,5467],{"__ignoreMap":3100},[3346,5454,5455],{"class":3348,"line":3349},[3346,5456,5457],{},"// Gradle (за замовчуванням UTF-8)\n",[3346,5459,5460],{"class":3348,"line":3355},[3346,5461,3503],{},[3346,5463,5464],{"class":3348,"line":3361},[3346,5465,5466],{},"    encoding 'UTF-8'\n",[3346,5468,5469],{"class":3348,"line":3367},[3346,5470,3494],{},[3092,5472,5474],{"className":4065,"code":5473,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven -->\n\u003Cconfiguration>\n    \u003Cencoding>UTF-8\u003C/encoding>\n\u003C/configuration>\n",[2968,5475,5476,5480,5488,5505],{"__ignoreMap":3100},[3346,5477,5478],{"class":3348,"line":3349},[3346,5479,5375],{"class":3997},[3346,5481,5482,5484,5486],{"class":3348,"line":3355},[3346,5483,4080],{"class":4079},[3346,5485,4155],{"class":4083},[3346,5487,4087],{"class":4079},[3346,5489,5490,5492,5494,5496,5499,5501,5503],{"class":3348,"line":3361},[3346,5491,4092],{"class":4079},[3346,5493,5446],{"class":4083},[3346,5495,4098],{"class":4079},[3346,5497,5498],{"class":4101},"UTF-8",[3346,5500,4105],{"class":4079},[3346,5502,5446],{"class":4083},[3346,5504,4087],{"class":4079},[3346,5506,5507,5509,5511],{"class":3348,"line":3367},[3346,5508,4105],{"class":4079},[3346,5510,4155],{"class":4083},[3346,5512,4087],{"class":4079},[2964,5514,5515,5518,5519,5521],{},[2979,5516,5517],{},"Зазвичай не потребує зміни."," Spotless за замовчуванням використовує ",[2968,5520,5498],{},". Змінюйте лише якщо проєкт має legacy-файли у Windows-1251 або ISO-8859-1.",[3070,5523],{},[3084,5525,5527,5528,5531],{"id":5526},"блок-java-java-форматування","Блок ",[2968,5529,5530],{},"java {}"," — Java-форматування",[5323,5533,5535,5538,5539,5542],{"id":5534},"target-targetexclude-вибір-файлів",[2968,5536,5537],{},"target"," / ",[2968,5540,5541],{},"targetExclude"," — вибір файлів",[3092,5544,5546],{"className":3465,"code":5545,"language":3467,"meta":3100,"style":3100},"// Gradle\nspotless {\n    java {\n        // Явно вказати цільові файли (за замовчуванням Spotless знаходить їх автоматично)\n        target 'src/**/*.java'\n        target 'src/main/java/**/*.java', 'src/test/java/**/*.java'\n\n        // Виключити файли або директорії\n        targetExclude 'src/generated/**', '**/BuildConfig.java'\n    }\n}\n",[2968,5547,5548,5552,5556,5560,5565,5569,5574,5578,5583,5588,5592],{"__ignoreMap":3100},[3346,5549,5550],{"class":3348,"line":3349},[3346,5551,5339],{},[3346,5553,5554],{"class":3348,"line":3355},[3346,5555,3503],{},[3346,5557,5558],{"class":3348,"line":3361},[3346,5559,3532],{},[3346,5561,5562],{"class":3348,"line":3367},[3346,5563,5564],{},"        // Явно вказати цільові файли (за замовчуванням Spotless знаходить їх автоматично)\n",[3346,5566,5567],{"class":3348,"line":3373},[3346,5568,3547],{},[3346,5570,5571],{"class":3348,"line":3379},[3346,5572,5573],{},"        target 'src/main/java/**/*.java', 'src/test/java/**/*.java'\n",[3346,5575,5576],{"class":3348,"line":3385},[3346,5577,3395],{"emptyLinePlaceholder":3394},[3346,5579,5580],{"class":3348,"line":3391},[3346,5581,5582],{},"        // Виключити файли або директорії\n",[3346,5584,5585],{"class":3348,"line":3398},[3346,5586,5587],{},"        targetExclude 'src/generated/**', '**/BuildConfig.java'\n",[3346,5589,5590],{"class":3348,"line":3404},[3346,5591,3739],{},[3346,5593,5594],{"class":3348,"line":3410},[3346,5595,3494],{},[3092,5597,5599],{"className":4065,"code":5598,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven -->\n\u003Cjava>\n    \u003Cincludes>\n        \u003Cinclude>src/main/java/**/*.java\u003C/include>\n        \u003Cinclude>src/test/java/**/*.java\u003C/include>\n    \u003C/includes>\n    \u003Cexcludes>\n        \u003Cexclude>src/generated/**/*.java\u003C/exclude>\n    \u003C/excludes>\n\u003C/java>\n",[2968,5600,5601,5605,5613,5621,5637,5653,5661,5670,5688,5696],{"__ignoreMap":3100},[3346,5602,5603],{"class":3348,"line":3349},[3346,5604,5375],{"class":3997},[3346,5606,5607,5609,5611],{"class":3348,"line":3355},[3346,5608,4080],{"class":4079},[3346,5610,4179],{"class":4083},[3346,5612,4087],{"class":4079},[3346,5614,5615,5617,5619],{"class":3348,"line":3361},[3346,5616,4092],{"class":4079},[3346,5618,4194],{"class":4083},[3346,5620,4087],{"class":4079},[3346,5622,5623,5625,5627,5629,5631,5633,5635],{"class":3348,"line":3367},[3346,5624,4176],{"class":4079},[3346,5626,4204],{"class":4083},[3346,5628,4098],{"class":4079},[3346,5630,4209],{"class":4101},[3346,5632,4105],{"class":4079},[3346,5634,4204],{"class":4083},[3346,5636,4087],{"class":4079},[3346,5638,5639,5641,5643,5645,5647,5649,5651],{"class":3348,"line":3373},[3346,5640,4176],{"class":4079},[3346,5642,4204],{"class":4083},[3346,5644,4098],{"class":4079},[3346,5646,4226],{"class":4101},[3346,5648,4105],{"class":4079},[3346,5650,4204],{"class":4083},[3346,5652,4087],{"class":4079},[3346,5654,5655,5657,5659],{"class":3348,"line":3379},[3346,5656,4656],{"class":4079},[3346,5658,4194],{"class":4083},[3346,5660,4087],{"class":4079},[3346,5662,5663,5665,5668],{"class":3348,"line":3385},[3346,5664,4092],{"class":4079},[3346,5666,5667],{"class":4083},"excludes",[3346,5669,4087],{"class":4079},[3346,5671,5672,5674,5677,5679,5682,5684,5686],{"class":3348,"line":3391},[3346,5673,4176],{"class":4079},[3346,5675,5676],{"class":4083},"exclude",[3346,5678,4098],{"class":4079},[3346,5680,5681],{"class":4101},"src/generated/**/*.java",[3346,5683,4105],{"class":4079},[3346,5685,5676],{"class":4083},[3346,5687,4087],{"class":4079},[3346,5689,5690,5692,5694],{"class":3348,"line":3398},[3346,5691,4656],{"class":4079},[3346,5693,5667],{"class":4083},[3346,5695,4087],{"class":4079},[3346,5697,5698,5700,5702],{"class":3348,"line":3404},[3346,5699,4105],{"class":4079},[3346,5701,4179],{"class":4083},[3346,5703,4087],{"class":4079},[3070,5705],{},[3084,5707,5709],{"id":5708},"форматери-java","Форматери Java",[2964,5711,5712,5713,4919,5715,5717],{},"Spotless підтримує кілька альтернативних форматерів. Одночасно може бути активований лише один основний форматер (хоча інші кроки — ",[2968,5714,4395],{},[2968,5716,4439],{}," тощо — поєднуються вільно).",[5323,5719,5721,5724],{"id":5720},"googlejavaformat-рекомендований-форматер",[2968,5722,5723],{},"googleJavaFormat()"," — рекомендований форматер",[2964,5726,5727],{},"Найпопулярніший вибір для Java-проєктів. Використовує ті самі правила, що і внутрішні інструменти Google.",[3092,5729,5731],{"className":3465,"code":5730,"language":3467,"meta":3100,"style":3100},"// Gradle — повна сигнатура з усіма параметрами\nspotless {\n    java {\n        googleJavaFormat()                 // остання підтримувана версія (1.35.0 у 2026)\n        // або з явною версією:\n        googleJavaFormat('1.35.0')\n            // Стиль відступу:\n            .style('GOOGLE')               // 2-пробільний відступ (за замовчуванням)\n            // .aosp()                     // 4-пробільний відступ (Android Open Source Project)\n            //\n            // Перенесення довгих рядкових літералів:\n            .reflowLongStrings()           // увімкнути (за замовчуванням вимкнено)\n            //\n            // Форматування Javadoc:\n            .formatJavadoc(true)           // true = форматувати (за замовчуванням true)\n            // .skipJavadocFormatting()    // еквівалент .formatJavadoc(false)\n            //\n            // Перегрупування imports:\n            .reorderImports(false)         // false = керувати imports через importOrder()\n            // .reorderImports(true)       // GJF перегруповує imports самостійно\n    }\n}\n",[2968,5732,5733,5738,5742,5746,5751,5756,5760,5765,5770,5778,5783,5788,5793,5797,5802,5807,5815,5819,5824,5829,5837,5841],{"__ignoreMap":3100},[3346,5734,5735],{"class":3348,"line":3349},[3346,5736,5737],{},"// Gradle — повна сигнатура з усіма параметрами\n",[3346,5739,5740],{"class":3348,"line":3355},[3346,5741,3503],{},[3346,5743,5744],{"class":3348,"line":3361},[3346,5745,3532],{},[3346,5747,5748],{"class":3348,"line":3367},[3346,5749,5750],{},"        googleJavaFormat()                 // остання підтримувана версія (1.35.0 у 2026)\n",[3346,5752,5753],{"class":3348,"line":3373},[3346,5754,5755],{},"        // або з явною версією:\n",[3346,5757,5758],{"class":3348,"line":3379},[3346,5759,3588],{},[3346,5761,5762],{"class":3348,"line":3385},[3346,5763,5764],{},"            // Стиль відступу:\n",[3346,5766,5767],{"class":3348,"line":3391},[3346,5768,5769],{},"            .style('GOOGLE')               // 2-пробільний відступ (за замовчуванням)\n",[3346,5771,5772,5775],{"class":3348,"line":3398},[3346,5773,5774],{},"            // .aosp()",[3346,5776,5777],{},"                     // 4-пробільний відступ (Android Open Source Project)\n",[3346,5779,5780],{"class":3348,"line":3404},[3346,5781,5782],{},"            //\n",[3346,5784,5785],{"class":3348,"line":3410},[3346,5786,5787],{},"            // Перенесення довгих рядкових літералів:\n",[3346,5789,5790],{"class":3348,"line":3416},[3346,5791,5792],{},"            .reflowLongStrings()           // увімкнути (за замовчуванням вимкнено)\n",[3346,5794,5795],{"class":3348,"line":3422},[3346,5796,5782],{},[3346,5798,5799],{"class":3348,"line":3427},[3346,5800,5801],{},"            // Форматування Javadoc:\n",[3346,5803,5804],{"class":3348,"line":3433},[3346,5805,5806],{},"            .formatJavadoc(true)           // true = форматувати (за замовчуванням true)\n",[3346,5808,5809,5812],{"class":3348,"line":3439},[3346,5810,5811],{},"            // .skipJavadocFormatting()",[3346,5813,5814],{},"    // еквівалент .formatJavadoc(false)\n",[3346,5816,5817],{"class":3348,"line":3550},[3346,5818,5782],{},[3346,5820,5821],{"class":3348,"line":3555},[3346,5822,5823],{},"            // Перегрупування imports:\n",[3346,5825,5826],{"class":3348,"line":3561},[3346,5827,5828],{},"            .reorderImports(false)         // false = керувати imports через importOrder()\n",[3346,5830,5831,5834],{"class":3348,"line":3567},[3346,5832,5833],{},"            // .reorderImports(true)",[3346,5835,5836],{},"       // GJF перегруповує imports самостійно\n",[3346,5838,5839],{"class":3348,"line":3573},[3346,5840,3739],{},[3346,5842,5843],{"class":3348,"line":3579},[3346,5844,3494],{},[3092,5846,5848],{"className":4065,"code":5847,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven — повна конфігурація -->\n\u003CgoogleJavaFormat>\n    \u003Cversion>1.35.0\u003C/version>\n    \u003C!-- GOOGLE (2 пробіли) або AOSP (4 пробіли) -->\n    \u003Cstyle>GOOGLE\u003C/style>\n    \u003C!-- Переносити довгі рядкові літерали -->\n    \u003CreflowLongStrings>true\u003C/reflowLongStrings>\n    \u003C!-- Форматувати Javadoc (true за замовчуванням) -->\n    \u003CformatJavadoc>true\u003C/formatJavadoc>\n    \u003C!-- false = imports керуємо через \u003CimportOrder> -->\n    \u003CreorderImports>false\u003C/reorderImports>\n    \u003C!-- Власний groupArtifact (не потрібно у більшості випадків) -->\n    \u003C!-- \u003CgroupArtifact>com.google.googlejavaformat:google-java-format\u003C/groupArtifact> -->\n\u003C/googleJavaFormat>\n",[2968,5849,5850,5855,5863,5879,5884,5900,5905,5921,5926,5942,5947,5963,5968,5973],{"__ignoreMap":3100},[3346,5851,5852],{"class":3348,"line":3349},[3346,5853,5854],{"class":3997},"\u003C!-- Maven — повна конфігурація -->\n",[3346,5856,5857,5859,5861],{"class":3348,"line":3355},[3346,5858,4080],{"class":4079},[3346,5860,4267],{"class":4083},[3346,5862,4087],{"class":4079},[3346,5864,5865,5867,5869,5871,5873,5875,5877],{"class":3348,"line":3361},[3346,5866,4092],{"class":4079},[3346,5868,4134],{"class":4083},[3346,5870,4098],{"class":4079},[3346,5872,3059],{"class":4101},[3346,5874,4105],{"class":4079},[3346,5876,4134],{"class":4083},[3346,5878,4087],{"class":4079},[3346,5880,5881],{"class":3348,"line":3367},[3346,5882,5883],{"class":3997},"    \u003C!-- GOOGLE (2 пробіли) або AOSP (4 пробіли) -->\n",[3346,5885,5886,5888,5890,5892,5894,5896,5898],{"class":3348,"line":3373},[3346,5887,4092],{"class":4079},[3346,5889,4292],{"class":4083},[3346,5891,4098],{"class":4079},[3346,5893,4297],{"class":4101},[3346,5895,4105],{"class":4079},[3346,5897,4292],{"class":4083},[3346,5899,4087],{"class":4079},[3346,5901,5902],{"class":3348,"line":3379},[3346,5903,5904],{"class":3997},"    \u003C!-- Переносити довгі рядкові літерали -->\n",[3346,5906,5907,5909,5911,5913,5915,5917,5919],{"class":3348,"line":3385},[3346,5908,4092],{"class":4079},[3346,5910,4310],{"class":4083},[3346,5912,4098],{"class":4079},[3346,5914,4315],{"class":4101},[3346,5916,4105],{"class":4079},[3346,5918,4310],{"class":4083},[3346,5920,4087],{"class":4079},[3346,5922,5923],{"class":3348,"line":3391},[3346,5924,5925],{"class":3997},"    \u003C!-- Форматувати Javadoc (true за замовчуванням) -->\n",[3346,5927,5928,5930,5932,5934,5936,5938,5940],{"class":3348,"line":3398},[3346,5929,4092],{"class":4079},[3346,5931,4328],{"class":4083},[3346,5933,4098],{"class":4079},[3346,5935,4315],{"class":4101},[3346,5937,4105],{"class":4079},[3346,5939,4328],{"class":4083},[3346,5941,4087],{"class":4079},[3346,5943,5944],{"class":3348,"line":3404},[3346,5945,5946],{"class":3997},"    \u003C!-- false = imports керуємо через \u003CimportOrder> -->\n",[3346,5948,5949,5951,5953,5955,5957,5959,5961],{"class":3348,"line":3410},[3346,5950,4092],{"class":4079},[3346,5952,4345],{"class":4083},[3346,5954,4098],{"class":4079},[3346,5956,4350],{"class":4101},[3346,5958,4105],{"class":4079},[3346,5960,4345],{"class":4083},[3346,5962,4087],{"class":4079},[3346,5964,5965],{"class":3348,"line":3416},[3346,5966,5967],{"class":3997},"    \u003C!-- Власний groupArtifact (не потрібно у більшості випадків) -->\n",[3346,5969,5970],{"class":3348,"line":3422},[3346,5971,5972],{"class":3997},"    \u003C!-- \u003CgroupArtifact>com.google.googlejavaformat:google-java-format\u003C/groupArtifact> -->\n",[3346,5974,5975,5977,5979],{"class":3348,"line":3427},[3346,5976,4105],{"class":4079},[3346,5978,4267],{"class":4083},[3346,5980,4087],{"class":4079},[2964,5982,5983],{},[2979,5984,5985],{},"Різниця GOOGLE vs AOSP:",[3181,5987,5988,5999],{},[3184,5989,5990],{},[3187,5991,5992,5994,5996],{},[3190,5993,3192],{},[3190,5995,4297],{},[3190,5997,5998],{},"AOSP",[3200,6000,6001,6010,6020],{},[3187,6002,6003,6005,6007],{},[3205,6004,3207],{},[3205,6006,3212],{},[3205,6008,6009],{},"4 пробіли",[3187,6011,6012,6015,6017],{},[3205,6013,6014],{},"Продовження рядка",[3205,6016,6009],{},[3205,6018,6019],{},"8 пробілів",[3187,6021,6022,6025,6028],{},[3205,6023,6024],{},"Використання",[3205,6026,6027],{},"Більшість Java-проєктів",[3205,6029,6030],{},"Android-проєкти",[3070,6032],{},[5323,6034,6036,6039],{"id":6035},"eclipse-форматер-eclipse",[2968,6037,6038],{},"eclipse()"," — форматер Eclipse",[2964,6041,6042],{},"Альтернативний форматер на основі Eclipse JDT. Дозволяє використовувати XML-файл конфігурації, аналогічний тому, що ви імпортували в IntelliJ:",[3092,6044,6046],{"className":3465,"code":6045,"language":3467,"meta":3100,"style":3100},"// Gradle\nspotless {\n    java {\n        eclipse()                                     // Eclipse за замовчуванням\n        eclipse('4.26')                               // конкретна версія Eclipse\n        eclipse().configFile('eclipse-format.xml')   // із зовнішнього файлу\n    }\n}\n",[2968,6047,6048,6052,6056,6060,6065,6070,6075,6079],{"__ignoreMap":3100},[3346,6049,6050],{"class":3348,"line":3349},[3346,6051,5339],{},[3346,6053,6054],{"class":3348,"line":3355},[3346,6055,3503],{},[3346,6057,6058],{"class":3348,"line":3361},[3346,6059,3532],{},[3346,6061,6062],{"class":3348,"line":3367},[3346,6063,6064],{},"        eclipse()                                     // Eclipse за замовчуванням\n",[3346,6066,6067],{"class":3348,"line":3373},[3346,6068,6069],{},"        eclipse('4.26')                               // конкретна версія Eclipse\n",[3346,6071,6072],{"class":3348,"line":3379},[3346,6073,6074],{},"        eclipse().configFile('eclipse-format.xml')   // із зовнішнього файлу\n",[3346,6076,6077],{"class":3348,"line":3385},[3346,6078,3739],{},[3346,6080,6081],{"class":3348,"line":3391},[3346,6082,3494],{},[3092,6084,6086],{"className":4065,"code":6085,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven -->\n\u003Ceclipse>\n    \u003Cversion>4.26\u003C/version>\n    \u003CconfigFile>eclipse-format.xml\u003C/configFile>\n\u003C/eclipse>\n",[2968,6087,6088,6092,6101,6118,6136],{"__ignoreMap":3100},[3346,6089,6090],{"class":3348,"line":3349},[3346,6091,5375],{"class":3997},[3346,6093,6094,6096,6099],{"class":3348,"line":3355},[3346,6095,4080],{"class":4079},[3346,6097,6098],{"class":4083},"eclipse",[3346,6100,4087],{"class":4079},[3346,6102,6103,6105,6107,6109,6112,6114,6116],{"class":3348,"line":3361},[3346,6104,4092],{"class":4079},[3346,6106,4134],{"class":4083},[3346,6108,4098],{"class":4079},[3346,6110,6111],{"class":4101},"4.26",[3346,6113,4105],{"class":4079},[3346,6115,4134],{"class":4083},[3346,6117,4087],{"class":4079},[3346,6119,6120,6122,6125,6127,6130,6132,6134],{"class":3348,"line":3367},[3346,6121,4092],{"class":4079},[3346,6123,6124],{"class":4083},"configFile",[3346,6126,4098],{"class":4079},[3346,6128,6129],{"class":4101},"eclipse-format.xml",[3346,6131,4105],{"class":4079},[3346,6133,6124],{"class":4083},[3346,6135,4087],{"class":4079},[3346,6137,6138,6140,6142],{"class":3348,"line":3373},[3346,6139,4105],{"class":4079},[3346,6141,6098],{"class":4083},[3346,6143,4087],{"class":4079},[2964,6145,6146,6148,6149,6151,6152,6155],{},[2979,6147,5413],{}," якщо команда вже має налаштований Eclipse formatter XML і хоче використати ті самі правила через Spotless. Файл ",[2968,6150,3153],{}," ",[2979,6153,6154],{},"не сумісний"," з форматером Eclipse — це два різних формати.",[3070,6157],{},[5323,6159,6161,6164],{"id":6160},"prettier-prettier-для-java-з-nodejs",[2968,6162,6163],{},"prettier()"," — Prettier для Java (з Node.js)",[3092,6166,6168],{"className":3465,"code":6167,"language":3467,"meta":3100,"style":3100},"// Gradle (потребує Node.js)\nspotless {\n    java {\n        prettier(['prettier': '3.2.5', 'prettier-plugin-java': '2.6.0'])\n            .config(['tabWidth': 4, 'printWidth': 120])\n    }\n}\n",[2968,6169,6170,6175,6179,6183,6188,6193,6197],{"__ignoreMap":3100},[3346,6171,6172],{"class":3348,"line":3349},[3346,6173,6174],{},"// Gradle (потребує Node.js)\n",[3346,6176,6177],{"class":3348,"line":3355},[3346,6178,3503],{},[3346,6180,6181],{"class":3348,"line":3361},[3346,6182,3532],{},[3346,6184,6185],{"class":3348,"line":3367},[3346,6186,6187],{},"        prettier(['prettier': '3.2.5', 'prettier-plugin-java': '2.6.0'])\n",[3346,6189,6190],{"class":3348,"line":3373},[3346,6191,6192],{},"            .config(['tabWidth': 4, 'printWidth': 120])\n",[3346,6194,6195],{"class":3348,"line":3379},[3346,6196,3739],{},[3346,6198,6199],{"class":3348,"line":3385},[3346,6200,3494],{},[2964,6202,6203,6206,6207,6209],{},[2979,6204,6205],{},"Рідко використовується для Java."," Prettier — стандарт у JavaScript/TypeScript. Для Java-проєктів ",[2968,6208,4267],{}," є кращим вибором.",[3070,6211],{},[3084,6213,6215],{"id":6214},"управління-imports","Управління imports",[2964,6217,6218,6219,6222],{},"Кроки управління imports застосовуються ",[2979,6220,6221],{},"після"," основного форматера і не конкурують між собою.",[5323,6224,6226,6229],{"id":6225},"importorder-порядок-груп-imports",[2968,6227,6228],{},"importOrder()"," — порядок груп imports",[3092,6231,6233],{"className":3465,"code":6232,"language":3467,"meta":3100,"style":3100},"// Gradle\nspotless {\n    java {\n        // Варіант 1: стандартний порядок Eclipse (мінімальна конфігурація)\n        importOrder()\n\n        // Варіант 2: явний порядок груп (рекомендований)\n        // Кожен рядок = окрема група з порожнім рядком між ними\n        // '' (порожній рядок) = \"всі imports, що не потрапили у жодну групу\"\n        // '\\\\#' = static imports (import static ...)\n        importOrder('java', 'javax', 'org', 'com', '', '\\\\#')\n\n        // Варіант 3: об'єднання груп БЕЗ порожнього рядка між ними (через '|')\n        // java та javax будуть в одній групі без роздільника\n        importOrder('java|javax', 'org', 'com', '', '\\\\#')\n\n        // Варіант 4: з файлу (формат Eclipse .importorder)\n        importOrderFile('config/eclipse.importorder')\n\n        // Варіант 5: static imports першими (стиль деяких Google-команд)\n        importOrder('\\\\#', 'java', 'javax', 'org', 'com', '')\n    }\n}\n",[2968,6234,6235,6239,6243,6247,6252,6257,6261,6266,6271,6276,6281,6285,6289,6294,6299,6304,6308,6313,6318,6322,6327,6332,6336],{"__ignoreMap":3100},[3346,6236,6237],{"class":3348,"line":3349},[3346,6238,5339],{},[3346,6240,6241],{"class":3348,"line":3355},[3346,6242,3503],{},[3346,6244,6245],{"class":3348,"line":3361},[3346,6246,3532],{},[3346,6248,6249],{"class":3348,"line":3367},[3346,6250,6251],{},"        // Варіант 1: стандартний порядок Eclipse (мінімальна конфігурація)\n",[3346,6253,6254],{"class":3348,"line":3373},[3346,6255,6256],{},"        importOrder()\n",[3346,6258,6259],{"class":3348,"line":3379},[3346,6260,3395],{"emptyLinePlaceholder":3394},[3346,6262,6263],{"class":3348,"line":3385},[3346,6264,6265],{},"        // Варіант 2: явний порядок груп (рекомендований)\n",[3346,6267,6268],{"class":3348,"line":3391},[3346,6269,6270],{},"        // Кожен рядок = окрема група з порожнім рядком між ними\n",[3346,6272,6273],{"class":3348,"line":3398},[3346,6274,6275],{},"        // '' (порожній рядок) = \"всі imports, що не потрапили у жодну групу\"\n",[3346,6277,6278],{"class":3348,"line":3404},[3346,6279,6280],{},"        // '\\\\#' = static imports (import static ...)\n",[3346,6282,6283],{"class":3348,"line":3410},[3346,6284,3647],{},[3346,6286,6287],{"class":3348,"line":3416},[3346,6288,3395],{"emptyLinePlaceholder":3394},[3346,6290,6291],{"class":3348,"line":3422},[3346,6292,6293],{},"        // Варіант 3: об'єднання груп БЕЗ порожнього рядка між ними (через '|')\n",[3346,6295,6296],{"class":3348,"line":3427},[3346,6297,6298],{},"        // java та javax будуть в одній групі без роздільника\n",[3346,6300,6301],{"class":3348,"line":3433},[3346,6302,6303],{},"        importOrder('java|javax', 'org', 'com', '', '\\\\#')\n",[3346,6305,6306],{"class":3348,"line":3439},[3346,6307,3395],{"emptyLinePlaceholder":3394},[3346,6309,6310],{"class":3348,"line":3550},[3346,6311,6312],{},"        // Варіант 4: з файлу (формат Eclipse .importorder)\n",[3346,6314,6315],{"class":3348,"line":3555},[3346,6316,6317],{},"        importOrderFile('config/eclipse.importorder')\n",[3346,6319,6320],{"class":3348,"line":3561},[3346,6321,3395],{"emptyLinePlaceholder":3394},[3346,6323,6324],{"class":3348,"line":3567},[3346,6325,6326],{},"        // Варіант 5: static imports першими (стиль деяких Google-команд)\n",[3346,6328,6329],{"class":3348,"line":3573},[3346,6330,6331],{},"        importOrder('\\\\#', 'java', 'javax', 'org', 'com', '')\n",[3346,6333,6334],{"class":3348,"line":3579},[3346,6335,3739],{},[3346,6337,6338],{"class":3348,"line":3585},[3346,6339,3494],{},[3092,6341,6343],{"className":4065,"code":6342,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven -->\n\u003CimportOrder>\n    \u003C!-- Роздільник груп — кома. '|' — без порожнього рядка між групами. -->\n    \u003C!-- '\\#' — static imports -->\n    \u003Corder>java|javax,org,com,,\\#\u003C/order>\n\n    \u003C!-- Або з файлу: -->\n    \u003C!-- \u003Cfile>config/eclipse.importorder\u003C/file> -->\n\u003C/importOrder>\n",[2968,6344,6345,6349,6357,6362,6367,6383,6387,6392,6397],{"__ignoreMap":3100},[3346,6346,6347],{"class":3348,"line":3349},[3346,6348,5375],{"class":3997},[3346,6350,6351,6353,6355],{"class":3348,"line":3355},[3346,6352,4080],{"class":4079},[3346,6354,4395],{"class":4083},[3346,6356,4087],{"class":4079},[3346,6358,6359],{"class":3348,"line":3361},[3346,6360,6361],{"class":3997},"    \u003C!-- Роздільник груп — кома. '|' — без порожнього рядка між групами. -->\n",[3346,6363,6364],{"class":3348,"line":3367},[3346,6365,6366],{"class":3997},"    \u003C!-- '\\#' — static imports -->\n",[3346,6368,6369,6371,6373,6375,6377,6379,6381],{"class":3348,"line":3373},[3346,6370,4092],{"class":4079},[3346,6372,4404],{"class":4083},[3346,6374,4098],{"class":4079},[3346,6376,4409],{"class":4101},[3346,6378,4105],{"class":4079},[3346,6380,4404],{"class":4083},[3346,6382,4087],{"class":4079},[3346,6384,6385],{"class":3348,"line":3379},[3346,6386,3395],{"emptyLinePlaceholder":3394},[3346,6388,6389],{"class":3348,"line":3385},[3346,6390,6391],{"class":3997},"    \u003C!-- Або з файлу: -->\n",[3346,6393,6394],{"class":3348,"line":3391},[3346,6395,6396],{"class":3997},"    \u003C!-- \u003Cfile>config/eclipse.importorder\u003C/file> -->\n",[3346,6398,6399,6401,6403],{"class":3348,"line":3398},[3346,6400,4105],{"class":4079},[3346,6402,4395],{"class":4083},[3346,6404,4087],{"class":4079},[2964,6406,6407,6413,6414,6417],{},[2979,6408,6409,6410],{},"Формат файлу ",[2968,6411,6412],{},".importorder"," (для ",[2968,6415,6416],{},"importOrderFile()","):",[3092,6419,6422],{"className":6420,"code":6421,"language":3097},[3095],"# Порядок imports для Spotless / Eclipse\n0=java\n1=javax\n2=org\n3=com\n4=\n5=\\#\n",[2968,6423,6421],{"__ignoreMap":3100},[3070,6425],{},[5323,6427,6429,6432],{"id":6428},"removeunusedimports-видалення-невикористаних-imports",[2968,6430,6431],{},"removeUnusedImports()"," — видалення невикористаних imports",[3092,6434,6436],{"className":3465,"code":6435,"language":3467,"meta":3100,"style":3100},"// Gradle\nspotless {\n    java {\n        // Стандартний движок (google-java-format)\n        removeUnusedImports()\n\n        // Альтернативний движок (підтримує ширший діапазон Java-версій)\n        // Корисний якщо googleJavaFormat не підтримує вашу версію синтаксису\n        removeUnusedImports('cleanthat-javaparser-unnecessaryimport')\n    }\n}\n",[2968,6437,6438,6442,6446,6450,6455,6459,6463,6468,6473,6478,6482],{"__ignoreMap":3100},[3346,6439,6440],{"class":3348,"line":3349},[3346,6441,5339],{},[3346,6443,6444],{"class":3348,"line":3355},[3346,6445,3503],{},[3346,6447,6448],{"class":3348,"line":3361},[3346,6449,3532],{},[3346,6451,6452],{"class":3348,"line":3367},[3346,6453,6454],{},"        // Стандартний движок (google-java-format)\n",[3346,6456,6457],{"class":3348,"line":3373},[3346,6458,3664],{},[3346,6460,6461],{"class":3348,"line":3379},[3346,6462,3395],{"emptyLinePlaceholder":3394},[3346,6464,6465],{"class":3348,"line":3385},[3346,6466,6467],{},"        // Альтернативний движок (підтримує ширший діапазон Java-версій)\n",[3346,6469,6470],{"class":3348,"line":3391},[3346,6471,6472],{},"        // Корисний якщо googleJavaFormat не підтримує вашу версію синтаксису\n",[3346,6474,6475],{"class":3348,"line":3398},[3346,6476,6477],{},"        removeUnusedImports('cleanthat-javaparser-unnecessaryimport')\n",[3346,6479,6480],{"class":3348,"line":3404},[3346,6481,3739],{},[3346,6483,6484],{"class":3348,"line":3410},[3346,6485,3494],{},[3092,6487,6489],{"className":4065,"code":6488,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven -->\n\u003CremoveUnusedImports/>\n\u003C!-- або з альтернативним движком: -->\n\u003CremoveUnusedImports>\n    \u003Cengine>cleanthat-javaparser-unnecessaryimport\u003C/engine>\n\u003C/removeUnusedImports>\n",[2968,6490,6491,6495,6503,6508,6516,6534],{"__ignoreMap":3100},[3346,6492,6493],{"class":3348,"line":3349},[3346,6494,5375],{"class":3997},[3346,6496,6497,6499,6501],{"class":3348,"line":3355},[3346,6498,4080],{"class":4079},[3346,6500,4439],{"class":4083},[3346,6502,4442],{"class":4079},[3346,6504,6505],{"class":3348,"line":3361},[3346,6506,6507],{"class":3997},"\u003C!-- або з альтернативним движком: -->\n",[3346,6509,6510,6512,6514],{"class":3348,"line":3367},[3346,6511,4080],{"class":4079},[3346,6513,4439],{"class":4083},[3346,6515,4087],{"class":4079},[3346,6517,6518,6520,6523,6525,6528,6530,6532],{"class":3348,"line":3373},[3346,6519,4092],{"class":4079},[3346,6521,6522],{"class":4083},"engine",[3346,6524,4098],{"class":4079},[3346,6526,6527],{"class":4101},"cleanthat-javaparser-unnecessaryimport",[3346,6529,4105],{"class":4079},[3346,6531,6522],{"class":4083},[3346,6533,4087],{"class":4079},[3346,6535,6536,6538,6540],{"class":3348,"line":3379},[3346,6537,4105],{"class":4079},[3346,6539,4439],{"class":4083},[3346,6541,4087],{"class":4079},[2964,6543,6544,6550,6551,6553],{},[2979,6545,6546,6547,6549],{},"Коли використовувати ",[2968,6548,6527],{},":"," якщо проєкт використовує Java 21+ pattern matching, records або sealed classes, а ",[2968,6552,4267],{}," ще не підтримує новий синтаксис у своєму парсері.",[3070,6555],{},[5323,6557,6559,6562],{"id":6558},"forbidwildcardimports-заборона-wildcard-imports",[2968,6560,6561],{},"forbidWildcardImports()"," — заборона wildcard imports",[3092,6564,6566],{"className":3465,"code":6565,"language":3467,"meta":3100,"style":3100},"// Gradle\nspotless {\n    java {\n        forbidWildcardImports()    // провалити білд при знахідці import java.util.*\n    }\n}\n",[2968,6567,6568,6572,6576,6580,6585,6589],{"__ignoreMap":3100},[3346,6569,6570],{"class":3348,"line":3349},[3346,6571,5339],{},[3346,6573,6574],{"class":3348,"line":3355},[3346,6575,3503],{},[3346,6577,6578],{"class":3348,"line":3361},[3346,6579,3532],{},[3346,6581,6582],{"class":3348,"line":3367},[3346,6583,6584],{},"        forbidWildcardImports()    // провалити білд при знахідці import java.util.*\n",[3346,6586,6587],{"class":3348,"line":3373},[3346,6588,3739],{},[3346,6590,6591],{"class":3348,"line":3379},[3346,6592,3494],{},[3092,6594,6596],{"className":4065,"code":6595,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven -->\n\u003CforbidWildcardImports/>\n",[2968,6597,6598,6602],{"__ignoreMap":3100},[3346,6599,6600],{"class":3348,"line":3349},[3346,6601,5375],{"class":3997},[3346,6603,6604,6606,6609],{"class":3348,"line":3355},[3346,6605,4080],{"class":4079},[3346,6607,6608],{"class":4083},"forbidWildcardImports",[3346,6610,4442],{"class":4079},[2964,6612,6613,6151,6616,6618],{},[2979,6614,6615],{},"Чому wildcard шкідливі:",[2968,6617,3279],{}," не показує, які саме класи використовуються. При рефакторингу важко зрозуміти залежності. Сучасні IDE і Google Style забороняють їх.",[3070,6620],{},[5323,6622,6624,6627],{"id":6623},"expandwildcardimports-розгортання-wildcard-у-explicit",[2968,6625,6626],{},"expandWildcardImports()"," — розгортання wildcard у explicit",[3092,6629,6631],{"className":3465,"code":6630,"language":3467,"meta":3100,"style":3100},"// Gradle — автоматично розгорнути import java.util.* у конкретні imports\nspotless {\n    java {\n        expandWildcardImports()\n    }\n}\n",[2968,6632,6633,6638,6642,6646,6651,6655],{"__ignoreMap":3100},[3346,6634,6635],{"class":3348,"line":3349},[3346,6636,6637],{},"// Gradle — автоматично розгорнути import java.util.* у конкретні imports\n",[3346,6639,6640],{"class":3348,"line":3355},[3346,6641,3503],{},[3346,6643,6644],{"class":3348,"line":3361},[3346,6645,3532],{},[3346,6647,6648],{"class":3348,"line":3367},[3346,6649,6650],{},"        expandWildcardImports()\n",[3346,6652,6653],{"class":3348,"line":3373},[3346,6654,3739],{},[3346,6656,6657],{"class":3348,"line":3379},[3346,6658,3494],{},[3092,6660,6662],{"className":4065,"code":6661,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven -->\n\u003CexpandWildcardImports/>\n",[2968,6663,6664,6668],{"__ignoreMap":3100},[3346,6665,6666],{"class":3348,"line":3349},[3346,6667,5375],{"class":3997},[3346,6669,6670,6672,6675],{"class":3348,"line":3355},[3346,6671,4080],{"class":4079},[3346,6673,6674],{"class":4083},"expandWildcardImports",[3346,6676,4442],{"class":4079},[2964,6678,6679,6682,6683,6686,6687,3068],{},[2979,6680,6681],{},"Корисний при міграції:"," один раз запустіть ",[2968,6684,6685],{},"spotlessApply"," з цією опцією, щоб розгорнути всі wildcard imports у проєкті, а потім замініть на ",[2968,6688,6561],{},[3070,6690],{},[5323,6692,6694,6697],{"id":6693},"forbidmoduleimports-заборона-модульних-imports-java-25",[2968,6695,6696],{},"forbidModuleImports()"," — заборона модульних imports (Java 25+)",[3092,6699,6701],{"className":3465,"code":6700,"language":3467,"meta":3100,"style":3100},"// Gradle (для проєктів на Java 25+)\nspotless {\n    java {\n        forbidModuleImports()\n    }\n}\n",[2968,6702,6703,6708,6712,6716,6721,6725],{"__ignoreMap":3100},[3346,6704,6705],{"class":3348,"line":3349},[3346,6706,6707],{},"// Gradle (для проєктів на Java 25+)\n",[3346,6709,6710],{"class":3348,"line":3355},[3346,6711,3503],{},[3346,6713,6714],{"class":3348,"line":3361},[3346,6715,3532],{},[3346,6717,6718],{"class":3348,"line":3367},[3346,6719,6720],{},"        forbidModuleImports()\n",[3346,6722,6723],{"class":3348,"line":3373},[3346,6724,3739],{},[3346,6726,6727],{"class":3348,"line":3379},[3346,6728,3494],{},[3092,6730,6732],{"className":4065,"code":6731,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven -->\n\u003CforbidModuleImports/>\n",[2968,6733,6734,6738],{"__ignoreMap":3100},[3346,6735,6736],{"class":3348,"line":3349},[3346,6737,5375],{"class":3997},[3346,6739,6740,6742,6745],{"class":3348,"line":3355},[3346,6741,4080],{"class":4079},[3346,6743,6744],{"class":4083},"forbidModuleImports",[3346,6746,4442],{"class":4079},[2964,6748,6749,6752,6753,6756],{},[2979,6750,6751],{},"Актуально з Java 25",", де з'явилися модульні imports (",[2968,6754,6755],{},"import module java.base","). Забороняє їх використання для сумісності зі старшими версіями.",[3070,6758],{},[3084,6760,6762],{"id":6761},"форматування-анотацій","Форматування анотацій",[5323,6764,6766,6769],{"id":6765},"formatannotations-коректне-розміщення-type-annotations",[2968,6767,6768],{},"formatAnnotations()"," — коректне розміщення type annotations",[3092,6771,6773],{"className":3465,"code":6772,"language":3467,"meta":3100,"style":3100},"// Gradle\nspotless {\n    java {\n        formatAnnotations()\n    }\n}\n",[2968,6774,6775,6779,6783,6787,6791,6795],{"__ignoreMap":3100},[3346,6776,6777],{"class":3348,"line":3349},[3346,6778,5339],{},[3346,6780,6781],{"class":3348,"line":3355},[3346,6782,3503],{},[3346,6784,6785],{"class":3348,"line":3361},[3346,6786,3532],{},[3346,6788,6789],{"class":3348,"line":3367},[3346,6790,3710],{},[3346,6792,6793],{"class":3348,"line":3373},[3346,6794,3739],{},[3346,6796,6797],{"class":3348,"line":3379},[3346,6798,3494],{},[3092,6800,6802],{"className":4065,"code":6801,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven -->\n\u003CformatAnnotations/>\n",[2968,6803,6804,6808],{"__ignoreMap":3100},[3346,6805,6806],{"class":3348,"line":3349},[3346,6807,5375],{"class":3997},[3346,6809,6810,6812,6814],{"class":3348,"line":3355},[3346,6811,4080],{"class":4079},[3346,6813,4472],{"class":4083},[3346,6815,4442],{"class":4079},[2964,6817,6818,6821,6822,4919,6825,4919,6828,4919,6831,6834,6835,6837],{},[2979,6819,6820],{},"Яку проблему вирішує:"," google-java-format іноді некоректно розміщує type annotations (",[2968,6823,6824],{},"@NonNull",[2968,6826,6827],{},"@Nullable",[2968,6829,6830],{},"@NotNull",[2968,6832,6833],{},"@Nonnull","). Крок ",[2968,6836,6768],{}," виправляє їх позицію відповідно до специфікації JSR 308.",[2964,6839,6840],{},[2979,6841,6842],{},"Приклад виправлення:",[3092,6844,6847],{"className":6845,"code":6846,"language":4179,"meta":3100,"style":3100},"language-java shiki shiki-themes light-plus dark-plus dark-plus","// До formatAnnotations() — некоректно\n@NonNull String\ngetTitle() { ... }\n\n// Після formatAnnotations() — коректно\n@NonNull\nString getTitle() { ... }\n",[2968,6848,6849,6854,6866,6874,6878,6883,6890],{"__ignoreMap":3100},[3346,6850,6851],{"class":3348,"line":3349},[3346,6852,6853],{"class":3997},"// До formatAnnotations() — некоректно\n",[3346,6855,6856,6859,6863],{"class":3348,"line":3355},[3346,6857,6858],{"class":4101},"@",[3346,6860,6862],{"class":6861},"sN1BT","NonNull",[3346,6864,6865],{"class":6861}," String\n",[3346,6867,6868,6871],{"class":3348,"line":3361},[3346,6869,6870],{"class":4003},"getTitle",[3346,6872,6873],{"class":4101},"() { ... }\n",[3346,6875,6876],{"class":3348,"line":3367},[3346,6877,3395],{"emptyLinePlaceholder":3394},[3346,6879,6880],{"class":3348,"line":3373},[3346,6881,6882],{"class":3997},"// Після formatAnnotations() — коректно\n",[3346,6884,6885,6887],{"class":3348,"line":3379},[3346,6886,6858],{"class":4101},[3346,6888,6889],{"class":6861},"NonNull\n",[3346,6891,6892,6894,6897],{"class":3348,"line":3385},[3346,6893,5249],{"class":6861},[3346,6895,6896],{"class":4003}," getTitle",[3346,6898,6873],{"class":4101},[3070,6900],{},[3084,6902,6904],{"id":6903},"базові-кроки-для-всіх-форматів","Базові кроки для всіх форматів",[2964,6906,6907],{},"Ці кроки застосовуються до будь-якого типу файлів, не лише Java:",[5323,6909,6911,6914],{"id":6910},"trimtrailingwhitespace-видалення-пробілів-у-кінці-рядків",[2968,6912,6913],{},"trimTrailingWhitespace()"," — видалення пробілів у кінці рядків",[3092,6916,6918],{"className":3465,"code":6917,"language":3467,"meta":3100,"style":3100},"spotless {\n    java {\n        trimTrailingWhitespace()   // видалити всі trailing spaces/tabs\n    }\n}\n",[2968,6919,6920,6924,6928,6933,6937],{"__ignoreMap":3100},[3346,6921,6922],{"class":3348,"line":3349},[3346,6923,3503],{},[3346,6925,6926],{"class":3348,"line":3355},[3346,6927,3532],{},[3346,6929,6930],{"class":3348,"line":3361},[3346,6931,6932],{},"        trimTrailingWhitespace()   // видалити всі trailing spaces/tabs\n",[3346,6934,6935],{"class":3348,"line":3367},[3346,6936,3739],{},[3346,6938,6939],{"class":3348,"line":3373},[3346,6940,3494],{},[2964,6942,6943,6946],{},[2979,6944,6945],{},"Чому важливо:"," trailing whitespace — класичне джерело diff-шуму. Git показує їх як зміни навіть якщо логічний вміст рядка не змінився.",[3070,6948],{},[5323,6950,6952,6955],{"id":6951},"endwithnewline-файл-завершується-символом-нового-рядка",[2968,6953,6954],{},"endWithNewline()"," — файл завершується символом нового рядка",[3092,6957,6959],{"className":3465,"code":6958,"language":3467,"meta":3100,"style":3100},"spotless {\n    java {\n        endWithNewline()    // файл має закінчуватися '\\n'\n    }\n}\n",[2968,6960,6961,6965,6969,6974,6978],{"__ignoreMap":3100},[3346,6962,6963],{"class":3348,"line":3349},[3346,6964,3503],{},[3346,6966,6967],{"class":3348,"line":3355},[3346,6968,3532],{},[3346,6970,6971],{"class":3348,"line":3361},[3346,6972,6973],{},"        endWithNewline()    // файл має закінчуватися '\\n'\n",[3346,6975,6976],{"class":3348,"line":3367},[3346,6977,3739],{},[3346,6979,6980],{"class":3348,"line":3373},[3346,6981,3494],{},[2964,6983,6984,6987,6988,4919,6991,4919,6994,6997],{},[2979,6985,6986],{},"Стандарт POSIX:"," кожен текстовий файл повинен завершуватися символом нового рядка. Без цього деякі UNIX-утиліти (",[2968,6989,6990],{},"cat",[2968,6992,6993],{},"tail",[2968,6995,6996],{},"wc",") дають несподівані результати, а git показує «no newline at end of file».",[3070,6999],{},[5323,7001,7003,7006],{"id":7002},"indent-заміна-відступів-рідко-потрібно-разом-з-gjf",[2968,7004,7005],{},"indent"," — заміна відступів (рідко потрібно разом з GJF)",[3092,7008,7010],{"className":3465,"code":7009,"language":3467,"meta":3100,"style":3100},"spotless {\n    format 'misc', {                 // зазвичай у misc, не у java\n        leadingTabsToSpaces(2)       // замінити leading tabs на 2 пробіли\n        // leadingSpacesToTabs()     // замінити leading spaces на tabs\n        // leadingTabsToSpaces(4)    // замінити tabs на 4 пробіли\n    }\n}\n",[2968,7011,7012,7016,7021,7026,7034,7042,7046],{"__ignoreMap":3100},[3346,7013,7014],{"class":3348,"line":3349},[3346,7015,3503],{},[3346,7017,7018],{"class":3348,"line":3355},[3346,7019,7020],{},"    format 'misc', {                 // зазвичай у misc, не у java\n",[3346,7022,7023],{"class":3348,"line":3361},[3346,7024,7025],{},"        leadingTabsToSpaces(2)       // замінити leading tabs на 2 пробіли\n",[3346,7027,7028,7031],{"class":3348,"line":3367},[3346,7029,7030],{},"        // leadingSpacesToTabs()",[3346,7032,7033],{},"     // замінити leading spaces на tabs\n",[3346,7035,7036,7039],{"class":3348,"line":3373},[3346,7037,7038],{},"        // leadingTabsToSpaces(4)",[3346,7040,7041],{},"    // замінити tabs на 4 пробіли\n",[3346,7043,7044],{"class":3348,"line":3379},[3346,7045,3739],{},[3346,7047,7048],{"class":3348,"line":3385},[3346,7049,3494],{},[2964,7051,7052,7057,7058,7060,7061,4919,7064,4919,7067,7070,7071,3068],{},[2979,7053,7054,7055],{},"Не потрібно у блоці ",[2968,7056,5530],{}," при використанні ",[2968,7059,5723],{}," — GJF вже встановлює правильні відступи. Корисно для ",[2968,7062,7063],{},".gradle",[2968,7065,7066],{},".yaml",[2968,7068,7069],{},".xml"," у блоці ",[2968,7072,7073],{},"misc",[3070,7075],{},[3084,7077,7079],{"id":7078},"заголовки-ліцензій","Заголовки ліцензій",[5323,7081,7083,7086],{"id":7082},"licenseheader-автоматична-вставка-заголовку-ліцензії",[2968,7084,7085],{},"licenseHeader"," — автоматична вставка заголовку ліцензії",[3092,7088,7090],{"className":3465,"code":7089,"language":3467,"meta":3100,"style":3100},"// Gradle — рядковий literal\nspotless {\n    java {\n        licenseHeader '/* Copyright (C) $YEAR Example Corp. All rights reserved. */'\n        // $YEAR — автоматично підставляється рік (з git history або поточний)\n    }\n}\n\n// Gradle — із зовнішнього файлу\nspotless {\n    java {\n        licenseHeaderFile 'config/license-header.txt'\n        // Або з явним роздільником (за замовчуванням Spotless визначає сам)\n        licenseHeaderFile('config/license-header.txt', 'package ')\n    }\n}\n",[2968,7091,7092,7097,7101,7105,7110,7115,7119,7123,7127,7132,7136,7140,7145,7150,7155,7159],{"__ignoreMap":3100},[3346,7093,7094],{"class":3348,"line":3349},[3346,7095,7096],{},"// Gradle — рядковий literal\n",[3346,7098,7099],{"class":3348,"line":3355},[3346,7100,3503],{},[3346,7102,7103],{"class":3348,"line":3361},[3346,7104,3532],{},[3346,7106,7107],{"class":3348,"line":3367},[3346,7108,7109],{},"        licenseHeader '/* Copyright (C) $YEAR Example Corp. All rights reserved. */'\n",[3346,7111,7112],{"class":3348,"line":3373},[3346,7113,7114],{},"        // $YEAR — автоматично підставляється рік (з git history або поточний)\n",[3346,7116,7117],{"class":3348,"line":3379},[3346,7118,3739],{},[3346,7120,7121],{"class":3348,"line":3385},[3346,7122,3494],{},[3346,7124,7125],{"class":3348,"line":3391},[3346,7126,3395],{"emptyLinePlaceholder":3394},[3346,7128,7129],{"class":3348,"line":3398},[3346,7130,7131],{},"// Gradle — із зовнішнього файлу\n",[3346,7133,7134],{"class":3348,"line":3404},[3346,7135,3503],{},[3346,7137,7138],{"class":3348,"line":3410},[3346,7139,3532],{},[3346,7141,7142],{"class":3348,"line":3416},[3346,7143,7144],{},"        licenseHeaderFile 'config/license-header.txt'\n",[3346,7146,7147],{"class":3348,"line":3422},[3346,7148,7149],{},"        // Або з явним роздільником (за замовчуванням Spotless визначає сам)\n",[3346,7151,7152],{"class":3348,"line":3427},[3346,7153,7154],{},"        licenseHeaderFile('config/license-header.txt', 'package ')\n",[3346,7156,7157],{"class":3348,"line":3433},[3346,7158,3739],{},[3346,7160,7161],{"class":3348,"line":3439},[3346,7162,3494],{},[3092,7164,7166],{"className":4065,"code":7165,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven -->\n\u003ClicenseHeader>\n    \u003Ccontent>/* Copyright (C) $YEAR Example Corp. All rights reserved. */\u003C/content>\n\u003C/licenseHeader>\n\n\u003C!-- або з файлу -->\n\u003ClicenseHeader>\n    \u003Cfile>${project.basedir}/config/license-header.txt\u003C/file>\n    \u003Cdelimiter>package \u003C/delimiter>\n\u003C/licenseHeader>\n",[2968,7167,7168,7172,7180,7198,7206,7210,7215,7223,7241,7259],{"__ignoreMap":3100},[3346,7169,7170],{"class":3348,"line":3349},[3346,7171,5375],{"class":3997},[3346,7173,7174,7176,7178],{"class":3348,"line":3355},[3346,7175,4080],{"class":4079},[3346,7177,7085],{"class":4083},[3346,7179,4087],{"class":4079},[3346,7181,7182,7184,7187,7189,7192,7194,7196],{"class":3348,"line":3361},[3346,7183,4092],{"class":4079},[3346,7185,7186],{"class":4083},"content",[3346,7188,4098],{"class":4079},[3346,7190,7191],{"class":4101},"/* Copyright (C) $YEAR Example Corp. All rights reserved. */",[3346,7193,4105],{"class":4079},[3346,7195,7186],{"class":4083},[3346,7197,4087],{"class":4079},[3346,7199,7200,7202,7204],{"class":3348,"line":3367},[3346,7201,4105],{"class":4079},[3346,7203,7085],{"class":4083},[3346,7205,4087],{"class":4079},[3346,7207,7208],{"class":3348,"line":3373},[3346,7209,3395],{"emptyLinePlaceholder":3394},[3346,7211,7212],{"class":3348,"line":3379},[3346,7213,7214],{"class":3997},"\u003C!-- або з файлу -->\n",[3346,7216,7217,7219,7221],{"class":3348,"line":3385},[3346,7218,4080],{"class":4079},[3346,7220,7085],{"class":4083},[3346,7222,4087],{"class":4079},[3346,7224,7225,7227,7230,7232,7235,7237,7239],{"class":3348,"line":3391},[3346,7226,4092],{"class":4079},[3346,7228,7229],{"class":4083},"file",[3346,7231,4098],{"class":4079},[3346,7233,7234],{"class":4101},"${project.basedir}/config/license-header.txt",[3346,7236,4105],{"class":4079},[3346,7238,7229],{"class":4083},[3346,7240,4087],{"class":4079},[3346,7242,7243,7245,7248,7250,7253,7255,7257],{"class":3348,"line":3398},[3346,7244,4092],{"class":4079},[3346,7246,7247],{"class":4083},"delimiter",[3346,7249,4098],{"class":4079},[3346,7251,7252],{"class":4101},"package ",[3346,7254,4105],{"class":4079},[3346,7256,7247],{"class":4083},[3346,7258,4087],{"class":4079},[3346,7260,7261,7263,7265],{"class":3348,"line":3404},[3346,7262,4105],{"class":4079},[3346,7264,7085],{"class":4083},[3346,7266,4087],{"class":4079},[2964,7268,7269],{},[2979,7270,6409,7271,6549],{},[2968,7272,7273],{},"config/license-header.txt",[3092,7275,7278],{"className":7276,"code":7277,"language":3097},[3095],"/*\n * Copyright (C) $YEAR Example Corp.\n *\n * Licensed under the Apache License, Version 2.0.\n */\n",[2968,7279,7277],{"__ignoreMap":3100},[2964,7281,7282,7287,7288,7291,7292,3068],{},[2979,7283,7284],{},[2968,7285,7286],{},"$YEAR"," — Spotless підставляє рік першого commit файлу з git history (або поточний рік для нових файлів). Це означає, що заголовок у старих файлах матиме ",[2968,7289,7290],{},"2023",", а у нових — ",[2968,7293,7294],{},"2026",[3070,7296],{},[3084,7298,7300],{"id":7299},"форматування-інших-типів-файлів","Форматування інших типів файлів",[5323,7302,7304,7307],{"id":7303},"format-misc-довільні-формати",[2968,7305,7306],{},"format 'misc', {}"," — довільні формати",[3092,7309,7311],{"className":3465,"code":7310,"language":3467,"meta":3100,"style":3100},"// Gradle\nspotless {\n    format 'misc', {\n        // Вибір файлів\n        target '*.gradle', '*.md', '*.yml', '*.yaml',\n               '.gitignore', '.gitattributes', '.editorconfig'\n\n        // Виключення\n        targetExclude 'build/**', '.gradle/**'\n\n        // Кроки форматування (ті самі, що доступні у java {})\n        trimTrailingWhitespace()\n        endWithNewline()\n        leadingTabsToSpaces(2)     // якщо yaml потребує пробілів\n    }\n\n    // Можна мати кілька форматів\n    format 'xml', {\n        target 'src/**/*.xml', 'config/**/*.xml'\n        trimTrailingWhitespace()\n        endWithNewline()\n        // Для XML можна підключити prettier або eclipse XML formatter\n    }\n}\n",[2968,7312,7313,7317,7321,7325,7330,7335,7340,7344,7349,7354,7358,7363,7367,7371,7376,7380,7384,7389,7394,7399,7403,7407,7412,7416],{"__ignoreMap":3100},[3346,7314,7315],{"class":3348,"line":3349},[3346,7316,5339],{},[3346,7318,7319],{"class":3348,"line":3355},[3346,7320,3503],{},[3346,7322,7323],{"class":3348,"line":3361},[3346,7324,3756],{},[3346,7326,7327],{"class":3348,"line":3367},[3346,7328,7329],{},"        // Вибір файлів\n",[3346,7331,7332],{"class":3348,"line":3373},[3346,7333,7334],{},"        target '*.gradle', '*.md', '*.yml', '*.yaml',\n",[3346,7336,7337],{"class":3348,"line":3379},[3346,7338,7339],{},"               '.gitignore', '.gitattributes', '.editorconfig'\n",[3346,7341,7342],{"class":3348,"line":3385},[3346,7343,3395],{"emptyLinePlaceholder":3394},[3346,7345,7346],{"class":3348,"line":3391},[3346,7347,7348],{},"        // Виключення\n",[3346,7350,7351],{"class":3348,"line":3398},[3346,7352,7353],{},"        targetExclude 'build/**', '.gradle/**'\n",[3346,7355,7356],{"class":3348,"line":3404},[3346,7357,3395],{"emptyLinePlaceholder":3394},[3346,7359,7360],{"class":3348,"line":3410},[3346,7361,7362],{},"        // Кроки форматування (ті самі, що доступні у java {})\n",[3346,7364,7365],{"class":3348,"line":3416},[3346,7366,3768],{},[3346,7368,7369],{"class":3348,"line":3422},[3346,7370,3774],{},[3346,7372,7373],{"class":3348,"line":3427},[3346,7374,7375],{},"        leadingTabsToSpaces(2)     // якщо yaml потребує пробілів\n",[3346,7377,7378],{"class":3348,"line":3433},[3346,7379,3739],{},[3346,7381,7382],{"class":3348,"line":3439},[3346,7383,3395],{"emptyLinePlaceholder":3394},[3346,7385,7386],{"class":3348,"line":3550},[3346,7387,7388],{},"    // Можна мати кілька форматів\n",[3346,7390,7391],{"class":3348,"line":3555},[3346,7392,7393],{},"    format 'xml', {\n",[3346,7395,7396],{"class":3348,"line":3561},[3346,7397,7398],{},"        target 'src/**/*.xml', 'config/**/*.xml'\n",[3346,7400,7401],{"class":3348,"line":3567},[3346,7402,3768],{},[3346,7404,7405],{"class":3348,"line":3573},[3346,7406,3774],{},[3346,7408,7409],{"class":3348,"line":3579},[3346,7410,7411],{},"        // Для XML можна підключити prettier або eclipse XML formatter\n",[3346,7413,7414],{"class":3348,"line":3585},[3346,7415,3739],{},[3346,7417,7418],{"class":3348,"line":3591},[3346,7419,3494],{},[3092,7421,7423],{"className":4065,"code":7422,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven -->\n\u003Cformats>\n    \u003Cformat>\n        \u003Cincludes>\n            \u003Cinclude>*.md\u003C/include>\n            \u003Cinclude>*.yml\u003C/include>\n            \u003Cinclude>.gitignore\u003C/include>\n        \u003C/includes>\n        \u003CtrimTrailingWhitespace/>\n        \u003CendWithNewline/>\n    \u003C/format>\n\u003C/formats>\n",[2968,7424,7425,7429,7437,7445,7453,7469,7486,7502,7510,7518,7526,7534],{"__ignoreMap":3100},[3346,7426,7427],{"class":3348,"line":3349},[3346,7428,5375],{"class":3997},[3346,7430,7431,7433,7435],{"class":3348,"line":3355},[3346,7432,4080],{"class":4079},[3346,7434,4521],{"class":4083},[3346,7436,4087],{"class":4079},[3346,7438,7439,7441,7443],{"class":3348,"line":3361},[3346,7440,4092],{"class":4079},[3346,7442,4530],{"class":4083},[3346,7444,4087],{"class":4079},[3346,7446,7447,7449,7451],{"class":3348,"line":3367},[3346,7448,4176],{"class":4079},[3346,7450,4194],{"class":4083},[3346,7452,4087],{"class":4079},[3346,7454,7455,7457,7459,7461,7463,7465,7467],{"class":3348,"line":3373},[3346,7456,4191],{"class":4079},[3346,7458,4204],{"class":4083},[3346,7460,4098],{"class":4079},[3346,7462,4569],{"class":4101},[3346,7464,4105],{"class":4079},[3346,7466,4204],{"class":4083},[3346,7468,4087],{"class":4079},[3346,7470,7471,7473,7475,7477,7480,7482,7484],{"class":3348,"line":3379},[3346,7472,4191],{"class":4079},[3346,7474,4204],{"class":4083},[3346,7476,4098],{"class":4079},[3346,7478,7479],{"class":4101},"*.yml",[3346,7481,4105],{"class":4079},[3346,7483,4204],{"class":4083},[3346,7485,4087],{"class":4079},[3346,7487,7488,7490,7492,7494,7496,7498,7500],{"class":3348,"line":3385},[3346,7489,4191],{"class":4079},[3346,7491,4204],{"class":4083},[3346,7493,4098],{"class":4079},[3346,7495,4586],{"class":4101},[3346,7497,4105],{"class":4079},[3346,7499,4204],{"class":4083},[3346,7501,4087],{"class":4079},[3346,7503,7504,7506,7508],{"class":3348,"line":3391},[3346,7505,4501],{"class":4079},[3346,7507,4194],{"class":4083},[3346,7509,4087],{"class":4079},[3346,7511,7512,7514,7516],{"class":3348,"line":3398},[3346,7513,4176],{"class":4079},[3346,7515,4485],{"class":4083},[3346,7517,4442],{"class":4079},[3346,7519,7520,7522,7524],{"class":3348,"line":3404},[3346,7521,4176],{"class":4079},[3346,7523,4494],{"class":4083},[3346,7525,4442],{"class":4079},[3346,7527,7528,7530,7532],{"class":3348,"line":3410},[3346,7529,4656],{"class":4079},[3346,7531,4530],{"class":4083},[3346,7533,4087],{"class":4079},[3346,7535,7536,7538,7540],{"class":3348,"line":3416},[3346,7537,4105],{"class":4079},[3346,7539,4521],{"class":4083},[3346,7541,4087],{"class":4079},[3070,7543],{},[3084,7545,7547,7548],{"id":7546},"очищення-коду-cleanthat","Очищення коду: ",[2968,7549,7550],{},"cleanthat",[2964,7552,7553,7556,7557,7560],{},[2979,7554,7555],{},"CleanThat"," — крок, що виконує автоматичний рефакторинг Java-коду: замінює застарілі конструкції на сучасніші еквіваленти. Застосовується ",[2979,7558,7559],{},"до"," основного форматера.",[3092,7562,7564],{"className":3465,"code":7563,"language":3467,"meta":3100,"style":3100},"// Gradle\nspotless {\n    java {\n        cleanthat()\n            // Конкретні мутатори (правила рефакторингу)\n            .addMutator('UnnecessaryModifier')    // видалити зайві модифікатори\n            .addMutator('StringReplaceAll')        // String.replaceAll → String.replace де можливо\n            .addMutator('PrimitiveWrapperInstantiation') // new Integer(x) → Integer.valueOf(x)\n\n        googleJavaFormat('1.35.0')  // CleanThat запускається першим, потім GJF\n    }\n}\n",[2968,7565,7566,7570,7574,7578,7583,7588,7593,7598,7603,7607,7612,7616],{"__ignoreMap":3100},[3346,7567,7568],{"class":3348,"line":3349},[3346,7569,5339],{},[3346,7571,7572],{"class":3348,"line":3355},[3346,7573,3503],{},[3346,7575,7576],{"class":3348,"line":3361},[3346,7577,3532],{},[3346,7579,7580],{"class":3348,"line":3367},[3346,7581,7582],{},"        cleanthat()\n",[3346,7584,7585],{"class":3348,"line":3373},[3346,7586,7587],{},"            // Конкретні мутатори (правила рефакторингу)\n",[3346,7589,7590],{"class":3348,"line":3379},[3346,7591,7592],{},"            .addMutator('UnnecessaryModifier')    // видалити зайві модифікатори\n",[3346,7594,7595],{"class":3348,"line":3385},[3346,7596,7597],{},"            .addMutator('StringReplaceAll')        // String.replaceAll → String.replace де можливо\n",[3346,7599,7600],{"class":3348,"line":3391},[3346,7601,7602],{},"            .addMutator('PrimitiveWrapperInstantiation') // new Integer(x) → Integer.valueOf(x)\n",[3346,7604,7605],{"class":3348,"line":3398},[3346,7606,3395],{"emptyLinePlaceholder":3394},[3346,7608,7609],{"class":3348,"line":3404},[3346,7610,7611],{},"        googleJavaFormat('1.35.0')  // CleanThat запускається першим, потім GJF\n",[3346,7613,7614],{"class":3348,"line":3410},[3346,7615,3739],{},[3346,7617,7618],{"class":3348,"line":3416},[3346,7619,3494],{},[3092,7621,7623],{"className":4065,"code":7622,"language":4067,"meta":3100,"style":3100},"\u003C!-- Maven -->\n\u003Ccleanthat>\n    \u003Cmutators>\n        \u003Cmutator>UnnecessaryModifier\u003C/mutator>\n        \u003Cmutator>StringReplaceAll\u003C/mutator>\n    \u003C/mutators>\n\u003C/cleanthat>\n\u003CgoogleJavaFormat>\n    \u003Cversion>1.35.0\u003C/version>\n\u003C/googleJavaFormat>\n",[2968,7624,7625,7629,7637,7646,7664,7681,7689,7697,7705,7721],{"__ignoreMap":3100},[3346,7626,7627],{"class":3348,"line":3349},[3346,7628,5375],{"class":3997},[3346,7630,7631,7633,7635],{"class":3348,"line":3355},[3346,7632,4080],{"class":4079},[3346,7634,7550],{"class":4083},[3346,7636,4087],{"class":4079},[3346,7638,7639,7641,7644],{"class":3348,"line":3361},[3346,7640,4092],{"class":4079},[3346,7642,7643],{"class":4083},"mutators",[3346,7645,4087],{"class":4079},[3346,7647,7648,7650,7653,7655,7658,7660,7662],{"class":3348,"line":3367},[3346,7649,4176],{"class":4079},[3346,7651,7652],{"class":4083},"mutator",[3346,7654,4098],{"class":4079},[3346,7656,7657],{"class":4101},"UnnecessaryModifier",[3346,7659,4105],{"class":4079},[3346,7661,7652],{"class":4083},[3346,7663,4087],{"class":4079},[3346,7665,7666,7668,7670,7672,7675,7677,7679],{"class":3348,"line":3373},[3346,7667,4176],{"class":4079},[3346,7669,7652],{"class":4083},[3346,7671,4098],{"class":4079},[3346,7673,7674],{"class":4101},"StringReplaceAll",[3346,7676,4105],{"class":4079},[3346,7678,7652],{"class":4083},[3346,7680,4087],{"class":4079},[3346,7682,7683,7685,7687],{"class":3348,"line":3379},[3346,7684,4656],{"class":4079},[3346,7686,7643],{"class":4083},[3346,7688,4087],{"class":4079},[3346,7690,7691,7693,7695],{"class":3348,"line":3385},[3346,7692,4105],{"class":4079},[3346,7694,7550],{"class":4083},[3346,7696,4087],{"class":4079},[3346,7698,7699,7701,7703],{"class":3348,"line":3391},[3346,7700,4080],{"class":4079},[3346,7702,4267],{"class":4083},[3346,7704,4087],{"class":4079},[3346,7706,7707,7709,7711,7713,7715,7717,7719],{"class":3348,"line":3398},[3346,7708,4092],{"class":4079},[3346,7710,4134],{"class":4083},[3346,7712,4098],{"class":4079},[3346,7714,3059],{"class":4101},[3346,7716,4105],{"class":4079},[3346,7718,4134],{"class":4083},[3346,7720,4087],{"class":4079},[3346,7722,7723,7725,7727],{"class":3348,"line":3404},[3346,7724,4105],{"class":4079},[3346,7726,4267],{"class":4083},[3346,7728,4087],{"class":4079},[5215,7730,7731,7732,7735,7736,7739,7740,3068],{},"CleanThat може змінити логіку коду (наприклад, ",[2968,7733,7734],{},"replaceAll"," з regex на ",[2968,7737,7738],{},"replace"," без regex — різна семантика). Вмикайте мутатори обережно і завжди перевіряйте diff після першого ",[2968,7741,6685],{},[3070,7743],{},[3084,7745,7747,7748,7750],{"id":7746},"довідкова-таблиця-всі-параметри-java-одним-поглядом","Довідкова таблиця: всі параметри ",[2968,7749,5530],{}," одним поглядом",[3181,7752,7753,7767],{},[3184,7754,7755],{},[3187,7756,7757,7760,7762,7764],{},[3190,7758,7759],{},"Крок",[3190,7761,4880],{},[3190,7763,5064],{},[3190,7765,7766],{},"Що робить",[3200,7768,7769,7787,7808,7826,7844,7862,7879,7897,7914,7933,7953,7970,7987,8004,8021,8039,8057,8075],{},[3187,7770,7771,7774,7779,7784],{},[3205,7772,7773],{},"Форматер",[3205,7775,7776],{},[2968,7777,7778],{},"googleJavaFormat('1.35.0')",[3205,7780,7781],{},[2968,7782,7783],{},"\u003CgoogleJavaFormat>",[3205,7785,7786],{},"Основне форматування",[3187,7788,7789,7792,7800,7805],{},[3205,7790,7791],{},"Стиль GJF",[3205,7793,7794,5538,7797],{},[2968,7795,7796],{},".style('GOOGLE')",[2968,7798,7799],{},".aosp()",[3205,7801,7802],{},[2968,7803,7804],{},"\u003Cstyle>GOOGLE\u003C/style>",[3205,7806,7807],{},"2 або 4 пробіли",[3187,7809,7810,7813,7818,7823],{},[3205,7811,7812],{},"Довгі рядки",[3205,7814,7815],{},[2968,7816,7817],{},".reflowLongStrings()",[3205,7819,7820],{},[2968,7821,7822],{},"\u003CreflowLongStrings>true",[3205,7824,7825],{},"Перенос рядкових літералів",[3187,7827,7828,7831,7836,7841],{},[3205,7829,7830],{},"Javadoc",[3205,7832,7833],{},[2968,7834,7835],{},".formatJavadoc(true)",[3205,7837,7838],{},[2968,7839,7840],{},"\u003CformatJavadoc>true",[3205,7842,7843],{},"Форматувати коментарі",[3187,7845,7846,7849,7854,7859],{},[3205,7847,7848],{},"Imports GJF",[3205,7850,7851],{},[2968,7852,7853],{},".reorderImports(false)",[3205,7855,7856],{},[2968,7857,7858],{},"\u003CreorderImports>false",[3205,7860,7861],{},"Відключити вбудований sort",[3187,7863,7864,7867,7871,7876],{},[3205,7865,7866],{},"Порядок imports",[3205,7868,7869],{},[2968,7870,5276],{},[3205,7872,7873],{},[2968,7874,7875],{},"\u003CimportOrder>\u003Corder>",[3205,7877,7878],{},"Групи imports",[3187,7880,7881,7884,7889,7894],{},[3205,7882,7883],{},"Файл порядку",[3205,7885,7886],{},[2968,7887,7888],{},"importOrderFile('file')",[3205,7890,7891],{},[2968,7892,7893],{},"\u003CimportOrder>\u003Cfile>",[3205,7895,7896],{},"Порядок з файлу Eclipse",[3187,7898,7899,7902,7906,7911],{},[3205,7900,7901],{},"Видалення imports",[3205,7903,7904],{},[2968,7905,6431],{},[3205,7907,7908],{},[2968,7909,7910],{},"\u003CremoveUnusedImports/>",[3205,7912,7913],{},"Прибрати невикористані",[3187,7915,7916,7919,7923,7928],{},[3205,7917,7918],{},"Wildcard заборона",[3205,7920,7921],{},[2968,7922,6561],{},[3205,7924,7925],{},[2968,7926,7927],{},"\u003CforbidWildcardImports/>",[3205,7929,7930,7931],{},"Блокувати ",[2968,7932,3279],{},[3187,7934,7935,7938,7942,7947],{},[3205,7936,7937],{},"Wildcard розгортання",[3205,7939,7940],{},[2968,7941,6626],{},[3205,7943,7944],{},[2968,7945,7946],{},"\u003CexpandWildcardImports/>",[3205,7948,7949,7952],{},[2968,7950,7951],{},"java.util.*"," → explicit",[3187,7954,7955,7958,7962,7967],{},[3205,7956,7957],{},"Module imports",[3205,7959,7960],{},[2968,7961,6696],{},[3205,7963,7964],{},[2968,7965,7966],{},"\u003CforbidModuleImports/>",[3205,7968,7969],{},"Java 25+: заборона module import",[3187,7971,7972,7975,7979,7984],{},[3205,7973,7974],{},"Анотації",[3205,7976,7977],{},[2968,7978,6768],{},[3205,7980,7981],{},[2968,7982,7983],{},"\u003CformatAnnotations/>",[3205,7985,7986],{},"Коректне розміщення @NonNull тощо",[3187,7988,7989,7992,7996,8001],{},[3205,7990,7991],{},"Пробіли в кінці",[3205,7993,7994],{},[2968,7995,6913],{},[3205,7997,7998],{},[2968,7999,8000],{},"\u003CtrimTrailingWhitespace/>",[3205,8002,8003],{},"Прибрати trailing spaces",[3187,8005,8006,8009,8013,8018],{},[3205,8007,8008],{},"Новий рядок",[3205,8010,8011],{},[2968,8012,6954],{},[3205,8014,8015],{},[2968,8016,8017],{},"\u003CendWithNewline/>",[3205,8019,8020],{},"EOF newline",[3187,8022,8023,8026,8031,8036],{},[3205,8024,8025],{},"Заголовок ліцензії",[3205,8027,8028],{},[2968,8029,8030],{},"licenseHeader '...'",[3205,8032,8033],{},[2968,8034,8035],{},"\u003ClicenseHeader>\u003Ccontent>",[3205,8037,8038],{},"Вставити copyright",[3187,8040,8041,8044,8049,8054],{},[3205,8042,8043],{},"Рефакторинг",[3205,8045,8046],{},[2968,8047,8048],{},"cleanthat()",[3205,8050,8051],{},[2968,8052,8053],{},"\u003Ccleanthat>",[3205,8055,8056],{},"Авто-рефакторинг конструкцій",[3187,8058,8059,8062,8067,8072],{},[3205,8060,8061],{},"Ціль",[3205,8063,8064],{},[2968,8065,8066],{},"target 'src/**/*.java'",[3205,8068,8069],{},[2968,8070,8071],{},"\u003Cincludes>\u003Cinclude>",[3205,8073,8074],{},"Обрані файли",[3187,8076,8077,8080,8085,8090],{},[3205,8078,8079],{},"Виключення",[3205,8081,8082],{},[2968,8083,8084],{},"targetExclude 'src/gen/**'",[3205,8086,8087],{},[2968,8088,8089],{},"\u003Cexcludes>\u003Cexclude>",[3205,8091,8092],{},"Файли поза перевіркою",[3070,8094],{},[3084,8096,8098],{"id":8097},"довідкова-таблиця-gradle-tasks-та-maven-goals","Довідкова таблиця: Gradle tasks та Maven goals",[3457,8100,8101,8210],{},[3460,8102,8104,8190],{"label":8103},"Gradle tasks",[3181,8105,8106,8115],{},[3184,8107,8108],{},[3187,8109,8110,8113],{},[3190,8111,8112],{},"Task",[3190,8114,7766],{},[3200,8116,8117,8126,8135,8145,8155,8168,8180],{},[3187,8118,8119,8123],{},[3205,8120,8121],{},[2968,8122,6685],{},[3205,8124,8125],{},"Виправити форматування всіх налаштованих форматів",[3187,8127,8128,8132],{},[3205,8129,8130],{},[2968,8131,5231],{},[3205,8133,8134],{},"Перевірити всі формати (не змінює файли; для CI)",[3187,8136,8137,8142],{},[3205,8138,8139],{},[2968,8140,8141],{},"spotlessJavaApply",[3205,8143,8144],{},"Виправити тільки Java-файли",[3187,8146,8147,8152],{},[3205,8148,8149],{},[2968,8150,8151],{},"spotlessJavaCheck",[3205,8153,8154],{},"Перевірити тільки Java-файли",[3187,8156,8157,8162],{},[3205,8158,8159],{},[2968,8160,8161],{},"spotlessMiscApply",[3205,8163,8164,8165,8167],{},"Виправити тільки ",[2968,8166,7073],{},"-формат",[3187,8169,8170,8175],{},[3205,8171,8172],{},[2968,8173,8174],{},"spotlessMiscCheck",[3205,8176,8177,8178,8167],{},"Перевірити тільки ",[2968,8179,7073],{},[3187,8181,8182,8187],{},[3205,8183,8184],{},[2968,8185,8186],{},"spotlessDiagnose",[3205,8188,8189],{},"Показати детальну інформацію про конфігурацію",[3092,8191,8193],{"className":3988,"code":8192,"language":3990,"meta":3100,"style":3100},"# Перегляд всіх Spotless tasks\n./gradlew tasks --group=spotless\n",[2968,8194,8195,8200],{"__ignoreMap":3100},[3346,8196,8197],{"class":3348,"line":3349},[3346,8198,8199],{"class":3997},"# Перегляд всіх Spotless tasks\n",[3346,8201,8202,8204,8207],{"class":3348,"line":3355},[3346,8203,4004],{"class":4003},[3346,8205,8206],{"class":4007}," tasks",[3346,8208,8209],{"class":4909}," --group=spotless\n",[3460,8211,8213,8256],{"label":8212},"Maven goals",[3181,8214,8215,8224],{},[3184,8216,8217],{},[3187,8218,8219,8222],{},[3190,8220,8221],{},"Goal",[3190,8223,7766],{},[3200,8225,8226,8236,8246],{},[3187,8227,8228,8233],{},[3205,8229,8230],{},[2968,8231,8232],{},"spotless:apply",[3205,8234,8235],{},"Виправити форматування",[3187,8237,8238,8243],{},[3205,8239,8240],{},[2968,8241,8242],{},"spotless:check",[3205,8244,8245],{},"Перевірити (для CI; не змінює файли)",[3187,8247,8248,8253],{},[3205,8249,8250],{},[2968,8251,8252],{},"spotless:diagnose",[3205,8254,8255],{},"Показати деталі конфігурації",[3092,8257,8259],{"className":3988,"code":8258,"language":3990,"meta":3100,"style":3100},"# Повний білд із перевіркою (check запускається у фазі verify)\nmvn clean verify\n\n# Тільки перевірка форматування\nmvn spotless:check\n\n# Виправити і одразу перевірити\nmvn spotless:apply spotless:check\n",[2968,8260,8261,8266,8274,8278,8283,8289,8293,8298],{"__ignoreMap":3100},[3346,8262,8263],{"class":3348,"line":3349},[3346,8264,8265],{"class":3997},"# Повний білд із перевіркою (check запускається у фазі verify)\n",[3346,8267,8268,8270,8272],{"class":3348,"line":3355},[3346,8269,4813],{"class":4003},[3346,8271,4864],{"class":4007},[3346,8273,4848],{"class":4007},[3346,8275,8276],{"class":3348,"line":3361},[3346,8277,3395],{"emptyLinePlaceholder":3394},[3346,8279,8280],{"class":3348,"line":3367},[3346,8281,8282],{"class":3997},"# Тільки перевірка форматування\n",[3346,8284,8285,8287],{"class":3348,"line":3373},[3346,8286,4813],{"class":4003},[3346,8288,4816],{"class":4007},[3346,8290,8291],{"class":3348,"line":3379},[3346,8292,3395],{"emptyLinePlaceholder":3394},[3346,8294,8295],{"class":3348,"line":3385},[3346,8296,8297],{"class":3997},"# Виправити і одразу перевірити\n",[3346,8299,8300,8302,8305],{"class":3348,"line":3391},[3346,8301,4813],{"class":4003},[3346,8303,8304],{"class":4007}," spotless:apply",[3346,8306,4816],{"class":4007},[3070,8308],{},[2959,8310,8312],{"id":8311},"spotbugs-статичний-аналіз-патернів-помилок","SpotBugs: Статичний аналіз патернів помилок",[2964,8314,8315,8316,8319,8320,8323],{},"Spotless гарантує ",[2979,8317,8318],{},"зовнішній вигляд"," коду. SpotBugs аналізує ",[2979,8321,8322],{},"семантику"," — він шукає патерни, що статистично часто призводять до помилок у production, навіть якщо код синтаксично коректний і компілюється без попереджень.",[2964,8325,8326,8327,8330],{},"SpotBugs є наступником FindBugs і аналізує скомпільований байт-код (",[2968,8328,8329],{},".class","), а не вихідний код. Це означає, що він бачить те, що компілятор вже обробив — без синтаксичного шуму.",[3084,8332,8334],{"id":8333},"які-дефекти-знаходить-spotbugs","Які дефекти знаходить SpotBugs",[3181,8336,8337,8350],{},[3184,8338,8339],{},[3187,8340,8341,8344,8347],{},[3190,8342,8343],{},"Категорія",[3190,8345,8346],{},"Код",[3190,8348,8349],{},"Приклад",[3200,8351,8352,8368,8384,8400,8414,8427],{},[3187,8353,8354,8357,8362],{},[3205,8355,8356],{},"Null pointer",[3205,8358,8359],{},[2968,8360,8361],{},"NP",[3205,8363,8364,8365],{},"Звернення до результату методу, що може повернути ",[2968,8366,8367],{},"null",[3187,8369,8370,8373,8378],{},[3205,8371,8372],{},"Ignored return value",[3205,8374,8375],{},[2968,8376,8377],{},"RV",[3205,8379,8380,8383],{},[2968,8381,8382],{},"stmt.executeUpdate()"," без збереження результату",[3187,8385,8386,8389,8394],{},[3205,8387,8388],{},"Resource leak",[3205,8390,8391],{},[2968,8392,8393],{},"OBL",[3205,8395,8396,8399],{},[2968,8397,8398],{},"Connection"," відкрита, але не закрита",[3187,8401,8402,8405,8409],{},[3205,8403,8404],{},"SQL injection",[3205,8406,8407],{},[2968,8408,2412],{},[3205,8410,8411],{},[2968,8412,8413],{},"\"SELECT * FROM t WHERE id = \" + id",[3187,8415,8416,8419,8424],{},[3205,8417,8418],{},"Bad synchronization",[3205,8420,8421],{},[2968,8422,8423],{},"IS",[3205,8425,8426],{},"Змінна змінюється без синхронізації",[3187,8428,8429,8432,8437],{},[3205,8430,8431],{},"Incorrect equals",[3205,8433,8434],{},[2968,8435,8436],{},"EC",[3205,8438,8439,8440,8442,8443],{},"Порівняння ",[2968,8441,5249],{}," через ",[2968,8444,8445],{},"==",[3084,8447,8449],{"id":8448},"gradle-конфігурація","Gradle конфігурація",[3092,8451,8453],{"className":3465,"code":8452,"language":3467,"meta":3100,"style":3100},"// build.gradle\nplugins {\n    id 'java'\n    id 'com.github.spotbugs' version '6.5.1'\n}\n\nspotbugs {\n    // Версія самого SpotBugs (не плагіна)\n    toolVersion = '4.8.6'\n\n    // Глибина аналізу: min, default, more, max\n    // 'max' — найретельніший, але повільніший\n    effort = 'max'\n\n    // Мінімальна серйозність для повідомлення: low, medium, high\n    // 'low' — показувати всі, включаючи незначні\n    reportLevel = 'low'\n\n    // false — провалити білд при знахідках (рекомендовано для CI)\n    ignoreFailures = false\n\n    // Файл виключень: паттерни помилок, які ігноруємо свідомо\n    excludeFilter = file('config/spotbugs-exclude.xml')\n}\n\n// Генерувати HTML-звіт (зручніший за XML для ручного перегляду)\ntasks.withType(com.github.spotbugs.snom.SpotBugsTask).configureEach {\n    reports {\n        html {\n            required = true\n            outputLocation = file(\"$buildDir/reports/spotbugs/${name}.html\")\n        }\n        xml.required = false\n    }\n}\n\n// Запускати SpotBugs при ./gradlew check\ntasks.named('check') {\n    dependsOn 'spotbugsMain'\n    // dependsOn 'spotbugsTest' // за потреби аналізувати і тести\n}\n",[2968,8454,8455,8459,8463,8467,8472,8476,8480,8485,8490,8495,8499,8504,8509,8514,8518,8523,8528,8533,8537,8542,8547,8551,8556,8561,8565,8569,8574,8579,8584,8589,8594,8599,8604,8609,8613,8617,8621,8626,8631,8636,8644],{"__ignoreMap":3100},[3346,8456,8457],{"class":3348,"line":3349},[3346,8458,3474],{},[3346,8460,8461],{"class":3348,"line":3355},[3346,8462,3479],{},[3346,8464,8465],{"class":3348,"line":3361},[3346,8466,3484],{},[3346,8468,8469],{"class":3348,"line":3367},[3346,8470,8471],{},"    id 'com.github.spotbugs' version '6.5.1'\n",[3346,8473,8474],{"class":3348,"line":3373},[3346,8475,3494],{},[3346,8477,8478],{"class":3348,"line":3379},[3346,8479,3395],{"emptyLinePlaceholder":3394},[3346,8481,8482],{"class":3348,"line":3385},[3346,8483,8484],{},"spotbugs {\n",[3346,8486,8487],{"class":3348,"line":3391},[3346,8488,8489],{},"    // Версія самого SpotBugs (не плагіна)\n",[3346,8491,8492],{"class":3348,"line":3398},[3346,8493,8494],{},"    toolVersion = '4.8.6'\n",[3346,8496,8497],{"class":3348,"line":3404},[3346,8498,3395],{"emptyLinePlaceholder":3394},[3346,8500,8501],{"class":3348,"line":3410},[3346,8502,8503],{},"    // Глибина аналізу: min, default, more, max\n",[3346,8505,8506],{"class":3348,"line":3416},[3346,8507,8508],{},"    // 'max' — найретельніший, але повільніший\n",[3346,8510,8511],{"class":3348,"line":3422},[3346,8512,8513],{},"    effort = 'max'\n",[3346,8515,8516],{"class":3348,"line":3427},[3346,8517,3395],{"emptyLinePlaceholder":3394},[3346,8519,8520],{"class":3348,"line":3433},[3346,8521,8522],{},"    // Мінімальна серйозність для повідомлення: low, medium, high\n",[3346,8524,8525],{"class":3348,"line":3439},[3346,8526,8527],{},"    // 'low' — показувати всі, включаючи незначні\n",[3346,8529,8530],{"class":3348,"line":3550},[3346,8531,8532],{},"    reportLevel = 'low'\n",[3346,8534,8535],{"class":3348,"line":3555},[3346,8536,3395],{"emptyLinePlaceholder":3394},[3346,8538,8539],{"class":3348,"line":3561},[3346,8540,8541],{},"    // false — провалити білд при знахідках (рекомендовано для CI)\n",[3346,8543,8544],{"class":3348,"line":3567},[3346,8545,8546],{},"    ignoreFailures = false\n",[3346,8548,8549],{"class":3348,"line":3573},[3346,8550,3395],{"emptyLinePlaceholder":3394},[3346,8552,8553],{"class":3348,"line":3579},[3346,8554,8555],{},"    // Файл виключень: паттерни помилок, які ігноруємо свідомо\n",[3346,8557,8558],{"class":3348,"line":3585},[3346,8559,8560],{},"    excludeFilter = file('config/spotbugs-exclude.xml')\n",[3346,8562,8563],{"class":3348,"line":3591},[3346,8564,3494],{},[3346,8566,8567],{"class":3348,"line":3597},[3346,8568,3395],{"emptyLinePlaceholder":3394},[3346,8570,8571],{"class":3348,"line":3603},[3346,8572,8573],{},"// Генерувати HTML-звіт (зручніший за XML для ручного перегляду)\n",[3346,8575,8576],{"class":3348,"line":3609},[3346,8577,8578],{},"tasks.withType(com.github.spotbugs.snom.SpotBugsTask).configureEach {\n",[3346,8580,8581],{"class":3348,"line":3615},[3346,8582,8583],{},"    reports {\n",[3346,8585,8586],{"class":3348,"line":3620},[3346,8587,8588],{},"        html {\n",[3346,8590,8591],{"class":3348,"line":3626},[3346,8592,8593],{},"            required = true\n",[3346,8595,8596],{"class":3348,"line":3632},[3346,8597,8598],{},"            outputLocation = file(\"$buildDir/reports/spotbugs/${name}.html\")\n",[3346,8600,8601],{"class":3348,"line":3638},[3346,8602,8603],{},"        }\n",[3346,8605,8606],{"class":3348,"line":3644},[3346,8607,8608],{},"        xml.required = false\n",[3346,8610,8611],{"class":3348,"line":3650},[3346,8612,3739],{},[3346,8614,8615],{"class":3348,"line":3655},[3346,8616,3494],{},[3346,8618,8619],{"class":3348,"line":3661},[3346,8620,3395],{"emptyLinePlaceholder":3394},[3346,8622,8623],{"class":3348,"line":3667},[3346,8624,8625],{},"// Запускати SpotBugs при ./gradlew check\n",[3346,8627,8628],{"class":3348,"line":3672},[3346,8629,8630],{},"tasks.named('check') {\n",[3346,8632,8633],{"class":3348,"line":3678},[3346,8634,8635],{},"    dependsOn 'spotbugsMain'\n",[3346,8637,8638,8641],{"class":3348,"line":3684},[3346,8639,8640],{},"    // dependsOn 'spotbugsTest'",[3346,8642,8643],{}," // за потреби аналізувати і тести\n",[3346,8645,8646],{"class":3348,"line":3690},[3346,8647,3494],{},[2964,8649,8650],{},[2979,8651,8652,8653,6549],{},"Файл виключень ",[2968,8654,8655],{},"config/spotbugs-exclude.xml",[3092,8657,8659],{"className":4065,"code":8658,"language":4067,"meta":3100,"style":3100},"\u003C?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\u003CFindBugsFilter>\n    \u003C!-- Ігнорувати все у тест-класах -->\n    \u003CMatch>\n        \u003CClass name=\"~.*Test\" />\n    \u003C/Match>\n    \u003C!-- Ігнорувати попередження стилю (залишаємо Spotless для цього) -->\n    \u003CMatch>\n        \u003CBug category=\"STYLE\" />\n    \u003C/Match>\n    \u003C!-- Ігнорувати конкретну помилку у конкретному класі -->\n    \u003C!--\n    \u003CMatch>\n        \u003CClass name=\"com.example.audiobook.db.ConnectionManager\" />\n        \u003CBug pattern=\"OBL_UNSATISFIED_OBLIGATION\" />\n    \u003C/Match>\n    -->\n\u003C/FindBugsFilter>\n",[2968,8660,8661,8689,8698,8703,8712,8730,8738,8743,8751,8768,8776,8781,8786,8791,8796,8801,8806,8811],{"__ignoreMap":3100},[3346,8662,8663,8666,8668,8672,8675,8678,8681,8683,8686],{"class":3348,"line":3349},[3346,8664,8665],{"class":4079},"\u003C?",[3346,8667,4067],{"class":4083},[3346,8669,8671],{"class":8670},"sa4r_"," version",[3346,8673,8674],{"class":4101},"=",[3346,8676,8677],{"class":4903},"\"1.0\"",[3346,8679,8680],{"class":8670}," encoding",[3346,8682,8674],{"class":4101},[3346,8684,8685],{"class":4903},"\"UTF-8\"",[3346,8687,8688],{"class":4079},"?>\n",[3346,8690,8691,8693,8696],{"class":3348,"line":3355},[3346,8692,4080],{"class":4079},[3346,8694,8695],{"class":4083},"FindBugsFilter",[3346,8697,4087],{"class":4079},[3346,8699,8700],{"class":3348,"line":3361},[3346,8701,8702],{"class":3997},"    \u003C!-- Ігнорувати все у тест-класах -->\n",[3346,8704,8705,8707,8710],{"class":3348,"line":3367},[3346,8706,4092],{"class":4079},[3346,8708,8709],{"class":4083},"Match",[3346,8711,4087],{"class":4079},[3346,8713,8714,8716,8719,8722,8724,8727],{"class":3348,"line":3373},[3346,8715,4176],{"class":4079},[3346,8717,8718],{"class":4083},"Class",[3346,8720,8721],{"class":8670}," name",[3346,8723,8674],{"class":4101},[3346,8725,8726],{"class":4903},"\"~.*Test\"",[3346,8728,8729],{"class":4079}," />\n",[3346,8731,8732,8734,8736],{"class":3348,"line":3379},[3346,8733,4656],{"class":4079},[3346,8735,8709],{"class":4083},[3346,8737,4087],{"class":4079},[3346,8739,8740],{"class":3348,"line":3385},[3346,8741,8742],{"class":3997},"    \u003C!-- Ігнорувати попередження стилю (залишаємо Spotless для цього) -->\n",[3346,8744,8745,8747,8749],{"class":3348,"line":3391},[3346,8746,4092],{"class":4079},[3346,8748,8709],{"class":4083},[3346,8750,4087],{"class":4079},[3346,8752,8753,8755,8758,8761,8763,8766],{"class":3348,"line":3398},[3346,8754,4176],{"class":4079},[3346,8756,8757],{"class":4083},"Bug",[3346,8759,8760],{"class":8670}," category",[3346,8762,8674],{"class":4101},[3346,8764,8765],{"class":4903},"\"STYLE\"",[3346,8767,8729],{"class":4079},[3346,8769,8770,8772,8774],{"class":3348,"line":3404},[3346,8771,4656],{"class":4079},[3346,8773,8709],{"class":4083},[3346,8775,4087],{"class":4079},[3346,8777,8778],{"class":3348,"line":3410},[3346,8779,8780],{"class":3997},"    \u003C!-- Ігнорувати конкретну помилку у конкретному класі -->\n",[3346,8782,8783],{"class":3348,"line":3416},[3346,8784,8785],{"class":3997},"    \u003C!--\n",[3346,8787,8788],{"class":3348,"line":3422},[3346,8789,8790],{"class":3997},"    \u003CMatch>\n",[3346,8792,8793],{"class":3348,"line":3427},[3346,8794,8795],{"class":3997},"        \u003CClass name=\"com.example.audiobook.db.ConnectionManager\" />\n",[3346,8797,8798],{"class":3348,"line":3433},[3346,8799,8800],{"class":3997},"        \u003CBug pattern=\"OBL_UNSATISFIED_OBLIGATION\" />\n",[3346,8802,8803],{"class":3348,"line":3439},[3346,8804,8805],{"class":3997},"    \u003C/Match>\n",[3346,8807,8808],{"class":3348,"line":3550},[3346,8809,8810],{"class":3997},"    -->\n",[3346,8812,8813,8815,8817],{"class":3348,"line":3555},[3346,8814,4105],{"class":4079},[3346,8816,8695],{"class":4083},[3346,8818,4087],{"class":4079},[3084,8820,8822],{"id":8821},"maven-конфігурація","Maven конфігурація",[3092,8824,8826],{"className":4065,"code":8825,"language":4067,"meta":3100,"style":3100},"\u003C!-- pom.xml → \u003Cbuild> → \u003Cplugins> -->\n\u003Cplugin>\n    \u003CgroupId>com.github.spotbugs\u003C/groupId>\n    \u003CartifactId>spotbugs-maven-plugin\u003C/artifactId>\n    \u003Cversion>4.9.3.0\u003C/version>\n    \u003Cconfiguration>\n        \u003C!-- Глибина аналізу -->\n        \u003Ceffort>Max\u003C/effort>\n        \u003C!-- Мінімальна серйозність: Low, Medium, High -->\n        \u003Cthreshold>Low\u003C/threshold>\n        \u003C!-- Генерувати XML звіт -->\n        \u003CxmlOutput>true\u003C/xmlOutput>\n        \u003C!-- Файл виключень -->\n        \u003CexcludeFilterFile>config/spotbugs-exclude.xml\u003C/excludeFilterFile>\n    \u003C/configuration>\n    \u003Cexecutions>\n        \u003Cexecution>\n            \u003Cgoals>\n                \u003C!-- check провалює білд при знахідках -->\n                \u003Cgoal>check\u003C/goal>\n            \u003C/goals>\n        \u003C/execution>\n    \u003C/executions>\n\u003C/plugin>\n",[2968,8827,8828,8832,8840,8857,8874,8891,8899,8904,8922,8927,8945,8950,8967,8972,8989,8997,9005,9013,9021,9026,9042,9050,9058,9066],{"__ignoreMap":3100},[3346,8829,8830],{"class":3348,"line":3349},[3346,8831,4074],{"class":3997},[3346,8833,8834,8836,8838],{"class":3348,"line":3355},[3346,8835,4080],{"class":4079},[3346,8837,4084],{"class":4083},[3346,8839,4087],{"class":4079},[3346,8841,8842,8844,8846,8848,8851,8853,8855],{"class":3348,"line":3361},[3346,8843,4092],{"class":4079},[3346,8845,4095],{"class":4083},[3346,8847,4098],{"class":4079},[3346,8849,8850],{"class":4101},"com.github.spotbugs",[3346,8852,4105],{"class":4079},[3346,8854,4095],{"class":4083},[3346,8856,4087],{"class":4079},[3346,8858,8859,8861,8863,8865,8868,8870,8872],{"class":3348,"line":3367},[3346,8860,4092],{"class":4079},[3346,8862,4116],{"class":4083},[3346,8864,4098],{"class":4079},[3346,8866,8867],{"class":4101},"spotbugs-maven-plugin",[3346,8869,4105],{"class":4079},[3346,8871,4116],{"class":4083},[3346,8873,4087],{"class":4079},[3346,8875,8876,8878,8880,8882,8885,8887,8889],{"class":3348,"line":3373},[3346,8877,4092],{"class":4079},[3346,8879,4134],{"class":4083},[3346,8881,4098],{"class":4079},[3346,8883,8884],{"class":4101},"4.9.3.0",[3346,8886,4105],{"class":4079},[3346,8888,4134],{"class":4083},[3346,8890,4087],{"class":4079},[3346,8892,8893,8895,8897],{"class":3348,"line":3379},[3346,8894,4092],{"class":4079},[3346,8896,4155],{"class":4083},[3346,8898,4087],{"class":4079},[3346,8900,8901],{"class":3348,"line":3385},[3346,8902,8903],{"class":3997},"        \u003C!-- Глибина аналізу -->\n",[3346,8905,8906,8908,8911,8913,8916,8918,8920],{"class":3348,"line":3391},[3346,8907,4176],{"class":4079},[3346,8909,8910],{"class":4083},"effort",[3346,8912,4098],{"class":4079},[3346,8914,8915],{"class":4101},"Max",[3346,8917,4105],{"class":4079},[3346,8919,8910],{"class":4083},[3346,8921,4087],{"class":4079},[3346,8923,8924],{"class":3348,"line":3398},[3346,8925,8926],{"class":3997},"        \u003C!-- Мінімальна серйозність: Low, Medium, High -->\n",[3346,8928,8929,8931,8934,8936,8939,8941,8943],{"class":3348,"line":3404},[3346,8930,4176],{"class":4079},[3346,8932,8933],{"class":4083},"threshold",[3346,8935,4098],{"class":4079},[3346,8937,8938],{"class":4101},"Low",[3346,8940,4105],{"class":4079},[3346,8942,8933],{"class":4083},[3346,8944,4087],{"class":4079},[3346,8946,8947],{"class":3348,"line":3410},[3346,8948,8949],{"class":3997},"        \u003C!-- Генерувати XML звіт -->\n",[3346,8951,8952,8954,8957,8959,8961,8963,8965],{"class":3348,"line":3416},[3346,8953,4176],{"class":4079},[3346,8955,8956],{"class":4083},"xmlOutput",[3346,8958,4098],{"class":4079},[3346,8960,4315],{"class":4101},[3346,8962,4105],{"class":4079},[3346,8964,8956],{"class":4083},[3346,8966,4087],{"class":4079},[3346,8968,8969],{"class":3348,"line":3422},[3346,8970,8971],{"class":3997},"        \u003C!-- Файл виключень -->\n",[3346,8973,8974,8976,8979,8981,8983,8985,8987],{"class":3348,"line":3427},[3346,8975,4176],{"class":4079},[3346,8977,8978],{"class":4083},"excludeFilterFile",[3346,8980,4098],{"class":4079},[3346,8982,8655],{"class":4101},[3346,8984,4105],{"class":4079},[3346,8986,8978],{"class":4083},[3346,8988,4087],{"class":4079},[3346,8990,8991,8993,8995],{"class":3348,"line":3433},[3346,8992,4656],{"class":4079},[3346,8994,4155],{"class":4083},[3346,8996,4087],{"class":4079},[3346,8998,8999,9001,9003],{"class":3348,"line":3439},[3346,9000,4092],{"class":4079},[3346,9002,4679],{"class":4083},[3346,9004,4087],{"class":4079},[3346,9006,9007,9009,9011],{"class":3348,"line":3550},[3346,9008,4176],{"class":4079},[3346,9010,4689],{"class":4083},[3346,9012,4087],{"class":4079},[3346,9014,9015,9017,9019],{"class":3348,"line":3555},[3346,9016,4191],{"class":4079},[3346,9018,4737],{"class":4083},[3346,9020,4087],{"class":4079},[3346,9022,9023],{"class":3348,"line":3561},[3346,9024,9025],{"class":3997},"                \u003C!-- check провалює білд при знахідках -->\n",[3346,9027,9028,9030,9032,9034,9036,9038,9040],{"class":3348,"line":3567},[3346,9029,4201],{"class":4079},[3346,9031,4747],{"class":4083},[3346,9033,4098],{"class":4079},[3346,9035,3327],{"class":4101},[3346,9037,4105],{"class":4079},[3346,9039,4747],{"class":4083},[3346,9041,4087],{"class":4079},[3346,9043,9044,9046,9048],{"class":3348,"line":3573},[3346,9045,4237],{"class":4079},[3346,9047,4737],{"class":4083},[3346,9049,4087],{"class":4079},[3346,9051,9052,9054,9056],{"class":3348,"line":3579},[3346,9053,4501],{"class":4079},[3346,9055,4689],{"class":4083},[3346,9057,4087],{"class":4079},[3346,9059,9060,9062,9064],{"class":3348,"line":3585},[3346,9061,4656],{"class":4079},[3346,9063,4679],{"class":4083},[3346,9065,4087],{"class":4079},[3346,9067,9068,9070,9072],{"class":3348,"line":3591},[3346,9069,4105],{"class":4079},[3346,9071,4084],{"class":4083},[3346,9073,4087],{"class":4079},[2964,9075,9076],{},[2979,9077,9078],{},"Команди:",[3092,9080,9082],{"className":3988,"code":9081,"language":3990,"meta":3100,"style":3100},"# Gradle\n./gradlew spotbugsMain           # аналіз src/main/java\n./gradlew spotbugsTest           # аналіз src/test/java\n# Звіт: build/reports/spotbugs/main.html\n\n# Maven\nmvn spotbugs:check               # аналіз і провал при знахідках\nmvn spotbugs:gui                 # відкрити GUI для перегляду результатів\nmvn spotbugs:spotbugs            # лише генерація звіту без провалу\n# Звіт: target/spotbugsXml.xml\n",[2968,9083,9084,9089,9099,9109,9114,9118,9123,9133,9143,9153],{"__ignoreMap":3100},[3346,9085,9086],{"class":3348,"line":3349},[3346,9087,9088],{"class":3997},"# Gradle\n",[3346,9090,9091,9093,9096],{"class":3348,"line":3355},[3346,9092,4004],{"class":4003},[3346,9094,9095],{"class":4007}," spotbugsMain",[3346,9097,9098],{"class":3997},"           # аналіз src/main/java\n",[3346,9100,9101,9103,9106],{"class":3348,"line":3361},[3346,9102,4004],{"class":4003},[3346,9104,9105],{"class":4007}," spotbugsTest",[3346,9107,9108],{"class":3997},"           # аналіз src/test/java\n",[3346,9110,9111],{"class":3348,"line":3367},[3346,9112,9113],{"class":3997},"# Звіт: build/reports/spotbugs/main.html\n",[3346,9115,9116],{"class":3348,"line":3373},[3346,9117,3395],{"emptyLinePlaceholder":3394},[3346,9119,9120],{"class":3348,"line":3379},[3346,9121,9122],{"class":3997},"# Maven\n",[3346,9124,9125,9127,9130],{"class":3348,"line":3385},[3346,9126,4813],{"class":4003},[3346,9128,9129],{"class":4007}," spotbugs:check",[3346,9131,9132],{"class":3997},"               # аналіз і провал при знахідках\n",[3346,9134,9135,9137,9140],{"class":3348,"line":3391},[3346,9136,4813],{"class":4003},[3346,9138,9139],{"class":4007}," spotbugs:gui",[3346,9141,9142],{"class":3997},"                 # відкрити GUI для перегляду результатів\n",[3346,9144,9145,9147,9150],{"class":3348,"line":3398},[3346,9146,4813],{"class":4003},[3346,9148,9149],{"class":4007}," spotbugs:spotbugs",[3346,9151,9152],{"class":3997},"            # лише генерація звіту без провалу\n",[3346,9154,9155],{"class":3348,"line":3404},[3346,9156,9157],{"class":3997},"# Звіт: target/spotbugsXml.xml\n",[3070,9159],{},[2959,9161,9163],{"id":9162},"sonarqube-архітектурний-аналіз-якості","SonarQube: Архітектурний аналіз якості",[2964,9165,9166],{},"SonarQube — це окрема платформа якості коду з веб-інтерфейсом, що дає цілісний погляд на стан проєкту. Якщо SpotBugs знаходить конкретні помилки, SonarQube відповідає на ширші питання: «Чи росте технічний борг?», «Яке покриття тестами?», «Де найбільше дублювання?»",[3084,9168,9170],{"id":9169},"spotbugs-vs-sonarqube-що-вибрати","SpotBugs vs SonarQube: що вибрати",[3181,9172,9173,9186],{},[3184,9174,9175],{},[3187,9176,9177,9180,9183],{},[3190,9178,9179],{},"Можливість",[3190,9181,9182],{},"SpotBugs",[3190,9184,9185],{},"SonarQube",[3200,9187,9188,9198,9208,9218,9227,9237,9246,9255,9264],{},[3187,9189,9190,9193,9196],{},[3205,9191,9192],{},"Bug patterns",[3205,9194,9195],{},"✅",[3205,9197,9195],{},[3187,9199,9200,9203,9206],{},[3205,9201,9202],{},"Code smells",[3205,9204,9205],{},"❌",[3205,9207,9195],{},[3187,9209,9210,9213,9216],{},[3205,9211,9212],{},"Security vulnerabilities",[3205,9214,9215],{},"обмежено",[3205,9217,9195],{},[3187,9219,9220,9223,9225],{},[3205,9221,9222],{},"Дублювання коду",[3205,9224,9205],{},[3205,9226,9195],{},[3187,9228,9229,9232,9234],{},[3205,9230,9231],{},"Покриття тестами",[3205,9233,9205],{},[3205,9235,9236],{},"✅ (через JaCoCo)",[3187,9238,9239,9242,9244],{},[3205,9240,9241],{},"Технічний борг (час)",[3205,9243,9205],{},[3205,9245,9195],{},[3187,9247,9248,9251,9253],{},[3205,9249,9250],{},"Quality Gate",[3205,9252,9205],{},[3205,9254,9195],{},[3187,9256,9257,9260,9262],{},[3205,9258,9259],{},"Historical trends",[3205,9261,9205],{},[3205,9263,9195],{},[3187,9265,9266,9269,9271],{},[3205,9267,9268],{},"Потребує окремого сервера",[3205,9270,9205],{},[3205,9272,9195],{},[2964,9274,9275],{},"Ці інструменти не конкурують — вони доповнюють одне одного. SpotBugs простіший у запуску, SonarQube дає глибший аналіз.",[3084,9277,9279],{"id":9278},"запуск-sonarqube-локально-docker","Запуск SonarQube локально (Docker)",[3092,9281,9283],{"className":3988,"code":9282,"language":3990,"meta":3100,"style":3100},"# Запуск контейнера з постійним зберіганням даних\ndocker run -d \\\n  --name sonarqube \\\n  -p 9000:9000 \\\n  -v sonarqube_data:/opt/sonarqube/data \\\n  -v sonarqube_logs:/opt/sonarqube/logs \\\n  sonarqube:community\n\n# SonarQube запускається ~30-60 секунд\n# Відкрийте: http://localhost:9000\n# Логін: admin / admin → обов'язково змінити пароль\n",[2968,9284,9285,9290,9305,9315,9325,9335,9344,9349,9353,9358,9363],{"__ignoreMap":3100},[3346,9286,9287],{"class":3348,"line":3349},[3346,9288,9289],{"class":3997},"# Запуск контейнера з постійним зберіганням даних\n",[3346,9291,9292,9295,9298,9301],{"class":3348,"line":3355},[3346,9293,9294],{"class":4003},"docker",[3346,9296,9297],{"class":4007}," run",[3346,9299,9300],{"class":4909}," -d",[3346,9302,9304],{"class":9303},"sjcCO"," \\\n",[3346,9306,9307,9310,9313],{"class":3348,"line":3361},[3346,9308,9309],{"class":4909},"  --name",[3346,9311,9312],{"class":4007}," sonarqube",[3346,9314,9304],{"class":9303},[3346,9316,9317,9320,9323],{"class":3348,"line":3367},[3346,9318,9319],{"class":4909},"  -p",[3346,9321,9322],{"class":4007}," 9000:9000",[3346,9324,9304],{"class":9303},[3346,9326,9327,9330,9333],{"class":3348,"line":3373},[3346,9328,9329],{"class":4909},"  -v",[3346,9331,9332],{"class":4007}," sonarqube_data:/opt/sonarqube/data",[3346,9334,9304],{"class":9303},[3346,9336,9337,9339,9342],{"class":3348,"line":3379},[3346,9338,9329],{"class":4909},[3346,9340,9341],{"class":4007}," sonarqube_logs:/opt/sonarqube/logs",[3346,9343,9304],{"class":9303},[3346,9345,9346],{"class":3348,"line":3385},[3346,9347,9348],{"class":4007},"  sonarqube:community\n",[3346,9350,9351],{"class":3348,"line":3391},[3346,9352,3395],{"emptyLinePlaceholder":3394},[3346,9354,9355],{"class":3348,"line":3398},[3346,9356,9357],{"class":3997},"# SonarQube запускається ~30-60 секунд\n",[3346,9359,9360],{"class":3348,"line":3404},[3346,9361,9362],{"class":3997},"# Відкрийте: http://localhost:9000\n",[3346,9364,9365],{"class":3348,"line":3410},[3346,9366,9367],{"class":3997},"# Логін: admin / admin → обов'язково змінити пароль\n",[2964,9369,9370],{},[2979,9371,9372],{},"Отримання токена аналізу:",[9374,9375,9376,9388,9394],"ol",{},[9377,9378,9379,3140,9382,3140,9385],"li",{},[2968,9380,9381],{},"My Account",[2968,9383,9384],{},"Security",[2968,9386,9387],{},"Generate Token",[9377,9389,9390,9391],{},"Тип токена: ",[2968,9392,9393],{},"Project Analysis Token",[9377,9395,9396,9397],{},"Збережіть як змінну середовища: ",[2968,9398,9399],{},"export SONAR_TOKEN=sqp_xxx...",[3043,9401,9402,9403,9406],{},"Для публічних GitHub-репозиторіїв існує ",[2979,9404,9405],{},"SonarQube Cloud"," (sonarcloud.io) — безкоштовна хмарна версія без необхідності підтримувати власний сервер. Підключення займає ~5 хвилин через GitHub OAuth.",[3084,9408,8449],{"id":9409},"gradle-конфігурація-1",[3092,9411,9413],{"className":3465,"code":9412,"language":3467,"meta":3100,"style":3100},"// build.gradle\nplugins {\n    id 'java'\n    id 'org.sonarqube' version '7.2.3.7755'\n    id 'jacoco'           // для звіту покриття тестами\n}\n\nsonar {\n    properties {\n        property 'sonar.projectKey',  'audiobook-platform'\n        property 'sonar.projectName', 'Audiobook Platform'\n        property 'sonar.host.url',    'http://localhost:9000'\n\n        // Токен береться зі змінної середовища SONAR_TOKEN\n        // Ніколи не хардкодьте токен у build.gradle!\n        property 'sonar.token', System.getenv('SONAR_TOKEN') ?: ''\n\n        // Шлях до JaCoCo XML-звіту (для покриття тестами)\n        property 'sonar.coverage.jacoco.xmlReportPaths',\n                 \"${buildDir}/reports/jacoco/test/jacocoTestReport.xml\"\n\n        // Виключити з аналізу генерований код\n        property 'sonar.exclusions', '**/generated/**,**/*Test*.java'\n    }\n}\n\n// JaCoCo: генерація звіту покриття\njacoco {\n    toolVersion = '0.8.12'\n}\n\njacocoTestReport {\n    dependsOn test       // спочатку запустити тести\n    reports {\n        xml.required  = true   // SonarQube читає XML\n        html.required = true   // для ручного перегляду\n    }\n}\n",[2968,9414,9415,9419,9423,9427,9432,9437,9441,9445,9450,9455,9460,9465,9470,9474,9479,9484,9489,9493,9498,9503,9508,9512,9517,9522,9526,9530,9534,9539,9544,9549,9553,9557,9562,9567,9571,9576,9581,9585],{"__ignoreMap":3100},[3346,9416,9417],{"class":3348,"line":3349},[3346,9418,3474],{},[3346,9420,9421],{"class":3348,"line":3355},[3346,9422,3479],{},[3346,9424,9425],{"class":3348,"line":3361},[3346,9426,3484],{},[3346,9428,9429],{"class":3348,"line":3367},[3346,9430,9431],{},"    id 'org.sonarqube' version '7.2.3.7755'\n",[3346,9433,9434],{"class":3348,"line":3373},[3346,9435,9436],{},"    id 'jacoco'           // для звіту покриття тестами\n",[3346,9438,9439],{"class":3348,"line":3379},[3346,9440,3494],{},[3346,9442,9443],{"class":3348,"line":3385},[3346,9444,3395],{"emptyLinePlaceholder":3394},[3346,9446,9447],{"class":3348,"line":3391},[3346,9448,9449],{},"sonar {\n",[3346,9451,9452],{"class":3348,"line":3398},[3346,9453,9454],{},"    properties {\n",[3346,9456,9457],{"class":3348,"line":3404},[3346,9458,9459],{},"        property 'sonar.projectKey',  'audiobook-platform'\n",[3346,9461,9462],{"class":3348,"line":3410},[3346,9463,9464],{},"        property 'sonar.projectName', 'Audiobook Platform'\n",[3346,9466,9467],{"class":3348,"line":3416},[3346,9468,9469],{},"        property 'sonar.host.url',    'http://localhost:9000'\n",[3346,9471,9472],{"class":3348,"line":3422},[3346,9473,3395],{"emptyLinePlaceholder":3394},[3346,9475,9476],{"class":3348,"line":3427},[3346,9477,9478],{},"        // Токен береться зі змінної середовища SONAR_TOKEN\n",[3346,9480,9481],{"class":3348,"line":3433},[3346,9482,9483],{},"        // Ніколи не хардкодьте токен у build.gradle!\n",[3346,9485,9486],{"class":3348,"line":3439},[3346,9487,9488],{},"        property 'sonar.token', System.getenv('SONAR_TOKEN') ?: ''\n",[3346,9490,9491],{"class":3348,"line":3550},[3346,9492,3395],{"emptyLinePlaceholder":3394},[3346,9494,9495],{"class":3348,"line":3555},[3346,9496,9497],{},"        // Шлях до JaCoCo XML-звіту (для покриття тестами)\n",[3346,9499,9500],{"class":3348,"line":3561},[3346,9501,9502],{},"        property 'sonar.coverage.jacoco.xmlReportPaths',\n",[3346,9504,9505],{"class":3348,"line":3567},[3346,9506,9507],{},"                 \"${buildDir}/reports/jacoco/test/jacocoTestReport.xml\"\n",[3346,9509,9510],{"class":3348,"line":3573},[3346,9511,3395],{"emptyLinePlaceholder":3394},[3346,9513,9514],{"class":3348,"line":3579},[3346,9515,9516],{},"        // Виключити з аналізу генерований код\n",[3346,9518,9519],{"class":3348,"line":3585},[3346,9520,9521],{},"        property 'sonar.exclusions', '**/generated/**,**/*Test*.java'\n",[3346,9523,9524],{"class":3348,"line":3591},[3346,9525,3739],{},[3346,9527,9528],{"class":3348,"line":3597},[3346,9529,3494],{},[3346,9531,9532],{"class":3348,"line":3603},[3346,9533,3395],{"emptyLinePlaceholder":3394},[3346,9535,9536],{"class":3348,"line":3609},[3346,9537,9538],{},"// JaCoCo: генерація звіту покриття\n",[3346,9540,9541],{"class":3348,"line":3615},[3346,9542,9543],{},"jacoco {\n",[3346,9545,9546],{"class":3348,"line":3620},[3346,9547,9548],{},"    toolVersion = '0.8.12'\n",[3346,9550,9551],{"class":3348,"line":3626},[3346,9552,3494],{},[3346,9554,9555],{"class":3348,"line":3632},[3346,9556,3395],{"emptyLinePlaceholder":3394},[3346,9558,9559],{"class":3348,"line":3638},[3346,9560,9561],{},"jacocoTestReport {\n",[3346,9563,9564],{"class":3348,"line":3644},[3346,9565,9566],{},"    dependsOn test       // спочатку запустити тести\n",[3346,9568,9569],{"class":3348,"line":3650},[3346,9570,8583],{},[3346,9572,9573],{"class":3348,"line":3655},[3346,9574,9575],{},"        xml.required  = true   // SonarQube читає XML\n",[3346,9577,9578],{"class":3348,"line":3661},[3346,9579,9580],{},"        html.required = true   // для ручного перегляду\n",[3346,9582,9583],{"class":3348,"line":3667},[3346,9584,3739],{},[3346,9586,9587],{"class":3348,"line":3672},[3346,9588,3494],{},[2964,9590,9591],{},[2979,9592,9593],{},"Команда аналізу:",[3092,9595,9597],{"className":3988,"code":9596,"language":3990,"meta":3100,"style":3100},"export SONAR_TOKEN=sqp_your_token_here\n\n# Запустити тести → сформувати JaCoCo-звіт → надіслати у SonarQube\n./gradlew test jacocoTestReport sonar\n",[2968,9598,9599,9613,9617,9622],{"__ignoreMap":3100},[3346,9600,9601,9604,9608,9610],{"class":3348,"line":3349},[3346,9602,9603],{"class":4909},"export",[3346,9605,9607],{"class":9606},"siwwj"," SONAR_TOKEN",[3346,9609,8674],{"class":4101},[3346,9611,9612],{"class":9606},"sqp_your_token_here\n",[3346,9614,9615],{"class":3348,"line":3355},[3346,9616,3395],{"emptyLinePlaceholder":3394},[3346,9618,9619],{"class":3348,"line":3361},[3346,9620,9621],{"class":3997},"# Запустити тести → сформувати JaCoCo-звіт → надіслати у SonarQube\n",[3346,9623,9624,9626,9629,9632],{"class":3348,"line":3367},[3346,9625,4004],{"class":4003},[3346,9627,9628],{"class":4007}," test",[3346,9630,9631],{"class":4007}," jacocoTestReport",[3346,9633,9634],{"class":4007}," sonar\n",[3084,9636,8822],{"id":9637},"maven-конфігурація-1",[3092,9639,9641],{"className":4065,"code":9640,"language":4067,"meta":3100,"style":3100},"\u003C!-- pom.xml → \u003Cproperties> -->\n\u003Cproperties>\n    \u003Csonar.projectKey>audiobook-platform\u003C/sonar.projectKey>\n    \u003Csonar.projectName>Audiobook Platform\u003C/sonar.projectName>\n    \u003Csonar.host.url>http://localhost:9000\u003C/sonar.host.url>\n    \u003C!-- Токен зі змінної середовища -->\n    \u003Csonar.token>${env.SONAR_TOKEN}\u003C/sonar.token>\n    \u003C!-- Шлях до JaCoCo XML-звіту -->\n    \u003Csonar.coverage.jacoco.xmlReportPaths>\n        ${project.build.directory}/site/jacoco/jacoco.xml\n    \u003C/sonar.coverage.jacoco.xmlReportPaths>\n    \u003Csonar.exclusions>**/generated/**\u003C/sonar.exclusions>\n\u003C/properties>\n\n\u003C!-- pom.xml → \u003Cbuild> → \u003Cplugins> -->\n\u003C!-- JaCoCo для генерації звіту покриття тестами -->\n\u003Cplugin>\n    \u003CgroupId>org.jacoco\u003C/groupId>\n    \u003CartifactId>jacoco-maven-plugin\u003C/artifactId>\n    \u003Cversion>0.8.12\u003C/version>\n    \u003Cexecutions>\n        \u003C!-- Інструментує тести для збору даних покриття -->\n        \u003Cexecution>\n            \u003Cgoals>\u003Cgoal>prepare-agent\u003C/goal>\u003C/goals>\n        \u003C/execution>\n        \u003C!-- Генерує звіт після виконання тестів -->\n        \u003Cexecution>\n            \u003Cid>report\u003C/id>\n            \u003Cphase>test\u003C/phase>\n            \u003Cgoals>\u003Cgoal>report\u003C/goal>\u003C/goals>\n        \u003C/execution>\n    \u003C/executions>\n\u003C/plugin>\n",[2968,9642,9643,9648,9657,9675,9693,9711,9716,9734,9739,9748,9753,9761,9779,9787,9791,9795,9800,9808,9825,9842,9859,9867,9872,9880,9907,9915,9920,9928,9945,9962,9986,9994,10002],{"__ignoreMap":3100},[3346,9644,9645],{"class":3348,"line":3349},[3346,9646,9647],{"class":3997},"\u003C!-- pom.xml → \u003Cproperties> -->\n",[3346,9649,9650,9652,9655],{"class":3348,"line":3355},[3346,9651,4080],{"class":4079},[3346,9653,9654],{"class":4083},"properties",[3346,9656,4087],{"class":4079},[3346,9658,9659,9661,9664,9666,9669,9671,9673],{"class":3348,"line":3361},[3346,9660,4092],{"class":4079},[3346,9662,9663],{"class":4083},"sonar.projectKey",[3346,9665,4098],{"class":4079},[3346,9667,9668],{"class":4101},"audiobook-platform",[3346,9670,4105],{"class":4079},[3346,9672,9663],{"class":4083},[3346,9674,4087],{"class":4079},[3346,9676,9677,9679,9682,9684,9687,9689,9691],{"class":3348,"line":3367},[3346,9678,4092],{"class":4079},[3346,9680,9681],{"class":4083},"sonar.projectName",[3346,9683,4098],{"class":4079},[3346,9685,9686],{"class":4101},"Audiobook Platform",[3346,9688,4105],{"class":4079},[3346,9690,9681],{"class":4083},[3346,9692,4087],{"class":4079},[3346,9694,9695,9697,9700,9702,9705,9707,9709],{"class":3348,"line":3373},[3346,9696,4092],{"class":4079},[3346,9698,9699],{"class":4083},"sonar.host.url",[3346,9701,4098],{"class":4079},[3346,9703,9704],{"class":4101},"http://localhost:9000",[3346,9706,4105],{"class":4079},[3346,9708,9699],{"class":4083},[3346,9710,4087],{"class":4079},[3346,9712,9713],{"class":3348,"line":3379},[3346,9714,9715],{"class":3997},"    \u003C!-- Токен зі змінної середовища -->\n",[3346,9717,9718,9720,9723,9725,9728,9730,9732],{"class":3348,"line":3385},[3346,9719,4092],{"class":4079},[3346,9721,9722],{"class":4083},"sonar.token",[3346,9724,4098],{"class":4079},[3346,9726,9727],{"class":4101},"${env.SONAR_TOKEN}",[3346,9729,4105],{"class":4079},[3346,9731,9722],{"class":4083},[3346,9733,4087],{"class":4079},[3346,9735,9736],{"class":3348,"line":3391},[3346,9737,9738],{"class":3997},"    \u003C!-- Шлях до JaCoCo XML-звіту -->\n",[3346,9740,9741,9743,9746],{"class":3348,"line":3398},[3346,9742,4092],{"class":4079},[3346,9744,9745],{"class":4083},"sonar.coverage.jacoco.xmlReportPaths",[3346,9747,4087],{"class":4079},[3346,9749,9750],{"class":3348,"line":3404},[3346,9751,9752],{"class":4101},"        ${project.build.directory}/site/jacoco/jacoco.xml\n",[3346,9754,9755,9757,9759],{"class":3348,"line":3410},[3346,9756,4656],{"class":4079},[3346,9758,9745],{"class":4083},[3346,9760,4087],{"class":4079},[3346,9762,9763,9765,9768,9770,9773,9775,9777],{"class":3348,"line":3416},[3346,9764,4092],{"class":4079},[3346,9766,9767],{"class":4083},"sonar.exclusions",[3346,9769,4098],{"class":4079},[3346,9771,9772],{"class":4101},"**/generated/**",[3346,9774,4105],{"class":4079},[3346,9776,9767],{"class":4083},[3346,9778,4087],{"class":4079},[3346,9780,9781,9783,9785],{"class":3348,"line":3422},[3346,9782,4105],{"class":4079},[3346,9784,9654],{"class":4083},[3346,9786,4087],{"class":4079},[3346,9788,9789],{"class":3348,"line":3427},[3346,9790,3395],{"emptyLinePlaceholder":3394},[3346,9792,9793],{"class":3348,"line":3433},[3346,9794,4074],{"class":3997},[3346,9796,9797],{"class":3348,"line":3439},[3346,9798,9799],{"class":3997},"\u003C!-- JaCoCo для генерації звіту покриття тестами -->\n",[3346,9801,9802,9804,9806],{"class":3348,"line":3550},[3346,9803,4080],{"class":4079},[3346,9805,4084],{"class":4083},[3346,9807,4087],{"class":4079},[3346,9809,9810,9812,9814,9816,9819,9821,9823],{"class":3348,"line":3555},[3346,9811,4092],{"class":4079},[3346,9813,4095],{"class":4083},[3346,9815,4098],{"class":4079},[3346,9817,9818],{"class":4101},"org.jacoco",[3346,9820,4105],{"class":4079},[3346,9822,4095],{"class":4083},[3346,9824,4087],{"class":4079},[3346,9826,9827,9829,9831,9833,9836,9838,9840],{"class":3348,"line":3561},[3346,9828,4092],{"class":4079},[3346,9830,4116],{"class":4083},[3346,9832,4098],{"class":4079},[3346,9834,9835],{"class":4101},"jacoco-maven-plugin",[3346,9837,4105],{"class":4079},[3346,9839,4116],{"class":4083},[3346,9841,4087],{"class":4079},[3346,9843,9844,9846,9848,9850,9853,9855,9857],{"class":3348,"line":3567},[3346,9845,4092],{"class":4079},[3346,9847,4134],{"class":4083},[3346,9849,4098],{"class":4079},[3346,9851,9852],{"class":4101},"0.8.12",[3346,9854,4105],{"class":4079},[3346,9856,4134],{"class":4083},[3346,9858,4087],{"class":4079},[3346,9860,9861,9863,9865],{"class":3348,"line":3573},[3346,9862,4092],{"class":4079},[3346,9864,4679],{"class":4083},[3346,9866,4087],{"class":4079},[3346,9868,9869],{"class":3348,"line":3579},[3346,9870,9871],{"class":3997},"        \u003C!-- Інструментує тести для збору даних покриття -->\n",[3346,9873,9874,9876,9878],{"class":3348,"line":3585},[3346,9875,4176],{"class":4079},[3346,9877,4689],{"class":4083},[3346,9879,4087],{"class":4079},[3346,9881,9882,9884,9886,9889,9891,9893,9896,9898,9900,9903,9905],{"class":3348,"line":3591},[3346,9883,4191],{"class":4079},[3346,9885,4737],{"class":4083},[3346,9887,9888],{"class":4079},">\u003C",[3346,9890,4747],{"class":4083},[3346,9892,4098],{"class":4079},[3346,9894,9895],{"class":4101},"prepare-agent",[3346,9897,4105],{"class":4079},[3346,9899,4747],{"class":4083},[3346,9901,9902],{"class":4079},">\u003C/",[3346,9904,4737],{"class":4083},[3346,9906,4087],{"class":4079},[3346,9908,9909,9911,9913],{"class":3348,"line":3597},[3346,9910,4501],{"class":4079},[3346,9912,4689],{"class":4083},[3346,9914,4087],{"class":4079},[3346,9916,9917],{"class":3348,"line":3603},[3346,9918,9919],{"class":3997},"        \u003C!-- Генерує звіт після виконання тестів -->\n",[3346,9921,9922,9924,9926],{"class":3348,"line":3609},[3346,9923,4176],{"class":4079},[3346,9925,4689],{"class":4083},[3346,9927,4087],{"class":4079},[3346,9929,9930,9932,9934,9936,9939,9941,9943],{"class":3348,"line":3615},[3346,9931,4191],{"class":4079},[3346,9933,4699],{"class":4083},[3346,9935,4098],{"class":4079},[3346,9937,9938],{"class":4101},"report",[3346,9940,4105],{"class":4079},[3346,9942,4699],{"class":4083},[3346,9944,4087],{"class":4079},[3346,9946,9947,9949,9951,9953,9956,9958,9960],{"class":3348,"line":3620},[3346,9948,4191],{"class":4079},[3346,9950,4718],{"class":4083},[3346,9952,4098],{"class":4079},[3346,9954,9955],{"class":4101},"test",[3346,9957,4105],{"class":4079},[3346,9959,4718],{"class":4083},[3346,9961,4087],{"class":4079},[3346,9963,9964,9966,9968,9970,9972,9974,9976,9978,9980,9982,9984],{"class":3348,"line":3626},[3346,9965,4191],{"class":4079},[3346,9967,4737],{"class":4083},[3346,9969,9888],{"class":4079},[3346,9971,4747],{"class":4083},[3346,9973,4098],{"class":4079},[3346,9975,9938],{"class":4101},[3346,9977,4105],{"class":4079},[3346,9979,4747],{"class":4083},[3346,9981,9902],{"class":4079},[3346,9983,4737],{"class":4083},[3346,9985,4087],{"class":4079},[3346,9987,9988,9990,9992],{"class":3348,"line":3632},[3346,9989,4501],{"class":4079},[3346,9991,4689],{"class":4083},[3346,9993,4087],{"class":4079},[3346,9995,9996,9998,10000],{"class":3348,"line":3638},[3346,9997,4656],{"class":4079},[3346,9999,4679],{"class":4083},[3346,10001,4087],{"class":4079},[3346,10003,10004,10006,10008],{"class":3348,"line":3644},[3346,10005,4105],{"class":4079},[3346,10007,4084],{"class":4083},[3346,10009,4087],{"class":4079},[2964,10011,10012],{},[2979,10013,9593],{},[3092,10015,10017],{"className":3988,"code":10016,"language":3990,"meta":3100,"style":3100},"export SONAR_TOKEN=sqp_your_token_here\n\n# Зібрати → тести → JaCoCo → SonarQube\nmvn clean verify sonar:sonar\n",[2968,10018,10019,10029,10033,10038],{"__ignoreMap":3100},[3346,10020,10021,10023,10025,10027],{"class":3348,"line":3349},[3346,10022,9603],{"class":4909},[3346,10024,9607],{"class":9606},[3346,10026,8674],{"class":4101},[3346,10028,9612],{"class":9606},[3346,10030,10031],{"class":3348,"line":3355},[3346,10032,3395],{"emptyLinePlaceholder":3394},[3346,10034,10035],{"class":3348,"line":3361},[3346,10036,10037],{"class":3997},"# Зібрати → тести → JaCoCo → SonarQube\n",[3346,10039,10040,10042,10044,10047],{"class":3348,"line":3367},[3346,10041,4813],{"class":4003},[3346,10043,4864],{"class":4007},[3346,10045,10046],{"class":4007}," verify",[3346,10048,10049],{"class":4007}," sonar:sonar\n",[3084,10051,9250],{"id":10052},"quality-gate",[2964,10054,10055,10057],{},[2979,10056,9250],{}," — набір порогових значень, що визначає, чи є код «готовим до релізу». За замовчуванням SonarQube Community включає такі умови:",[3181,10059,10060,10070],{},[3184,10061,10062],{},[3187,10063,10064,10067],{},[3190,10065,10066],{},"Метрика",[3190,10068,10069],{},"Стандартний поріг",[3200,10071,10072,10080,10088,10096,10102],{},[3187,10073,10074,10077],{},[3205,10075,10076],{},"Покриття нового коду тестами",[3205,10078,10079],{},"≥ 80%",[3187,10081,10082,10085],{},[3205,10083,10084],{},"Дублювання нового коду",[3205,10086,10087],{},"≤ 3%",[3187,10089,10090,10093],{},[3205,10091,10092],{},"Критичні баги",[3205,10094,10095],{},"0",[3187,10097,10098,10100],{},[3205,10099,9212],{},[3205,10101,10095],{},[3187,10103,10104,10106],{},[3205,10105,9202],{},[3205,10107,10108],{},"≤ A rating",[2964,10110,10111,10112,10115,10116,10119,10120,10123],{},"Якщо Quality Gate не пройдено — ",[2968,10113,10114],{},"sonar"," task або ",[2968,10117,10118],{},"sonar:sonar"," мета повертає ненульовий код завершення, і CI pipeline зупиняється. Це і є ",[2979,10121,10122],{},"архітектурний блокер",": не можна злити PR, якщо новий код знижує якість проєкту.",[3070,10125],{},[2959,10127,10129],{"id":10128},"повна-картина-pipeline-якості-коду","Повна картина: Pipeline якості коду",[3337,10131,10132],{},[3092,10133,10135],{"className":3341,"code":10134,"language":3337,"meta":3100,"style":3100},"graph TD\n    DEV[\"👨‍💻 Розробник\u003Cbr/>пише код\"]\n    IDEA[\"IntelliJ IDEA\u003Cbr/>Ctrl+Alt+L\u003Cbr/>Google Style XML\"]\n    APPLY[\"spotlessApply\u003Cbr/>Автовиправлення\"]\n    COMMIT[\"git push / PR\"]\n\n    subgraph CI [\"🤖 CI Pipeline (GitHub Actions)\"]\n        SL[\"spotlessCheck\u003Cbr/>Форматування?\"]\n        SB[\"SpotBugs\u003Cbr/>Патерни помилок?\"]\n        TEST[\"Unit Tests\u003Cbr/>+ JaCoCo coverage\"]\n        SONAR[\"SonarQube\u003Cbr/>Quality Gate?\"]\n    end\n\n    PASS[\"✅ PR Ready to Merge\"]\n    FAIL[\"❌ CI Failed\u003Cbr/>Повідомлення розробнику\"]\n\n    DEV --> IDEA --> APPLY --> COMMIT\n    COMMIT --> SL\n    SL -- \"fail\" --> FAIL\n    SL -- \"pass\" --> SB\n    SB -- \"fail\" --> FAIL\n    SB -- \"pass\" --> TEST\n    TEST --> SONAR\n    SONAR -- \"Quality Gate fail\" --> FAIL\n    SONAR -- \"Quality Gate pass\" --> PASS\n\n    style FAIL fill:#ef4444,stroke:#b91c1c,color:#ffffff\n    style PASS fill:#22c55e,stroke:#15803d,color:#ffffff\n",[2968,10136,10137,10141,10146,10151,10156,10161,10165,10170,10175,10180,10185,10190,10195,10199,10204,10209,10213,10218,10223,10228,10233,10238,10243,10248,10253,10258,10262,10266],{"__ignoreMap":3100},[3346,10138,10139],{"class":3348,"line":3349},[3346,10140,3352],{},[3346,10142,10143],{"class":3348,"line":3355},[3346,10144,10145],{},"    DEV[\"👨‍💻 Розробник\u003Cbr/>пише код\"]\n",[3346,10147,10148],{"class":3348,"line":3361},[3346,10149,10150],{},"    IDEA[\"IntelliJ IDEA\u003Cbr/>Ctrl+Alt+L\u003Cbr/>Google Style XML\"]\n",[3346,10152,10153],{"class":3348,"line":3367},[3346,10154,10155],{},"    APPLY[\"spotlessApply\u003Cbr/>Автовиправлення\"]\n",[3346,10157,10158],{"class":3348,"line":3373},[3346,10159,10160],{},"    COMMIT[\"git push / PR\"]\n",[3346,10162,10163],{"class":3348,"line":3379},[3346,10164,3395],{"emptyLinePlaceholder":3394},[3346,10166,10167],{"class":3348,"line":3385},[3346,10168,10169],{},"    subgraph CI [\"🤖 CI Pipeline (GitHub Actions)\"]\n",[3346,10171,10172],{"class":3348,"line":3391},[3346,10173,10174],{},"        SL[\"spotlessCheck\u003Cbr/>Форматування?\"]\n",[3346,10176,10177],{"class":3348,"line":3398},[3346,10178,10179],{},"        SB[\"SpotBugs\u003Cbr/>Патерни помилок?\"]\n",[3346,10181,10182],{"class":3348,"line":3404},[3346,10183,10184],{},"        TEST[\"Unit Tests\u003Cbr/>+ JaCoCo coverage\"]\n",[3346,10186,10187],{"class":3348,"line":3410},[3346,10188,10189],{},"        SONAR[\"SonarQube\u003Cbr/>Quality Gate?\"]\n",[3346,10191,10192],{"class":3348,"line":3416},[3346,10193,10194],{},"    end\n",[3346,10196,10197],{"class":3348,"line":3422},[3346,10198,3395],{"emptyLinePlaceholder":3394},[3346,10200,10201],{"class":3348,"line":3427},[3346,10202,10203],{},"    PASS[\"✅ PR Ready to Merge\"]\n",[3346,10205,10206],{"class":3348,"line":3433},[3346,10207,10208],{},"    FAIL[\"❌ CI Failed\u003Cbr/>Повідомлення розробнику\"]\n",[3346,10210,10211],{"class":3348,"line":3439},[3346,10212,3395],{"emptyLinePlaceholder":3394},[3346,10214,10215],{"class":3348,"line":3550},[3346,10216,10217],{},"    DEV --> IDEA --> APPLY --> COMMIT\n",[3346,10219,10220],{"class":3348,"line":3555},[3346,10221,10222],{},"    COMMIT --> SL\n",[3346,10224,10225],{"class":3348,"line":3561},[3346,10226,10227],{},"    SL -- \"fail\" --> FAIL\n",[3346,10229,10230],{"class":3348,"line":3567},[3346,10231,10232],{},"    SL -- \"pass\" --> SB\n",[3346,10234,10235],{"class":3348,"line":3573},[3346,10236,10237],{},"    SB -- \"fail\" --> FAIL\n",[3346,10239,10240],{"class":3348,"line":3579},[3346,10241,10242],{},"    SB -- \"pass\" --> TEST\n",[3346,10244,10245],{"class":3348,"line":3585},[3346,10246,10247],{},"    TEST --> SONAR\n",[3346,10249,10250],{"class":3348,"line":3591},[3346,10251,10252],{},"    SONAR -- \"Quality Gate fail\" --> FAIL\n",[3346,10254,10255],{"class":3348,"line":3597},[3346,10256,10257],{},"    SONAR -- \"Quality Gate pass\" --> PASS\n",[3346,10259,10260],{"class":3348,"line":3603},[3346,10261,3395],{"emptyLinePlaceholder":3394},[3346,10263,10264],{"class":3348,"line":3609},[3346,10265,3430],{},[3346,10267,10268],{"class":3348,"line":3615},[3346,10269,10270],{},"    style PASS fill:#22c55e,stroke:#15803d,color:#ffffff\n",[3070,10272],{},[2959,10274,10276],{"id":10275},"підсумок","Підсумок",[2964,10278,10279,10280,10283],{},"Три рівні контролю якості, що ми розглянули, утворюють ",[2979,10281,10282],{},"ешелонований захист",": кожен шар виявляє свій клас проблем, і жоден не дублює інший.",[3181,10285,10286,10302],{},[3184,10287,10288],{},[3187,10289,10290,10293,10296,10299],{},[3190,10291,10292],{},"Рівень",[3190,10294,10295],{},"Інструмент",[3190,10297,10298],{},"Виявляє",[3190,10300,10301],{},"Коли спрацьовує",[3200,10303,10304,10318,10331],{},[3187,10305,10306,10309,10312,10315],{},[3205,10307,10308],{},"1. Форматування",[3205,10310,10311],{},"Spotless + GJF",[3205,10313,10314],{},"Відступи, imports, пробіли",[3205,10316,10317],{},"Кожен білд/commit",[3187,10319,10320,10323,10325,10328],{},[3205,10321,10322],{},"2. Патерни помилок",[3205,10324,9182],{},[3205,10326,10327],{},"Null pointer, resource leak, SQL-ін'єкції",[3205,10329,10330],{},"Кожен білд",[3187,10332,10333,10336,10338,10341],{},[3205,10334,10335],{},"3. Архітектурна якість",[3205,10337,9185],{},[3205,10339,10340],{},"Складність, дублювання, coverage, борг",[3205,10342,10343],{},"При PR/commit",[2964,10345,10346,10347,10349,10350,10353],{},"Рекомендований мінімальний набір для навчального проєкту — ",[2979,10348,3311],{}," (завжди) + ",[2979,10351,10352],{},"IntelliJ XML"," (для зручності в IDE). SpotBugs і SonarQube доречні у командній розробці та production-проєктах.",[2964,10355,10356,10357,10359],{},"Починаючи з наступної статті, весь код модуля дотримується Google Java Style: 2-пробільний відступ, 100-символьний рядок, відсортовані imports. ",[2968,10358,5231],{}," є обов'язковою умовою для будь-якого прикладу.",[4292,10361,10362],{},"html .light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html.light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html.dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html pre.shiki code .spJ8K, html code.shiki .spJ8K{--shiki-light:#008000;--shiki-default:#6A9955;--shiki-dark:#6A9955}html pre.shiki code .s8Opu, html code.shiki .s8Opu{--shiki-light:#795E26;--shiki-default:#DCDCAA;--shiki-dark:#DCDCAA}html pre.shiki code .sbdoH, html code.shiki .sbdoH{--shiki-light:#A31515;--shiki-default:#CE9178;--shiki-dark:#CE9178}html pre.shiki code .s0P7L, html code.shiki .s0P7L{--shiki-light:#800000;--shiki-default:#808080;--shiki-dark:#808080}html pre.shiki code .sKtos, html code.shiki .sKtos{--shiki-light:#800000;--shiki-default:#569CD6;--shiki-dark:#569CD6}html pre.shiki code .sHH4Y, html code.shiki .sHH4Y{--shiki-light:#000000;--shiki-default:#D4D4D4;--shiki-dark:#D4D4D4}html pre.shiki code .su9tN, html code.shiki .su9tN{--shiki-light:#0000FF;--shiki-default:#CE9178;--shiki-dark:#CE9178}html pre.shiki code .su1O8, html code.shiki .su1O8{--shiki-light:#0000FF;--shiki-default:#569CD6;--shiki-dark:#569CD6}html pre.shiki code .sN1BT, html code.shiki .sN1BT{--shiki-light:#267F99;--shiki-default:#4EC9B0;--shiki-dark:#4EC9B0}html pre.shiki code .sa4r_, html code.shiki .sa4r_{--shiki-light:#E50000;--shiki-default:#9CDCFE;--shiki-dark:#9CDCFE}html pre.shiki code .sjcCO, html code.shiki .sjcCO{--shiki-light:#EE0000;--shiki-default:#D7BA7D;--shiki-dark:#D7BA7D}html pre.shiki code .siwwj, html code.shiki .siwwj{--shiki-light:#001080;--shiki-default:#9CDCFE;--shiki-dark:#9CDCFE}",{"title":3100,"searchDepth":3355,"depth":3355,"links":10364},[10365,10366,10375,10382,10399,10404,10411,10412],{"id":2961,"depth":3355,"text":2962},{"id":3074,"depth":3355,"text":3075,"children":10367},[10368,10369,10370,10371,10372,10373,10374],{"id":3086,"depth":3361,"text":3087},{"id":3109,"depth":3361,"text":3110},{"id":3123,"depth":3361,"text":3124},{"id":3132,"depth":3361,"text":3133},{"id":3146,"depth":3361,"text":3147},{"id":3162,"depth":3361,"text":3163},{"id":3175,"depth":3361,"text":3176},{"id":3305,"depth":3355,"text":3306,"children":10376},[10377,10378,10379,10380,10381],{"id":3334,"depth":3361,"text":3335},{"id":3454,"depth":3361,"text":3455},{"id":4061,"depth":3361,"text":4062},{"id":4871,"depth":3361,"text":4872},{"id":5237,"depth":3361,"text":5238},{"id":5304,"depth":3355,"text":5305,"children":10383},[10384,10386,10388,10389,10390,10391,10392,10393,10394,10396,10398],{"id":5311,"depth":3361,"text":10385},"Глобальні параметри блоку spotless {}",{"id":5526,"depth":3361,"text":10387},"Блок java {} — Java-форматування",{"id":5708,"depth":3361,"text":5709},{"id":6214,"depth":3361,"text":6215},{"id":6761,"depth":3361,"text":6762},{"id":6903,"depth":3361,"text":6904},{"id":7078,"depth":3361,"text":7079},{"id":7299,"depth":3361,"text":7300},{"id":7546,"depth":3361,"text":10395},"Очищення коду: cleanthat",{"id":7746,"depth":3361,"text":10397},"Довідкова таблиця: всі параметри java {} одним поглядом",{"id":8097,"depth":3361,"text":8098},{"id":8311,"depth":3355,"text":8312,"children":10400},[10401,10402,10403],{"id":8333,"depth":3361,"text":8334},{"id":8448,"depth":3361,"text":8449},{"id":8821,"depth":3361,"text":8822},{"id":9162,"depth":3355,"text":9163,"children":10405},[10406,10407,10408,10409,10410],{"id":9169,"depth":3361,"text":9170},{"id":9278,"depth":3361,"text":9279},{"id":9409,"depth":3361,"text":8449},{"id":9637,"depth":3361,"text":8822},{"id":10052,"depth":3361,"text":9250},{"id":10128,"depth":3355,"text":10129},{"id":10275,"depth":3355,"text":10276},"Автоматизований контроль форматування та статичний аналіз у Java-проєкті: налаштування Google Java Style для IntelliJ IDEA, Spotless для Gradle та Maven, SpotBugs та SonarQube для виявлення дефектів.","md",null,{},{"title":2289,"description":10413},"WvvxkpzAzgiWxCF78pv9dPF9MZA8YKWuPGLJXJ2rt4A",[10420,10422],{"title":2285,"path":2286,"stem":2287,"description":10421,"children":-1},"Архітектура JDBC, перший робочий CRUD для аудіоплатформи, виявлення критичних проблем наївного підходу та їх усунення через PreparedStatement і try-with-resources.",{"title":2293,"path":2294,"stem":2295,"description":10423,"children":-1},"Чому створення Connection — дорога операція, як влаштований патерн Object Pool, реалізація власного пулу з'єднань з нуля через BlockingQueue та Proxy-обгортку, і чим наша реалізація відрізняється від HikariCP.",1777909225576]