[{"data":1,"prerenderedAt":11796},["ShallowReactive",2],{"navigation_docs":3,"-java-pr2-mvvm-guice-integration":3135,"-java-pr2-mvvm-guice-integration-surround":11791},[4,1669,1826,2280,2461,2668,2790,2840,2897,2931,3057,3094,3131],{"title":5,"icon":6,"path":7,"stem":8,"children":9},"C#","i-devicon-csharp","\u002Fcsharp","01.csharp",[10,13,60,90,120,202,219,253,379,404,457,650,1346,1636,1665],{"title":11,"path":7,"stem":12},"C# та .NET","01.csharp\u002Findex",{"title":14,"icon":15,"path":16,"stem":17,"children":18,"page":59},"Fundamentals","i-lucide-book-open","\u002Fcsharp\u002Ffundamentals","01.csharp\u002F01.fundamentals",[19,23,27,31,35,39,43,47,51,55],{"title":20,"path":21,"stem":22},"Вступ до екосистеми .NET","\u002Fcsharp\u002Ffundamentals\u002Fintroduction-to-ecosystem","01.csharp\u002F01.fundamentals\u002F01.introduction-to-ecosystem",{"title":24,"path":25,"stem":26},"Структура програми на C#","\u002Fcsharp\u002Ffundamentals\u002Fprogram-structure","01.csharp\u002F01.fundamentals\u002F02.program-structure",{"title":28,"path":29,"stem":30},"Змінні та Типи Даних","\u002Fcsharp\u002Ffundamentals\u002Fvariables-data-types","01.csharp\u002F01.fundamentals\u002F03.variables-data-types",{"title":32,"path":33,"stem":34},"Масиви","\u002Fcsharp\u002Ffundamentals\u002Farrays","01.csharp\u002F01.fundamentals\u002F04.arrays",{"title":36,"path":37,"stem":38},"Strings & Text Handling","\u002Fcsharp\u002Ffundamentals\u002Fstrings-text-handling","01.csharp\u002F01.fundamentals\u002F05.strings-text-handling",{"title":40,"path":41,"stem":42},"Дати і Час","\u002Fcsharp\u002Ffundamentals\u002Fdates-time-handling","01.csharp\u002F01.fundamentals\u002F06.dates-time-handling",{"title":44,"path":45,"stem":46},"Потік Керування","\u002Fcsharp\u002Ffundamentals\u002Fcontrol-flow","01.csharp\u002F01.fundamentals\u002F07.control-flow",{"title":48,"path":49,"stem":50},"Методи","\u002Fcsharp\u002Ffundamentals\u002Fmethods","01.csharp\u002F01.fundamentals\u002F08.methods",{"title":52,"path":53,"stem":54},"Основи Відлагодження","\u002Fcsharp\u002Ffundamentals\u002Fdebugging-basics","01.csharp\u002F01.fundamentals\u002F09.debugging-basics",{"title":56,"path":57,"stem":58},"Інтерактивна Консоль (Classic)","\u002Fcsharp\u002Ffundamentals\u002Finteractive-console","01.csharp\u002F01.fundamentals\u002F10.interactive-console",false,{"title":61,"icon":62,"path":63,"stem":64,"children":65,"page":59},"OOP","i-lucide-box","\u002Fcsharp\u002Foop","01.csharp\u002F02.oop",[66,70,74,78,82,86],{"title":67,"path":68,"stem":69},"Package Management (Управління Пакетами)","\u002Fcsharp\u002Foop\u002Fpackage-management","01.csharp\u002F02.oop\u002F01.package-management",{"title":71,"path":72,"stem":73},"Класи та Об'єкти","\u002Fcsharp\u002Foop\u002Fclasses-objects","01.csharp\u002F02.oop\u002F02.classes-objects",{"title":75,"path":76,"stem":77},"Властивості та Поля","\u002Fcsharp\u002Foop\u002Fproperties-fields","01.csharp\u002F02.oop\u002F03.properties-fields",{"title":79,"path":80,"stem":81},"Стовпи ООП","\u002Fcsharp\u002Foop\u002Foop-pillars","01.csharp\u002F02.oop\u002F04.oop-pillars",{"title":83,"path":84,"stem":85},"Advanced Types","\u002Fcsharp\u002Foop\u002Fadvanced-types","01.csharp\u002F02.oop\u002F05.advanced-types",{"title":87,"path":88,"stem":89},"Namespaces (Простори Імен)","\u002Fcsharp\u002Foop\u002Fnamespaces","01.csharp\u002F02.oop\u002F06.namespaces",{"title":91,"icon":92,"path":93,"stem":94,"children":95,"page":59},"Advanced Core","i-lucide-zap","\u002Fcsharp\u002Fadvanced-core","01.csharp\u002F03.advanced-core",[96,100,104,108,112,116],{"title":97,"path":98,"stem":99},"Generics (Узагальнення)","\u002Fcsharp\u002Fadvanced-core\u002Fgenerics","01.csharp\u002F03.advanced-core\u002F01.generics",{"title":101,"path":102,"stem":103},"Делегати, Події та Лямбда-вирази","\u002Fcsharp\u002Fadvanced-core\u002Fdelegates-events-lambdas","01.csharp\u002F03.advanced-core\u002F02.delegates-events-lambdas",{"title":105,"path":106,"stem":107},"Interfaces Deep Dive (Інтерфейси: Поглиблений Розгляд)","\u002Fcsharp\u002Fadvanced-core\u002Finterfaces-deep-dive","01.csharp\u002F03.advanced-core\u002F03.interfaces-deep-dive",{"title":109,"path":110,"stem":111},"Обробка Винятків","\u002Fcsharp\u002Fadvanced-core\u002Fexception-handling","01.csharp\u002F03.advanced-core\u002F04.exception-handling",{"title":113,"path":114,"stem":115},"Pattern Matching","\u002Fcsharp\u002Fadvanced-core\u002Fpattern-matching","01.csharp\u002F03.advanced-core\u002F05.pattern-matching",{"title":117,"path":118,"stem":119},"Додаткові Можливості C#","\u002Fcsharp\u002Fadvanced-core\u002Fadditional-features","01.csharp\u002F03.advanced-core\u002F06.additional-features",{"title":121,"icon":122,"path":123,"stem":124,"children":125,"page":59},"Architecture Best Practices","i-lucide-building-2","\u002Fcsharp\u002Farchitecture-best-practices","01.csharp\u002F04.architecture-best-practices",[126,130,149,153,157,161,165,169],{"title":127,"path":128,"stem":129},"Software Design Principles (Частина 1)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fsoftware-design-principles","01.csharp\u002F04.architecture-best-practices\u002F01.software-design-principles",{"title":131,"icon":132,"path":133,"stem":134,"children":135,"page":59},"Design Patterns","i-lucide-folder","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdesign-patterns","01.csharp\u002F04.architecture-best-practices\u002F02.design-patterns",[136],{"title":137,"icon":132,"path":138,"stem":139,"children":140,"page":59},"Creational","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdesign-patterns\u002Fcreational","01.csharp\u002F04.architecture-best-practices\u002F02.design-patterns\u002Fcreational",[141,145],{"title":142,"path":143,"stem":144},"Singleton (Одинак)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdesign-patterns\u002Fcreational\u002Fsingleton","01.csharp\u002F04.architecture-best-practices\u002F02.design-patterns\u002Fcreational\u002F01.singleton",{"title":146,"path":147,"stem":148},"Builder (Будівельник)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdesign-patterns\u002Fcreational\u002Fbuilder","01.csharp\u002F04.architecture-best-practices\u002F02.design-patterns\u002Fcreational\u002F02.builder",{"title":150,"path":151,"stem":152},"Building Professional CLIs","\u002Fcsharp\u002Farchitecture-best-practices\u002Fbuilding-professional-clis","01.csharp\u002F04.architecture-best-practices\u002F03.building-professional-clis",{"title":154,"path":155,"stem":156},"Validation & Flow Control","\u002Fcsharp\u002Farchitecture-best-practices\u002Fvalidation-flow-control","01.csharp\u002F04.architecture-best-practices\u002F04.validation-flow-control",{"title":158,"path":159,"stem":160},"The Modern .NET Host (Microsoft.Extensions)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fmodern-dotnet-host","01.csharp\u002F04.architecture-best-practices\u002F05.modern-dotnet-host",{"title":162,"path":163,"stem":164},"Data Mapper: Repository та DAO патерни (Частина 1)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdata-mapper-part1","01.csharp\u002F04.architecture-best-practices\u002F06.data-mapper-part1",{"title":166,"path":167,"stem":168},"Data Mapper: Repository та DAO патерни (Частина 2)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdata-mapper-part2","01.csharp\u002F04.architecture-best-practices\u002F07.data-mapper-part2",{"title":170,"icon":132,"path":171,"stem":172,"children":173,"page":59},"Di Ioc","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc",[174,178,182,186,190,194,198],{"title":175,"path":176,"stem":177},"Проблема залежностей та Інверсія Контролю","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fthe-dependency-problem","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F01.the-dependency-problem",{"title":179,"path":180,"stem":181},"Будуємо власний Service Container","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fbuild-your-own-container","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F02.build-your-own-container",{"title":183,"path":184,"stem":185},"Service Locator: Паттерн та Анти-паттерн","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fservice-locator-pattern","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F03.service-locator-pattern",{"title":187,"path":188,"stem":189},"Паттерни Dependency Injection","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fdependency-injection-patterns","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F04.dependency-injection-patterns",{"title":191,"path":192,"stem":193},"Microsoft DI: IServiceCollection та IServiceProvider","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fmicrosoft-di-deep-dive","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F05.microsoft-di-deep-dive",{"title":195,"path":196,"stem":197},"Service Lifetimes та Scopes","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fservice-lifetimes-and-scopes","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F06.service-lifetimes-and-scopes",{"title":199,"path":200,"stem":201},"DI Анти-паттерни та Найкращі Практики","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fdi-anti-patterns-and-best-practices","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F07.di-anti-patterns-and-best-practices",{"title":203,"icon":132,"path":204,"stem":205,"children":206,"page":59},"Standard Library","\u002Fcsharp\u002Fstandard-library","01.csharp\u002F05.standard-library",[207,211,215],{"title":208,"path":209,"stem":210},"Collections (Колекції)","\u002Fcsharp\u002Fstandard-library\u002Fcollections","01.csharp\u002F05.standard-library\u002F01.collections",{"title":212,"path":213,"stem":214},"High Performance Types (Високопродуктивні Типи)","\u002Fcsharp\u002Fstandard-library\u002Fhigh-performance-types","01.csharp\u002F05.standard-library\u002F02.high-performance-types",{"title":216,"path":217,"stem":218},"LINQ (Language Integrated Query)","\u002Fcsharp\u002Fstandard-library\u002Flinq","01.csharp\u002F05.standard-library\u002F03.linq",{"title":220,"icon":221,"path":222,"stem":223,"children":224,"page":59},"System Internals Concurrency","i-lucide-server","\u002Fcsharp\u002Fsystem-internals-concurrency","01.csharp\u002F06.system-internals-concurrency",[225,229,233,237,241,245,249],{"title":226,"path":227,"stem":228},"Memory Management","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fmemory-management","01.csharp\u002F06.system-internals-concurrency\u002F01.memory-management",{"title":230,"path":231,"stem":232},"Reflection API: System.Type та Метадані","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Freflection-fundamentals","01.csharp\u002F06.system-internals-concurrency\u002F02.reflection-fundamentals",{"title":234,"path":235,"stem":236},"Attributes та Dynamic Language Runtime","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fattributes-dynamic","01.csharp\u002F06.system-internals-concurrency\u002F03.attributes-dynamic",{"title":238,"path":239,"stem":240},"Expression Trees: Швидка Альтернатива Рефлексії","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fexpression-trees-compiled","01.csharp\u002F06.system-internals-concurrency\u002F04.expression-trees-compiled",{"title":242,"path":243,"stem":244},"Source Generators: Compile-Time Code Generation","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fsource-generators","01.csharp\u002F06.system-internals-concurrency\u002F05.source-generators",{"title":246,"path":247,"stem":248},"Multithreading Fundamentals","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fmultithreading-fundamentals","01.csharp\u002F06.system-internals-concurrency\u002F06.multithreading-fundamentals",{"title":250,"path":251,"stem":252},"Synchronization Primitives","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fsynchronization-primitives","01.csharp\u002F06.system-internals-concurrency\u002F07.synchronization-primitives",{"title":254,"icon":255,"path":256,"stem":257,"children":258,"page":59},"System Programming Windows","i-lucide-cpu","\u002Fcsharp\u002Fsystem-programming-windows","01.csharp\u002F07.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},"Як Працює Операційна Система","\u002Fcsharp\u002Fsystem-programming-windows\u002Fhow-os-works","01.csharp\u002F07.system-programming-windows\u002F01.how-os-works",{"title":264,"path":265,"stem":266},"Процеси в .NET — API та Запуск","\u002Fcsharp\u002Fsystem-programming-windows\u002Fprocesses-in-dotnet","01.csharp\u002F07.system-programming-windows\u002F02.processes-in-dotnet",{"title":268,"path":269,"stem":270},"Процеси в .NET — IPC та Міжпроцесна Комунікація","\u002Fcsharp\u002Fsystem-programming-windows\u002F02a.processes-ipc","01.csharp\u002F07.system-programming-windows\u002F02a.processes-ipc",{"title":272,"path":273,"stem":274},"Application Domains та Збірки — AppDomain і AssemblyLoadContext","\u002Fcsharp\u002Fsystem-programming-windows\u002Fappdomains-assemblies","01.csharp\u002F07.system-programming-windows\u002F03.appdomains-assemblies",{"title":276,"path":277,"stem":278},"Application Domains та Збірки — Plug-in Система з Hot-Reload","\u002Fcsharp\u002Fsystem-programming-windows\u002F03a.appdomains-plugin-system","01.csharp\u002F07.system-programming-windows\u002F03a.appdomains-plugin-system",{"title":280,"path":281,"stem":282},"Потоки — Основи та API Thread","\u002Fcsharp\u002Fsystem-programming-windows\u002Fthread-fundamentals","01.csharp\u002F07.system-programming-windows\u002F04.thread-fundamentals",{"title":284,"path":285,"stem":286},"Потоки — Lifecycle, Пріоритети та Безпечне Завершення","\u002Fcsharp\u002Fsystem-programming-windows\u002F04a.thread-lifecycle-priorities","01.csharp\u002F07.system-programming-windows\u002F04a.thread-lifecycle-priorities",{"title":288,"path":289,"stem":290},"Проблеми Спільного Стану — Race Condition та Data Race","\u002Fcsharp\u002Fsystem-programming-windows\u002Fshared-state-problems","01.csharp\u002F07.system-programming-windows\u002F05.shared-state-problems",{"title":292,"path":293,"stem":294},"Проблеми Спільного Стану — Memory Model та volatile","\u002Fcsharp\u002Fsystem-programming-windows\u002F05a.shared-state-memory-model","01.csharp\u002F07.system-programming-windows\u002F05a.shared-state-memory-model",{"title":296,"path":297,"stem":298},"Синхронізація — Monitor, lock та еволюція примітивів","\u002Fcsharp\u002Fsystem-programming-windows\u002Fsynchronization-fundamentals","01.csharp\u002F07.system-programming-windows\u002F06.synchronization-fundamentals",{"title":300,"path":301,"stem":302},"Синхронізація — Наскрізний Приклад та Deadlock Detection","\u002Fcsharp\u002Fsystem-programming-windows\u002F06a.synchronization-walkthrough","01.csharp\u002F07.system-programming-windows\u002F06a.synchronization-walkthrough",{"title":304,"path":305,"stem":306},"Синхронізація — Mutex, Semaphore та Event-Based Primitives","\u002Fcsharp\u002Fsystem-programming-windows\u002Fsynchronization-advanced","01.csharp\u002F07.system-programming-windows\u002F07.synchronization-advanced",{"title":308,"path":309,"stem":310},"Синхронізація — Interlocked, Volatile та Lock-Free Структури","\u002Fcsharp\u002Fsystem-programming-windows\u002F07a.synchronization-advanced-walkthrough","01.csharp\u002F07.system-programming-windows\u002F07a.synchronization-advanced-walkthrough",{"title":312,"path":313,"stem":314},"Interlocked, CAS та Lock-Free Структури","\u002Fcsharp\u002Fsystem-programming-windows\u002Finterlocked-cas-lockfree","01.csharp\u002F07.system-programming-windows\u002F08.interlocked-cas-lockfree",{"title":316,"path":317,"stem":318},"Volatile, Memory Model та Spinning","\u002Fcsharp\u002Fsystem-programming-windows\u002F08a.volatile-memory-model","01.csharp\u002F07.system-programming-windows\u002F08a.volatile-memory-model",{"title":320,"path":321,"stem":322},"ThreadPool — Пул Потоків для Ефективного Виконання","\u002Fcsharp\u002Fsystem-programming-windows\u002Fthread-pool","01.csharp\u002F07.system-programming-windows\u002F09.thread-pool",{"title":324,"path":325,"stem":326},"ThreadPool — Просунуті Сценарії та Внутрішня Будова","\u002Fcsharp\u002Fsystem-programming-windows\u002F09a.thread-pool-advanced","01.csharp\u002F07.system-programming-windows\u002F09a.thread-pool-advanced",{"title":328,"path":329,"stem":330},"Concurrent та Immutable Collections","\u002Fcsharp\u002Fsystem-programming-windows\u002Fconcurrent-collections","01.csharp\u002F07.system-programming-windows\u002F10.concurrent-collections",{"title":332,"path":333,"stem":334},"TPL, Task та Композиція — Від Thread до Task","\u002Fcsharp\u002Fsystem-programming-windows\u002Ftpl-parallel-plinq","01.csharp\u002F07.system-programming-windows\u002F11.tpl-parallel-plinq",{"title":336,"path":337,"stem":338},"Parallel Class та PLINQ — Data Parallelism","\u002Fcsharp\u002Fsystem-programming-windows\u002F11a.tpl-parallel-plinq-advanced","01.csharp\u002F07.system-programming-windows\u002F11a.tpl-parallel-plinq-advanced",{"title":340,"path":341,"stem":342},"Async\u002FAwait — Фундамент Асинхронного Програмування","\u002Fcsharp\u002Fsystem-programming-windows\u002Fasync-fundamentals","01.csharp\u002F07.system-programming-windows\u002F12.async-fundamentals",{"title":344,"path":345,"stem":346},"SynchronizationContext та ConfigureAwait — Контекст Виконання","\u002Fcsharp\u002Fsystem-programming-windows\u002Fasync-context-configureawait","01.csharp\u002F07.system-programming-windows\u002F13.async-context-configureawait",{"title":348,"path":349,"stem":350},"Async — Просунуті Паттерни","\u002Fcsharp\u002Fsystem-programming-windows\u002Fasync-advanced","01.csharp\u002F07.system-programming-windows\u002F14.async-advanced",{"title":352,"path":353,"stem":354},"System.Threading.Channels — Async Producer-Consumer","\u002Fcsharp\u002Fsystem-programming-windows\u002Fchannels","01.csharp\u002F07.system-programming-windows\u002F15.channels",{"title":356,"path":357,"stem":358},"Асинхронна Синхронізація","\u002Fcsharp\u002Fsystem-programming-windows\u002Fasync-synchronization","01.csharp\u002F07.system-programming-windows\u002F16.async-synchronization",{"title":360,"path":361,"stem":362},"Unsafe Code та Вказівники","\u002Fcsharp\u002Fsystem-programming-windows\u002Funsafe-code","01.csharp\u002F07.system-programming-windows\u002F17.unsafe-code",{"title":364,"path":365,"stem":366},"P\u002FInvoke та Windows API — Міст між .NET та Native Code","\u002Fcsharp\u002Fsystem-programming-windows\u002Fpinvoke-winapi","01.csharp\u002F07.system-programming-windows\u002F18.pinvoke-winapi",{"title":368,"path":369,"stem":370},"Реєстр Windows — Центральна База Конфігурації Системи","\u002Fcsharp\u002Fsystem-programming-windows\u002Fwindows-registry","01.csharp\u002F07.system-programming-windows\u002F19.windows-registry",{"title":372,"path":373,"stem":374},"Windows Hooks, Hotkeys та Services — Глибока Інтеграція з ОС","\u002Fcsharp\u002Fsystem-programming-windows\u002Fwindows-hooks-services","01.csharp\u002F07.system-programming-windows\u002F20.windows-hooks-services",{"title":376,"path":377,"stem":378},"Системне Програмування C# (Windows) — 07.system-programming-windows","\u002Fcsharp\u002Fsystem-programming-windows\u002Fimplementation_plan","01.csharp\u002F07.system-programming-windows\u002Fimplementation_plan",{"title":380,"icon":132,"path":381,"stem":382,"children":383,"page":59},"Io","\u002Fcsharp\u002Fio","01.csharp\u002F08.io",[384,388,392,396,400],{"title":385,"path":386,"stem":387},"8.1.1. Основи роботи з файловою системою","\u002Fcsharp\u002Fio\u002Ffile-system-basics","01.csharp\u002F08.io\u002F01.file-system-basics",{"title":389,"path":390,"stem":391},"8.1.2. Потоки (Streams) та Серіалізація Даних","\u002Fcsharp\u002Fio\u002Fstreams-serialization","01.csharp\u002F08.io\u002F02.streams-serialization",{"title":393,"path":394,"stem":395},"8.2.1. JSON Serialization з System.Text.Json","\u002Fcsharp\u002Fio\u002Fjson-serialization","01.csharp\u002F08.io\u002F03.json-serialization",{"title":397,"path":398,"stem":399},"8.2.2. XML Serialization та LINQ to XML","\u002Fcsharp\u002Fio\u002Fxml-serialization","01.csharp\u002F08.io\u002F04.xml-serialization",{"title":401,"path":402,"stem":403},"8.2.3. Binary Serialization: MessagePack та Protocol Buffers","\u002Fcsharp\u002Fio\u002Fbinary-serialization","01.csharp\u002F08.io\u002F05.binary-serialization",{"title":405,"icon":132,"path":406,"stem":407,"children":408,"page":59},"Ado Net","\u002Fcsharp\u002Fado-net","01.csharp\u002F09.ado-net",[409,413,417,421,425,429,433,437,441,445,449,453],{"title":410,"path":411,"stem":412},"9.1. Введення в ADO.NET","\u002Fcsharp\u002Fado-net\u002Fintroduction-to-adonet","01.csharp\u002F09.ado-net\u002F01.introduction-to-adonet",{"title":414,"path":415,"stem":416},"9.2. Клас DbConnection — з'єднання з базою даних","\u002Fcsharp\u002Fado-net\u002Fconnection","01.csharp\u002F09.ado-net\u002F02.connection",{"title":418,"path":419,"stem":420},"9.3. Клас DbCommand — виконання SQL-запитів","\u002Fcsharp\u002Fado-net\u002Fcommand-and-queries","01.csharp\u002F09.ado-net\u002F03.command-and-queries",{"title":422,"path":423,"stem":424},"9.4. Клас DbDataReader — ефективне читання даних","\u002Fcsharp\u002Fado-net\u002Fdatareader","01.csharp\u002F09.ado-net\u002F04.datareader",{"title":426,"path":427,"stem":428},"9.5. Параметризовані запити та захист від SQL Injection","\u002Fcsharp\u002Fado-net\u002Fparameters-and-sql-injection","01.csharp\u002F09.ado-net\u002F05.parameters-and-sql-injection",{"title":430,"path":431,"stem":432},"9.6. Транзакції в ADO.NET","\u002Fcsharp\u002Fado-net\u002Ftransactions","01.csharp\u002F09.ado-net\u002F06.transactions",{"title":434,"path":435,"stem":436},"9.7. DbProviderFactory — провайдер-незалежний код","\u002Fcsharp\u002Fado-net\u002Fprovider-factory","01.csharp\u002F09.ado-net\u002F07.provider-factory",{"title":438,"path":439,"stem":440},"9.8. Асинхронний доступ до даних","\u002Fcsharp\u002Fado-net\u002Fasync-data-access","01.csharp\u002F09.ado-net\u002F08.async-data-access",{"title":442,"path":443,"stem":444},"9.9. Від'єднаний режим: DataSet, DataTable, DataRow","\u002Fcsharp\u002Fado-net\u002Fdisconnected-mode-dataset","01.csharp\u002F09.ado-net\u002F09.disconnected-mode-dataset",{"title":446,"path":447,"stem":448},"9.10. DataAdapter — міст між DataSet та базою даних","\u002Fcsharp\u002Fado-net\u002Fdata-adapter","01.csharp\u002F09.ado-net\u002F10.data-adapter",{"title":450,"path":451,"stem":452},"9.11. Data Mapper та Repository: Архітектура доступу до даних","\u002Fcsharp\u002Fado-net\u002Fdata-mapper-repository","01.csharp\u002F09.ado-net\u002F11.data-mapper-repository",{"title":454,"path":455,"stem":456},"9.12. Identity Map, Unit of Work та Specification Pattern","\u002Fcsharp\u002Fado-net\u002Fadvanced-patterns","01.csharp\u002F09.ado-net\u002F12.advanced-patterns",{"title":458,"icon":255,"path":459,"stem":460,"children":461,"page":59},"Ef Core","\u002Fcsharp\u002Fef-core","01.csharp\u002F10.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 до об'єктів","\u002Fcsharp\u002Fef-core\u002Fwhat-is-orm","01.csharp\u002F10.ef-core\u002F01.what-is-orm",{"title":467,"path":468,"stem":469},"Перший проєкт — від нуля до CRUD","\u002Fcsharp\u002Fef-core\u002Ffirst-project","01.csharp\u002F10.ef-core\u002F02.first-project",{"title":471,"path":472,"stem":473},"DbContext — Серце EF Core","\u002Fcsharp\u002Fef-core\u002Fdbcontext-deep-dive","01.csharp\u002F10.ef-core\u002F03.dbcontext-deep-dive",{"title":475,"path":476,"stem":477},"Провайдери баз даних — Архітектура та Вибір СУБД","\u002Fcsharp\u002Fef-core\u002Fdatabase-providers","01.csharp\u002F10.ef-core\u002F04.database-providers",{"title":479,"path":480,"stem":481},"Конвенції EF Core — Магія без конфігурації","\u002Fcsharp\u002Fef-core\u002Fconventions","01.csharp\u002F10.ef-core\u002F05.conventions",{"title":483,"path":484,"stem":485},"Fluent API та Data Annotations — Явна конфігурація моделі","\u002Fcsharp\u002Fef-core\u002Ffluent-api-vs-annotations","01.csharp\u002F10.ef-core\u002F06.fluent-api-vs-annotations",{"title":487,"path":488,"stem":489},"Зв'язки — One-to-One та One-to-Many","\u002Fcsharp\u002Fef-core\u002Frelationships-basics","01.csharp\u002F10.ef-core\u002F07.relationships-basics",{"title":491,"path":492,"stem":493},"Зв'язки Advanced — Many-to-Many та Складні Сценарії","\u002Fcsharp\u002Fef-core\u002Frelationships-advanced","01.csharp\u002F10.ef-core\u002F08.relationships-advanced",{"title":495,"path":496,"stem":497},"Властивості — Типи, Конвертери, Компаратори (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fproperty-configuration-part1","01.csharp\u002F10.ef-core\u002F09.property-configuration-part1",{"title":499,"path":500,"stem":501},"Властивості — Value Comparers, Generators, Shadow Properties (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fproperty-configuration-part2","01.csharp\u002F10.ef-core\u002F09.property-configuration-part2",{"title":503,"path":504,"stem":505},"Складні типи — Owned Types та Complex Types (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fcomplex-types-owned-part1","01.csharp\u002F10.ef-core\u002F10.complex-types-owned-part1",{"title":507,"path":508,"stem":509},"Складні типи — Complex Types, Keyless Entities, Порівняння (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fcomplex-types-owned-part2","01.csharp\u002F10.ef-core\u002F10.complex-types-owned-part2",{"title":511,"path":512,"stem":513},"JSON Columns — Складні дані у JSON (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fjson-columns-part1","01.csharp\u002F10.ef-core\u002F11.json-columns-part1",{"title":515,"path":516,"stem":517},"JSON Columns — Value Comparers, Індекси, Провайдери (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fjson-columns-part2","01.csharp\u002F10.ef-core\u002F11.json-columns-part2",{"title":519,"path":520,"stem":521},"Успадкування — Абстрактні класи та TPH (Частина 1)","\u002Fcsharp\u002Fef-core\u002Finheritance-part1","01.csharp\u002F10.ef-core\u002F12.inheritance-part1",{"title":523,"path":524,"stem":525},"Успадкування — TPT, TPC та Порівняння Стратегій (Частина 2)","\u002Fcsharp\u002Fef-core\u002Finheritance-part2","01.csharp\u002F10.ef-core\u002F12.inheritance-part2",{"title":527,"path":528,"stem":529,"children":530},"Індекси, Обмеження та Схема (Частина 1)","\u002Fcsharp\u002Fef-core\u002Findexes-constraints-part1","01.csharp\u002F10.ef-core\u002F13.indexes-constraints-part1",[531],{"title":527,"path":528,"stem":529},{"title":533,"path":534,"stem":535,"children":536},"Індекси, Обмеження та Схема (Частина 2)","\u002Fcsharp\u002Fef-core\u002Findexes-constraints-part2","01.csharp\u002F10.ef-core\u002F13.indexes-constraints-part2",[537],{"title":533,"path":534,"stem":535},{"title":539,"path":540,"stem":541},"Seed Data — Початкові Дані (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fseeding-part1","01.csharp\u002F10.ef-core\u002F14.seeding-part1",{"title":543,"path":544,"stem":545},"Seed Data — SQL-скрипти, Bogus та Стратегії (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fseeding-part2","01.csharp\u002F10.ef-core\u002F14.seeding-part2",{"title":547,"path":548,"stem":549},"Global Query Filters — Глобальні Фільтри (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fglobal-query-filters-part1","01.csharp\u002F10.ef-core\u002F15.global-query-filters-part1",{"title":551,"path":552,"stem":553},"Global Query Filters — Підводні камені та Інтеграція (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fglobal-query-filters-part2","01.csharp\u002F10.ef-core\u002F15.global-query-filters-part2",{"title":555,"path":556,"stem":557},"LINQ-запити в EF Core (Частина 1)","\u002Fcsharp\u002Fef-core\u002Flinq-queries-part1","01.csharp\u002F10.ef-core\u002F16.linq-queries-part1",{"title":559,"path":560,"stem":561},"LINQ-запити в EF Core (Частина 2)","\u002Fcsharp\u002Fef-core\u002Flinq-queries-part2","01.csharp\u002F10.ef-core\u002F16.linq-queries-part2",{"title":563,"path":564,"stem":565},"Завантаження Пов'язаних Даних (Частина 1)","\u002Fcsharp\u002Fef-core\u002Floading-related-data-part1","01.csharp\u002F10.ef-core\u002F17.loading-related-data-part1",{"title":567,"path":568,"stem":569},"Завантаження Пов'язаних Даних (Частина 2)","\u002Fcsharp\u002Fef-core\u002Floading-related-data-part2","01.csharp\u002F10.ef-core\u002F17.loading-related-data-part2",{"title":571,"path":572,"stem":573},"Raw SQL, Views та Stored Procedures (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fraw-sql-part1","01.csharp\u002F10.ef-core\u002F18.raw-sql-part1",{"title":575,"path":576,"stem":577},"Raw SQL — Stored Procedures, DbFunction та Bulk Operations (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fraw-sql-part2","01.csharp\u002F10.ef-core\u002F18.raw-sql-part2",{"title":579,"path":580,"stem":581},"Продвинуті Запити — Compiled Queries, Bulk та Оптимізація (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fadvanced-queries-part1","01.csharp\u002F10.ef-core\u002F19.advanced-queries-part1",{"title":583,"path":584,"stem":585},"Продвинуті Запити — Query Tags, Bulk та Interceptors (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fadvanced-queries-part2","01.csharp\u002F10.ef-core\u002F19.advanced-queries-part2",{"title":587,"path":588,"stem":589},"Change Tracker — Відстеження Змін (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fchange-tracking-part1","01.csharp\u002F10.ef-core\u002F20.change-tracking-part1",{"title":591,"path":592,"stem":593},"Change Tracker — Графи Об'єктів та Disconnected (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fchange-tracking-part2","01.csharp\u002F10.ef-core\u002F20.change-tracking-part2",{"title":595,"path":596,"stem":597},"Збереження Даних та Транзакції (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fsaving-data-part1","01.csharp\u002F10.ef-core\u002F21.saving-data-part1",{"title":599,"path":600,"stem":601},"Збереження Даних — Concurrency та Outbox (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fsaving-data-part2","01.csharp\u002F10.ef-core\u002F21.saving-data-part2",{"title":603,"path":604,"stem":605},"Конкурентність та Блокування (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fconcurrency-part1","01.csharp\u002F10.ef-core\u002F22.concurrency-part1",{"title":607,"path":608,"stem":609},"Конкурентність — Дедлоки та Queue Processing (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fconcurrency-part2","01.csharp\u002F10.ef-core\u002F22.concurrency-part2",{"title":611,"path":612,"stem":613},"Міграції в EF Core — Основи (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fmigrations-basics-part1","01.csharp\u002F10.ef-core\u002F23.migrations-basics-part1",{"title":615,"path":616,"stem":617},"Міграції в EF Core — Основи (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fmigrations-basics-part2","01.csharp\u002F10.ef-core\u002F23.migrations-basics-part2",{"title":619,"path":620,"stem":621},"Міграції — Просунуті Сценарії (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fmigrations-advanced-part1","01.csharp\u002F10.ef-core\u002F24.migrations-advanced-part1",{"title":623,"path":624,"stem":625},"Міграції — Просунуті Сценарії (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fmigrations-advanced-part2","01.csharp\u002F10.ef-core\u002F24.migrations-advanced-part2",{"title":627,"path":628,"stem":629},"Управління Схемою та Database-First (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fschema-management-part1","01.csharp\u002F10.ef-core\u002F25.schema-management-part1",{"title":631,"path":632,"stem":633},"Управління Схемою та Database-First (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fschema-management-part2","01.csharp\u002F10.ef-core\u002F25.schema-management-part2",{"title":635,"path":636,"stem":637},"Продуктивність EF Core — Основи (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fperformance-fundamentals-part1","01.csharp\u002F10.ef-core\u002F26.performance-fundamentals-part1",{"title":639,"path":640,"stem":641},"Interceptors в EF Core (Частина 1)","\u002Fcsharp\u002Fef-core\u002Finterceptors-part1","01.csharp\u002F10.ef-core\u002F29.interceptors-part1",{"title":643,"path":644,"stem":645},"Interceptors в EF Core — Connection, Transaction та Materialization (Частина 2)","\u002Fcsharp\u002Fef-core\u002Finterceptors-part2","01.csharp\u002F10.ef-core\u002F29.interceptors-part2",{"title":647,"path":648,"stem":649},"План вивчення Entity Framework Core — Повний курс","\u002Fcsharp\u002Fef-core\u002Fimplementation_plan","01.csharp\u002F10.ef-core\u002Fimplementation_plan",{"title":651,"icon":652,"path":653,"stem":654,"children":655,"page":59},"ASP.NET","i-devicon-dotnetcore","\u002Fcsharp\u002Faspnet","01.csharp\u002F11.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","\u002Fcsharp\u002Faspnet\u002Fminimal-api","01.csharp\u002F11.aspnet\u002F01.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 та еволюція фреймворку","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fintroduction","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F01.introduction",{"title":667,"path":668,"stem":669},"Перший додаток на ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Ffirst-application","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F02.first-application",{"title":671,"path":672,"stem":673},"WebApplication, Builder та Dependency Injection","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fwebapplication-builder","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F03.webapplication-builder",{"title":675,"path":676,"stem":677},"Конвеєр запитів та Middleware","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Frequest-pipeline-middleware","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F04.request-pipeline-middleware",{"title":679,"path":680,"stem":681},"Маршрутизація в ASP.NET Core: Основи","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Frouting-basics","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F05.routing-basics",{"title":683,"path":684,"stem":685},"Маршрутизація в ASP.NET Core: Розширені можливості","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Frouting-advanced","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F06.routing-advanced",{"title":687,"path":688,"stem":689},"Статичні файли в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fstatic-files","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F07.static-files",{"title":691,"path":692,"stem":693},"Статичні Активи: MapStaticAssets (ASP.NET Core 9.0)","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fstatic-assets","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F08.static-assets",{"title":695,"path":696,"stem":697},"Конфігурація в ASP.NET Core: Основи","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fconfiguration-fundamentals","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F09.configuration-fundamentals",{"title":699,"path":700,"stem":701},"Конфігурація: Паттерн Options","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fconfiguration-options","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F10.configuration-options",{"title":703,"path":704,"stem":705},"Логування в ASP.NET Core: Основи","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Flogging-basics","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F11.logging-basics",{"title":707,"path":708,"stem":709},"Логування: Serilog та Middleware","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Flogging-advanced","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F12.logging-advanced",{"title":711,"path":712,"stem":713},"Управління станом: HttpContext.Items та Cookies","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fstate-management","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F13.state-management",{"title":715,"path":716,"stem":717},"Стан сесії: Sessions","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fsession-state","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F14.session-state",{"title":719,"path":720,"stem":721},"Структура проєкту: від хаосу до архітектури","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fproject-structure","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F15.project-structure",{"title":723,"path":724,"stem":725},"Scalar у Minimal API: повний проєкт і Fluent OpenAPI","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fscalar-openapi-fluent","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F16.scalar-openapi-fluent",{"title":727,"path":728,"stem":729},"Swagger \u002F Swashbuckle у Minimal API: окремий класичний шлях","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fswagger-swashbuckle","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F17.swagger-swashbuckle",{"title":731,"icon":658,"path":732,"stem":733,"children":734,"page":59},"API","\u002Fcsharp\u002Faspnet\u002Fapi","01.csharp\u002F11.aspnet\u002F02.api",[735,739,743,747,751,755,759,763,767,771,775,779,783,787],{"title":736,"path":737,"stem":738},"Що таке API. Клієнт-серверна архітектура","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fwhat-is-api","01.csharp\u002F11.aspnet\u002F02.api\u002F01.what-is-api",{"title":740,"path":741,"stem":742},"Формати даних: JSON, XML, TOML та бінарні формати","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fdata-formats","01.csharp\u002F11.aspnet\u002F02.api\u002F02.data-formats",{"title":744,"path":745,"stem":746},"Парадигми API та концепція REST","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fapi-paradigms-rest","01.csharp\u002F11.aspnet\u002F02.api\u002F03.api-paradigms-rest",{"title":748,"path":749,"stem":750},"HTTP-методи, статус-коди та заголовки","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fhttp-methods-status-codes","01.csharp\u002F11.aspnet\u002F02.api\u002F04.http-methods-status-codes",{"title":752,"path":753,"stem":754},"Організація HTTP API за принципами REST","\u002Fcsharp\u002Faspnet\u002Fapi\u002Frest-organizing","01.csharp\u002F11.aspnet\u002F02.api\u002F05.rest-organizing",{"title":756,"path":757,"stem":758},"Номенклатура URL та CRUD-операції","\u002Fcsharp\u002Faspnet\u002Fapi\u002Furl-nomenclature-crud","01.csharp\u002F11.aspnet\u002F02.api\u002F06.url-nomenclature-crud",{"title":760,"path":761,"stem":762},"Правила дизайну: іменування та стандарти","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fapi-design-naming","01.csharp\u002F11.aspnet\u002F02.api\u002F07.api-design-naming",{"title":764,"path":765,"stem":766},"Валідація, ліміти та обробка помилок","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fapi-design-validation","01.csharp\u002F11.aspnet\u002F02.api\u002F08.api-design-validation",{"title":768,"path":769,"stem":770},"Обробка помилок у Minimal API","\u002Fcsharp\u002Faspnet\u002Fapi\u002Ferror-handling-http","01.csharp\u002F11.aspnet\u002F02.api\u002F09.error-handling-http",{"title":772,"path":773,"stem":774},"Ідемпотентність та синхронізація стану","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fidempotency-sync","01.csharp\u002F11.aspnet\u002F02.api\u002F10.idempotency-sync",{"title":776,"path":777,"stem":778},"Пагінація та організація списків","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fpagination-lists","01.csharp\u002F11.aspnet\u002F02.api\u002F11.pagination-lists",{"title":780,"path":781,"stem":782},"Безпека API, кешування та інтернаціоналізація","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fsecurity-auth","01.csharp\u002F11.aspnet\u002F02.api\u002F12.security-auth",{"title":784,"path":785,"stem":786},"Процес проєктування API та документування","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fapi-design-process","01.csharp\u002F11.aspnet\u002F02.api\u002F13.api-design-process",{"title":788,"path":789,"stem":790},"OpenAPI: контракт, специфікація та документація API","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fopenapi","01.csharp\u002F11.aspnet\u002F02.api\u002F14.openapi",{"title":792,"icon":793,"path":794,"stem":795,"children":796,"page":59},"Auth","i-lucide-shield-check","\u002Fcsharp\u002Faspnet\u002Fauth","01.csharp\u002F11.aspnet\u002F03.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},"Основи аутентифікації та авторизації","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fauth-fundamentals","01.csharp\u002F11.aspnet\u002F03.auth\u002F01.auth-fundamentals",{"title":802,"path":803,"stem":804},"JWT-аутентифікація","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fjwt-authentication","01.csharp\u002F11.aspnet\u002F03.auth\u002F02.jwt-authentication",{"title":806,"path":807,"stem":808},"Авторизація: ролі, політики та resource-based доступ","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fauthorization-policies","01.csharp\u002F11.aspnet\u002F03.auth\u002F03.authorization-policies",{"title":810,"path":811,"stem":812},"Cookie-аутентифікація та ASP.NET Core Identity","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fcookie-auth-identity","01.csharp\u002F11.aspnet\u002F03.auth\u002F04.cookie-auth-identity",{"title":814,"path":815,"stem":816},"JWT + Refresh Tokens (HttpOnly Cookie)","\u002Fcsharp\u002Faspnet\u002Fauth\u002F04b.identity-auth-jwt","01.csharp\u002F11.aspnet\u002F03.auth\u002F04b.identity-auth-jwt",{"title":818,"path":819,"stem":820},"Identity: Підтвердження Email та Скидання Пароля","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fidentity-email-confirmation","01.csharp\u002F11.aspnet\u002F03.auth\u002F05.identity-email-confirmation",{"title":822,"path":823,"stem":824},"Identity: Двофакторна Аутентифікація (2FA)","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fidentity-two-factor","01.csharp\u002F11.aspnet\u002F03.auth\u002F06.identity-two-factor",{"title":826,"path":827,"stem":828},"Identity: Внутрішня Архітектура та Кастомізація","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fidentity-internals","01.csharp\u002F11.aspnet\u002F03.auth\u002F07.identity-internals",{"title":830,"path":831,"stem":832},"OAuth 2.0 та зовнішні провайдери","\u002Fcsharp\u002Faspnet\u002Fauth\u002Foauth-external-providers","01.csharp\u002F11.aspnet\u002F03.auth\u002F08.oauth-external-providers",{"title":834,"path":835,"stem":836},"Безпека на практиці: CORS, HTTPS та захист від атак","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fsecurity-hardening","01.csharp\u002F11.aspnet\u002F03.auth\u002F09.security-hardening",{"title":838,"path":839,"stem":840},"Теорія OAuth 2.0: Поняття, Аналогії та Флоу","\u002Fcsharp\u002Faspnet\u002Fauth\u002Foauth-theory","01.csharp\u002F11.aspnet\u002F03.auth\u002F10.oauth-theory",{"title":842,"path":843,"stem":844},"OIDC, OAuth 2.0 та Keycloak в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Foidc-keycloak","01.csharp\u002F11.aspnet\u002F03.auth\u002F10.oidc-keycloak",{"title":846,"path":847,"stem":848},"API Keys аутентифікація в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fapi-keys","01.csharp\u002F11.aspnet\u002F03.auth\u002F11.api-keys",{"title":850,"path":851,"stem":852},"Rate Limiting та Throttling в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Frate-limiting","01.csharp\u002F11.aspnet\u002F03.auth\u002F12.rate-limiting",{"title":854,"path":855,"stem":856},"Refresh Token Rotation в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Frefresh-token-rotation","01.csharp\u002F11.aspnet\u002F03.auth\u002F13.refresh-token-rotation",{"title":858,"path":859,"stem":860},"Certificate Authentication та mTLS в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fcertificate-auth","01.csharp\u002F11.aspnet\u002F03.auth\u002F14.certificate-auth",{"title":862,"path":863,"stem":864},"RBAC, ABAC та ReBAC в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Frbac-abac-rebac","01.csharp\u002F11.aspnet\u002F03.auth\u002F15.rbac-abac-rebac",{"title":866,"path":867,"stem":868},"Multi-tenancy та ізоляція даних в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fmulti-tenancy","01.csharp\u002F11.aspnet\u002F03.auth\u002F16.multi-tenancy",{"title":870,"icon":871,"path":872,"stem":873,"children":874,"page":59},"Нотифікації","i-lucide-bell","\u002Fcsharp\u002Faspnet\u002Fnotifications","01.csharp\u002F11.aspnet\u002F04.notifications",[875,879,883,887,891,895,899,903,907,911,915,919,923],{"title":876,"path":877,"stem":878},"In-App нотифікації через базу даних","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fin-app-database-notifications","01.csharp\u002F11.aspnet\u002F04.notifications\u002F01.in-app-database-notifications",{"title":880,"path":881,"stem":882},"Polling: Регулярний запит оновлень","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fpolling","01.csharp\u002F11.aspnet\u002F04.notifications\u002F02.polling",{"title":884,"path":885,"stem":886},"Server-Sent Events: Однострімовий push від сервера","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fserver-sent-events","01.csharp\u002F11.aspnet\u002F04.notifications\u002F03.server-sent-events",{"title":888,"path":889,"stem":890},"WebSockets: Двостороннє з'єднання в реальному часі","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fwebsockets","01.csharp\u002F11.aspnet\u002F04.notifications\u002F04.websockets",{"title":892,"path":893,"stem":894},"SignalR: Абстракція над транспортами реального часу","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fsignalr","01.csharp\u002F11.aspnet\u002F04.notifications\u002F05.signalr",{"title":896,"path":897,"stem":898},"Background Services: Фонові задачі в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fbackground-services","01.csharp\u002F11.aspnet\u002F04.notifications\u002F06.background-services",{"title":900,"path":901,"stem":902},"Web Push нотифікації","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fweb-push","01.csharp\u002F11.aspnet\u002F04.notifications\u002F07.web-push",{"title":904,"path":905,"stem":906},"Email нотифікації","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Femail-notifications","01.csharp\u002F11.aspnet\u002F04.notifications\u002F08.email-notifications",{"title":908,"path":909,"stem":910},"Порівняння підходів: Як вибрати правильну технологію нотифікацій","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fchoosing-the-right-approach","01.csharp\u002F11.aspnet\u002F04.notifications\u002F09.choosing-the-right-approach",{"title":912,"path":913,"stem":914},"Hangfire: Надійне планування фонових задач","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fhangfire","01.csharp\u002F11.aspnet\u002F04.notifications\u002F10.hangfire",{"title":916,"path":917,"stem":918},"Практика: Конвертація зображень у WebP через Hangfire","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fhangfire-image-webp","01.csharp\u002F11.aspnet\u002F04.notifications\u002F11.hangfire-image-webp",{"title":920,"path":921,"stem":922},"Практика: Підготовка відео до HLS-стрімінгу через Hangfire","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fhangfire-video-hls","01.csharp\u002F11.aspnet\u002F04.notifications\u002F12.hangfire-video-hls",{"title":924,"path":925,"stem":926},"Telegram-нотифікації: від одного повідомлення до масових розсилок і мульти-канального підходу","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Ftelegram-notifications","01.csharp\u002F11.aspnet\u002F04.notifications\u002F13.telegram-notifications",{"title":928,"icon":929,"path":930,"stem":931,"children":932,"page":59},"Інтернаціоналізація","i-lucide-languages","\u002Fcsharp\u002Faspnet\u002Fi18n","01.csharp\u002F11.aspnet\u002F05.i18n",[933,937],{"title":934,"path":935,"stem":936},"Інтернаціоналізація (i18n) у Minimal API: від A до Я","\u002Fcsharp\u002Faspnet\u002Fi18n\u002Finternationalization","01.csharp\u002F11.aspnet\u002F05.i18n\u002F01.internationalization",{"title":938,"path":939,"stem":940},"Humanizer: людиномовні рядки у .NET","\u002Fcsharp\u002Faspnet\u002Fi18n\u002Fhumanizer","01.csharp\u002F11.aspnet\u002F05.i18n\u002F02.humanizer",{"title":942,"icon":943,"path":944,"stem":945,"children":946,"page":59},"Кешування","i-lucide-layers","\u002Fcsharp\u002Faspnet\u002Fcaching","01.csharp\u002F11.aspnet\u002F06.caching",[947,951,955,959,963],{"title":948,"path":949,"stem":950},"Огляд кешування: чотири рівні і коли що обирати","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Fcaching","01.csharp\u002F11.aspnet\u002F06.caching\u002F01.caching",{"title":952,"path":953,"stem":954},"IMemoryCache: кеш в оперативній пам'яті","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Fmemory-cache","01.csharp\u002F11.aspnet\u002F06.caching\u002F02.memory-cache",{"title":956,"path":957,"stem":958},"IDistributedCache і Redis: розподілений кеш","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Fdistributed-cache","01.csharp\u002F11.aspnet\u002F06.caching\u002F03.distributed-cache",{"title":960,"path":961,"stem":962},"Response Cache: HTTP-кешування через Cache-Control","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Fresponse-cache","01.csharp\u002F11.aspnet\u002F06.caching\u002F04.response-cache",{"title":964,"path":965,"stem":966},"Output Cache: серверний кеш HTTP-відповідей (.NET 7+)","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Foutput-cache","01.csharp\u002F11.aspnet\u002F06.caching\u002F05.output-cache",{"title":968,"icon":969,"path":970,"stem":971,"children":972,"page":59},"Тестування","i-lucide-test-tube","\u002Fcsharp\u002Faspnet\u002Ftesting","01.csharp\u002F11.aspnet\u002F07.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},"Що таке тестування? Від інтуїції до науки","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fwhat-is-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F01.what-is-testing",{"title":978,"path":979,"stem":980},"Піраміда тестування — Стратегія, а не Догма","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftesting-pyramid","01.csharp\u002F11.aspnet\u002F07.testing\u002F02.testing-pyramid",{"title":982,"path":983,"stem":984},"Дві Школи Тестування — Лондон проти Детройту","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftesting-schools","01.csharp\u002F11.aspnet\u002F07.testing\u002F03.testing-schools",{"title":986,"path":987,"stem":988},"TDD та BDD — Тести як Дизайн-інструмент","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftdd-and-bdd","01.csharp\u002F11.aspnet\u002F07.testing\u002F04.tdd-and-bdd",{"title":990,"path":991,"stem":992},"Що саме тестувати — Техніки аналізу та Циклomatична складність","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fwhat-to-test","01.csharp\u002F11.aspnet\u002F07.testing\u002F05.what-to-test",{"title":994,"path":995,"stem":996},"Тестові Фреймворки — Навіщо вони і що всередині","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftest-frameworks","01.csharp\u002F11.aspnet\u002F07.testing\u002F06.test-frameworks",{"title":998,"path":999,"stem":1000},"xUnit — Факти, Теорії та Lifecycle тестів","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fxunit-basics","01.csharp\u002F11.aspnet\u002F07.testing\u002F07.xunit-basics",{"title":1002,"path":1003,"stem":1004},"xUnit Advanced — Fixtures, Кастомізація та Розширення","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fxunit-advanced","01.csharp\u002F11.aspnet\u002F07.testing\u002F08.xunit-advanced",{"title":1006,"path":1007,"stem":1008},"Moq — Глибоке занурення в мокування","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fmocking-with-moq","01.csharp\u002F11.aspnet\u002F07.testing\u002F09.mocking-with-moq",{"title":1010,"path":1011,"stem":1012},"Тестування Баз Даних — EF Core, SQLite та Testcontainers","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fdatabase-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F10.database-testing",{"title":1014,"path":1015,"stem":1016},"Integration Testing — Частина 1 [Теорія та WebApplicationFactory]","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fintegration-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F11.integration-testing",{"title":1018,"path":1019,"stem":1020},"Інтеграційне тестування — Практика","\u002Fcsharp\u002Faspnet\u002Ftesting\u002F11a.integration-testing-practice","01.csharp\u002F11.aspnet\u002F07.testing\u002F11a.integration-testing-practice",{"title":1022,"path":1023,"stem":1024},"Integration Testing — Частина 2 [Просунуті Сценарії та Testcontainers]","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fintegration-testing-advanced","01.csharp\u002F11.aspnet\u002F07.testing\u002F12.integration-testing-advanced",{"title":1026,"path":1027,"stem":1028},"Професійний Postman: Колекції, Змінні та GitHub Інтеграція","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fpostman-professional","01.csharp\u002F11.aspnet\u002F07.testing\u002F13.postman-professional",{"title":1030,"path":1031,"stem":1032},"HttpClient у Тестах Частина 1: Архітектура та MockHttpMessageHandler","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fhttpclient-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F14.httpclient-testing",{"title":1034,"path":1035,"stem":1036},"HttpClient у Тестах Частина 2: WireMock.Net та Resilience","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fwiremock-net","01.csharp\u002F11.aspnet\u002F07.testing\u002F15.wiremock-net",{"title":1038,"path":1039,"stem":1040},"Патерни та Анти-патерни Тестування: Test Smells","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftesting-patterns","01.csharp\u002F11.aspnet\u002F07.testing\u002F16.testing-patterns",{"title":1042,"path":1043,"stem":1044},"Просунуті інструменти: Time, Snapshots та Властивості","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fadvanced-testing-tools","01.csharp\u002F11.aspnet\u002F07.testing\u002F17.advanced-testing-tools",{"title":1046,"path":1047,"stem":1048},"Тестування Архітектури з NetArchTest","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Farchitecture-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F18.architecture-testing",{"title":1050,"path":1051,"stem":1052},"Тестування Продуктивності: BenchmarkDotNet, NBomber та k6","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fperformance-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F19.performance-testing",{"title":1054,"path":1055,"stem":1056},"Залишок плану для курсу \"Тестування ASP.NET Minimal API\"","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fremaining_plan","01.csharp\u002F11.aspnet\u002F07.testing\u002Fremaining_plan",{"title":1058,"icon":1059,"path":1060,"stem":1061,"children":1062,"page":59},"Платежі","i-lucide-credit-card","\u002Fcsharp\u002Faspnet\u002Fpayments","01.csharp\u002F11.aspnet\u002F08.payments",[1063,1067,1071,1075,1079,1083,1087,1091,1095,1099,1103,1107],{"title":1064,"path":1065,"stem":1066},"Основи платіжної інфраструктури","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fpayment-fundamentals","01.csharp\u002F11.aspnet\u002F08.payments\u002F01.payment-fundamentals",{"title":1068,"path":1069,"stem":1070},"Методи оплати в Україні","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fpayment-methods-ukraine","01.csharp\u002F11.aspnet\u002F08.payments\u002F02.payment-methods-ukraine",{"title":1072,"path":1073,"stem":1074},"PCI DSS та безпека платежів","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fpci-dss-security","01.csharp\u002F11.aspnet\u002F08.payments\u002F03.pci-dss-security",{"title":1076,"path":1077,"stem":1078},"Архітектура платіжної підсистеми","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fpayment-architecture","01.csharp\u002F11.aspnet\u002F08.payments\u002F04.payment-architecture",{"title":1080,"path":1081,"stem":1082},"Інтеграція LiqPay (ПриватБанк)","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fliqpay-integration","01.csharp\u002F11.aspnet\u002F08.payments\u002F05.liqpay-integration",{"title":1084,"path":1085,"stem":1086},"Інтеграція Monobank Acquiring API","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fmonobank-acquiring","01.csharp\u002F11.aspnet\u002F08.payments\u002F06.monobank-acquiring",{"title":1088,"path":1089,"stem":1090},"Інтеграція Stripe","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fstripe-integration","01.csharp\u002F11.aspnet\u002F08.payments\u002F07.stripe-integration",{"title":1092,"path":1093,"stem":1094},"Webhooks — глибоке занурення","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fwebhooks-deep-dive","01.csharp\u002F11.aspnet\u002F08.payments\u002F08.webhooks-deep-dive",{"title":1096,"path":1097,"stem":1098},"Підписки та рекурентні платежі","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fsubscriptions-recurring","01.csharp\u002F11.aspnet\u002F08.payments\u002F09.subscriptions-recurring",{"title":1100,"path":1101,"stem":1102},"Повернення коштів та диспути","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Frefunds-disputes","01.csharp\u002F11.aspnet\u002F08.payments\u002F10.refunds-disputes",{"title":1104,"path":1105,"stem":1106},"Тестування платіжних інтеграцій","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Ftesting-payments","01.csharp\u002F11.aspnet\u002F08.payments\u002F11.testing-payments",{"title":1108,"path":1109,"stem":1110},"Чекліст виходу в Production","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fproduction-checklist","01.csharp\u002F11.aspnet\u002F08.payments\u002F12.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","\u002Fcsharp\u002Faspnet\u002Flibraries","01.csharp\u002F11.aspnet\u002F09.libraries",[1130,1134,1138,1142,1146,1150,1154,1158,1162,1166,1170,1174,1178],{"title":1131,"path":1132,"stem":1133},"Валідація з FluentValidation в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Ffluent-validation","01.csharp\u002F11.aspnet\u002F09.libraries\u002F01.fluent-validation",{"title":1135,"path":1136,"stem":1137},"Маппінг об","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fmapster","01.csharp\u002F11.aspnet\u002F09.libraries\u002F02.mapster",{"title":1139,"path":1140,"stem":1141},"Обробка помилок з ErrorOr та Result Pattern в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Ferroror-result-pattern","01.csharp\u002F11.aspnet\u002F09.libraries\u002F03.erroror-result-pattern",{"title":1143,"path":1144,"stem":1145},"Структуроване логування з Serilog в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fserilog","01.csharp\u002F11.aspnet\u002F09.libraries\u002F04.serilog",{"title":1147,"path":1148,"stem":1149},"CQRS та Mediator з MediatR в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fmediatr","01.csharp\u002F11.aspnet\u002F09.libraries\u002F05.mediatr",{"title":1151,"path":1152,"stem":1153},"Відмовостійкість з Polly в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fpolly","01.csharp\u002F11.aspnet\u002F09.libraries\u002F06.polly",{"title":1155,"path":1156,"stem":1157},"Health Checks в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fhealth-checks","01.csharp\u002F11.aspnet\u002F09.libraries\u002F07.health-checks",{"title":1159,"path":1160,"stem":1161},"Feature Management та Feature Flags в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Ffeature-management","01.csharp\u002F11.aspnet\u002F09.libraries\u002F08.feature-management",{"title":1163,"path":1164,"stem":1165},"Відправка Email з FluentEmail в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Ffluent-email","01.csharp\u002F11.aspnet\u002F09.libraries\u002F09.fluent-email",{"title":1167,"path":1168,"stem":1169},"Генерація PDF з QuestPDF в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fquest-pdf","01.csharp\u002F11.aspnet\u002F09.libraries\u002F10.quest-pdf",{"title":1171,"path":1172,"stem":1173},"Генерація тестових даних з Bogus в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fbogus","01.csharp\u002F11.aspnet\u002F09.libraries\u002F11.bogus",{"title":1175,"path":1176,"stem":1177},"Humanizer та Guard Clauses в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fhumanizer-guard","01.csharp\u002F11.aspnet\u002F09.libraries\u002F12.humanizer-guard",{"title":1179,"path":1180,"stem":1181},"План модуля 10.libraries — Популярні бібліотеки ASP.NET","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fplan","01.csharp\u002F11.aspnet\u002F09.libraries\u002Fplan",{"title":1183,"icon":1184,"path":1185,"stem":1186,"children":1187,"page":59},"Razor Pages","i-lucide-layout-template","\u002Fcsharp\u002Faspnet\u002Frazor-pages","01.csharp\u002F11.aspnet\u002F10.razor-pages",[1188,1192,1196,1200,1204,1208],{"title":1189,"path":1190,"stem":1191},"Від Minimal API до Razor Pages: концептуальний перехід","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Ffrom-minimal-api","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F01.from-minimal-api",{"title":1193,"path":1194,"stem":1195},"PageModel: логіка сторінки Razor Pages","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Fpage-model","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F02.page-model",{"title":1197,"path":1198,"stem":1199},"Razor синтаксис: шаблонізатор у .cshtml","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Frazor-syntax","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F03.razor-syntax",{"title":1201,"path":1202,"stem":1203},"Tag Helpers: типізований HTML","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Ftag-helpers","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F04.tag-helpers",{"title":1205,"path":1206,"stem":1207},"Форми і валідація: повний цикл обробки даних","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Fforms-validation","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F05.forms-validation",{"title":1209,"path":1210,"stem":1211},"Практичний проєкт: TaskManager на Razor Pages","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Fproject-task-manager","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F06.project-task-manager",{"title":1213,"path":1214,"stem":1215,"children":1216,"page":59},"ASP.NET Core MVC","\u002Fcsharp\u002Faspnet\u002Fmvc","01.csharp\u002F11.aspnet\u002F11.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: архітектура, що змінила веб","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fmvc-pattern","01.csharp\u002F11.aspnet\u002F11.mvc\u002F01.mvc-pattern",{"title":1222,"path":1223,"stem":1224},"Від Razor Pages до MVC: концептуальний перехід","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Ffrom-razor-pages","01.csharp\u002F11.aspnet\u002F11.mvc\u002F02.from-razor-pages",{"title":1226,"path":1227,"stem":1228},"Controllers та Actions: серце MVC","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fcontrollers-actions","01.csharp\u002F11.aspnet\u002F11.mvc\u002F03.controllers-actions",{"title":1230,"path":1231,"stem":1232},"Маршрутизація в MVC: Convention vs Attribute Routing","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Frouting-mvc","01.csharp\u002F11.aspnet\u002F11.mvc\u002F04.routing-mvc",{"title":1234,"path":1235,"stem":1236},"Model Binding: від HTTP до C#","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fmodel-binding","01.csharp\u002F11.aspnet\u002F11.mvc\u002F05.model-binding",{"title":1238,"path":1239,"stem":1240},"Views, ViewData, ViewBag, TempData і ViewModel","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fviews-viewdata-tempdata","01.csharp\u002F11.aspnet\u002F11.mvc\u002F06.views-viewdata-tempdata",{"title":1242,"path":1243,"stem":1244},"Filters: аспектно-орієнтоване програмування в MVC","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Ffilters","01.csharp\u002F11.aspnet\u002F11.mvc\u002F07.filters",{"title":1246,"path":1247,"stem":1248},"Areas: структурування великих застосунків","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fareas","01.csharp\u002F11.aspnet\u002F11.mvc\u002F08.areas",{"title":1250,"path":1251,"stem":1252},"View Components: повторювані незалежні блоки UI","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fview-components","01.csharp\u002F11.aspnet\u002F11.mvc\u002F09.view-components",{"title":1254,"path":1255,"stem":1256},"Display та Editor Templates","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fdisplay-editor-templates","01.csharp\u002F11.aspnet\u002F11.mvc\u002F10.display-editor-templates",{"title":1258,"path":1259,"stem":1260},"Валідація: IValidatableObject та FluentValidation","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fvalidation-advanced","01.csharp\u002F11.aspnet\u002F11.mvc\u002F11.validation-advanced",{"title":1262,"path":1263,"stem":1264},"HTMX: інтерактивність через HTML-атрибути","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fhtmx","01.csharp\u002F11.aspnet\u002F11.mvc\u002F12.htmx",{"title":1266,"path":1267,"stem":1268},"HTMX у ASP.NET Core MVC: серверна інтеграція","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fajax-htmx-mvc","01.csharp\u002F11.aspnet\u002F11.mvc\u002F13.ajax-htmx-mvc",{"title":1270,"path":1271,"stem":1272},"Практичний проєкт: Каталог товарів з HTMX","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fhtmx-project","01.csharp\u002F11.aspnet\u002F11.mvc\u002F14.htmx-project",{"title":1274,"path":1275,"stem":1276},"Завантаження та обробка файлів","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Ffile-upload","01.csharp\u002F11.aspnet\u002F11.mvc\u002F15.file-upload",{"title":1278,"path":1279,"stem":1280},"Глобалізація та Локалізація MVC","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fglobalization-localization","01.csharp\u002F11.aspnet\u002F11.mvc\u002F16.globalization-localization",{"title":1282,"path":1283,"stem":1284},"Підсумковий проєкт: Блог-платформа","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fmvc-project","01.csharp\u002F11.aspnet\u002F11.mvc\u002F17.mvc-project",{"title":1286,"path":1287,"stem":1288},"План курсу: ASP.NET Core MVC","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fplan","01.csharp\u002F11.aspnet\u002F11.mvc\u002Fplan",{"title":1290,"path":1291,"stem":1292,"children":1293,"page":59},"Web Api","\u002Fcsharp\u002Faspnet\u002Fweb-api","01.csharp\u002F11.aspnet\u002F12.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","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Ffrom-minimal-api-to-controllers","01.csharp\u002F11.aspnet\u002F12.web-api\u002F01.from-minimal-api-to-controllers",{"title":1299,"path":1300,"stem":1301},"ControllerBase, ActionResult\u003CT> та Response Types","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fcontroller-base-actionresult","01.csharp\u002F11.aspnet\u002F12.web-api\u002F02.controller-base-actionresult",{"title":1303,"path":1304,"stem":1305},"Content Negotiation - JSON, XML та власні форматери","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fcontent-negotiation","01.csharp\u002F11.aspnet\u002F12.web-api\u002F03.content-negotiation",{"title":1307,"path":1308,"stem":1309},"Версіонування API","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fapi-versioning","01.csharp\u002F11.aspnet\u002F12.web-api\u002F04.api-versioning",{"title":1311,"path":1312,"stem":1313},"ProblemDetails та структурована обробка помилок","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fproblemdetails-error-handling","01.csharp\u002F11.aspnet\u002F12.web-api\u002F05.problemdetails-error-handling",{"title":1315,"path":1316,"stem":1317},"Фільтри у Web API контексті","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Ffilters-for-api","01.csharp\u002F11.aspnet\u002F12.web-api\u002F06.filters-for-api",{"title":1319,"path":1320,"stem":1321},"Пагінація, фільтрація та сортування","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fpagination-filtering-sorting","01.csharp\u002F11.aspnet\u002F12.web-api\u002F07.pagination-filtering-sorting",{"title":1323,"path":1324,"stem":1325},"HATEOAS та Resource Expansion","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fhateoas-resource-expansion","01.csharp\u002F11.aspnet\u002F12.web-api\u002F08.hateoas-resource-expansion",{"title":1327,"path":1328,"stem":1329},"Гібридна архітектура - Minimal API + Controllers","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fminimal-api-vs-controllers-hybrid","01.csharp\u002F11.aspnet\u002F12.web-api\u002F09.minimal-api-vs-controllers-hybrid",{"title":1331,"path":1332,"stem":1333},"Документація API - Swashbuckle, NSwag та генерація клієнтів","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fapi-documentation-generation","01.csharp\u002F11.aspnet\u002F12.web-api\u002F10.api-documentation-generation",{"title":1335,"path":1336,"stem":1337},"Health Checks та моніторинг API","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fhealth-checks-monitoring","01.csharp\u002F11.aspnet\u002F12.web-api\u002F11.health-checks-monitoring",{"title":1339,"path":1340,"stem":1341},"Підсумковий проєкт - Production-Ready REST API","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fweb-api-project","01.csharp\u002F11.aspnet\u002F12.web-api\u002F12.web-api-project",{"title":1343,"path":1344,"stem":1345},"План курсу: ASP.NET Core Web API (Controllers)","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fplan","01.csharp\u002F11.aspnet\u002F12.web-api\u002Fplan",{"title":1347,"icon":1348,"path":1349,"stem":1350,"children":1351,"page":59},"Desktop UI","i-lucide-app-window","\u002Fcsharp\u002Fdesktop-ui","01.csharp\u002F12.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},"Що таке десктопна розробка?","\u002Fcsharp\u002Fdesktop-ui\u002Fwhat-is-desktop-dev","01.csharp\u002F12.desktop-ui\u002F01.what-is-desktop-dev",{"title":1357,"path":1358,"stem":1359},"Архітектура WPF — як влаштований графічний інтерфейс","\u002Fcsharp\u002Fdesktop-ui\u002Fwpf-architecture","01.csharp\u002F12.desktop-ui\u002F02.wpf-architecture",{"title":1361,"path":1362,"stem":1363},"Перший WPF-проєкт — від нуля до вікна","\u002Fcsharp\u002Fdesktop-ui\u002Ffirst-wpf-app","01.csharp\u002F12.desktop-ui\u002F03.first-wpf-app",{"title":1365,"path":1366,"stem":1367},"Перший Avalonia-проєкт: WPF для всіх платформ","\u002Fcsharp\u002Fdesktop-ui\u002F03a.first-avalonia-app","01.csharp\u002F12.desktop-ui\u002F03a.first-avalonia-app",{"title":1369,"path":1370,"stem":1371},"XAML: декларативний інтерфейс","\u002Fcsharp\u002Fdesktop-ui\u002Fxaml-basics","01.csharp\u002F12.desktop-ui\u002F04.xaml-basics",{"title":1373,"path":1374,"stem":1375},"Fluent UI у WPF — сучасний дизайн Windows 11","\u002Fcsharp\u002Fdesktop-ui\u002F04a.wpf-fluent-ui","01.csharp\u002F12.desktop-ui\u002F04a.wpf-fluent-ui",{"title":1377,"path":1378,"stem":1379},"WPF UI — сучасна бібліотека Fluent контролів","\u002Fcsharp\u002Fdesktop-ui\u002F04b.wpf-ui-library","01.csharp\u002F12.desktop-ui\u002F04b.wpf-ui-library",{"title":1381,"path":1382,"stem":1383},"HandyControl — велика бібліотека UI контролів для WPF","\u002Fcsharp\u002Fdesktop-ui\u002F04c.handycontrol-library","01.csharp\u002F12.desktop-ui\u002F04c.handycontrol-library",{"title":1385,"path":1386,"stem":1387},"Простори імен та ресурси XAML","\u002Fcsharp\u002Fdesktop-ui\u002Fxaml-namespaces-resources","01.csharp\u002F12.desktop-ui\u002F05.xaml-namespaces-resources",{"title":1389,"path":1390,"stem":1391},"XAML в Avalonia: ключові відмінності від WPF","\u002Fcsharp\u002Fdesktop-ui\u002F05a.avalonia-xaml-differences","01.csharp\u002F12.desktop-ui\u002F05a.avalonia-xaml-differences",{"title":1393,"path":1394,"stem":1395},"Розширення розмітки XAML (Markup Extensions)","\u002Fcsharp\u002Fdesktop-ui\u002Fxaml-markup-extensions","01.csharp\u002F12.desktop-ui\u002F06.xaml-markup-extensions",{"title":1397,"path":1398,"stem":1399},"Панелі Layout: StackPanel, WrapPanel, DockPanel","\u002Fcsharp\u002Fdesktop-ui\u002Flayout-panels-part1","01.csharp\u002F12.desktop-ui\u002F07.layout-panels-part1",{"title":1401,"path":1402,"stem":1403},"Grid, Canvas, UniformGrid","\u002Fcsharp\u002Fdesktop-ui\u002Flayout-panels-part2","01.csharp\u002F12.desktop-ui\u002F07.layout-panels-part2",{"title":1405,"path":1406,"stem":1407},"Просунуті техніки Layout","\u002Fcsharp\u002Fdesktop-ui\u002Flayout-advanced","01.csharp\u002F12.desktop-ui\u002F08.layout-advanced",{"title":1409,"path":1410,"stem":1411},"Адаптивний Layout та найкращі практики","\u002Fcsharp\u002Fdesktop-ui\u002Flayout-responsive","01.csharp\u002F12.desktop-ui\u002F09.layout-responsive",{"title":1413,"path":1414,"stem":1415},"Layout в Avalonia: відмінності та нові можливості","\u002Fcsharp\u002Fdesktop-ui\u002F09a.layout-avalonia","01.csharp\u002F12.desktop-ui\u002F09a.layout-avalonia",{"title":1417,"path":1418,"stem":1419},"Button, Image, ProgressBar та інші базові контроли","\u002Fcsharp\u002Fdesktop-ui\u002Fbasic-controls","01.csharp\u002F12.desktop-ui\u002F10.basic-controls",{"title":1421,"path":1422,"stem":1423},"Контроли в Avalonia: відмінності від WPF","\u002Fcsharp\u002Fdesktop-ui\u002F10a.controls-avalonia","01.csharp\u002F12.desktop-ui\u002F10a.controls-avalonia",{"title":1425,"path":1426,"stem":1427},"Текстові контроли — TextBlock, TextBox, RichTextBox","\u002Fcsharp\u002Fdesktop-ui\u002Ftext-controls","01.csharp\u002F12.desktop-ui\u002F11.text-controls",{"title":1429,"path":1430,"stem":1431},"Контроли вибору — CheckBox, RadioButton, ComboBox, ListBox, DatePicker","\u002Fcsharp\u002Fdesktop-ui\u002Fselection-controls","01.csharp\u002F12.desktop-ui\u002F12.selection-controls",{"title":1433,"path":1434,"stem":1435},"Content Model — GroupBox, Expander, TabControl, StatusBar","\u002Fcsharp\u002Fdesktop-ui\u002Fcontent-controls","01.csharp\u002F12.desktop-ui\u002F13.content-controls",{"title":1437,"path":1438,"stem":1439},"UI\u002FUX принципи десктопних застосунків","\u002Fcsharp\u002Fdesktop-ui\u002F13a.ui-ux-principles","01.csharp\u002F12.desktop-ui\u002F13a.ui-ux-principles",{"title":1441,"path":1442,"stem":1443},"Dependency Properties — Концепція та Value Resolution","\u002Fcsharp\u002Fdesktop-ui\u002Fdependency-properties-part1","01.csharp\u002F12.desktop-ui\u002F14.dependency-properties-part1",{"title":1445,"path":1446,"stem":1447},"Avalonia Property System — StyledProperty та DirectProperty","\u002Fcsharp\u002Fdesktop-ui\u002F14a.avalonia-property-system","01.csharp\u002F12.desktop-ui\u002F14a.avalonia-property-system",{"title":1449,"path":1450,"stem":1451},"Attached Properties — Властивості без меж","\u002Fcsharp\u002Fdesktop-ui\u002Fattached-properties","01.csharp\u002F12.desktop-ui\u002F15.attached-properties",{"title":1453,"path":1454,"stem":1455},"Routed Events — Маршрутизація подій у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Frouted-events","01.csharp\u002F12.desktop-ui\u002F16.routed-events",{"title":1457,"path":1458,"stem":1459},"Data Binding — Від Code-Behind до Декларативності","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-binding-basics-part1","01.csharp\u002F12.desktop-ui\u002F17.data-binding-basics-part1",{"title":1461,"path":1462,"stem":1463},"INotifyPropertyChanged — Живе оновлення UI","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-binding-basics-part2","01.csharp\u002F12.desktop-ui\u002F17.data-binding-basics-part2",{"title":1465,"path":1466,"stem":1467},"Compiled Bindings в Avalonia — Безпека на етапі компіляції","\u002Fcsharp\u002Fdesktop-ui\u002F17a.avalonia-compiled-bindings","01.csharp\u002F12.desktop-ui\u002F17a.avalonia-compiled-bindings",{"title":1469,"path":1470,"stem":1471},"Просунутий Data Binding — ElementName, RelativeSource, MultiBinding","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-binding-advanced","01.csharp\u002F12.desktop-ui\u002F18.data-binding-advanced",{"title":1473,"path":1474,"stem":1475},"Value Converters — Перетворення типів даних у Data Binding","\u002Fcsharp\u002Fdesktop-ui\u002Fvalue-converters","01.csharp\u002F12.desktop-ui\u002F19.value-converters",{"title":1477,"path":1478,"stem":1479},"Data Templates — Візуалізація об'єктів у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-templates","01.csharp\u002F12.desktop-ui\u002F20.data-templates",{"title":1481,"path":1482,"stem":1483},"Collections Binding Part 1 — ObservableCollection та ItemsControl","\u002Fcsharp\u002Fdesktop-ui\u002Fcollections-binding-part1","01.csharp\u002F12.desktop-ui\u002F21.collections-binding-part1",{"title":1485,"path":1486,"stem":1487},"Collections Binding Part 2 — ICollectionView, Filtering, Sorting та Virtualization","\u002Fcsharp\u002Fdesktop-ui\u002Fcollections-binding-part2","01.csharp\u002F12.desktop-ui\u002F21.collections-binding-part2",{"title":1489,"path":1490,"stem":1491},"MVVM Pattern — Від Spaghetti Code до архітектури","\u002Fcsharp\u002Fdesktop-ui\u002Fmvvm-pattern","01.csharp\u002F12.desktop-ui\u002F22.mvvm-pattern",{"title":1493,"path":1494,"stem":1495},"ViewModel Implementation — Від BaseViewModel до валідації","\u002Fcsharp\u002Fdesktop-ui\u002Fviewmodel-implementation","01.csharp\u002F12.desktop-ui\u002F23.viewmodel-implementation",{"title":1497,"path":1498,"stem":1499},"Commands — Від event handlers до декларативних команд","\u002Fcsharp\u002Fdesktop-ui\u002Fcommands","01.csharp\u002F12.desktop-ui\u002F24.commands",{"title":1501,"path":1502,"stem":1503},"MVVM Toolkit — MVVM без boilerplate через Source Generators","\u002Fcsharp\u002Fdesktop-ui\u002Fmvvm-toolkit","01.csharp\u002F12.desktop-ui\u002F25.mvvm-toolkit",{"title":1505,"path":1506,"stem":1507},"Messenger Pattern — Комунікація між ViewModel без прямих посилань","\u002Fcsharp\u002Fdesktop-ui\u002Fmessenger-pattern","01.csharp\u002F12.desktop-ui\u002F26.messenger-pattern",{"title":1509,"path":1510,"stem":1511},"Стилі WPF — CSS для десктопу","\u002Fcsharp\u002Fdesktop-ui\u002Fstyles-basics","01.csharp\u002F12.desktop-ui\u002F27.styles-basics",{"title":1513,"path":1514,"stem":1515},"CSS-like стилі Avalonia","\u002Fcsharp\u002Fdesktop-ui\u002F27a.avalonia-css-styling","01.csharp\u002F12.desktop-ui\u002F27a.avalonia-css-styling",{"title":1517,"path":1518,"stem":1519},"Control Templates — Частина 1. Концепція та TemplateBinding","\u002Fcsharp\u002Fdesktop-ui\u002Fcontrol-templates-part1","01.csharp\u002F12.desktop-ui\u002F28.control-templates-part1",{"title":1521,"path":1522,"stem":1523},"Control Templates — Частина 2. Named Parts та ContentPresenter","\u002Fcsharp\u002Fdesktop-ui\u002Fcontrol-templates-part2","01.csharp\u002F12.desktop-ui\u002F28.control-templates-part2",{"title":1525,"path":1526,"stem":1527},"Control Themes в Avalonia — нова ера стилізації","\u002Fcsharp\u002Fdesktop-ui\u002F28a.avalonia-control-themes","01.csharp\u002F12.desktop-ui\u002F28a.avalonia-control-themes",{"title":1529,"path":1530,"stem":1531},"Triggers та Visual State Manager у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Ftriggers-visual-states","01.csharp\u002F12.desktop-ui\u002F29.triggers-visual-states",{"title":1533,"path":1534,"stem":1535},"Pseudo-classes в Avalonia — замість WPF Triggers","\u002Fcsharp\u002Fdesktop-ui\u002F29a.avalonia-pseudo-classes","01.csharp\u002F12.desktop-ui\u002F29a.avalonia-pseudo-classes",{"title":1537,"path":1538,"stem":1539},"Теми та ресурсні словники у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fresources-themes","01.csharp\u002F12.desktop-ui\u002F30.resources-themes",{"title":1541,"path":1542,"stem":1543},"Avalonia Themes — Fluent Design та система тематизації","\u002Fcsharp\u002Fdesktop-ui\u002F30a.avalonia-themes-fluent","01.csharp\u002F12.desktop-ui\u002F30a.avalonia-themes-fluent",{"title":1545,"path":1546,"stem":1547},"Контроли колекцій — глибоке занурення","\u002Fcsharp\u002Fdesktop-ui\u002Fcollection-controls","01.csharp\u002F12.desktop-ui\u002F31.collection-controls",{"title":1549,"path":1550,"stem":1551},"DataGrid — колонки та базове відображення","\u002Fcsharp\u002Fdesktop-ui\u002Fdatagrid-part1","01.csharp\u002F12.desktop-ui\u002F32.datagrid-part1",{"title":1553,"path":1554,"stem":1555},"DataGrid — сортування, фільтрація, редагування","\u002Fcsharp\u002Fdesktop-ui\u002Fdatagrid-part2","01.csharp\u002F12.desktop-ui\u002F32.datagrid-part2",{"title":1557,"path":1558,"stem":1559},"TreeView та GridView","\u002Fcsharp\u002Fdesktop-ui\u002Ftreeview-listview","01.csharp\u002F12.desktop-ui\u002F33.treeview-listview",{"title":1561,"path":1562,"stem":1563},"Меню, Toolbar, ContextMenu, StatusBar","\u002Fcsharp\u002Fdesktop-ui\u002Fmenus-toolbars","01.csharp\u002F12.desktop-ui\u002F34.menus-toolbars",{"title":1565,"path":1566,"stem":1567},"Навігація та керування вікнами. Частина 1: вікна та сторінки","\u002Fcsharp\u002Fdesktop-ui\u002Fnavigation-windows-part1","01.csharp\u002F12.desktop-ui\u002F35.navigation-windows-part1",{"title":1569,"path":1570,"stem":1571},"Навігація та керування вікнами. Частина 2: MVVM-навігація","\u002Fcsharp\u002Fdesktop-ui\u002Fnavigation-windows-part2","01.csharp\u002F12.desktop-ui\u002F35.navigation-windows-part2",{"title":1573,"path":1574,"stem":1575},"Avalonia — Навігація та діалоги","\u002Fcsharp\u002Fdesktop-ui\u002F35a.avalonia-navigation-dialogs","01.csharp\u002F12.desktop-ui\u002F35a.avalonia-navigation-dialogs",{"title":1577,"path":1578,"stem":1579},"Діалоги та File Pickers у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fdialogs-file-pickers","01.csharp\u002F12.desktop-ui\u002F36.dialogs-file-pickers",{"title":1581,"path":1582,"stem":1583},"UserControl: компонентний підхід у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fuser-controls","01.csharp\u002F12.desktop-ui\u002F37.user-controls",{"title":1585,"path":1586,"stem":1587},"Custom Controls: Lookless Controls у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fcustom-controls","01.csharp\u002F12.desktop-ui\u002F38.custom-controls",{"title":1589,"path":1590,"stem":1591},"Avalonia TemplatedControl — Lookless Controls","\u002Fcsharp\u002Fdesktop-ui\u002F38a.avalonia-templated-controls","01.csharp\u002F12.desktop-ui\u002F38a.avalonia-templated-controls",{"title":1593,"path":1594,"stem":1595},"Анімації у WPF: Storyboard та Easing Functions","\u002Fcsharp\u002Fdesktop-ui\u002Fanimations-transitions","01.csharp\u002F12.desktop-ui\u002F39.animations-transitions",{"title":1597,"path":1598,"stem":1599},"Анімації в Avalonia","\u002Fcsharp\u002Fdesktop-ui\u002F39a.avalonia-animations","01.csharp\u002F12.desktop-ui\u002F39a.avalonia-animations",{"title":1601,"path":1602,"stem":1603},"2D Графіка та Мультимедіа у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fmedia-graphics","01.csharp\u002F12.desktop-ui\u002F40.media-graphics",{"title":1605,"path":1606,"stem":1607},"Dependency Injection у WPF та Avalonia","\u002Fcsharp\u002Fdesktop-ui\u002Fdi-integration","01.csharp\u002F12.desktop-ui\u002F41.di-integration",{"title":1609,"path":1610,"stem":1611},"SQLite та EF Core у десктопних додатках","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-persistence-part1","01.csharp\u002F12.desktop-ui\u002F42.data-persistence-part1",{"title":1613,"path":1614,"stem":1615},"Repository Pattern та Unit of Work","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-persistence-part2","01.csharp\u002F12.desktop-ui\u002F43.data-persistence-part2",{"title":1617,"path":1618,"stem":1619},"Тестування ViewModels","\u002Fcsharp\u002Fdesktop-ui\u002Fviewmodel-testing","01.csharp\u002F12.desktop-ui\u002F44.viewmodel-testing",{"title":1621,"path":1622,"stem":1623},"Avalonia Headless Testing — тестування UI без вікон","\u002Fcsharp\u002Fdesktop-ui\u002F44a.avalonia-headless-testing","01.csharp\u002F12.desktop-ui\u002F44a.avalonia-headless-testing",{"title":1625,"path":1626,"stem":1627},"Кросплатформна розробка з Avalonia","\u002Fcsharp\u002Fdesktop-ui\u002Favalonia-cross-platform","01.csharp\u002F12.desktop-ui\u002F45.avalonia-cross-platform",{"title":1629,"path":1630,"stem":1631},"Пакування та розгортання Avalonia додатків","\u002Fcsharp\u002Fdesktop-ui\u002Favalonia-packaging-deployment","01.csharp\u002F12.desktop-ui\u002F46.avalonia-packaging-deployment",{"title":1633,"path":1634,"stem":1635},"Розгортання WPF застосунків","\u002Fcsharp\u002Fdesktop-ui\u002Fwpf-packaging-deployment","01.csharp\u002F12.desktop-ui\u002F47.wpf-packaging-deployment",{"title":1637,"icon":658,"path":1638,"stem":1639,"children":1640,"page":59},"Network Programming","\u002Fcsharp\u002Fnetwork-programming","01.csharp\u002F13.network-programming",[1641,1645,1649,1653,1657,1661],{"title":1642,"path":1643,"stem":1644},"Основи комп'ютерних мереж","\u002Fcsharp\u002Fnetwork-programming\u002Ffoundations","01.csharp\u002F13.network-programming\u002F01.foundations",{"title":1646,"path":1647,"stem":1648},"Модель OSI та стек TCP\u002FIP","\u002Fcsharp\u002Fnetwork-programming\u002Fosi-model","01.csharp\u002F13.network-programming\u002F02.osi-model",{"title":1650,"path":1651,"stem":1652},"IP-протокол та адресація","\u002Fcsharp\u002Fnetwork-programming\u002Fip-addressing","01.csharp\u002F13.network-programming\u002F03.ip-addressing",{"title":1654,"path":1655,"stem":1656},"UDP — протокол без з'єднання","\u002Fcsharp\u002Fnetwork-programming\u002Fudp","01.csharp\u002F13.network-programming\u002F05.udp",{"title":1658,"path":1659,"stem":1660},"UDP Broadcast та Multicast","\u002Fcsharp\u002Fnetwork-programming\u002Fudp-broadcast-multicast","01.csharp\u002F13.network-programming\u002F06.udp-broadcast-multicast",{"title":1662,"path":1663,"stem":1664},"HTTP — протокол вебу","\u002Fcsharp\u002Fnetwork-programming\u002Fhttp-fundamentals","01.csharp\u002F13.network-programming\u002F07.http-fundamentals",{"title":1666,"path":1667,"stem":1668},"C# & .NET: The Ultimate Roadmap","\u002Fcsharp\u002Froadmap","01.csharp\u002Froadmap",{"title":1670,"icon":1671,"path":1672,"stem":1673,"children":1674,"page":59},"C++","i-devicon-cplusplus","\u002Fcpp","02.cpp",[1675,1679,1683,1687,1691,1695,1699,1703,1707,1710,1714,1718,1722,1726,1730,1734,1738,1742,1746,1750,1754,1758,1762,1766,1770,1774,1778,1782,1786,1790,1794,1798,1802,1806,1810,1814,1818,1822],{"title":1676,"path":1677,"stem":1678},"Вступ у програмування та алгоритми","\u002Fcpp\u002Fintro-algorithms","02.cpp\u002F01.intro-algorithms",{"title":1680,"path":1681,"stem":1682},"Code Style: угоди про оформлення коду","\u002Fcpp\u002Fcode-style","02.cpp\u002F02.code-style",{"title":1684,"path":1685,"stem":1686},"Середовище розробки та перший проєкт","\u002Fcpp\u002Fide-setup","02.cpp\u002F03.ide-setup",{"title":1688,"path":1689,"stem":1690},"Вивід даних на екран","\u002Fcpp\u002Fdata-output","02.cpp\u002F04.data-output",{"title":1692,"path":1693,"stem":1694},"Типи даних, змінні та константи","\u002Fcpp\u002Fdata-types-variables","02.cpp\u002F05.data-types-variables",{"title":1696,"path":1697,"stem":1698},"Ввід даних з клавіатури","\u002Fcpp\u002Fdata-input","02.cpp\u002F06.data-input",{"title":1700,"path":1701,"stem":1702},"Оператори, перетворення типів та логічні операції","\u002Fcpp\u002Foperators-type-conversion","02.cpp\u002F07.operators-type-conversion",{"title":1704,"path":1705,"stem":1706},"Цикли","\u002Fcpp\u002Floops","02.cpp\u002F08.loops",{"title":32,"path":1708,"stem":1709},"\u002Fcpp\u002Farrays","02.cpp\u002F09.arrays",{"title":1711,"path":1712,"stem":1713},"Алгоритми сортування та аналіз складності","\u002Fcpp\u002Fsorting","02.cpp\u002F10.sorting",{"title":1715,"path":1716,"stem":1717},"Алгоритми пошуку","\u002Fcpp\u002Fsearching","02.cpp\u002F11.searching",{"title":1719,"path":1720,"stem":1721},"Функції: основи","\u002Fcpp\u002Ffunctions-basics","02.cpp\u002F12.functions-basics",{"title":1723,"path":1724,"stem":1725},"Функції: прототипи, область видимості та додаткові можливості","\u002Fcpp\u002Ffunctions-scope","02.cpp\u002F13.functions-scope",{"title":1727,"path":1728,"stem":1729},"Функції: перевантаження та шаблони","\u002Fcpp\u002Ffunctions-overloading-templates","02.cpp\u002F14.functions-overloading-templates",{"title":1731,"path":1732,"stem":1733},"Вказівники: основи","\u002Fcpp\u002Fpointers-basics","02.cpp\u002F15.pointers-basics",{"title":1735,"path":1736,"stem":1737},"Посилання (References)","\u002Fcpp\u002Freferences","02.cpp\u002F16.references",{"title":1739,"path":1740,"stem":1741},"Вказівники, const і масиви","\u002Fcpp\u002Fpointers-const-arrays","02.cpp\u002F17.pointers-const-arrays",{"title":1743,"path":1744,"stem":1745},"Адресна арифметика","\u002Fcpp\u002Fpointer-arithmetic","02.cpp\u002F18.pointer-arithmetic",{"title":1747,"path":1748,"stem":1749},"Динамічна пам'ять","\u002Fcpp\u002Fdynamic-memory","02.cpp\u002F19.dynamic-memory",{"title":1751,"path":1752,"stem":1753},"Вказівники типу void","\u002Fcpp\u002Fvoid-pointers","02.cpp\u002F20.void-pointers",{"title":1755,"path":1756,"stem":1757},"Вказівники на вказівники","\u002Fcpp\u002Fpointers-to-pointers","02.cpp\u002F21.pointers-to-pointers",{"title":1759,"path":1760,"stem":1761},"Оператор доступу до членів через вказівник (->)","\u002Fcpp\u002Fmember-access-operator","02.cpp\u002F22.member-access-operator",{"title":1763,"path":1764,"stem":1765},"Цикл for-each (Range-based for)","\u002Fcpp\u002Fforeach-loop","02.cpp\u002F23.foreach-loop",{"title":1767,"path":1768,"stem":1769},"Вказівники на функції","\u002Fcpp\u002Ffunction-pointers","02.cpp\u002F24.function-pointers",{"title":1771,"path":1772,"stem":1773},"Лямбда-вирази","\u002Fcpp\u002Flambdas","02.cpp\u002F25.lambdas",{"title":1775,"path":1776,"stem":1777},"Лямбда-захоплення","\u002Fcpp\u002Flambda-captures","02.cpp\u002F26.lambda-captures",{"title":1779,"path":1780,"stem":1781},"Еліпсис","\u002Fcpp\u002Fellipsis","02.cpp\u002F27.ellipsis",{"title":1783,"path":1784,"stem":1785},"Безпечні альтернативи еліпсису","\u002Fcpp\u002F27a.ellipsis","02.cpp\u002F27a.ellipsis",{"title":1787,"path":1788,"stem":1789},"Аргументи командного рядка","\u002Fcpp\u002Fcommand-line-arguments","02.cpp\u002F28.command-line-arguments",{"title":1791,"path":1792,"stem":1793},"Перерахування (enum)","\u002Fcpp\u002Fenum","02.cpp\u002F29.enum",{"title":1795,"path":1796,"stem":1797},"Класи-перерахування (enum class)","\u002Fcpp\u002Fenum-class","02.cpp\u002F30.enum-class",{"title":1799,"path":1800,"stem":1801},"Псевдоніми типів (typedef і using)","\u002Fcpp\u002Ftype-aliases","02.cpp\u002F31.type-aliases",{"title":1803,"path":1804,"stem":1805},"Системи числення та двійкова арифметика","\u002Fcpp\u002Fnumber-systems","02.cpp\u002F32.number-systems",{"title":1807,"path":1808,"stem":1809},"Структури (struct): агрегування даних","\u002Fcpp\u002Fstruct","02.cpp\u002F33.struct",{"title":1811,"path":1812,"stem":1813},"Структури у функціях","\u002Fcpp\u002Fstruct-functions","02.cpp\u002F34.struct-functions",{"title":1815,"path":1816,"stem":1817},"Масиви структур і вкладені структури","\u002Fcpp\u002Fstruct-arrays","02.cpp\u002F35.struct-arrays",{"title":1819,"path":1820,"stem":1821},"Патерни struct та межі застосування","\u002Fcpp\u002Fstruct-patterns","02.cpp\u002F36.struct-patterns",{"title":1823,"path":1824,"stem":1825},"План навчання: Курс C++ — Продовження (Статті 29–60+)","\u002Fcpp\u002Fcurriculum-plan","02.cpp\u002Fcurriculum-plan",{"title":1827,"icon":1828,"path":1829,"stem":1830,"children":1831,"page":59},"JavaScript","i-devicon-javascript","\u002Fjavascript","03.javascript",[1832,1858,1912,1934,2238,2276],{"title":1833,"icon":1834,"path":1835,"stem":1836,"children":1837,"page":59},"Events","i-lucide-mouse-pointer-click","\u002Fjavascript\u002Fevents","03.javascript\u002F01.events",[1838,1842,1846,1850,1854],{"title":1839,"path":1840,"stem":1841},"Вступ до подій браузера","\u002Fjavascript\u002Fevents\u002Fintro","03.javascript\u002F01.events\u002F01.intro",{"title":1843,"path":1844,"stem":1845},"Бульбашковий механізм (Bubbling) та занурення (Capturing)","\u002Fjavascript\u002Fevents\u002Fbubbling-capturing","03.javascript\u002F01.events\u002F02.bubbling-capturing",{"title":1847,"path":1848,"stem":1849},"Делегування подій (Event Delegation)","\u002Fjavascript\u002Fevents\u002Fdelegate-events","03.javascript\u002F01.events\u002F03.delegate-events",{"title":1851,"path":1852,"stem":1853},"Типові дії браузера та preventDefault()","\u002Fjavascript\u002Fevents\u002Fprevent-default","03.javascript\u002F01.events\u002F04.prevent-default",{"title":1855,"path":1856,"stem":1857},"Запуск користувацьких подій (Custom Events)","\u002Fjavascript\u002Fevents\u002Fcustom-events","03.javascript\u002F01.events\u002F05.custom-events",{"title":1859,"icon":1860,"path":1861,"stem":1862,"children":1863,"page":59},"Network","i-lucide-globe","\u002Fjavascript\u002Fnetwork","03.javascript\u002F02.network",[1864,1868,1872,1876,1880,1884,1888,1892,1896,1900,1904,1908],{"title":1865,"path":1866,"stem":1867},"Fetch API - Сучасний підхід до HTTP-запитів","\u002Fjavascript\u002Fnetwork\u002F01-fetch-api","03.javascript\u002F02.network\u002F01-fetch-api",{"title":1869,"path":1870,"stem":1871},"FormData - Робота з формами та файлами","\u002Fjavascript\u002Fnetwork\u002F02-formdata","03.javascript\u002F02.network\u002F02-formdata",{"title":1873,"path":1874,"stem":1875},"Відстеження прогресу завантаження","\u002Fjavascript\u002Fnetwork\u002F03-download-progress","03.javascript\u002F02.network\u002F03-download-progress",{"title":1877,"path":1878,"stem":1879},"Переривання fetch-запитів","\u002Fjavascript\u002Fnetwork\u002F04-abort-requests","03.javascript\u002F02.network\u002F04-abort-requests",{"title":1881,"path":1882,"stem":1883},"CORS - Запити між різними джерелами","\u002Fjavascript\u002Fnetwork\u002F05-cors","03.javascript\u002F02.network\u002F05-cors",{"title":1885,"path":1886,"stem":1887},"Fetch API - Повний довідник опцій","\u002Fjavascript\u002Fnetwork\u002F06-fetch-options","03.javascript\u002F02.network\u002F06-fetch-options",{"title":1889,"path":1890,"stem":1891},"URL Objects - Робота з посиланнями","\u002Fjavascript\u002Fnetwork\u002F07-url-objects","03.javascript\u002F02.network\u002F07-url-objects",{"title":1893,"path":1894,"stem":1895},"XMLHttpRequest - AJAX та низькорівневі запити","\u002Fjavascript\u002Fnetwork\u002F08-xmlhttprequest","03.javascript\u002F02.network\u002F08-xmlhttprequest",{"title":1897,"path":1898,"stem":1899},"Відновлюване завантаження файлів","\u002Fjavascript\u002Fnetwork\u002F09-resumable-upload","03.javascript\u002F02.network\u002F09-resumable-upload",{"title":1901,"path":1902,"stem":1903},"Cookies, document.cookie та світ після \"Cookiepocalypse\"","\u002Fjavascript\u002Fnetwork\u002F10-cookies","03.javascript\u002F02.network\u002F10-cookies",{"title":1905,"path":1906,"stem":1907},"js-cookie: Керування Cookies без Болю","\u002Fjavascript\u002Fnetwork\u002F11-js-cookie","03.javascript\u002F02.network\u002F11-js-cookie",{"title":1909,"path":1910,"stem":1911},"Axios: Потужний HTTP-клієнт для JavaScript","\u002Fjavascript\u002Fnetwork\u002F12-axios","03.javascript\u002F02.network\u002F12-axios",{"title":1913,"icon":1914,"path":1915,"stem":1916,"children":1917,"page":59},"Bom","i-lucide-monitor","\u002Fjavascript\u002Fbom","03.javascript\u002F03.bom",[1918,1922,1926,1930],{"title":1919,"path":1920,"stem":1921},"LocalStorage, SessionStorage та patterns збереження даних","\u002Fjavascript\u002Fbom\u002F01-localstorage","03.javascript\u002F03.bom\u002F01-localstorage",{"title":1923,"path":1924,"stem":1925},"Location Object - Керування адресою сторінки","\u002Fjavascript\u002Fbom\u002F02-location-object","03.javascript\u002F03.bom\u002F02-location-object",{"title":1927,"path":1928,"stem":1929},"History API - Керування історією браузера","\u002Fjavascript\u002Fbom\u002F03-history-api","03.javascript\u002F03.bom\u002F03-history-api",{"title":1931,"path":1932,"stem":1933},"Navigator Object - Ідентифікація та Можливості Пристрою","\u002Fjavascript\u002Fbom\u002F04-navigator-object","03.javascript\u002F03.bom\u002F04-navigator-object",{"title":1935,"icon":1936,"path":1937,"stem":1938,"children":1939},"React","i-devicon-react","\u002Fjavascript\u002Freact","03.javascript\u002F04.react\u002Findex",[1940,1941,1945,1949,1953,1957,2020,2055,2207],{"title":1935,"path":1937,"stem":1938},{"title":1942,"path":1943,"stem":1944},"Робота з Формами в React","\u002Fjavascript\u002Freact\u002Freact-forms","03.javascript\u002F04.react\u002F01.react-forms",{"title":1946,"path":1947,"stem":1948},"React Hook Form: Професійна Робота з Формами","\u002Fjavascript\u002Freact\u002Freact-hook-form","03.javascript\u002F04.react\u002F02.react-hook-form",{"title":1950,"path":1951,"stem":1952},"React Hook Form: Глибоке Розуміння Архітектури та Оптимізації","\u002Fjavascript\u002Freact\u002Freact-hook-form-new","03.javascript\u002F04.react\u002F02.react-hook-form-new",{"title":1954,"path":1955,"stem":1956},"Axios та React: Професійна Архітектура Запитів","\u002Fjavascript\u002Freact\u002Fdata-fetching-axios","03.javascript\u002F04.react\u002F03.data-fetching-axios",{"title":1958,"icon":132,"path":1959,"stem":1960,"children":1961},"Tanstack Query","\u002Fjavascript\u002Freact\u002Ftanstack-query","03.javascript\u002F04.react\u002F04.tanstack-query\u002Findex",[1962,1964,1968,1972,1976,1980,1984,1988,1992,1996,2000,2004,2008,2012,2016],{"title":1963,"path":1959,"stem":1960},"TanStack Query: Майстерність Керування Станом Сервера",{"title":1965,"path":1966,"stem":1967},"Парадигма Server State: Чому useEffect недостатньо","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fserver-state-paradigm","03.javascript\u002F04.react\u002F04.tanstack-query\u002F01.server-state-paradigm",{"title":1969,"path":1970,"stem":1971},"Встановлення та Налаштування: Фундамент","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Finstallation-and-devtools","03.javascript\u002F04.react\u002F04.tanstack-query\u002F02.installation-and-devtools",{"title":1973,"path":1974,"stem":1975},"Основи Запитів та Магія Ключів","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fquery-basics-and-keys","03.javascript\u002F04.react\u002F04.tanstack-query\u002F03.query-basics-and-keys",{"title":1977,"path":1978,"stem":1979},"Синхронізація Даних: Життєвий Цикл Запиту","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fdata-synchronization","03.javascript\u002F04.react\u002F04.tanstack-query\u002F04.data-synchronization",{"title":1981,"path":1982,"stem":1983},"Мутації та Інвалідація: Зміна Даних","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fmutations-and-invalidation","03.javascript\u002F04.react\u002F04.tanstack-query\u002F05.mutations-and-invalidation",{"title":1985,"path":1986,"stem":1987},"Оптимістичні Оновлення: Швидше за Світло","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Foptimistic-updates","03.javascript\u002F04.react\u002F04.tanstack-query\u002F06.optimistic-updates",{"title":1989,"path":1990,"stem":1991},"Пагінація та Infinite Scroll","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fpagination-and-load-more","03.javascript\u002F04.react\u002F04.tanstack-query\u002F07.pagination-and-load-more",{"title":1993,"path":1994,"stem":1995},"Просунуті Патерни та Оптимізація","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fadvanced-patterns","03.javascript\u002F04.react\u002F04.tanstack-query\u002F08.advanced-patterns",{"title":1997,"path":1998,"stem":1999},"Архітектура та Best Practices","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Farchitecture-and-best-practices","03.javascript\u002F04.react\u002F04.tanstack-query\u002F09.architecture-and-best-practices",{"title":2001,"path":2002,"stem":2003},"Server-Side Rendering (SSR) та Гідратація","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fserver-side-rendering","03.javascript\u002F04.react\u002F04.tanstack-query\u002F10.server-side-rendering",{"title":2005,"path":2006,"stem":2007},"Стратегії Тестування","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Ftesting-strategies","03.javascript\u002F04.react\u002F04.tanstack-query\u002F11.testing-strategies",{"title":2009,"path":2010,"stem":2011},"Аутентифікація та Обробка Помилок","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fauthentication-and-errors","03.javascript\u002F04.react\u002F04.tanstack-query\u002F12.authentication-and-errors",{"title":2013,"path":2014,"stem":2015},"React Suspense та Майбутнє","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Freact-suspense","03.javascript\u002F04.react\u002F04.tanstack-query\u002F13.react-suspense",{"title":2017,"path":2018,"stem":2019},"Глибоке Занурення в Продуктивність","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fperformance-deep-dive","03.javascript\u002F04.react\u002F04.tanstack-query\u002F14.performance-deep-dive",{"title":2021,"icon":1936,"path":2022,"stem":2023,"children":2024},"React Router","\u002Fjavascript\u002Freact\u002Freact-router","03.javascript\u002F04.react\u002F05.react-router\u002Findex",[2025,2027,2031,2035,2039,2043,2047,2051],{"title":2026,"path":2022,"stem":2023},"React Router: Навігаційна система сучасного вебу",{"title":2028,"path":2029,"stem":2030},"Налаштування та Базовий Роутинг","\u002Fjavascript\u002Freact\u002Freact-router\u002Fsetup-and-basic-routing","03.javascript\u002F04.react\u002F05.react-router\u002F01.setup-and-basic-routing",{"title":2032,"path":2033,"stem":2034},"Динамічна Навігація","\u002Fjavascript\u002Freact\u002Freact-router\u002Fnavigation-and-links","03.javascript\u002F04.react\u002F05.react-router\u002F02.navigation-and-links",{"title":2036,"path":2037,"stem":2038},"Вкладені Маршрути та Макети","\u002Fjavascript\u002Freact\u002Freact-router\u002Fnested-routes-and-layouts","03.javascript\u002F04.react\u002F05.react-router\u002F03.nested-routes-and-layouts",{"title":2040,"path":2041,"stem":2042},"Динамічні Маршрути та Параметри","\u002Fjavascript\u002Freact\u002Freact-router\u002Fdynamic-routing","03.javascript\u002F04.react\u002F05.react-router\u002F04.dynamic-routing",{"title":2044,"path":2045,"stem":2046},"Data APIs: Loaders та Actions","\u002Fjavascript\u002Freact\u002Freact-router\u002Fdata-loading","03.javascript\u002F04.react\u002F05.react-router\u002F05.data-loading",{"title":2048,"path":2049,"stem":2050},"Просунуті Патерни","\u002Fjavascript\u002Freact\u002Freact-router\u002Fadvanced-patterns","03.javascript\u002F04.react\u002F05.react-router\u002F06.advanced-patterns",{"title":2052,"path":2053,"stem":2054},"Legacy Routing: Компонентний підхід","\u002Fjavascript\u002Freact\u002Freact-router\u002Flegacy-routing","03.javascript\u002F04.react\u002F05.react-router\u002F07.legacy-routing",{"title":2056,"icon":132,"path":2057,"stem":2058,"children":2059},"Redux","\u002Fjavascript\u002Freact\u002Fredux","03.javascript\u002F04.react\u002F06.redux\u002Findex",[2060,2062,2078,2107,2116,2137,2153,2182],{"title":2061,"path":2057,"stem":2058},"Redux: Еволюція управління станом",{"title":14,"icon":15,"path":2063,"stem":2064,"children":2065,"page":59},"\u002Fjavascript\u002Freact\u002Fredux\u002Ffundamentals","03.javascript\u002F04.react\u002F06.redux\u002F01.fundamentals",[2066,2070,2074],{"title":2067,"path":2068,"stem":2069},"Вступ до State Management","\u002Fjavascript\u002Freact\u002Fredux\u002Ffundamentals\u002Fintro-state-management","03.javascript\u002F04.react\u002F06.redux\u002F01.fundamentals\u002F01.intro-state-management",{"title":2071,"path":2072,"stem":2073},"Філософія Redux та Три Принципи","\u002Fjavascript\u002Freact\u002Fredux\u002Ffundamentals\u002Fredux-philosophy","03.javascript\u002F04.react\u002F06.redux\u002F01.fundamentals\u002F02.redux-philosophy",{"title":2075,"path":2076,"stem":2077},"Чисті функції та Іммутабельність","\u002Fjavascript\u002Freact\u002Fredux\u002Ffundamentals\u002Fpure-functions-immutability","03.javascript\u002F04.react\u002F06.redux\u002F01.fundamentals\u002F03.pure-functions-immutability",{"title":2079,"icon":132,"path":2080,"stem":2081,"children":2082,"page":59},"Classic Redux","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux",[2083,2087,2091,2095,2099,2103],{"title":2084,"path":2085,"stem":2086},"Створення Store (Classic Redux)","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Fstore-setup","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F01.store-setup",{"title":2088,"path":2089,"stem":2090},"Actions, Constants та Action Creators","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Factions-constants","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F02.actions-constants",{"title":2092,"path":2093,"stem":2094},"Логіка Reducers","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Freducers","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F03.reducers",{"title":2096,"path":2097,"stem":2098},"Комбінування Reducers (Root Reducer)","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Fdata-flow","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F04.data-flow",{"title":2100,"path":2101,"stem":2102},"Підключення до React (React-Redux)","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Freact-redux-connection","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F05.react-redux-connection",{"title":2104,"path":2105,"stem":2106},"Middleware та Асинхронність (Redux Thunk)","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Fmiddleware-thunk","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F06.middleware-thunk",{"title":2108,"icon":132,"path":2109,"stem":2110,"children":2111,"page":59},"Transition To Rtk","\u002Fjavascript\u002Freact\u002Fredux\u002Ftransition-to-rtk","03.javascript\u002F04.react\u002F06.redux\u002F03.transition-to-rtk",[2112],{"title":2113,"path":2114,"stem":2115},"Проблеми класичного Redux","\u002Fjavascript\u002Freact\u002Fredux\u002Ftransition-to-rtk\u002Fproblems-with-classic","03.javascript\u002F04.react\u002F06.redux\u002F03.transition-to-rtk\u002F01.problems-with-classic",{"title":2117,"icon":132,"path":2118,"stem":2119,"children":2120,"page":59},"Redux Toolkit","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit",[2121,2125,2129,2133],{"title":2122,"path":2123,"stem":2124},"Налаштування Store з configureStore","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit\u002Fconfigure-store","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit\u002F01.configure-store",{"title":2126,"path":2127,"stem":2128},"createSlice: Революція в Redux","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit\u002Fcreate-slice","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit\u002F02.create-slice",{"title":2130,"path":2131,"stem":2132},"Асинхронність з createAsyncThunk","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit\u002Fasync-thunks","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit\u002F03.async-thunks",{"title":2134,"path":2135,"stem":2136},"04. Entity Adapter: Керування нормалізованим станом","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit\u002Fentity-adapter","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit\u002F04.entity-adapter",{"title":2138,"icon":92,"path":2139,"stem":2140,"children":2141,"page":59},"Advanced","\u002Fjavascript\u002Freact\u002Fredux\u002Fadvanced","03.javascript\u002F04.react\u002F06.redux\u002F05.advanced",[2142,2146,2150],{"title":2143,"path":2144,"stem":2145},"Мемоізація та Селектори: Повний Гайд по Reselect","\u002Fjavascript\u002Freact\u002Fredux\u002Fadvanced\u002Fselectors-reselect","03.javascript\u002F04.react\u002F06.redux\u002F05.advanced\u002F01.selectors-reselect",{"title":2147,"path":2148,"stem":2149},"RTK Query: Архітектура Серверного Кешу","\u002Fjavascript\u002Freact\u002Fredux\u002Fadvanced\u002Frtk-query-intro","03.javascript\u002F04.react\u002F06.redux\u002F05.advanced\u002F02.rtk-query-intro",{"title":1997,"path":2151,"stem":2152},"\u002Fjavascript\u002Freact\u002Fredux\u002Fadvanced\u002Farchitecture-best-practices","03.javascript\u002F04.react\u002F06.redux\u002F05.advanced\u002F03.architecture-best-practices",{"title":2154,"icon":132,"path":2155,"stem":2156,"children":2157,"page":59},"Project Kanban","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban",[2158,2162,2166,2170,2174,2178],{"title":2159,"path":2160,"stem":2161},"Проєкт: Kanban Board (Trello Clone)","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Fproject-overview","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F01.project-overview",{"title":2163,"path":2164,"stem":2165},"Налаштування та Типізація","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Fsetup-and-types","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F02.setup-and-types",{"title":2167,"path":2168,"stem":2169},"Board Slice: Серце Дошки","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Fboard-slice","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F03.board-slice",{"title":2171,"path":2172,"stem":2173},"Логіка Drag & Drop","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Fdrag-and-drop-logic","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F04.drag-and-drop-logic",{"title":2175,"path":2176,"stem":2177},"Інтеграція з RTK Query","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Frtk-query-integration","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F05.rtk-query-integration",{"title":2179,"path":2180,"stem":2181},"Optimistic Updates","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Foptimistic-updates","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F06.optimistic-updates",{"title":2183,"icon":132,"path":2184,"stem":2185,"children":2186,"page":59},"Testing","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting","03.javascript\u002F04.react\u002F06.redux\u002F07.testing",[2187,2191,2195,2199,2203],{"title":2188,"path":2189,"stem":2190},"Тестування Redux","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Fintro-testing","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F01.intro-testing",{"title":2192,"path":2193,"stem":2194},"Тестування Reducers","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Ftesting-reducers","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F02.testing-reducers",{"title":2196,"path":2197,"stem":2198},"Тестування Селекторів","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Ftesting-selectors","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F03.testing-selectors",{"title":2200,"path":2201,"stem":2202},"Тестування Компонентів (Integration)","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Ftesting-components","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F04.testing-components",{"title":2204,"path":2205,"stem":2206},"Тестування Async Thunks","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Ftesting-thunks","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F05.testing-thunks",{"title":2208,"icon":132,"path":2209,"stem":2210,"children":2211},"Ui Libraries","\u002Fjavascript\u002Freact\u002Fui-libraries","03.javascript\u002F04.react\u002F07.ui-libraries\u002Findex",[2212,2214,2218,2222,2226,2230,2234],{"title":2213,"path":2209,"stem":2210},"UI Бібліотеки в React",{"title":2215,"path":2216,"stem":2217},"Вступ до UI Бібліотек: Навіщо Винаходити Велосипед Двічі?","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fintroduction-to-ui-libraries","03.javascript\u002F04.react\u002F07.ui-libraries\u002F01.introduction-to-ui-libraries",{"title":2219,"path":2220,"stem":2221},"Філософія shadcn\u002Fui: \"Not a Component Library\"","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-philosophy","03.javascript\u002F04.react\u002F07.ui-libraries\u002F02.shadcn-philosophy",{"title":2223,"path":2224,"stem":2225},"Установка та Налаштування shadcn\u002Fui","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-installation","03.javascript\u002F04.react\u002F07.ui-libraries\u002F03.shadcn-installation",{"title":2227,"path":2228,"stem":2229},"Базові Компоненти shadcn\u002Fui: Фундамент Інтерфейсу","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-components-basics","03.javascript\u002F04.react\u002F07.ui-libraries\u002F04.shadcn-components-basics",{"title":2231,"path":2232,"stem":2233},"Компоненти Форм: Побудова Інтерактивних Form","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-components-forms","03.javascript\u002F04.react\u002F07.ui-libraries\u002F05.shadcn-components-forms",{"title":2235,"path":2236,"stem":2237},"Складні Компоненти: Dialog, Dropdown, Table та Command","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-components-advanced","03.javascript\u002F04.react\u002F07.ui-libraries\u002F06.shadcn-components-advanced",{"title":2239,"icon":2240,"path":2241,"stem":2242,"children":2243,"page":59},"TypeScript","i-devicon-typescript","\u002Fjavascript\u002Ftypescript","03.javascript\u002F05.typescript",[2244,2248,2252,2256,2260,2264,2268,2272],{"title":2245,"path":2246,"stem":2247},"TypeScript: Броня для вашого коду","\u002Fjavascript\u002Ftypescript\u002Fintro-and-basic-types","03.javascript\u002F05.typescript\u002F01.intro-and-basic-types",{"title":2249,"path":2250,"stem":2251},"Майстерність Моделювання Даних: Інтерфейси та Просунуті Типи","\u002Fjavascript\u002Ftypescript\u002Finterfaces-and-advanced-types","03.javascript\u002F05.typescript\u002F02.interfaces-and-advanced-types",{"title":2253,"path":2254,"stem":2255},"Алхімія Типів: Generics та Utility Types","\u002Fjavascript\u002Ftypescript\u002Fgenerics-and-utilities","03.javascript\u002F05.typescript\u002F03.generics-and-utilities",{"title":2257,"path":2258,"stem":2259},"Архітектура та Шаблони: Класи в TypeScript","\u002Fjavascript\u002Ftypescript\u002Fclasses-and-oop","03.javascript\u002F05.typescript\u002F04.classes-and-oop",{"title":2261,"path":2262,"stem":2263},"Продакшн та Екосистема: Advanced Config & Workflow","\u002Fjavascript\u002Ftypescript\u002Fadvanced-patterns-and-config","03.javascript\u002F05.typescript\u002F05.advanced-patterns-and-config",{"title":2265,"path":2266,"stem":2267},"TypeScript у світі React","\u002Fjavascript\u002Ftypescript\u002Freact-basics","03.javascript\u002F05.typescript\u002F06.react-basics",{"title":2269,"path":2270,"stem":2271},"React + TypeScript: Продвинуті патерни","\u002Fjavascript\u002Ftypescript\u002Freact-advanced","03.javascript\u002F05.typescript\u002F07.react-advanced",{"title":2273,"path":2274,"stem":2275},"React + TypeScript: Екосистема та бібліотеки","\u002Fjavascript\u002Ftypescript\u002Freact-ecosystem","03.javascript\u002F05.typescript\u002F08.react-ecosystem",{"title":2277,"path":2278,"stem":2279},"Atomic Design","\u002Fjavascript\u002Fatomic-design","03.javascript\u002F2.atomic-design",{"title":2281,"icon":2282,"path":2283,"stem":2284,"children":2285,"page":59},"Java","i-devicon-java","\u002Fjava","04.java",[2286,2289,2292,2296,2300,2304,2308],{"title":162,"path":2287,"stem":2288},"\u002Fjava\u002Fdata-mapper-part1","04.java\u002F01.data-mapper-part1",{"title":166,"path":2290,"stem":2291},"\u002Fjava\u002Fdata-mapper-part2","04.java\u002F02.data-mapper-part2",{"title":2293,"path":2294,"stem":2295},"Service Layer: Організація бізнес-логіки","\u002Fjava\u002Fservice-layer","04.java\u002F03.service-layer",{"title":2297,"path":2298,"stem":2299},"Rich Domain Model та State Pattern","\u002Fjava\u002Frich-domain-model","04.java\u002F04.rich-domain-model",{"title":2301,"path":2302,"stem":2303},"Патерни для складної бізнес-логіки","\u002Fjava\u002Fbusiness-logic-patterns","04.java\u002F05.business-logic-patterns",{"title":2305,"path":2306,"stem":2307},"Обробка помилок та валідація","\u002Fjava\u002Ferror-handling-validation","04.java\u002F06.error-handling-validation",{"title":2309,"path":2310,"stem":2311,"children":2312,"page":59},"Проектування баз даних","\u002Fjava\u002Fpr2","04.java\u002Fpr2",[2313,2317,2321,2325,2329,2333,2337,2341,2345,2349,2353,2357,2361,2365,2369,2373,2377,2381,2385,2389,2393,2397,2401,2405,2409,2413,2417,2421,2425,2429,2433,2437,2441,2445,2449,2453,2457],{"title":2314,"path":2315,"stem":2316},"Концептуальне моделювання: Мистецтво розуміння предметної області","\u002Fjava\u002Fpr2\u002Fconceptual-modeling","04.java\u002Fpr2\u002F01.conceptual-modeling",{"title":2318,"path":2319,"stem":2320},"Логічне моделювання: Від бізнес-ідей до структур даних","\u002Fjava\u002Fpr2\u002Flogical-modeling","04.java\u002Fpr2\u002F02.logical-modeling",{"title":2322,"path":2323,"stem":2324},"Нормалізація: Гігієна даних та боротьба з аномаліями","\u002Fjava\u002Fpr2\u002Fnormalization","04.java\u002Fpr2\u002F03.normalization",{"title":2326,"path":2327,"stem":2328},"Фізична схема: Від абстракції до DDL","\u002Fjava\u002Fpr2\u002Fphysical-schema","04.java\u002Fpr2\u002F04.physical-schema",{"title":2330,"path":2331,"stem":2332},"Архітектурна класифікація таблиць","\u002Fjava\u002Fpr2\u002Ftable-classification","04.java\u002Fpr2\u002F05.table-classification",{"title":2334,"path":2335,"stem":2336},"Database Migrations: Версіонування схеми з Flyway","\u002Fjava\u002Fpr2\u002Fdatabase-migrations","04.java\u002Fpr2\u002F06.database-migrations",{"title":2338,"path":2339,"stem":2340},"А що, якби це була не реляційна БД?","\u002Fjava\u002Fpr2\u002Fbeyond-relational","04.java\u002Fpr2\u002F07.beyond-relational",{"title":2342,"path":2343,"stem":2344},"Object-Relational Impedance Mismatch: Два світи, що не хочуть дружити","\u002Fjava\u002Fpr2\u002Fimpedance-mismatch","04.java\u002Fpr2\u002F09.impedance-mismatch",{"title":2346,"path":2347,"stem":2348},"JDBC: Перший контакт із базою даних","\u002Fjava\u002Fpr2\u002Fjdbc-fundamentals","04.java\u002Fpr2\u002F10.jdbc-fundamentals",{"title":2350,"path":2351,"stem":2352},"Якість коду: Spotless, SpotBugs та SonarQube","\u002Fjava\u002Fpr2\u002F10a.code-quality","04.java\u002Fpr2\u002F10a.code-quality",{"title":2354,"path":2355,"stem":2356},"Connection Pool: Патерн Object Pool для JDBC-з'єднань","\u002Fjava\u002Fpr2\u002Fconnection-pool","04.java\u002Fpr2\u002F11.connection-pool",{"title":2358,"path":2359,"stem":2360},"Row Data Gateway: Об'єкт як обгортка рядка таблиці","\u002Fjava\u002Fpr2\u002Frow-data-gateway","04.java\u002Fpr2\u002F12.row-data-gateway",{"title":2362,"path":2363,"stem":2364},"Table Data Gateway: Фасад таблиці як архітектурний відступ","\u002Fjava\u002Fpr2\u002Ftable-data-gateway","04.java\u002Fpr2\u002F13.table-data-gateway",{"title":2366,"path":2367,"stem":2368},"Repository + Data Mapper: Правильна шарова архітектура з JDBC","\u002Fjava\u002Fpr2\u002Frepository-data-mapper","04.java\u002Fpr2\u002F14.repository-data-mapper",{"title":2370,"path":2371,"stem":2372},"Identity Map: Кешування сутностей у рамках сесії","\u002Fjava\u002Fpr2\u002Fidentity-map","04.java\u002Fpr2\u002F15.identity-map",{"title":2374,"path":2375,"stem":2376},"Unit of Work: Відстеження змін і координація JDBC-транзакцій","\u002Fjava\u002Fpr2\u002Funit-of-work","04.java\u002Fpr2\u002F16.unit-of-work",{"title":2378,"path":2379,"stem":2380},"Strategy: Замінювані SQL-стратегії для підтримки різних СУБД","\u002Fjava\u002Fpr2\u002Fstrategy-sql","04.java\u002Fpr2\u002F17.strategy-sql",{"title":2382,"path":2383,"stem":2384},"Proxy: Lazy Loading для One-To-Many колекцій","\u002Fjava\u002Fpr2\u002Fproxy-lazy-loading","04.java\u002Fpr2\u002F18.proxy-lazy-loading",{"title":2386,"path":2387,"stem":2388},"Generic Repository через Java Reflection: анотації та динамічний SQL","\u002Fjava\u002Fpr2\u002Fgeneric-repository-reflection","04.java\u002Fpr2\u002F19.generic-repository-reflection",{"title":2390,"path":2391,"stem":2392},"Specification Pattern: Композиція бізнес-правил для складних запитів","\u002Fjava\u002Fpr2\u002Fspecification-pattern","04.java\u002Fpr2\u002F20.specification-pattern",{"title":2394,"path":2395,"stem":2396},"Розширені можливості Specification Pattern: підзапити, агрегації та гібридний підхід","\u002Fjava\u002Fpr2\u002F20a.advanced-specifications","04.java\u002Fpr2\u002F20a.advanced-specifications",{"title":2398,"path":2399,"stem":2400},"Асинхронність у JDBC: Від блокуючих викликів до CompletableFuture","\u002Fjava\u002Fpr2\u002Fasynchronous-jdbc","04.java\u002Fpr2\u002F21.asynchronous-jdbc",{"title":2402,"path":2403,"stem":2404},"Інтеграційне тестування JDBC-репозиторіїв: Embedded H2 та патерн AAA","\u002Fjava\u002Fpr2\u002Fintegration-testing-h2","04.java\u002Fpr2\u002F22.integration-testing-h2",{"title":2406,"path":2407,"stem":2408},"Testcontainers: Тестування з реальною PostgreSQL у Docker-контейнерах","\u002Fjava\u002Fpr2\u002Fintegration-testing-testcontainers","04.java\u002Fpr2\u002F23.integration-testing-testcontainers",{"title":2410,"path":2411,"stem":2412},"Google Guice: Впровадження залежностей у JavaFX-проєкті","\u002Fjava\u002Fpr2\u002Fdependency-injection-guice","04.java\u002Fpr2\u002F24.dependency-injection-guice",{"title":2414,"path":2415,"stem":2416},"JavaFX: Основи побудови графічних інтерфейсів","\u002Fjava\u002Fpr2\u002Fjavafx-fundamentals","04.java\u002Fpr2\u002F25.javafx-fundamentals",{"title":2418,"path":2419,"stem":2420},"Properties та Bindings: Реактивність у JavaFX","\u002Fjava\u002Fpr2\u002Fjavafx-properties-bindings","04.java\u002Fpr2\u002F26.javafx-properties-bindings",{"title":2422,"path":2423,"stem":2424},"MVC vs MVP vs MVVM: Еволюція архітектурних патернів UI","\u002Fjava\u002Fpr2\u002Fui-architecture-patterns","04.java\u002Fpr2\u002F27.ui-architecture-patterns",{"title":2426,"path":2427,"stem":2428},"MVVM на практиці: Побудова ViewModel","\u002Fjava\u002Fpr2\u002Fmvvm-viewmodel-implementation","04.java\u002Fpr2\u002F28.mvvm-viewmodel-implementation",{"title":2430,"path":2431,"stem":2432},"View та Controller: Зв'язування з ViewModel через FXML","\u002Fjava\u002Fpr2\u002Fmvvm-view-controller","04.java\u002Fpr2\u002F29.mvvm-view-controller",{"title":2434,"path":2435,"stem":2436},"Інтеграція MVVM з Guice: Автоматична ін'єкція залежностей","\u002Fjava\u002Fpr2\u002Fmvvm-guice-integration","04.java\u002Fpr2\u002F30.mvvm-guice-integration",{"title":2438,"path":2439,"stem":2440},"Валідація та обробка помилок у MVVM","\u002Fjava\u002Fpr2\u002Fmvvm-validation-error-handling","04.java\u002Fpr2\u002F31.mvvm-validation-error-handling",{"title":2442,"path":2443,"stem":2444},"Навігація та управління екранами у JavaFX MVVM","\u002Fjava\u002Fpr2\u002Fmvvm-navigation-screen-management","04.java\u002Fpr2\u002F32.mvvm-navigation-screen-management",{"title":2446,"path":2447,"stem":2448},"Тестування JavaFX MVVM-додатків","\u002Fjava\u002Fpr2\u002Fmvvm-testing","04.java\u002Fpr2\u002F33.mvvm-testing",{"title":2450,"path":2451,"stem":2452},"Стилізація та теми у JavaFX: CSS та User Experience","\u002Fjava\u002Fpr2\u002Fjavafx-styling-themes","04.java\u002Fpr2\u002F34.javafx-styling-themes",{"title":2454,"path":2455,"stem":2456},"AtlantaFX: Сучасні теми для JavaFX додатків","\u002Fjava\u002Fpr2\u002Fatlantafx-modern-themes","04.java\u002Fpr2\u002F35.atlantafx-modern-themes",{"title":2458,"path":2459,"stem":2460},"Пакування та розповсюдження JavaFX-додатків","\u002Fjava\u002Fpr2\u002Fjar-packaging-distribution","04.java\u002Fpr2\u002F36.jar-packaging-distribution",{"title":2462,"icon":2463,"path":2464,"stem":2465,"children":2466,"page":59},"Бази даних","i-lucide-database","\u002Fdatabases","06.databases",[2467,2497,2520,2557,2586,2604,2638,2650,2659],{"title":2468,"icon":2469,"path":2470,"stem":2471,"children":2472,"page":59},"Intro","i-lucide-play","\u002Fdatabases\u002Fintro","06.databases\u002F01.intro",[2473,2477,2481,2485,2489,2493],{"title":2474,"path":2475,"stem":2476},"Введення в теорію баз даних","\u002Fdatabases\u002Fintro\u002Fintroduction-to-databases","06.databases\u002F01.intro\u002F01.introduction-to-databases",{"title":2478,"path":2479,"stem":2480},"Реляційна модель даних","\u002Fdatabases\u002Fintro\u002Frelational-model-theory","06.databases\u002F01.intro\u002F02.relational-model-theory",{"title":2482,"path":2483,"stem":2484},"ER-моделювання","\u002Fdatabases\u002Fintro\u002Fer-modeling","06.databases\u002F01.intro\u002F03.er-modeling",{"title":2486,"path":2487,"stem":2488},"Логічне проектування БД","\u002Fdatabases\u002Fintro\u002Flogical-schema","06.databases\u002F01.intro\u002F04.logical-schema",{"title":2490,"path":2491,"stem":2492},"Класифікація таблиць","\u002Fdatabases\u002Fintro\u002Ftable-classification","06.databases\u002F01.intro\u002F05.table-classification",{"title":2494,"path":2495,"stem":2496},"PlantUML для баз даних","\u002Fdatabases\u002Fintro\u002Fplantuml-diagrams","06.databases\u002F01.intro\u002F06.plantuml-diagrams",{"title":2498,"icon":2463,"path":2499,"stem":2500,"children":2501,"page":59},"MS SQL Server Start","\u002Fdatabases\u002Fms-sql-server-start","06.databases\u002F02.ms-sql-server-start",[2502,2506,2512,2516],{"title":2503,"path":2504,"stem":2505},"Типи даних у MS SQL Server","\u002Fdatabases\u002Fms-sql-server-start\u002Fdata-types","06.databases\u002F02.ms-sql-server-start\u002F01.data-types",{"title":2507,"path":2508,"stem":2509,"children":2510},"Індекси у MS SQL Server","\u002Fdatabases\u002Fms-sql-server-start\u002Fsql-indexes","06.databases\u002F02.ms-sql-server-start\u002F02.sql-indexes",[2511],{"title":2507,"path":2508,"stem":2509},{"title":2513,"path":2514,"stem":2515},"Системні бази даних MS SQL Server","\u002Fdatabases\u002Fms-sql-server-start\u002Fsystem-databases","06.databases\u002F02.ms-sql-server-start\u002F03.system-databases",{"title":2517,"path":2518,"stem":2519},"Огляд мови SQL та запитів","\u002Fdatabases\u002Fms-sql-server-start\u002Fsql-queries-overview","06.databases\u002F02.ms-sql-server-start\u002F04.sql-queries-overview",{"title":2521,"icon":2463,"path":2522,"stem":2523,"children":2524,"page":59},"SQL","\u002Fdatabases\u002Fsql","06.databases\u002F03.sql",[2525,2529,2533,2537,2541,2545,2549,2553],{"title":2526,"path":2527,"stem":2528},"Налаштування демонстраційної бази даних","\u002Fdatabases\u002Fsql\u002Fsample-database-setup","06.databases\u002F03.sql\u002F00.sample-database-setup",{"title":2530,"path":2531,"stem":2532},"DDL - Створення таблиць (CREATE TABLE)","\u002Fdatabases\u002Fsql\u002Fddl-create-table","06.databases\u002F03.sql\u002F01.ddl-create-table",{"title":2534,"path":2535,"stem":2536},"DDL - Зміна та видалення таблиць (ALTER, DROP)","\u002Fdatabases\u002Fsql\u002Fddl-alter-drop-table","06.databases\u002F03.sql\u002F02.ddl-alter-drop-table",{"title":2538,"path":2539,"stem":2540},"SELECT запити - Основи","\u002Fdatabases\u002Fsql\u002Fselect-queries-fundamentals","06.databases\u002F03.sql\u002F03.select-queries-fundamentals",{"title":2542,"path":2543,"stem":2544},"SELECT запити - Розширені можливості","\u002Fdatabases\u002Fsql\u002Fselect-queries-advanced","06.databases\u002F03.sql\u002F04.select-queries-advanced",{"title":2546,"path":2547,"stem":2548},"INSERT запити - Додавання даних","\u002Fdatabases\u002Fsql\u002Finsert-queries","06.databases\u002F03.sql\u002F05.insert-queries",{"title":2550,"path":2551,"stem":2552},"UPDATE та DELETE запити","\u002Fdatabases\u002Fsql\u002Fupdate-delete-queries","06.databases\u002F03.sql\u002F06.update-delete-queries",{"title":2554,"path":2555,"stem":2556},"Транзакції в SQL","\u002Fdatabases\u002Fsql\u002Ftransactions","06.databases\u002F03.sql\u002F07.transactions",{"title":2558,"icon":2463,"path":2559,"stem":2560,"children":2561,"page":59},"Multi Table Databases","\u002Fdatabases\u002Fmulti-table-databases","06.databases\u002F04.multi-table-databases",[2562,2566,2570,2574,2578,2582],{"title":2563,"path":2564,"stem":2565},"Зв'язки та нормалізація БД","\u002Fdatabases\u002Fmulti-table-databases\u002Frelationships-and-normalization","06.databases\u002F04.multi-table-databases\u002F00.relationships-and-normalization",{"title":2567,"path":2568,"stem":2569},"INNER JOIN - З'єднання таблиць","\u002Fdatabases\u002Fmulti-table-databases\u002Finner-join","06.databases\u002F04.multi-table-databases\u002F01.inner-join",{"title":2571,"path":2572,"stem":2573},"OUTER JOINs - LEFT, RIGHT, FULL","\u002Fdatabases\u002Fmulti-table-databases\u002Fouter-joins","06.databases\u002F04.multi-table-databases\u002F02.outer-joins",{"title":2575,"path":2576,"stem":2577},"CROSS та SELF JOINs","\u002Fdatabases\u002Fmulti-table-databases\u002Fcross-self-joins","06.databases\u002F04.multi-table-databases\u002F03.cross-self-joins",{"title":2579,"path":2580,"stem":2581},"Підзапити (Subqueries)","\u002Fdatabases\u002Fmulti-table-databases\u002Fsubqueries","06.databases\u002F04.multi-table-databases\u002F04.subqueries",{"title":2583,"path":2584,"stem":2585},"Агрегації з JOIN","\u002Fdatabases\u002Fmulti-table-databases\u002Faggregations-with-joins","06.databases\u002F04.multi-table-databases\u002F05.aggregations-with-joins",{"title":2587,"icon":2588,"path":2589,"stem":2590,"children":2591,"page":59},"Aggregate Functions","i-lucide-calculator","\u002Fdatabases\u002Faggregate-functions","06.databases\u002F05.aggregate-functions",[2592,2596,2600],{"title":2593,"path":2594,"stem":2595},"Функції агрегування в MS SQL Server","\u002Fdatabases\u002Faggregate-functions\u002Fintroduction-aggregate-functions","06.databases\u002F05.aggregate-functions\u002F01.introduction-aggregate-functions",{"title":2597,"path":2598,"stem":2599},"Групування даних в MS SQL Server","\u002Fdatabases\u002Faggregate-functions\u002Fgrouping-data","06.databases\u002F05.aggregate-functions\u002F02.grouping-data",{"title":2601,"path":2602,"stem":2603},"Підзапити з агрегатними функціями","\u002Fdatabases\u002Faggregate-functions\u002Fsubqueries-aggregates","06.databases\u002F05.aggregate-functions\u002F03.subqueries-aggregates",{"title":2605,"icon":2606,"path":2607,"stem":2608,"children":2609,"page":59},"Тригери та зберігаємі процедури","i-lucide-database-zap","\u002Fdatabases\u002Ftriggers-stored-procedures","06.databases\u002F07.triggers-stored-procedures",[2610,2614,2618,2622,2626,2630,2634],{"title":2611,"path":2612,"stem":2613},"DML-тригери","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fdml-triggers","06.databases\u002F07.triggers-stored-procedures\u002F01.dml-triggers",{"title":2615,"path":2616,"stem":2617},"DDL-тригери","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fddl-triggers","06.databases\u002F07.triggers-stored-procedures\u002F02.ddl-triggers",{"title":2619,"path":2620,"stem":2621},"Transact-SQL розширення","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Ftransact-sql-extensions","06.databases\u002F07.triggers-stored-procedures\u002F03.transact-sql-extensions",{"title":2623,"path":2624,"stem":2625},"Транзакції","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Ftransactions","06.databases\u002F07.triggers-stored-procedures\u002F04.transactions",{"title":2627,"path":2628,"stem":2629},"Зберігаємі процедури","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fstored-procedures","06.databases\u002F07.triggers-stored-procedures\u002F05.stored-procedures",{"title":2631,"path":2632,"stem":2633},"Користувацькі функції","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fuser-defined-functions","06.databases\u002F07.triggers-stored-procedures\u002F06.user-defined-functions",{"title":2635,"path":2636,"stem":2637},"Безпека баз даних","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fsecurity","06.databases\u002F07.triggers-stored-procedures\u002F08.security",{"title":2635,"icon":793,"path":2639,"stem":2640,"children":2641,"page":59},"\u002Fdatabases\u002Fsecurity","06.databases\u002F08.security",[2642,2646],{"title":2643,"path":2644,"stem":2645},"Вступ до безпеки баз даних","\u002Fdatabases\u002Fsecurity\u002Fintroduction","06.databases\u002F08.security\u002F01.introduction",{"title":2647,"path":2648,"stem":2649},"Системні представлення та метадані","\u002Fdatabases\u002Fsecurity\u002Fsystem-views","06.databases\u002F08.security\u002F02.system-views",{"title":2651,"icon":2652,"path":2653,"stem":2654,"children":2655,"page":59},"Резервне копіювання та відновлення","i-lucide-database-backup","\u002Fdatabases\u002Fbackup-recovery","06.databases\u002F09.backup-recovery",[2656],{"title":2651,"path":2657,"stem":2658},"\u002Fdatabases\u002Fbackup-recovery\u002Fbackup-restore","06.databases\u002F09.backup-recovery\u002F01.backup-restore",{"title":2660,"icon":2661,"path":2662,"stem":2663,"children":2664,"page":59},"Повнотекстовий пошук","i-lucide-search","\u002Fdatabases\u002Ffull-text-search","06.databases\u002F10.full-text-search",[2665],{"title":2660,"path":2666,"stem":2667},"\u002Fdatabases\u002Ffull-text-search\u002Ffull-text-search","06.databases\u002F10.full-text-search\u002F01.full-text-search",{"title":2669,"icon":2670,"path":2671,"stem":2672,"children":2673,"page":59},"Tools","i-lucide-wrench","\u002Ftools","07.tools",[2674,2750],{"title":2675,"icon":2676,"path":2677,"stem":2678,"children":2679},"Docker","i-simple-icons-docker","\u002Ftools\u002Fdocker","07.tools\u002F01.docker\u002Findex",[2680,2682,2686,2690,2694,2698,2702,2706,2710,2714,2718,2722,2726,2730,2734,2738,2742,2746],{"title":2681,"path":2677,"stem":2678},"Docker: від нуля до production",{"title":2683,"path":2684,"stem":2685},"Контейнеризація — від проблеми до рішення","\u002Ftools\u002Fdocker\u002Fcontainerization-concept","07.tools\u002F01.docker\u002F01.containerization-concept",{"title":2687,"path":2688,"stem":2689},"Docker — що це і навіщо?","\u002Ftools\u002Fdocker\u002Fdocker-what-and-why","07.tools\u002F01.docker\u002F02.docker-what-and-why",{"title":2691,"path":2692,"stem":2693},"Архітектура Docker Engine","\u002Ftools\u002Fdocker\u002Fdocker-architecture","07.tools\u002F01.docker\u002F03.docker-architecture",{"title":2695,"path":2696,"stem":2697},"Встановлення Docker","\u002Ftools\u002Fdocker\u002Finstallation","07.tools\u002F01.docker\u002F04.installation",{"title":2699,"path":2700,"stem":2701},"Перший контейнер — docker run","\u002Ftools\u002Fdocker\u002Ffirst-container","07.tools\u002F01.docker\u002F05.first-container",{"title":2703,"path":2704,"stem":2705},"Життєвий цикл контейнера","\u002Ftools\u002Fdocker\u002Fcontainer-lifecycle","07.tools\u002F01.docker\u002F06.container-lifecycle",{"title":2707,"path":2708,"stem":2709},"Docker Images — фундаментальні концепції","\u002Ftools\u002Fdocker\u002Fdocker-images-fundamentals","07.tools\u002F01.docker\u002F07.docker-images-fundamentals",{"title":2711,"path":2712,"stem":2713},"Dockerfile — основи","\u002Ftools\u002Fdocker\u002Fdockerfile-basics","07.tools\u002F01.docker\u002F08.dockerfile-basics",{"title":2715,"path":2716,"stem":2717},"Dockerfile — просунуті техніки","\u002Ftools\u002Fdocker\u002Fdockerfile-advanced","07.tools\u002F01.docker\u002F09.dockerfile-advanced",{"title":2719,"path":2720,"stem":2721},"Build Context та кешування шарів","\u002Ftools\u002Fdocker\u002Fbuild-context-and-cache","07.tools\u002F01.docker\u002F10.build-context-and-cache",{"title":2723,"path":2724,"stem":2725},"Реєстри Docker-образів","\u002Ftools\u002Fdocker\u002Fimage-registries","07.tools\u002F01.docker\u002F11.image-registries",{"title":2727,"path":2728,"stem":2729},"Контейнеризація .NET додатків","\u002Ftools\u002Fdocker\u002Fdotnet-containerization","07.tools\u002F01.docker\u002F12.dotnet-containerization",{"title":2731,"path":2732,"stem":2733},"Томи та збереження даних","\u002Ftools\u002Fdocker\u002Fvolumes-and-data","07.tools\u002F01.docker\u002F13.volumes-and-data",{"title":2735,"path":2736,"stem":2737},"Основи мережі в Docker","\u002Ftools\u002Fdocker\u002Fnetworking-basics","07.tools\u002F01.docker\u002F14.networking-basics",{"title":2739,"path":2740,"stem":2741},"Змінні оточення та конфігурація","\u002Ftools\u002Fdocker\u002Fenvironment-and-configuration","07.tools\u002F01.docker\u002F15.environment-and-configuration",{"title":2743,"path":2744,"stem":2745},"Docker Compose — оркестрація контейнерів","\u002Ftools\u002Fdocker\u002Fdocker-compose-basics","07.tools\u002F01.docker\u002F16.docker-compose-basics",{"title":2747,"path":2748,"stem":2749},"Docker Compose — Multi-Service застосунки","\u002Ftools\u002Fdocker\u002Fcompose-multi-service","07.tools\u002F01.docker\u002F17.compose-multi-service",{"title":2751,"icon":2752,"path":2753,"stem":2754,"children":2755},"Kubernetes","simple-icons:kubernetes","\u002Ftools\u002Fkubernetes","07.tools\u002F02.kubernetes\u002Findex",[2756,2758,2762,2766,2770,2774,2778,2782,2786],{"title":2757,"path":2753,"stem":2754},"Kubernetes: від розробки до production",{"title":2759,"path":2760,"stem":2761},"Kubernetes — коли Docker Compose більше не вистачає","\u002Ftools\u002Fkubernetes\u002Fwhy-kubernetes","07.tools\u002F02.kubernetes\u002F01.why-kubernetes",{"title":2763,"path":2764,"stem":2765},"Архітектура Kubernetes — анатомія кластера","\u002Ftools\u002Fkubernetes\u002Fkubernetes-architecture","07.tools\u002F02.kubernetes\u002F02.kubernetes-architecture",{"title":2767,"path":2768,"stem":2769},"Локальне середовище — minikube, kind та k3s","\u002Ftools\u002Fkubernetes\u002Flocal-environment","07.tools\u002F02.kubernetes\u002F03.local-environment",{"title":2771,"path":2772,"stem":2773},"Pod — атомарна одиниця Kubernetes","\u002Ftools\u002Fkubernetes\u002Fpods-and-containers","07.tools\u002F02.kubernetes\u002F04.pods-and-containers",{"title":2775,"path":2776,"stem":2777},"Патерни використання Pod","\u002Ftools\u002Fkubernetes\u002Fpod-patterns","07.tools\u002F02.kubernetes\u002F05.pod-patterns",{"title":2779,"path":2780,"stem":2781},"Deployment — декларативне управління Pod","\u002Ftools\u002Fkubernetes\u002Fdeployment-basics","07.tools\u002F02.kubernetes\u002F06.deployment-basics",{"title":2783,"path":2784,"stem":2785},"Rolling Updates та управління життєвим циклом Deployment","\u002Ftools\u002Fkubernetes\u002Fdeployment-rolling-updates","07.tools\u002F02.kubernetes\u002F07.deployment-rolling-updates",{"title":2787,"path":2788,"stem":2789},"Service — мережева абстракція для Pod","\u002Ftools\u002Fkubernetes\u002Fservices-networking","07.tools\u002F02.kubernetes\u002F08.services-networking",{"title":2791,"icon":2792,"path":2793,"stem":2794,"children":2795,"page":59},"Software Engineering","i-lucide-code-2","\u002Fsoftware-engineering","09.software-engineering",[2796,2800,2804,2808,2812,2816,2820,2824,2828,2832,2836],{"title":2797,"path":2798,"stem":2799},"1. Аналіз предметної області. Експертні знання та складність","\u002Fsoftware-engineering\u002Fintro-subdomains","09.software-engineering\u002F01.intro-subdomains",{"title":2801,"path":2802,"stem":2803},"2. Обмежені контексти. Інтеграція обмежених контекстів","\u002Fsoftware-engineering\u002Fintegrating-limited-contexts","09.software-engineering\u002F02.integrating-limited-contexts",{"title":2805,"path":2806,"stem":2807},"3. Реалізація простої бізнес-логіки","\u002Fsoftware-engineering\u002Fsimple","09.software-engineering\u002F03.simple",{"title":2809,"path":2810,"stem":2811},"4. Опрацювання складної бізнес-логіки","\u002Fsoftware-engineering\u002Fcomplex-business-logic","09.software-engineering\u002F04.complex-business-logic",{"title":2813,"path":2814,"stem":2815},"5. Моделювання фактора часу. Подієво-орієнтована архітектура.","\u002Fsoftware-engineering\u002Fmodelling-the-time-factor","09.software-engineering\u002F05.modelling-the-time-factor",{"title":2817,"path":2818,"stem":2819},"6. Архітектурні патерни","\u002Fsoftware-engineering\u002Farchitectural-patterns","09.software-engineering\u002F06.architectural-patterns",{"title":2821,"path":2822,"stem":2823},"Паттерни взаємодії","\u002Fsoftware-engineering\u002Fpatterns-of-interaction","09.software-engineering\u002F07.patterns-of-interaction",{"title":2825,"path":2826,"stem":2827},"Евристика проєктування","\u002Fsoftware-engineering\u002Fdesign-heuristics","09.software-engineering\u002F08.design-heuristics",{"title":2829,"path":2830,"stem":2831},"Еволюція проєктних рішень","\u002Fsoftware-engineering\u002Fevolution-of-design-solutions","09.software-engineering\u002F09.evolution-of-design-solutions",{"title":2833,"path":2834,"stem":2835},"EventStorming","\u002Fsoftware-engineering\u002Feventstorming","09.software-engineering\u002F10.eventstorming",{"title":2837,"path":2838,"stem":2839},"DDD на практиці","\u002Fsoftware-engineering\u002Fddd-in-practice","09.software-engineering\u002F11.ddd-in-practice",{"title":2841,"icon":943,"path":2842,"stem":2843,"children":2844,"page":59},"DDD","\u002Fddd","10.ddd",[2845,2849,2853,2857,2861,2865,2869,2873,2877,2881,2885,2889,2893],{"title":2846,"path":2847,"stem":2848},"Аналіз предметної області","\u002Fddd\u002Fdomain-analysis","10.ddd\u002F01.domain-analysis",{"title":2850,"path":2851,"stem":2852},"Експертні знання про предметну область","\u002Fddd\u002Fdomain-expert-knowledge","10.ddd\u002F02.domain-expert-knowledge",{"title":2854,"path":2855,"stem":2856},"Як осмислити складність предметної області","\u002Fddd\u002Fmanaging-domain-complexity","10.ddd\u002F03.managing-domain-complexity",{"title":2858,"path":2859,"stem":2860},"Інтеграція обмежених контекстів","\u002Fddd\u002Fbounded-context-integration","10.ddd\u002F04.bounded-context-integration",{"title":2862,"path":2863,"stem":2864},"Реалізація простої бізнес-логіки","\u002Fddd\u002Fsimple-business-logic","10.ddd\u002F05.simple-business-logic",{"title":2866,"path":2867,"stem":2868},"Обробка складної бізнес-логіки","\u002Fddd\u002Fcomplex-business-logic","10.ddd\u002F06.complex-business-logic",{"title":2870,"path":2871,"stem":2872},"Моделювання фактора часу","\u002Fddd\u002Ftime-modeling","10.ddd\u002F07.time-modeling",{"title":2874,"path":2875,"stem":2876},"Глава 8. Архітектурні Патерни","\u002Fddd\u002Farchitectural-patterns","10.ddd\u002F08.architectural-patterns",{"title":2878,"path":2879,"stem":2880},"Глава 9. Патерни Взаємодії","\u002Fddd\u002Finteraction-patterns","10.ddd\u002F09.interaction-patterns",{"title":2882,"path":2883,"stem":2884},"Глава 10. Проектні Евристики","\u002Fddd\u002Fdesign-heuristics","10.ddd\u002F10.design-heuristics",{"title":2886,"path":2887,"stem":2888},"Глава 11. Еволюція Проектних Рішень","\u002Fddd\u002Fevolution-of-design-decisions","10.ddd\u002F11.evolution-of-design-decisions",{"title":2890,"path":2891,"stem":2892},"Глава 12. EventStorming","\u002Fddd\u002Fevent-storming","10.ddd\u002F12.event-storming",{"title":2894,"path":2895,"stem":2896},"Глава 13. DDD на Практиці","\u002Fddd\u002Fddd-in-practice","10.ddd\u002F13.ddd-in-practice",{"title":2898,"icon":2899,"path":2900,"stem":2901,"children":2902,"page":59},"Media Streaming","i-lucide-video","\u002Fmedia-streaming","11.media-streaming",[2903,2907,2911,2915,2919,2923,2927],{"title":2904,"path":2905,"stem":2906},"01. Магія Стрімінгу: Що відбувається, коли ви натискаєте \"Play\"","\u002Fmedia-streaming\u002Fintroduction","11.media-streaming\u002F01.introduction",{"title":2908,"path":2909,"stem":2910},"02. Анатомія Медіа: Кодеки, Контейнери та Стиснення","\u002Fmedia-streaming\u002Faudio-video-anatomy","11.media-streaming\u002F02.audio-video-anatomy",{"title":2912,"path":2913,"stem":2914},"03. The Gym: FFmpeg Deep Dive","\u002Fmedia-streaming\u002Fffmpeg-gym","11.media-streaming\u002F03.ffmpeg-gym",{"title":2916,"path":2917,"stem":2918},"04. HLS Protocol: HTTP Live Streaming у Деталях","\u002Fmedia-streaming\u002Fhls-protocol","11.media-streaming\u002F04.hls-protocol",{"title":2920,"path":2921,"stem":2922},"05. DASH Protocol: Відкритий Стандарт","\u002Fmedia-streaming\u002Fdash-protocol","11.media-streaming\u002F05.dash-protocol",{"title":2924,"path":2925,"stem":2926},"06. Масштабування: CDN та Adaptive Bitrate","\u002Fmedia-streaming\u002Fcdn-and-adaptive-bitrate","11.media-streaming\u002F06.cdn-and-adaptive-bitrate",{"title":2928,"path":2929,"stem":2930},"07. Війна із Затримкою (Latency)","\u002Fmedia-streaming\u002Frealtime-latency","11.media-streaming\u002F07.realtime-latency",{"title":2932,"icon":2933,"path":2934,"stem":2935,"children":2936,"page":59},"HTML & CSS","i-devicon-html5","\u002Fhtml-css","12.html-css",[2937,2941,2945,2949,2953,2957,2961,2965,2969,2973,2977,2981,2985,2989,2993,2997,3001,3005,3009,3013,3017,3021,3025,3029,3033,3037,3041,3045,3049,3053],{"title":2938,"path":2939,"stem":2940},"Вступ до HTML. Структура документа","\u002Fhtml-css\u002Fintro-html-structure","12.html-css\u002F01.intro-html-structure",{"title":2942,"path":2943,"stem":2944},"Форматування тексту в HTML","\u002Fhtml-css\u002Fhtml-text-formatting","12.html-css\u002F02.html-text-formatting",{"title":2946,"path":2947,"stem":2948},"Посилання та зображення в HTML","\u002Fhtml-css\u002Fhtml-links-images","12.html-css\u002F03.html-links-images",{"title":2950,"path":2951,"stem":2952},"Списки та таблиці в HTML","\u002Fhtml-css\u002Fhtml-lists-tables","12.html-css\u002F04.html-lists-tables",{"title":2954,"path":2955,"stem":2956},"Форми в HTML","\u002Fhtml-css\u002Fhtml-forms","12.html-css\u002F05.html-forms",{"title":2958,"path":2959,"stem":2960},"Семантичні елементи HTML5","\u002Fhtml-css\u002Fhtml-semantic-elements","12.html-css\u002F06.html-semantic-elements",{"title":2962,"path":2963,"stem":2964},"Мультимедіа та розширені елементи HTML","\u002Fhtml-css\u002Fhtml-multimedia-advanced","12.html-css\u002F07.html-multimedia-advanced",{"title":2966,"path":2967,"stem":2968},"Мікророзмітка та SEO в HTML","\u002Fhtml-css\u002Fhtml-microdata-seo","12.html-css\u002F08.html-microdata-seo",{"title":2970,"path":2971,"stem":2972},"Вступ до CSS. Селектори та специфічність","\u002Fhtml-css\u002Fcss-intro-selectors","12.html-css\u002F09.css-intro-selectors",{"title":2974,"path":2975,"stem":2976},"Блокова модель CSS. Відступи. Box Sizing","\u002Fhtml-css\u002Fcss-box-model","12.html-css\u002F10.css-box-model",{"title":2978,"path":2979,"stem":2980},"Розміри у CSS: повний довідник одиниць і ключових слів","\u002Fhtml-css\u002F10a.css-sizing","12.html-css\u002F10a.css-sizing",{"title":2982,"path":2983,"stem":2984},"Типографіка в CSS. Шрифти та текст","\u002Fhtml-css\u002Fcss-typography","12.html-css\u002F11.css-typography",{"title":2986,"path":2987,"stem":2988},"Кольори та фони в CSS","\u002Fhtml-css\u002Fcss-colors-backgrounds","12.html-css\u002F12.css-colors-backgrounds",{"title":2990,"path":2991,"stem":2992},"Тіні та фільтри в CSS","\u002Fhtml-css\u002F12b.css-shadows-filters","12.html-css\u002F12b.css-shadows-filters",{"title":2994,"path":2995,"stem":2996},"CSS Flexbox: Фундамент гнучких макетів","\u002Fhtml-css\u002Fcss-flexbox-fundamentals","12.html-css\u002F13.css-flexbox-fundamentals",{"title":2998,"path":2999,"stem":3000},"CSS Flexbox: Вирівнювання та Позиціонування","\u002Fhtml-css\u002Fcss-flexbox-alignment-sizing-and-patterns","12.html-css\u002F14.css-flexbox-alignment-sizing-and-patterns",{"title":3002,"path":3003,"stem":3004},"CSS Grid. Двовимірний макет. Частина 1","\u002Fhtml-css\u002Fcss-layout-grid","12.html-css\u002F15.css-layout-grid",{"title":3006,"path":3007,"stem":3008},"CSS Grid. Двовимірний макет. Частина 2","\u002Fhtml-css\u002Fcss-layout-grid-advanced","12.html-css\u002F16.css-layout-grid-advanced",{"title":3010,"path":3011,"stem":3012},"Позиціонування в CSS. Z-index. Stacking Context","\u002Fhtml-css\u002Fcss-positioning","12.html-css\u002F17.css-positioning",{"title":3014,"path":3015,"stem":3016},"CSS Анімації та Переходи","\u002Fhtml-css\u002Fcss-animations-transitions","12.html-css\u002F18.css-animations-transitions",{"title":3018,"path":3019,"stem":3020},"Адаптивний дизайн. Media Queries. Частина 1","\u002Fhtml-css\u002Fcss-responsive-media-queries","12.html-css\u002F19.css-responsive-media-queries",{"title":3022,"path":3023,"stem":3024},"Адаптивний дизайн. Частина 2: clamp(), Container Queries, @layer","\u002Fhtml-css\u002Fcss-responsive-advanced","12.html-css\u002F20.css-responsive-advanced",{"title":3026,"path":3027,"stem":3028},"CSS Custom Properties. Методології. Сучасний CSS","\u002Fhtml-css\u002Fcss-variables-methodologies","12.html-css\u002F21.css-variables-methodologies",{"title":3030,"path":3031,"stem":3032},"Сучасний CSS 2023–2025: Нові можливості","\u002Fhtml-css\u002Fcss-modern-features","12.html-css\u002F22.css-modern-features",{"title":3034,"path":3035,"stem":3036},"CSS Nesting, @layer, @scope та @property: нативний препроцесор","\u002Fhtml-css\u002F22a.css-nesting-modern-syntax","12.html-css\u002F22a.css-nesting-modern-syntax",{"title":3038,"path":3039,"stem":3040},"CSS для форм та інтерактивних станів","\u002Fhtml-css\u002Fcss-forms-interactive-states","12.html-css\u002F23.css-forms-interactive-states",{"title":3042,"path":3043,"stem":3044},"Доступність у CSS (CSS Accessibility)","\u002Fhtml-css\u002Fcss-accessibility","12.html-css\u002F24.css-accessibility",{"title":3046,"path":3047,"stem":3048},"CSS-функції та сучасні sizing primitives","\u002Fhtml-css\u002Fcss-functions-sizing","12.html-css\u002F25.css-functions-sizing",{"title":3050,"path":3051,"stem":3052},"Rendering Pipeline і CSS Performance","\u002Fhtml-css\u002Fcss-rendering-performance","12.html-css\u002F26.css-rendering-performance",{"title":3054,"path":3055,"stem":3056},"CSS Best Practices: типові ситуації та правильні рішення","\u002Fhtml-css\u002Fcss-best-practices","12.html-css\u002F27.css-best-practices",{"title":3058,"path":3059,"stem":3060,"children":3061,"page":59},"AWS","\u002Faws","13.aws",[3062,3066,3070,3074,3078,3082,3086,3090],{"title":3063,"path":3064,"stem":3065},"Реєстрація AWS акаунту та студентські програми","\u002Faws\u002Faccount-registration","13.aws\u002F00.account-registration",{"title":3067,"path":3068,"stem":3069},"Вступ до хмарних обчислень та AWS","\u002Faws\u002Fintroduction-to-cloud","13.aws\u002F01.introduction-to-cloud",{"title":3071,"path":3072,"stem":3073},"AWS IAM — Identity and Access Management","\u002Faws\u002Fiam","13.aws\u002F02.iam",{"title":3075,"path":3076,"stem":3077},"Docker та контейнеризація в AWS — ECR, ECS та Fargate","\u002Faws\u002Fdocker-ecs","13.aws\u002F03.docker-ecs",{"title":3079,"path":3080,"stem":3081},"Amazon EC2 — Elastic Compute Cloud","\u002Faws\u002Fec2","13.aws\u002F04.ec2",{"title":3083,"path":3084,"stem":3085},"Elastic Load Balancing та Auto Scaling","\u002Faws\u002Falb-asg","13.aws\u002F05.alb-asg",{"title":3087,"path":3088,"stem":3089},"Amazon S3 — Simple Storage Service","\u002Faws\u002Fs3","13.aws\u002F06.s3",{"title":3091,"path":3092,"stem":3093},"Amazon CloudFront — Content Delivery Network","\u002Faws\u002Fcloudfront","13.aws\u002F07.cloudfront",{"title":3095,"path":3096,"stem":3097,"children":3098,"page":59},"Tailwind","\u002Ftailwind","21.tailwind",[3099,3103,3107,3111,3115,3119,3123,3127],{"title":3100,"path":3101,"stem":3102},"Що таке Tailwind CSS і навіщо він потрібен","\u002Ftailwind\u002Ftailwind-intro-philosophy","21.tailwind\u002F01.tailwind-intro-philosophy",{"title":3104,"path":3105,"stem":3106},"Встановлення та налаштування Tailwind CSS v4","\u002Ftailwind\u002Ftailwind-installation-setup","21.tailwind\u002F02.tailwind-installation-setup",{"title":3108,"path":3109,"stem":3110},"Utility-класи: основи та система Tailwind","\u002Ftailwind\u002Ftailwind-utility-classes-core","21.tailwind\u002F03.tailwind-utility-classes-core",{"title":3112,"path":3113,"stem":3114},"Layout: Flexbox та Grid через Tailwind","\u002Ftailwind\u002Ftailwind-flexbox-grid","21.tailwind\u002F04.tailwind-flexbox-grid",{"title":3116,"path":3117,"stem":3118},"Кастомізація теми через @theme у Tailwind v4","\u002Ftailwind\u002Ftailwind-theme-customization","21.tailwind\u002F05.tailwind-theme-customization",{"title":3120,"path":3121,"stem":3122},"Варіанти: hover, focus, responsive, dark mode та нові v4","\u002Ftailwind\u002Ftailwind-variants-states","21.tailwind\u002F06.tailwind-variants-states",{"title":3124,"path":3125,"stem":3126},"Типографіка та система кольорів у Tailwind v4","\u002Ftailwind\u002Ftailwind-typography-colors","21.tailwind\u002F07.tailwind-typography-colors",{"title":3128,"path":3129,"stem":3130},"Компоненти та повторюваність: @apply, @utility та патерни","\u002Ftailwind\u002Ftailwind-components-patterns","21.tailwind\u002F08.tailwind-components-patterns",{"title":3132,"path":3133,"stem":3134},"Тестування компонентів діаграм","\u002Ftest-components","98.test-components",{"id":3136,"title":2434,"body":3137,"description":11785,"extension":11786,"links":11787,"meta":11788,"navigation":4023,"path":2435,"seo":11789,"stem":2436,"__hash__":11790},"docs\u002F04.java\u002Fpr2\u002F30.mvvm-guice-integration.md",{"type":3138,"value":3139,"toc":11736},"minimark",[3140,3144,3149,3157,3160,3171,3178,3662,3665,3682,3688,3697,3707,3718,3721,3728,3731,3735,3738,3743,3749,3782,3788,3809,3815,3819,3876,3881,3980,3987,3989,3993,3996,4000,4006,4541,4545,4563,4575,4583,4600,4606,4610,4613,4967,4970,5046,5055,5057,5061,5070,5074,5077,5153,5156,5209,5221,5225,5230,5278,5284,5352,5360,5364,5622,5625,5703,5720,5722,5726,5732,5736,6880,6884,6903,6930,6946,6972,6988,7027,7031,7328,7330,7334,7337,7341,7347,7352,7372,7377,7492,7495,7574,7578,7583,7587,7601,7605,7650,7656,7737,7741,7744,7780,7787,7793,7816,7819,7825,7827,7831,7837,7841,8680,8684,8705,8714,8730,8744,8759,8768,8776,8790,8794,8803,8805,8809,8820,8916,8926,8930,8937,8942,9072,9078,9083,9119,9122,9127,9204,9209,9331,9346,9352,9354,9358,9361,9365,9903,9907,10368,10370,10374,10377,10381,10968,10973,10984,10988,10991,11290,11293,11485,11501,11503,11507,11647,11649,11653,11656,11665,11671,11682,11691,11697,11709,11723,11729,11732],[3141,3142,2434],"h1",{"id":3143},"інтеграція-mvvm-з-guice-автоматична-інєкція-залежностей",[3145,3146,3148],"h2",{"id":3147},"вступ-проблема-ручного-створення-viewmodel","Вступ: Проблема ручного створення ViewModel",[3150,3151,3152,3153],"p",{},"У попередніх статтях ми побудували повноцінну MVVM-архітектуру: ViewModel з Properties та Commands, Controller з Bindings, Repository для доступу до даних. Але є одна проблема, яку ми обходили стороною: ",[3154,3155,3156],"strong",{},"хто створює всі ці об'єкти?",[3150,3158,3159],{},"Розглянемо типовий граф залежностей нашого додатку:",[3161,3162,3167],"pre",{"className":3163,"code":3165,"language":3166},[3164],"language-text","AudiobookListController\n    ↓ залежить від\nAudiobookListViewModel\n    ↓ залежить від\nAudiobookService\n    ↓ залежить від\nAudiobookRepository (JdbcAudiobookRepository)\n    ↓ залежить від\nDataSource (HikariCP Connection Pool)\n","text",[3168,3169,3165],"code",{"__ignoreMap":3170},"",[3150,3172,3173,3174,3177],{},"Якщо створювати ці об'єкти вручну у ",[3168,3175,3176],{},"Application.start()",", код виглядатиме так:",[3161,3179,3183],{"className":3180,"code":3181,"language":3182,"meta":3170,"style":3170},"language-java shiki shiki-themes light-plus dark-plus dark-plus","@Override\npublic void start(Stage primaryStage) throws Exception {\n    \u002F\u002F Створення DataSource\n    HikariConfig config = new HikariConfig();\n    config.setJdbcUrl(\"jdbc:h2:mem:audiobook\");\n    config.setUsername(\"sa\");\n    config.setPassword(\"\");\n    DataSource dataSource = new HikariDataSource(config);\n    \n    \u002F\u002F Створення Repository\n    AudiobookRepository audiobookRepository = new JdbcAudiobookRepository(dataSource);\n    AuthorRepository authorRepository = new JdbcAuthorRepository(dataSource);\n    GenreRepository genreRepository = new JdbcGenreRepository(dataSource);\n    \n    \u002F\u002F Створення Service\n    AudiobookService audiobookService = new AudiobookService(\n        audiobookRepository, \n        authorRepository, \n        genreRepository\n    );\n    \n    \u002F\u002F Створення ViewModel\n    AudiobookListViewModel viewModel = new AudiobookListViewModel(audiobookService);\n    \n    \u002F\u002F Завантаження FXML\n    FXMLLoader loader = new FXMLLoader(getClass().getResource(\"audiobook-list-view.fxml\"));\n    Parent root = loader.load();\n    \n    \u002F\u002F Передача ViewModel у Controller\n    AudiobookListController controller = loader.getController();\n    controller.setViewModel(viewModel);\n    \n    \u002F\u002F Налаштування Scene\n    Scene scene = new Scene(root, 900, 600);\n    primaryStage.setScene(scene);\n    primaryStage.show();\n}\n","java",[3168,3184,3185,3198,3221,3228,3251,3272,3289,3306,3325,3331,3337,3356,3374,3392,3397,3403,3422,3428,3434,3440,3446,3451,3457,3476,3481,3487,3522,3543,3548,3554,3574,3588,3593,3599,3630,3644,3656],{"__ignoreMap":3170},[3186,3187,3190,3194],"span",{"class":3188,"line":3189},"line",1,[3186,3191,3193],{"class":3192},"sHH4Y","@",[3186,3195,3197],{"class":3196},"sN1BT","Override\n",[3186,3199,3201,3205,3208,3212,3215,3218],{"class":3188,"line":3200},2,[3186,3202,3204],{"class":3203},"su1O8","public",[3186,3206,3207],{"class":3196}," void",[3186,3209,3211],{"class":3210},"s8Opu"," start",[3186,3213,3214],{"class":3192},"(",[3186,3216,3217],{"class":3196},"Stage",[3186,3219,3220],{"class":3192}," primaryStage) throws Exception {\n",[3186,3222,3224],{"class":3188,"line":3223},3,[3186,3225,3227],{"class":3226},"spJ8K","    \u002F\u002F Створення DataSource\n",[3186,3229,3231,3234,3238,3241,3245,3248],{"class":3188,"line":3230},4,[3186,3232,3233],{"class":3196},"    HikariConfig",[3186,3235,3237],{"class":3236},"siwwj"," config",[3186,3239,3240],{"class":3192}," = ",[3186,3242,3244],{"class":3243},"s8xlr","new",[3186,3246,3247],{"class":3210}," HikariConfig",[3186,3249,3250],{"class":3192},"();\n",[3186,3252,3254,3257,3260,3263,3265,3269],{"class":3188,"line":3253},5,[3186,3255,3256],{"class":3236},"    config",[3186,3258,3259],{"class":3192},".",[3186,3261,3262],{"class":3210},"setJdbcUrl",[3186,3264,3214],{"class":3192},[3186,3266,3268],{"class":3267},"sbdoH","\"jdbc:h2:mem:audiobook\"",[3186,3270,3271],{"class":3192},");\n",[3186,3273,3275,3277,3279,3282,3284,3287],{"class":3188,"line":3274},6,[3186,3276,3256],{"class":3236},[3186,3278,3259],{"class":3192},[3186,3280,3281],{"class":3210},"setUsername",[3186,3283,3214],{"class":3192},[3186,3285,3286],{"class":3267},"\"sa\"",[3186,3288,3271],{"class":3192},[3186,3290,3292,3294,3296,3299,3301,3304],{"class":3188,"line":3291},7,[3186,3293,3256],{"class":3236},[3186,3295,3259],{"class":3192},[3186,3297,3298],{"class":3210},"setPassword",[3186,3300,3214],{"class":3192},[3186,3302,3303],{"class":3267},"\"\"",[3186,3305,3271],{"class":3192},[3186,3307,3309,3312,3315,3317,3319,3322],{"class":3188,"line":3308},8,[3186,3310,3311],{"class":3196},"    DataSource",[3186,3313,3314],{"class":3236}," dataSource",[3186,3316,3240],{"class":3192},[3186,3318,3244],{"class":3243},[3186,3320,3321],{"class":3210}," HikariDataSource",[3186,3323,3324],{"class":3192},"(config);\n",[3186,3326,3328],{"class":3188,"line":3327},9,[3186,3329,3330],{"class":3192},"    \n",[3186,3332,3334],{"class":3188,"line":3333},10,[3186,3335,3336],{"class":3226},"    \u002F\u002F Створення Repository\n",[3186,3338,3340,3343,3346,3348,3350,3353],{"class":3188,"line":3339},11,[3186,3341,3342],{"class":3196},"    AudiobookRepository",[3186,3344,3345],{"class":3236}," audiobookRepository",[3186,3347,3240],{"class":3192},[3186,3349,3244],{"class":3243},[3186,3351,3352],{"class":3210}," JdbcAudiobookRepository",[3186,3354,3355],{"class":3192},"(dataSource);\n",[3186,3357,3359,3362,3365,3367,3369,3372],{"class":3188,"line":3358},12,[3186,3360,3361],{"class":3196},"    AuthorRepository",[3186,3363,3364],{"class":3236}," authorRepository",[3186,3366,3240],{"class":3192},[3186,3368,3244],{"class":3243},[3186,3370,3371],{"class":3210}," JdbcAuthorRepository",[3186,3373,3355],{"class":3192},[3186,3375,3377,3380,3383,3385,3387,3390],{"class":3188,"line":3376},13,[3186,3378,3379],{"class":3196},"    GenreRepository",[3186,3381,3382],{"class":3236}," genreRepository",[3186,3384,3240],{"class":3192},[3186,3386,3244],{"class":3243},[3186,3388,3389],{"class":3210}," JdbcGenreRepository",[3186,3391,3355],{"class":3192},[3186,3393,3395],{"class":3188,"line":3394},14,[3186,3396,3330],{"class":3192},[3186,3398,3400],{"class":3188,"line":3399},15,[3186,3401,3402],{"class":3226},"    \u002F\u002F Створення Service\n",[3186,3404,3406,3409,3412,3414,3416,3419],{"class":3188,"line":3405},16,[3186,3407,3408],{"class":3196},"    AudiobookService",[3186,3410,3411],{"class":3236}," audiobookService",[3186,3413,3240],{"class":3192},[3186,3415,3244],{"class":3243},[3186,3417,3418],{"class":3210}," AudiobookService",[3186,3420,3421],{"class":3192},"(\n",[3186,3423,3425],{"class":3188,"line":3424},17,[3186,3426,3427],{"class":3192},"        audiobookRepository, \n",[3186,3429,3431],{"class":3188,"line":3430},18,[3186,3432,3433],{"class":3192},"        authorRepository, \n",[3186,3435,3437],{"class":3188,"line":3436},19,[3186,3438,3439],{"class":3192},"        genreRepository\n",[3186,3441,3443],{"class":3188,"line":3442},20,[3186,3444,3445],{"class":3192},"    );\n",[3186,3447,3449],{"class":3188,"line":3448},21,[3186,3450,3330],{"class":3192},[3186,3452,3454],{"class":3188,"line":3453},22,[3186,3455,3456],{"class":3226},"    \u002F\u002F Створення ViewModel\n",[3186,3458,3460,3463,3466,3468,3470,3473],{"class":3188,"line":3459},23,[3186,3461,3462],{"class":3196},"    AudiobookListViewModel",[3186,3464,3465],{"class":3236}," viewModel",[3186,3467,3240],{"class":3192},[3186,3469,3244],{"class":3243},[3186,3471,3472],{"class":3210}," AudiobookListViewModel",[3186,3474,3475],{"class":3192},"(audiobookService);\n",[3186,3477,3479],{"class":3188,"line":3478},24,[3186,3480,3330],{"class":3192},[3186,3482,3484],{"class":3188,"line":3483},25,[3186,3485,3486],{"class":3226},"    \u002F\u002F Завантаження FXML\n",[3186,3488,3490,3493,3496,3498,3500,3503,3505,3508,3511,3514,3516,3519],{"class":3188,"line":3489},26,[3186,3491,3492],{"class":3196},"    FXMLLoader",[3186,3494,3495],{"class":3236}," loader",[3186,3497,3240],{"class":3192},[3186,3499,3244],{"class":3243},[3186,3501,3502],{"class":3210}," FXMLLoader",[3186,3504,3214],{"class":3192},[3186,3506,3507],{"class":3210},"getClass",[3186,3509,3510],{"class":3192},"().",[3186,3512,3513],{"class":3210},"getResource",[3186,3515,3214],{"class":3192},[3186,3517,3518],{"class":3267},"\"audiobook-list-view.fxml\"",[3186,3520,3521],{"class":3192},"));\n",[3186,3523,3525,3528,3531,3533,3536,3538,3541],{"class":3188,"line":3524},27,[3186,3526,3527],{"class":3196},"    Parent",[3186,3529,3530],{"class":3236}," root",[3186,3532,3240],{"class":3192},[3186,3534,3535],{"class":3236},"loader",[3186,3537,3259],{"class":3192},[3186,3539,3540],{"class":3210},"load",[3186,3542,3250],{"class":3192},[3186,3544,3546],{"class":3188,"line":3545},28,[3186,3547,3330],{"class":3192},[3186,3549,3551],{"class":3188,"line":3550},29,[3186,3552,3553],{"class":3226},"    \u002F\u002F Передача ViewModel у Controller\n",[3186,3555,3557,3560,3563,3565,3567,3569,3572],{"class":3188,"line":3556},30,[3186,3558,3559],{"class":3196},"    AudiobookListController",[3186,3561,3562],{"class":3236}," controller",[3186,3564,3240],{"class":3192},[3186,3566,3535],{"class":3236},[3186,3568,3259],{"class":3192},[3186,3570,3571],{"class":3210},"getController",[3186,3573,3250],{"class":3192},[3186,3575,3577,3580,3582,3585],{"class":3188,"line":3576},31,[3186,3578,3579],{"class":3236},"    controller",[3186,3581,3259],{"class":3192},[3186,3583,3584],{"class":3210},"setViewModel",[3186,3586,3587],{"class":3192},"(viewModel);\n",[3186,3589,3591],{"class":3188,"line":3590},32,[3186,3592,3330],{"class":3192},[3186,3594,3596],{"class":3188,"line":3595},33,[3186,3597,3598],{"class":3226},"    \u002F\u002F Налаштування Scene\n",[3186,3600,3602,3605,3608,3610,3612,3615,3618,3622,3625,3628],{"class":3188,"line":3601},34,[3186,3603,3604],{"class":3196},"    Scene",[3186,3606,3607],{"class":3236}," scene",[3186,3609,3240],{"class":3192},[3186,3611,3244],{"class":3243},[3186,3613,3614],{"class":3210}," Scene",[3186,3616,3617],{"class":3192},"(root, ",[3186,3619,3621],{"class":3620},"sJj4R","900",[3186,3623,3624],{"class":3192},", ",[3186,3626,3627],{"class":3620},"600",[3186,3629,3271],{"class":3192},[3186,3631,3633,3636,3638,3641],{"class":3188,"line":3632},35,[3186,3634,3635],{"class":3236},"    primaryStage",[3186,3637,3259],{"class":3192},[3186,3639,3640],{"class":3210},"setScene",[3186,3642,3643],{"class":3192},"(scene);\n",[3186,3645,3647,3649,3651,3654],{"class":3188,"line":3646},36,[3186,3648,3635],{"class":3236},[3186,3650,3259],{"class":3192},[3186,3652,3653],{"class":3210},"show",[3186,3655,3250],{"class":3192},[3186,3657,3659],{"class":3188,"line":3658},37,[3186,3660,3661],{"class":3192},"}\n",[3150,3663,3664],{},"Цей код працює, але має кілька критичних проблем:",[3150,3666,3667,3670,3671,3674,3675,3678,3679,3259],{},[3154,3668,3669],{},"Проблема 1: Жорстке кодування залежностей."," Якщо ми захочемо замінити ",[3168,3672,3673],{},"JdbcAudiobookRepository"," на ",[3168,3676,3677],{},"InMemoryAudiobookRepository"," для тестування, доведеться змінювати код у ",[3168,3680,3681],{},"start()",[3150,3683,3684,3687],{},[3154,3685,3686],{},"Проблема 2: Дублювання коду."," Якщо у нас 10 екранів, кожен з власним Controller та ViewModel, доведеться повторювати цей код 10 разів.",[3150,3689,3690,3693,3694,3696],{},[3154,3691,3692],{},"Проблема 3: Порушення Single Responsibility Principle."," ",[3168,3695,3176],{}," не повинен знати, як створювати Repository, як конфігурувати Connection Pool, як передавати залежності. Його відповідальність — запустити додаток.",[3150,3698,3699,3702,3703,3706],{},[3154,3700,3701],{},"Проблема 4: Складність тестування."," Щоб протестувати ",[3168,3704,3705],{},"AudiobookListViewModel",", потрібно створити весь ланцюжок залежностей вручну. Це робить тести громіздкими та крихкими.",[3150,3708,3709,3710,3713,3714,3717],{},"Рішення всіх цих проблем — ",[3154,3711,3712],{},"Dependency Injection (DI)"," через ",[3154,3715,3716],{},"Google Guice",". Guice автоматично створює об'єкти, резолвить залежності, керує життєвим циклом (Singleton vs Prototype), дозволяє легко замінювати реалізації для тестування.",[3150,3719,3720],{},"У статті 24 ми вже розглядали основи Guice. Тепер застосуємо його для повної інтеграції з MVVM-архітектурою JavaFX-додатку.",[3722,3723,3724,3727],"note",{},[3154,3725,3726],{},"Guice vs Spring:"," Guice — це легковаговий DI-контейнер від Google, що фокусується на простоті та швидкості. Spring — це повноцінний фреймворк з DI, AOP, транзакціями, веб-сервером. Для desktop JavaFX-додатків Guice — оптимальний вибір: мінімальні залежності, швидкий старт, відсутність \"магії\" Spring.",[3729,3730],"hr",{},[3145,3732,3734],{"id":3733},"архітектура-інтеграції-guice-javafx","Архітектура інтеграції: Guice + JavaFX",[3150,3736,3737],{},"Перш ніж писати код, розглянемо архітектуру інтеграції Guice з JavaFX. Ключові компоненти:",[3739,3740,3742],"h3",{"id":3741},"компоненти-інтеграції","Компоненти інтеграції",[3150,3744,3745,3748],{},[3154,3746,3747],{},"1. Guice Injector"," — центральний контейнер, що створює об'єкти та впроваджує залежності. Створюється один раз при старті додатку:",[3161,3750,3752],{"className":3180,"code":3751,"language":3182,"meta":3170,"style":3170},"Injector injector = Guice.createInjector(new AudiobookModule());\n",[3168,3753,3754],{"__ignoreMap":3170},[3186,3755,3756,3759,3762,3764,3767,3769,3772,3774,3776,3779],{"class":3188,"line":3189},[3186,3757,3758],{"class":3196},"Injector",[3186,3760,3761],{"class":3236}," injector",[3186,3763,3240],{"class":3192},[3186,3765,3766],{"class":3236},"Guice",[3186,3768,3259],{"class":3192},[3186,3770,3771],{"class":3210},"createInjector",[3186,3773,3214],{"class":3192},[3186,3775,3244],{"class":3243},[3186,3777,3778],{"class":3210}," AudiobookModule",[3186,3780,3781],{"class":3192},"());\n",[3150,3783,3784,3787],{},[3154,3785,3786],{},"2. Guice Module"," — конфігурація bindings: які інтерфейси прив'язані до яких реалізацій, які об'єкти є Singleton, як створювати складні залежності (Provider).",[3150,3789,3790,3793,3794,3797,3798,3800,3801,3804,3805,3808],{},[3154,3791,3792],{},"3. ControllerFactory"," — міст між ",[3168,3795,3796],{},"FXMLLoader"," та Guice. За замовчуванням ",[3168,3799,3796],{}," створює Controllers через рефлексію (",[3168,3802,3803],{},"Class.newInstance()","), що не підтримує Constructor Injection. ",[3168,3806,3807],{},"ControllerFactory"," делегує створення Controllers до Guice Injector.",[3150,3810,3811,3814],{},[3154,3812,3813],{},"4. @Inject анотації"," — маркери для Guice: які конструктори, поля або методи потребують ін'єкції залежностей.",[3739,3816,3818],{"id":3817},"діаграма-архітектури","Діаграма архітектури",[3820,3821,3822,3870],"mermaid",{},[3150,3823,3824,3825,3832,3833,3836,3837,3843,3844,3849,3850,3852,3853,3856,3857,3859,3860,3863,3864,3866,3867],{},"graph TB\nApp",[3186,3826,3827,3828,3831],{},"AudiobookApp",[3829,3830],"br",{},"Application.start","\nInjector",[3186,3834,3835],{},"Guice Injector","\nModule",[3186,3838,3839,3840,3842],{},"AudiobookModule",[3829,3841],{},"configure bindings","\nFactory",[3186,3845,3807,3846,3848],{},[3829,3847],{},"injector::getInstance","\nLoader",[3186,3851,3796],{},"\nCtrl",[3186,3854,3855],{},"AudiobookListController","\nVM",[3186,3858,3705],{},"\nSvc",[3186,3861,3862],{},"AudiobookService","\nRepo",[3186,3865,3673],{},"\nDS",[3186,3868,3869],{},"HikariCP DataSource",[3161,3871,3874],{"className":3872,"code":3873,"language":3166},[3164],"App -->|1. createInjector| Injector\nInjector -->|reads| Module\nApp -->|2. setControllerFactory| Factory\nFactory -->|uses| Injector\nApp -->|3. load FXML| Loader\nLoader -->|4. create Controller| Factory\nFactory -->|5. getInstance| Ctrl\nInjector -->|6. inject| VM\nInjector -->|7. inject| Svc\nInjector -->|8. inject| Repo\nInjector -->|9. inject| DS\n\nCtrl -.depends on.-> VM\nVM -.depends on.-> Svc\nSvc -.depends on.-> Repo\nRepo -.depends on.-> DS\n\nstyle App fill:#e1f5ff\nstyle Injector fill:#fff4e1\nstyle Module fill:#f0f0f0\nstyle Factory fill:#ffe1e1\n",[3168,3875,3873],{"__ignoreMap":3170},[3150,3877,3878],{},[3154,3879,3880],{},"Пояснення потоку:",[3882,3883,3884,3893,3904,3919,3927,3935,3943,3949,3962,3968],"ol",{},[3885,3886,3887,3890,3891,3259],"li",{},[3168,3888,3889],{},"AudiobookApp.start()"," створює Guice Injector з ",[3168,3892,3839],{},[3885,3894,3895,3897,3898,3624,3901,3259],{},[3168,3896,3839],{}," конфігурує bindings: ",[3168,3899,3900],{},"AudiobookRepository → JdbcAudiobookRepository",[3168,3902,3903],{},"DataSource → HikariDataSource",[3885,3905,3906,3909,3910,3912,3913,3915,3916,3259],{},[3168,3907,3908],{},"Application"," встановлює ",[3168,3911,3807],{}," для ",[3168,3914,3796],{},": ",[3168,3917,3918],{},"loader.setControllerFactory(injector::getInstance)",[3885,3920,3921,3923,3924,3259],{},[3168,3922,3796],{}," парсить FXML, знаходить ",[3168,3925,3926],{},"fx:controller=\"AudiobookListController\"",[3885,3928,3929,3931,3932,3259],{},[3168,3930,3796],{}," викликає ",[3168,3933,3934],{},"ControllerFactory.call(AudiobookListController.class)",[3885,3936,3937,3939,3940,3259],{},[3168,3938,3807],{}," делегує до ",[3168,3941,3942],{},"Injector.getInstance(AudiobookListController.class)",[3885,3944,3945,3946,3259],{},"Guice аналізує конструктор ",[3168,3947,3948],{},"AudiobookListController(@Inject AudiobookListViewModel viewModel)",[3885,3950,3951,3952,3954,3955,3954,3957,3954,3959,3259],{},"Guice рекурсивно резолвить залежності: створює ",[3168,3953,3705],{}," → ",[3168,3956,3862],{},[3168,3958,3673],{},[3168,3960,3961],{},"HikariDataSource",[3885,3963,3964,3965,3967],{},"Guice повертає повністю ініціалізований ",[3168,3966,3855],{}," з усіма залежностями.",[3885,3969,3970,3972,3973,3976,3977,3259],{},[3168,3971,3796],{}," ін'єктує ",[3168,3974,3975],{},"@FXML"," поля (UI-елементи) та викликає ",[3168,3978,3979],{},"initialize()",[3981,3982,3983,3986],"tip",{},[3154,3984,3985],{},"Переваги цієї архітектури:"," Додавання нового екрану зводиться до створення FXML, Controller та ViewModel. Guice автоматично резолвить всі залежності — не потрібно писати код створення об'єктів.",[3729,3988],{},[3145,3990,3992],{"id":3991},"guice-module-конфігурація-залежностей","Guice Module: Конфігурація залежностей",[3150,3994,3995],{},"Guice Module — це клас, що конфігурує bindings: які інтерфейси прив'язані до яких реалізацій, які об'єкти є Singleton, як створювати складні залежності.",[3739,3997,3999],{"id":3998},"створення-audiobookmodule","Створення AudiobookModule",[3150,4001,4002,4003,4005],{},"Створимо ",[3168,4004,3839],{}," — центральну конфігурацію нашого додатку:",[3161,4007,4009],{"className":3180,"code":4008,"language":3182,"meta":3170,"style":3170},"package dev.kostyl.audiobook.infrastructure;\n\nimport com.google.inject.AbstractModule;\nimport com.google.inject.Provides;\nimport com.google.inject.Singleton;\nimport dev.kostyl.audiobook.repository.AudiobookRepository;\nimport dev.kostyl.audiobook.repository.AuthorRepository;\nimport dev.kostyl.audiobook.repository.GenreRepository;\nimport dev.kostyl.audiobook.repository.jdbc.JdbcAudiobookRepository;\nimport dev.kostyl.audiobook.repository.jdbc.JdbcAuthorRepository;\nimport dev.kostyl.audiobook.repository.jdbc.JdbcGenreRepository;\nimport dev.kostyl.audiobook.service.AudiobookService;\nimport dev.kostyl.audiobook.viewmodel.AudiobookListViewModel;\n\nimport javax.sql.DataSource;\n\npublic class AudiobookModule extends AbstractModule {\n    \n    @Override\n    protected void configure() {\n        \u002F\u002F ===== Data Access Layer =====\n        \n        \u002F\u002F Repositories: інтерфейс → реалізація\n        bind(AudiobookRepository.class).to(JdbcAudiobookRepository.class);\n        bind(AuthorRepository.class).to(JdbcAuthorRepository.class);\n        bind(GenreRepository.class).to(JdbcGenreRepository.class);\n        \n        \u002F\u002F ===== Service Layer =====\n        \n        \u002F\u002F Services як Singleton (stateless, можна перевикористовувати)\n        bind(AudiobookService.class).in(Singleton.class);\n        \n        \u002F\u002F ===== Presentation Layer =====\n        \n        \u002F\u002F ViewModels як Singleton (зберігають стан між переходами)\n        bind(AudiobookListViewModel.class).in(Singleton.class);\n    }\n    \n    \u002F\u002F ===== Provider Methods =====\n    \n    @Provides\n    @Singleton\n    DataSource provideDataSource() {\n        HikariConfig config = new HikariConfig();\n        config.setJdbcUrl(\"jdbc:h2:mem:audiobook;DB_CLOSE_DELAY=-1\");\n        config.setUsername(\"sa\");\n        config.setPassword(\"\");\n        config.setMaximumPoolSize(10);\n        config.setMinimumIdle(2);\n        config.setConnectionTimeout(30000);\n        \n        return new HikariDataSource(config);\n    }\n}\n",[3168,4010,4011,4019,4025,4033,4040,4047,4054,4061,4068,4075,4082,4089,4096,4103,4107,4114,4118,4136,4140,4147,4160,4165,4170,4175,4206,4234,4262,4266,4271,4275,4280,4308,4312,4317,4321,4326,4352,4357,4362,4368,4373,4381,4389,4399,4415,4432,4447,4462,4479,4496,4513,4518,4531,4536],{"__ignoreMap":3170},[3186,4012,4013,4016],{"class":3188,"line":3189},[3186,4014,4015],{"class":3203},"package",[3186,4017,4018],{"class":3192}," dev.kostyl.audiobook.infrastructure;\n",[3186,4020,4021],{"class":3188,"line":3200},[3186,4022,4024],{"emptyLinePlaceholder":4023},true,"\n",[3186,4026,4027,4030],{"class":3188,"line":3223},[3186,4028,4029],{"class":3203},"import",[3186,4031,4032],{"class":3192}," com.google.inject.AbstractModule;\n",[3186,4034,4035,4037],{"class":3188,"line":3230},[3186,4036,4029],{"class":3203},[3186,4038,4039],{"class":3192}," com.google.inject.Provides;\n",[3186,4041,4042,4044],{"class":3188,"line":3253},[3186,4043,4029],{"class":3203},[3186,4045,4046],{"class":3192}," com.google.inject.Singleton;\n",[3186,4048,4049,4051],{"class":3188,"line":3274},[3186,4050,4029],{"class":3203},[3186,4052,4053],{"class":3192}," dev.kostyl.audiobook.repository.AudiobookRepository;\n",[3186,4055,4056,4058],{"class":3188,"line":3291},[3186,4057,4029],{"class":3203},[3186,4059,4060],{"class":3192}," dev.kostyl.audiobook.repository.AuthorRepository;\n",[3186,4062,4063,4065],{"class":3188,"line":3308},[3186,4064,4029],{"class":3203},[3186,4066,4067],{"class":3192}," dev.kostyl.audiobook.repository.GenreRepository;\n",[3186,4069,4070,4072],{"class":3188,"line":3327},[3186,4071,4029],{"class":3203},[3186,4073,4074],{"class":3192}," dev.kostyl.audiobook.repository.jdbc.JdbcAudiobookRepository;\n",[3186,4076,4077,4079],{"class":3188,"line":3333},[3186,4078,4029],{"class":3203},[3186,4080,4081],{"class":3192}," dev.kostyl.audiobook.repository.jdbc.JdbcAuthorRepository;\n",[3186,4083,4084,4086],{"class":3188,"line":3339},[3186,4085,4029],{"class":3203},[3186,4087,4088],{"class":3192}," dev.kostyl.audiobook.repository.jdbc.JdbcGenreRepository;\n",[3186,4090,4091,4093],{"class":3188,"line":3358},[3186,4092,4029],{"class":3203},[3186,4094,4095],{"class":3192}," dev.kostyl.audiobook.service.AudiobookService;\n",[3186,4097,4098,4100],{"class":3188,"line":3376},[3186,4099,4029],{"class":3203},[3186,4101,4102],{"class":3192}," dev.kostyl.audiobook.viewmodel.AudiobookListViewModel;\n",[3186,4104,4105],{"class":3188,"line":3394},[3186,4106,4024],{"emptyLinePlaceholder":4023},[3186,4108,4109,4111],{"class":3188,"line":3399},[3186,4110,4029],{"class":3203},[3186,4112,4113],{"class":3192}," javax.sql.DataSource;\n",[3186,4115,4116],{"class":3188,"line":3405},[3186,4117,4024],{"emptyLinePlaceholder":4023},[3186,4119,4120,4122,4125,4127,4130,4133],{"class":3188,"line":3424},[3186,4121,3204],{"class":3203},[3186,4123,4124],{"class":3203}," class",[3186,4126,3778],{"class":3196},[3186,4128,4129],{"class":3203}," extends",[3186,4131,4132],{"class":3196}," AbstractModule",[3186,4134,4135],{"class":3192}," {\n",[3186,4137,4138],{"class":3188,"line":3430},[3186,4139,3330],{"class":3192},[3186,4141,4142,4145],{"class":3188,"line":3436},[3186,4143,4144],{"class":3192},"    @",[3186,4146,3197],{"class":3196},[3186,4148,4149,4152,4154,4157],{"class":3188,"line":3442},[3186,4150,4151],{"class":3203},"    protected",[3186,4153,3207],{"class":3196},[3186,4155,4156],{"class":3210}," configure",[3186,4158,4159],{"class":3192},"() {\n",[3186,4161,4162],{"class":3188,"line":3448},[3186,4163,4164],{"class":3226},"        \u002F\u002F ===== Data Access Layer =====\n",[3186,4166,4167],{"class":3188,"line":3453},[3186,4168,4169],{"class":3192},"        \n",[3186,4171,4172],{"class":3188,"line":3459},[3186,4173,4174],{"class":3226},"        \u002F\u002F Repositories: інтерфейс → реалізація\n",[3186,4176,4177,4180,4182,4185,4187,4190,4193,4196,4198,4200,4202,4204],{"class":3188,"line":3478},[3186,4178,4179],{"class":3210},"        bind",[3186,4181,3214],{"class":3192},[3186,4183,4184],{"class":3236},"AudiobookRepository",[3186,4186,3259],{"class":3192},[3186,4188,4189],{"class":3236},"class",[3186,4191,4192],{"class":3192},").",[3186,4194,4195],{"class":3210},"to",[3186,4197,3214],{"class":3192},[3186,4199,3673],{"class":3236},[3186,4201,3259],{"class":3192},[3186,4203,4189],{"class":3236},[3186,4205,3271],{"class":3192},[3186,4207,4208,4210,4212,4215,4217,4219,4221,4223,4225,4228,4230,4232],{"class":3188,"line":3483},[3186,4209,4179],{"class":3210},[3186,4211,3214],{"class":3192},[3186,4213,4214],{"class":3236},"AuthorRepository",[3186,4216,3259],{"class":3192},[3186,4218,4189],{"class":3236},[3186,4220,4192],{"class":3192},[3186,4222,4195],{"class":3210},[3186,4224,3214],{"class":3192},[3186,4226,4227],{"class":3236},"JdbcAuthorRepository",[3186,4229,3259],{"class":3192},[3186,4231,4189],{"class":3236},[3186,4233,3271],{"class":3192},[3186,4235,4236,4238,4240,4243,4245,4247,4249,4251,4253,4256,4258,4260],{"class":3188,"line":3489},[3186,4237,4179],{"class":3210},[3186,4239,3214],{"class":3192},[3186,4241,4242],{"class":3236},"GenreRepository",[3186,4244,3259],{"class":3192},[3186,4246,4189],{"class":3236},[3186,4248,4192],{"class":3192},[3186,4250,4195],{"class":3210},[3186,4252,3214],{"class":3192},[3186,4254,4255],{"class":3236},"JdbcGenreRepository",[3186,4257,3259],{"class":3192},[3186,4259,4189],{"class":3236},[3186,4261,3271],{"class":3192},[3186,4263,4264],{"class":3188,"line":3524},[3186,4265,4169],{"class":3192},[3186,4267,4268],{"class":3188,"line":3545},[3186,4269,4270],{"class":3226},"        \u002F\u002F ===== Service Layer =====\n",[3186,4272,4273],{"class":3188,"line":3550},[3186,4274,4169],{"class":3192},[3186,4276,4277],{"class":3188,"line":3556},[3186,4278,4279],{"class":3226},"        \u002F\u002F Services як Singleton (stateless, можна перевикористовувати)\n",[3186,4281,4282,4284,4286,4288,4290,4292,4294,4297,4299,4302,4304,4306],{"class":3188,"line":3576},[3186,4283,4179],{"class":3210},[3186,4285,3214],{"class":3192},[3186,4287,3862],{"class":3236},[3186,4289,3259],{"class":3192},[3186,4291,4189],{"class":3236},[3186,4293,4192],{"class":3192},[3186,4295,4296],{"class":3210},"in",[3186,4298,3214],{"class":3192},[3186,4300,4301],{"class":3236},"Singleton",[3186,4303,3259],{"class":3192},[3186,4305,4189],{"class":3236},[3186,4307,3271],{"class":3192},[3186,4309,4310],{"class":3188,"line":3590},[3186,4311,4169],{"class":3192},[3186,4313,4314],{"class":3188,"line":3595},[3186,4315,4316],{"class":3226},"        \u002F\u002F ===== Presentation Layer =====\n",[3186,4318,4319],{"class":3188,"line":3601},[3186,4320,4169],{"class":3192},[3186,4322,4323],{"class":3188,"line":3632},[3186,4324,4325],{"class":3226},"        \u002F\u002F ViewModels як Singleton (зберігають стан між переходами)\n",[3186,4327,4328,4330,4332,4334,4336,4338,4340,4342,4344,4346,4348,4350],{"class":3188,"line":3646},[3186,4329,4179],{"class":3210},[3186,4331,3214],{"class":3192},[3186,4333,3705],{"class":3236},[3186,4335,3259],{"class":3192},[3186,4337,4189],{"class":3236},[3186,4339,4192],{"class":3192},[3186,4341,4296],{"class":3210},[3186,4343,3214],{"class":3192},[3186,4345,4301],{"class":3236},[3186,4347,3259],{"class":3192},[3186,4349,4189],{"class":3236},[3186,4351,3271],{"class":3192},[3186,4353,4354],{"class":3188,"line":3658},[3186,4355,4356],{"class":3192},"    }\n",[3186,4358,4360],{"class":3188,"line":4359},38,[3186,4361,3330],{"class":3192},[3186,4363,4365],{"class":3188,"line":4364},39,[3186,4366,4367],{"class":3226},"    \u002F\u002F ===== Provider Methods =====\n",[3186,4369,4371],{"class":3188,"line":4370},40,[3186,4372,3330],{"class":3192},[3186,4374,4376,4378],{"class":3188,"line":4375},41,[3186,4377,4144],{"class":3192},[3186,4379,4380],{"class":3196},"Provides\n",[3186,4382,4384,4386],{"class":3188,"line":4383},42,[3186,4385,4144],{"class":3192},[3186,4387,4388],{"class":3196},"Singleton\n",[3186,4390,4392,4394,4397],{"class":3188,"line":4391},43,[3186,4393,3311],{"class":3196},[3186,4395,4396],{"class":3210}," provideDataSource",[3186,4398,4159],{"class":3192},[3186,4400,4402,4405,4407,4409,4411,4413],{"class":3188,"line":4401},44,[3186,4403,4404],{"class":3196},"        HikariConfig",[3186,4406,3237],{"class":3236},[3186,4408,3240],{"class":3192},[3186,4410,3244],{"class":3243},[3186,4412,3247],{"class":3210},[3186,4414,3250],{"class":3192},[3186,4416,4418,4421,4423,4425,4427,4430],{"class":3188,"line":4417},45,[3186,4419,4420],{"class":3236},"        config",[3186,4422,3259],{"class":3192},[3186,4424,3262],{"class":3210},[3186,4426,3214],{"class":3192},[3186,4428,4429],{"class":3267},"\"jdbc:h2:mem:audiobook;DB_CLOSE_DELAY=-1\"",[3186,4431,3271],{"class":3192},[3186,4433,4435,4437,4439,4441,4443,4445],{"class":3188,"line":4434},46,[3186,4436,4420],{"class":3236},[3186,4438,3259],{"class":3192},[3186,4440,3281],{"class":3210},[3186,4442,3214],{"class":3192},[3186,4444,3286],{"class":3267},[3186,4446,3271],{"class":3192},[3186,4448,4450,4452,4454,4456,4458,4460],{"class":3188,"line":4449},47,[3186,4451,4420],{"class":3236},[3186,4453,3259],{"class":3192},[3186,4455,3298],{"class":3210},[3186,4457,3214],{"class":3192},[3186,4459,3303],{"class":3267},[3186,4461,3271],{"class":3192},[3186,4463,4465,4467,4469,4472,4474,4477],{"class":3188,"line":4464},48,[3186,4466,4420],{"class":3236},[3186,4468,3259],{"class":3192},[3186,4470,4471],{"class":3210},"setMaximumPoolSize",[3186,4473,3214],{"class":3192},[3186,4475,4476],{"class":3620},"10",[3186,4478,3271],{"class":3192},[3186,4480,4482,4484,4486,4489,4491,4494],{"class":3188,"line":4481},49,[3186,4483,4420],{"class":3236},[3186,4485,3259],{"class":3192},[3186,4487,4488],{"class":3210},"setMinimumIdle",[3186,4490,3214],{"class":3192},[3186,4492,4493],{"class":3620},"2",[3186,4495,3271],{"class":3192},[3186,4497,4499,4501,4503,4506,4508,4511],{"class":3188,"line":4498},50,[3186,4500,4420],{"class":3236},[3186,4502,3259],{"class":3192},[3186,4504,4505],{"class":3210},"setConnectionTimeout",[3186,4507,3214],{"class":3192},[3186,4509,4510],{"class":3620},"30000",[3186,4512,3271],{"class":3192},[3186,4514,4516],{"class":3188,"line":4515},51,[3186,4517,4169],{"class":3192},[3186,4519,4521,4524,4527,4529],{"class":3188,"line":4520},52,[3186,4522,4523],{"class":3243},"        return",[3186,4525,4526],{"class":3243}," new",[3186,4528,3321],{"class":3210},[3186,4530,3324],{"class":3192},[3186,4532,4534],{"class":3188,"line":4533},53,[3186,4535,4356],{"class":3192},[3186,4537,4539],{"class":3188,"line":4538},54,[3186,4540,3661],{"class":3192},[3739,4542,4544],{"id":4543},"розбір-конфігурації","Розбір конфігурації",[3150,4546,4547,3693,4550,4553,4554,4556,4557,4559,4560,4562],{},[3154,4548,4549],{},"Рядки 20-25: Bindings для Repositories.",[3168,4551,4552],{},"bind(AudiobookRepository.class).to(JdbcAudiobookRepository.class)"," означає: \"коли хтось запитує ",[3168,4555,4184],{},", створи екземпляр ",[3168,4558,3673],{},"\". Це дозволяє легко замінити реалізацію (наприклад, на ",[3168,4561,3677],{}," для тестів) без зміни коду, що використовує Repository.",[3150,4564,4565,3693,4568,4571,4572,4574],{},[3154,4566,4567],{},"Рядок 30: Service як Singleton.",[3168,4569,4570],{},"bind(AudiobookService.class).in(Singleton.class)"," — Guice створить один екземпляр ",[3168,4573,3862],{}," при першому запиті та повертатиме його для всіх наступних запитів. Це оптимізація: Service не має стану (stateless), тому немає сенсу створювати новий екземпляр кожного разу.",[3150,4576,4577,3693,4580,4582],{},[3154,4578,4579],{},"Рядок 35: ViewModel як Singleton.",[3168,4581,3705],{}," — Singleton, бо він зберігає стан (список аудіокниг, фільтри, вибраний елемент), що має зберігатися між переходами між екранами. Якщо користувач переходить з \"Audiobook List\" до \"Audiobook Details\" і повертається назад, список має залишитися у тому самому стані (прокрутка, фільтри, вибір).",[3150,4584,4585,3693,4588,4591,4592,4595,4596,4599],{},[3154,4586,4587],{},"Рядки 41-53: Provider Method для DataSource.",[3168,4589,4590],{},"@Provides"," метод — це спосіб створити складний об'єкт, що потребує конфігурації. Guice викличе цей метод один раз (через ",[3168,4593,4594],{},"@Singleton",") та збереже результат. Provider Method зручніший за ",[3168,4597,4598],{},"bind().toProvider()",", коли логіка створення проста.",[3722,4601,4602,4605],{},[3154,4603,4604],{},"Чому DataSource — Singleton?"," Connection Pool (HikariCP) — це дорогий ресурс: він створює з'єднання з базою даних, керує їхнім життєвим циклом. Створювати новий Pool при кожному запиті — це витік ресурсів. Один Pool на весь додаток — стандартна практика.",[3739,4607,4609],{"id":4608},"альтернатива-provider-class","Альтернатива: Provider Class",[3150,4611,4612],{},"Якщо логіка створення DataSource складна (читання конфігурації з файлу, міграції через Flyway), краще винести її у окремий Provider:",[3161,4614,4616],{"className":3180,"code":4615,"language":3182,"meta":3170,"style":3170},"public class HikariDataSourceProvider implements Provider\u003CDataSource> {\n    \n    @Override\n    public DataSource get() {\n        \u002F\u002F Читання конфігурації з application.properties\n        Properties props = loadProperties(\"application.properties\");\n        \n        HikariConfig config = new HikariConfig();\n        config.setJdbcUrl(props.getProperty(\"db.url\"));\n        config.setUsername(props.getProperty(\"db.username\"));\n        config.setPassword(props.getProperty(\"db.password\"));\n        config.setMaximumPoolSize(Integer.parseInt(props.getProperty(\"db.pool.size\")));\n        \n        DataSource dataSource = new HikariDataSource(config);\n        \n        \u002F\u002F Виконання міграцій через Flyway\n        Flyway flyway = Flyway.configure()\n            .dataSource(dataSource)\n            .locations(\"classpath:db\u002Fmigration\")\n            .load();\n        flyway.migrate();\n        \n        return dataSource;\n    }\n    \n    private Properties loadProperties(String filename) {\n        \u002F\u002F Реалізація завантаження properties\n        \u002F\u002F ...\n    }\n}\n",[3168,4617,4618,4642,4646,4652,4665,4670,4690,4694,4708,4733,4756,4779,4813,4817,4832,4836,4841,4862,4873,4888,4896,4908,4912,4919,4923,4927,4949,4954,4959,4963],{"__ignoreMap":3170},[3186,4619,4620,4622,4624,4627,4630,4633,4636,4639],{"class":3188,"line":3189},[3186,4621,3204],{"class":3203},[3186,4623,4124],{"class":3203},[3186,4625,4626],{"class":3196}," HikariDataSourceProvider",[3186,4628,4629],{"class":3203}," implements",[3186,4631,4632],{"class":3196}," Provider",[3186,4634,4635],{"class":3192},"\u003C",[3186,4637,4638],{"class":3196},"DataSource",[3186,4640,4641],{"class":3192},"> {\n",[3186,4643,4644],{"class":3188,"line":3200},[3186,4645,3330],{"class":3192},[3186,4647,4648,4650],{"class":3188,"line":3223},[3186,4649,4144],{"class":3192},[3186,4651,3197],{"class":3196},[3186,4653,4654,4657,4660,4663],{"class":3188,"line":3230},[3186,4655,4656],{"class":3203},"    public",[3186,4658,4659],{"class":3196}," DataSource",[3186,4661,4662],{"class":3210}," get",[3186,4664,4159],{"class":3192},[3186,4666,4667],{"class":3188,"line":3253},[3186,4668,4669],{"class":3226},"        \u002F\u002F Читання конфігурації з application.properties\n",[3186,4671,4672,4675,4678,4680,4683,4685,4688],{"class":3188,"line":3274},[3186,4673,4674],{"class":3196},"        Properties",[3186,4676,4677],{"class":3236}," props",[3186,4679,3240],{"class":3192},[3186,4681,4682],{"class":3210},"loadProperties",[3186,4684,3214],{"class":3192},[3186,4686,4687],{"class":3267},"\"application.properties\"",[3186,4689,3271],{"class":3192},[3186,4691,4692],{"class":3188,"line":3291},[3186,4693,4169],{"class":3192},[3186,4695,4696,4698,4700,4702,4704,4706],{"class":3188,"line":3308},[3186,4697,4404],{"class":3196},[3186,4699,3237],{"class":3236},[3186,4701,3240],{"class":3192},[3186,4703,3244],{"class":3243},[3186,4705,3247],{"class":3210},[3186,4707,3250],{"class":3192},[3186,4709,4710,4712,4714,4716,4718,4721,4723,4726,4728,4731],{"class":3188,"line":3327},[3186,4711,4420],{"class":3236},[3186,4713,3259],{"class":3192},[3186,4715,3262],{"class":3210},[3186,4717,3214],{"class":3192},[3186,4719,4720],{"class":3236},"props",[3186,4722,3259],{"class":3192},[3186,4724,4725],{"class":3210},"getProperty",[3186,4727,3214],{"class":3192},[3186,4729,4730],{"class":3267},"\"db.url\"",[3186,4732,3521],{"class":3192},[3186,4734,4735,4737,4739,4741,4743,4745,4747,4749,4751,4754],{"class":3188,"line":3333},[3186,4736,4420],{"class":3236},[3186,4738,3259],{"class":3192},[3186,4740,3281],{"class":3210},[3186,4742,3214],{"class":3192},[3186,4744,4720],{"class":3236},[3186,4746,3259],{"class":3192},[3186,4748,4725],{"class":3210},[3186,4750,3214],{"class":3192},[3186,4752,4753],{"class":3267},"\"db.username\"",[3186,4755,3521],{"class":3192},[3186,4757,4758,4760,4762,4764,4766,4768,4770,4772,4774,4777],{"class":3188,"line":3339},[3186,4759,4420],{"class":3236},[3186,4761,3259],{"class":3192},[3186,4763,3298],{"class":3210},[3186,4765,3214],{"class":3192},[3186,4767,4720],{"class":3236},[3186,4769,3259],{"class":3192},[3186,4771,4725],{"class":3210},[3186,4773,3214],{"class":3192},[3186,4775,4776],{"class":3267},"\"db.password\"",[3186,4778,3521],{"class":3192},[3186,4780,4781,4783,4785,4787,4789,4792,4794,4797,4799,4801,4803,4805,4807,4810],{"class":3188,"line":3358},[3186,4782,4420],{"class":3236},[3186,4784,3259],{"class":3192},[3186,4786,4471],{"class":3210},[3186,4788,3214],{"class":3192},[3186,4790,4791],{"class":3236},"Integer",[3186,4793,3259],{"class":3192},[3186,4795,4796],{"class":3210},"parseInt",[3186,4798,3214],{"class":3192},[3186,4800,4720],{"class":3236},[3186,4802,3259],{"class":3192},[3186,4804,4725],{"class":3210},[3186,4806,3214],{"class":3192},[3186,4808,4809],{"class":3267},"\"db.pool.size\"",[3186,4811,4812],{"class":3192},")));\n",[3186,4814,4815],{"class":3188,"line":3376},[3186,4816,4169],{"class":3192},[3186,4818,4819,4822,4824,4826,4828,4830],{"class":3188,"line":3394},[3186,4820,4821],{"class":3196},"        DataSource",[3186,4823,3314],{"class":3236},[3186,4825,3240],{"class":3192},[3186,4827,3244],{"class":3243},[3186,4829,3321],{"class":3210},[3186,4831,3324],{"class":3192},[3186,4833,4834],{"class":3188,"line":3399},[3186,4835,4169],{"class":3192},[3186,4837,4838],{"class":3188,"line":3405},[3186,4839,4840],{"class":3226},"        \u002F\u002F Виконання міграцій через Flyway\n",[3186,4842,4843,4846,4849,4851,4854,4856,4859],{"class":3188,"line":3424},[3186,4844,4845],{"class":3196},"        Flyway",[3186,4847,4848],{"class":3236}," flyway",[3186,4850,3240],{"class":3192},[3186,4852,4853],{"class":3236},"Flyway",[3186,4855,3259],{"class":3192},[3186,4857,4858],{"class":3210},"configure",[3186,4860,4861],{"class":3192},"()\n",[3186,4863,4864,4867,4870],{"class":3188,"line":3430},[3186,4865,4866],{"class":3192},"            .",[3186,4868,4869],{"class":3210},"dataSource",[3186,4871,4872],{"class":3192},"(dataSource)\n",[3186,4874,4875,4877,4880,4882,4885],{"class":3188,"line":3436},[3186,4876,4866],{"class":3192},[3186,4878,4879],{"class":3210},"locations",[3186,4881,3214],{"class":3192},[3186,4883,4884],{"class":3267},"\"classpath:db\u002Fmigration\"",[3186,4886,4887],{"class":3192},")\n",[3186,4889,4890,4892,4894],{"class":3188,"line":3442},[3186,4891,4866],{"class":3192},[3186,4893,3540],{"class":3210},[3186,4895,3250],{"class":3192},[3186,4897,4898,4901,4903,4906],{"class":3188,"line":3448},[3186,4899,4900],{"class":3236},"        flyway",[3186,4902,3259],{"class":3192},[3186,4904,4905],{"class":3210},"migrate",[3186,4907,3250],{"class":3192},[3186,4909,4910],{"class":3188,"line":3453},[3186,4911,4169],{"class":3192},[3186,4913,4914,4916],{"class":3188,"line":3459},[3186,4915,4523],{"class":3243},[3186,4917,4918],{"class":3192}," dataSource;\n",[3186,4920,4921],{"class":3188,"line":3478},[3186,4922,4356],{"class":3192},[3186,4924,4925],{"class":3188,"line":3483},[3186,4926,3330],{"class":3192},[3186,4928,4929,4932,4935,4938,4940,4943,4946],{"class":3188,"line":3489},[3186,4930,4931],{"class":3203},"    private",[3186,4933,4934],{"class":3196}," Properties",[3186,4936,4937],{"class":3210}," loadProperties",[3186,4939,3214],{"class":3192},[3186,4941,4942],{"class":3196},"String",[3186,4944,4945],{"class":3236}," filename",[3186,4947,4948],{"class":3192},") {\n",[3186,4950,4951],{"class":3188,"line":3524},[3186,4952,4953],{"class":3226},"        \u002F\u002F Реалізація завантаження properties\n",[3186,4955,4956],{"class":3188,"line":3545},[3186,4957,4958],{"class":3226},"        \u002F\u002F ...\n",[3186,4960,4961],{"class":3188,"line":3550},[3186,4962,4356],{"class":3192},[3186,4964,4965],{"class":3188,"line":3556},[3186,4966,3661],{"class":3192},[3150,4968,4969],{},"Використання у Module:",[3161,4971,4973],{"className":3180,"code":4972,"language":3182,"meta":3170,"style":3170},"@Override\nprotected void configure() {\n    bind(DataSource.class)\n        .toProvider(HikariDataSourceProvider.class)\n        .in(Singleton.class);\n}\n",[3168,4974,4975,4981,4992,5007,5026,5042],{"__ignoreMap":3170},[3186,4976,4977,4979],{"class":3188,"line":3189},[3186,4978,3193],{"class":3192},[3186,4980,3197],{"class":3196},[3186,4982,4983,4986,4988,4990],{"class":3188,"line":3200},[3186,4984,4985],{"class":3203},"protected",[3186,4987,3207],{"class":3196},[3186,4989,4156],{"class":3210},[3186,4991,4159],{"class":3192},[3186,4993,4994,4997,4999,5001,5003,5005],{"class":3188,"line":3223},[3186,4995,4996],{"class":3210},"    bind",[3186,4998,3214],{"class":3192},[3186,5000,4638],{"class":3236},[3186,5002,3259],{"class":3192},[3186,5004,4189],{"class":3236},[3186,5006,4887],{"class":3192},[3186,5008,5009,5012,5015,5017,5020,5022,5024],{"class":3188,"line":3230},[3186,5010,5011],{"class":3192},"        .",[3186,5013,5014],{"class":3210},"toProvider",[3186,5016,3214],{"class":3192},[3186,5018,5019],{"class":3236},"HikariDataSourceProvider",[3186,5021,3259],{"class":3192},[3186,5023,4189],{"class":3236},[3186,5025,4887],{"class":3192},[3186,5027,5028,5030,5032,5034,5036,5038,5040],{"class":3188,"line":3253},[3186,5029,5011],{"class":3192},[3186,5031,4296],{"class":3210},[3186,5033,3214],{"class":3192},[3186,5035,4301],{"class":3236},[3186,5037,3259],{"class":3192},[3186,5039,4189],{"class":3236},[3186,5041,3271],{"class":3192},[3186,5043,5044],{"class":3188,"line":3274},[3186,5045,3661],{"class":3192},[3981,5047,5048,5051,5052,5054],{},[3154,5049,5050],{},"Provider Method vs Provider Class:"," Використовуйте ",[3168,5053,4590],{}," метод для простої логіки (2-10 рядків). Використовуйте Provider Class для складної логіки (читання конфігурації, міграції, валідація).",[3729,5056],{},[3145,5058,5060],{"id":5059},"controllerfactory-міст-між-fxmlloader-та-guice","ControllerFactory: Міст між FXMLLoader та Guice",[3150,5062,5063,5065,5066,5069],{},[3168,5064,3796],{}," за замовчуванням створює Controllers через рефлексію: ",[3168,5067,5068],{},"controllerClass.getDeclaredConstructor().newInstance()",". Це працює лише для конструкторів без параметрів. Але наші Controllers мають залежності (ViewModel), тому потрібен конструктор з параметрами.",[3739,5071,5073],{"id":5072},"проблема-fxmlloader-не-знає-про-guice","Проблема: FXMLLoader не знає про Guice",[3150,5075,5076],{},"Розглянемо Controller з Constructor Injection:",[3161,5078,5080],{"className":3180,"code":5079,"language":3182,"meta":3170,"style":3170},"public class AudiobookListController {\n    private final AudiobookListViewModel viewModel;\n    \n    @Inject\n    public AudiobookListController(AudiobookListViewModel viewModel) {\n        this.viewModel = viewModel;\n    }\n}\n",[3168,5081,5082,5093,5107,5111,5118,5132,5145,5149],{"__ignoreMap":3170},[3186,5083,5084,5086,5088,5091],{"class":3188,"line":3189},[3186,5085,3204],{"class":3203},[3186,5087,4124],{"class":3203},[3186,5089,5090],{"class":3196}," AudiobookListController",[3186,5092,4135],{"class":3192},[3186,5094,5095,5097,5100,5102,5104],{"class":3188,"line":3200},[3186,5096,4931],{"class":3203},[3186,5098,5099],{"class":3203}," final",[3186,5101,3472],{"class":3196},[3186,5103,3465],{"class":3236},[3186,5105,5106],{"class":3192},";\n",[3186,5108,5109],{"class":3188,"line":3223},[3186,5110,3330],{"class":3192},[3186,5112,5113,5115],{"class":3188,"line":3230},[3186,5114,4144],{"class":3192},[3186,5116,5117],{"class":3196},"Inject\n",[3186,5119,5120,5122,5124,5126,5128,5130],{"class":3188,"line":3253},[3186,5121,4656],{"class":3203},[3186,5123,5090],{"class":3210},[3186,5125,3214],{"class":3192},[3186,5127,3705],{"class":3196},[3186,5129,3465],{"class":3236},[3186,5131,4948],{"class":3192},[3186,5133,5134,5137,5139,5142],{"class":3188,"line":3274},[3186,5135,5136],{"class":3203},"        this",[3186,5138,3259],{"class":3192},[3186,5140,5141],{"class":3236},"viewModel",[3186,5143,5144],{"class":3192}," = viewModel;\n",[3186,5146,5147],{"class":3188,"line":3291},[3186,5148,4356],{"class":3192},[3186,5150,5151],{"class":3188,"line":3308},[3186,5152,3661],{"class":3192},[3150,5154,5155],{},"Якщо завантажити FXML без ControllerFactory:",[3161,5157,5159],{"className":3180,"code":5158,"language":3182,"meta":3170,"style":3170},"FXMLLoader loader = new FXMLLoader(getClass().getResource(\"AudiobookListView.fxml\"));\nParent root = loader.load(); \u002F\u002F Викине виняток!\n",[3168,5160,5161,5188],{"__ignoreMap":3170},[3186,5162,5163,5165,5167,5169,5171,5173,5175,5177,5179,5181,5183,5186],{"class":3188,"line":3189},[3186,5164,3796],{"class":3196},[3186,5166,3495],{"class":3236},[3186,5168,3240],{"class":3192},[3186,5170,3244],{"class":3243},[3186,5172,3502],{"class":3210},[3186,5174,3214],{"class":3192},[3186,5176,3507],{"class":3210},[3186,5178,3510],{"class":3192},[3186,5180,3513],{"class":3210},[3186,5182,3214],{"class":3192},[3186,5184,5185],{"class":3267},"\"AudiobookListView.fxml\"",[3186,5187,3521],{"class":3192},[3186,5189,5190,5193,5195,5197,5199,5201,5203,5206],{"class":3188,"line":3200},[3186,5191,5192],{"class":3196},"Parent",[3186,5194,3530],{"class":3236},[3186,5196,3240],{"class":3192},[3186,5198,3535],{"class":3236},[3186,5200,3259],{"class":3192},[3186,5202,3540],{"class":3210},[3186,5204,5205],{"class":3192},"(); ",[3186,5207,5208],{"class":3226},"\u002F\u002F Викине виняток!\n",[3150,5210,5211,5213,5214,5217,5218,3259],{},[3168,5212,3796],{}," спробує викликати ",[3168,5215,5216],{},"new AudiobookListController()",", але такого конструктора немає → ",[3168,5219,5220],{},"InstantiationException",[3739,5222,5224],{"id":5223},"рішення-setcontrollerfactory","Рішення: setControllerFactory()",[3150,5226,5227,5229],{},[3168,5228,3796],{}," підтримує кастомну фабрику для створення Controllers:",[3161,5231,5233],{"className":3180,"code":5232,"language":3182,"meta":3170,"style":3170},"public interface Callback\u003CP, R> {\n    R call(P param);\n}\n",[3168,5234,5235,5257,5274],{"__ignoreMap":3170},[3186,5236,5237,5239,5242,5245,5247,5250,5252,5255],{"class":3188,"line":3189},[3186,5238,3204],{"class":3203},[3186,5240,5241],{"class":3203}," interface",[3186,5243,5244],{"class":3196}," Callback",[3186,5246,4635],{"class":3192},[3186,5248,5249],{"class":3196},"P",[3186,5251,3624],{"class":3192},[3186,5253,5254],{"class":3196},"R",[3186,5256,4641],{"class":3192},[3186,5258,5259,5262,5265,5267,5269,5272],{"class":3188,"line":3200},[3186,5260,5261],{"class":3196},"    R",[3186,5263,5264],{"class":3210}," call",[3186,5266,3214],{"class":3192},[3186,5268,5249],{"class":3196},[3186,5270,5271],{"class":3236}," param",[3186,5273,3271],{"class":3192},[3186,5275,5276],{"class":3188,"line":3223},[3186,5277,3661],{"class":3192},[3150,5279,5280,5283],{},[3168,5281,5282],{},"FXMLLoader.setControllerFactory(Callback\u003CClass\u003C?>, Object>)"," приймає функцію, що отримує клас Controller та повертає його екземпляр. Ми можемо передати туди Guice Injector:",[3161,5285,5287],{"className":3180,"code":5286,"language":3182,"meta":3170,"style":3170},"FXMLLoader loader = new FXMLLoader(getClass().getResource(\"AudiobookListView.fxml\"));\nloader.setControllerFactory(injector::getInstance);\nParent root = loader.load(); \u002F\u002F Працює!\n",[3168,5288,5289,5315,5333],{"__ignoreMap":3170},[3186,5290,5291,5293,5295,5297,5299,5301,5303,5305,5307,5309,5311,5313],{"class":3188,"line":3189},[3186,5292,3796],{"class":3196},[3186,5294,3495],{"class":3236},[3186,5296,3240],{"class":3192},[3186,5298,3244],{"class":3243},[3186,5300,3502],{"class":3210},[3186,5302,3214],{"class":3192},[3186,5304,3507],{"class":3210},[3186,5306,3510],{"class":3192},[3186,5308,3513],{"class":3210},[3186,5310,3214],{"class":3192},[3186,5312,5185],{"class":3267},[3186,5314,3521],{"class":3192},[3186,5316,5317,5319,5321,5324,5327,5330],{"class":3188,"line":3200},[3186,5318,3535],{"class":3236},[3186,5320,3259],{"class":3192},[3186,5322,5323],{"class":3210},"setControllerFactory",[3186,5325,5326],{"class":3192},"(injector",[3186,5328,5329],{"class":3243},"::",[3186,5331,5332],{"class":3192},"getInstance);\n",[3186,5334,5335,5337,5339,5341,5343,5345,5347,5349],{"class":3188,"line":3223},[3186,5336,5192],{"class":3196},[3186,5338,3530],{"class":3236},[3186,5340,3240],{"class":3192},[3186,5342,3535],{"class":3236},[3186,5344,3259],{"class":3192},[3186,5346,3540],{"class":3210},[3186,5348,5205],{"class":3192},[3186,5350,5351],{"class":3226},"\u002F\u002F Працює!\n",[3150,5353,5354,5356,5357,3259],{},[3168,5355,3848],{}," — це method reference, еквівалент лямбди ",[3168,5358,5359],{},"clazz -> injector.getInstance(clazz)",[3739,5361,5363],{"id":5362},"повний-приклад-завантаження-fxml-з-guice","Повний приклад: Завантаження FXML з Guice",[3161,5365,5367],{"className":3180,"code":5366,"language":3182,"meta":3170,"style":3170},"public class FxmlLoaderHelper {\n    private final Injector injector;\n    \n    @Inject\n    public FxmlLoaderHelper(Injector injector) {\n        this.injector = injector;\n    }\n    \n    public Parent load(String fxmlPath) throws IOException {\n        FXMLLoader loader = new FXMLLoader(getClass().getResource(fxmlPath));\n        loader.setControllerFactory(injector::getInstance);\n        return loader.load();\n    }\n    \n    public \u003CT> T loadWithController(String fxmlPath) throws IOException {\n        FXMLLoader loader = new FXMLLoader(getClass().getResource(fxmlPath));\n        loader.setControllerFactory(injector::getInstance);\n        loader.load();\n        return loader.getController();\n    }\n}\n",[3168,5368,5369,5380,5393,5397,5403,5417,5429,5433,5437,5465,5489,5504,5516,5520,5524,5556,5578,5592,5602,5614,5618],{"__ignoreMap":3170},[3186,5370,5371,5373,5375,5378],{"class":3188,"line":3189},[3186,5372,3204],{"class":3203},[3186,5374,4124],{"class":3203},[3186,5376,5377],{"class":3196}," FxmlLoaderHelper",[3186,5379,4135],{"class":3192},[3186,5381,5382,5384,5386,5389,5391],{"class":3188,"line":3200},[3186,5383,4931],{"class":3203},[3186,5385,5099],{"class":3203},[3186,5387,5388],{"class":3196}," Injector",[3186,5390,3761],{"class":3236},[3186,5392,5106],{"class":3192},[3186,5394,5395],{"class":3188,"line":3223},[3186,5396,3330],{"class":3192},[3186,5398,5399,5401],{"class":3188,"line":3230},[3186,5400,4144],{"class":3192},[3186,5402,5117],{"class":3196},[3186,5404,5405,5407,5409,5411,5413,5415],{"class":3188,"line":3253},[3186,5406,4656],{"class":3203},[3186,5408,5377],{"class":3210},[3186,5410,3214],{"class":3192},[3186,5412,3758],{"class":3196},[3186,5414,3761],{"class":3236},[3186,5416,4948],{"class":3192},[3186,5418,5419,5421,5423,5426],{"class":3188,"line":3274},[3186,5420,5136],{"class":3203},[3186,5422,3259],{"class":3192},[3186,5424,5425],{"class":3236},"injector",[3186,5427,5428],{"class":3192}," = injector;\n",[3186,5430,5431],{"class":3188,"line":3291},[3186,5432,4356],{"class":3192},[3186,5434,5435],{"class":3188,"line":3308},[3186,5436,3330],{"class":3192},[3186,5438,5439,5441,5444,5447,5449,5451,5454,5457,5460,5463],{"class":3188,"line":3327},[3186,5440,4656],{"class":3203},[3186,5442,5443],{"class":3196}," Parent",[3186,5445,5446],{"class":3210}," load",[3186,5448,3214],{"class":3192},[3186,5450,4942],{"class":3196},[3186,5452,5453],{"class":3236}," fxmlPath",[3186,5455,5456],{"class":3192},") ",[3186,5458,5459],{"class":3203},"throws",[3186,5461,5462],{"class":3196}," IOException",[3186,5464,4135],{"class":3192},[3186,5466,5467,5470,5472,5474,5476,5478,5480,5482,5484,5486],{"class":3188,"line":3333},[3186,5468,5469],{"class":3196},"        FXMLLoader",[3186,5471,3495],{"class":3236},[3186,5473,3240],{"class":3192},[3186,5475,3244],{"class":3243},[3186,5477,3502],{"class":3210},[3186,5479,3214],{"class":3192},[3186,5481,3507],{"class":3210},[3186,5483,3510],{"class":3192},[3186,5485,3513],{"class":3210},[3186,5487,5488],{"class":3192},"(fxmlPath));\n",[3186,5490,5491,5494,5496,5498,5500,5502],{"class":3188,"line":3339},[3186,5492,5493],{"class":3236},"        loader",[3186,5495,3259],{"class":3192},[3186,5497,5323],{"class":3210},[3186,5499,5326],{"class":3192},[3186,5501,5329],{"class":3243},[3186,5503,5332],{"class":3192},[3186,5505,5506,5508,5510,5512,5514],{"class":3188,"line":3358},[3186,5507,4523],{"class":3243},[3186,5509,3495],{"class":3236},[3186,5511,3259],{"class":3192},[3186,5513,3540],{"class":3210},[3186,5515,3250],{"class":3192},[3186,5517,5518],{"class":3188,"line":3376},[3186,5519,4356],{"class":3192},[3186,5521,5522],{"class":3188,"line":3394},[3186,5523,3330],{"class":3192},[3186,5525,5526,5528,5531,5534,5537,5539,5542,5544,5546,5548,5550,5552,5554],{"class":3188,"line":3399},[3186,5527,4656],{"class":3203},[3186,5529,5530],{"class":3192}," \u003C",[3186,5532,5533],{"class":3196},"T",[3186,5535,5536],{"class":3192},"> ",[3186,5538,5533],{"class":3196},[3186,5540,5541],{"class":3210}," loadWithController",[3186,5543,3214],{"class":3192},[3186,5545,4942],{"class":3196},[3186,5547,5453],{"class":3236},[3186,5549,5456],{"class":3192},[3186,5551,5459],{"class":3203},[3186,5553,5462],{"class":3196},[3186,5555,4135],{"class":3192},[3186,5557,5558,5560,5562,5564,5566,5568,5570,5572,5574,5576],{"class":3188,"line":3405},[3186,5559,5469],{"class":3196},[3186,5561,3495],{"class":3236},[3186,5563,3240],{"class":3192},[3186,5565,3244],{"class":3243},[3186,5567,3502],{"class":3210},[3186,5569,3214],{"class":3192},[3186,5571,3507],{"class":3210},[3186,5573,3510],{"class":3192},[3186,5575,3513],{"class":3210},[3186,5577,5488],{"class":3192},[3186,5579,5580,5582,5584,5586,5588,5590],{"class":3188,"line":3424},[3186,5581,5493],{"class":3236},[3186,5583,3259],{"class":3192},[3186,5585,5323],{"class":3210},[3186,5587,5326],{"class":3192},[3186,5589,5329],{"class":3243},[3186,5591,5332],{"class":3192},[3186,5593,5594,5596,5598,5600],{"class":3188,"line":3430},[3186,5595,5493],{"class":3236},[3186,5597,3259],{"class":3192},[3186,5599,3540],{"class":3210},[3186,5601,3250],{"class":3192},[3186,5603,5604,5606,5608,5610,5612],{"class":3188,"line":3436},[3186,5605,4523],{"class":3243},[3186,5607,3495],{"class":3236},[3186,5609,3259],{"class":3192},[3186,5611,3571],{"class":3210},[3186,5613,3250],{"class":3192},[3186,5615,5616],{"class":3188,"line":3442},[3186,5617,4356],{"class":3192},[3186,5619,5620],{"class":3188,"line":3448},[3186,5621,3661],{"class":3192},[3150,5623,5624],{},"Використання:",[3161,5626,5628],{"className":3180,"code":5627,"language":3182,"meta":3170,"style":3170},"@Inject\nprivate FxmlLoaderHelper loaderHelper;\n\npublic void showAudiobookList() throws IOException {\n    Parent root = loaderHelper.load(\"\u002Ffxml\u002Faudiobook-list-view.fxml\");\n    scene.setRoot(root);\n}\n",[3168,5629,5630,5636,5648,5652,5664,5686,5699],{"__ignoreMap":3170},[3186,5631,5632,5634],{"class":3188,"line":3189},[3186,5633,3193],{"class":3192},[3186,5635,5117],{"class":3196},[3186,5637,5638,5641,5643,5646],{"class":3188,"line":3200},[3186,5639,5640],{"class":3203},"private",[3186,5642,5377],{"class":3196},[3186,5644,5645],{"class":3236}," loaderHelper",[3186,5647,5106],{"class":3192},[3186,5649,5650],{"class":3188,"line":3223},[3186,5651,4024],{"emptyLinePlaceholder":4023},[3186,5653,5654,5656,5658,5661],{"class":3188,"line":3230},[3186,5655,3204],{"class":3203},[3186,5657,3207],{"class":3196},[3186,5659,5660],{"class":3210}," showAudiobookList",[3186,5662,5663],{"class":3192},"() throws IOException {\n",[3186,5665,5666,5668,5670,5672,5675,5677,5679,5681,5684],{"class":3188,"line":3253},[3186,5667,3527],{"class":3196},[3186,5669,3530],{"class":3236},[3186,5671,3240],{"class":3192},[3186,5673,5674],{"class":3236},"loaderHelper",[3186,5676,3259],{"class":3192},[3186,5678,3540],{"class":3210},[3186,5680,3214],{"class":3192},[3186,5682,5683],{"class":3267},"\"\u002Ffxml\u002Faudiobook-list-view.fxml\"",[3186,5685,3271],{"class":3192},[3186,5687,5688,5691,5693,5696],{"class":3188,"line":3274},[3186,5689,5690],{"class":3236},"    scene",[3186,5692,3259],{"class":3192},[3186,5694,5695],{"class":3210},"setRoot",[3186,5697,5698],{"class":3192},"(root);\n",[3186,5700,5701],{"class":3188,"line":3291},[3186,5702,3661],{"class":3192},[5704,5705,5706,5709,5710,3624,5713,3624,5716,5719],"warning",{},[3154,5707,5708],{},"Не створюйте FXMLLoader вручну у кожному місці."," Винесіть логіку завантаження у окремий клас (",[3168,5711,5712],{},"FxmlLoaderHelper",[3168,5714,5715],{},"ViewLoader",[3168,5717,5718],{},"Navigator",") та ін'єктуйте Injector туди. Це дозволить централізовано керувати завантаженням Views.",[3729,5721],{},[3145,5723,5725],{"id":5724},"constructor-injection-інєкція-viewmodel-у-controller","Constructor Injection: Ін'єкція ViewModel у Controller",[3150,5727,5728,5729,5731],{},"Тепер, коли Guice інтегрований з ",[3168,5730,3796],{},", можемо використовувати Constructor Injection у Controllers.",[3739,5733,5735],{"id":5734},"controller-з-constructor-injection","Controller з Constructor Injection",[3161,5737,5739],{"className":3180,"code":5738,"language":3182,"meta":3170,"style":3170},"package dev.kostyl.audiobook.controller;\n\nimport com.google.inject.Inject;\nimport dev.kostyl.audiobook.viewmodel.AudiobookListViewModel;\nimport dev.kostyl.audiobook.viewmodel.AudiobookViewModel;\nimport javafx.fxml.FXML;\nimport javafx.scene.control.*;\n\npublic class AudiobookListController {\n    \n    \u002F\u002F ===== FXML-ін'єкція UI-елементів =====\n    \n    @FXML private TableView\u003CAudiobookViewModel> audiobookTable;\n    @FXML private TableColumn\u003CAudiobookViewModel, String> titleColumn;\n    @FXML private TableColumn\u003CAudiobookViewModel, String> authorColumn;\n    @FXML private TableColumn\u003CAudiobookViewModel, String> genreColumn;\n    @FXML private TableColumn\u003CAudiobookViewModel, String> durationColumn;\n    \n    @FXML private TextField searchField;\n    @FXML private ComboBox\u003CString> genreComboBox;\n    \n    @FXML private Button addButton;\n    @FXML private Button editButton;\n    @FXML private Button deleteButton;\n    \n    @FXML private ProgressIndicator loadingIndicator;\n    @FXML private Label statusLabel;\n    @FXML private Label countLabel;\n    \n    \u002F\u002F ===== Guice-ін'єкція залежностей =====\n    \n    private final AudiobookListViewModel viewModel;\n    \n    @Inject\n    public AudiobookListController(AudiobookListViewModel viewModel) {\n        this.viewModel = viewModel;\n    }\n    \n    \u002F\u002F ===== Lifecycle =====\n    \n    @FXML\n    public void initialize() {\n        setupTableColumns();\n        setupBindings();\n        setupEventHandlers();\n        \n        \u002F\u002F Завантаження даних\n        viewModel.initialize();\n    }\n    \n    \u002F\u002F ===== Налаштування =====\n    \n    private void setupTableColumns() {\n        titleColumn.setCellValueFactory(cellData -> \n            cellData.getValue().titleProperty());\n        authorColumn.setCellValueFactory(cellData -> \n            cellData.getValue().authorNameProperty());\n        genreColumn.setCellValueFactory(cellData -> \n            cellData.getValue().genreNameProperty());\n        durationColumn.setCellValueFactory(cellData -> \n            cellData.getValue().formattedDurationProperty());\n    }\n    \n    private void setupBindings() {\n        \u002F\u002F Таблиця\n        audiobookTable.setItems(viewModel.getSortedAudiobooks());\n        audiobookTable.getSelectionModel().selectedItemProperty()\n            .bindBidirectional(viewModel.selectedAudiobookProperty());\n        \n        \u002F\u002F Фільтри\n        searchField.textProperty().bindBidirectional(viewModel.searchQueryProperty());\n        genreComboBox.valueProperty().bindBidirectional(viewModel.selectedGenreProperty());\n        \n        \u002F\u002F Індикатори\n        loadingIndicator.visibleProperty().bind(viewModel.isLoadingProperty());\n        statusLabel.textProperty().bind(viewModel.statusMessageProperty());\n        \n        \u002F\u002F Кнопки\n        editButton.disableProperty().bind(viewModel.selectedAudiobookProperty().isNull());\n        deleteButton.disableProperty().bind(viewModel.selectedAudiobookProperty().isNull());\n    }\n    \n    private void setupEventHandlers() {\n        addButton.setOnAction(e -> onAddClicked());\n        editButton.setOnAction(e -> onEditClicked());\n        deleteButton.setOnAction(e -> onDeleteClicked());\n    }\n    \n    \u002F\u002F ===== Event Handlers =====\n    \n    @FXML\n    private void onAddClicked() {\n        viewModel.addNew();\n    }\n    \n    @FXML\n    private void onEditClicked() {\n        viewModel.editSelected();\n    }\n    \n    @FXML\n    private void onDeleteClicked() {\n        viewModel.deleteSelected();\n    }\n}\n",[3168,5740,5741,5748,5752,5759,5765,5772,5779,5786,5790,5800,5804,5809,5813,5838,5864,5889,5914,5939,5943,5959,5981,5985,6001,6016,6031,6035,6051,6067,6082,6086,6091,6095,6107,6111,6117,6131,6141,6145,6149,6154,6158,6165,6176,6183,6190,6197,6201,6206,6218,6222,6226,6231,6235,6246,6265,6283,6299,6315,6331,6347,6363,6379,6384,6389,6401,6407,6429,6446,6465,6470,6476,6502,6528,6533,6539,6566,6591,6596,6602,6632,6660,6665,6670,6682,6703,6721,6739,6744,6749,6755,6760,6767,6778,6790,6795,6800,6807,6818,6830,6835,6840,6847,6858,6870,6875],{"__ignoreMap":3170},[3186,5742,5743,5745],{"class":3188,"line":3189},[3186,5744,4015],{"class":3203},[3186,5746,5747],{"class":3192}," dev.kostyl.audiobook.controller;\n",[3186,5749,5750],{"class":3188,"line":3200},[3186,5751,4024],{"emptyLinePlaceholder":4023},[3186,5753,5754,5756],{"class":3188,"line":3223},[3186,5755,4029],{"class":3203},[3186,5757,5758],{"class":3192}," com.google.inject.Inject;\n",[3186,5760,5761,5763],{"class":3188,"line":3230},[3186,5762,4029],{"class":3203},[3186,5764,4102],{"class":3192},[3186,5766,5767,5769],{"class":3188,"line":3253},[3186,5768,4029],{"class":3203},[3186,5770,5771],{"class":3192}," dev.kostyl.audiobook.viewmodel.AudiobookViewModel;\n",[3186,5773,5774,5776],{"class":3188,"line":3274},[3186,5775,4029],{"class":3203},[3186,5777,5778],{"class":3192}," javafx.fxml.FXML;\n",[3186,5780,5781,5783],{"class":3188,"line":3291},[3186,5782,4029],{"class":3203},[3186,5784,5785],{"class":3192}," javafx.scene.control.*;\n",[3186,5787,5788],{"class":3188,"line":3308},[3186,5789,4024],{"emptyLinePlaceholder":4023},[3186,5791,5792,5794,5796,5798],{"class":3188,"line":3327},[3186,5793,3204],{"class":3203},[3186,5795,4124],{"class":3203},[3186,5797,5090],{"class":3196},[3186,5799,4135],{"class":3192},[3186,5801,5802],{"class":3188,"line":3333},[3186,5803,3330],{"class":3192},[3186,5805,5806],{"class":3188,"line":3339},[3186,5807,5808],{"class":3226},"    \u002F\u002F ===== FXML-ін'єкція UI-елементів =====\n",[3186,5810,5811],{"class":3188,"line":3358},[3186,5812,3330],{"class":3192},[3186,5814,5815,5817,5820,5823,5826,5828,5831,5833,5836],{"class":3188,"line":3376},[3186,5816,4144],{"class":3192},[3186,5818,5819],{"class":3196},"FXML",[3186,5821,5822],{"class":3203}," private",[3186,5824,5825],{"class":3196}," TableView",[3186,5827,4635],{"class":3192},[3186,5829,5830],{"class":3196},"AudiobookViewModel",[3186,5832,5536],{"class":3192},[3186,5834,5835],{"class":3236},"audiobookTable",[3186,5837,5106],{"class":3192},[3186,5839,5840,5842,5844,5846,5849,5851,5853,5855,5857,5859,5862],{"class":3188,"line":3394},[3186,5841,4144],{"class":3192},[3186,5843,5819],{"class":3196},[3186,5845,5822],{"class":3203},[3186,5847,5848],{"class":3196}," TableColumn",[3186,5850,4635],{"class":3192},[3186,5852,5830],{"class":3196},[3186,5854,3624],{"class":3192},[3186,5856,4942],{"class":3196},[3186,5858,5536],{"class":3192},[3186,5860,5861],{"class":3236},"titleColumn",[3186,5863,5106],{"class":3192},[3186,5865,5866,5868,5870,5872,5874,5876,5878,5880,5882,5884,5887],{"class":3188,"line":3399},[3186,5867,4144],{"class":3192},[3186,5869,5819],{"class":3196},[3186,5871,5822],{"class":3203},[3186,5873,5848],{"class":3196},[3186,5875,4635],{"class":3192},[3186,5877,5830],{"class":3196},[3186,5879,3624],{"class":3192},[3186,5881,4942],{"class":3196},[3186,5883,5536],{"class":3192},[3186,5885,5886],{"class":3236},"authorColumn",[3186,5888,5106],{"class":3192},[3186,5890,5891,5893,5895,5897,5899,5901,5903,5905,5907,5909,5912],{"class":3188,"line":3405},[3186,5892,4144],{"class":3192},[3186,5894,5819],{"class":3196},[3186,5896,5822],{"class":3203},[3186,5898,5848],{"class":3196},[3186,5900,4635],{"class":3192},[3186,5902,5830],{"class":3196},[3186,5904,3624],{"class":3192},[3186,5906,4942],{"class":3196},[3186,5908,5536],{"class":3192},[3186,5910,5911],{"class":3236},"genreColumn",[3186,5913,5106],{"class":3192},[3186,5915,5916,5918,5920,5922,5924,5926,5928,5930,5932,5934,5937],{"class":3188,"line":3424},[3186,5917,4144],{"class":3192},[3186,5919,5819],{"class":3196},[3186,5921,5822],{"class":3203},[3186,5923,5848],{"class":3196},[3186,5925,4635],{"class":3192},[3186,5927,5830],{"class":3196},[3186,5929,3624],{"class":3192},[3186,5931,4942],{"class":3196},[3186,5933,5536],{"class":3192},[3186,5935,5936],{"class":3236},"durationColumn",[3186,5938,5106],{"class":3192},[3186,5940,5941],{"class":3188,"line":3430},[3186,5942,3330],{"class":3192},[3186,5944,5945,5947,5949,5951,5954,5957],{"class":3188,"line":3436},[3186,5946,4144],{"class":3192},[3186,5948,5819],{"class":3196},[3186,5950,5822],{"class":3203},[3186,5952,5953],{"class":3196}," TextField",[3186,5955,5956],{"class":3236}," searchField",[3186,5958,5106],{"class":3192},[3186,5960,5961,5963,5965,5967,5970,5972,5974,5976,5979],{"class":3188,"line":3442},[3186,5962,4144],{"class":3192},[3186,5964,5819],{"class":3196},[3186,5966,5822],{"class":3203},[3186,5968,5969],{"class":3196}," ComboBox",[3186,5971,4635],{"class":3192},[3186,5973,4942],{"class":3196},[3186,5975,5536],{"class":3192},[3186,5977,5978],{"class":3236},"genreComboBox",[3186,5980,5106],{"class":3192},[3186,5982,5983],{"class":3188,"line":3448},[3186,5984,3330],{"class":3192},[3186,5986,5987,5989,5991,5993,5996,5999],{"class":3188,"line":3453},[3186,5988,4144],{"class":3192},[3186,5990,5819],{"class":3196},[3186,5992,5822],{"class":3203},[3186,5994,5995],{"class":3196}," Button",[3186,5997,5998],{"class":3236}," addButton",[3186,6000,5106],{"class":3192},[3186,6002,6003,6005,6007,6009,6011,6014],{"class":3188,"line":3459},[3186,6004,4144],{"class":3192},[3186,6006,5819],{"class":3196},[3186,6008,5822],{"class":3203},[3186,6010,5995],{"class":3196},[3186,6012,6013],{"class":3236}," editButton",[3186,6015,5106],{"class":3192},[3186,6017,6018,6020,6022,6024,6026,6029],{"class":3188,"line":3478},[3186,6019,4144],{"class":3192},[3186,6021,5819],{"class":3196},[3186,6023,5822],{"class":3203},[3186,6025,5995],{"class":3196},[3186,6027,6028],{"class":3236}," deleteButton",[3186,6030,5106],{"class":3192},[3186,6032,6033],{"class":3188,"line":3483},[3186,6034,3330],{"class":3192},[3186,6036,6037,6039,6041,6043,6046,6049],{"class":3188,"line":3489},[3186,6038,4144],{"class":3192},[3186,6040,5819],{"class":3196},[3186,6042,5822],{"class":3203},[3186,6044,6045],{"class":3196}," ProgressIndicator",[3186,6047,6048],{"class":3236}," loadingIndicator",[3186,6050,5106],{"class":3192},[3186,6052,6053,6055,6057,6059,6062,6065],{"class":3188,"line":3524},[3186,6054,4144],{"class":3192},[3186,6056,5819],{"class":3196},[3186,6058,5822],{"class":3203},[3186,6060,6061],{"class":3196}," Label",[3186,6063,6064],{"class":3236}," statusLabel",[3186,6066,5106],{"class":3192},[3186,6068,6069,6071,6073,6075,6077,6080],{"class":3188,"line":3545},[3186,6070,4144],{"class":3192},[3186,6072,5819],{"class":3196},[3186,6074,5822],{"class":3203},[3186,6076,6061],{"class":3196},[3186,6078,6079],{"class":3236}," countLabel",[3186,6081,5106],{"class":3192},[3186,6083,6084],{"class":3188,"line":3550},[3186,6085,3330],{"class":3192},[3186,6087,6088],{"class":3188,"line":3556},[3186,6089,6090],{"class":3226},"    \u002F\u002F ===== Guice-ін'єкція залежностей =====\n",[3186,6092,6093],{"class":3188,"line":3576},[3186,6094,3330],{"class":3192},[3186,6096,6097,6099,6101,6103,6105],{"class":3188,"line":3590},[3186,6098,4931],{"class":3203},[3186,6100,5099],{"class":3203},[3186,6102,3472],{"class":3196},[3186,6104,3465],{"class":3236},[3186,6106,5106],{"class":3192},[3186,6108,6109],{"class":3188,"line":3595},[3186,6110,3330],{"class":3192},[3186,6112,6113,6115],{"class":3188,"line":3601},[3186,6114,4144],{"class":3192},[3186,6116,5117],{"class":3196},[3186,6118,6119,6121,6123,6125,6127,6129],{"class":3188,"line":3632},[3186,6120,4656],{"class":3203},[3186,6122,5090],{"class":3210},[3186,6124,3214],{"class":3192},[3186,6126,3705],{"class":3196},[3186,6128,3465],{"class":3236},[3186,6130,4948],{"class":3192},[3186,6132,6133,6135,6137,6139],{"class":3188,"line":3646},[3186,6134,5136],{"class":3203},[3186,6136,3259],{"class":3192},[3186,6138,5141],{"class":3236},[3186,6140,5144],{"class":3192},[3186,6142,6143],{"class":3188,"line":3658},[3186,6144,4356],{"class":3192},[3186,6146,6147],{"class":3188,"line":4359},[3186,6148,3330],{"class":3192},[3186,6150,6151],{"class":3188,"line":4364},[3186,6152,6153],{"class":3226},"    \u002F\u002F ===== Lifecycle =====\n",[3186,6155,6156],{"class":3188,"line":4370},[3186,6157,3330],{"class":3192},[3186,6159,6160,6162],{"class":3188,"line":4375},[3186,6161,4144],{"class":3192},[3186,6163,6164],{"class":3196},"FXML\n",[3186,6166,6167,6169,6171,6174],{"class":3188,"line":4383},[3186,6168,4656],{"class":3203},[3186,6170,3207],{"class":3196},[3186,6172,6173],{"class":3210}," initialize",[3186,6175,4159],{"class":3192},[3186,6177,6178,6181],{"class":3188,"line":4391},[3186,6179,6180],{"class":3210},"        setupTableColumns",[3186,6182,3250],{"class":3192},[3186,6184,6185,6188],{"class":3188,"line":4401},[3186,6186,6187],{"class":3210},"        setupBindings",[3186,6189,3250],{"class":3192},[3186,6191,6192,6195],{"class":3188,"line":4417},[3186,6193,6194],{"class":3210},"        setupEventHandlers",[3186,6196,3250],{"class":3192},[3186,6198,6199],{"class":3188,"line":4434},[3186,6200,4169],{"class":3192},[3186,6202,6203],{"class":3188,"line":4449},[3186,6204,6205],{"class":3226},"        \u002F\u002F Завантаження даних\n",[3186,6207,6208,6211,6213,6216],{"class":3188,"line":4464},[3186,6209,6210],{"class":3236},"        viewModel",[3186,6212,3259],{"class":3192},[3186,6214,6215],{"class":3210},"initialize",[3186,6217,3250],{"class":3192},[3186,6219,6220],{"class":3188,"line":4481},[3186,6221,4356],{"class":3192},[3186,6223,6224],{"class":3188,"line":4498},[3186,6225,3330],{"class":3192},[3186,6227,6228],{"class":3188,"line":4515},[3186,6229,6230],{"class":3226},"    \u002F\u002F ===== Налаштування =====\n",[3186,6232,6233],{"class":3188,"line":4520},[3186,6234,3330],{"class":3192},[3186,6236,6237,6239,6241,6244],{"class":3188,"line":4533},[3186,6238,4931],{"class":3203},[3186,6240,3207],{"class":3196},[3186,6242,6243],{"class":3210}," setupTableColumns",[3186,6245,4159],{"class":3192},[3186,6247,6248,6251,6253,6256,6259,6262],{"class":3188,"line":4538},[3186,6249,6250],{"class":3236},"        titleColumn",[3186,6252,3259],{"class":3192},[3186,6254,6255],{"class":3210},"setCellValueFactory",[3186,6257,6258],{"class":3192},"(cellData ",[3186,6260,6261],{"class":3203},"->",[3186,6263,6264],{"class":3192}," \n",[3186,6266,6268,6271,6273,6276,6278,6281],{"class":3188,"line":6267},55,[3186,6269,6270],{"class":3236},"            cellData",[3186,6272,3259],{"class":3192},[3186,6274,6275],{"class":3210},"getValue",[3186,6277,3510],{"class":3192},[3186,6279,6280],{"class":3210},"titleProperty",[3186,6282,3781],{"class":3192},[3186,6284,6286,6289,6291,6293,6295,6297],{"class":3188,"line":6285},56,[3186,6287,6288],{"class":3236},"        authorColumn",[3186,6290,3259],{"class":3192},[3186,6292,6255],{"class":3210},[3186,6294,6258],{"class":3192},[3186,6296,6261],{"class":3203},[3186,6298,6264],{"class":3192},[3186,6300,6302,6304,6306,6308,6310,6313],{"class":3188,"line":6301},57,[3186,6303,6270],{"class":3236},[3186,6305,3259],{"class":3192},[3186,6307,6275],{"class":3210},[3186,6309,3510],{"class":3192},[3186,6311,6312],{"class":3210},"authorNameProperty",[3186,6314,3781],{"class":3192},[3186,6316,6318,6321,6323,6325,6327,6329],{"class":3188,"line":6317},58,[3186,6319,6320],{"class":3236},"        genreColumn",[3186,6322,3259],{"class":3192},[3186,6324,6255],{"class":3210},[3186,6326,6258],{"class":3192},[3186,6328,6261],{"class":3203},[3186,6330,6264],{"class":3192},[3186,6332,6334,6336,6338,6340,6342,6345],{"class":3188,"line":6333},59,[3186,6335,6270],{"class":3236},[3186,6337,3259],{"class":3192},[3186,6339,6275],{"class":3210},[3186,6341,3510],{"class":3192},[3186,6343,6344],{"class":3210},"genreNameProperty",[3186,6346,3781],{"class":3192},[3186,6348,6350,6353,6355,6357,6359,6361],{"class":3188,"line":6349},60,[3186,6351,6352],{"class":3236},"        durationColumn",[3186,6354,3259],{"class":3192},[3186,6356,6255],{"class":3210},[3186,6358,6258],{"class":3192},[3186,6360,6261],{"class":3203},[3186,6362,6264],{"class":3192},[3186,6364,6366,6368,6370,6372,6374,6377],{"class":3188,"line":6365},61,[3186,6367,6270],{"class":3236},[3186,6369,3259],{"class":3192},[3186,6371,6275],{"class":3210},[3186,6373,3510],{"class":3192},[3186,6375,6376],{"class":3210},"formattedDurationProperty",[3186,6378,3781],{"class":3192},[3186,6380,6382],{"class":3188,"line":6381},62,[3186,6383,4356],{"class":3192},[3186,6385,6387],{"class":3188,"line":6386},63,[3186,6388,3330],{"class":3192},[3186,6390,6392,6394,6396,6399],{"class":3188,"line":6391},64,[3186,6393,4931],{"class":3203},[3186,6395,3207],{"class":3196},[3186,6397,6398],{"class":3210}," setupBindings",[3186,6400,4159],{"class":3192},[3186,6402,6404],{"class":3188,"line":6403},65,[3186,6405,6406],{"class":3226},"        \u002F\u002F Таблиця\n",[3186,6408,6410,6413,6415,6418,6420,6422,6424,6427],{"class":3188,"line":6409},66,[3186,6411,6412],{"class":3236},"        audiobookTable",[3186,6414,3259],{"class":3192},[3186,6416,6417],{"class":3210},"setItems",[3186,6419,3214],{"class":3192},[3186,6421,5141],{"class":3236},[3186,6423,3259],{"class":3192},[3186,6425,6426],{"class":3210},"getSortedAudiobooks",[3186,6428,3781],{"class":3192},[3186,6430,6432,6434,6436,6439,6441,6444],{"class":3188,"line":6431},67,[3186,6433,6412],{"class":3236},[3186,6435,3259],{"class":3192},[3186,6437,6438],{"class":3210},"getSelectionModel",[3186,6440,3510],{"class":3192},[3186,6442,6443],{"class":3210},"selectedItemProperty",[3186,6445,4861],{"class":3192},[3186,6447,6449,6451,6454,6456,6458,6460,6463],{"class":3188,"line":6448},68,[3186,6450,4866],{"class":3192},[3186,6452,6453],{"class":3210},"bindBidirectional",[3186,6455,3214],{"class":3192},[3186,6457,5141],{"class":3236},[3186,6459,3259],{"class":3192},[3186,6461,6462],{"class":3210},"selectedAudiobookProperty",[3186,6464,3781],{"class":3192},[3186,6466,6468],{"class":3188,"line":6467},69,[3186,6469,4169],{"class":3192},[3186,6471,6473],{"class":3188,"line":6472},70,[3186,6474,6475],{"class":3226},"        \u002F\u002F Фільтри\n",[3186,6477,6479,6482,6484,6487,6489,6491,6493,6495,6497,6500],{"class":3188,"line":6478},71,[3186,6480,6481],{"class":3236},"        searchField",[3186,6483,3259],{"class":3192},[3186,6485,6486],{"class":3210},"textProperty",[3186,6488,3510],{"class":3192},[3186,6490,6453],{"class":3210},[3186,6492,3214],{"class":3192},[3186,6494,5141],{"class":3236},[3186,6496,3259],{"class":3192},[3186,6498,6499],{"class":3210},"searchQueryProperty",[3186,6501,3781],{"class":3192},[3186,6503,6505,6508,6510,6513,6515,6517,6519,6521,6523,6526],{"class":3188,"line":6504},72,[3186,6506,6507],{"class":3236},"        genreComboBox",[3186,6509,3259],{"class":3192},[3186,6511,6512],{"class":3210},"valueProperty",[3186,6514,3510],{"class":3192},[3186,6516,6453],{"class":3210},[3186,6518,3214],{"class":3192},[3186,6520,5141],{"class":3236},[3186,6522,3259],{"class":3192},[3186,6524,6525],{"class":3210},"selectedGenreProperty",[3186,6527,3781],{"class":3192},[3186,6529,6531],{"class":3188,"line":6530},73,[3186,6532,4169],{"class":3192},[3186,6534,6536],{"class":3188,"line":6535},74,[3186,6537,6538],{"class":3226},"        \u002F\u002F Індикатори\n",[3186,6540,6542,6545,6547,6550,6552,6555,6557,6559,6561,6564],{"class":3188,"line":6541},75,[3186,6543,6544],{"class":3236},"        loadingIndicator",[3186,6546,3259],{"class":3192},[3186,6548,6549],{"class":3210},"visibleProperty",[3186,6551,3510],{"class":3192},[3186,6553,6554],{"class":3210},"bind",[3186,6556,3214],{"class":3192},[3186,6558,5141],{"class":3236},[3186,6560,3259],{"class":3192},[3186,6562,6563],{"class":3210},"isLoadingProperty",[3186,6565,3781],{"class":3192},[3186,6567,6569,6572,6574,6576,6578,6580,6582,6584,6586,6589],{"class":3188,"line":6568},76,[3186,6570,6571],{"class":3236},"        statusLabel",[3186,6573,3259],{"class":3192},[3186,6575,6486],{"class":3210},[3186,6577,3510],{"class":3192},[3186,6579,6554],{"class":3210},[3186,6581,3214],{"class":3192},[3186,6583,5141],{"class":3236},[3186,6585,3259],{"class":3192},[3186,6587,6588],{"class":3210},"statusMessageProperty",[3186,6590,3781],{"class":3192},[3186,6592,6594],{"class":3188,"line":6593},77,[3186,6595,4169],{"class":3192},[3186,6597,6599],{"class":3188,"line":6598},78,[3186,6600,6601],{"class":3226},"        \u002F\u002F Кнопки\n",[3186,6603,6605,6608,6610,6613,6615,6617,6619,6621,6623,6625,6627,6630],{"class":3188,"line":6604},79,[3186,6606,6607],{"class":3236},"        editButton",[3186,6609,3259],{"class":3192},[3186,6611,6612],{"class":3210},"disableProperty",[3186,6614,3510],{"class":3192},[3186,6616,6554],{"class":3210},[3186,6618,3214],{"class":3192},[3186,6620,5141],{"class":3236},[3186,6622,3259],{"class":3192},[3186,6624,6462],{"class":3210},[3186,6626,3510],{"class":3192},[3186,6628,6629],{"class":3210},"isNull",[3186,6631,3781],{"class":3192},[3186,6633,6635,6638,6640,6642,6644,6646,6648,6650,6652,6654,6656,6658],{"class":3188,"line":6634},80,[3186,6636,6637],{"class":3236},"        deleteButton",[3186,6639,3259],{"class":3192},[3186,6641,6612],{"class":3210},[3186,6643,3510],{"class":3192},[3186,6645,6554],{"class":3210},[3186,6647,3214],{"class":3192},[3186,6649,5141],{"class":3236},[3186,6651,3259],{"class":3192},[3186,6653,6462],{"class":3210},[3186,6655,3510],{"class":3192},[3186,6657,6629],{"class":3210},[3186,6659,3781],{"class":3192},[3186,6661,6663],{"class":3188,"line":6662},81,[3186,6664,4356],{"class":3192},[3186,6666,6668],{"class":3188,"line":6667},82,[3186,6669,3330],{"class":3192},[3186,6671,6673,6675,6677,6680],{"class":3188,"line":6672},83,[3186,6674,4931],{"class":3203},[3186,6676,3207],{"class":3196},[3186,6678,6679],{"class":3210}," setupEventHandlers",[3186,6681,4159],{"class":3192},[3186,6683,6685,6688,6690,6693,6696,6698,6701],{"class":3188,"line":6684},84,[3186,6686,6687],{"class":3236},"        addButton",[3186,6689,3259],{"class":3192},[3186,6691,6692],{"class":3210},"setOnAction",[3186,6694,6695],{"class":3192},"(e ",[3186,6697,6261],{"class":3203},[3186,6699,6700],{"class":3210}," onAddClicked",[3186,6702,3781],{"class":3192},[3186,6704,6706,6708,6710,6712,6714,6716,6719],{"class":3188,"line":6705},85,[3186,6707,6607],{"class":3236},[3186,6709,3259],{"class":3192},[3186,6711,6692],{"class":3210},[3186,6713,6695],{"class":3192},[3186,6715,6261],{"class":3203},[3186,6717,6718],{"class":3210}," onEditClicked",[3186,6720,3781],{"class":3192},[3186,6722,6724,6726,6728,6730,6732,6734,6737],{"class":3188,"line":6723},86,[3186,6725,6637],{"class":3236},[3186,6727,3259],{"class":3192},[3186,6729,6692],{"class":3210},[3186,6731,6695],{"class":3192},[3186,6733,6261],{"class":3203},[3186,6735,6736],{"class":3210}," onDeleteClicked",[3186,6738,3781],{"class":3192},[3186,6740,6742],{"class":3188,"line":6741},87,[3186,6743,4356],{"class":3192},[3186,6745,6747],{"class":3188,"line":6746},88,[3186,6748,3330],{"class":3192},[3186,6750,6752],{"class":3188,"line":6751},89,[3186,6753,6754],{"class":3226},"    \u002F\u002F ===== Event Handlers =====\n",[3186,6756,6758],{"class":3188,"line":6757},90,[3186,6759,3330],{"class":3192},[3186,6761,6763,6765],{"class":3188,"line":6762},91,[3186,6764,4144],{"class":3192},[3186,6766,6164],{"class":3196},[3186,6768,6770,6772,6774,6776],{"class":3188,"line":6769},92,[3186,6771,4931],{"class":3203},[3186,6773,3207],{"class":3196},[3186,6775,6700],{"class":3210},[3186,6777,4159],{"class":3192},[3186,6779,6781,6783,6785,6788],{"class":3188,"line":6780},93,[3186,6782,6210],{"class":3236},[3186,6784,3259],{"class":3192},[3186,6786,6787],{"class":3210},"addNew",[3186,6789,3250],{"class":3192},[3186,6791,6793],{"class":3188,"line":6792},94,[3186,6794,4356],{"class":3192},[3186,6796,6798],{"class":3188,"line":6797},95,[3186,6799,3330],{"class":3192},[3186,6801,6803,6805],{"class":3188,"line":6802},96,[3186,6804,4144],{"class":3192},[3186,6806,6164],{"class":3196},[3186,6808,6810,6812,6814,6816],{"class":3188,"line":6809},97,[3186,6811,4931],{"class":3203},[3186,6813,3207],{"class":3196},[3186,6815,6718],{"class":3210},[3186,6817,4159],{"class":3192},[3186,6819,6821,6823,6825,6828],{"class":3188,"line":6820},98,[3186,6822,6210],{"class":3236},[3186,6824,3259],{"class":3192},[3186,6826,6827],{"class":3210},"editSelected",[3186,6829,3250],{"class":3192},[3186,6831,6833],{"class":3188,"line":6832},99,[3186,6834,4356],{"class":3192},[3186,6836,6838],{"class":3188,"line":6837},100,[3186,6839,3330],{"class":3192},[3186,6841,6843,6845],{"class":3188,"line":6842},101,[3186,6844,4144],{"class":3192},[3186,6846,6164],{"class":3196},[3186,6848,6850,6852,6854,6856],{"class":3188,"line":6849},102,[3186,6851,4931],{"class":3203},[3186,6853,3207],{"class":3196},[3186,6855,6736],{"class":3210},[3186,6857,4159],{"class":3192},[3186,6859,6861,6863,6865,6868],{"class":3188,"line":6860},103,[3186,6862,6210],{"class":3236},[3186,6864,3259],{"class":3192},[3186,6866,6867],{"class":3210},"deleteSelected",[3186,6869,3250],{"class":3192},[3186,6871,6873],{"class":3188,"line":6872},104,[3186,6874,4356],{"class":3192},[3186,6876,6878],{"class":3188,"line":6877},105,[3186,6879,3661],{"class":3192},[3739,6881,6883],{"id":6882},"розбір-коду","Розбір коду",[3150,6885,6886,6889,6890,6892,6893,6896,6897,6900,6901,3259],{},[3154,6887,6888],{},"Рядки 13-28: @FXML поля."," Ці поля ін'єктуються ",[3168,6891,3796],{}," після парсингу FXML. Це відбувається ",[3154,6894,6895],{},"після"," виклику конструктора, але ",[3154,6898,6899],{},"до"," виклику ",[3168,6902,3979],{},[3150,6904,6905,6908,6909,6912,6913,6915,6916,6918,6919,6921,6922,6924,6925,6927,6928,3259],{},[3154,6906,6907],{},"Рядки 32-37: Constructor Injection."," Конструктор з ",[3168,6910,6911],{},"@Inject"," — це сигнал для Guice: \"створи ",[3168,6914,3705],{}," та передай його сюди\". Guice рекурсивно резолвить залежності: якщо ",[3168,6917,3705],{}," залежить від ",[3168,6920,3862],{},", Guice створить ",[3168,6923,3862],{}," (і його залежності), потім створить ",[3168,6926,3705],{},", потім створить ",[3168,6929,3855],{},[3150,6931,6932,6939,6940,6942,6943,6945],{},[3154,6933,6934,6935,6938],{},"Рядок 32: ",[3168,6936,6937],{},"final"," для залежностей."," Поле ",[3168,6941,5141],{}," оголошене як ",[3168,6944,6937],{}," — це гарантує, що воно буде ініціалізоване у конструкторі та не зміниться протягом життя Controller. Це immutability — важлива практика для безпеки та передбачуваності коду.",[3150,6947,6948,6953,6954,3693,6956,6958,6959,6961,6962,6964,6965,6967,6968,6971],{},[3154,6949,6950,6951,3259],{},"Рядки 41-48: ",[3168,6952,3979],{}," Цей метод викликається ",[3168,6955,3796],{},[3154,6957,6895],{}," ін'єкції ",[3168,6960,3975],{}," полів. На цей момент ",[3168,6963,5141],{}," вже ініціалізований (через конструктор), і всі UI-елементи вже ін'єктовані (через ",[3168,6966,3975],{},"). Тут ми налаштовуємо Bindings та викликаємо ",[3168,6969,6970],{},"viewModel.initialize()"," для завантаження даних.",[3150,6973,6974,6977,6978,3624,6981,3624,6984,6987],{},[3154,6975,6976],{},"Рядки 52-84: Налаштування."," Методи ",[3168,6979,6980],{},"setupTableColumns()",[3168,6982,6983],{},"setupBindings()",[3168,6985,6986],{},"setupEventHandlers()"," розбивають логіку ініціалізації на логічні блоки. Це покращує читабельність: замість одного методу на 100 рядків — три методи по 20-30 рядків.",[3722,6989,6990,6995,7017],{},[3150,6991,6992],{},[3154,6993,6994],{},"Порядок ініціалізації Controller:",[3882,6996,6997,7003,7010],{},[3885,6998,6999,7000,7002],{},"Guice викликає конструктор → ",[3168,7001,5141],{}," ініціалізований.",[3885,7004,7005,3972,7007,7009],{},[3168,7006,3796],{},[3168,7008,3975],{}," поля → UI-елементи ініціалізовані.",[3885,7011,7012,3931,7014,7016],{},[3168,7013,3796],{},[3168,7015,3979],{}," → Bindings налаштовані, дані завантажені.",[3150,7018,7019,7020,7022,7023,7026],{},"Якщо спробувати звернутися до ",[3168,7021,3975],{}," полів у конструкторі, отримаєте ",[3168,7024,7025],{},"NullPointerException"," — вони ще не ін'єктовані.",[3739,7028,7030],{"id":7029},"порівняння-setter-injection-vs-constructor-injection","Порівняння: Setter Injection vs Constructor Injection",[7032,7033,7034,7037,7165,7170,7191,7194,7302,7307],"tabs",{},[3150,7035,7036],{},"== Setter Injection (без Guice)",[3161,7038,7040],{"className":3180,"code":7039,"language":3182,"meta":3170,"style":3170},"public class AudiobookListController {\n    private AudiobookListViewModel viewModel;\n    \n    public void setViewModel(AudiobookListViewModel viewModel) {\n        this.viewModel = viewModel;\n        setupBindings();\n    }\n    \n    @FXML\n    public void initialize() {\n        \u002F\u002F viewModel може бути null!\n        if (viewModel != null) {\n            viewModel.initialize();\n        }\n    }\n}\n",[3168,7041,7042,7052,7062,7066,7083,7093,7099,7103,7107,7113,7123,7128,7141,7152,7157,7161],{"__ignoreMap":3170},[3186,7043,7044,7046,7048,7050],{"class":3188,"line":3189},[3186,7045,3204],{"class":3203},[3186,7047,4124],{"class":3203},[3186,7049,5090],{"class":3196},[3186,7051,4135],{"class":3192},[3186,7053,7054,7056,7058,7060],{"class":3188,"line":3200},[3186,7055,4931],{"class":3203},[3186,7057,3472],{"class":3196},[3186,7059,3465],{"class":3236},[3186,7061,5106],{"class":3192},[3186,7063,7064],{"class":3188,"line":3223},[3186,7065,3330],{"class":3192},[3186,7067,7068,7070,7072,7075,7077,7079,7081],{"class":3188,"line":3230},[3186,7069,4656],{"class":3203},[3186,7071,3207],{"class":3196},[3186,7073,7074],{"class":3210}," setViewModel",[3186,7076,3214],{"class":3192},[3186,7078,3705],{"class":3196},[3186,7080,3465],{"class":3236},[3186,7082,4948],{"class":3192},[3186,7084,7085,7087,7089,7091],{"class":3188,"line":3253},[3186,7086,5136],{"class":3203},[3186,7088,3259],{"class":3192},[3186,7090,5141],{"class":3236},[3186,7092,5144],{"class":3192},[3186,7094,7095,7097],{"class":3188,"line":3274},[3186,7096,6187],{"class":3210},[3186,7098,3250],{"class":3192},[3186,7100,7101],{"class":3188,"line":3291},[3186,7102,4356],{"class":3192},[3186,7104,7105],{"class":3188,"line":3308},[3186,7106,3330],{"class":3192},[3186,7108,7109,7111],{"class":3188,"line":3327},[3186,7110,4144],{"class":3192},[3186,7112,6164],{"class":3196},[3186,7114,7115,7117,7119,7121],{"class":3188,"line":3333},[3186,7116,4656],{"class":3203},[3186,7118,3207],{"class":3196},[3186,7120,6173],{"class":3210},[3186,7122,4159],{"class":3192},[3186,7124,7125],{"class":3188,"line":3339},[3186,7126,7127],{"class":3226},"        \u002F\u002F viewModel може бути null!\n",[3186,7129,7130,7133,7136,7139],{"class":3188,"line":3358},[3186,7131,7132],{"class":3243},"        if",[3186,7134,7135],{"class":3192}," (viewModel != ",[3186,7137,7138],{"class":3203},"null",[3186,7140,4948],{"class":3192},[3186,7142,7143,7146,7148,7150],{"class":3188,"line":3376},[3186,7144,7145],{"class":3236},"            viewModel",[3186,7147,3259],{"class":3192},[3186,7149,6215],{"class":3210},[3186,7151,3250],{"class":3192},[3186,7153,7154],{"class":3188,"line":3394},[3186,7155,7156],{"class":3192},"        }\n",[3186,7158,7159],{"class":3188,"line":3399},[3186,7160,4356],{"class":3192},[3186,7162,7163],{"class":3188,"line":3405},[3186,7164,3661],{"class":3192},[3150,7166,7167],{},[3154,7168,7169],{},"Недоліки:",[7171,7172,7173,7181,7188],"ul",{},[3885,7174,7175,7177,7178,7180],{},[3168,7176,5141],{}," може бути ",[3168,7179,7138],{}," → потрібні перевірки.",[3885,7182,7183,7184,7187],{},"Хтось має викликати ",[3168,7185,7186],{},"setViewModel()"," вручну після завантаження FXML.",[3885,7189,7190],{},"Неявна залежність: з сигнатури класу не зрозуміло, що потрібен ViewModel.",[3150,7192,7193],{},"== Constructor Injection (з Guice)",[3161,7195,7197],{"className":3180,"code":7196,"language":3182,"meta":3170,"style":3170},"public class AudiobookListController {\n    private final AudiobookListViewModel viewModel;\n    \n    @Inject\n    public AudiobookListController(AudiobookListViewModel viewModel) {\n        this.viewModel = viewModel;\n    }\n    \n    @FXML\n    public void initialize() {\n        \u002F\u002F viewModel гарантовано не null\n        viewModel.initialize();\n    }\n}\n",[3168,7198,7199,7209,7221,7225,7231,7245,7255,7259,7263,7269,7279,7284,7294,7298],{"__ignoreMap":3170},[3186,7200,7201,7203,7205,7207],{"class":3188,"line":3189},[3186,7202,3204],{"class":3203},[3186,7204,4124],{"class":3203},[3186,7206,5090],{"class":3196},[3186,7208,4135],{"class":3192},[3186,7210,7211,7213,7215,7217,7219],{"class":3188,"line":3200},[3186,7212,4931],{"class":3203},[3186,7214,5099],{"class":3203},[3186,7216,3472],{"class":3196},[3186,7218,3465],{"class":3236},[3186,7220,5106],{"class":3192},[3186,7222,7223],{"class":3188,"line":3223},[3186,7224,3330],{"class":3192},[3186,7226,7227,7229],{"class":3188,"line":3230},[3186,7228,4144],{"class":3192},[3186,7230,5117],{"class":3196},[3186,7232,7233,7235,7237,7239,7241,7243],{"class":3188,"line":3253},[3186,7234,4656],{"class":3203},[3186,7236,5090],{"class":3210},[3186,7238,3214],{"class":3192},[3186,7240,3705],{"class":3196},[3186,7242,3465],{"class":3236},[3186,7244,4948],{"class":3192},[3186,7246,7247,7249,7251,7253],{"class":3188,"line":3274},[3186,7248,5136],{"class":3203},[3186,7250,3259],{"class":3192},[3186,7252,5141],{"class":3236},[3186,7254,5144],{"class":3192},[3186,7256,7257],{"class":3188,"line":3291},[3186,7258,4356],{"class":3192},[3186,7260,7261],{"class":3188,"line":3308},[3186,7262,3330],{"class":3192},[3186,7264,7265,7267],{"class":3188,"line":3327},[3186,7266,4144],{"class":3192},[3186,7268,6164],{"class":3196},[3186,7270,7271,7273,7275,7277],{"class":3188,"line":3333},[3186,7272,4656],{"class":3203},[3186,7274,3207],{"class":3196},[3186,7276,6173],{"class":3210},[3186,7278,4159],{"class":3192},[3186,7280,7281],{"class":3188,"line":3339},[3186,7282,7283],{"class":3226},"        \u002F\u002F viewModel гарантовано не null\n",[3186,7285,7286,7288,7290,7292],{"class":3188,"line":3358},[3186,7287,6210],{"class":3236},[3186,7289,3259],{"class":3192},[3186,7291,6215],{"class":3210},[3186,7293,3250],{"class":3192},[3186,7295,7296],{"class":3188,"line":3376},[3186,7297,4356],{"class":3192},[3186,7299,7300],{"class":3188,"line":3394},[3186,7301,3661],{"class":3192},[3150,7303,7304],{},[3154,7305,7306],{},"Переваги:",[7171,7308,7309,7316,7319,7322],{},[3885,7310,7311,7313,7314,4192],{},[3168,7312,5141],{}," гарантовано ініціалізований (через ",[3168,7315,6937],{},[3885,7317,7318],{},"Явна залежність: з конструктора зрозуміло, що потрібен ViewModel.",[3885,7320,7321],{},"Автоматична ін'єкція через Guice — не потрібен ручний код.",[3885,7323,7324,7325,3259],{},"Легко тестувати: ",[3168,7326,7327],{},"new AudiobookListController(mockViewModel)",[3729,7329],{},[3145,7331,7333],{"id":7332},"scopes-singleton-vs-prototype","Scopes: Singleton vs Prototype",[3150,7335,7336],{},"Guice підтримує різні scopes (області видимості) для об'єктів. Два найважливіші для JavaFX-додатків:",[3739,7338,7340],{"id":7339},"singleton-scope","Singleton Scope",[3150,7342,7343,7346],{},[3154,7344,7345],{},"Визначення:"," Guice створює один екземпляр об'єкта при першому запиті та повертає його для всіх наступних запитів.",[3150,7348,7349],{},[3154,7350,7351],{},"Коли використовувати:",[7171,7353,7354,7360,7366],{},[3885,7355,7356,7359],{},[3154,7357,7358],{},"Stateless об'єкти:"," Repository, Service, Validator — вони не зберігають стан між викликами.",[3885,7361,7362,7365],{},[3154,7363,7364],{},"Дорогі ресурси:"," DataSource (Connection Pool), ExecutorService — створення нового екземпляра кожного разу — це витік ресурсів.",[3885,7367,7368,7371],{},[3154,7369,7370],{},"Глобальний стан:"," ViewModel для головних екранів, що зберігає стан між переходами (список аудіокниг, фільтри, вибір).",[3150,7373,7374],{},[3154,7375,7376],{},"Приклад:",[3161,7378,7380],{"className":3180,"code":7379,"language":3182,"meta":3170,"style":3170},"@Override\nprotected void configure() {\n    bind(AudiobookRepository.class).to(JdbcAudiobookRepository.class).in(Singleton.class);\n    bind(AudiobookService.class).in(Singleton.class);\n    bind(AudiobookListViewModel.class).in(Singleton.class);\n}\n",[3168,7381,7382,7388,7398,7436,7462,7488],{"__ignoreMap":3170},[3186,7383,7384,7386],{"class":3188,"line":3189},[3186,7385,3193],{"class":3192},[3186,7387,3197],{"class":3196},[3186,7389,7390,7392,7394,7396],{"class":3188,"line":3200},[3186,7391,4985],{"class":3203},[3186,7393,3207],{"class":3196},[3186,7395,4156],{"class":3210},[3186,7397,4159],{"class":3192},[3186,7399,7400,7402,7404,7406,7408,7410,7412,7414,7416,7418,7420,7422,7424,7426,7428,7430,7432,7434],{"class":3188,"line":3223},[3186,7401,4996],{"class":3210},[3186,7403,3214],{"class":3192},[3186,7405,4184],{"class":3236},[3186,7407,3259],{"class":3192},[3186,7409,4189],{"class":3236},[3186,7411,4192],{"class":3192},[3186,7413,4195],{"class":3210},[3186,7415,3214],{"class":3192},[3186,7417,3673],{"class":3236},[3186,7419,3259],{"class":3192},[3186,7421,4189],{"class":3236},[3186,7423,4192],{"class":3192},[3186,7425,4296],{"class":3210},[3186,7427,3214],{"class":3192},[3186,7429,4301],{"class":3236},[3186,7431,3259],{"class":3192},[3186,7433,4189],{"class":3236},[3186,7435,3271],{"class":3192},[3186,7437,7438,7440,7442,7444,7446,7448,7450,7452,7454,7456,7458,7460],{"class":3188,"line":3230},[3186,7439,4996],{"class":3210},[3186,7441,3214],{"class":3192},[3186,7443,3862],{"class":3236},[3186,7445,3259],{"class":3192},[3186,7447,4189],{"class":3236},[3186,7449,4192],{"class":3192},[3186,7451,4296],{"class":3210},[3186,7453,3214],{"class":3192},[3186,7455,4301],{"class":3236},[3186,7457,3259],{"class":3192},[3186,7459,4189],{"class":3236},[3186,7461,3271],{"class":3192},[3186,7463,7464,7466,7468,7470,7472,7474,7476,7478,7480,7482,7484,7486],{"class":3188,"line":3253},[3186,7465,4996],{"class":3210},[3186,7467,3214],{"class":3192},[3186,7469,3705],{"class":3236},[3186,7471,3259],{"class":3192},[3186,7473,4189],{"class":3236},[3186,7475,4192],{"class":3192},[3186,7477,4296],{"class":3210},[3186,7479,3214],{"class":3192},[3186,7481,4301],{"class":3236},[3186,7483,3259],{"class":3192},[3186,7485,4189],{"class":3236},[3186,7487,3271],{"class":3192},[3186,7489,7490],{"class":3188,"line":3274},[3186,7491,3661],{"class":3192},[3150,7493,7494],{},"Альтернативний синтаксис через анотацію:",[3161,7496,7498],{"className":3180,"code":7497,"language":3182,"meta":3170,"style":3170},"@Singleton\npublic class AudiobookService {\n    private final AudiobookRepository repository;\n    \n    @Inject\n    public AudiobookService(AudiobookRepository repository) {\n        this.repository = repository;\n    }\n}\n",[3168,7499,7500,7506,7516,7530,7534,7540,7554,7566,7570],{"__ignoreMap":3170},[3186,7501,7502,7504],{"class":3188,"line":3189},[3186,7503,3193],{"class":3192},[3186,7505,4388],{"class":3196},[3186,7507,7508,7510,7512,7514],{"class":3188,"line":3200},[3186,7509,3204],{"class":3203},[3186,7511,4124],{"class":3203},[3186,7513,3418],{"class":3196},[3186,7515,4135],{"class":3192},[3186,7517,7518,7520,7522,7525,7528],{"class":3188,"line":3223},[3186,7519,4931],{"class":3203},[3186,7521,5099],{"class":3203},[3186,7523,7524],{"class":3196}," AudiobookRepository",[3186,7526,7527],{"class":3236}," repository",[3186,7529,5106],{"class":3192},[3186,7531,7532],{"class":3188,"line":3230},[3186,7533,3330],{"class":3192},[3186,7535,7536,7538],{"class":3188,"line":3253},[3186,7537,4144],{"class":3192},[3186,7539,5117],{"class":3196},[3186,7541,7542,7544,7546,7548,7550,7552],{"class":3188,"line":3274},[3186,7543,4656],{"class":3203},[3186,7545,3418],{"class":3210},[3186,7547,3214],{"class":3192},[3186,7549,4184],{"class":3196},[3186,7551,7527],{"class":3236},[3186,7553,4948],{"class":3192},[3186,7555,7556,7558,7560,7563],{"class":3188,"line":3291},[3186,7557,5136],{"class":3203},[3186,7559,3259],{"class":3192},[3186,7561,7562],{"class":3236},"repository",[3186,7564,7565],{"class":3192}," = repository;\n",[3186,7567,7568],{"class":3188,"line":3308},[3186,7569,4356],{"class":3192},[3186,7571,7572],{"class":3188,"line":3327},[3186,7573,3661],{"class":3192},[3739,7575,7577],{"id":7576},"prototype-scope-no-scope","Prototype Scope (No Scope)",[3150,7579,7580,7582],{},[3154,7581,7345],{}," Guice створює новий екземпляр об'єкта при кожному запиті.",[3150,7584,7585],{},[3154,7586,7351],{},[7171,7588,7589,7595],{},[3885,7590,7591,7594],{},[3154,7592,7593],{},"Stateful об'єкти:"," ViewModel для діалогів, форм — кожне відкриття діалогу має мати свій стан.",[3885,7596,7597,7600],{},[3154,7598,7599],{},"Короткоживучі об'єкти:"," Task, Command — створюються для одноразового виконання.",[3150,7602,7603],{},[3154,7604,7376],{},[3161,7606,7608],{"className":3180,"code":7607,"language":3182,"meta":3170,"style":3170},"@Override\nprotected void configure() {\n    \u002F\u002F Без .in(Singleton.class) — це Prototype scope\n    bind(AudiobookFormViewModel.class);\n}\n",[3168,7609,7610,7616,7626,7631,7646],{"__ignoreMap":3170},[3186,7611,7612,7614],{"class":3188,"line":3189},[3186,7613,3193],{"class":3192},[3186,7615,3197],{"class":3196},[3186,7617,7618,7620,7622,7624],{"class":3188,"line":3200},[3186,7619,4985],{"class":3203},[3186,7621,3207],{"class":3196},[3186,7623,4156],{"class":3210},[3186,7625,4159],{"class":3192},[3186,7627,7628],{"class":3188,"line":3223},[3186,7629,7630],{"class":3226},"    \u002F\u002F Без .in(Singleton.class) — це Prototype scope\n",[3186,7632,7633,7635,7637,7640,7642,7644],{"class":3188,"line":3230},[3186,7634,4996],{"class":3210},[3186,7636,3214],{"class":3192},[3186,7638,7639],{"class":3236},"AudiobookFormViewModel",[3186,7641,3259],{"class":3192},[3186,7643,4189],{"class":3236},[3186,7645,3271],{"class":3192},[3186,7647,7648],{"class":3188,"line":3253},[3186,7649,3661],{"class":3192},[3150,7651,7652,7653,7655],{},"Кожного разу, коли хтось запитує ",[3168,7654,7639],{},", Guice створить новий екземпляр:",[3161,7657,7659],{"className":3180,"code":7658,"language":3182,"meta":3170,"style":3170},"AudiobookFormViewModel vm1 = injector.getInstance(AudiobookFormViewModel.class);\nAudiobookFormViewModel vm2 = injector.getInstance(AudiobookFormViewModel.class);\n\nSystem.out.println(vm1 == vm2); \u002F\u002F false — різні екземпляри\n",[3168,7660,7661,7687,7712,7716],{"__ignoreMap":3170},[3186,7662,7663,7665,7668,7670,7672,7674,7677,7679,7681,7683,7685],{"class":3188,"line":3189},[3186,7664,7639],{"class":3196},[3186,7666,7667],{"class":3236}," vm1",[3186,7669,3240],{"class":3192},[3186,7671,5425],{"class":3236},[3186,7673,3259],{"class":3192},[3186,7675,7676],{"class":3210},"getInstance",[3186,7678,3214],{"class":3192},[3186,7680,7639],{"class":3236},[3186,7682,3259],{"class":3192},[3186,7684,4189],{"class":3236},[3186,7686,3271],{"class":3192},[3186,7688,7689,7691,7694,7696,7698,7700,7702,7704,7706,7708,7710],{"class":3188,"line":3200},[3186,7690,7639],{"class":3196},[3186,7692,7693],{"class":3236}," vm2",[3186,7695,3240],{"class":3192},[3186,7697,5425],{"class":3236},[3186,7699,3259],{"class":3192},[3186,7701,7676],{"class":3210},[3186,7703,3214],{"class":3192},[3186,7705,7639],{"class":3236},[3186,7707,3259],{"class":3192},[3186,7709,4189],{"class":3236},[3186,7711,3271],{"class":3192},[3186,7713,7714],{"class":3188,"line":3223},[3186,7715,4024],{"emptyLinePlaceholder":4023},[3186,7717,7718,7721,7723,7726,7728,7731,7734],{"class":3188,"line":3230},[3186,7719,7720],{"class":3236},"System",[3186,7722,3259],{"class":3192},[3186,7724,7725],{"class":3236},"out",[3186,7727,3259],{"class":3192},[3186,7729,7730],{"class":3210},"println",[3186,7732,7733],{"class":3192},"(vm1 == vm2); ",[3186,7735,7736],{"class":3226},"\u002F\u002F false — різні екземпляри\n",[3739,7738,7740],{"id":7739},"проблема-singleton-viewmodel-для-діалогів","Проблема: Singleton ViewModel для діалогів",[3150,7742,7743],{},"Розглянемо проблемний сценарій:",[3161,7745,7747],{"className":3180,"code":7746,"language":3182,"meta":3170,"style":3170},"\u002F\u002F AudiobookFormViewModel — Singleton\nbind(AudiobookFormViewModel.class).in(Singleton.class);\n",[3168,7748,7749,7754],{"__ignoreMap":3170},[3186,7750,7751],{"class":3188,"line":3189},[3186,7752,7753],{"class":3226},"\u002F\u002F AudiobookFormViewModel — Singleton\n",[3186,7755,7756,7758,7760,7762,7764,7766,7768,7770,7772,7774,7776,7778],{"class":3188,"line":3200},[3186,7757,6554],{"class":3210},[3186,7759,3214],{"class":3192},[3186,7761,7639],{"class":3236},[3186,7763,3259],{"class":3192},[3186,7765,4189],{"class":3236},[3186,7767,4192],{"class":3192},[3186,7769,4296],{"class":3210},[3186,7771,3214],{"class":3192},[3186,7773,4301],{"class":3236},[3186,7775,3259],{"class":3192},[3186,7777,4189],{"class":3236},[3186,7779,3271],{"class":3192},[3150,7781,7782,7783,7786],{},"Користувач відкриває діалог \"Add Audiobook\" → вводить дані → закриває діалог. Потім знову відкриває діалог → ",[3154,7784,7785],{},"бачить попередні дані","! Це тому, що ViewModel — Singleton, і він зберігає стан між відкриттями.",[3150,7788,7789,7792],{},[3154,7790,7791],{},"Рішення:"," ViewModel для діалогів має бути Prototype:",[3161,7794,7796],{"className":3180,"code":7795,"language":3182,"meta":3170,"style":3170},"bind(AudiobookFormViewModel.class); \u002F\u002F Без Singleton\n",[3168,7797,7798],{"__ignoreMap":3170},[3186,7799,7800,7802,7804,7806,7808,7810,7813],{"class":3188,"line":3189},[3186,7801,6554],{"class":3210},[3186,7803,3214],{"class":3192},[3186,7805,7639],{"class":3236},[3186,7807,3259],{"class":3192},[3186,7809,4189],{"class":3236},[3186,7811,7812],{"class":3192},"); ",[3186,7814,7815],{"class":3226},"\u002F\u002F Без Singleton\n",[3150,7817,7818],{},"Тепер кожне відкриття діалогу створює новий ViewModel з чистим станом.",[5704,7820,7821,7824],{},[3154,7822,7823],{},"Будьте обережні з Singleton ViewModel."," Переконайтеся, що стан не конфліктує між різними використаннями. Якщо ViewModel містить стан форми (введені дані, валідаційні помилки), він має бути Prototype.",[3729,7826],{},[3145,7828,7830],{"id":7829},"application-entry-point-ініціалізація-guice","Application Entry Point: Ініціалізація Guice",[3150,7832,7833,7834,7836],{},"Тепер з'єднаємо всі частини разом: створимо ",[3168,7835,3908],{}," клас, що ініціалізує Guice Injector, завантажує головний екран та керує життєвим циклом додатку.",[3739,7838,7840],{"id":7839},"audiobookapp-точка-входу","AudiobookApp: Точка входу",[3161,7842,7844],{"className":3180,"code":7843,"language":3182,"meta":3170,"style":3170},"package dev.kostyl.audiobook;\n\nimport com.google.inject.Guice;\nimport com.google.inject.Injector;\nimport dev.kostyl.audiobook.infrastructure.AudiobookModule;\nimport javafx.application.Application;\nimport javafx.fxml.FXMLLoader;\nimport javafx.scene.Parent;\nimport javafx.scene.Scene;\nimport javafx.stage.Stage;\n\nimport javax.sql.DataSource;\nimport java.io.IOException;\n\npublic class AudiobookApp extends Application {\n    \n    private Injector injector;\n    \n    @Override\n    public void init() throws Exception {\n        \u002F\u002F Створення Guice Injector з модулем конфігурації\n        injector = Guice.createInjector(new AudiobookModule());\n        \n        \u002F\u002F Ініціалізація бази даних (міграції через Flyway)\n        initializeDatabase();\n    }\n    \n    @Override\n    public void start(Stage primaryStage) throws IOException {\n        \u002F\u002F Завантаження головного екрану через FXMLLoader з ControllerFactory\n        FXMLLoader loader = new FXMLLoader(\n            getClass().getResource(\"\u002Ffxml\u002Faudiobook-list-view.fxml\")\n        );\n        loader.setControllerFactory(injector::getInstance);\n        \n        Parent root = loader.load();\n        \n        \u002F\u002F Налаштування Scene\n        Scene scene = new Scene(root, 1000, 700);\n        scene.getStylesheets().add(\n            getClass().getResource(\"\u002Fcss\u002Fstyles.css\").toExternalForm()\n        );\n        \n        \u002F\u002F Налаштування Stage\n        primaryStage.setTitle(\"Audiobook Platform\");\n        primaryStage.setScene(scene);\n        primaryStage.setMinWidth(800);\n        primaryStage.setMinHeight(600);\n        primaryStage.show();\n    }\n    \n    @Override\n    public void stop() throws Exception {\n        \u002F\u002F Закриття ресурсів при завершенні додатку\n        closeResources();\n    }\n    \n    private void initializeDatabase() {\n        DataSource dataSource = injector.getInstance(DataSource.class);\n        \n        \u002F\u002F Виконання міграцій через Flyway\n        Flyway flyway = Flyway.configure()\n            .dataSource(dataSource)\n            .locations(\"classpath:db\u002Fmigration\")\n            .load();\n        \n        flyway.migrate();\n        \n        System.out.println(\"Database initialized successfully\");\n    }\n    \n    private void closeResources() {\n        try {\n            \u002F\u002F Закриття Connection Pool\n            DataSource dataSource = injector.getInstance(DataSource.class);\n            if (dataSource instanceof HikariDataSource) {\n                ((HikariDataSource) dataSource).close();\n                System.out.println(\"DataSource closed\");\n            }\n        } catch (Exception e) {\n            System.err.println(\"Error closing resources: \" + e.getMessage());\n        }\n    }\n    \n    public static void main(String[] args) {\n        launch(args);\n    }\n}\n",[3168,7845,7846,7853,7857,7864,7871,7878,7885,7892,7899,7906,7913,7917,7923,7930,7934,7950,7954,7964,7968,7974,7993,7998,8017,8021,8026,8033,8037,8041,8047,8070,8075,8089,8104,8109,8123,8127,8144,8148,8153,8178,8195,8215,8219,8223,8228,8245,8255,8271,8286,8296,8300,8304,8310,8327,8332,8339,8343,8347,8358,8382,8386,8390,8406,8414,8426,8434,8438,8448,8452,8472,8476,8480,8491,8498,8503,8528,8542,8552,8572,8577,8596,8628,8632,8636,8640,8664,8672,8676],{"__ignoreMap":3170},[3186,7847,7848,7850],{"class":3188,"line":3189},[3186,7849,4015],{"class":3203},[3186,7851,7852],{"class":3192}," dev.kostyl.audiobook;\n",[3186,7854,7855],{"class":3188,"line":3200},[3186,7856,4024],{"emptyLinePlaceholder":4023},[3186,7858,7859,7861],{"class":3188,"line":3223},[3186,7860,4029],{"class":3203},[3186,7862,7863],{"class":3192}," com.google.inject.Guice;\n",[3186,7865,7866,7868],{"class":3188,"line":3230},[3186,7867,4029],{"class":3203},[3186,7869,7870],{"class":3192}," com.google.inject.Injector;\n",[3186,7872,7873,7875],{"class":3188,"line":3253},[3186,7874,4029],{"class":3203},[3186,7876,7877],{"class":3192}," dev.kostyl.audiobook.infrastructure.AudiobookModule;\n",[3186,7879,7880,7882],{"class":3188,"line":3274},[3186,7881,4029],{"class":3203},[3186,7883,7884],{"class":3192}," javafx.application.Application;\n",[3186,7886,7887,7889],{"class":3188,"line":3291},[3186,7888,4029],{"class":3203},[3186,7890,7891],{"class":3192}," javafx.fxml.FXMLLoader;\n",[3186,7893,7894,7896],{"class":3188,"line":3308},[3186,7895,4029],{"class":3203},[3186,7897,7898],{"class":3192}," javafx.scene.Parent;\n",[3186,7900,7901,7903],{"class":3188,"line":3327},[3186,7902,4029],{"class":3203},[3186,7904,7905],{"class":3192}," javafx.scene.Scene;\n",[3186,7907,7908,7910],{"class":3188,"line":3333},[3186,7909,4029],{"class":3203},[3186,7911,7912],{"class":3192}," javafx.stage.Stage;\n",[3186,7914,7915],{"class":3188,"line":3339},[3186,7916,4024],{"emptyLinePlaceholder":4023},[3186,7918,7919,7921],{"class":3188,"line":3358},[3186,7920,4029],{"class":3203},[3186,7922,4113],{"class":3192},[3186,7924,7925,7927],{"class":3188,"line":3376},[3186,7926,4029],{"class":3203},[3186,7928,7929],{"class":3192}," java.io.IOException;\n",[3186,7931,7932],{"class":3188,"line":3394},[3186,7933,4024],{"emptyLinePlaceholder":4023},[3186,7935,7936,7938,7940,7943,7945,7948],{"class":3188,"line":3399},[3186,7937,3204],{"class":3203},[3186,7939,4124],{"class":3203},[3186,7941,7942],{"class":3196}," AudiobookApp",[3186,7944,4129],{"class":3203},[3186,7946,7947],{"class":3196}," Application",[3186,7949,4135],{"class":3192},[3186,7951,7952],{"class":3188,"line":3405},[3186,7953,3330],{"class":3192},[3186,7955,7956,7958,7960,7962],{"class":3188,"line":3424},[3186,7957,4931],{"class":3203},[3186,7959,5388],{"class":3196},[3186,7961,3761],{"class":3236},[3186,7963,5106],{"class":3192},[3186,7965,7966],{"class":3188,"line":3430},[3186,7967,3330],{"class":3192},[3186,7969,7970,7972],{"class":3188,"line":3436},[3186,7971,4144],{"class":3192},[3186,7973,3197],{"class":3196},[3186,7975,7976,7978,7980,7983,7986,7988,7991],{"class":3188,"line":3442},[3186,7977,4656],{"class":3203},[3186,7979,3207],{"class":3196},[3186,7981,7982],{"class":3210}," init",[3186,7984,7985],{"class":3192},"() ",[3186,7987,5459],{"class":3203},[3186,7989,7990],{"class":3196}," Exception",[3186,7992,4135],{"class":3192},[3186,7994,7995],{"class":3188,"line":3448},[3186,7996,7997],{"class":3226},"        \u002F\u002F Створення Guice Injector з модулем конфігурації\n",[3186,7999,8000,8003,8005,8007,8009,8011,8013,8015],{"class":3188,"line":3453},[3186,8001,8002],{"class":3192},"        injector = ",[3186,8004,3766],{"class":3236},[3186,8006,3259],{"class":3192},[3186,8008,3771],{"class":3210},[3186,8010,3214],{"class":3192},[3186,8012,3244],{"class":3243},[3186,8014,3778],{"class":3210},[3186,8016,3781],{"class":3192},[3186,8018,8019],{"class":3188,"line":3459},[3186,8020,4169],{"class":3192},[3186,8022,8023],{"class":3188,"line":3478},[3186,8024,8025],{"class":3226},"        \u002F\u002F Ініціалізація бази даних (міграції через Flyway)\n",[3186,8027,8028,8031],{"class":3188,"line":3483},[3186,8029,8030],{"class":3210},"        initializeDatabase",[3186,8032,3250],{"class":3192},[3186,8034,8035],{"class":3188,"line":3489},[3186,8036,4356],{"class":3192},[3186,8038,8039],{"class":3188,"line":3524},[3186,8040,3330],{"class":3192},[3186,8042,8043,8045],{"class":3188,"line":3545},[3186,8044,4144],{"class":3192},[3186,8046,3197],{"class":3196},[3186,8048,8049,8051,8053,8055,8057,8059,8062,8064,8066,8068],{"class":3188,"line":3550},[3186,8050,4656],{"class":3203},[3186,8052,3207],{"class":3196},[3186,8054,3211],{"class":3210},[3186,8056,3214],{"class":3192},[3186,8058,3217],{"class":3196},[3186,8060,8061],{"class":3236}," primaryStage",[3186,8063,5456],{"class":3192},[3186,8065,5459],{"class":3203},[3186,8067,5462],{"class":3196},[3186,8069,4135],{"class":3192},[3186,8071,8072],{"class":3188,"line":3556},[3186,8073,8074],{"class":3226},"        \u002F\u002F Завантаження головного екрану через FXMLLoader з ControllerFactory\n",[3186,8076,8077,8079,8081,8083,8085,8087],{"class":3188,"line":3576},[3186,8078,5469],{"class":3196},[3186,8080,3495],{"class":3236},[3186,8082,3240],{"class":3192},[3186,8084,3244],{"class":3243},[3186,8086,3502],{"class":3210},[3186,8088,3421],{"class":3192},[3186,8090,8091,8094,8096,8098,8100,8102],{"class":3188,"line":3590},[3186,8092,8093],{"class":3210},"            getClass",[3186,8095,3510],{"class":3192},[3186,8097,3513],{"class":3210},[3186,8099,3214],{"class":3192},[3186,8101,5683],{"class":3267},[3186,8103,4887],{"class":3192},[3186,8105,8106],{"class":3188,"line":3595},[3186,8107,8108],{"class":3192},"        );\n",[3186,8110,8111,8113,8115,8117,8119,8121],{"class":3188,"line":3601},[3186,8112,5493],{"class":3236},[3186,8114,3259],{"class":3192},[3186,8116,5323],{"class":3210},[3186,8118,5326],{"class":3192},[3186,8120,5329],{"class":3243},[3186,8122,5332],{"class":3192},[3186,8124,8125],{"class":3188,"line":3632},[3186,8126,4169],{"class":3192},[3186,8128,8129,8132,8134,8136,8138,8140,8142],{"class":3188,"line":3646},[3186,8130,8131],{"class":3196},"        Parent",[3186,8133,3530],{"class":3236},[3186,8135,3240],{"class":3192},[3186,8137,3535],{"class":3236},[3186,8139,3259],{"class":3192},[3186,8141,3540],{"class":3210},[3186,8143,3250],{"class":3192},[3186,8145,8146],{"class":3188,"line":3658},[3186,8147,4169],{"class":3192},[3186,8149,8150],{"class":3188,"line":4359},[3186,8151,8152],{"class":3226},"        \u002F\u002F Налаштування Scene\n",[3186,8154,8155,8158,8160,8162,8164,8166,8168,8171,8173,8176],{"class":3188,"line":4364},[3186,8156,8157],{"class":3196},"        Scene",[3186,8159,3607],{"class":3236},[3186,8161,3240],{"class":3192},[3186,8163,3244],{"class":3243},[3186,8165,3614],{"class":3210},[3186,8167,3617],{"class":3192},[3186,8169,8170],{"class":3620},"1000",[3186,8172,3624],{"class":3192},[3186,8174,8175],{"class":3620},"700",[3186,8177,3271],{"class":3192},[3186,8179,8180,8183,8185,8188,8190,8193],{"class":3188,"line":4370},[3186,8181,8182],{"class":3236},"        scene",[3186,8184,3259],{"class":3192},[3186,8186,8187],{"class":3210},"getStylesheets",[3186,8189,3510],{"class":3192},[3186,8191,8192],{"class":3210},"add",[3186,8194,3421],{"class":3192},[3186,8196,8197,8199,8201,8203,8205,8208,8210,8213],{"class":3188,"line":4375},[3186,8198,8093],{"class":3210},[3186,8200,3510],{"class":3192},[3186,8202,3513],{"class":3210},[3186,8204,3214],{"class":3192},[3186,8206,8207],{"class":3267},"\"\u002Fcss\u002Fstyles.css\"",[3186,8209,4192],{"class":3192},[3186,8211,8212],{"class":3210},"toExternalForm",[3186,8214,4861],{"class":3192},[3186,8216,8217],{"class":3188,"line":4383},[3186,8218,8108],{"class":3192},[3186,8220,8221],{"class":3188,"line":4391},[3186,8222,4169],{"class":3192},[3186,8224,8225],{"class":3188,"line":4401},[3186,8226,8227],{"class":3226},"        \u002F\u002F Налаштування Stage\n",[3186,8229,8230,8233,8235,8238,8240,8243],{"class":3188,"line":4417},[3186,8231,8232],{"class":3236},"        primaryStage",[3186,8234,3259],{"class":3192},[3186,8236,8237],{"class":3210},"setTitle",[3186,8239,3214],{"class":3192},[3186,8241,8242],{"class":3267},"\"Audiobook Platform\"",[3186,8244,3271],{"class":3192},[3186,8246,8247,8249,8251,8253],{"class":3188,"line":4434},[3186,8248,8232],{"class":3236},[3186,8250,3259],{"class":3192},[3186,8252,3640],{"class":3210},[3186,8254,3643],{"class":3192},[3186,8256,8257,8259,8261,8264,8266,8269],{"class":3188,"line":4449},[3186,8258,8232],{"class":3236},[3186,8260,3259],{"class":3192},[3186,8262,8263],{"class":3210},"setMinWidth",[3186,8265,3214],{"class":3192},[3186,8267,8268],{"class":3620},"800",[3186,8270,3271],{"class":3192},[3186,8272,8273,8275,8277,8280,8282,8284],{"class":3188,"line":4464},[3186,8274,8232],{"class":3236},[3186,8276,3259],{"class":3192},[3186,8278,8279],{"class":3210},"setMinHeight",[3186,8281,3214],{"class":3192},[3186,8283,3627],{"class":3620},[3186,8285,3271],{"class":3192},[3186,8287,8288,8290,8292,8294],{"class":3188,"line":4481},[3186,8289,8232],{"class":3236},[3186,8291,3259],{"class":3192},[3186,8293,3653],{"class":3210},[3186,8295,3250],{"class":3192},[3186,8297,8298],{"class":3188,"line":4498},[3186,8299,4356],{"class":3192},[3186,8301,8302],{"class":3188,"line":4515},[3186,8303,3330],{"class":3192},[3186,8305,8306,8308],{"class":3188,"line":4520},[3186,8307,4144],{"class":3192},[3186,8309,3197],{"class":3196},[3186,8311,8312,8314,8316,8319,8321,8323,8325],{"class":3188,"line":4533},[3186,8313,4656],{"class":3203},[3186,8315,3207],{"class":3196},[3186,8317,8318],{"class":3210}," stop",[3186,8320,7985],{"class":3192},[3186,8322,5459],{"class":3203},[3186,8324,7990],{"class":3196},[3186,8326,4135],{"class":3192},[3186,8328,8329],{"class":3188,"line":4538},[3186,8330,8331],{"class":3226},"        \u002F\u002F Закриття ресурсів при завершенні додатку\n",[3186,8333,8334,8337],{"class":3188,"line":6267},[3186,8335,8336],{"class":3210},"        closeResources",[3186,8338,3250],{"class":3192},[3186,8340,8341],{"class":3188,"line":6285},[3186,8342,4356],{"class":3192},[3186,8344,8345],{"class":3188,"line":6301},[3186,8346,3330],{"class":3192},[3186,8348,8349,8351,8353,8356],{"class":3188,"line":6317},[3186,8350,4931],{"class":3203},[3186,8352,3207],{"class":3196},[3186,8354,8355],{"class":3210}," initializeDatabase",[3186,8357,4159],{"class":3192},[3186,8359,8360,8362,8364,8366,8368,8370,8372,8374,8376,8378,8380],{"class":3188,"line":6333},[3186,8361,4821],{"class":3196},[3186,8363,3314],{"class":3236},[3186,8365,3240],{"class":3192},[3186,8367,5425],{"class":3236},[3186,8369,3259],{"class":3192},[3186,8371,7676],{"class":3210},[3186,8373,3214],{"class":3192},[3186,8375,4638],{"class":3236},[3186,8377,3259],{"class":3192},[3186,8379,4189],{"class":3236},[3186,8381,3271],{"class":3192},[3186,8383,8384],{"class":3188,"line":6349},[3186,8385,4169],{"class":3192},[3186,8387,8388],{"class":3188,"line":6365},[3186,8389,4840],{"class":3226},[3186,8391,8392,8394,8396,8398,8400,8402,8404],{"class":3188,"line":6381},[3186,8393,4845],{"class":3196},[3186,8395,4848],{"class":3236},[3186,8397,3240],{"class":3192},[3186,8399,4853],{"class":3236},[3186,8401,3259],{"class":3192},[3186,8403,4858],{"class":3210},[3186,8405,4861],{"class":3192},[3186,8407,8408,8410,8412],{"class":3188,"line":6386},[3186,8409,4866],{"class":3192},[3186,8411,4869],{"class":3210},[3186,8413,4872],{"class":3192},[3186,8415,8416,8418,8420,8422,8424],{"class":3188,"line":6391},[3186,8417,4866],{"class":3192},[3186,8419,4879],{"class":3210},[3186,8421,3214],{"class":3192},[3186,8423,4884],{"class":3267},[3186,8425,4887],{"class":3192},[3186,8427,8428,8430,8432],{"class":3188,"line":6403},[3186,8429,4866],{"class":3192},[3186,8431,3540],{"class":3210},[3186,8433,3250],{"class":3192},[3186,8435,8436],{"class":3188,"line":6409},[3186,8437,4169],{"class":3192},[3186,8439,8440,8442,8444,8446],{"class":3188,"line":6431},[3186,8441,4900],{"class":3236},[3186,8443,3259],{"class":3192},[3186,8445,4905],{"class":3210},[3186,8447,3250],{"class":3192},[3186,8449,8450],{"class":3188,"line":6448},[3186,8451,4169],{"class":3192},[3186,8453,8454,8457,8459,8461,8463,8465,8467,8470],{"class":3188,"line":6467},[3186,8455,8456],{"class":3236},"        System",[3186,8458,3259],{"class":3192},[3186,8460,7725],{"class":3236},[3186,8462,3259],{"class":3192},[3186,8464,7730],{"class":3210},[3186,8466,3214],{"class":3192},[3186,8468,8469],{"class":3267},"\"Database initialized successfully\"",[3186,8471,3271],{"class":3192},[3186,8473,8474],{"class":3188,"line":6472},[3186,8475,4356],{"class":3192},[3186,8477,8478],{"class":3188,"line":6478},[3186,8479,3330],{"class":3192},[3186,8481,8482,8484,8486,8489],{"class":3188,"line":6504},[3186,8483,4931],{"class":3203},[3186,8485,3207],{"class":3196},[3186,8487,8488],{"class":3210}," closeResources",[3186,8490,4159],{"class":3192},[3186,8492,8493,8496],{"class":3188,"line":6530},[3186,8494,8495],{"class":3243},"        try",[3186,8497,4135],{"class":3192},[3186,8499,8500],{"class":3188,"line":6535},[3186,8501,8502],{"class":3226},"            \u002F\u002F Закриття Connection Pool\n",[3186,8504,8505,8508,8510,8512,8514,8516,8518,8520,8522,8524,8526],{"class":3188,"line":6541},[3186,8506,8507],{"class":3196},"            DataSource",[3186,8509,3314],{"class":3236},[3186,8511,3240],{"class":3192},[3186,8513,5425],{"class":3236},[3186,8515,3259],{"class":3192},[3186,8517,7676],{"class":3210},[3186,8519,3214],{"class":3192},[3186,8521,4638],{"class":3236},[3186,8523,3259],{"class":3192},[3186,8525,4189],{"class":3236},[3186,8527,3271],{"class":3192},[3186,8529,8530,8533,8536,8539],{"class":3188,"line":6568},[3186,8531,8532],{"class":3243},"            if",[3186,8534,8535],{"class":3192}," (dataSource ",[3186,8537,8538],{"class":3203},"instanceof",[3186,8540,8541],{"class":3192}," HikariDataSource) {\n",[3186,8543,8544,8547,8550],{"class":3188,"line":6593},[3186,8545,8546],{"class":3192},"                ((HikariDataSource) dataSource).",[3186,8548,8549],{"class":3210},"close",[3186,8551,3250],{"class":3192},[3186,8553,8554,8557,8559,8561,8563,8565,8567,8570],{"class":3188,"line":6598},[3186,8555,8556],{"class":3236},"                System",[3186,8558,3259],{"class":3192},[3186,8560,7725],{"class":3236},[3186,8562,3259],{"class":3192},[3186,8564,7730],{"class":3210},[3186,8566,3214],{"class":3192},[3186,8568,8569],{"class":3267},"\"DataSource closed\"",[3186,8571,3271],{"class":3192},[3186,8573,8574],{"class":3188,"line":6604},[3186,8575,8576],{"class":3192},"            }\n",[3186,8578,8579,8582,8585,8588,8591,8594],{"class":3188,"line":6634},[3186,8580,8581],{"class":3192},"        } ",[3186,8583,8584],{"class":3243},"catch",[3186,8586,8587],{"class":3192}," (",[3186,8589,8590],{"class":3196},"Exception",[3186,8592,8593],{"class":3236}," e",[3186,8595,4948],{"class":3192},[3186,8597,8598,8601,8603,8606,8608,8610,8612,8615,8618,8621,8623,8626],{"class":3188,"line":6662},[3186,8599,8600],{"class":3236},"            System",[3186,8602,3259],{"class":3192},[3186,8604,8605],{"class":3236},"err",[3186,8607,3259],{"class":3192},[3186,8609,7730],{"class":3210},[3186,8611,3214],{"class":3192},[3186,8613,8614],{"class":3267},"\"Error closing resources: \"",[3186,8616,8617],{"class":3192}," + ",[3186,8619,8620],{"class":3236},"e",[3186,8622,3259],{"class":3192},[3186,8624,8625],{"class":3210},"getMessage",[3186,8627,3781],{"class":3192},[3186,8629,8630],{"class":3188,"line":6667},[3186,8631,7156],{"class":3192},[3186,8633,8634],{"class":3188,"line":6672},[3186,8635,4356],{"class":3192},[3186,8637,8638],{"class":3188,"line":6684},[3186,8639,3330],{"class":3192},[3186,8641,8642,8644,8647,8649,8652,8654,8656,8659,8662],{"class":3188,"line":6705},[3186,8643,4656],{"class":3203},[3186,8645,8646],{"class":3203}," static",[3186,8648,3207],{"class":3196},[3186,8650,8651],{"class":3210}," main",[3186,8653,3214],{"class":3192},[3186,8655,4942],{"class":3196},[3186,8657,8658],{"class":3192},"[] ",[3186,8660,8661],{"class":3236},"args",[3186,8663,4948],{"class":3192},[3186,8665,8666,8669],{"class":3188,"line":6723},[3186,8667,8668],{"class":3210},"        launch",[3186,8670,8671],{"class":3192},"(args);\n",[3186,8673,8674],{"class":3188,"line":6741},[3186,8675,4356],{"class":3192},[3186,8677,8678],{"class":3188,"line":6746},[3186,8679,3661],{"class":3192},[3739,8681,8683],{"id":8682},"розбір-життєвого-циклу-application","Розбір життєвого циклу Application",[3150,8685,8686,8692,8693,8695,8696,3693,8699,8701,8702,8704],{},[3154,8687,8688,8689,3259],{},"Рядки 20-25: ",[3168,8690,8691],{},"init()"," Метод ",[3168,8694,8691],{}," викликається ",[3154,8697,8698],{},"перед",[3168,8700,3681],{},", у фоновому потоці. Це ідеальне місце для ініціалізації Guice Injector та виконання міграцій бази даних. Якщо ",[3168,8703,8691],{}," викине виняток, додаток не запуститься.",[3150,8706,8707,3693,8710,8713],{},[3154,8708,8709],{},"Рядок 22: Створення Injector.",[3168,8711,8712],{},"Guice.createInjector(new AudiobookModule())"," — це точка, де Guice читає конфігурацію з Module та будує граф залежностей. Це відбувається один раз при старті додатку.",[3150,8715,8716,8692,8721,8723,8724,8726,8727,8729],{},[3154,8717,8718,8719,3259],{},"Рядки 28-49: ",[3168,8720,3681],{},[3168,8722,3681],{}," викликається у JavaFX Application Thread. Тут завантажуємо головний екран через ",[3168,8725,3796],{}," з ",[3168,8728,3807],{},", налаштовуємо Scene та Stage, показуємо вікно.",[3150,8731,8732,3693,8735,8737,8738,8740,8741,8743],{},[3154,8733,8734],{},"Рядок 34: ControllerFactory.",[3168,8736,3918],{}," — це міст між ",[3168,8739,3796],{}," та Guice. Тепер ",[3168,8742,3796],{}," створюватиме Controllers через Guice, що автоматично ін'єктує всі залежності.",[3150,8745,8746,8692,8752,8754,8755,8758],{},[3154,8747,8748,8749,3259],{},"Рядки 51-54: ",[3168,8750,8751],{},"stop()",[3168,8753,8751],{}," викликається при закритті додатку (користувач натиснув \"X\" або викликав ",[3168,8756,8757],{},"Platform.exit()","). Тут закриваємо ресурси: Connection Pool, ExecutorService, файли.",[3150,8760,8761,8764,8765,8767],{},[3154,8762,8763],{},"Рядки 56-67: Ініціалізація бази даних."," Отримуємо ",[3168,8766,4638],{}," з Injector та виконуємо міграції через Flyway. Це гарантує, що база даних має актуальну схему перед запуском додатку.",[3150,8769,8770,8764,8773,8775],{},[3154,8771,8772],{},"Рядки 69-81: Закриття ресурсів.",[3168,8774,4638],{}," та закриваємо його. HikariCP автоматично закриє всі активні з'єднання та звільнить ресурси.",[3722,8777,8778,8781,8782,8784,8785,8787,8788,3259],{},[3154,8779,8780],{},"Чому init() у фоновому потоці?"," JavaFX викликає ",[3168,8783,8691],{}," у фоновому потоці, щоб не блокувати UI Thread. Це дозволяє виконувати довгі операції (міграції, завантаження конфігурації) без заморожування інтерфейсу. Але будьте обережні: не створюйте JavaFX-об'єкти (Node, Scene) у ",[3168,8786,8691],{}," — вони мають створюватися у ",[3168,8789,3681],{},[3739,8791,8793],{"id":8792},"структура-проєкту","Структура проєкту",[8795,8796,8797],"code-tree",{},[3161,8798,8801],{"className":8799,"code":8800,"language":3166},[3164],"audiobook-platform\u002F\n├── src\u002F\n│   ├── main\u002F\n│   │   ├── java\u002F\n│   │   │   └── dev\u002Fkostyl\u002Faudiobook\u002F\n│   │   │       ├── AudiobookApp.java              # Application Entry Point\n│   │   │       ├── controller\u002F\n│   │   │       │   ├── AudiobookListController.java\n│   │   │       │   └── AudiobookFormController.java\n│   │   │       ├── viewmodel\u002F\n│   │   │       │   ├── AudiobookListViewModel.java\n│   │   │       │   └── AudiobookFormViewModel.java\n│   │   │       ├── service\u002F\n│   │   │       │   └── AudiobookService.java\n│   │   │       ├── repository\u002F\n│   │   │       │   ├── AudiobookRepository.java   # Interface\n│   │   │       │   └── jdbc\u002F\n│   │   │       │       └── JdbcAudiobookRepository.java\n│   │   │       ├── domain\u002F\n│   │   │       │   ├── Audiobook.java\n│   │   │       │   ├── Author.java\n│   │   │       │   └── Genre.java\n│   │   │       └── infrastructure\u002F\n│   │   │           ├── AudiobookModule.java       # Guice Module\n│   │   │           └── HikariDataSourceProvider.java\n│   │   └── resources\u002F\n│   │       ├── fxml\u002F\n│   │       │   ├── audiobook-list-view.fxml\n│   │       │   └── audiobook-form-view.fxml\n│   │       ├── css\u002F\n│   │       │   └── styles.css\n│   │       └── db\u002Fmigration\u002F\n│   │           ├── V1__Create_authors.sql\n│   │           ├── V2__Create_genres.sql\n│   │           └── V3__Create_audiobooks.sql\n│   └── test\u002F\n│       └── java\u002F\n│           └── dev\u002Fkostyl\u002Faudiobook\u002F\n│               ├── viewmodel\u002F\n│               │   └── AudiobookListViewModelTest.java\n│               └── repository\u002F\n│                   └── JdbcAudiobookRepositoryTest.java\n└── pom.xml\n",[3168,8802,8800],{"__ignoreMap":3170},[3729,8804],{},[3145,8806,8808],{"id":8807},"assistedinject-параметризовані-viewmodel","AssistedInject: Параметризовані ViewModel",[3150,8810,8811,8812,8815,8816,8819],{},"Іноді ViewModel потребує параметр при створенні. Наприклад, ",[3168,8813,8814],{},"AudiobookDetailViewModel"," має показувати деталі конкретної аудіокниги — йому потрібен ",[3168,8817,8818],{},"Audiobook"," у конструкторі:",[3161,8821,8823],{"className":3180,"code":8822,"language":3182,"meta":3170,"style":3170},"public class AudiobookDetailViewModel {\n    private final Audiobook audiobook;\n    private final AudiobookRepository repository;\n    \n    public AudiobookDetailViewModel(Audiobook audiobook, AudiobookRepository repository) {\n        this.audiobook = audiobook;\n        this.repository = repository;\n    }\n}\n",[3168,8824,8825,8836,8850,8862,8866,8886,8898,8908,8912],{"__ignoreMap":3170},[3186,8826,8827,8829,8831,8834],{"class":3188,"line":3189},[3186,8828,3204],{"class":3203},[3186,8830,4124],{"class":3203},[3186,8832,8833],{"class":3196}," AudiobookDetailViewModel",[3186,8835,4135],{"class":3192},[3186,8837,8838,8840,8842,8845,8848],{"class":3188,"line":3200},[3186,8839,4931],{"class":3203},[3186,8841,5099],{"class":3203},[3186,8843,8844],{"class":3196}," Audiobook",[3186,8846,8847],{"class":3236}," audiobook",[3186,8849,5106],{"class":3192},[3186,8851,8852,8854,8856,8858,8860],{"class":3188,"line":3223},[3186,8853,4931],{"class":3203},[3186,8855,5099],{"class":3203},[3186,8857,7524],{"class":3196},[3186,8859,7527],{"class":3236},[3186,8861,5106],{"class":3192},[3186,8863,8864],{"class":3188,"line":3230},[3186,8865,3330],{"class":3192},[3186,8867,8868,8870,8872,8874,8876,8878,8880,8882,8884],{"class":3188,"line":3253},[3186,8869,4656],{"class":3203},[3186,8871,8833],{"class":3210},[3186,8873,3214],{"class":3192},[3186,8875,8818],{"class":3196},[3186,8877,8847],{"class":3236},[3186,8879,3624],{"class":3192},[3186,8881,4184],{"class":3196},[3186,8883,7527],{"class":3236},[3186,8885,4948],{"class":3192},[3186,8887,8888,8890,8892,8895],{"class":3188,"line":3274},[3186,8889,5136],{"class":3203},[3186,8891,3259],{"class":3192},[3186,8893,8894],{"class":3236},"audiobook",[3186,8896,8897],{"class":3192}," = audiobook;\n",[3186,8899,8900,8902,8904,8906],{"class":3188,"line":3291},[3186,8901,5136],{"class":3203},[3186,8903,3259],{"class":3192},[3186,8905,7562],{"class":3236},[3186,8907,7565],{"class":3192},[3186,8909,8910],{"class":3188,"line":3308},[3186,8911,4356],{"class":3192},[3186,8913,8914],{"class":3188,"line":3327},[3186,8915,3661],{"class":3192},[3150,8917,8918,8919,8921,8922,8925],{},"Але як передати ",[3168,8920,8894],{}," через Guice? ",[3168,8923,8924],{},"injector.getInstance(AudiobookDetailViewModel.class)"," не приймає параметрів.",[3739,8927,8929],{"id":8928},"рішення-assistedinject","Рішення: AssistedInject",[3150,8931,8932,8933,8936],{},"Guice підтримує ",[3154,8934,8935],{},"AssistedInject"," — механізм для створення об'єктів з параметрами через фабрику.",[3150,8938,8939],{},[3154,8940,8941],{},"Крок 1: Анотація @Assisted для параметрів",[3161,8943,8945],{"className":3180,"code":8944,"language":3182,"meta":3170,"style":3170},"import com.google.inject.assistedinject.Assisted;\nimport com.google.inject.Inject;\n\npublic class AudiobookDetailViewModel {\n    private final Audiobook audiobook;\n    private final AudiobookRepository repository;\n    \n    @Inject\n    public AudiobookDetailViewModel(\n        @Assisted Audiobook audiobook,\n        AudiobookRepository repository\n    ) {\n        this.audiobook = audiobook;\n        this.repository = repository;\n    }\n}\n",[3168,8946,8947,8954,8960,8964,8974,8986,8998,9002,9008,9016,9031,9039,9044,9054,9064,9068],{"__ignoreMap":3170},[3186,8948,8949,8951],{"class":3188,"line":3189},[3186,8950,4029],{"class":3203},[3186,8952,8953],{"class":3192}," com.google.inject.assistedinject.Assisted;\n",[3186,8955,8956,8958],{"class":3188,"line":3200},[3186,8957,4029],{"class":3203},[3186,8959,5758],{"class":3192},[3186,8961,8962],{"class":3188,"line":3223},[3186,8963,4024],{"emptyLinePlaceholder":4023},[3186,8965,8966,8968,8970,8972],{"class":3188,"line":3230},[3186,8967,3204],{"class":3203},[3186,8969,4124],{"class":3203},[3186,8971,8833],{"class":3196},[3186,8973,4135],{"class":3192},[3186,8975,8976,8978,8980,8982,8984],{"class":3188,"line":3253},[3186,8977,4931],{"class":3203},[3186,8979,5099],{"class":3203},[3186,8981,8844],{"class":3196},[3186,8983,8847],{"class":3236},[3186,8985,5106],{"class":3192},[3186,8987,8988,8990,8992,8994,8996],{"class":3188,"line":3274},[3186,8989,4931],{"class":3203},[3186,8991,5099],{"class":3203},[3186,8993,7524],{"class":3196},[3186,8995,7527],{"class":3236},[3186,8997,5106],{"class":3192},[3186,8999,9000],{"class":3188,"line":3291},[3186,9001,3330],{"class":3192},[3186,9003,9004,9006],{"class":3188,"line":3308},[3186,9005,4144],{"class":3192},[3186,9007,5117],{"class":3196},[3186,9009,9010,9012,9014],{"class":3188,"line":3327},[3186,9011,4656],{"class":3203},[3186,9013,8833],{"class":3210},[3186,9015,3421],{"class":3192},[3186,9017,9018,9021,9024,9026,9028],{"class":3188,"line":3333},[3186,9019,9020],{"class":3192},"        @",[3186,9022,9023],{"class":3196},"Assisted",[3186,9025,8844],{"class":3196},[3186,9027,8847],{"class":3236},[3186,9029,9030],{"class":3192},",\n",[3186,9032,9033,9036],{"class":3188,"line":3339},[3186,9034,9035],{"class":3196},"        AudiobookRepository",[3186,9037,9038],{"class":3236}," repository\n",[3186,9040,9041],{"class":3188,"line":3358},[3186,9042,9043],{"class":3192},"    ) {\n",[3186,9045,9046,9048,9050,9052],{"class":3188,"line":3376},[3186,9047,5136],{"class":3203},[3186,9049,3259],{"class":3192},[3186,9051,8894],{"class":3236},[3186,9053,8897],{"class":3192},[3186,9055,9056,9058,9060,9062],{"class":3188,"line":3394},[3186,9057,5136],{"class":3203},[3186,9059,3259],{"class":3192},[3186,9061,7562],{"class":3236},[3186,9063,7565],{"class":3192},[3186,9065,9066],{"class":3188,"line":3399},[3186,9067,4356],{"class":3192},[3186,9069,9070],{"class":3188,"line":3405},[3186,9071,3661],{"class":3192},[3150,9073,9074,9077],{},[3168,9075,9076],{},"@Assisted"," означає: \"цей параметр буде переданий вручну, не резолви його через Guice\".",[3150,9079,9080],{},[3154,9081,9082],{},"Крок 2: Створення Factory інтерфейсу",[3161,9084,9086],{"className":3180,"code":9085,"language":3182,"meta":3170,"style":3170},"public interface AudiobookDetailViewModelFactory {\n    AudiobookDetailViewModel create(Audiobook audiobook);\n}\n",[3168,9087,9088,9099,9115],{"__ignoreMap":3170},[3186,9089,9090,9092,9094,9097],{"class":3188,"line":3189},[3186,9091,3204],{"class":3203},[3186,9093,5241],{"class":3203},[3186,9095,9096],{"class":3196}," AudiobookDetailViewModelFactory",[3186,9098,4135],{"class":3192},[3186,9100,9101,9104,9107,9109,9111,9113],{"class":3188,"line":3200},[3186,9102,9103],{"class":3196},"    AudiobookDetailViewModel",[3186,9105,9106],{"class":3210}," create",[3186,9108,3214],{"class":3192},[3186,9110,8818],{"class":3196},[3186,9112,8847],{"class":3236},[3186,9114,3271],{"class":3192},[3186,9116,9117],{"class":3188,"line":3223},[3186,9118,3661],{"class":3192},[3150,9120,9121],{},"Guice автоматично створить реалізацію цього інтерфейсу.",[3150,9123,9124],{},[3154,9125,9126],{},"Крок 3: Реєстрація Factory у Module",[3161,9128,9130],{"className":3180,"code":9129,"language":3182,"meta":3170,"style":3170},"import com.google.inject.assistedinject.FactoryModuleBuilder;\n\n@Override\nprotected void configure() {\n    \u002F\u002F ... інші bindings\n    \n    install(new FactoryModuleBuilder()\n        .build(AudiobookDetailViewModelFactory.class));\n}\n",[3168,9131,9132,9139,9143,9149,9159,9164,9168,9182,9200],{"__ignoreMap":3170},[3186,9133,9134,9136],{"class":3188,"line":3189},[3186,9135,4029],{"class":3203},[3186,9137,9138],{"class":3192}," com.google.inject.assistedinject.FactoryModuleBuilder;\n",[3186,9140,9141],{"class":3188,"line":3200},[3186,9142,4024],{"emptyLinePlaceholder":4023},[3186,9144,9145,9147],{"class":3188,"line":3223},[3186,9146,3193],{"class":3192},[3186,9148,3197],{"class":3196},[3186,9150,9151,9153,9155,9157],{"class":3188,"line":3230},[3186,9152,4985],{"class":3203},[3186,9154,3207],{"class":3196},[3186,9156,4156],{"class":3210},[3186,9158,4159],{"class":3192},[3186,9160,9161],{"class":3188,"line":3253},[3186,9162,9163],{"class":3226},"    \u002F\u002F ... інші bindings\n",[3186,9165,9166],{"class":3188,"line":3274},[3186,9167,3330],{"class":3192},[3186,9169,9170,9173,9175,9177,9180],{"class":3188,"line":3291},[3186,9171,9172],{"class":3210},"    install",[3186,9174,3214],{"class":3192},[3186,9176,3244],{"class":3243},[3186,9178,9179],{"class":3210}," FactoryModuleBuilder",[3186,9181,4861],{"class":3192},[3186,9183,9184,9186,9189,9191,9194,9196,9198],{"class":3188,"line":3308},[3186,9185,5011],{"class":3192},[3186,9187,9188],{"class":3210},"build",[3186,9190,3214],{"class":3192},[3186,9192,9193],{"class":3236},"AudiobookDetailViewModelFactory",[3186,9195,3259],{"class":3192},[3186,9197,4189],{"class":3236},[3186,9199,3521],{"class":3192},[3186,9201,9202],{"class":3188,"line":3327},[3186,9203,3661],{"class":3192},[3150,9205,9206],{},[3154,9207,9208],{},"Крок 4: Використання Factory",[3161,9210,9212],{"className":3180,"code":9211,"language":3182,"meta":3170,"style":3170},"public class AudiobookListController {\n    private final AudiobookDetailViewModelFactory detailViewModelFactory;\n    \n    @Inject\n    public AudiobookListController(AudiobookDetailViewModelFactory factory) {\n        this.detailViewModelFactory = factory;\n    }\n    \n    private void onShowDetails(Audiobook audiobook) {\n        AudiobookDetailViewModel viewModel = detailViewModelFactory.create(audiobook);\n        \u002F\u002F Відкрити діалог з цим ViewModel\n    }\n}\n",[3168,9213,9214,9224,9237,9241,9247,9262,9274,9278,9282,9299,9318,9323,9327],{"__ignoreMap":3170},[3186,9215,9216,9218,9220,9222],{"class":3188,"line":3189},[3186,9217,3204],{"class":3203},[3186,9219,4124],{"class":3203},[3186,9221,5090],{"class":3196},[3186,9223,4135],{"class":3192},[3186,9225,9226,9228,9230,9232,9235],{"class":3188,"line":3200},[3186,9227,4931],{"class":3203},[3186,9229,5099],{"class":3203},[3186,9231,9096],{"class":3196},[3186,9233,9234],{"class":3236}," detailViewModelFactory",[3186,9236,5106],{"class":3192},[3186,9238,9239],{"class":3188,"line":3223},[3186,9240,3330],{"class":3192},[3186,9242,9243,9245],{"class":3188,"line":3230},[3186,9244,4144],{"class":3192},[3186,9246,5117],{"class":3196},[3186,9248,9249,9251,9253,9255,9257,9260],{"class":3188,"line":3253},[3186,9250,4656],{"class":3203},[3186,9252,5090],{"class":3210},[3186,9254,3214],{"class":3192},[3186,9256,9193],{"class":3196},[3186,9258,9259],{"class":3236}," factory",[3186,9261,4948],{"class":3192},[3186,9263,9264,9266,9268,9271],{"class":3188,"line":3274},[3186,9265,5136],{"class":3203},[3186,9267,3259],{"class":3192},[3186,9269,9270],{"class":3236},"detailViewModelFactory",[3186,9272,9273],{"class":3192}," = factory;\n",[3186,9275,9276],{"class":3188,"line":3291},[3186,9277,4356],{"class":3192},[3186,9279,9280],{"class":3188,"line":3308},[3186,9281,3330],{"class":3192},[3186,9283,9284,9286,9288,9291,9293,9295,9297],{"class":3188,"line":3327},[3186,9285,4931],{"class":3203},[3186,9287,3207],{"class":3196},[3186,9289,9290],{"class":3210}," onShowDetails",[3186,9292,3214],{"class":3192},[3186,9294,8818],{"class":3196},[3186,9296,8847],{"class":3236},[3186,9298,4948],{"class":3192},[3186,9300,9301,9304,9306,9308,9310,9312,9315],{"class":3188,"line":3333},[3186,9302,9303],{"class":3196},"        AudiobookDetailViewModel",[3186,9305,3465],{"class":3236},[3186,9307,3240],{"class":3192},[3186,9309,9270],{"class":3236},[3186,9311,3259],{"class":3192},[3186,9313,9314],{"class":3210},"create",[3186,9316,9317],{"class":3192},"(audiobook);\n",[3186,9319,9320],{"class":3188,"line":3339},[3186,9321,9322],{"class":3226},"        \u002F\u002F Відкрити діалог з цим ViewModel\n",[3186,9324,9325],{"class":3188,"line":3358},[3186,9326,4356],{"class":3192},[3186,9328,9329],{"class":3188,"line":3376},[3186,9330,3661],{"class":3192},[3150,9332,9333,9334,9336,9337,9339,9340,9342,9343,3259],{},"Guice автоматично ін'єктує ",[3168,9335,4184],{}," у ",[3168,9338,8814],{},", а ",[3168,9341,8894],{}," передається через ",[3168,9344,9345],{},"factory.create(audiobook)",[3981,9347,9348,9351],{},[3154,9349,9350],{},"AssistedInject для діалогів:"," Це ідеальний підхід для ViewModel діалогів, що потребують параметрів (редагування існуючого об'єкта, показ деталей). Factory створює новий ViewModel з параметром, Guice ін'єктує решту залежностей.",[3729,9353],{},[3145,9355,9357],{"id":9356},"повний-приклад-додаток-з-кількома-екранами","Повний приклад: Додаток з кількома екранами",[3150,9359,9360],{},"Розглянемо повний приклад додатку з трьома екранами: список аудіокниг, форма додавання, деталі аудіокниги.",[3739,9362,9364],{"id":9363},"audiobookmodule-повна-конфігурація","AudiobookModule: Повна конфігурація",[3161,9366,9368],{"className":3180,"code":9367,"language":3182,"meta":3170,"style":3170},"package dev.kostyl.audiobook.infrastructure;\n\nimport com.google.inject.AbstractModule;\nimport com.google.inject.Provides;\nimport com.google.inject.Singleton;\nimport com.google.inject.assistedinject.FactoryModuleBuilder;\nimport com.zaxxer.hikari.HikariConfig;\nimport com.zaxxer.hikari.HikariDataSource;\nimport dev.kostyl.audiobook.repository.*;\nimport dev.kostyl.audiobook.repository.jdbc.*;\nimport dev.kostyl.audiobook.service.AudiobookService;\nimport dev.kostyl.audiobook.viewmodel.*;\n\nimport javax.sql.DataSource;\n\npublic class AudiobookModule extends AbstractModule {\n    \n    @Override\n    protected void configure() {\n        \u002F\u002F ===== Data Access Layer =====\n        \n        bind(AudiobookRepository.class).to(JdbcAudiobookRepository.class);\n        bind(AuthorRepository.class).to(JdbcAuthorRepository.class);\n        bind(GenreRepository.class).to(JdbcGenreRepository.class);\n        \n        \u002F\u002F ===== Service Layer =====\n        \n        bind(AudiobookService.class).in(Singleton.class);\n        \n        \u002F\u002F ===== Presentation Layer =====\n        \n        \u002F\u002F Singleton ViewModels (зберігають стан між переходами)\n        bind(AudiobookListViewModel.class).in(Singleton.class);\n        \n        \u002F\u002F Prototype ViewModels (новий екземпляр для кожного відкриття)\n        bind(AudiobookFormViewModel.class); \u002F\u002F Без Singleton\n        \n        \u002F\u002F AssistedInject Factory для параметризованих ViewModel\n        install(new FactoryModuleBuilder()\n            .build(AudiobookDetailViewModelFactory.class));\n    }\n    \n    @Provides\n    @Singleton\n    DataSource provideDataSource() {\n        HikariConfig config = new HikariConfig();\n        config.setJdbcUrl(\"jdbc:h2:mem:audiobook;DB_CLOSE_DELAY=-1\");\n        config.setUsername(\"sa\");\n        config.setPassword(\"\");\n        config.setMaximumPoolSize(10);\n        config.setMinimumIdle(2);\n        config.setConnectionTimeout(30000);\n        config.setIdleTimeout(600000);\n        config.setMaxLifetime(1800000);\n        \n        return new HikariDataSource(config);\n    }\n}\n",[3168,9369,9370,9376,9380,9386,9392,9398,9404,9411,9418,9425,9432,9438,9445,9449,9455,9459,9473,9477,9483,9493,9497,9501,9527,9553,9579,9583,9587,9591,9617,9621,9625,9629,9634,9660,9664,9669,9685,9689,9694,9707,9723,9727,9731,9737,9743,9751,9765,9779,9793,9807,9821,9835,9849,9865,9881,9885,9895,9899],{"__ignoreMap":3170},[3186,9371,9372,9374],{"class":3188,"line":3189},[3186,9373,4015],{"class":3203},[3186,9375,4018],{"class":3192},[3186,9377,9378],{"class":3188,"line":3200},[3186,9379,4024],{"emptyLinePlaceholder":4023},[3186,9381,9382,9384],{"class":3188,"line":3223},[3186,9383,4029],{"class":3203},[3186,9385,4032],{"class":3192},[3186,9387,9388,9390],{"class":3188,"line":3230},[3186,9389,4029],{"class":3203},[3186,9391,4039],{"class":3192},[3186,9393,9394,9396],{"class":3188,"line":3253},[3186,9395,4029],{"class":3203},[3186,9397,4046],{"class":3192},[3186,9399,9400,9402],{"class":3188,"line":3274},[3186,9401,4029],{"class":3203},[3186,9403,9138],{"class":3192},[3186,9405,9406,9408],{"class":3188,"line":3291},[3186,9407,4029],{"class":3203},[3186,9409,9410],{"class":3192}," com.zaxxer.hikari.HikariConfig;\n",[3186,9412,9413,9415],{"class":3188,"line":3308},[3186,9414,4029],{"class":3203},[3186,9416,9417],{"class":3192}," com.zaxxer.hikari.HikariDataSource;\n",[3186,9419,9420,9422],{"class":3188,"line":3327},[3186,9421,4029],{"class":3203},[3186,9423,9424],{"class":3192}," dev.kostyl.audiobook.repository.*;\n",[3186,9426,9427,9429],{"class":3188,"line":3333},[3186,9428,4029],{"class":3203},[3186,9430,9431],{"class":3192}," dev.kostyl.audiobook.repository.jdbc.*;\n",[3186,9433,9434,9436],{"class":3188,"line":3339},[3186,9435,4029],{"class":3203},[3186,9437,4095],{"class":3192},[3186,9439,9440,9442],{"class":3188,"line":3358},[3186,9441,4029],{"class":3203},[3186,9443,9444],{"class":3192}," dev.kostyl.audiobook.viewmodel.*;\n",[3186,9446,9447],{"class":3188,"line":3376},[3186,9448,4024],{"emptyLinePlaceholder":4023},[3186,9450,9451,9453],{"class":3188,"line":3394},[3186,9452,4029],{"class":3203},[3186,9454,4113],{"class":3192},[3186,9456,9457],{"class":3188,"line":3399},[3186,9458,4024],{"emptyLinePlaceholder":4023},[3186,9460,9461,9463,9465,9467,9469,9471],{"class":3188,"line":3405},[3186,9462,3204],{"class":3203},[3186,9464,4124],{"class":3203},[3186,9466,3778],{"class":3196},[3186,9468,4129],{"class":3203},[3186,9470,4132],{"class":3196},[3186,9472,4135],{"class":3192},[3186,9474,9475],{"class":3188,"line":3424},[3186,9476,3330],{"class":3192},[3186,9478,9479,9481],{"class":3188,"line":3430},[3186,9480,4144],{"class":3192},[3186,9482,3197],{"class":3196},[3186,9484,9485,9487,9489,9491],{"class":3188,"line":3436},[3186,9486,4151],{"class":3203},[3186,9488,3207],{"class":3196},[3186,9490,4156],{"class":3210},[3186,9492,4159],{"class":3192},[3186,9494,9495],{"class":3188,"line":3442},[3186,9496,4164],{"class":3226},[3186,9498,9499],{"class":3188,"line":3448},[3186,9500,4169],{"class":3192},[3186,9502,9503,9505,9507,9509,9511,9513,9515,9517,9519,9521,9523,9525],{"class":3188,"line":3453},[3186,9504,4179],{"class":3210},[3186,9506,3214],{"class":3192},[3186,9508,4184],{"class":3236},[3186,9510,3259],{"class":3192},[3186,9512,4189],{"class":3236},[3186,9514,4192],{"class":3192},[3186,9516,4195],{"class":3210},[3186,9518,3214],{"class":3192},[3186,9520,3673],{"class":3236},[3186,9522,3259],{"class":3192},[3186,9524,4189],{"class":3236},[3186,9526,3271],{"class":3192},[3186,9528,9529,9531,9533,9535,9537,9539,9541,9543,9545,9547,9549,9551],{"class":3188,"line":3459},[3186,9530,4179],{"class":3210},[3186,9532,3214],{"class":3192},[3186,9534,4214],{"class":3236},[3186,9536,3259],{"class":3192},[3186,9538,4189],{"class":3236},[3186,9540,4192],{"class":3192},[3186,9542,4195],{"class":3210},[3186,9544,3214],{"class":3192},[3186,9546,4227],{"class":3236},[3186,9548,3259],{"class":3192},[3186,9550,4189],{"class":3236},[3186,9552,3271],{"class":3192},[3186,9554,9555,9557,9559,9561,9563,9565,9567,9569,9571,9573,9575,9577],{"class":3188,"line":3478},[3186,9556,4179],{"class":3210},[3186,9558,3214],{"class":3192},[3186,9560,4242],{"class":3236},[3186,9562,3259],{"class":3192},[3186,9564,4189],{"class":3236},[3186,9566,4192],{"class":3192},[3186,9568,4195],{"class":3210},[3186,9570,3214],{"class":3192},[3186,9572,4255],{"class":3236},[3186,9574,3259],{"class":3192},[3186,9576,4189],{"class":3236},[3186,9578,3271],{"class":3192},[3186,9580,9581],{"class":3188,"line":3483},[3186,9582,4169],{"class":3192},[3186,9584,9585],{"class":3188,"line":3489},[3186,9586,4270],{"class":3226},[3186,9588,9589],{"class":3188,"line":3524},[3186,9590,4169],{"class":3192},[3186,9592,9593,9595,9597,9599,9601,9603,9605,9607,9609,9611,9613,9615],{"class":3188,"line":3545},[3186,9594,4179],{"class":3210},[3186,9596,3214],{"class":3192},[3186,9598,3862],{"class":3236},[3186,9600,3259],{"class":3192},[3186,9602,4189],{"class":3236},[3186,9604,4192],{"class":3192},[3186,9606,4296],{"class":3210},[3186,9608,3214],{"class":3192},[3186,9610,4301],{"class":3236},[3186,9612,3259],{"class":3192},[3186,9614,4189],{"class":3236},[3186,9616,3271],{"class":3192},[3186,9618,9619],{"class":3188,"line":3550},[3186,9620,4169],{"class":3192},[3186,9622,9623],{"class":3188,"line":3556},[3186,9624,4316],{"class":3226},[3186,9626,9627],{"class":3188,"line":3576},[3186,9628,4169],{"class":3192},[3186,9630,9631],{"class":3188,"line":3590},[3186,9632,9633],{"class":3226},"        \u002F\u002F Singleton ViewModels (зберігають стан між переходами)\n",[3186,9635,9636,9638,9640,9642,9644,9646,9648,9650,9652,9654,9656,9658],{"class":3188,"line":3595},[3186,9637,4179],{"class":3210},[3186,9639,3214],{"class":3192},[3186,9641,3705],{"class":3236},[3186,9643,3259],{"class":3192},[3186,9645,4189],{"class":3236},[3186,9647,4192],{"class":3192},[3186,9649,4296],{"class":3210},[3186,9651,3214],{"class":3192},[3186,9653,4301],{"class":3236},[3186,9655,3259],{"class":3192},[3186,9657,4189],{"class":3236},[3186,9659,3271],{"class":3192},[3186,9661,9662],{"class":3188,"line":3601},[3186,9663,4169],{"class":3192},[3186,9665,9666],{"class":3188,"line":3632},[3186,9667,9668],{"class":3226},"        \u002F\u002F Prototype ViewModels (новий екземпляр для кожного відкриття)\n",[3186,9670,9671,9673,9675,9677,9679,9681,9683],{"class":3188,"line":3646},[3186,9672,4179],{"class":3210},[3186,9674,3214],{"class":3192},[3186,9676,7639],{"class":3236},[3186,9678,3259],{"class":3192},[3186,9680,4189],{"class":3236},[3186,9682,7812],{"class":3192},[3186,9684,7815],{"class":3226},[3186,9686,9687],{"class":3188,"line":3658},[3186,9688,4169],{"class":3192},[3186,9690,9691],{"class":3188,"line":4359},[3186,9692,9693],{"class":3226},"        \u002F\u002F AssistedInject Factory для параметризованих ViewModel\n",[3186,9695,9696,9699,9701,9703,9705],{"class":3188,"line":4364},[3186,9697,9698],{"class":3210},"        install",[3186,9700,3214],{"class":3192},[3186,9702,3244],{"class":3243},[3186,9704,9179],{"class":3210},[3186,9706,4861],{"class":3192},[3186,9708,9709,9711,9713,9715,9717,9719,9721],{"class":3188,"line":4370},[3186,9710,4866],{"class":3192},[3186,9712,9188],{"class":3210},[3186,9714,3214],{"class":3192},[3186,9716,9193],{"class":3236},[3186,9718,3259],{"class":3192},[3186,9720,4189],{"class":3236},[3186,9722,3521],{"class":3192},[3186,9724,9725],{"class":3188,"line":4375},[3186,9726,4356],{"class":3192},[3186,9728,9729],{"class":3188,"line":4383},[3186,9730,3330],{"class":3192},[3186,9732,9733,9735],{"class":3188,"line":4391},[3186,9734,4144],{"class":3192},[3186,9736,4380],{"class":3196},[3186,9738,9739,9741],{"class":3188,"line":4401},[3186,9740,4144],{"class":3192},[3186,9742,4388],{"class":3196},[3186,9744,9745,9747,9749],{"class":3188,"line":4417},[3186,9746,3311],{"class":3196},[3186,9748,4396],{"class":3210},[3186,9750,4159],{"class":3192},[3186,9752,9753,9755,9757,9759,9761,9763],{"class":3188,"line":4434},[3186,9754,4404],{"class":3196},[3186,9756,3237],{"class":3236},[3186,9758,3240],{"class":3192},[3186,9760,3244],{"class":3243},[3186,9762,3247],{"class":3210},[3186,9764,3250],{"class":3192},[3186,9766,9767,9769,9771,9773,9775,9777],{"class":3188,"line":4449},[3186,9768,4420],{"class":3236},[3186,9770,3259],{"class":3192},[3186,9772,3262],{"class":3210},[3186,9774,3214],{"class":3192},[3186,9776,4429],{"class":3267},[3186,9778,3271],{"class":3192},[3186,9780,9781,9783,9785,9787,9789,9791],{"class":3188,"line":4464},[3186,9782,4420],{"class":3236},[3186,9784,3259],{"class":3192},[3186,9786,3281],{"class":3210},[3186,9788,3214],{"class":3192},[3186,9790,3286],{"class":3267},[3186,9792,3271],{"class":3192},[3186,9794,9795,9797,9799,9801,9803,9805],{"class":3188,"line":4481},[3186,9796,4420],{"class":3236},[3186,9798,3259],{"class":3192},[3186,9800,3298],{"class":3210},[3186,9802,3214],{"class":3192},[3186,9804,3303],{"class":3267},[3186,9806,3271],{"class":3192},[3186,9808,9809,9811,9813,9815,9817,9819],{"class":3188,"line":4498},[3186,9810,4420],{"class":3236},[3186,9812,3259],{"class":3192},[3186,9814,4471],{"class":3210},[3186,9816,3214],{"class":3192},[3186,9818,4476],{"class":3620},[3186,9820,3271],{"class":3192},[3186,9822,9823,9825,9827,9829,9831,9833],{"class":3188,"line":4515},[3186,9824,4420],{"class":3236},[3186,9826,3259],{"class":3192},[3186,9828,4488],{"class":3210},[3186,9830,3214],{"class":3192},[3186,9832,4493],{"class":3620},[3186,9834,3271],{"class":3192},[3186,9836,9837,9839,9841,9843,9845,9847],{"class":3188,"line":4520},[3186,9838,4420],{"class":3236},[3186,9840,3259],{"class":3192},[3186,9842,4505],{"class":3210},[3186,9844,3214],{"class":3192},[3186,9846,4510],{"class":3620},[3186,9848,3271],{"class":3192},[3186,9850,9851,9853,9855,9858,9860,9863],{"class":3188,"line":4533},[3186,9852,4420],{"class":3236},[3186,9854,3259],{"class":3192},[3186,9856,9857],{"class":3210},"setIdleTimeout",[3186,9859,3214],{"class":3192},[3186,9861,9862],{"class":3620},"600000",[3186,9864,3271],{"class":3192},[3186,9866,9867,9869,9871,9874,9876,9879],{"class":3188,"line":4538},[3186,9868,4420],{"class":3236},[3186,9870,3259],{"class":3192},[3186,9872,9873],{"class":3210},"setMaxLifetime",[3186,9875,3214],{"class":3192},[3186,9877,9878],{"class":3620},"1800000",[3186,9880,3271],{"class":3192},[3186,9882,9883],{"class":3188,"line":6267},[3186,9884,4169],{"class":3192},[3186,9886,9887,9889,9891,9893],{"class":3188,"line":6285},[3186,9888,4523],{"class":3243},[3186,9890,4526],{"class":3243},[3186,9892,3321],{"class":3210},[3186,9894,3324],{"class":3192},[3186,9896,9897],{"class":3188,"line":6301},[3186,9898,4356],{"class":3192},[3186,9900,9901],{"class":3188,"line":6317},[3186,9902,3661],{"class":3192},[3739,9904,9906],{"id":9905},"порівняння-до-та-після-guice","Порівняння: До та після Guice",[7032,9908,9909,9912,10182,10187,10201,10204,10350,10354],{},[3150,9910,9911],{},"== Без Guice (ручне створення)",[3161,9913,9915],{"className":3180,"code":9914,"language":3182,"meta":3170,"style":3170},"@Override\npublic void start(Stage primaryStage) {\n    \u002F\u002F Створення всіх залежностей вручну\n    DataSource dataSource = createDataSource();\n    \n    AudiobookRepository audiobookRepo = new JdbcAudiobookRepository(dataSource);\n    AuthorRepository authorRepo = new JdbcAuthorRepository(dataSource);\n    GenreRepository genreRepo = new JdbcGenreRepository(dataSource);\n    \n    AudiobookService service = new AudiobookService(\n        audiobookRepo, authorRepo, genreRepo\n    );\n    \n    AudiobookListViewModel viewModel = new AudiobookListViewModel(service);\n    \n    \u002F\u002F Завантаження FXML\n    FXMLLoader loader = new FXMLLoader(getClass().getResource(\"audiobook-list-view.fxml\"));\n    Parent root = loader.load();\n    \n    \u002F\u002F Передача ViewModel у Controller\n    AudiobookListController controller = loader.getController();\n    controller.setViewModel(viewModel);\n    \n    \u002F\u002F Налаштування Scene\n    Scene scene = new Scene(root);\n    primaryStage.setScene(scene);\n    primaryStage.show();\n}\n",[3168,9916,9917,9923,9938,9943,9956,9960,9975,9990,10005,10009,10024,10029,10033,10037,10052,10056,10060,10086,10102,10106,10110,10126,10136,10140,10144,10158,10168,10178],{"__ignoreMap":3170},[3186,9918,9919,9921],{"class":3188,"line":3189},[3186,9920,3193],{"class":3192},[3186,9922,3197],{"class":3196},[3186,9924,9925,9927,9929,9931,9933,9935],{"class":3188,"line":3200},[3186,9926,3204],{"class":3203},[3186,9928,3207],{"class":3196},[3186,9930,3211],{"class":3210},[3186,9932,3214],{"class":3192},[3186,9934,3217],{"class":3196},[3186,9936,9937],{"class":3192}," primaryStage) {\n",[3186,9939,9940],{"class":3188,"line":3223},[3186,9941,9942],{"class":3226},"    \u002F\u002F Створення всіх залежностей вручну\n",[3186,9944,9945,9947,9949,9951,9954],{"class":3188,"line":3230},[3186,9946,3311],{"class":3196},[3186,9948,3314],{"class":3236},[3186,9950,3240],{"class":3192},[3186,9952,9953],{"class":3210},"createDataSource",[3186,9955,3250],{"class":3192},[3186,9957,9958],{"class":3188,"line":3253},[3186,9959,3330],{"class":3192},[3186,9961,9962,9964,9967,9969,9971,9973],{"class":3188,"line":3274},[3186,9963,3342],{"class":3196},[3186,9965,9966],{"class":3236}," audiobookRepo",[3186,9968,3240],{"class":3192},[3186,9970,3244],{"class":3243},[3186,9972,3352],{"class":3210},[3186,9974,3355],{"class":3192},[3186,9976,9977,9979,9982,9984,9986,9988],{"class":3188,"line":3291},[3186,9978,3361],{"class":3196},[3186,9980,9981],{"class":3236}," authorRepo",[3186,9983,3240],{"class":3192},[3186,9985,3244],{"class":3243},[3186,9987,3371],{"class":3210},[3186,9989,3355],{"class":3192},[3186,9991,9992,9994,9997,9999,10001,10003],{"class":3188,"line":3308},[3186,9993,3379],{"class":3196},[3186,9995,9996],{"class":3236}," genreRepo",[3186,9998,3240],{"class":3192},[3186,10000,3244],{"class":3243},[3186,10002,3389],{"class":3210},[3186,10004,3355],{"class":3192},[3186,10006,10007],{"class":3188,"line":3327},[3186,10008,3330],{"class":3192},[3186,10010,10011,10013,10016,10018,10020,10022],{"class":3188,"line":3333},[3186,10012,3408],{"class":3196},[3186,10014,10015],{"class":3236}," service",[3186,10017,3240],{"class":3192},[3186,10019,3244],{"class":3243},[3186,10021,3418],{"class":3210},[3186,10023,3421],{"class":3192},[3186,10025,10026],{"class":3188,"line":3339},[3186,10027,10028],{"class":3192},"        audiobookRepo, authorRepo, genreRepo\n",[3186,10030,10031],{"class":3188,"line":3358},[3186,10032,3445],{"class":3192},[3186,10034,10035],{"class":3188,"line":3376},[3186,10036,3330],{"class":3192},[3186,10038,10039,10041,10043,10045,10047,10049],{"class":3188,"line":3394},[3186,10040,3462],{"class":3196},[3186,10042,3465],{"class":3236},[3186,10044,3240],{"class":3192},[3186,10046,3244],{"class":3243},[3186,10048,3472],{"class":3210},[3186,10050,10051],{"class":3192},"(service);\n",[3186,10053,10054],{"class":3188,"line":3399},[3186,10055,3330],{"class":3192},[3186,10057,10058],{"class":3188,"line":3405},[3186,10059,3486],{"class":3226},[3186,10061,10062,10064,10066,10068,10070,10072,10074,10076,10078,10080,10082,10084],{"class":3188,"line":3424},[3186,10063,3492],{"class":3196},[3186,10065,3495],{"class":3236},[3186,10067,3240],{"class":3192},[3186,10069,3244],{"class":3243},[3186,10071,3502],{"class":3210},[3186,10073,3214],{"class":3192},[3186,10075,3507],{"class":3210},[3186,10077,3510],{"class":3192},[3186,10079,3513],{"class":3210},[3186,10081,3214],{"class":3192},[3186,10083,3518],{"class":3267},[3186,10085,3521],{"class":3192},[3186,10087,10088,10090,10092,10094,10096,10098,10100],{"class":3188,"line":3430},[3186,10089,3527],{"class":3196},[3186,10091,3530],{"class":3236},[3186,10093,3240],{"class":3192},[3186,10095,3535],{"class":3236},[3186,10097,3259],{"class":3192},[3186,10099,3540],{"class":3210},[3186,10101,3250],{"class":3192},[3186,10103,10104],{"class":3188,"line":3436},[3186,10105,3330],{"class":3192},[3186,10107,10108],{"class":3188,"line":3442},[3186,10109,3553],{"class":3226},[3186,10111,10112,10114,10116,10118,10120,10122,10124],{"class":3188,"line":3448},[3186,10113,3559],{"class":3196},[3186,10115,3562],{"class":3236},[3186,10117,3240],{"class":3192},[3186,10119,3535],{"class":3236},[3186,10121,3259],{"class":3192},[3186,10123,3571],{"class":3210},[3186,10125,3250],{"class":3192},[3186,10127,10128,10130,10132,10134],{"class":3188,"line":3453},[3186,10129,3579],{"class":3236},[3186,10131,3259],{"class":3192},[3186,10133,3584],{"class":3210},[3186,10135,3587],{"class":3192},[3186,10137,10138],{"class":3188,"line":3459},[3186,10139,3330],{"class":3192},[3186,10141,10142],{"class":3188,"line":3478},[3186,10143,3598],{"class":3226},[3186,10145,10146,10148,10150,10152,10154,10156],{"class":3188,"line":3483},[3186,10147,3604],{"class":3196},[3186,10149,3607],{"class":3236},[3186,10151,3240],{"class":3192},[3186,10153,3244],{"class":3243},[3186,10155,3614],{"class":3210},[3186,10157,5698],{"class":3192},[3186,10159,10160,10162,10164,10166],{"class":3188,"line":3489},[3186,10161,3635],{"class":3236},[3186,10163,3259],{"class":3192},[3186,10165,3640],{"class":3210},[3186,10167,3643],{"class":3192},[3186,10169,10170,10172,10174,10176],{"class":3188,"line":3524},[3186,10171,3635],{"class":3236},[3186,10173,3259],{"class":3192},[3186,10175,3653],{"class":3210},[3186,10177,3250],{"class":3192},[3186,10179,10180],{"class":3188,"line":3545},[3186,10181,3661],{"class":3192},[3150,10183,10184],{},[3154,10185,10186],{},"Проблеми:",[7171,10188,10189,10192,10195,10198],{},[3885,10190,10191],{},"20+ рядків коду для створення залежностей.",[3885,10193,10194],{},"Жорстке кодування реалізацій.",[3885,10196,10197],{},"Складно тестувати.",[3885,10199,10200],{},"Дублювання для кожного екрану.",[3150,10202,10203],{},"== З Guice (автоматична ін'єкція)",[3161,10205,10207],{"className":3180,"code":10206,"language":3182,"meta":3170,"style":3170},"@Override\npublic void start(Stage primaryStage) throws IOException {\n    \u002F\u002F Завантаження FXML з ControllerFactory\n    FXMLLoader loader = new FXMLLoader(\n        getClass().getResource(\"\u002Ffxml\u002FAudiobookListView.fxml\")\n    );\n    loader.setControllerFactory(injector::getInstance);\n    \n    Parent root = loader.load();\n    \n    \u002F\u002F Налаштування Scene\n    Scene scene = new Scene(root);\n    primaryStage.setScene(scene);\n    primaryStage.show();\n}\n",[3168,10208,10209,10215,10230,10235,10249,10265,10269,10284,10288,10304,10308,10312,10326,10336,10346],{"__ignoreMap":3170},[3186,10210,10211,10213],{"class":3188,"line":3189},[3186,10212,3193],{"class":3192},[3186,10214,3197],{"class":3196},[3186,10216,10217,10219,10221,10223,10225,10227],{"class":3188,"line":3200},[3186,10218,3204],{"class":3203},[3186,10220,3207],{"class":3196},[3186,10222,3211],{"class":3210},[3186,10224,3214],{"class":3192},[3186,10226,3217],{"class":3196},[3186,10228,10229],{"class":3192}," primaryStage) throws IOException {\n",[3186,10231,10232],{"class":3188,"line":3223},[3186,10233,10234],{"class":3226},"    \u002F\u002F Завантаження FXML з ControllerFactory\n",[3186,10236,10237,10239,10241,10243,10245,10247],{"class":3188,"line":3230},[3186,10238,3492],{"class":3196},[3186,10240,3495],{"class":3236},[3186,10242,3240],{"class":3192},[3186,10244,3244],{"class":3243},[3186,10246,3502],{"class":3210},[3186,10248,3421],{"class":3192},[3186,10250,10251,10254,10256,10258,10260,10263],{"class":3188,"line":3253},[3186,10252,10253],{"class":3210},"        getClass",[3186,10255,3510],{"class":3192},[3186,10257,3513],{"class":3210},[3186,10259,3214],{"class":3192},[3186,10261,10262],{"class":3267},"\"\u002Ffxml\u002FAudiobookListView.fxml\"",[3186,10264,4887],{"class":3192},[3186,10266,10267],{"class":3188,"line":3274},[3186,10268,3445],{"class":3192},[3186,10270,10271,10274,10276,10278,10280,10282],{"class":3188,"line":3291},[3186,10272,10273],{"class":3236},"    loader",[3186,10275,3259],{"class":3192},[3186,10277,5323],{"class":3210},[3186,10279,5326],{"class":3192},[3186,10281,5329],{"class":3243},[3186,10283,5332],{"class":3192},[3186,10285,10286],{"class":3188,"line":3308},[3186,10287,3330],{"class":3192},[3186,10289,10290,10292,10294,10296,10298,10300,10302],{"class":3188,"line":3327},[3186,10291,3527],{"class":3196},[3186,10293,3530],{"class":3236},[3186,10295,3240],{"class":3192},[3186,10297,3535],{"class":3236},[3186,10299,3259],{"class":3192},[3186,10301,3540],{"class":3210},[3186,10303,3250],{"class":3192},[3186,10305,10306],{"class":3188,"line":3333},[3186,10307,3330],{"class":3192},[3186,10309,10310],{"class":3188,"line":3339},[3186,10311,3598],{"class":3226},[3186,10313,10314,10316,10318,10320,10322,10324],{"class":3188,"line":3358},[3186,10315,3604],{"class":3196},[3186,10317,3607],{"class":3236},[3186,10319,3240],{"class":3192},[3186,10321,3244],{"class":3243},[3186,10323,3614],{"class":3210},[3186,10325,5698],{"class":3192},[3186,10327,10328,10330,10332,10334],{"class":3188,"line":3376},[3186,10329,3635],{"class":3236},[3186,10331,3259],{"class":3192},[3186,10333,3640],{"class":3210},[3186,10335,3643],{"class":3192},[3186,10337,10338,10340,10342,10344],{"class":3188,"line":3394},[3186,10339,3635],{"class":3236},[3186,10341,3259],{"class":3192},[3186,10343,3653],{"class":3210},[3186,10345,3250],{"class":3192},[3186,10347,10348],{"class":3188,"line":3399},[3186,10349,3661],{"class":3192},[3150,10351,10352],{},[3154,10353,7306],{},[7171,10355,10356,10359,10362,10365],{},[3885,10357,10358],{},"10 рядків коду замість 20+.",[3885,10360,10361],{},"Guice автоматично резолвить всі залежності.",[3885,10363,10364],{},"Легко замінити реалізації у Module.",[3885,10366,10367],{},"Легко тестувати (mock залежності).",[3729,10369],{},[3145,10371,10373],{"id":10372},"тестування-з-guice-mock-залежностей","Тестування з Guice: Mock залежностей",[3150,10375,10376],{},"Одна з найбільших переваг Dependency Injection — легкість тестування. Ми можемо замінити реальні залежності на mock-об'єкти для unit-тестів.",[3739,10378,10380],{"id":10379},"unit-тестування-viewmodel-з-mock-repository","Unit-тестування ViewModel з mock Repository",[3161,10382,10384],{"className":3180,"code":10383,"language":3182,"meta":3170,"style":3170},"package dev.kostyl.audiobook.viewmodel;\n\nimport dev.kostyl.audiobook.domain.Audiobook;\nimport dev.kostyl.audiobook.repository.AudiobookRepository;\nimport dev.kostyl.audiobook.service.AudiobookService;\nimport org.junit.jupiter.api.BeforeEach;\nimport org.junit.jupiter.api.Test;\nimport org.junit.jupiter.api.extension.ExtendWith;\nimport org.mockito.Mock;\nimport org.mockito.junit.jupiter.MockitoExtension;\n\nimport java.util.List;\n\nimport static org.junit.jupiter.api.Assertions.*;\nimport static org.mockito.Mockito.*;\n\n@ExtendWith(MockitoExtension.class)\nclass AudiobookListViewModelTest {\n    \n    @Mock\n    private AudiobookRepository repository;\n    \n    @Mock\n    private AudiobookService service;\n    \n    private AudiobookListViewModel viewModel;\n    \n    @BeforeEach\n    void setUp() {\n        \u002F\u002F Створення ViewModel з mock залежностями\n        viewModel = new AudiobookListViewModel(service);\n    }\n    \n    @Test\n    void shouldLoadAudiobooks() {\n        \u002F\u002F Given\n        List\u003CAudiobook> audiobooks = List.of(\n            new Audiobook(\"Title 1\", author1, genre1, 3600),\n            new Audiobook(\"Title 2\", author2, genre2, 7200)\n        );\n        when(service.getAllAudiobooks()).thenReturn(audiobooks);\n        \n        \u002F\u002F When\n        viewModel.loadAudiobooks();\n        \n        \u002F\u002F Then\n        assertEquals(2, viewModel.getAudiobooks().size());\n        verify(service).getAllAudiobooks();\n    }\n    \n    @Test\n    void shouldFilterAudiobooksBySearchQuery() {\n        \u002F\u002F Given\n        viewModel.getAudiobooks().addAll(\n            new AudiobookViewModel(audiobook1), \u002F\u002F title: \"1984\"\n            new AudiobookViewModel(audiobook2)  \u002F\u002F title: \"Brave New World\"\n        );\n        \n        \u002F\u002F When\n        viewModel.searchQueryProperty().set(\"1984\");\n        \n        \u002F\u002F Then\n        assertEquals(1, viewModel.getFilteredAudiobooks().size());\n        assertEquals(\"1984\", viewModel.getFilteredAudiobooks().get(0).getTitle());\n    }\n}\n",[3168,10385,10386,10393,10397,10404,10410,10416,10423,10430,10437,10444,10451,10455,10462,10466,10475,10484,10488,10506,10515,10519,10526,10536,10540,10546,10556,10560,10570,10574,10581,10591,10596,10607,10611,10615,10622,10631,10636,10662,10683,10702,10706,10730,10734,10739,10750,10754,10759,10784,10796,10800,10804,10810,10819,10823,10838,10851,10863,10867,10871,10875,10895,10899,10903,10927,10960,10964],{"__ignoreMap":3170},[3186,10387,10388,10390],{"class":3188,"line":3189},[3186,10389,4015],{"class":3203},[3186,10391,10392],{"class":3192}," dev.kostyl.audiobook.viewmodel;\n",[3186,10394,10395],{"class":3188,"line":3200},[3186,10396,4024],{"emptyLinePlaceholder":4023},[3186,10398,10399,10401],{"class":3188,"line":3223},[3186,10400,4029],{"class":3203},[3186,10402,10403],{"class":3192}," dev.kostyl.audiobook.domain.Audiobook;\n",[3186,10405,10406,10408],{"class":3188,"line":3230},[3186,10407,4029],{"class":3203},[3186,10409,4053],{"class":3192},[3186,10411,10412,10414],{"class":3188,"line":3253},[3186,10413,4029],{"class":3203},[3186,10415,4095],{"class":3192},[3186,10417,10418,10420],{"class":3188,"line":3274},[3186,10419,4029],{"class":3203},[3186,10421,10422],{"class":3192}," org.junit.jupiter.api.BeforeEach;\n",[3186,10424,10425,10427],{"class":3188,"line":3291},[3186,10426,4029],{"class":3203},[3186,10428,10429],{"class":3192}," org.junit.jupiter.api.Test;\n",[3186,10431,10432,10434],{"class":3188,"line":3308},[3186,10433,4029],{"class":3203},[3186,10435,10436],{"class":3192}," org.junit.jupiter.api.extension.ExtendWith;\n",[3186,10438,10439,10441],{"class":3188,"line":3327},[3186,10440,4029],{"class":3203},[3186,10442,10443],{"class":3192}," org.mockito.Mock;\n",[3186,10445,10446,10448],{"class":3188,"line":3333},[3186,10447,4029],{"class":3203},[3186,10449,10450],{"class":3192}," org.mockito.junit.jupiter.MockitoExtension;\n",[3186,10452,10453],{"class":3188,"line":3339},[3186,10454,4024],{"emptyLinePlaceholder":4023},[3186,10456,10457,10459],{"class":3188,"line":3358},[3186,10458,4029],{"class":3203},[3186,10460,10461],{"class":3192}," java.util.List;\n",[3186,10463,10464],{"class":3188,"line":3376},[3186,10465,4024],{"emptyLinePlaceholder":4023},[3186,10467,10468,10470,10472],{"class":3188,"line":3394},[3186,10469,4029],{"class":3203},[3186,10471,8646],{"class":3203},[3186,10473,10474],{"class":3192}," org.junit.jupiter.api.Assertions.*;\n",[3186,10476,10477,10479,10481],{"class":3188,"line":3399},[3186,10478,4029],{"class":3203},[3186,10480,8646],{"class":3203},[3186,10482,10483],{"class":3192}," org.mockito.Mockito.*;\n",[3186,10485,10486],{"class":3188,"line":3405},[3186,10487,4024],{"emptyLinePlaceholder":4023},[3186,10489,10490,10492,10495,10497,10500,10502,10504],{"class":3188,"line":3424},[3186,10491,3193],{"class":3192},[3186,10493,10494],{"class":3196},"ExtendWith",[3186,10496,3214],{"class":3192},[3186,10498,10499],{"class":3236},"MockitoExtension",[3186,10501,3259],{"class":3192},[3186,10503,4189],{"class":3236},[3186,10505,4887],{"class":3192},[3186,10507,10508,10510,10513],{"class":3188,"line":3430},[3186,10509,4189],{"class":3203},[3186,10511,10512],{"class":3196}," AudiobookListViewModelTest",[3186,10514,4135],{"class":3192},[3186,10516,10517],{"class":3188,"line":3436},[3186,10518,3330],{"class":3192},[3186,10520,10521,10523],{"class":3188,"line":3442},[3186,10522,4144],{"class":3192},[3186,10524,10525],{"class":3196},"Mock\n",[3186,10527,10528,10530,10532,10534],{"class":3188,"line":3448},[3186,10529,4931],{"class":3203},[3186,10531,7524],{"class":3196},[3186,10533,7527],{"class":3236},[3186,10535,5106],{"class":3192},[3186,10537,10538],{"class":3188,"line":3453},[3186,10539,3330],{"class":3192},[3186,10541,10542,10544],{"class":3188,"line":3459},[3186,10543,4144],{"class":3192},[3186,10545,10525],{"class":3196},[3186,10547,10548,10550,10552,10554],{"class":3188,"line":3478},[3186,10549,4931],{"class":3203},[3186,10551,3418],{"class":3196},[3186,10553,10015],{"class":3236},[3186,10555,5106],{"class":3192},[3186,10557,10558],{"class":3188,"line":3483},[3186,10559,3330],{"class":3192},[3186,10561,10562,10564,10566,10568],{"class":3188,"line":3489},[3186,10563,4931],{"class":3203},[3186,10565,3472],{"class":3196},[3186,10567,3465],{"class":3236},[3186,10569,5106],{"class":3192},[3186,10571,10572],{"class":3188,"line":3524},[3186,10573,3330],{"class":3192},[3186,10575,10576,10578],{"class":3188,"line":3545},[3186,10577,4144],{"class":3192},[3186,10579,10580],{"class":3196},"BeforeEach\n",[3186,10582,10583,10586,10589],{"class":3188,"line":3550},[3186,10584,10585],{"class":3196},"    void",[3186,10587,10588],{"class":3210}," setUp",[3186,10590,4159],{"class":3192},[3186,10592,10593],{"class":3188,"line":3556},[3186,10594,10595],{"class":3226},"        \u002F\u002F Створення ViewModel з mock залежностями\n",[3186,10597,10598,10601,10603,10605],{"class":3188,"line":3576},[3186,10599,10600],{"class":3192},"        viewModel = ",[3186,10602,3244],{"class":3243},[3186,10604,3472],{"class":3210},[3186,10606,10051],{"class":3192},[3186,10608,10609],{"class":3188,"line":3590},[3186,10610,4356],{"class":3192},[3186,10612,10613],{"class":3188,"line":3595},[3186,10614,3330],{"class":3192},[3186,10616,10617,10619],{"class":3188,"line":3601},[3186,10618,4144],{"class":3192},[3186,10620,10621],{"class":3196},"Test\n",[3186,10623,10624,10626,10629],{"class":3188,"line":3632},[3186,10625,10585],{"class":3196},[3186,10627,10628],{"class":3210}," shouldLoadAudiobooks",[3186,10630,4159],{"class":3192},[3186,10632,10633],{"class":3188,"line":3646},[3186,10634,10635],{"class":3226},"        \u002F\u002F Given\n",[3186,10637,10638,10641,10643,10645,10647,10650,10652,10655,10657,10660],{"class":3188,"line":3658},[3186,10639,10640],{"class":3196},"        List",[3186,10642,4635],{"class":3192},[3186,10644,8818],{"class":3196},[3186,10646,5536],{"class":3192},[3186,10648,10649],{"class":3236},"audiobooks",[3186,10651,3240],{"class":3192},[3186,10653,10654],{"class":3236},"List",[3186,10656,3259],{"class":3192},[3186,10658,10659],{"class":3210},"of",[3186,10661,3421],{"class":3192},[3186,10663,10664,10667,10669,10671,10674,10677,10680],{"class":3188,"line":4359},[3186,10665,10666],{"class":3243},"            new",[3186,10668,8844],{"class":3210},[3186,10670,3214],{"class":3192},[3186,10672,10673],{"class":3267},"\"Title 1\"",[3186,10675,10676],{"class":3192},", author1, genre1, ",[3186,10678,10679],{"class":3620},"3600",[3186,10681,10682],{"class":3192},"),\n",[3186,10684,10685,10687,10689,10691,10694,10697,10700],{"class":3188,"line":4364},[3186,10686,10666],{"class":3243},[3186,10688,8844],{"class":3210},[3186,10690,3214],{"class":3192},[3186,10692,10693],{"class":3267},"\"Title 2\"",[3186,10695,10696],{"class":3192},", author2, genre2, ",[3186,10698,10699],{"class":3620},"7200",[3186,10701,4887],{"class":3192},[3186,10703,10704],{"class":3188,"line":4370},[3186,10705,8108],{"class":3192},[3186,10707,10708,10711,10713,10716,10718,10721,10724,10727],{"class":3188,"line":4375},[3186,10709,10710],{"class":3210},"        when",[3186,10712,3214],{"class":3192},[3186,10714,10715],{"class":3236},"service",[3186,10717,3259],{"class":3192},[3186,10719,10720],{"class":3210},"getAllAudiobooks",[3186,10722,10723],{"class":3192},"()).",[3186,10725,10726],{"class":3210},"thenReturn",[3186,10728,10729],{"class":3192},"(audiobooks);\n",[3186,10731,10732],{"class":3188,"line":4383},[3186,10733,4169],{"class":3192},[3186,10735,10736],{"class":3188,"line":4391},[3186,10737,10738],{"class":3226},"        \u002F\u002F When\n",[3186,10740,10741,10743,10745,10748],{"class":3188,"line":4401},[3186,10742,6210],{"class":3236},[3186,10744,3259],{"class":3192},[3186,10746,10747],{"class":3210},"loadAudiobooks",[3186,10749,3250],{"class":3192},[3186,10751,10752],{"class":3188,"line":4417},[3186,10753,4169],{"class":3192},[3186,10755,10756],{"class":3188,"line":4434},[3186,10757,10758],{"class":3226},"        \u002F\u002F Then\n",[3186,10760,10761,10764,10766,10768,10770,10772,10774,10777,10779,10782],{"class":3188,"line":4449},[3186,10762,10763],{"class":3210},"        assertEquals",[3186,10765,3214],{"class":3192},[3186,10767,4493],{"class":3620},[3186,10769,3624],{"class":3192},[3186,10771,5141],{"class":3236},[3186,10773,3259],{"class":3192},[3186,10775,10776],{"class":3210},"getAudiobooks",[3186,10778,3510],{"class":3192},[3186,10780,10781],{"class":3210},"size",[3186,10783,3781],{"class":3192},[3186,10785,10786,10789,10792,10794],{"class":3188,"line":4464},[3186,10787,10788],{"class":3210},"        verify",[3186,10790,10791],{"class":3192},"(service).",[3186,10793,10720],{"class":3210},[3186,10795,3250],{"class":3192},[3186,10797,10798],{"class":3188,"line":4481},[3186,10799,4356],{"class":3192},[3186,10801,10802],{"class":3188,"line":4498},[3186,10803,3330],{"class":3192},[3186,10805,10806,10808],{"class":3188,"line":4515},[3186,10807,4144],{"class":3192},[3186,10809,10621],{"class":3196},[3186,10811,10812,10814,10817],{"class":3188,"line":4520},[3186,10813,10585],{"class":3196},[3186,10815,10816],{"class":3210}," shouldFilterAudiobooksBySearchQuery",[3186,10818,4159],{"class":3192},[3186,10820,10821],{"class":3188,"line":4533},[3186,10822,10635],{"class":3226},[3186,10824,10825,10827,10829,10831,10833,10836],{"class":3188,"line":4538},[3186,10826,6210],{"class":3236},[3186,10828,3259],{"class":3192},[3186,10830,10776],{"class":3210},[3186,10832,3510],{"class":3192},[3186,10834,10835],{"class":3210},"addAll",[3186,10837,3421],{"class":3192},[3186,10839,10840,10842,10845,10848],{"class":3188,"line":6267},[3186,10841,10666],{"class":3243},[3186,10843,10844],{"class":3210}," AudiobookViewModel",[3186,10846,10847],{"class":3192},"(audiobook1), ",[3186,10849,10850],{"class":3226},"\u002F\u002F title: \"1984\"\n",[3186,10852,10853,10855,10857,10860],{"class":3188,"line":6285},[3186,10854,10666],{"class":3243},[3186,10856,10844],{"class":3210},[3186,10858,10859],{"class":3192},"(audiobook2)  ",[3186,10861,10862],{"class":3226},"\u002F\u002F title: \"Brave New World\"\n",[3186,10864,10865],{"class":3188,"line":6301},[3186,10866,8108],{"class":3192},[3186,10868,10869],{"class":3188,"line":6317},[3186,10870,4169],{"class":3192},[3186,10872,10873],{"class":3188,"line":6333},[3186,10874,10738],{"class":3226},[3186,10876,10877,10879,10881,10883,10885,10888,10890,10893],{"class":3188,"line":6349},[3186,10878,6210],{"class":3236},[3186,10880,3259],{"class":3192},[3186,10882,6499],{"class":3210},[3186,10884,3510],{"class":3192},[3186,10886,10887],{"class":3210},"set",[3186,10889,3214],{"class":3192},[3186,10891,10892],{"class":3267},"\"1984\"",[3186,10894,3271],{"class":3192},[3186,10896,10897],{"class":3188,"line":6365},[3186,10898,4169],{"class":3192},[3186,10900,10901],{"class":3188,"line":6381},[3186,10902,10758],{"class":3226},[3186,10904,10905,10907,10909,10912,10914,10916,10918,10921,10923,10925],{"class":3188,"line":6386},[3186,10906,10763],{"class":3210},[3186,10908,3214],{"class":3192},[3186,10910,10911],{"class":3620},"1",[3186,10913,3624],{"class":3192},[3186,10915,5141],{"class":3236},[3186,10917,3259],{"class":3192},[3186,10919,10920],{"class":3210},"getFilteredAudiobooks",[3186,10922,3510],{"class":3192},[3186,10924,10781],{"class":3210},[3186,10926,3781],{"class":3192},[3186,10928,10929,10931,10933,10935,10937,10939,10941,10943,10945,10948,10950,10953,10955,10958],{"class":3188,"line":6391},[3186,10930,10763],{"class":3210},[3186,10932,3214],{"class":3192},[3186,10934,10892],{"class":3267},[3186,10936,3624],{"class":3192},[3186,10938,5141],{"class":3236},[3186,10940,3259],{"class":3192},[3186,10942,10920],{"class":3210},[3186,10944,3510],{"class":3192},[3186,10946,10947],{"class":3210},"get",[3186,10949,3214],{"class":3192},[3186,10951,10952],{"class":3620},"0",[3186,10954,4192],{"class":3192},[3186,10956,10957],{"class":3210},"getTitle",[3186,10959,3781],{"class":3192},[3186,10961,10962],{"class":3188,"line":6403},[3186,10963,4356],{"class":3192},[3186,10965,10966],{"class":3188,"line":6409},[3186,10967,3661],{"class":3192},[3150,10969,10970],{},[3154,10971,10972],{},"Переваги тестування з DI:",[7171,10974,10975,10978,10981],{},[3885,10976,10977],{},"Не потрібно створювати реальну базу даних для unit-тестів.",[3885,10979,10980],{},"Mock залежності передаються через конструктор — просто та зрозуміло.",[3885,10982,10983],{},"Тести швидкі (без IO-операцій) та ізольовані.",[3739,10985,10987],{"id":10986},"інтеграційне-тестування-з-test-module","Інтеграційне тестування з Test Module",[3150,10989,10990],{},"Для інтеграційних тестів створимо окремий Guice Module з тестовими залежностями:",[3161,10992,10994],{"className":3180,"code":10993,"language":3182,"meta":3170,"style":3170},"package dev.kostyl.audiobook.infrastructure;\n\nimport com.google.inject.AbstractModule;\nimport com.google.inject.Provides;\nimport com.google.inject.Singleton;\nimport dev.kostyl.audiobook.repository.AudiobookRepository;\nimport dev.kostyl.audiobook.repository.inmemory.InMemoryAudiobookRepository;\n\nimport javax.sql.DataSource;\n\npublic class TestAudiobookModule extends AbstractModule {\n    \n    @Override\n    protected void configure() {\n        \u002F\u002F Використовуємо InMemory реалізації для тестів\n        bind(AudiobookRepository.class).to(InMemoryAudiobookRepository.class);\n        \n        \u002F\u002F Решта конфігурації як у production Module\n        bind(AudiobookService.class).in(Singleton.class);\n        bind(AudiobookListViewModel.class).in(Singleton.class);\n    }\n    \n    @Provides\n    @Singleton\n    DataSource provideDataSource() {\n        \u002F\u002F H2 in-memory database для тестів\n        HikariConfig config = new HikariConfig();\n        config.setJdbcUrl(\"jdbc:h2:mem:test;DB_CLOSE_DELAY=-1\");\n        config.setUsername(\"sa\");\n        config.setPassword(\"\");\n        \n        return new HikariDataSource(config);\n    }\n}\n",[3168,10995,10996,11002,11006,11012,11018,11024,11030,11037,11041,11047,11051,11066,11070,11076,11086,11091,11117,11121,11126,11152,11178,11182,11186,11192,11198,11206,11211,11225,11240,11254,11268,11272,11282,11286],{"__ignoreMap":3170},[3186,10997,10998,11000],{"class":3188,"line":3189},[3186,10999,4015],{"class":3203},[3186,11001,4018],{"class":3192},[3186,11003,11004],{"class":3188,"line":3200},[3186,11005,4024],{"emptyLinePlaceholder":4023},[3186,11007,11008,11010],{"class":3188,"line":3223},[3186,11009,4029],{"class":3203},[3186,11011,4032],{"class":3192},[3186,11013,11014,11016],{"class":3188,"line":3230},[3186,11015,4029],{"class":3203},[3186,11017,4039],{"class":3192},[3186,11019,11020,11022],{"class":3188,"line":3253},[3186,11021,4029],{"class":3203},[3186,11023,4046],{"class":3192},[3186,11025,11026,11028],{"class":3188,"line":3274},[3186,11027,4029],{"class":3203},[3186,11029,4053],{"class":3192},[3186,11031,11032,11034],{"class":3188,"line":3291},[3186,11033,4029],{"class":3203},[3186,11035,11036],{"class":3192}," dev.kostyl.audiobook.repository.inmemory.InMemoryAudiobookRepository;\n",[3186,11038,11039],{"class":3188,"line":3308},[3186,11040,4024],{"emptyLinePlaceholder":4023},[3186,11042,11043,11045],{"class":3188,"line":3327},[3186,11044,4029],{"class":3203},[3186,11046,4113],{"class":3192},[3186,11048,11049],{"class":3188,"line":3333},[3186,11050,4024],{"emptyLinePlaceholder":4023},[3186,11052,11053,11055,11057,11060,11062,11064],{"class":3188,"line":3339},[3186,11054,3204],{"class":3203},[3186,11056,4124],{"class":3203},[3186,11058,11059],{"class":3196}," TestAudiobookModule",[3186,11061,4129],{"class":3203},[3186,11063,4132],{"class":3196},[3186,11065,4135],{"class":3192},[3186,11067,11068],{"class":3188,"line":3358},[3186,11069,3330],{"class":3192},[3186,11071,11072,11074],{"class":3188,"line":3376},[3186,11073,4144],{"class":3192},[3186,11075,3197],{"class":3196},[3186,11077,11078,11080,11082,11084],{"class":3188,"line":3394},[3186,11079,4151],{"class":3203},[3186,11081,3207],{"class":3196},[3186,11083,4156],{"class":3210},[3186,11085,4159],{"class":3192},[3186,11087,11088],{"class":3188,"line":3399},[3186,11089,11090],{"class":3226},"        \u002F\u002F Використовуємо InMemory реалізації для тестів\n",[3186,11092,11093,11095,11097,11099,11101,11103,11105,11107,11109,11111,11113,11115],{"class":3188,"line":3405},[3186,11094,4179],{"class":3210},[3186,11096,3214],{"class":3192},[3186,11098,4184],{"class":3236},[3186,11100,3259],{"class":3192},[3186,11102,4189],{"class":3236},[3186,11104,4192],{"class":3192},[3186,11106,4195],{"class":3210},[3186,11108,3214],{"class":3192},[3186,11110,3677],{"class":3236},[3186,11112,3259],{"class":3192},[3186,11114,4189],{"class":3236},[3186,11116,3271],{"class":3192},[3186,11118,11119],{"class":3188,"line":3424},[3186,11120,4169],{"class":3192},[3186,11122,11123],{"class":3188,"line":3430},[3186,11124,11125],{"class":3226},"        \u002F\u002F Решта конфігурації як у production Module\n",[3186,11127,11128,11130,11132,11134,11136,11138,11140,11142,11144,11146,11148,11150],{"class":3188,"line":3436},[3186,11129,4179],{"class":3210},[3186,11131,3214],{"class":3192},[3186,11133,3862],{"class":3236},[3186,11135,3259],{"class":3192},[3186,11137,4189],{"class":3236},[3186,11139,4192],{"class":3192},[3186,11141,4296],{"class":3210},[3186,11143,3214],{"class":3192},[3186,11145,4301],{"class":3236},[3186,11147,3259],{"class":3192},[3186,11149,4189],{"class":3236},[3186,11151,3271],{"class":3192},[3186,11153,11154,11156,11158,11160,11162,11164,11166,11168,11170,11172,11174,11176],{"class":3188,"line":3442},[3186,11155,4179],{"class":3210},[3186,11157,3214],{"class":3192},[3186,11159,3705],{"class":3236},[3186,11161,3259],{"class":3192},[3186,11163,4189],{"class":3236},[3186,11165,4192],{"class":3192},[3186,11167,4296],{"class":3210},[3186,11169,3214],{"class":3192},[3186,11171,4301],{"class":3236},[3186,11173,3259],{"class":3192},[3186,11175,4189],{"class":3236},[3186,11177,3271],{"class":3192},[3186,11179,11180],{"class":3188,"line":3448},[3186,11181,4356],{"class":3192},[3186,11183,11184],{"class":3188,"line":3453},[3186,11185,3330],{"class":3192},[3186,11187,11188,11190],{"class":3188,"line":3459},[3186,11189,4144],{"class":3192},[3186,11191,4380],{"class":3196},[3186,11193,11194,11196],{"class":3188,"line":3478},[3186,11195,4144],{"class":3192},[3186,11197,4388],{"class":3196},[3186,11199,11200,11202,11204],{"class":3188,"line":3483},[3186,11201,3311],{"class":3196},[3186,11203,4396],{"class":3210},[3186,11205,4159],{"class":3192},[3186,11207,11208],{"class":3188,"line":3489},[3186,11209,11210],{"class":3226},"        \u002F\u002F H2 in-memory database для тестів\n",[3186,11212,11213,11215,11217,11219,11221,11223],{"class":3188,"line":3524},[3186,11214,4404],{"class":3196},[3186,11216,3237],{"class":3236},[3186,11218,3240],{"class":3192},[3186,11220,3244],{"class":3243},[3186,11222,3247],{"class":3210},[3186,11224,3250],{"class":3192},[3186,11226,11227,11229,11231,11233,11235,11238],{"class":3188,"line":3545},[3186,11228,4420],{"class":3236},[3186,11230,3259],{"class":3192},[3186,11232,3262],{"class":3210},[3186,11234,3214],{"class":3192},[3186,11236,11237],{"class":3267},"\"jdbc:h2:mem:test;DB_CLOSE_DELAY=-1\"",[3186,11239,3271],{"class":3192},[3186,11241,11242,11244,11246,11248,11250,11252],{"class":3188,"line":3550},[3186,11243,4420],{"class":3236},[3186,11245,3259],{"class":3192},[3186,11247,3281],{"class":3210},[3186,11249,3214],{"class":3192},[3186,11251,3286],{"class":3267},[3186,11253,3271],{"class":3192},[3186,11255,11256,11258,11260,11262,11264,11266],{"class":3188,"line":3556},[3186,11257,4420],{"class":3236},[3186,11259,3259],{"class":3192},[3186,11261,3298],{"class":3210},[3186,11263,3214],{"class":3192},[3186,11265,3303],{"class":3267},[3186,11267,3271],{"class":3192},[3186,11269,11270],{"class":3188,"line":3576},[3186,11271,4169],{"class":3192},[3186,11273,11274,11276,11278,11280],{"class":3188,"line":3590},[3186,11275,4523],{"class":3243},[3186,11277,4526],{"class":3243},[3186,11279,3321],{"class":3210},[3186,11281,3324],{"class":3192},[3186,11283,11284],{"class":3188,"line":3595},[3186,11285,4356],{"class":3192},[3186,11287,11288],{"class":3188,"line":3601},[3186,11289,3661],{"class":3192},[3150,11291,11292],{},"Використання у тестах:",[3161,11294,11296],{"className":3180,"code":11295,"language":3182,"meta":3170,"style":3170},"@ExtendWith(GuiceExtension.class)\n@GuiceModules(TestAudiobookModule.class)\nclass AudiobookListViewModelIntegrationTest {\n    \n    @Inject\n    private AudiobookListViewModel viewModel;\n    \n    @Inject\n    private AudiobookRepository repository;\n    \n    @Test\n    void shouldLoadAudiobooksFromRepository() {\n        \u002F\u002F Given\n        repository.save(audiobook1);\n        repository.save(audiobook2);\n        \n        \u002F\u002F When\n        viewModel.loadAudiobooks();\n        \n        \u002F\u002F Then\n        assertEquals(2, viewModel.getAudiobooks().size());\n    }\n}\n",[3168,11297,11298,11315,11333,11342,11346,11352,11362,11366,11372,11382,11386,11392,11401,11405,11418,11429,11433,11437,11447,11451,11455,11477,11481],{"__ignoreMap":3170},[3186,11299,11300,11302,11304,11306,11309,11311,11313],{"class":3188,"line":3189},[3186,11301,3193],{"class":3192},[3186,11303,10494],{"class":3196},[3186,11305,3214],{"class":3192},[3186,11307,11308],{"class":3236},"GuiceExtension",[3186,11310,3259],{"class":3192},[3186,11312,4189],{"class":3236},[3186,11314,4887],{"class":3192},[3186,11316,11317,11319,11322,11324,11327,11329,11331],{"class":3188,"line":3200},[3186,11318,3193],{"class":3192},[3186,11320,11321],{"class":3196},"GuiceModules",[3186,11323,3214],{"class":3192},[3186,11325,11326],{"class":3236},"TestAudiobookModule",[3186,11328,3259],{"class":3192},[3186,11330,4189],{"class":3236},[3186,11332,4887],{"class":3192},[3186,11334,11335,11337,11340],{"class":3188,"line":3223},[3186,11336,4189],{"class":3203},[3186,11338,11339],{"class":3196}," AudiobookListViewModelIntegrationTest",[3186,11341,4135],{"class":3192},[3186,11343,11344],{"class":3188,"line":3230},[3186,11345,3330],{"class":3192},[3186,11347,11348,11350],{"class":3188,"line":3253},[3186,11349,4144],{"class":3192},[3186,11351,5117],{"class":3196},[3186,11353,11354,11356,11358,11360],{"class":3188,"line":3274},[3186,11355,4931],{"class":3203},[3186,11357,3472],{"class":3196},[3186,11359,3465],{"class":3236},[3186,11361,5106],{"class":3192},[3186,11363,11364],{"class":3188,"line":3291},[3186,11365,3330],{"class":3192},[3186,11367,11368,11370],{"class":3188,"line":3308},[3186,11369,4144],{"class":3192},[3186,11371,5117],{"class":3196},[3186,11373,11374,11376,11378,11380],{"class":3188,"line":3327},[3186,11375,4931],{"class":3203},[3186,11377,7524],{"class":3196},[3186,11379,7527],{"class":3236},[3186,11381,5106],{"class":3192},[3186,11383,11384],{"class":3188,"line":3333},[3186,11385,3330],{"class":3192},[3186,11387,11388,11390],{"class":3188,"line":3339},[3186,11389,4144],{"class":3192},[3186,11391,10621],{"class":3196},[3186,11393,11394,11396,11399],{"class":3188,"line":3358},[3186,11395,10585],{"class":3196},[3186,11397,11398],{"class":3210}," shouldLoadAudiobooksFromRepository",[3186,11400,4159],{"class":3192},[3186,11402,11403],{"class":3188,"line":3376},[3186,11404,10635],{"class":3226},[3186,11406,11407,11410,11412,11415],{"class":3188,"line":3394},[3186,11408,11409],{"class":3236},"        repository",[3186,11411,3259],{"class":3192},[3186,11413,11414],{"class":3210},"save",[3186,11416,11417],{"class":3192},"(audiobook1);\n",[3186,11419,11420,11422,11424,11426],{"class":3188,"line":3399},[3186,11421,11409],{"class":3236},[3186,11423,3259],{"class":3192},[3186,11425,11414],{"class":3210},[3186,11427,11428],{"class":3192},"(audiobook2);\n",[3186,11430,11431],{"class":3188,"line":3405},[3186,11432,4169],{"class":3192},[3186,11434,11435],{"class":3188,"line":3424},[3186,11436,10738],{"class":3226},[3186,11438,11439,11441,11443,11445],{"class":3188,"line":3430},[3186,11440,6210],{"class":3236},[3186,11442,3259],{"class":3192},[3186,11444,10747],{"class":3210},[3186,11446,3250],{"class":3192},[3186,11448,11449],{"class":3188,"line":3436},[3186,11450,4169],{"class":3192},[3186,11452,11453],{"class":3188,"line":3442},[3186,11454,10758],{"class":3226},[3186,11456,11457,11459,11461,11463,11465,11467,11469,11471,11473,11475],{"class":3188,"line":3448},[3186,11458,10763],{"class":3210},[3186,11460,3214],{"class":3192},[3186,11462,4493],{"class":3620},[3186,11464,3624],{"class":3192},[3186,11466,5141],{"class":3236},[3186,11468,3259],{"class":3192},[3186,11470,10776],{"class":3210},[3186,11472,3510],{"class":3192},[3186,11474,10781],{"class":3210},[3186,11476,3781],{"class":3192},[3186,11478,11479],{"class":3188,"line":3453},[3186,11480,4356],{"class":3192},[3186,11482,11483],{"class":3188,"line":3459},[3186,11484,3661],{"class":3192},[3981,11486,11487,11490,11491,11493,11494,11496,11497,11500],{},[3154,11488,11489],{},"Test Module для різних середовищ:"," Створіть окремі Modules для production (",[3168,11492,3839],{},"), тестів (",[3168,11495,11326],{},"), розробки (",[3168,11498,11499],{},"DevAudiobookModule"," з логуванням SQL). Це дозволить легко перемикатися між конфігураціями.",[3729,11502],{},[3145,11504,11506],{"id":11505},"практичні-завдання","Практичні завдання",[11508,11509,11510,11514,11529,11541,11560,11564,11572,11582,11595,11599,11613,11627],"steps",{},[3739,11511,11513],{"id":11512},"рівень-1-базова-інтеграція-guice","Рівень 1: Базова інтеграція Guice",[3150,11515,11516,11519,11520,11523,11524,3624,11526,3259],{},[3154,11517,11518],{},"Завдання 1.1:"," Створіть ",[3168,11521,11522],{},"AuthorModule"," (Guice Module) з bindings для ",[3168,11525,4214],{},[3168,11527,11528],{},"AuthorService",[3150,11530,11531,11519,11534,11537,11538,3259],{},[3154,11532,11533],{},"Завдання 1.2:",[3168,11535,11536],{},"AuthorListController"," з Constructor Injection для ",[3168,11539,11540],{},"AuthorListViewModel",[3150,11542,11543,11546,11547,11549,11550,11552,11553,3713,11556,8726,11558,3259],{},[3154,11544,11545],{},"Завдання 1.3:"," Модифікуйте ",[3168,11548,3827],{},", щоб він використовував ",[3168,11551,11522],{}," та завантажував ",[3168,11554,11555],{},"author-list-view.fxml",[3168,11557,3796],{},[3168,11559,3807],{},[3739,11561,11563],{"id":11562},"рівень-2-scopes-та-lifecycle","Рівень 2: Scopes та Lifecycle",[3150,11565,11566,11519,11569,11571],{},[3154,11567,11568],{},"Завдання 2.1:",[3168,11570,7639],{}," як Prototype scope (новий екземпляр для кожного відкриття форми). Переконайтеся, що при повторному відкритті форми стан скидається.",[3150,11573,11574,11577,11578,11581],{},[3154,11575,11576],{},"Завдання 2.2:"," Додайте у ",[3168,11579,11580],{},"AudiobookApp.stop()"," логіку закриття ExecutorService (якщо він використовується у ViewModel для асинхронних операцій).",[3150,11583,11584,11587,11588,11591,11592,3259],{},[3154,11585,11586],{},"Завдання 2.3:"," Створіть Provider для ",[3168,11589,11590],{},"ExecutorService",", що конфігурує ThreadPool з параметрами з файлу ",[3168,11593,11594],{},"application.properties",[3739,11596,11598],{"id":11597},"рівень-3-assistedinject-та-складні-сценарії","Рівень 3: AssistedInject та складні сценарії",[3150,11600,11601,11519,11604,11606,11607,11609,11610,11612],{},[3154,11602,11603],{},"Завдання 3.1:",[3168,11605,8814],{},", що приймає ",[3168,11608,8818],{}," у конструкторі через ",[3168,11611,9076],{},". Створіть Factory для нього та зареєструйте у Module.",[3150,11614,11615,11618,11619,11621,11622,11624,11625,3259],{},[3154,11616,11617],{},"Завдання 3.2:"," Реалізуйте діалог редагування аудіокниги: при натисканні \"Edit\" у ",[3168,11620,3855],{}," відкривається діалог з ",[3168,11623,7639],{},", створеним через Factory з параметром ",[3168,11626,8818],{},[3150,11628,11629,11519,11632,11634,11635,11638,11639,8726,11641,11643,11644,11646],{},[3154,11630,11631],{},"Завдання 3.3:",[3168,11633,5718],{}," клас з методом ",[3168,11636,11637],{},"navigateTo(String screenId)",", що завантажує FXML через ",[3168,11640,3796],{},[3168,11642,3807],{}," та змінює Root Node у Scene. Ін'єктуйте ",[3168,11645,5718],{}," у Controllers для навігації між екранами.",[3729,11648],{},[3145,11650,11652],{"id":11651},"підсумок","Підсумок",[3150,11654,11655],{},"У цій статті ми інтегрували MVVM-архітектуру з Google Guice для автоматичної ін'єкції залежностей. Ключові висновки:",[3150,11657,11658,11661,11662,11664],{},[3154,11659,11660],{},"Guice вирішує проблему ручного створення об'єктів."," Замість 20+ рядків коду для створення графу залежностей у ",[3168,11663,3176],{},", Guice автоматично резолвить всі залежності через конфігурацію у Module.",[3150,11666,11667,11670],{},[3154,11668,11669],{},"Guice Module — це центральна конфігурація додатку."," Тут визначаємо bindings (інтерфейс → реалізація), scopes (Singleton vs Prototype), Provider Methods для складних об'єктів (DataSource, ExecutorService).",[3150,11672,11673,3693,11676,11678,11679,11681],{},[3154,11674,11675],{},"ControllerFactory — це міст між FXMLLoader та Guice.",[3168,11677,3918],{}," дозволяє ",[3168,11680,3796],{}," створювати Controllers через Guice, що автоматично ін'єктує всі залежності (ViewModel, Navigator).",[3150,11683,11684,11687,11688,11690],{},[3154,11685,11686],{},"Constructor Injection — рекомендований підхід."," Явні залежності у конструкторі, immutability через ",[3168,11689,6937],{},", легкість тестування (передача mock через конструктор).",[3150,11692,11693,11696],{},[3154,11694,11695],{},"Scopes керують життєвим циклом об'єктів."," Singleton для stateless об'єктів (Repository, Service) та глобального стану (ViewModel головних екранів). Prototype для stateful об'єктів (ViewModel діалогів, форм).",[3150,11698,11699,11702,11703,11705,11706,11708],{},[3154,11700,11701],{},"AssistedInject для параметризованих ViewModel."," Коли ViewModel потребує параметр (наприклад, ",[3168,11704,8818],{}," для редагування), використовуємо Factory з ",[3168,11707,9076],{}," анотацією. Guice автоматично ін'єктує решту залежностей.",[3150,11710,11711,3693,11714,11716,11717,11719,11720,11722],{},[3154,11712,11713],{},"Application lifecycle: init() → start() → stop().",[3168,11715,8691],{}," — створення Injector та міграції БД (фоновий потік). ",[3168,11718,3681],{}," — завантаження UI (JavaFX Thread). ",[3168,11721,8751],{}," — закриття ресурсів (Connection Pool, ExecutorService).",[3150,11724,11725,11728],{},[3154,11726,11727],{},"Тестування стає простішим."," Unit-тести з mock залежностями (Mockito). Інтеграційні тести з Test Module (InMemory Repository, H2 database). Легко замінити реалізації без зміни коду.",[3150,11730,11731],{},"У наступній статті ми розглянемо валідацію та обробку помилок у MVVM: як валідувати введення користувача на рівні ViewModel, як відображати помилки у View, як обробляти винятки з Repository та Service, та як створити централізовану систему валідації через Validator Pattern.",[11733,11734,11735],"style",{},"html pre.shiki code .sHH4Y, html code.shiki .sHH4Y{--shiki-light:#000000;--shiki-default:#D4D4D4;--shiki-dark:#D4D4D4}html pre.shiki code .sN1BT, html code.shiki .sN1BT{--shiki-light:#267F99;--shiki-default:#4EC9B0;--shiki-dark:#4EC9B0}html pre.shiki code .su1O8, html code.shiki .su1O8{--shiki-light:#0000FF;--shiki-default:#569CD6;--shiki-dark:#569CD6}html pre.shiki code .s8Opu, html code.shiki .s8Opu{--shiki-light:#795E26;--shiki-default:#DCDCAA;--shiki-dark:#DCDCAA}html pre.shiki code .spJ8K, html code.shiki .spJ8K{--shiki-light:#008000;--shiki-default:#6A9955;--shiki-dark:#6A9955}html pre.shiki code .siwwj, html code.shiki .siwwj{--shiki-light:#001080;--shiki-default:#9CDCFE;--shiki-dark:#9CDCFE}html pre.shiki code .s8xlr, html code.shiki .s8xlr{--shiki-light:#AF00DB;--shiki-default:#C586C0;--shiki-dark:#C586C0}html pre.shiki code .sbdoH, html code.shiki .sbdoH{--shiki-light:#A31515;--shiki-default:#CE9178;--shiki-dark:#CE9178}html pre.shiki code .sJj4R, html code.shiki .sJj4R{--shiki-light:#098658;--shiki-default:#B5CEA8;--shiki-dark:#B5CEA8}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);}",{"title":3170,"searchDepth":3200,"depth":3200,"links":11737},[11738,11739,11743,11748,11753,11758,11763,11768,11771,11775,11779,11784],{"id":3147,"depth":3200,"text":3148},{"id":3733,"depth":3200,"text":3734,"children":11740},[11741,11742],{"id":3741,"depth":3223,"text":3742},{"id":3817,"depth":3223,"text":3818},{"id":3991,"depth":3200,"text":3992,"children":11744},[11745,11746,11747],{"id":3998,"depth":3223,"text":3999},{"id":4543,"depth":3223,"text":4544},{"id":4608,"depth":3223,"text":4609},{"id":5059,"depth":3200,"text":5060,"children":11749},[11750,11751,11752],{"id":5072,"depth":3223,"text":5073},{"id":5223,"depth":3223,"text":5224},{"id":5362,"depth":3223,"text":5363},{"id":5724,"depth":3200,"text":5725,"children":11754},[11755,11756,11757],{"id":5734,"depth":3223,"text":5735},{"id":6882,"depth":3223,"text":6883},{"id":7029,"depth":3223,"text":7030},{"id":7332,"depth":3200,"text":7333,"children":11759},[11760,11761,11762],{"id":7339,"depth":3223,"text":7340},{"id":7576,"depth":3223,"text":7577},{"id":7739,"depth":3223,"text":7740},{"id":7829,"depth":3200,"text":7830,"children":11764},[11765,11766,11767],{"id":7839,"depth":3223,"text":7840},{"id":8682,"depth":3223,"text":8683},{"id":8792,"depth":3223,"text":8793},{"id":8807,"depth":3200,"text":8808,"children":11769},[11770],{"id":8928,"depth":3223,"text":8929},{"id":9356,"depth":3200,"text":9357,"children":11772},[11773,11774],{"id":9363,"depth":3223,"text":9364},{"id":9905,"depth":3223,"text":9906},{"id":10372,"depth":3200,"text":10373,"children":11776},[11777,11778],{"id":10379,"depth":3223,"text":10380},{"id":10986,"depth":3223,"text":10987},{"id":11505,"depth":3200,"text":11506,"children":11780},[11781,11782,11783],{"id":11512,"depth":3223,"text":11513},{"id":11562,"depth":3223,"text":11563},{"id":11597,"depth":3223,"text":11598},{"id":11651,"depth":3200,"text":11652},"Від ручного створення об'єктів до Dependency Injection: Guice Module, ControllerFactory, Constructor Injection, Scopes (Singleton vs Prototype), AssistedInject для параметризованих ViewModel.","md",null,{},{"title":2434,"description":11785},"ib_RRdyH18xKthkLTzvcy2B20GkjGaX34u6eCg7kdno",[11792,11794],{"title":2430,"path":2431,"stem":2432,"description":11793,"children":-1},"Від ViewModel до UI: FXML як декларативний опис інтерфейсу, мінімальний Controller з Bindings, ін'єкція ViewModel, FXMLLoader та ControllerFactory, обробка подій та навігація.",{"title":2438,"path":2439,"stem":2440,"description":11795,"children":-1},"Від реактивної валідації Properties до централізованої системи Validators: валідація на рівні ViewModel, Validator Pattern, CompositeValidator, асинхронна валідація унікальності, обробка помилок Repository та Service, відображення помилок у View.",1778998390354]