[{"data":1,"prerenderedAt":9484},["ShallowReactive",2],{"navigation_docs":3,"-java-pr2-mvvm-view-controller":3135,"-java-pr2-mvvm-view-controller-surround":9479},[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":2430,"body":3137,"description":9473,"extension":9474,"links":9475,"meta":9476,"navigation":3310,"path":2431,"seo":9477,"stem":2432,"__hash__":9478},"docs\u002F04.java\u002Fpr2\u002F29.mvvm-view-controller.md",{"type":3138,"value":3139,"toc":9431},"minimark",[3140,3144,3149,3158,3170,3184,3191,3213,3228,3235,3238,3242,3249,3254,3261,4482,4486,4499,4520,4532,4549,4561,4583,4607,4617,4638,4707,4709,4713,4724,4728,5446,5451,5469,5479,5491,5503,5512,5531,5535,5541,6274,6279,6298,6311,6331,6352,6369,6394,6418,6431,6441,6443,6447,6450,6456,6460,6463,6472,6509,6524,6529,6573,6576,6585,6589,6595,7023,7028,7040,7052,7066,7077,7091,7098,7102,7107,7157,7164,7166,7170,7179,7186,7190,7204,7264,7270,7341,7354,7358,7367,7416,7426,7493,7506,7510,7517,8001,8006,8026,8045,8071,8077,8088,8092,8098,8431,8436,8448,8465,8476,8488,8503,8520,8522,8526,8531,8597,8601,8614,8619,8687,8689,8693,8696,8700,8769,8774,8858,8871,8875,8940,8946,8956,8960,9069,9082,9087,9089,9093,9096,9209,9211,9215,9346,9348,9352,9355,9361,9373,9392,9401,9410,9424,9427],[3141,3142,2430],"h1",{"id":3143},"view-та-controller-звязування-з-viewmodel-через-fxml",[3145,3146,3148],"h2",{"id":3147},"вступ-роль-controller-у-mvvm","Вступ: Роль Controller у MVVM",[3150,3151,3152,3153,3157],"p",{},"У попередній статті ми побудували ",[3154,3155,3156],"code",{},"AudiobookListViewModel"," — повноцінний ViewModel з Properties, Commands, фільтрацією та асинхронністю. Але цей ViewModel існує у вакуумі — він не підключений до жодного UI-елемента. Користувач не бачить список аудіокниг, не може натиснути кнопку \"Delete\", не бачить індикатор завантаження.",[3150,3159,3160,3161,3165,3166,3169],{},"Саме тут на сцену виходить ",[3162,3163,3164],"strong",{},"View"," (FXML-розмітка) та ",[3162,3167,3168],{},"Controller"," (Java-клас, що з'єднує View з ViewModel). Але їхня роль у MVVM радикально відрізняється від того, що ми бачили у MVC або навіть у класичних JavaFX-додатках без архітектурного патерну.",[3150,3171,3172,3175,3176,3179,3180,3183],{},[3162,3173,3174],{},"У MVP"," Controller (Presenter) був \"розумним\" — він містив всю логіку, викликав методи View, оновлював UI вручну. У ",[3162,3177,3178],{},"MVVM"," Controller стає \"дурним\" — він лише ",[3162,3181,3182],{},"ініціалізує Bindings"," між View та ViewModel. Вся логіка залишається у ViewModel, весь UI описаний у FXML, а Controller — це тонкий шар \"клею\", що з'єднує їх.",[3150,3185,3186,3187,3190],{},"Ця стаття про те, ",[3162,3188,3189],{},"як саме виглядає цей \"клей\"",". Ми розглянемо:",[3192,3193,3194,3198,3201,3204,3207,3210],"ul",{},[3195,3196,3197],"li",{},"Як структурувати FXML для MVVM-додатку.",[3195,3199,3200],{},"Як мінімізувати код у Controller.",[3195,3202,3203],{},"Як ініціалізувати Bindings між UI-елементами та Properties ViewModel.",[3195,3205,3206],{},"Як передати ViewModel у Controller (ін'єкція залежностей).",[3195,3208,3209],{},"Як обробляти події (кліки кнопок) через делегування до ViewModel.",[3195,3211,3212],{},"Як організувати навігацію між екранами.",[3150,3214,3215,3216,3219,3220,3223,3224,3227],{},"Але перш за все, потрібно зрозуміти фундаментальний принцип: ",[3162,3217,3218],{},"Controller у MVVM не містить логіки",". Якщо ви пишете ",[3154,3221,3222],{},"if",", ",[3154,3225,3226],{},"for",", обчислення, SQL-запити у Controller — ви порушуєте MVVM. Весь цей код має бути у ViewModel.",[3229,3230,3231,3234],"note",{},[3162,3232,3233],{},"Controller у MVVM — це адаптер."," Його єдина відповідальність — з'єднати FXML-елементи (які мають JavaFX Properties) з ViewModel Properties через Bindings. Якщо Controller перевищує 100-150 рядків коду, це сигнал, що щось не так — логіка просочилася з ViewModel у Controller.",[3236,3237],"hr",{},[3145,3239,3241],{"id":3240},"fxml-декларативний-опис-ui","FXML: Декларативний опис UI",[3150,3243,3244,3245,3248],{},"Перш ніж писати Controller, створимо FXML — XML-розмітку, що описує структуру нашого екрану. FXML — це не просто \"HTML для JavaFX\". Це ",[3162,3246,3247],{},"декларативний спосіб"," описати, що відображається, без змішування з логікою того, як воно працює.",[3250,3251,3253],"h3",{"id":3252},"структура-fxml-для-audiobooklistview","Структура FXML для AudiobookListView",[3150,3255,3256,3257,3260],{},"Створимо файл ",[3154,3258,3259],{},"audiobook-list-view.fxml"," для екрану зі списком аудіокниг:",[3262,3263,3268],"pre",{"className":3264,"code":3265,"language":3266,"meta":3267,"style":3267},"language-xml shiki shiki-themes light-plus dark-plus dark-plus","\u003C?xml version=\"1.0\" encoding=\"UTF-8\"?>\n\n\u003C?import javafx.scene.control.*?>\n\u003C?import javafx.scene.layout.*?>\n\u003C?import javafx.geometry.Insets?>\n\n\u003CBorderPane xmlns:fx=\"http:\u002F\u002Fjavafx.com\u002Ffxml\"\n            fx:controller=\"dev.kostyl.audiobook.controller.AudiobookListController\"\n            prefWidth=\"900\" prefHeight=\"600\">\n    \n    \u003C!-- Top: Панель інструментів -->\n    \u003Ctop>\n        \u003CVBox spacing=\"10\" style=\"-fx-background-color: #f5f5f5; -fx-padding: 10;\">\n            \n            \u003C!-- Рядок 1: Кнопки дій -->\n            \u003CHBox spacing=\"10\" alignment=\"CENTER_LEFT\">\n                \u003CButton fx:id=\"addButton\" text=\"Add Audiobook\" \n                        styleClass=\"primary-button\"\u002F>\n                \u003CButton fx:id=\"editButton\" text=\"Edit\" \n                        styleClass=\"secondary-button\"\u002F>\n                \u003CButton fx:id=\"deleteButton\" text=\"Delete\" \n                        styleClass=\"danger-button\"\u002F>\n                \n                \u003CRegion HBox.hgrow=\"ALWAYS\"\u002F> \u003C!-- Spacer -->\n                \n                \u003CButton fx:id=\"refreshButton\" text=\"Refresh\" \n                        styleClass=\"secondary-button\"\u002F>\n            \u003C\u002FHBox>\n            \n            \u003C!-- Рядок 2: Фільтри -->\n            \u003CHBox spacing=\"10\" alignment=\"CENTER_LEFT\">\n                \u003CLabel text=\"Search:\"\u002F>\n                \u003CTextField fx:id=\"searchField\" \n                           promptText=\"Search by title or author...\" \n                           prefWidth=\"300\"\u002F>\n                \n                \u003CLabel text=\"Genre:\"\u002F>\n                \u003CComboBox fx:id=\"genreComboBox\" \n                          promptText=\"All genres\" \n                          prefWidth=\"150\"\u002F>\n            \u003C\u002FHBox>\n            \n        \u003C\u002FVBox>\n    \u003C\u002Ftop>\n    \n    \u003C!-- Center: Таблиця аудіокниг -->\n    \u003Ccenter>\n        \u003CStackPane>\n            \n            \u003C!-- Таблиця -->\n            \u003CTableView fx:id=\"audiobookTable\">\n                \u003Ccolumns>\n                    \u003CTableColumn fx:id=\"titleColumn\" text=\"Title\" \n                                 prefWidth=\"250\" minWidth=\"150\"\u002F>\n                    \u003CTableColumn fx:id=\"authorColumn\" text=\"Author\" \n                                 prefWidth=\"200\" minWidth=\"120\"\u002F>\n                    \u003CTableColumn fx:id=\"genreColumn\" text=\"Genre\" \n                                 prefWidth=\"150\" minWidth=\"100\"\u002F>\n                    \u003CTableColumn fx:id=\"durationColumn\" text=\"Duration\" \n                                 prefWidth=\"120\" minWidth=\"80\"\u002F>\n                    \u003CTableColumn fx:id=\"releaseYearColumn\" text=\"Year\" \n                                 prefWidth=\"80\" minWidth=\"60\"\u002F>\n                \u003C\u002Fcolumns>\n                \n                \u003Cplaceholder>\n                    \u003CLabel text=\"No audiobooks found\" \n                           style=\"-fx-text-fill: gray;\"\u002F>\n                \u003C\u002Fplaceholder>\n            \u003C\u002FTableView>\n            \n            \u003C!-- Індикатор завантаження (поверх таблиці) -->\n            \u003CProgressIndicator fx:id=\"loadingIndicator\" \n                               maxWidth=\"100\" maxHeight=\"100\"\n                               visible=\"false\"\u002F>\n            \n        \u003C\u002FStackPane>\n    \u003C\u002Fcenter>\n    \n    \u003C!-- Bottom: Статус-бар -->\n    \u003Cbottom>\n        \u003CHBox spacing=\"10\" alignment=\"CENTER_LEFT\" \n              style=\"-fx-background-color: #e0e0e0; -fx-padding: 5 10;\">\n            \n            \u003CLabel fx:id=\"statusLabel\" text=\"Ready\"\u002F>\n            \n            \u003CRegion HBox.hgrow=\"ALWAYS\"\u002F> \u003C!-- Spacer -->\n            \n            \u003CLabel fx:id=\"countLabel\" text=\"0 audiobooks\"\u002F>\n            \n        \u003C\u002FHBox>\n    \u003C\u002Fbottom>\n    \n\u003C\u002FBorderPane>\n","xml","",[3154,3269,3270,3305,3312,3328,3342,3356,3361,3378,3389,3411,3417,3424,3435,3462,3468,3474,3499,3527,3541,3564,3576,3599,3611,3617,3639,3644,3667,3678,3688,3693,3699,3720,3737,3754,3767,3780,3785,3801,3818,3831,3844,3853,3858,3868,3878,3883,3889,3899,3909,3914,3920,3937,3947,3972,3992,4015,4034,4057,4075,4098,4116,4139,4157,4167,4172,4182,4198,4211,4220,4229,4234,4240,4257,4275,4288,4293,4302,4311,4316,4322,4332,4353,4366,4371,4394,4399,4416,4421,4444,4449,4458,4467,4472],{"__ignoreMap":3267},[3271,3272,3275,3279,3282,3286,3290,3294,3297,3299,3302],"span",{"class":3273,"line":3274},"line",1,[3271,3276,3278],{"class":3277},"s0P7L","\u003C?",[3271,3280,3266],{"class":3281},"sKtos",[3271,3283,3285],{"class":3284},"sa4r_"," version",[3271,3287,3289],{"class":3288},"sHH4Y","=",[3271,3291,3293],{"class":3292},"su9tN","\"1.0\"",[3271,3295,3296],{"class":3284}," encoding",[3271,3298,3289],{"class":3288},[3271,3300,3301],{"class":3292},"\"UTF-8\"",[3271,3303,3304],{"class":3277},"?>\n",[3271,3306,3308],{"class":3273,"line":3307},2,[3271,3309,3311],{"emptyLinePlaceholder":3310},true,"\n",[3271,3313,3315,3317,3320,3323,3326],{"class":3273,"line":3314},3,[3271,3316,3278],{"class":3277},[3271,3318,3319],{"class":3281},"import",[3271,3321,3322],{"class":3284}," javafx",[3271,3324,3325],{"class":3288},".scene.control.*",[3271,3327,3304],{"class":3277},[3271,3329,3331,3333,3335,3337,3340],{"class":3273,"line":3330},4,[3271,3332,3278],{"class":3277},[3271,3334,3319],{"class":3281},[3271,3336,3322],{"class":3284},[3271,3338,3339],{"class":3288},".scene.layout.*",[3271,3341,3304],{"class":3277},[3271,3343,3345,3347,3349,3351,3354],{"class":3273,"line":3344},5,[3271,3346,3278],{"class":3277},[3271,3348,3319],{"class":3281},[3271,3350,3322],{"class":3284},[3271,3352,3353],{"class":3288},".geometry.Insets",[3271,3355,3304],{"class":3277},[3271,3357,3359],{"class":3273,"line":3358},6,[3271,3360,3311],{"emptyLinePlaceholder":3310},[3271,3362,3364,3367,3370,3373,3375],{"class":3273,"line":3363},7,[3271,3365,3366],{"class":3277},"\u003C",[3271,3368,3369],{"class":3281},"BorderPane",[3271,3371,3372],{"class":3284}," xmlns:fx",[3271,3374,3289],{"class":3288},[3271,3376,3377],{"class":3292},"\"http:\u002F\u002Fjavafx.com\u002Ffxml\"\n",[3271,3379,3381,3384,3386],{"class":3273,"line":3380},8,[3271,3382,3383],{"class":3284},"            fx:controller",[3271,3385,3289],{"class":3288},[3271,3387,3388],{"class":3292},"\"dev.kostyl.audiobook.controller.AudiobookListController\"\n",[3271,3390,3392,3395,3397,3400,3403,3405,3408],{"class":3273,"line":3391},9,[3271,3393,3394],{"class":3284},"            prefWidth",[3271,3396,3289],{"class":3288},[3271,3398,3399],{"class":3292},"\"900\"",[3271,3401,3402],{"class":3284}," prefHeight",[3271,3404,3289],{"class":3288},[3271,3406,3407],{"class":3292},"\"600\"",[3271,3409,3410],{"class":3277},">\n",[3271,3412,3414],{"class":3273,"line":3413},10,[3271,3415,3416],{"class":3288},"    \n",[3271,3418,3420],{"class":3273,"line":3419},11,[3271,3421,3423],{"class":3422},"spJ8K","    \u003C!-- Top: Панель інструментів -->\n",[3271,3425,3427,3430,3433],{"class":3273,"line":3426},12,[3271,3428,3429],{"class":3277},"    \u003C",[3271,3431,3432],{"class":3281},"top",[3271,3434,3410],{"class":3277},[3271,3436,3438,3441,3444,3447,3449,3452,3455,3457,3460],{"class":3273,"line":3437},13,[3271,3439,3440],{"class":3277},"        \u003C",[3271,3442,3443],{"class":3281},"VBox",[3271,3445,3446],{"class":3284}," spacing",[3271,3448,3289],{"class":3288},[3271,3450,3451],{"class":3292},"\"10\"",[3271,3453,3454],{"class":3284}," style",[3271,3456,3289],{"class":3288},[3271,3458,3459],{"class":3292},"\"-fx-background-color: #f5f5f5; -fx-padding: 10;\"",[3271,3461,3410],{"class":3277},[3271,3463,3465],{"class":3273,"line":3464},14,[3271,3466,3467],{"class":3288},"            \n",[3271,3469,3471],{"class":3273,"line":3470},15,[3271,3472,3473],{"class":3422},"            \u003C!-- Рядок 1: Кнопки дій -->\n",[3271,3475,3477,3480,3483,3485,3487,3489,3492,3494,3497],{"class":3273,"line":3476},16,[3271,3478,3479],{"class":3277},"            \u003C",[3271,3481,3482],{"class":3281},"HBox",[3271,3484,3446],{"class":3284},[3271,3486,3289],{"class":3288},[3271,3488,3451],{"class":3292},[3271,3490,3491],{"class":3284}," alignment",[3271,3493,3289],{"class":3288},[3271,3495,3496],{"class":3292},"\"CENTER_LEFT\"",[3271,3498,3410],{"class":3277},[3271,3500,3502,3505,3508,3511,3513,3516,3519,3521,3524],{"class":3273,"line":3501},17,[3271,3503,3504],{"class":3277},"                \u003C",[3271,3506,3507],{"class":3281},"Button",[3271,3509,3510],{"class":3284}," fx:id",[3271,3512,3289],{"class":3288},[3271,3514,3515],{"class":3292},"\"addButton\"",[3271,3517,3518],{"class":3284}," text",[3271,3520,3289],{"class":3288},[3271,3522,3523],{"class":3292},"\"Add Audiobook\"",[3271,3525,3526],{"class":3288}," \n",[3271,3528,3530,3533,3535,3538],{"class":3273,"line":3529},18,[3271,3531,3532],{"class":3284},"                        styleClass",[3271,3534,3289],{"class":3288},[3271,3536,3537],{"class":3292},"\"primary-button\"",[3271,3539,3540],{"class":3277},"\u002F>\n",[3271,3542,3544,3546,3548,3550,3552,3555,3557,3559,3562],{"class":3273,"line":3543},19,[3271,3545,3504],{"class":3277},[3271,3547,3507],{"class":3281},[3271,3549,3510],{"class":3284},[3271,3551,3289],{"class":3288},[3271,3553,3554],{"class":3292},"\"editButton\"",[3271,3556,3518],{"class":3284},[3271,3558,3289],{"class":3288},[3271,3560,3561],{"class":3292},"\"Edit\"",[3271,3563,3526],{"class":3288},[3271,3565,3567,3569,3571,3574],{"class":3273,"line":3566},20,[3271,3568,3532],{"class":3284},[3271,3570,3289],{"class":3288},[3271,3572,3573],{"class":3292},"\"secondary-button\"",[3271,3575,3540],{"class":3277},[3271,3577,3579,3581,3583,3585,3587,3590,3592,3594,3597],{"class":3273,"line":3578},21,[3271,3580,3504],{"class":3277},[3271,3582,3507],{"class":3281},[3271,3584,3510],{"class":3284},[3271,3586,3289],{"class":3288},[3271,3588,3589],{"class":3292},"\"deleteButton\"",[3271,3591,3518],{"class":3284},[3271,3593,3289],{"class":3288},[3271,3595,3596],{"class":3292},"\"Delete\"",[3271,3598,3526],{"class":3288},[3271,3600,3602,3604,3606,3609],{"class":3273,"line":3601},22,[3271,3603,3532],{"class":3284},[3271,3605,3289],{"class":3288},[3271,3607,3608],{"class":3292},"\"danger-button\"",[3271,3610,3540],{"class":3277},[3271,3612,3614],{"class":3273,"line":3613},23,[3271,3615,3616],{"class":3288},"                \n",[3271,3618,3620,3622,3625,3628,3630,3633,3636],{"class":3273,"line":3619},24,[3271,3621,3504],{"class":3277},[3271,3623,3624],{"class":3281},"Region",[3271,3626,3627],{"class":3284}," HBox.hgrow",[3271,3629,3289],{"class":3288},[3271,3631,3632],{"class":3292},"\"ALWAYS\"",[3271,3634,3635],{"class":3277},"\u002F>",[3271,3637,3638],{"class":3422}," \u003C!-- Spacer -->\n",[3271,3640,3642],{"class":3273,"line":3641},25,[3271,3643,3616],{"class":3288},[3271,3645,3647,3649,3651,3653,3655,3658,3660,3662,3665],{"class":3273,"line":3646},26,[3271,3648,3504],{"class":3277},[3271,3650,3507],{"class":3281},[3271,3652,3510],{"class":3284},[3271,3654,3289],{"class":3288},[3271,3656,3657],{"class":3292},"\"refreshButton\"",[3271,3659,3518],{"class":3284},[3271,3661,3289],{"class":3288},[3271,3663,3664],{"class":3292},"\"Refresh\"",[3271,3666,3526],{"class":3288},[3271,3668,3670,3672,3674,3676],{"class":3273,"line":3669},27,[3271,3671,3532],{"class":3284},[3271,3673,3289],{"class":3288},[3271,3675,3573],{"class":3292},[3271,3677,3540],{"class":3277},[3271,3679,3681,3684,3686],{"class":3273,"line":3680},28,[3271,3682,3683],{"class":3277},"            \u003C\u002F",[3271,3685,3482],{"class":3281},[3271,3687,3410],{"class":3277},[3271,3689,3691],{"class":3273,"line":3690},29,[3271,3692,3467],{"class":3288},[3271,3694,3696],{"class":3273,"line":3695},30,[3271,3697,3698],{"class":3422},"            \u003C!-- Рядок 2: Фільтри -->\n",[3271,3700,3702,3704,3706,3708,3710,3712,3714,3716,3718],{"class":3273,"line":3701},31,[3271,3703,3479],{"class":3277},[3271,3705,3482],{"class":3281},[3271,3707,3446],{"class":3284},[3271,3709,3289],{"class":3288},[3271,3711,3451],{"class":3292},[3271,3713,3491],{"class":3284},[3271,3715,3289],{"class":3288},[3271,3717,3496],{"class":3292},[3271,3719,3410],{"class":3277},[3271,3721,3723,3725,3728,3730,3732,3735],{"class":3273,"line":3722},32,[3271,3724,3504],{"class":3277},[3271,3726,3727],{"class":3281},"Label",[3271,3729,3518],{"class":3284},[3271,3731,3289],{"class":3288},[3271,3733,3734],{"class":3292},"\"Search:\"",[3271,3736,3540],{"class":3277},[3271,3738,3740,3742,3745,3747,3749,3752],{"class":3273,"line":3739},33,[3271,3741,3504],{"class":3277},[3271,3743,3744],{"class":3281},"TextField",[3271,3746,3510],{"class":3284},[3271,3748,3289],{"class":3288},[3271,3750,3751],{"class":3292},"\"searchField\"",[3271,3753,3526],{"class":3288},[3271,3755,3757,3760,3762,3765],{"class":3273,"line":3756},34,[3271,3758,3759],{"class":3284},"                           promptText",[3271,3761,3289],{"class":3288},[3271,3763,3764],{"class":3292},"\"Search by title or author...\"",[3271,3766,3526],{"class":3288},[3271,3768,3770,3773,3775,3778],{"class":3273,"line":3769},35,[3271,3771,3772],{"class":3284},"                           prefWidth",[3271,3774,3289],{"class":3288},[3271,3776,3777],{"class":3292},"\"300\"",[3271,3779,3540],{"class":3277},[3271,3781,3783],{"class":3273,"line":3782},36,[3271,3784,3616],{"class":3288},[3271,3786,3788,3790,3792,3794,3796,3799],{"class":3273,"line":3787},37,[3271,3789,3504],{"class":3277},[3271,3791,3727],{"class":3281},[3271,3793,3518],{"class":3284},[3271,3795,3289],{"class":3288},[3271,3797,3798],{"class":3292},"\"Genre:\"",[3271,3800,3540],{"class":3277},[3271,3802,3804,3806,3809,3811,3813,3816],{"class":3273,"line":3803},38,[3271,3805,3504],{"class":3277},[3271,3807,3808],{"class":3281},"ComboBox",[3271,3810,3510],{"class":3284},[3271,3812,3289],{"class":3288},[3271,3814,3815],{"class":3292},"\"genreComboBox\"",[3271,3817,3526],{"class":3288},[3271,3819,3821,3824,3826,3829],{"class":3273,"line":3820},39,[3271,3822,3823],{"class":3284},"                          promptText",[3271,3825,3289],{"class":3288},[3271,3827,3828],{"class":3292},"\"All genres\"",[3271,3830,3526],{"class":3288},[3271,3832,3834,3837,3839,3842],{"class":3273,"line":3833},40,[3271,3835,3836],{"class":3284},"                          prefWidth",[3271,3838,3289],{"class":3288},[3271,3840,3841],{"class":3292},"\"150\"",[3271,3843,3540],{"class":3277},[3271,3845,3847,3849,3851],{"class":3273,"line":3846},41,[3271,3848,3683],{"class":3277},[3271,3850,3482],{"class":3281},[3271,3852,3410],{"class":3277},[3271,3854,3856],{"class":3273,"line":3855},42,[3271,3857,3467],{"class":3288},[3271,3859,3861,3864,3866],{"class":3273,"line":3860},43,[3271,3862,3863],{"class":3277},"        \u003C\u002F",[3271,3865,3443],{"class":3281},[3271,3867,3410],{"class":3277},[3271,3869,3871,3874,3876],{"class":3273,"line":3870},44,[3271,3872,3873],{"class":3277},"    \u003C\u002F",[3271,3875,3432],{"class":3281},[3271,3877,3410],{"class":3277},[3271,3879,3881],{"class":3273,"line":3880},45,[3271,3882,3416],{"class":3288},[3271,3884,3886],{"class":3273,"line":3885},46,[3271,3887,3888],{"class":3422},"    \u003C!-- Center: Таблиця аудіокниг -->\n",[3271,3890,3892,3894,3897],{"class":3273,"line":3891},47,[3271,3893,3429],{"class":3277},[3271,3895,3896],{"class":3281},"center",[3271,3898,3410],{"class":3277},[3271,3900,3902,3904,3907],{"class":3273,"line":3901},48,[3271,3903,3440],{"class":3277},[3271,3905,3906],{"class":3281},"StackPane",[3271,3908,3410],{"class":3277},[3271,3910,3912],{"class":3273,"line":3911},49,[3271,3913,3467],{"class":3288},[3271,3915,3917],{"class":3273,"line":3916},50,[3271,3918,3919],{"class":3422},"            \u003C!-- Таблиця -->\n",[3271,3921,3923,3925,3928,3930,3932,3935],{"class":3273,"line":3922},51,[3271,3924,3479],{"class":3277},[3271,3926,3927],{"class":3281},"TableView",[3271,3929,3510],{"class":3284},[3271,3931,3289],{"class":3288},[3271,3933,3934],{"class":3292},"\"audiobookTable\"",[3271,3936,3410],{"class":3277},[3271,3938,3940,3942,3945],{"class":3273,"line":3939},52,[3271,3941,3504],{"class":3277},[3271,3943,3944],{"class":3281},"columns",[3271,3946,3410],{"class":3277},[3271,3948,3950,3953,3956,3958,3960,3963,3965,3967,3970],{"class":3273,"line":3949},53,[3271,3951,3952],{"class":3277},"                    \u003C",[3271,3954,3955],{"class":3281},"TableColumn",[3271,3957,3510],{"class":3284},[3271,3959,3289],{"class":3288},[3271,3961,3962],{"class":3292},"\"titleColumn\"",[3271,3964,3518],{"class":3284},[3271,3966,3289],{"class":3288},[3271,3968,3969],{"class":3292},"\"Title\"",[3271,3971,3526],{"class":3288},[3271,3973,3975,3978,3980,3983,3986,3988,3990],{"class":3273,"line":3974},54,[3271,3976,3977],{"class":3284},"                                 prefWidth",[3271,3979,3289],{"class":3288},[3271,3981,3982],{"class":3292},"\"250\"",[3271,3984,3985],{"class":3284}," minWidth",[3271,3987,3289],{"class":3288},[3271,3989,3841],{"class":3292},[3271,3991,3540],{"class":3277},[3271,3993,3995,3997,3999,4001,4003,4006,4008,4010,4013],{"class":3273,"line":3994},55,[3271,3996,3952],{"class":3277},[3271,3998,3955],{"class":3281},[3271,4000,3510],{"class":3284},[3271,4002,3289],{"class":3288},[3271,4004,4005],{"class":3292},"\"authorColumn\"",[3271,4007,3518],{"class":3284},[3271,4009,3289],{"class":3288},[3271,4011,4012],{"class":3292},"\"Author\"",[3271,4014,3526],{"class":3288},[3271,4016,4018,4020,4022,4025,4027,4029,4032],{"class":3273,"line":4017},56,[3271,4019,3977],{"class":3284},[3271,4021,3289],{"class":3288},[3271,4023,4024],{"class":3292},"\"200\"",[3271,4026,3985],{"class":3284},[3271,4028,3289],{"class":3288},[3271,4030,4031],{"class":3292},"\"120\"",[3271,4033,3540],{"class":3277},[3271,4035,4037,4039,4041,4043,4045,4048,4050,4052,4055],{"class":3273,"line":4036},57,[3271,4038,3952],{"class":3277},[3271,4040,3955],{"class":3281},[3271,4042,3510],{"class":3284},[3271,4044,3289],{"class":3288},[3271,4046,4047],{"class":3292},"\"genreColumn\"",[3271,4049,3518],{"class":3284},[3271,4051,3289],{"class":3288},[3271,4053,4054],{"class":3292},"\"Genre\"",[3271,4056,3526],{"class":3288},[3271,4058,4060,4062,4064,4066,4068,4070,4073],{"class":3273,"line":4059},58,[3271,4061,3977],{"class":3284},[3271,4063,3289],{"class":3288},[3271,4065,3841],{"class":3292},[3271,4067,3985],{"class":3284},[3271,4069,3289],{"class":3288},[3271,4071,4072],{"class":3292},"\"100\"",[3271,4074,3540],{"class":3277},[3271,4076,4078,4080,4082,4084,4086,4089,4091,4093,4096],{"class":3273,"line":4077},59,[3271,4079,3952],{"class":3277},[3271,4081,3955],{"class":3281},[3271,4083,3510],{"class":3284},[3271,4085,3289],{"class":3288},[3271,4087,4088],{"class":3292},"\"durationColumn\"",[3271,4090,3518],{"class":3284},[3271,4092,3289],{"class":3288},[3271,4094,4095],{"class":3292},"\"Duration\"",[3271,4097,3526],{"class":3288},[3271,4099,4101,4103,4105,4107,4109,4111,4114],{"class":3273,"line":4100},60,[3271,4102,3977],{"class":3284},[3271,4104,3289],{"class":3288},[3271,4106,4031],{"class":3292},[3271,4108,3985],{"class":3284},[3271,4110,3289],{"class":3288},[3271,4112,4113],{"class":3292},"\"80\"",[3271,4115,3540],{"class":3277},[3271,4117,4119,4121,4123,4125,4127,4130,4132,4134,4137],{"class":3273,"line":4118},61,[3271,4120,3952],{"class":3277},[3271,4122,3955],{"class":3281},[3271,4124,3510],{"class":3284},[3271,4126,3289],{"class":3288},[3271,4128,4129],{"class":3292},"\"releaseYearColumn\"",[3271,4131,3518],{"class":3284},[3271,4133,3289],{"class":3288},[3271,4135,4136],{"class":3292},"\"Year\"",[3271,4138,3526],{"class":3288},[3271,4140,4142,4144,4146,4148,4150,4152,4155],{"class":3273,"line":4141},62,[3271,4143,3977],{"class":3284},[3271,4145,3289],{"class":3288},[3271,4147,4113],{"class":3292},[3271,4149,3985],{"class":3284},[3271,4151,3289],{"class":3288},[3271,4153,4154],{"class":3292},"\"60\"",[3271,4156,3540],{"class":3277},[3271,4158,4160,4163,4165],{"class":3273,"line":4159},63,[3271,4161,4162],{"class":3277},"                \u003C\u002F",[3271,4164,3944],{"class":3281},[3271,4166,3410],{"class":3277},[3271,4168,4170],{"class":3273,"line":4169},64,[3271,4171,3616],{"class":3288},[3271,4173,4175,4177,4180],{"class":3273,"line":4174},65,[3271,4176,3504],{"class":3277},[3271,4178,4179],{"class":3281},"placeholder",[3271,4181,3410],{"class":3277},[3271,4183,4185,4187,4189,4191,4193,4196],{"class":3273,"line":4184},66,[3271,4186,3952],{"class":3277},[3271,4188,3727],{"class":3281},[3271,4190,3518],{"class":3284},[3271,4192,3289],{"class":3288},[3271,4194,4195],{"class":3292},"\"No audiobooks found\"",[3271,4197,3526],{"class":3288},[3271,4199,4201,4204,4206,4209],{"class":3273,"line":4200},67,[3271,4202,4203],{"class":3284},"                           style",[3271,4205,3289],{"class":3288},[3271,4207,4208],{"class":3292},"\"-fx-text-fill: gray;\"",[3271,4210,3540],{"class":3277},[3271,4212,4214,4216,4218],{"class":3273,"line":4213},68,[3271,4215,4162],{"class":3277},[3271,4217,4179],{"class":3281},[3271,4219,3410],{"class":3277},[3271,4221,4223,4225,4227],{"class":3273,"line":4222},69,[3271,4224,3683],{"class":3277},[3271,4226,3927],{"class":3281},[3271,4228,3410],{"class":3277},[3271,4230,4232],{"class":3273,"line":4231},70,[3271,4233,3467],{"class":3288},[3271,4235,4237],{"class":3273,"line":4236},71,[3271,4238,4239],{"class":3422},"            \u003C!-- Індикатор завантаження (поверх таблиці) -->\n",[3271,4241,4243,4245,4248,4250,4252,4255],{"class":3273,"line":4242},72,[3271,4244,3479],{"class":3277},[3271,4246,4247],{"class":3281},"ProgressIndicator",[3271,4249,3510],{"class":3284},[3271,4251,3289],{"class":3288},[3271,4253,4254],{"class":3292},"\"loadingIndicator\"",[3271,4256,3526],{"class":3288},[3271,4258,4260,4263,4265,4267,4270,4272],{"class":3273,"line":4259},73,[3271,4261,4262],{"class":3284},"                               maxWidth",[3271,4264,3289],{"class":3288},[3271,4266,4072],{"class":3292},[3271,4268,4269],{"class":3284}," maxHeight",[3271,4271,3289],{"class":3288},[3271,4273,4274],{"class":3292},"\"100\"\n",[3271,4276,4278,4281,4283,4286],{"class":3273,"line":4277},74,[3271,4279,4280],{"class":3284},"                               visible",[3271,4282,3289],{"class":3288},[3271,4284,4285],{"class":3292},"\"false\"",[3271,4287,3540],{"class":3277},[3271,4289,4291],{"class":3273,"line":4290},75,[3271,4292,3467],{"class":3288},[3271,4294,4296,4298,4300],{"class":3273,"line":4295},76,[3271,4297,3863],{"class":3277},[3271,4299,3906],{"class":3281},[3271,4301,3410],{"class":3277},[3271,4303,4305,4307,4309],{"class":3273,"line":4304},77,[3271,4306,3873],{"class":3277},[3271,4308,3896],{"class":3281},[3271,4310,3410],{"class":3277},[3271,4312,4314],{"class":3273,"line":4313},78,[3271,4315,3416],{"class":3288},[3271,4317,4319],{"class":3273,"line":4318},79,[3271,4320,4321],{"class":3422},"    \u003C!-- Bottom: Статус-бар -->\n",[3271,4323,4325,4327,4330],{"class":3273,"line":4324},80,[3271,4326,3429],{"class":3277},[3271,4328,4329],{"class":3281},"bottom",[3271,4331,3410],{"class":3277},[3271,4333,4335,4337,4339,4341,4343,4345,4347,4349,4351],{"class":3273,"line":4334},81,[3271,4336,3440],{"class":3277},[3271,4338,3482],{"class":3281},[3271,4340,3446],{"class":3284},[3271,4342,3289],{"class":3288},[3271,4344,3451],{"class":3292},[3271,4346,3491],{"class":3284},[3271,4348,3289],{"class":3288},[3271,4350,3496],{"class":3292},[3271,4352,3526],{"class":3288},[3271,4354,4356,4359,4361,4364],{"class":3273,"line":4355},82,[3271,4357,4358],{"class":3284},"              style",[3271,4360,3289],{"class":3288},[3271,4362,4363],{"class":3292},"\"-fx-background-color: #e0e0e0; -fx-padding: 5 10;\"",[3271,4365,3410],{"class":3277},[3271,4367,4369],{"class":3273,"line":4368},83,[3271,4370,3467],{"class":3288},[3271,4372,4374,4376,4378,4380,4382,4385,4387,4389,4392],{"class":3273,"line":4373},84,[3271,4375,3479],{"class":3277},[3271,4377,3727],{"class":3281},[3271,4379,3510],{"class":3284},[3271,4381,3289],{"class":3288},[3271,4383,4384],{"class":3292},"\"statusLabel\"",[3271,4386,3518],{"class":3284},[3271,4388,3289],{"class":3288},[3271,4390,4391],{"class":3292},"\"Ready\"",[3271,4393,3540],{"class":3277},[3271,4395,4397],{"class":3273,"line":4396},85,[3271,4398,3467],{"class":3288},[3271,4400,4402,4404,4406,4408,4410,4412,4414],{"class":3273,"line":4401},86,[3271,4403,3479],{"class":3277},[3271,4405,3624],{"class":3281},[3271,4407,3627],{"class":3284},[3271,4409,3289],{"class":3288},[3271,4411,3632],{"class":3292},[3271,4413,3635],{"class":3277},[3271,4415,3638],{"class":3422},[3271,4417,4419],{"class":3273,"line":4418},87,[3271,4420,3467],{"class":3288},[3271,4422,4424,4426,4428,4430,4432,4435,4437,4439,4442],{"class":3273,"line":4423},88,[3271,4425,3479],{"class":3277},[3271,4427,3727],{"class":3281},[3271,4429,3510],{"class":3284},[3271,4431,3289],{"class":3288},[3271,4433,4434],{"class":3292},"\"countLabel\"",[3271,4436,3518],{"class":3284},[3271,4438,3289],{"class":3288},[3271,4440,4441],{"class":3292},"\"0 audiobooks\"",[3271,4443,3540],{"class":3277},[3271,4445,4447],{"class":3273,"line":4446},89,[3271,4448,3467],{"class":3288},[3271,4450,4452,4454,4456],{"class":3273,"line":4451},90,[3271,4453,3863],{"class":3277},[3271,4455,3482],{"class":3281},[3271,4457,3410],{"class":3277},[3271,4459,4461,4463,4465],{"class":3273,"line":4460},91,[3271,4462,3873],{"class":3277},[3271,4464,4329],{"class":3281},[3271,4466,3410],{"class":3277},[3271,4468,4470],{"class":3273,"line":4469},92,[3271,4471,3416],{"class":3288},[3271,4473,4475,4478,4480],{"class":3273,"line":4474},93,[3271,4476,4477],{"class":3277},"\u003C\u002F",[3271,4479,3369],{"class":3281},[3271,4481,3410],{"class":3277},[3250,4483,4485],{"id":4484},"розбір-fxml-структури","Розбір FXML-структури",[3150,4487,4488,4491,4492,4495,4496,4498],{},[3162,4489,4490],{},"Рядки 1-5: XML-декларація та імпорти."," ",[3154,4493,4494],{},"\u003C?import>"," — це аналог ",[3154,4497,3319],{}," у Java. Імпортуємо пакети JavaFX, що містять компоненти, які використовуємо.",[3150,4500,4501,4491,4504,4507,4508,4511,4512,4515,4516,4519],{},[3162,4502,4503],{},"Рядки 7-9: Кореневий елемент.",[3154,4505,4506],{},"\u003CBorderPane>"," — це Root Node нашого View. Атрибут ",[3154,4509,4510],{},"fx:controller"," вказує клас Controller, що керує цим FXML. ",[3154,4513,4514],{},"prefWidth"," та ",[3154,4517,4518],{},"prefHeight"," — початкові розміри вікна.",[3150,4521,4522,4491,4525,4528,4529,4531],{},[3162,4523,4524],{},"Рядки 12-42: Top — панель інструментів.",[3154,4526,4527],{},"\u003Ctop>"," — це одна з п'яти зон BorderPane. Всередині ",[3154,4530,3443],{}," з двома рядками: кнопки дій (Add, Edit, Delete, Refresh) та фільтри (TextField для пошуку, ComboBox для жанру).",[3150,4533,4534,4541,4542,4544,4545,4548],{},[3162,4535,4536,4537,4540],{},"Рядок 16: ",[3154,4538,4539],{},"fx:id"," — зв'язок з Controller."," Кожен елемент, до якого потрібен доступ з Java-коду, має ",[3154,4543,4539],{},". Це ім'я поля у Controller, що буде автоматично ініціалізоване через ",[3154,4546,4547],{},"@FXML",".",[3150,4550,4551,4491,4557,4560],{},[3162,4552,4553,4554,4556],{},"Рядок 24: ",[3154,4555,3624],{}," як Spacer.",[3154,4558,4559],{},"\u003CRegion HBox.hgrow=\"ALWAYS\"\u002F>"," — це порожній елемент, що розтягується на весь доступний простір. Він штовхає кнопку \"Refresh\" до правого краю.",[3150,4562,4563,4491,4566,4569,4570,4572,4573,4575,4576,4578,4579,4582],{},[3162,4564,4565],{},"Рядки 46-72: Center — таблиця.",[3154,4567,4568],{},"\u003Ccenter>"," містить ",[3154,4571,3906],{}," (елементи один на одному). Знизу — ",[3154,4574,3927],{},", зверху — ",[3154,4577,4247],{}," (спочатку невидимий). Коли ",[3154,4580,4581],{},"isLoading = true",", індикатор з'являється поверх таблиці.",[3150,4584,4585,4588,4589,4592,4593,4595,4596,4599,4600,4602,4603,4606],{},[3162,4586,4587],{},"Рядки 48-60: TableView з колонками."," Кожна ",[3154,4590,4591],{},"\u003CTableColumn>"," має ",[3154,4594,4539],{}," (для доступу з Controller), ",[3154,4597,4598],{},"text"," (заголовок), ",[3154,4601,4514],{}," (бажана ширина) та ",[3154,4604,4605],{},"minWidth"," (мінімальна ширина при зміні розміру).",[3150,4608,4609,4612,4613,4616],{},[3162,4610,4611],{},"Рядки 62-65: Placeholder."," Коли таблиця порожня, показується ",[3154,4614,4615],{},"\u003Cplaceholder>"," — Label з текстом \"No audiobooks found\".",[3150,4618,4619,4491,4622,4569,4625,4627,4628,4631,4632,4635,4636,4556],{},[3162,4620,4621],{},"Рядки 77-88: Bottom — статус-бар.",[3154,4623,4624],{},"\u003Cbottom>",[3154,4626,3482],{}," з двома Labels: ",[3154,4629,4630],{},"statusLabel"," (повідомлення) та ",[3154,4633,4634],{},"countLabel"," (кількість аудіокниг). Між ними — ",[3154,4637,3624],{},[4639,4640,4641,4651,4704],"tip",{},[3150,4642,4643,4646,4647,4650],{},[3162,4644,4645],{},"CSS-класи у FXML:"," Атрибут ",[3154,4648,4649],{},"styleClass=\"primary-button\""," додає CSS-клас до елемента. У окремому CSS-файлі ви можете визначити стилі для цих класів:",[3262,4652,4656],{"className":4653,"code":4654,"language":4655,"meta":3267,"style":3267},"language-css shiki shiki-themes light-plus dark-plus dark-plus",".primary-button {\n    -fx-background-color: #3b82f6;\n    -fx-text-fill: white;\n    -fx-font-weight: bold;\n}\n","css",[3154,4657,4658,4667,4679,4689,4699],{"__ignoreMap":3267},[3271,4659,4660,4664],{"class":3273,"line":3274},[3271,4661,4663],{"class":4662},"sqdDX",".primary-button",[3271,4665,4666],{"class":3288}," {\n",[3271,4668,4669,4672,4676],{"class":3273,"line":3307},[3271,4670,4671],{"class":3288},"    -fx-background-color: ",[3271,4673,4675],{"class":4674},"sDUd3","#3b82f6",[3271,4677,4678],{"class":3288},";\n",[3271,4680,4681,4684,4687],{"class":3273,"line":3314},[3271,4682,4683],{"class":3288},"    -fx-text-fill: ",[3271,4685,4686],{"class":4674},"white",[3271,4688,4678],{"class":3288},[3271,4690,4691,4694,4697],{"class":3273,"line":3330},[3271,4692,4693],{"class":3288},"    -fx-font-weight: ",[3271,4695,4696],{"class":4674},"bold",[3271,4698,4678],{"class":3288},[3271,4700,4701],{"class":3273,"line":3344},[3271,4702,4703],{"class":3288},"}\n",[3150,4705,4706],{},"Це розділяє структуру (FXML) та стилізацію (CSS).",[3236,4708],{},[3145,4710,4712],{"id":4711},"controller-мінімальний-код-з-bindings","Controller: Мінімальний код з Bindings",[3150,4714,4715,4716,4719,4720,4723],{},"Тепер створимо ",[3154,4717,4718],{},"AudiobookListController"," — Java-клас, що з'єднує FXML з ViewModel. Пам'ятайте: Controller у MVVM ",[3162,4721,4722],{},"не містить логіки"," — лише ініціалізацію Bindings та делегування подій.",[3250,4725,4727],{"id":4726},"структура-controller","Структура Controller",[3262,4729,4733],{"className":4730,"code":4731,"language":4732,"meta":3267,"style":3267},"language-java shiki shiki-themes light-plus dark-plus dark-plus","package dev.kostyl.audiobook.controller;\n\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    @FXML private TableColumn\u003CAudiobookViewModel, Integer> releaseYearColumn;\n    \n    @FXML private TextField searchField;\n    @FXML private ComboBox\u003CGenre> genreComboBox;\n    \n    @FXML private Button addButton;\n    @FXML private Button editButton;\n    @FXML private Button deleteButton;\n    @FXML private Button refreshButton;\n    \n    @FXML private ProgressIndicator loadingIndicator;\n    @FXML private Label statusLabel;\n    @FXML private Label countLabel;\n    \n    \u002F\u002F ===== ViewModel =====\n    \n    private AudiobookListViewModel viewModel;\n    \n    \u002F\u002F ===== Lifecycle =====\n    \n    @FXML\n    public void initialize() {\n        \u002F\u002F Викликається автоматично після завантаження FXML\n        \u002F\u002F Тут ініціалізуємо Bindings\n    }\n    \n    public void setViewModel(AudiobookListViewModel viewModel) {\n        this.viewModel = viewModel;\n        setupBindings();\n        viewModel.initialize(); \u002F\u002F Завантаження даних\n    }\n    \n    \u002F\u002F ===== Ініціалізація Bindings =====\n    \n    private void setupBindings() {\n        \u002F\u002F Буде реалізовано далі\n    }\n    \n    \u002F\u002F ===== Event Handlers =====\n    \n    @FXML\n    private void onAddClicked() {\n        \u002F\u002F Делегування до ViewModel або Navigator\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    @FXML\n    private void onRefreshClicked() {\n        viewModel.refresh();\n    }\n}\n","java",[3154,4734,4735,4744,4748,4755,4762,4769,4776,4780,4794,4798,4803,4807,4835,4862,4887,4912,4937,4963,4967,4983,5006,5010,5026,5041,5056,5071,5075,5091,5107,5122,5126,5131,5135,5148,5152,5157,5161,5168,5183,5188,5193,5198,5202,5221,5234,5242,5258,5262,5266,5271,5275,5286,5291,5295,5299,5304,5308,5314,5325,5330,5334,5338,5344,5355,5366,5370,5374,5380,5391,5402,5406,5410,5416,5427,5438,5442],{"__ignoreMap":3267},[3271,4736,4737,4741],{"class":3273,"line":3274},[3271,4738,4740],{"class":4739},"su1O8","package",[3271,4742,4743],{"class":3288}," dev.kostyl.audiobook.controller;\n",[3271,4745,4746],{"class":3273,"line":3307},[3271,4747,3311],{"emptyLinePlaceholder":3310},[3271,4749,4750,4752],{"class":3273,"line":3314},[3271,4751,3319],{"class":4739},[3271,4753,4754],{"class":3288}," dev.kostyl.audiobook.viewmodel.AudiobookListViewModel;\n",[3271,4756,4757,4759],{"class":3273,"line":3330},[3271,4758,3319],{"class":4739},[3271,4760,4761],{"class":3288}," dev.kostyl.audiobook.viewmodel.AudiobookViewModel;\n",[3271,4763,4764,4766],{"class":3273,"line":3344},[3271,4765,3319],{"class":4739},[3271,4767,4768],{"class":3288}," javafx.fxml.FXML;\n",[3271,4770,4771,4773],{"class":3273,"line":3358},[3271,4772,3319],{"class":4739},[3271,4774,4775],{"class":3288}," javafx.scene.control.*;\n",[3271,4777,4778],{"class":3273,"line":3363},[3271,4779,3311],{"emptyLinePlaceholder":3310},[3271,4781,4782,4785,4788,4792],{"class":3273,"line":3380},[3271,4783,4784],{"class":4739},"public",[3271,4786,4787],{"class":4739}," class",[3271,4789,4791],{"class":4790},"sN1BT"," AudiobookListController",[3271,4793,4666],{"class":3288},[3271,4795,4796],{"class":3273,"line":3391},[3271,4797,3416],{"class":3288},[3271,4799,4800],{"class":3273,"line":3413},[3271,4801,4802],{"class":3422},"    \u002F\u002F ===== FXML-ін'єкція UI-елементів =====\n",[3271,4804,4805],{"class":3273,"line":3419},[3271,4806,3416],{"class":3288},[3271,4808,4809,4812,4815,4818,4821,4823,4826,4829,4833],{"class":3273,"line":3426},[3271,4810,4811],{"class":3288},"    @",[3271,4813,4814],{"class":4790},"FXML",[3271,4816,4817],{"class":4739}," private",[3271,4819,4820],{"class":4790}," TableView",[3271,4822,3366],{"class":3288},[3271,4824,4825],{"class":4790},"AudiobookViewModel",[3271,4827,4828],{"class":3288},"> ",[3271,4830,4832],{"class":4831},"siwwj","audiobookTable",[3271,4834,4678],{"class":3288},[3271,4836,4837,4839,4841,4843,4846,4848,4850,4852,4855,4857,4860],{"class":3273,"line":3437},[3271,4838,4811],{"class":3288},[3271,4840,4814],{"class":4790},[3271,4842,4817],{"class":4739},[3271,4844,4845],{"class":4790}," TableColumn",[3271,4847,3366],{"class":3288},[3271,4849,4825],{"class":4790},[3271,4851,3223],{"class":3288},[3271,4853,4854],{"class":4790},"String",[3271,4856,4828],{"class":3288},[3271,4858,4859],{"class":4831},"titleColumn",[3271,4861,4678],{"class":3288},[3271,4863,4864,4866,4868,4870,4872,4874,4876,4878,4880,4882,4885],{"class":3273,"line":3464},[3271,4865,4811],{"class":3288},[3271,4867,4814],{"class":4790},[3271,4869,4817],{"class":4739},[3271,4871,4845],{"class":4790},[3271,4873,3366],{"class":3288},[3271,4875,4825],{"class":4790},[3271,4877,3223],{"class":3288},[3271,4879,4854],{"class":4790},[3271,4881,4828],{"class":3288},[3271,4883,4884],{"class":4831},"authorColumn",[3271,4886,4678],{"class":3288},[3271,4888,4889,4891,4893,4895,4897,4899,4901,4903,4905,4907,4910],{"class":3273,"line":3470},[3271,4890,4811],{"class":3288},[3271,4892,4814],{"class":4790},[3271,4894,4817],{"class":4739},[3271,4896,4845],{"class":4790},[3271,4898,3366],{"class":3288},[3271,4900,4825],{"class":4790},[3271,4902,3223],{"class":3288},[3271,4904,4854],{"class":4790},[3271,4906,4828],{"class":3288},[3271,4908,4909],{"class":4831},"genreColumn",[3271,4911,4678],{"class":3288},[3271,4913,4914,4916,4918,4920,4922,4924,4926,4928,4930,4932,4935],{"class":3273,"line":3476},[3271,4915,4811],{"class":3288},[3271,4917,4814],{"class":4790},[3271,4919,4817],{"class":4739},[3271,4921,4845],{"class":4790},[3271,4923,3366],{"class":3288},[3271,4925,4825],{"class":4790},[3271,4927,3223],{"class":3288},[3271,4929,4854],{"class":4790},[3271,4931,4828],{"class":3288},[3271,4933,4934],{"class":4831},"durationColumn",[3271,4936,4678],{"class":3288},[3271,4938,4939,4941,4943,4945,4947,4949,4951,4953,4956,4958,4961],{"class":3273,"line":3501},[3271,4940,4811],{"class":3288},[3271,4942,4814],{"class":4790},[3271,4944,4817],{"class":4739},[3271,4946,4845],{"class":4790},[3271,4948,3366],{"class":3288},[3271,4950,4825],{"class":4790},[3271,4952,3223],{"class":3288},[3271,4954,4955],{"class":4790},"Integer",[3271,4957,4828],{"class":3288},[3271,4959,4960],{"class":4831},"releaseYearColumn",[3271,4962,4678],{"class":3288},[3271,4964,4965],{"class":3273,"line":3529},[3271,4966,3416],{"class":3288},[3271,4968,4969,4971,4973,4975,4978,4981],{"class":3273,"line":3543},[3271,4970,4811],{"class":3288},[3271,4972,4814],{"class":4790},[3271,4974,4817],{"class":4739},[3271,4976,4977],{"class":4790}," TextField",[3271,4979,4980],{"class":4831}," searchField",[3271,4982,4678],{"class":3288},[3271,4984,4985,4987,4989,4991,4994,4996,4999,5001,5004],{"class":3273,"line":3566},[3271,4986,4811],{"class":3288},[3271,4988,4814],{"class":4790},[3271,4990,4817],{"class":4739},[3271,4992,4993],{"class":4790}," ComboBox",[3271,4995,3366],{"class":3288},[3271,4997,4998],{"class":4790},"Genre",[3271,5000,4828],{"class":3288},[3271,5002,5003],{"class":4831},"genreComboBox",[3271,5005,4678],{"class":3288},[3271,5007,5008],{"class":3273,"line":3578},[3271,5009,3416],{"class":3288},[3271,5011,5012,5014,5016,5018,5021,5024],{"class":3273,"line":3601},[3271,5013,4811],{"class":3288},[3271,5015,4814],{"class":4790},[3271,5017,4817],{"class":4739},[3271,5019,5020],{"class":4790}," Button",[3271,5022,5023],{"class":4831}," addButton",[3271,5025,4678],{"class":3288},[3271,5027,5028,5030,5032,5034,5036,5039],{"class":3273,"line":3613},[3271,5029,4811],{"class":3288},[3271,5031,4814],{"class":4790},[3271,5033,4817],{"class":4739},[3271,5035,5020],{"class":4790},[3271,5037,5038],{"class":4831}," editButton",[3271,5040,4678],{"class":3288},[3271,5042,5043,5045,5047,5049,5051,5054],{"class":3273,"line":3619},[3271,5044,4811],{"class":3288},[3271,5046,4814],{"class":4790},[3271,5048,4817],{"class":4739},[3271,5050,5020],{"class":4790},[3271,5052,5053],{"class":4831}," deleteButton",[3271,5055,4678],{"class":3288},[3271,5057,5058,5060,5062,5064,5066,5069],{"class":3273,"line":3641},[3271,5059,4811],{"class":3288},[3271,5061,4814],{"class":4790},[3271,5063,4817],{"class":4739},[3271,5065,5020],{"class":4790},[3271,5067,5068],{"class":4831}," refreshButton",[3271,5070,4678],{"class":3288},[3271,5072,5073],{"class":3273,"line":3646},[3271,5074,3416],{"class":3288},[3271,5076,5077,5079,5081,5083,5086,5089],{"class":3273,"line":3669},[3271,5078,4811],{"class":3288},[3271,5080,4814],{"class":4790},[3271,5082,4817],{"class":4739},[3271,5084,5085],{"class":4790}," ProgressIndicator",[3271,5087,5088],{"class":4831}," loadingIndicator",[3271,5090,4678],{"class":3288},[3271,5092,5093,5095,5097,5099,5102,5105],{"class":3273,"line":3680},[3271,5094,4811],{"class":3288},[3271,5096,4814],{"class":4790},[3271,5098,4817],{"class":4739},[3271,5100,5101],{"class":4790}," Label",[3271,5103,5104],{"class":4831}," statusLabel",[3271,5106,4678],{"class":3288},[3271,5108,5109,5111,5113,5115,5117,5120],{"class":3273,"line":3690},[3271,5110,4811],{"class":3288},[3271,5112,4814],{"class":4790},[3271,5114,4817],{"class":4739},[3271,5116,5101],{"class":4790},[3271,5118,5119],{"class":4831}," countLabel",[3271,5121,4678],{"class":3288},[3271,5123,5124],{"class":3273,"line":3695},[3271,5125,3416],{"class":3288},[3271,5127,5128],{"class":3273,"line":3701},[3271,5129,5130],{"class":3422},"    \u002F\u002F ===== ViewModel =====\n",[3271,5132,5133],{"class":3273,"line":3722},[3271,5134,3416],{"class":3288},[3271,5136,5137,5140,5143,5146],{"class":3273,"line":3739},[3271,5138,5139],{"class":4739},"    private",[3271,5141,5142],{"class":4790}," AudiobookListViewModel",[3271,5144,5145],{"class":4831}," viewModel",[3271,5147,4678],{"class":3288},[3271,5149,5150],{"class":3273,"line":3756},[3271,5151,3416],{"class":3288},[3271,5153,5154],{"class":3273,"line":3769},[3271,5155,5156],{"class":3422},"    \u002F\u002F ===== Lifecycle =====\n",[3271,5158,5159],{"class":3273,"line":3782},[3271,5160,3416],{"class":3288},[3271,5162,5163,5165],{"class":3273,"line":3787},[3271,5164,4811],{"class":3288},[3271,5166,5167],{"class":4790},"FXML\n",[3271,5169,5170,5173,5176,5180],{"class":3273,"line":3803},[3271,5171,5172],{"class":4739},"    public",[3271,5174,5175],{"class":4790}," void",[3271,5177,5179],{"class":5178},"s8Opu"," initialize",[3271,5181,5182],{"class":3288},"() {\n",[3271,5184,5185],{"class":3273,"line":3820},[3271,5186,5187],{"class":3422},"        \u002F\u002F Викликається автоматично після завантаження FXML\n",[3271,5189,5190],{"class":3273,"line":3833},[3271,5191,5192],{"class":3422},"        \u002F\u002F Тут ініціалізуємо Bindings\n",[3271,5194,5195],{"class":3273,"line":3846},[3271,5196,5197],{"class":3288},"    }\n",[3271,5199,5200],{"class":3273,"line":3855},[3271,5201,3416],{"class":3288},[3271,5203,5204,5206,5208,5211,5214,5216,5218],{"class":3273,"line":3860},[3271,5205,5172],{"class":4739},[3271,5207,5175],{"class":4790},[3271,5209,5210],{"class":5178}," setViewModel",[3271,5212,5213],{"class":3288},"(",[3271,5215,3156],{"class":4790},[3271,5217,5145],{"class":4831},[3271,5219,5220],{"class":3288},") {\n",[3271,5222,5223,5226,5228,5231],{"class":3273,"line":3870},[3271,5224,5225],{"class":4739},"        this",[3271,5227,4548],{"class":3288},[3271,5229,5230],{"class":4831},"viewModel",[3271,5232,5233],{"class":3288}," = viewModel;\n",[3271,5235,5236,5239],{"class":3273,"line":3880},[3271,5237,5238],{"class":5178},"        setupBindings",[3271,5240,5241],{"class":3288},"();\n",[3271,5243,5244,5247,5249,5252,5255],{"class":3273,"line":3885},[3271,5245,5246],{"class":4831},"        viewModel",[3271,5248,4548],{"class":3288},[3271,5250,5251],{"class":5178},"initialize",[3271,5253,5254],{"class":3288},"(); ",[3271,5256,5257],{"class":3422},"\u002F\u002F Завантаження даних\n",[3271,5259,5260],{"class":3273,"line":3891},[3271,5261,5197],{"class":3288},[3271,5263,5264],{"class":3273,"line":3901},[3271,5265,3416],{"class":3288},[3271,5267,5268],{"class":3273,"line":3911},[3271,5269,5270],{"class":3422},"    \u002F\u002F ===== Ініціалізація Bindings =====\n",[3271,5272,5273],{"class":3273,"line":3916},[3271,5274,3416],{"class":3288},[3271,5276,5277,5279,5281,5284],{"class":3273,"line":3922},[3271,5278,5139],{"class":4739},[3271,5280,5175],{"class":4790},[3271,5282,5283],{"class":5178}," setupBindings",[3271,5285,5182],{"class":3288},[3271,5287,5288],{"class":3273,"line":3939},[3271,5289,5290],{"class":3422},"        \u002F\u002F Буде реалізовано далі\n",[3271,5292,5293],{"class":3273,"line":3949},[3271,5294,5197],{"class":3288},[3271,5296,5297],{"class":3273,"line":3974},[3271,5298,3416],{"class":3288},[3271,5300,5301],{"class":3273,"line":3994},[3271,5302,5303],{"class":3422},"    \u002F\u002F ===== Event Handlers =====\n",[3271,5305,5306],{"class":3273,"line":4017},[3271,5307,3416],{"class":3288},[3271,5309,5310,5312],{"class":3273,"line":4036},[3271,5311,4811],{"class":3288},[3271,5313,5167],{"class":4790},[3271,5315,5316,5318,5320,5323],{"class":3273,"line":4059},[3271,5317,5139],{"class":4739},[3271,5319,5175],{"class":4790},[3271,5321,5322],{"class":5178}," onAddClicked",[3271,5324,5182],{"class":3288},[3271,5326,5327],{"class":3273,"line":4077},[3271,5328,5329],{"class":3422},"        \u002F\u002F Делегування до ViewModel або Navigator\n",[3271,5331,5332],{"class":3273,"line":4100},[3271,5333,5197],{"class":3288},[3271,5335,5336],{"class":3273,"line":4118},[3271,5337,3416],{"class":3288},[3271,5339,5340,5342],{"class":3273,"line":4141},[3271,5341,4811],{"class":3288},[3271,5343,5167],{"class":4790},[3271,5345,5346,5348,5350,5353],{"class":3273,"line":4159},[3271,5347,5139],{"class":4739},[3271,5349,5175],{"class":4790},[3271,5351,5352],{"class":5178}," onEditClicked",[3271,5354,5182],{"class":3288},[3271,5356,5357,5359,5361,5364],{"class":3273,"line":4169},[3271,5358,5246],{"class":4831},[3271,5360,4548],{"class":3288},[3271,5362,5363],{"class":5178},"editSelected",[3271,5365,5241],{"class":3288},[3271,5367,5368],{"class":3273,"line":4174},[3271,5369,5197],{"class":3288},[3271,5371,5372],{"class":3273,"line":4184},[3271,5373,3416],{"class":3288},[3271,5375,5376,5378],{"class":3273,"line":4200},[3271,5377,4811],{"class":3288},[3271,5379,5167],{"class":4790},[3271,5381,5382,5384,5386,5389],{"class":3273,"line":4213},[3271,5383,5139],{"class":4739},[3271,5385,5175],{"class":4790},[3271,5387,5388],{"class":5178}," onDeleteClicked",[3271,5390,5182],{"class":3288},[3271,5392,5393,5395,5397,5400],{"class":3273,"line":4222},[3271,5394,5246],{"class":4831},[3271,5396,4548],{"class":3288},[3271,5398,5399],{"class":5178},"deleteSelected",[3271,5401,5241],{"class":3288},[3271,5403,5404],{"class":3273,"line":4231},[3271,5405,5197],{"class":3288},[3271,5407,5408],{"class":3273,"line":4236},[3271,5409,3416],{"class":3288},[3271,5411,5412,5414],{"class":3273,"line":4242},[3271,5413,4811],{"class":3288},[3271,5415,5167],{"class":4790},[3271,5417,5418,5420,5422,5425],{"class":3273,"line":4259},[3271,5419,5139],{"class":4739},[3271,5421,5175],{"class":4790},[3271,5423,5424],{"class":5178}," onRefreshClicked",[3271,5426,5182],{"class":3288},[3271,5428,5429,5431,5433,5436],{"class":3273,"line":4277},[3271,5430,5246],{"class":4831},[3271,5432,4548],{"class":3288},[3271,5434,5435],{"class":5178},"refresh",[3271,5437,5241],{"class":3288},[3271,5439,5440],{"class":3273,"line":4290},[3271,5441,5197],{"class":3288},[3271,5443,5444],{"class":3273,"line":4295},[3271,5445,4703],{"class":3288},[3150,5447,5448],{},[3162,5449,5450],{},"Розбір структури:",[3150,5452,5453,5456,5457,5459,5460,5462,5463,5466,5467,4548],{},[3162,5454,5455],{},"Рядки 12-29: FXML-ін'єкція."," Кожне поле з ",[3154,5458,4547],{}," автоматично ініціалізується FXMLLoader, коли він знаходить елемент з відповідним ",[3154,5461,4539],{}," у FXML. Ім'я поля має ",[3162,5464,5465],{},"точно співпадати"," з ",[3154,5468,4539],{},[3150,5470,5471,5474,5475,5478],{},[3162,5472,5473],{},"Рядок 33: ViewModel."," Це єдина залежність Controller. Вона передається ззовні через ",[3154,5476,5477],{},"setViewModel()"," (Setter Injection). У реальному додатку це буде через Guice (Constructor Injection).",[3150,5480,5481,5487,5488,5490],{},[3162,5482,5483,5484,4548],{},"Рядки 38-42: ",[3154,5485,5486],{},"initialize()"," Цей метод викликається автоматично після завантаження FXML та ін'єкції всіх ",[3154,5489,4547],{}," полів. Тут можна ініціалізувати те, що не залежить від ViewModel (наприклад, налаштування колонок таблиці).",[3150,5492,5493,5498,5499,5502],{},[3162,5494,5495,5496,4548],{},"Рядки 44-48: ",[3154,5497,5477],{}," Цей метод викликається ззовні (з Application або через Guice) для передачі ViewModel. Після отримання ViewModel ми ініціалізуємо Bindings та викликаємо ",[3154,5500,5501],{},"viewModel.initialize()"," для завантаження даних.",[3150,5504,5505,5511],{},[3162,5506,5507,5508,4548],{},"Рядки 52-55: ",[3154,5509,5510],{},"setupBindings()"," Тут відбувається магія MVVM — з'єднання UI-елементів з ViewModel Properties через Bindings. Це серце Controller.",[3150,5513,5514,5517,5518,5520,5521,5524,5525,5527,5528,4548],{},[3162,5515,5516],{},"Рядки 59-73: Event Handlers."," Методи з ",[3154,5519,4547],{},", що викликаються при натисканні кнопок. Вони ",[3162,5522,5523],{},"не містять логіки"," — лише делегують виклик до ViewModel. Зверніть увагу: жодних ",[3154,5526,3222],{},", жодних обчислень — лише ",[3154,5529,5530],{},"viewModel.method()",[3250,5532,5534],{"id":5533},"ініціалізація-bindings-setupbindings","Ініціалізація Bindings: setupBindings()",[3150,5536,5537,5538,5540],{},"Тепер реалізуємо ",[3154,5539,5510],{}," — найважливіший метод Controller:",[3262,5542,5544],{"className":4730,"code":5543,"language":4732,"meta":3267,"style":3267},"private void setupBindings() {\n    \u002F\u002F ===== Налаштування колонок таблиці =====\n    \n    titleColumn.setCellValueFactory(cellData -> \n        cellData.getValue().titleProperty());\n    \n    authorColumn.setCellValueFactory(cellData -> \n        cellData.getValue().authorNameProperty());\n    \n    genreColumn.setCellValueFactory(cellData -> \n        cellData.getValue().genreNameProperty());\n    \n    durationColumn.setCellValueFactory(cellData -> \n        cellData.getValue().formattedDurationProperty());\n    \n    releaseYearColumn.setCellValueFactory(cellData -> \n        cellData.getValue().releaseYearProperty().asObject());\n    \n    \u002F\u002F ===== Bindings: ViewModel → View (Unidirectional) =====\n    \n    \u002F\u002F Таблиця показує sortedAudiobooks з ViewModel\n    audiobookTable.setItems(viewModel.getSortedAudiobooks());\n    \n    \u002F\u002F Прив'язка сортування таблиці до SortedList\n    viewModel.getSortedAudiobooks().comparatorProperty()\n        .bind(audiobookTable.comparatorProperty());\n    \n    \u002F\u002F Індикатор завантаження видимий, коли isLoading = true\n    loadingIndicator.visibleProperty().bind(viewModel.isLoadingProperty());\n    \n    \u002F\u002F Статус-бар показує statusMessage\n    statusLabel.textProperty().bind(viewModel.statusMessageProperty());\n    \n    \u002F\u002F Лічильник показує \"Showing X of Y audiobooks\"\n    countLabel.textProperty().bind(\n        Bindings.concat(\"Showing \", \n            viewModel.filteredCountProperty(), \n            \" of \", \n            viewModel.totalCountProperty(), \n            \" audiobooks\")\n    );\n    \n    \u002F\u002F Кнопки Edit та Delete активні лише коли щось обрано\n    editButton.disableProperty().bind(\n        viewModel.deleteButtonEnabledProperty().not());\n    deleteButton.disableProperty().bind(\n        viewModel.deleteButtonEnabledProperty().not());\n    \n    \u002F\u002F ===== Bindings: View ↔ ViewModel (Bidirectional) =====\n    \n    \u002F\u002F Пошуковий запит синхронізується в обидва боки\n    searchField.textProperty().bindBidirectional(\n        viewModel.searchQueryProperty());\n    \n    \u002F\u002F Обраний жанр синхронізується в обидва боки\n    genreComboBox.valueProperty().bindBidirectional(\n        viewModel.selectedGenreProperty());\n    \n    \u002F\u002F Обраний елемент таблиці синхронізується в обидва боки\n    audiobookTable.getSelectionModel().selectedItemProperty()\n        .bindBidirectional(viewModel.selectedAudiobookProperty());\n    \n    \u002F\u002F ===== Завантаження жанрів у ComboBox =====\n    \n    \u002F\u002F У реальному додатку це буде через окремий ViewModel\n    genreComboBox.getItems().addAll(\n        new Genre(\"Fiction\"),\n        new Genre(\"Non-Fiction\"),\n        new Genre(\"Science\"),\n        new Genre(\"History\")\n    );\n}\n",[3154,5545,5546,5557,5562,5566,5584,5603,5607,5622,5637,5641,5656,5671,5675,5690,5705,5709,5724,5744,5748,5753,5757,5762,5783,5787,5792,5809,5827,5831,5836,5861,5865,5870,5895,5899,5904,5920,5939,5952,5959,5970,5978,5983,5987,5992,6008,6024,6039,6053,6057,6062,6066,6071,6087,6098,6102,6107,6123,6134,6138,6143,6159,6176,6180,6185,6189,6194,6210,6227,6240,6253,6266,6270],{"__ignoreMap":3267},[3271,5547,5548,5551,5553,5555],{"class":3273,"line":3274},[3271,5549,5550],{"class":4739},"private",[3271,5552,5175],{"class":4790},[3271,5554,5283],{"class":5178},[3271,5556,5182],{"class":3288},[3271,5558,5559],{"class":3273,"line":3307},[3271,5560,5561],{"class":3422},"    \u002F\u002F ===== Налаштування колонок таблиці =====\n",[3271,5563,5564],{"class":3273,"line":3314},[3271,5565,3416],{"class":3288},[3271,5567,5568,5571,5573,5576,5579,5582],{"class":3273,"line":3330},[3271,5569,5570],{"class":4831},"    titleColumn",[3271,5572,4548],{"class":3288},[3271,5574,5575],{"class":5178},"setCellValueFactory",[3271,5577,5578],{"class":3288},"(cellData ",[3271,5580,5581],{"class":4739},"->",[3271,5583,3526],{"class":3288},[3271,5585,5586,5589,5591,5594,5597,5600],{"class":3273,"line":3344},[3271,5587,5588],{"class":4831},"        cellData",[3271,5590,4548],{"class":3288},[3271,5592,5593],{"class":5178},"getValue",[3271,5595,5596],{"class":3288},"().",[3271,5598,5599],{"class":5178},"titleProperty",[3271,5601,5602],{"class":3288},"());\n",[3271,5604,5605],{"class":3273,"line":3358},[3271,5606,3416],{"class":3288},[3271,5608,5609,5612,5614,5616,5618,5620],{"class":3273,"line":3363},[3271,5610,5611],{"class":4831},"    authorColumn",[3271,5613,4548],{"class":3288},[3271,5615,5575],{"class":5178},[3271,5617,5578],{"class":3288},[3271,5619,5581],{"class":4739},[3271,5621,3526],{"class":3288},[3271,5623,5624,5626,5628,5630,5632,5635],{"class":3273,"line":3380},[3271,5625,5588],{"class":4831},[3271,5627,4548],{"class":3288},[3271,5629,5593],{"class":5178},[3271,5631,5596],{"class":3288},[3271,5633,5634],{"class":5178},"authorNameProperty",[3271,5636,5602],{"class":3288},[3271,5638,5639],{"class":3273,"line":3391},[3271,5640,3416],{"class":3288},[3271,5642,5643,5646,5648,5650,5652,5654],{"class":3273,"line":3413},[3271,5644,5645],{"class":4831},"    genreColumn",[3271,5647,4548],{"class":3288},[3271,5649,5575],{"class":5178},[3271,5651,5578],{"class":3288},[3271,5653,5581],{"class":4739},[3271,5655,3526],{"class":3288},[3271,5657,5658,5660,5662,5664,5666,5669],{"class":3273,"line":3419},[3271,5659,5588],{"class":4831},[3271,5661,4548],{"class":3288},[3271,5663,5593],{"class":5178},[3271,5665,5596],{"class":3288},[3271,5667,5668],{"class":5178},"genreNameProperty",[3271,5670,5602],{"class":3288},[3271,5672,5673],{"class":3273,"line":3426},[3271,5674,3416],{"class":3288},[3271,5676,5677,5680,5682,5684,5686,5688],{"class":3273,"line":3437},[3271,5678,5679],{"class":4831},"    durationColumn",[3271,5681,4548],{"class":3288},[3271,5683,5575],{"class":5178},[3271,5685,5578],{"class":3288},[3271,5687,5581],{"class":4739},[3271,5689,3526],{"class":3288},[3271,5691,5692,5694,5696,5698,5700,5703],{"class":3273,"line":3464},[3271,5693,5588],{"class":4831},[3271,5695,4548],{"class":3288},[3271,5697,5593],{"class":5178},[3271,5699,5596],{"class":3288},[3271,5701,5702],{"class":5178},"formattedDurationProperty",[3271,5704,5602],{"class":3288},[3271,5706,5707],{"class":3273,"line":3470},[3271,5708,3416],{"class":3288},[3271,5710,5711,5714,5716,5718,5720,5722],{"class":3273,"line":3476},[3271,5712,5713],{"class":4831},"    releaseYearColumn",[3271,5715,4548],{"class":3288},[3271,5717,5575],{"class":5178},[3271,5719,5578],{"class":3288},[3271,5721,5581],{"class":4739},[3271,5723,3526],{"class":3288},[3271,5725,5726,5728,5730,5732,5734,5737,5739,5742],{"class":3273,"line":3501},[3271,5727,5588],{"class":4831},[3271,5729,4548],{"class":3288},[3271,5731,5593],{"class":5178},[3271,5733,5596],{"class":3288},[3271,5735,5736],{"class":5178},"releaseYearProperty",[3271,5738,5596],{"class":3288},[3271,5740,5741],{"class":5178},"asObject",[3271,5743,5602],{"class":3288},[3271,5745,5746],{"class":3273,"line":3529},[3271,5747,3416],{"class":3288},[3271,5749,5750],{"class":3273,"line":3543},[3271,5751,5752],{"class":3422},"    \u002F\u002F ===== Bindings: ViewModel → View (Unidirectional) =====\n",[3271,5754,5755],{"class":3273,"line":3566},[3271,5756,3416],{"class":3288},[3271,5758,5759],{"class":3273,"line":3578},[3271,5760,5761],{"class":3422},"    \u002F\u002F Таблиця показує sortedAudiobooks з ViewModel\n",[3271,5763,5764,5767,5769,5772,5774,5776,5778,5781],{"class":3273,"line":3601},[3271,5765,5766],{"class":4831},"    audiobookTable",[3271,5768,4548],{"class":3288},[3271,5770,5771],{"class":5178},"setItems",[3271,5773,5213],{"class":3288},[3271,5775,5230],{"class":4831},[3271,5777,4548],{"class":3288},[3271,5779,5780],{"class":5178},"getSortedAudiobooks",[3271,5782,5602],{"class":3288},[3271,5784,5785],{"class":3273,"line":3613},[3271,5786,3416],{"class":3288},[3271,5788,5789],{"class":3273,"line":3619},[3271,5790,5791],{"class":3422},"    \u002F\u002F Прив'язка сортування таблиці до SortedList\n",[3271,5793,5794,5797,5799,5801,5803,5806],{"class":3273,"line":3641},[3271,5795,5796],{"class":4831},"    viewModel",[3271,5798,4548],{"class":3288},[3271,5800,5780],{"class":5178},[3271,5802,5596],{"class":3288},[3271,5804,5805],{"class":5178},"comparatorProperty",[3271,5807,5808],{"class":3288},"()\n",[3271,5810,5811,5814,5817,5819,5821,5823,5825],{"class":3273,"line":3646},[3271,5812,5813],{"class":3288},"        .",[3271,5815,5816],{"class":5178},"bind",[3271,5818,5213],{"class":3288},[3271,5820,4832],{"class":4831},[3271,5822,4548],{"class":3288},[3271,5824,5805],{"class":5178},[3271,5826,5602],{"class":3288},[3271,5828,5829],{"class":3273,"line":3669},[3271,5830,3416],{"class":3288},[3271,5832,5833],{"class":3273,"line":3680},[3271,5834,5835],{"class":3422},"    \u002F\u002F Індикатор завантаження видимий, коли isLoading = true\n",[3271,5837,5838,5841,5843,5846,5848,5850,5852,5854,5856,5859],{"class":3273,"line":3690},[3271,5839,5840],{"class":4831},"    loadingIndicator",[3271,5842,4548],{"class":3288},[3271,5844,5845],{"class":5178},"visibleProperty",[3271,5847,5596],{"class":3288},[3271,5849,5816],{"class":5178},[3271,5851,5213],{"class":3288},[3271,5853,5230],{"class":4831},[3271,5855,4548],{"class":3288},[3271,5857,5858],{"class":5178},"isLoadingProperty",[3271,5860,5602],{"class":3288},[3271,5862,5863],{"class":3273,"line":3695},[3271,5864,3416],{"class":3288},[3271,5866,5867],{"class":3273,"line":3701},[3271,5868,5869],{"class":3422},"    \u002F\u002F Статус-бар показує statusMessage\n",[3271,5871,5872,5875,5877,5880,5882,5884,5886,5888,5890,5893],{"class":3273,"line":3722},[3271,5873,5874],{"class":4831},"    statusLabel",[3271,5876,4548],{"class":3288},[3271,5878,5879],{"class":5178},"textProperty",[3271,5881,5596],{"class":3288},[3271,5883,5816],{"class":5178},[3271,5885,5213],{"class":3288},[3271,5887,5230],{"class":4831},[3271,5889,4548],{"class":3288},[3271,5891,5892],{"class":5178},"statusMessageProperty",[3271,5894,5602],{"class":3288},[3271,5896,5897],{"class":3273,"line":3739},[3271,5898,3416],{"class":3288},[3271,5900,5901],{"class":3273,"line":3756},[3271,5902,5903],{"class":3422},"    \u002F\u002F Лічильник показує \"Showing X of Y audiobooks\"\n",[3271,5905,5906,5909,5911,5913,5915,5917],{"class":3273,"line":3769},[3271,5907,5908],{"class":4831},"    countLabel",[3271,5910,4548],{"class":3288},[3271,5912,5879],{"class":5178},[3271,5914,5596],{"class":3288},[3271,5916,5816],{"class":5178},[3271,5918,5919],{"class":3288},"(\n",[3271,5921,5922,5925,5927,5930,5932,5936],{"class":3273,"line":3782},[3271,5923,5924],{"class":4831},"        Bindings",[3271,5926,4548],{"class":3288},[3271,5928,5929],{"class":5178},"concat",[3271,5931,5213],{"class":3288},[3271,5933,5935],{"class":5934},"sbdoH","\"Showing \"",[3271,5937,5938],{"class":3288},", \n",[3271,5940,5941,5944,5946,5949],{"class":3273,"line":3787},[3271,5942,5943],{"class":4831},"            viewModel",[3271,5945,4548],{"class":3288},[3271,5947,5948],{"class":5178},"filteredCountProperty",[3271,5950,5951],{"class":3288},"(), \n",[3271,5953,5954,5957],{"class":3273,"line":3803},[3271,5955,5956],{"class":5934},"            \" of \"",[3271,5958,5938],{"class":3288},[3271,5960,5961,5963,5965,5968],{"class":3273,"line":3820},[3271,5962,5943],{"class":4831},[3271,5964,4548],{"class":3288},[3271,5966,5967],{"class":5178},"totalCountProperty",[3271,5969,5951],{"class":3288},[3271,5971,5972,5975],{"class":3273,"line":3833},[3271,5973,5974],{"class":5934},"            \" audiobooks\"",[3271,5976,5977],{"class":3288},")\n",[3271,5979,5980],{"class":3273,"line":3846},[3271,5981,5982],{"class":3288},"    );\n",[3271,5984,5985],{"class":3273,"line":3855},[3271,5986,3416],{"class":3288},[3271,5988,5989],{"class":3273,"line":3860},[3271,5990,5991],{"class":3422},"    \u002F\u002F Кнопки Edit та Delete активні лише коли щось обрано\n",[3271,5993,5994,5997,5999,6002,6004,6006],{"class":3273,"line":3870},[3271,5995,5996],{"class":4831},"    editButton",[3271,5998,4548],{"class":3288},[3271,6000,6001],{"class":5178},"disableProperty",[3271,6003,5596],{"class":3288},[3271,6005,5816],{"class":5178},[3271,6007,5919],{"class":3288},[3271,6009,6010,6012,6014,6017,6019,6022],{"class":3273,"line":3880},[3271,6011,5246],{"class":4831},[3271,6013,4548],{"class":3288},[3271,6015,6016],{"class":5178},"deleteButtonEnabledProperty",[3271,6018,5596],{"class":3288},[3271,6020,6021],{"class":5178},"not",[3271,6023,5602],{"class":3288},[3271,6025,6026,6029,6031,6033,6035,6037],{"class":3273,"line":3885},[3271,6027,6028],{"class":4831},"    deleteButton",[3271,6030,4548],{"class":3288},[3271,6032,6001],{"class":5178},[3271,6034,5596],{"class":3288},[3271,6036,5816],{"class":5178},[3271,6038,5919],{"class":3288},[3271,6040,6041,6043,6045,6047,6049,6051],{"class":3273,"line":3891},[3271,6042,5246],{"class":4831},[3271,6044,4548],{"class":3288},[3271,6046,6016],{"class":5178},[3271,6048,5596],{"class":3288},[3271,6050,6021],{"class":5178},[3271,6052,5602],{"class":3288},[3271,6054,6055],{"class":3273,"line":3901},[3271,6056,3416],{"class":3288},[3271,6058,6059],{"class":3273,"line":3911},[3271,6060,6061],{"class":3422},"    \u002F\u002F ===== Bindings: View ↔ ViewModel (Bidirectional) =====\n",[3271,6063,6064],{"class":3273,"line":3916},[3271,6065,3416],{"class":3288},[3271,6067,6068],{"class":3273,"line":3922},[3271,6069,6070],{"class":3422},"    \u002F\u002F Пошуковий запит синхронізується в обидва боки\n",[3271,6072,6073,6076,6078,6080,6082,6085],{"class":3273,"line":3939},[3271,6074,6075],{"class":4831},"    searchField",[3271,6077,4548],{"class":3288},[3271,6079,5879],{"class":5178},[3271,6081,5596],{"class":3288},[3271,6083,6084],{"class":5178},"bindBidirectional",[3271,6086,5919],{"class":3288},[3271,6088,6089,6091,6093,6096],{"class":3273,"line":3949},[3271,6090,5246],{"class":4831},[3271,6092,4548],{"class":3288},[3271,6094,6095],{"class":5178},"searchQueryProperty",[3271,6097,5602],{"class":3288},[3271,6099,6100],{"class":3273,"line":3974},[3271,6101,3416],{"class":3288},[3271,6103,6104],{"class":3273,"line":3994},[3271,6105,6106],{"class":3422},"    \u002F\u002F Обраний жанр синхронізується в обидва боки\n",[3271,6108,6109,6112,6114,6117,6119,6121],{"class":3273,"line":4017},[3271,6110,6111],{"class":4831},"    genreComboBox",[3271,6113,4548],{"class":3288},[3271,6115,6116],{"class":5178},"valueProperty",[3271,6118,5596],{"class":3288},[3271,6120,6084],{"class":5178},[3271,6122,5919],{"class":3288},[3271,6124,6125,6127,6129,6132],{"class":3273,"line":4036},[3271,6126,5246],{"class":4831},[3271,6128,4548],{"class":3288},[3271,6130,6131],{"class":5178},"selectedGenreProperty",[3271,6133,5602],{"class":3288},[3271,6135,6136],{"class":3273,"line":4059},[3271,6137,3416],{"class":3288},[3271,6139,6140],{"class":3273,"line":4077},[3271,6141,6142],{"class":3422},"    \u002F\u002F Обраний елемент таблиці синхронізується в обидва боки\n",[3271,6144,6145,6147,6149,6152,6154,6157],{"class":3273,"line":4100},[3271,6146,5766],{"class":4831},[3271,6148,4548],{"class":3288},[3271,6150,6151],{"class":5178},"getSelectionModel",[3271,6153,5596],{"class":3288},[3271,6155,6156],{"class":5178},"selectedItemProperty",[3271,6158,5808],{"class":3288},[3271,6160,6161,6163,6165,6167,6169,6171,6174],{"class":3273,"line":4118},[3271,6162,5813],{"class":3288},[3271,6164,6084],{"class":5178},[3271,6166,5213],{"class":3288},[3271,6168,5230],{"class":4831},[3271,6170,4548],{"class":3288},[3271,6172,6173],{"class":5178},"selectedAudiobookProperty",[3271,6175,5602],{"class":3288},[3271,6177,6178],{"class":3273,"line":4141},[3271,6179,3416],{"class":3288},[3271,6181,6182],{"class":3273,"line":4159},[3271,6183,6184],{"class":3422},"    \u002F\u002F ===== Завантаження жанрів у ComboBox =====\n",[3271,6186,6187],{"class":3273,"line":4169},[3271,6188,3416],{"class":3288},[3271,6190,6191],{"class":3273,"line":4174},[3271,6192,6193],{"class":3422},"    \u002F\u002F У реальному додатку це буде через окремий ViewModel\n",[3271,6195,6196,6198,6200,6203,6205,6208],{"class":3273,"line":4184},[3271,6197,6111],{"class":4831},[3271,6199,4548],{"class":3288},[3271,6201,6202],{"class":5178},"getItems",[3271,6204,5596],{"class":3288},[3271,6206,6207],{"class":5178},"addAll",[3271,6209,5919],{"class":3288},[3271,6211,6212,6216,6219,6221,6224],{"class":3273,"line":4200},[3271,6213,6215],{"class":6214},"s8xlr","        new",[3271,6217,6218],{"class":5178}," Genre",[3271,6220,5213],{"class":3288},[3271,6222,6223],{"class":5934},"\"Fiction\"",[3271,6225,6226],{"class":3288},"),\n",[3271,6228,6229,6231,6233,6235,6238],{"class":3273,"line":4213},[3271,6230,6215],{"class":6214},[3271,6232,6218],{"class":5178},[3271,6234,5213],{"class":3288},[3271,6236,6237],{"class":5934},"\"Non-Fiction\"",[3271,6239,6226],{"class":3288},[3271,6241,6242,6244,6246,6248,6251],{"class":3273,"line":4222},[3271,6243,6215],{"class":6214},[3271,6245,6218],{"class":5178},[3271,6247,5213],{"class":3288},[3271,6249,6250],{"class":5934},"\"Science\"",[3271,6252,6226],{"class":3288},[3271,6254,6255,6257,6259,6261,6264],{"class":3273,"line":4231},[3271,6256,6215],{"class":6214},[3271,6258,6218],{"class":5178},[3271,6260,5213],{"class":3288},[3271,6262,6263],{"class":5934},"\"History\"",[3271,6265,5977],{"class":3288},[3271,6267,6268],{"class":3273,"line":4236},[3271,6269,5982],{"class":3288},[3271,6271,6272],{"class":3273,"line":4242},[3271,6273,4703],{"class":3288},[3150,6275,6276],{},[3162,6277,6278],{},"Розбір Bindings:",[3150,6280,6281,4491,6284,6287,6288,6291,6292,6294,6295,6297],{},[3162,6282,6283],{},"Рядки 4-17: Налаштування колонок.",[3154,6285,6286],{},"setCellValueFactory()"," вказує, яку Property використовувати для кожної колонки. Лямбда-вираз ",[3154,6289,6290],{},"cellData -> cellData.getValue().titleProperty()"," означає: \"для кожного рядка візьми ",[3154,6293,4825],{}," та поверни його ",[3154,6296,5599],{},"\". TableView автоматично підключається до цієї Property через Binding.",[3150,6299,6300,4491,6303,6306,6307,6310],{},[3162,6301,6302],{},"Рядок 22: Підключення таблиці до даних.",[3154,6304,6305],{},"setItems(viewModel.getSortedAudiobooks())"," — це не Binding у класичному сенсі, але це встановлює зв'язок: таблиця показує ",[3154,6308,6309],{},"sortedAudiobooks",". Коли ця колекція змінюється (додавання\u002Fвидалення елементів), таблиця автоматично оновлюється.",[3150,6312,6313,6316,6317,6319,6320,6323,6324,6326,6327,6330],{},[3162,6314,6315],{},"Рядки 25-26: Прив'язка сортування."," Коли користувач клікає на заголовок колонки, ",[3154,6318,3927],{}," змінює свій ",[3154,6321,6322],{},"comparator",". Ми прив'язуємо цей ",[3154,6325,6322],{}," до ",[3154,6328,6329],{},"SortedList"," у ViewModel, щоб сортування застосовувалося до даних.",[3150,6332,6333,4491,6336,6339,6340,6343,6344,6347,6348,6351],{},[3162,6334,6335],{},"Рядок 29: Unidirectional Binding.",[3154,6337,6338],{},"loadingIndicator.visibleProperty().bind(...)"," — це однонаправлена прив'язка. Коли ",[3154,6341,6342],{},"viewModel.isLoading"," змінюється, ",[3154,6345,6346],{},"loadingIndicator.visible"," автоматично оновлюється. Але зміна ",[3154,6349,6350],{},"visible"," вручну неможлива (викине виняток).",[3150,6353,6354,4491,6357,6360,6361,6364,6365,6368],{},[3162,6355,6356],{},"Рядки 35-41: Computed Binding.",[3154,6358,6359],{},"Bindings.concat()"," створює рядок, що автоматично оновлюється при зміні ",[3154,6362,6363],{},"filteredCount"," або ",[3154,6366,6367],{},"totalCount",". Це декларативний спосіб: \"countLabel завжди показує 'Showing X of Y audiobooks'\".",[3150,6370,6371,4491,6374,6377,6378,6381,6382,6385,6386,6389,6390,6393],{},[3162,6372,6373],{},"Рядки 44-47: Binding з інверсією.",[3154,6375,6376],{},"editButton.disableProperty().bind(viewModel.deleteButtonEnabledProperty().not())"," — кнопка ",[3162,6379,6380],{},"неактивна"," (",[3154,6383,6384],{},"disable = true","), коли ",[3154,6387,6388],{},"deleteButtonEnabled = false",". ",[3154,6391,6392],{},".not()"," інвертує Boolean Binding.",[3150,6395,6396,4491,6399,6402,6403,6406,6407,6410,6411,6414,6415,6417],{},[3162,6397,6398],{},"Рядки 52-53: Bidirectional Binding.",[3154,6400,6401],{},"bindBidirectional()"," синхронізує дві Properties в обидва боки. Користувач вводить текст у ",[3154,6404,6405],{},"searchField"," → ",[3154,6408,6409],{},"viewModel.searchQuery"," оновлюється. ViewModel змінює ",[3154,6412,6413],{},"searchQuery"," програмно → ",[3154,6416,6405],{}," оновлюється.",[3150,6419,6420,6423,6424,6410,6427,6430],{},[3162,6421,6422],{},"Рядки 61-62: Синхронізація вибору."," Користувач обирає рядок у таблиці → ",[3154,6425,6426],{},"viewModel.selectedAudiobook",[3154,6428,6429],{},"selectedAudiobook"," програмно → таблиця оновлює вибір.",[3229,6432,6433,6436,6437,6440],{},[3162,6434,6435],{},"Чому так багато Bindings?"," Кожен Binding — це декларація залежності: \"цей UI-елемент завжди відображає цю Property\". Це може здаватися багатослівним, але це ціна за автоматичну синхронізацію. Альтернатива (MVP) — десятки методів ",[3154,6438,6439],{},"view.setX()"," у Presenter.",[3236,6442],{},[3145,6444,6446],{"id":6445},"event-handlers-делегування-до-viewmodel","Event Handlers: Делегування до ViewModel",[3150,6448,6449],{},"Тепер, коли Bindings налаштовані, розглянемо обробку подій — що відбувається, коли користувач натискає кнопку або вибирає елемент у таблиці.",[3150,6451,6452,6453,6455],{},"У класичному JavaFX-додатку без архітектурного патерну Controller містив би всю логіку обробки подій: перевірку валідності, виклики до бази даних, оновлення UI. У MVVM Controller ",[3162,6454,4722],{}," — він лише делегує виклик до ViewModel.",[3250,6457,6459],{"id":6458},"привязка-event-handlers-у-fxml","Прив'язка Event Handlers у FXML",[3150,6461,6462],{},"Є два способи прив'язати обробник події до кнопки:",[3150,6464,6465],{},[3162,6466,6467,6468,6471],{},"Спосіб 1: Через атрибут ",[3154,6469,6470],{},"onAction"," у FXML",[3262,6473,6475],{"className":3264,"code":6474,"language":3266,"meta":3267,"style":3267},"\u003CButton fx:id=\"deleteButton\" text=\"Delete\" \n        onAction=\"#onDeleteClicked\"\u002F>\n",[3154,6476,6477,6497],{"__ignoreMap":3267},[3271,6478,6479,6481,6483,6485,6487,6489,6491,6493,6495],{"class":3273,"line":3274},[3271,6480,3366],{"class":3277},[3271,6482,3507],{"class":3281},[3271,6484,3510],{"class":3284},[3271,6486,3289],{"class":3288},[3271,6488,3589],{"class":3292},[3271,6490,3518],{"class":3284},[3271,6492,3289],{"class":3288},[3271,6494,3596],{"class":3292},[3271,6496,3526],{"class":3288},[3271,6498,6499,6502,6504,6507],{"class":3273,"line":3307},[3271,6500,6501],{"class":3284},"        onAction",[3271,6503,3289],{"class":3288},[3271,6505,6506],{"class":3292},"\"#onDeleteClicked\"",[3271,6508,3540],{"class":3277},[3150,6510,6511,6512,6515,6516,6519,6520,6523],{},"Атрибут ",[3154,6513,6514],{},"onAction=\"#onDeleteClicked\""," означає: \"коли користувач натискає цю кнопку, викликай метод ",[3154,6517,6518],{},"onDeleteClicked()"," у Controller\". Символ ",[3154,6521,6522],{},"#"," вказує на метод Controller.",[3150,6525,6526],{},[3162,6527,6528],{},"Спосіб 2: Програмно у Controller",[3262,6530,6532],{"className":4730,"code":6531,"language":4732,"meta":3267,"style":3267},"@FXML\npublic void initialize() {\n    deleteButton.setOnAction(event -> onDeleteClicked());\n}\n",[3154,6533,6534,6541,6551,6569],{"__ignoreMap":3267},[3271,6535,6536,6539],{"class":3273,"line":3274},[3271,6537,6538],{"class":3288},"@",[3271,6540,5167],{"class":4790},[3271,6542,6543,6545,6547,6549],{"class":3273,"line":3307},[3271,6544,4784],{"class":4739},[3271,6546,5175],{"class":4790},[3271,6548,5179],{"class":5178},[3271,6550,5182],{"class":3288},[3271,6552,6553,6555,6557,6560,6563,6565,6567],{"class":3273,"line":3314},[3271,6554,6028],{"class":4831},[3271,6556,4548],{"class":3288},[3271,6558,6559],{"class":5178},"setOnAction",[3271,6561,6562],{"class":3288},"(event ",[3271,6564,5581],{"class":4739},[3271,6566,5388],{"class":5178},[3271,6568,5602],{"class":3288},[3271,6570,6571],{"class":3273,"line":3330},[3271,6572,4703],{"class":3288},[3150,6574,6575],{},"Цей підхід корисний, коли потрібна додаткова логіка (наприклад, передача параметрів) або коли обробник встановлюється динамічно.",[4639,6577,6578,6581,6582,6584],{},[3162,6579,6580],{},"Рекомендація:"," Використовуйте атрибут ",[3154,6583,6470],{}," у FXML для простих випадків (кнопки, меню). Програмний підхід — для складних сценаріїв (динамічні обробники, передача контексту).",[3250,6586,6588],{"id":6587},"реалізація-event-handlers-у-controller","Реалізація Event Handlers у Controller",[3150,6590,6591,6592,6594],{},"Тепер реалізуємо методи-обробники у ",[3154,6593,4718],{},":",[3262,6596,6598],{"className":4730,"code":6597,"language":4732,"meta":3267,"style":3267},"\u002F\u002F ===== Event Handlers =====\n\n@FXML\nprivate void onAddClicked() {\n    \u002F\u002F Навігація до форми додавання (буде реалізовано у статті про навігацію)\n    \u002F\u002F navigator.navigateTo(\"audiobook-form\");\n    \n    \u002F\u002F Поки що — заглушка\n    System.out.println(\"Add button clicked\");\n}\n\n@FXML\nprivate void onEditClicked() {\n    viewModel.editSelected();\n}\n\n@FXML\nprivate void onDeleteClicked() {\n    \u002F\u002F Показуємо діалог підтвердження\n    Alert alert = new Alert(Alert.AlertType.CONFIRMATION);\n    alert.setTitle(\"Delete Audiobook\");\n    alert.setHeaderText(\"Are you sure you want to delete this audiobook?\");\n    alert.setContentText(viewModel.getSelectedAudiobook().getTitle());\n    \n    Optional\u003CButtonType> result = alert.showAndWait();\n    if (result.isPresent() && result.get() == ButtonType.OK) {\n        viewModel.deleteSelected();\n    }\n}\n\n@FXML\nprivate void onRefreshClicked() {\n    viewModel.refresh();\n}\n\n@FXML\nprivate void onTableRowDoubleClicked(MouseEvent event) {\n    if (event.getClickCount() == 2 && viewModel.getSelectedAudiobook() != null) {\n        viewModel.editSelected();\n    }\n}\n",[3154,6599,6600,6605,6609,6615,6625,6630,6635,6639,6644,6667,6671,6675,6681,6691,6701,6705,6709,6715,6725,6730,6764,6781,6797,6822,6826,6853,6889,6899,6903,6907,6911,6917,6927,6937,6941,6945,6951,6968,7005,7015,7019],{"__ignoreMap":3267},[3271,6601,6602],{"class":3273,"line":3274},[3271,6603,6604],{"class":3422},"\u002F\u002F ===== Event Handlers =====\n",[3271,6606,6607],{"class":3273,"line":3307},[3271,6608,3311],{"emptyLinePlaceholder":3310},[3271,6610,6611,6613],{"class":3273,"line":3314},[3271,6612,6538],{"class":3288},[3271,6614,5167],{"class":4790},[3271,6616,6617,6619,6621,6623],{"class":3273,"line":3330},[3271,6618,5550],{"class":4739},[3271,6620,5175],{"class":4790},[3271,6622,5322],{"class":5178},[3271,6624,5182],{"class":3288},[3271,6626,6627],{"class":3273,"line":3344},[3271,6628,6629],{"class":3422},"    \u002F\u002F Навігація до форми додавання (буде реалізовано у статті про навігацію)\n",[3271,6631,6632],{"class":3273,"line":3358},[3271,6633,6634],{"class":3422},"    \u002F\u002F navigator.navigateTo(\"audiobook-form\");\n",[3271,6636,6637],{"class":3273,"line":3363},[3271,6638,3416],{"class":3288},[3271,6640,6641],{"class":3273,"line":3380},[3271,6642,6643],{"class":3422},"    \u002F\u002F Поки що — заглушка\n",[3271,6645,6646,6649,6651,6654,6656,6659,6661,6664],{"class":3273,"line":3391},[3271,6647,6648],{"class":4831},"    System",[3271,6650,4548],{"class":3288},[3271,6652,6653],{"class":4831},"out",[3271,6655,4548],{"class":3288},[3271,6657,6658],{"class":5178},"println",[3271,6660,5213],{"class":3288},[3271,6662,6663],{"class":5934},"\"Add button clicked\"",[3271,6665,6666],{"class":3288},");\n",[3271,6668,6669],{"class":3273,"line":3413},[3271,6670,4703],{"class":3288},[3271,6672,6673],{"class":3273,"line":3419},[3271,6674,3311],{"emptyLinePlaceholder":3310},[3271,6676,6677,6679],{"class":3273,"line":3426},[3271,6678,6538],{"class":3288},[3271,6680,5167],{"class":4790},[3271,6682,6683,6685,6687,6689],{"class":3273,"line":3437},[3271,6684,5550],{"class":4739},[3271,6686,5175],{"class":4790},[3271,6688,5352],{"class":5178},[3271,6690,5182],{"class":3288},[3271,6692,6693,6695,6697,6699],{"class":3273,"line":3464},[3271,6694,5796],{"class":4831},[3271,6696,4548],{"class":3288},[3271,6698,5363],{"class":5178},[3271,6700,5241],{"class":3288},[3271,6702,6703],{"class":3273,"line":3470},[3271,6704,4703],{"class":3288},[3271,6706,6707],{"class":3273,"line":3476},[3271,6708,3311],{"emptyLinePlaceholder":3310},[3271,6710,6711,6713],{"class":3273,"line":3501},[3271,6712,6538],{"class":3288},[3271,6714,5167],{"class":4790},[3271,6716,6717,6719,6721,6723],{"class":3273,"line":3529},[3271,6718,5550],{"class":4739},[3271,6720,5175],{"class":4790},[3271,6722,5388],{"class":5178},[3271,6724,5182],{"class":3288},[3271,6726,6727],{"class":3273,"line":3543},[3271,6728,6729],{"class":3422},"    \u002F\u002F Показуємо діалог підтвердження\n",[3271,6731,6732,6735,6738,6741,6744,6747,6749,6752,6754,6757,6759,6762],{"class":3273,"line":3566},[3271,6733,6734],{"class":4790},"    Alert",[3271,6736,6737],{"class":4831}," alert",[3271,6739,6740],{"class":3288}," = ",[3271,6742,6743],{"class":6214},"new",[3271,6745,6746],{"class":5178}," Alert",[3271,6748,5213],{"class":3288},[3271,6750,6751],{"class":4831},"Alert",[3271,6753,4548],{"class":3288},[3271,6755,6756],{"class":4831},"AlertType",[3271,6758,4548],{"class":3288},[3271,6760,6761],{"class":4831},"CONFIRMATION",[3271,6763,6666],{"class":3288},[3271,6765,6766,6769,6771,6774,6776,6779],{"class":3273,"line":3578},[3271,6767,6768],{"class":4831},"    alert",[3271,6770,4548],{"class":3288},[3271,6772,6773],{"class":5178},"setTitle",[3271,6775,5213],{"class":3288},[3271,6777,6778],{"class":5934},"\"Delete Audiobook\"",[3271,6780,6666],{"class":3288},[3271,6782,6783,6785,6787,6790,6792,6795],{"class":3273,"line":3601},[3271,6784,6768],{"class":4831},[3271,6786,4548],{"class":3288},[3271,6788,6789],{"class":5178},"setHeaderText",[3271,6791,5213],{"class":3288},[3271,6793,6794],{"class":5934},"\"Are you sure you want to delete this audiobook?\"",[3271,6796,6666],{"class":3288},[3271,6798,6799,6801,6803,6806,6808,6810,6812,6815,6817,6820],{"class":3273,"line":3613},[3271,6800,6768],{"class":4831},[3271,6802,4548],{"class":3288},[3271,6804,6805],{"class":5178},"setContentText",[3271,6807,5213],{"class":3288},[3271,6809,5230],{"class":4831},[3271,6811,4548],{"class":3288},[3271,6813,6814],{"class":5178},"getSelectedAudiobook",[3271,6816,5596],{"class":3288},[3271,6818,6819],{"class":5178},"getTitle",[3271,6821,5602],{"class":3288},[3271,6823,6824],{"class":3273,"line":3619},[3271,6825,3416],{"class":3288},[3271,6827,6828,6831,6833,6836,6838,6841,6843,6846,6848,6851],{"class":3273,"line":3641},[3271,6829,6830],{"class":4790},"    Optional",[3271,6832,3366],{"class":3288},[3271,6834,6835],{"class":4790},"ButtonType",[3271,6837,4828],{"class":3288},[3271,6839,6840],{"class":4831},"result",[3271,6842,6740],{"class":3288},[3271,6844,6845],{"class":4831},"alert",[3271,6847,4548],{"class":3288},[3271,6849,6850],{"class":5178},"showAndWait",[3271,6852,5241],{"class":3288},[3271,6854,6855,6858,6860,6862,6864,6867,6870,6872,6874,6877,6880,6882,6884,6887],{"class":3273,"line":3646},[3271,6856,6857],{"class":6214},"    if",[3271,6859,6381],{"class":3288},[3271,6861,6840],{"class":4831},[3271,6863,4548],{"class":3288},[3271,6865,6866],{"class":5178},"isPresent",[3271,6868,6869],{"class":3288},"() && ",[3271,6871,6840],{"class":4831},[3271,6873,4548],{"class":3288},[3271,6875,6876],{"class":5178},"get",[3271,6878,6879],{"class":3288},"() == ",[3271,6881,6835],{"class":4831},[3271,6883,4548],{"class":3288},[3271,6885,6886],{"class":4831},"OK",[3271,6888,5220],{"class":3288},[3271,6890,6891,6893,6895,6897],{"class":3273,"line":3669},[3271,6892,5246],{"class":4831},[3271,6894,4548],{"class":3288},[3271,6896,5399],{"class":5178},[3271,6898,5241],{"class":3288},[3271,6900,6901],{"class":3273,"line":3680},[3271,6902,5197],{"class":3288},[3271,6904,6905],{"class":3273,"line":3690},[3271,6906,4703],{"class":3288},[3271,6908,6909],{"class":3273,"line":3695},[3271,6910,3311],{"emptyLinePlaceholder":3310},[3271,6912,6913,6915],{"class":3273,"line":3701},[3271,6914,6538],{"class":3288},[3271,6916,5167],{"class":4790},[3271,6918,6919,6921,6923,6925],{"class":3273,"line":3722},[3271,6920,5550],{"class":4739},[3271,6922,5175],{"class":4790},[3271,6924,5424],{"class":5178},[3271,6926,5182],{"class":3288},[3271,6928,6929,6931,6933,6935],{"class":3273,"line":3739},[3271,6930,5796],{"class":4831},[3271,6932,4548],{"class":3288},[3271,6934,5435],{"class":5178},[3271,6936,5241],{"class":3288},[3271,6938,6939],{"class":3273,"line":3756},[3271,6940,4703],{"class":3288},[3271,6942,6943],{"class":3273,"line":3769},[3271,6944,3311],{"emptyLinePlaceholder":3310},[3271,6946,6947,6949],{"class":3273,"line":3782},[3271,6948,6538],{"class":3288},[3271,6950,5167],{"class":4790},[3271,6952,6953,6955,6957,6960,6962,6965],{"class":3273,"line":3787},[3271,6954,5550],{"class":4739},[3271,6956,5175],{"class":4790},[3271,6958,6959],{"class":5178}," onTableRowDoubleClicked",[3271,6961,5213],{"class":3288},[3271,6963,6964],{"class":4790},"MouseEvent",[3271,6966,6967],{"class":3288}," event) {\n",[3271,6969,6970,6972,6974,6977,6979,6982,6984,6988,6991,6993,6995,6997,7000,7003],{"class":3273,"line":3803},[3271,6971,6857],{"class":6214},[3271,6973,6381],{"class":3288},[3271,6975,6976],{"class":4831},"event",[3271,6978,4548],{"class":3288},[3271,6980,6981],{"class":5178},"getClickCount",[3271,6983,6879],{"class":3288},[3271,6985,6987],{"class":6986},"sJj4R","2",[3271,6989,6990],{"class":3288}," && ",[3271,6992,5230],{"class":4831},[3271,6994,4548],{"class":3288},[3271,6996,6814],{"class":5178},[3271,6998,6999],{"class":3288},"() != ",[3271,7001,7002],{"class":4739},"null",[3271,7004,5220],{"class":3288},[3271,7006,7007,7009,7011,7013],{"class":3273,"line":3820},[3271,7008,5246],{"class":4831},[3271,7010,4548],{"class":3288},[3271,7012,5363],{"class":5178},[3271,7014,5241],{"class":3288},[3271,7016,7017],{"class":3273,"line":3833},[3271,7018,5197],{"class":3288},[3271,7020,7021],{"class":3273,"line":3846},[3271,7022,4703],{"class":3288},[3150,7024,7025],{},[3162,7026,7027],{},"Розбір обробників:",[3150,7029,7030,7035,7036,7039],{},[3162,7031,7032],{},[3154,7033,7034],{},"onAddClicked()"," — навігація до форми додавання. У реальному додатку тут буде виклик ",[3154,7037,7038],{},"Navigator.navigateTo(\"audiobook-form\")",". Поки що це заглушка.",[3150,7041,7042,7047,7048,7051],{},[3162,7043,7044],{},[3154,7045,7046],{},"onEditClicked()"," — делегування до ViewModel. Метод ",[3154,7049,7050],{},"viewModel.editSelected()"," містить всю логіку: перевірку, чи щось обрано, завантаження даних, навігацію до форми редагування. Controller лише викликає цей метод.",[3150,7053,7054,7058,7059,7062,7063,4548],{},[3162,7055,7056],{},[3154,7057,6518],{}," — єдиний обробник з \"логікою\". Але це не бізнес-логіка — це ",[3162,7060,7061],{},"UI-логіка",": показати діалог підтвердження. Це допустимо у Controller, бо діалог — це частина View. Після підтвердження — делегування до ",[3154,7064,7065],{},"viewModel.deleteSelected()",[3150,7067,7068,7073,7074,4548],{},[3162,7069,7070],{},[3154,7071,7072],{},"onRefreshClicked()"," — найпростіший обробник. Лише виклик ",[3154,7075,7076],{},"viewModel.refresh()",[3150,7078,7079,7084,7085,7088,7089,4548],{},[3162,7080,7081],{},[3154,7082,7083],{},"onTableRowDoubleClicked()"," — обробка подвійного кліку на рядку таблиці. Перевіряємо кількість кліків (",[3154,7086,7087],{},"event.getClickCount() == 2",") та наявність вибраного елемента, потім викликаємо ",[3154,7090,7050],{},[7092,7093,7094,7097],"warning",{},[3162,7095,7096],{},"Діалоги у Controller — виняток з правила."," Показ діалогів підтвердження (Alert, Dialog) — це UI-логіка, тому вона може бути у Controller. Але якщо діалог складний (форма з валідацією), створіть окремий ViewModel для нього.",[3250,7099,7101],{"id":7100},"привязка-обробника-подвійного-кліку","Прив'язка обробника подвійного кліку",[3150,7103,7104,7105,6594],{},"Для обробки подвійного кліку на рядку таблиці потрібно програмно встановити обробник у ",[3154,7106,5486],{},[3262,7108,7110],{"className":4730,"code":7109,"language":4732,"meta":3267,"style":3267},"@FXML\npublic void initialize() {\n    \u002F\u002F Обробка подвійного кліку на рядку таблиці\n    audiobookTable.setOnMouseClicked(this::onTableRowDoubleClicked);\n}\n",[3154,7111,7112,7118,7128,7133,7153],{"__ignoreMap":3267},[3271,7113,7114,7116],{"class":3273,"line":3274},[3271,7115,6538],{"class":3288},[3271,7117,5167],{"class":4790},[3271,7119,7120,7122,7124,7126],{"class":3273,"line":3307},[3271,7121,4784],{"class":4739},[3271,7123,5175],{"class":4790},[3271,7125,5179],{"class":5178},[3271,7127,5182],{"class":3288},[3271,7129,7130],{"class":3273,"line":3314},[3271,7131,7132],{"class":3422},"    \u002F\u002F Обробка подвійного кліку на рядку таблиці\n",[3271,7134,7135,7137,7139,7142,7144,7147,7150],{"class":3273,"line":3330},[3271,7136,5766],{"class":4831},[3271,7138,4548],{"class":3288},[3271,7140,7141],{"class":5178},"setOnMouseClicked",[3271,7143,5213],{"class":3288},[3271,7145,7146],{"class":4739},"this",[3271,7148,7149],{"class":6214},"::",[3271,7151,7152],{"class":3288},"onTableRowDoubleClicked);\n",[3271,7154,7155],{"class":3273,"line":3344},[3271,7156,4703],{"class":3288},[3150,7158,7159,7160,7163],{},"Альтернативно, можна використати атрибут ",[3154,7161,7162],{},"onMouseClicked"," у FXML, але для складних подій (подвійний клік, drag-and-drop) програмний підхід зручніший.",[3236,7165],{},[3145,7167,7169],{"id":7168},"інєкція-viewmodel-від-setter-injection-до-guice","Ін'єкція ViewModel: Від Setter Injection до Guice",[3150,7171,7172,7173,7175,7176,7178],{},"У попередніх прикладах ми передавали ViewModel через метод ",[3154,7174,5477],{},". Це працює, але не масштабується: хто викликає ",[3154,7177,5477],{},"? Хто створює ViewModel? Як передати залежності у ViewModel (Repository, Service)?",[3150,7180,7181,7182,7185],{},"Рішення — ",[3162,7183,7184],{},"Dependency Injection через Guice",". Ми вже розглядали Guice у статті 24, тепер застосуємо його для інтеграції Controller та ViewModel.",[3250,7187,7189],{"id":7188},"проблема-хто-створює-controller","Проблема: Хто створює Controller?",[3150,7191,7192,7193,7196,7197,7199,7200,7203],{},"JavaFX використовує ",[3154,7194,7195],{},"FXMLLoader"," для завантаження FXML та створення Controller. За замовчуванням ",[3154,7198,7195],{}," створює Controller через ",[3162,7201,7202],{},"рефлексію"," — викликає конструктор без параметрів:",[3262,7205,7207],{"className":4730,"code":7206,"language":4732,"meta":3267,"style":3267},"FXMLLoader loader = new FXMLLoader(getClass().getResource(\"audiobook-list-view.fxml\"));\nParent root = loader.load(); \u002F\u002F Тут створюється Controller через new AudiobookListController()\n",[3154,7208,7209,7241],{"__ignoreMap":3267},[3271,7210,7211,7213,7216,7218,7220,7223,7225,7228,7230,7233,7235,7238],{"class":3273,"line":3274},[3271,7212,7195],{"class":4790},[3271,7214,7215],{"class":4831}," loader",[3271,7217,6740],{"class":3288},[3271,7219,6743],{"class":6214},[3271,7221,7222],{"class":5178}," FXMLLoader",[3271,7224,5213],{"class":3288},[3271,7226,7227],{"class":5178},"getClass",[3271,7229,5596],{"class":3288},[3271,7231,7232],{"class":5178},"getResource",[3271,7234,5213],{"class":3288},[3271,7236,7237],{"class":5934},"\"audiobook-list-view.fxml\"",[3271,7239,7240],{"class":3288},"));\n",[3271,7242,7243,7246,7249,7251,7254,7256,7259,7261],{"class":3273,"line":3307},[3271,7244,7245],{"class":4790},"Parent",[3271,7247,7248],{"class":4831}," root",[3271,7250,6740],{"class":3288},[3271,7252,7253],{"class":4831},"loader",[3271,7255,4548],{"class":3288},[3271,7257,7258],{"class":5178},"load",[3271,7260,5254],{"class":3288},[3271,7262,7263],{"class":3422},"\u002F\u002F Тут створюється Controller через new AudiobookListController()\n",[3150,7265,7266,7267,6594],{},"Але якщо Controller має залежності (ViewModel, Navigator), конструктор без параметрів не підходить. Потрібен ",[3162,7268,7269],{},"Constructor Injection",[3262,7271,7273],{"className":4730,"code":7272,"language":4732,"meta":3267,"style":3267},"public class AudiobookListController {\n    private final AudiobookListViewModel viewModel;\n    \n    @Inject\n    public AudiobookListController(AudiobookListViewModel viewModel) {\n        this.viewModel = viewModel;\n    }\n}\n",[3154,7274,7275,7285,7298,7302,7309,7323,7333,7337],{"__ignoreMap":3267},[3271,7276,7277,7279,7281,7283],{"class":3273,"line":3274},[3271,7278,4784],{"class":4739},[3271,7280,4787],{"class":4739},[3271,7282,4791],{"class":4790},[3271,7284,4666],{"class":3288},[3271,7286,7287,7289,7292,7294,7296],{"class":3273,"line":3307},[3271,7288,5139],{"class":4739},[3271,7290,7291],{"class":4739}," final",[3271,7293,5142],{"class":4790},[3271,7295,5145],{"class":4831},[3271,7297,4678],{"class":3288},[3271,7299,7300],{"class":3273,"line":3314},[3271,7301,3416],{"class":3288},[3271,7303,7304,7306],{"class":3273,"line":3330},[3271,7305,4811],{"class":3288},[3271,7307,7308],{"class":4790},"Inject\n",[3271,7310,7311,7313,7315,7317,7319,7321],{"class":3273,"line":3344},[3271,7312,5172],{"class":4739},[3271,7314,4791],{"class":5178},[3271,7316,5213],{"class":3288},[3271,7318,3156],{"class":4790},[3271,7320,5145],{"class":4831},[3271,7322,5220],{"class":3288},[3271,7324,7325,7327,7329,7331],{"class":3273,"line":3358},[3271,7326,5225],{"class":4739},[3271,7328,4548],{"class":3288},[3271,7330,5230],{"class":4831},[3271,7332,5233],{"class":3288},[3271,7334,7335],{"class":3273,"line":3363},[3271,7336,5197],{"class":3288},[3271,7338,7339],{"class":3273,"line":3380},[3271,7340,4703],{"class":3288},[3150,7342,7343,7344,7346,7347,7350,7351,7353],{},"Але ",[3154,7345,7195],{}," не знає про Guice — він не може викликати ",[3154,7348,7349],{},"injector.getInstance(AudiobookListController.class)",". Потрібен міст між ",[3154,7352,7195],{}," та Guice.",[3250,7355,7357],{"id":7356},"рішення-controllerfactory","Рішення: ControllerFactory",[3150,7359,7360,7362,7363,7366],{},[3154,7361,7195],{}," підтримує ",[3162,7364,7365],{},"ControllerFactory"," — інтерфейс, що дозволяє кастомізувати створення Controller:",[3262,7368,7370],{"className":4730,"code":7369,"language":4732,"meta":3267,"style":3267},"public interface Callback\u003CP, R> {\n    R call(P param);\n}\n",[3154,7371,7372,7395,7412],{"__ignoreMap":3267},[3271,7373,7374,7376,7379,7382,7384,7387,7389,7392],{"class":3273,"line":3274},[3271,7375,4784],{"class":4739},[3271,7377,7378],{"class":4739}," interface",[3271,7380,7381],{"class":4790}," Callback",[3271,7383,3366],{"class":3288},[3271,7385,7386],{"class":4790},"P",[3271,7388,3223],{"class":3288},[3271,7390,7391],{"class":4790},"R",[3271,7393,7394],{"class":3288},"> {\n",[3271,7396,7397,7400,7403,7405,7407,7410],{"class":3273,"line":3307},[3271,7398,7399],{"class":4790},"    R",[3271,7401,7402],{"class":5178}," call",[3271,7404,5213],{"class":3288},[3271,7406,7386],{"class":4790},[3271,7408,7409],{"class":4831}," param",[3271,7411,6666],{"class":3288},[3271,7413,7414],{"class":3273,"line":3314},[3271,7415,4703],{"class":3288},[3150,7417,7418,7421,7422,7425],{},[3154,7419,7420],{},"FXMLLoader.setControllerFactory()"," приймає ",[3154,7423,7424],{},"Callback\u003CClass\u003C?>, Object>"," — функцію, що отримує клас Controller та повертає його екземпляр. Ми можемо передати туди Guice Injector:",[3262,7427,7429],{"className":4730,"code":7428,"language":4732,"meta":3267,"style":3267},"FXMLLoader loader = new FXMLLoader(getClass().getResource(\"audiobook-list-view.fxml\"));\nloader.setControllerFactory(injector::getInstance); \u002F\u002F Guice створює Controller\nParent root = loader.load();\n",[3154,7430,7431,7457,7477],{"__ignoreMap":3267},[3271,7432,7433,7435,7437,7439,7441,7443,7445,7447,7449,7451,7453,7455],{"class":3273,"line":3274},[3271,7434,7195],{"class":4790},[3271,7436,7215],{"class":4831},[3271,7438,6740],{"class":3288},[3271,7440,6743],{"class":6214},[3271,7442,7222],{"class":5178},[3271,7444,5213],{"class":3288},[3271,7446,7227],{"class":5178},[3271,7448,5596],{"class":3288},[3271,7450,7232],{"class":5178},[3271,7452,5213],{"class":3288},[3271,7454,7237],{"class":5934},[3271,7456,7240],{"class":3288},[3271,7458,7459,7461,7463,7466,7469,7471,7474],{"class":3273,"line":3307},[3271,7460,7253],{"class":4831},[3271,7462,4548],{"class":3288},[3271,7464,7465],{"class":5178},"setControllerFactory",[3271,7467,7468],{"class":3288},"(injector",[3271,7470,7149],{"class":6214},[3271,7472,7473],{"class":3288},"getInstance); ",[3271,7475,7476],{"class":3422},"\u002F\u002F Guice створює Controller\n",[3271,7478,7479,7481,7483,7485,7487,7489,7491],{"class":3273,"line":3314},[3271,7480,7245],{"class":4790},[3271,7482,7248],{"class":4831},[3271,7484,6740],{"class":3288},[3271,7486,7253],{"class":4831},[3271,7488,4548],{"class":3288},[3271,7490,7258],{"class":5178},[3271,7492,5241],{"class":3288},[3150,7494,7495,7496,7498,7499,7502,7503,7505],{},"Тепер, коли ",[3154,7497,7195],{}," зустрічає атрибут ",[3154,7500,7501],{},"fx:controller=\"dev.kostyl.audiobook.controller.AudiobookListController\"",", він викликає ",[3154,7504,7349],{},", і Guice автоматично ін'єктує всі залежності (ViewModel, Navigator).",[3250,7507,7509],{"id":7508},"повний-приклад-application-з-guice","Повний приклад: Application з Guice",[3150,7511,7512,7513,7516],{},"Створимо клас ",[3154,7514,7515],{},"AudiobookApp"," — точку входу у додаток:",[3262,7518,7520],{"className":4730,"code":7519,"language":4732,"meta":3267,"style":3267},"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\npublic class AudiobookApp extends Application {\n    \n    private Injector injector;\n    \n    @Override\n    public void init() {\n        \u002F\u002F Створення Guice Injector з модулем конфігурації\n        injector = Guice.createInjector(new AudiobookModule());\n    }\n    \n    @Override\n    public void start(Stage primaryStage) throws Exception {\n        \u002F\u002F Завантаження FXML з 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 та Stage\n        Scene scene = new Scene(root, 900, 600);\n        scene.getStylesheets().add(\n            getClass().getResource(\"\u002Fcss\u002Fstyles.css\").toExternalForm()\n        );\n        \n        primaryStage.setTitle(\"Audiobook Platform\");\n        primaryStage.setScene(scene);\n        primaryStage.show();\n    }\n    \n    @Override\n    public void stop() {\n        \u002F\u002F Закриття ресурсів (Connection Pool, тощо)\n        \u002F\u002F injector.getInstance(DataSource.class).close();\n    }\n    \n    public static void main(String[] args) {\n        launch(args);\n    }\n}\n",[3154,7521,7522,7529,7533,7540,7547,7554,7561,7568,7575,7582,7589,7593,7610,7614,7626,7630,7637,7648,7653,7675,7679,7683,7689,7717,7722,7737,7753,7758,7774,7779,7796,7800,7805,7833,7850,7871,7875,7879,7895,7907,7918,7922,7926,7932,7943,7948,7953,7957,7961,7985,7993,7997],{"__ignoreMap":3267},[3271,7523,7524,7526],{"class":3273,"line":3274},[3271,7525,4740],{"class":4739},[3271,7527,7528],{"class":3288}," dev.kostyl.audiobook;\n",[3271,7530,7531],{"class":3273,"line":3307},[3271,7532,3311],{"emptyLinePlaceholder":3310},[3271,7534,7535,7537],{"class":3273,"line":3314},[3271,7536,3319],{"class":4739},[3271,7538,7539],{"class":3288}," com.google.inject.Guice;\n",[3271,7541,7542,7544],{"class":3273,"line":3330},[3271,7543,3319],{"class":4739},[3271,7545,7546],{"class":3288}," com.google.inject.Injector;\n",[3271,7548,7549,7551],{"class":3273,"line":3344},[3271,7550,3319],{"class":4739},[3271,7552,7553],{"class":3288}," dev.kostyl.audiobook.infrastructure.AudiobookModule;\n",[3271,7555,7556,7558],{"class":3273,"line":3358},[3271,7557,3319],{"class":4739},[3271,7559,7560],{"class":3288}," javafx.application.Application;\n",[3271,7562,7563,7565],{"class":3273,"line":3363},[3271,7564,3319],{"class":4739},[3271,7566,7567],{"class":3288}," javafx.fxml.FXMLLoader;\n",[3271,7569,7570,7572],{"class":3273,"line":3380},[3271,7571,3319],{"class":4739},[3271,7573,7574],{"class":3288}," javafx.scene.Parent;\n",[3271,7576,7577,7579],{"class":3273,"line":3391},[3271,7578,3319],{"class":4739},[3271,7580,7581],{"class":3288}," javafx.scene.Scene;\n",[3271,7583,7584,7586],{"class":3273,"line":3413},[3271,7585,3319],{"class":4739},[3271,7587,7588],{"class":3288}," javafx.stage.Stage;\n",[3271,7590,7591],{"class":3273,"line":3419},[3271,7592,3311],{"emptyLinePlaceholder":3310},[3271,7594,7595,7597,7599,7602,7605,7608],{"class":3273,"line":3426},[3271,7596,4784],{"class":4739},[3271,7598,4787],{"class":4739},[3271,7600,7601],{"class":4790}," AudiobookApp",[3271,7603,7604],{"class":4739}," extends",[3271,7606,7607],{"class":4790}," Application",[3271,7609,4666],{"class":3288},[3271,7611,7612],{"class":3273,"line":3437},[3271,7613,3416],{"class":3288},[3271,7615,7616,7618,7621,7624],{"class":3273,"line":3464},[3271,7617,5139],{"class":4739},[3271,7619,7620],{"class":4790}," Injector",[3271,7622,7623],{"class":4831}," injector",[3271,7625,4678],{"class":3288},[3271,7627,7628],{"class":3273,"line":3470},[3271,7629,3416],{"class":3288},[3271,7631,7632,7634],{"class":3273,"line":3476},[3271,7633,4811],{"class":3288},[3271,7635,7636],{"class":4790},"Override\n",[3271,7638,7639,7641,7643,7646],{"class":3273,"line":3501},[3271,7640,5172],{"class":4739},[3271,7642,5175],{"class":4790},[3271,7644,7645],{"class":5178}," init",[3271,7647,5182],{"class":3288},[3271,7649,7650],{"class":3273,"line":3529},[3271,7651,7652],{"class":3422},"        \u002F\u002F Створення Guice Injector з модулем конфігурації\n",[3271,7654,7655,7658,7661,7663,7666,7668,7670,7673],{"class":3273,"line":3543},[3271,7656,7657],{"class":3288},"        injector = ",[3271,7659,7660],{"class":4831},"Guice",[3271,7662,4548],{"class":3288},[3271,7664,7665],{"class":5178},"createInjector",[3271,7667,5213],{"class":3288},[3271,7669,6743],{"class":6214},[3271,7671,7672],{"class":5178}," AudiobookModule",[3271,7674,5602],{"class":3288},[3271,7676,7677],{"class":3273,"line":3566},[3271,7678,5197],{"class":3288},[3271,7680,7681],{"class":3273,"line":3578},[3271,7682,3416],{"class":3288},[3271,7684,7685,7687],{"class":3273,"line":3601},[3271,7686,4811],{"class":3288},[3271,7688,7636],{"class":4790},[3271,7690,7691,7693,7695,7698,7700,7703,7706,7709,7712,7715],{"class":3273,"line":3613},[3271,7692,5172],{"class":4739},[3271,7694,5175],{"class":4790},[3271,7696,7697],{"class":5178}," start",[3271,7699,5213],{"class":3288},[3271,7701,7702],{"class":4790},"Stage",[3271,7704,7705],{"class":4831}," primaryStage",[3271,7707,7708],{"class":3288},") ",[3271,7710,7711],{"class":4739},"throws",[3271,7713,7714],{"class":4790}," Exception",[3271,7716,4666],{"class":3288},[3271,7718,7719],{"class":3273,"line":3619},[3271,7720,7721],{"class":3422},"        \u002F\u002F Завантаження FXML з ControllerFactory\n",[3271,7723,7724,7727,7729,7731,7733,7735],{"class":3273,"line":3641},[3271,7725,7726],{"class":4790},"        FXMLLoader",[3271,7728,7215],{"class":4831},[3271,7730,6740],{"class":3288},[3271,7732,6743],{"class":6214},[3271,7734,7222],{"class":5178},[3271,7736,5919],{"class":3288},[3271,7738,7739,7742,7744,7746,7748,7751],{"class":3273,"line":3646},[3271,7740,7741],{"class":5178},"            getClass",[3271,7743,5596],{"class":3288},[3271,7745,7232],{"class":5178},[3271,7747,5213],{"class":3288},[3271,7749,7750],{"class":5934},"\"\u002Ffxml\u002Faudiobook-list-view.fxml\"",[3271,7752,5977],{"class":3288},[3271,7754,7755],{"class":3273,"line":3669},[3271,7756,7757],{"class":3288},"        );\n",[3271,7759,7760,7763,7765,7767,7769,7771],{"class":3273,"line":3680},[3271,7761,7762],{"class":4831},"        loader",[3271,7764,4548],{"class":3288},[3271,7766,7465],{"class":5178},[3271,7768,7468],{"class":3288},[3271,7770,7149],{"class":6214},[3271,7772,7773],{"class":3288},"getInstance);\n",[3271,7775,7776],{"class":3273,"line":3690},[3271,7777,7778],{"class":3288},"        \n",[3271,7780,7781,7784,7786,7788,7790,7792,7794],{"class":3273,"line":3695},[3271,7782,7783],{"class":4790},"        Parent",[3271,7785,7248],{"class":4831},[3271,7787,6740],{"class":3288},[3271,7789,7253],{"class":4831},[3271,7791,4548],{"class":3288},[3271,7793,7258],{"class":5178},[3271,7795,5241],{"class":3288},[3271,7797,7798],{"class":3273,"line":3701},[3271,7799,7778],{"class":3288},[3271,7801,7802],{"class":3273,"line":3722},[3271,7803,7804],{"class":3422},"        \u002F\u002F Налаштування Scene та Stage\n",[3271,7806,7807,7810,7813,7815,7817,7820,7823,7826,7828,7831],{"class":3273,"line":3739},[3271,7808,7809],{"class":4790},"        Scene",[3271,7811,7812],{"class":4831}," scene",[3271,7814,6740],{"class":3288},[3271,7816,6743],{"class":6214},[3271,7818,7819],{"class":5178}," Scene",[3271,7821,7822],{"class":3288},"(root, ",[3271,7824,7825],{"class":6986},"900",[3271,7827,3223],{"class":3288},[3271,7829,7830],{"class":6986},"600",[3271,7832,6666],{"class":3288},[3271,7834,7835,7838,7840,7843,7845,7848],{"class":3273,"line":3756},[3271,7836,7837],{"class":4831},"        scene",[3271,7839,4548],{"class":3288},[3271,7841,7842],{"class":5178},"getStylesheets",[3271,7844,5596],{"class":3288},[3271,7846,7847],{"class":5178},"add",[3271,7849,5919],{"class":3288},[3271,7851,7852,7854,7856,7858,7860,7863,7866,7869],{"class":3273,"line":3769},[3271,7853,7741],{"class":5178},[3271,7855,5596],{"class":3288},[3271,7857,7232],{"class":5178},[3271,7859,5213],{"class":3288},[3271,7861,7862],{"class":5934},"\"\u002Fcss\u002Fstyles.css\"",[3271,7864,7865],{"class":3288},").",[3271,7867,7868],{"class":5178},"toExternalForm",[3271,7870,5808],{"class":3288},[3271,7872,7873],{"class":3273,"line":3782},[3271,7874,7757],{"class":3288},[3271,7876,7877],{"class":3273,"line":3787},[3271,7878,7778],{"class":3288},[3271,7880,7881,7884,7886,7888,7890,7893],{"class":3273,"line":3803},[3271,7882,7883],{"class":4831},"        primaryStage",[3271,7885,4548],{"class":3288},[3271,7887,6773],{"class":5178},[3271,7889,5213],{"class":3288},[3271,7891,7892],{"class":5934},"\"Audiobook Platform\"",[3271,7894,6666],{"class":3288},[3271,7896,7897,7899,7901,7904],{"class":3273,"line":3820},[3271,7898,7883],{"class":4831},[3271,7900,4548],{"class":3288},[3271,7902,7903],{"class":5178},"setScene",[3271,7905,7906],{"class":3288},"(scene);\n",[3271,7908,7909,7911,7913,7916],{"class":3273,"line":3833},[3271,7910,7883],{"class":4831},[3271,7912,4548],{"class":3288},[3271,7914,7915],{"class":5178},"show",[3271,7917,5241],{"class":3288},[3271,7919,7920],{"class":3273,"line":3846},[3271,7921,5197],{"class":3288},[3271,7923,7924],{"class":3273,"line":3855},[3271,7925,3416],{"class":3288},[3271,7927,7928,7930],{"class":3273,"line":3860},[3271,7929,4811],{"class":3288},[3271,7931,7636],{"class":4790},[3271,7933,7934,7936,7938,7941],{"class":3273,"line":3870},[3271,7935,5172],{"class":4739},[3271,7937,5175],{"class":4790},[3271,7939,7940],{"class":5178}," stop",[3271,7942,5182],{"class":3288},[3271,7944,7945],{"class":3273,"line":3880},[3271,7946,7947],{"class":3422},"        \u002F\u002F Закриття ресурсів (Connection Pool, тощо)\n",[3271,7949,7950],{"class":3273,"line":3885},[3271,7951,7952],{"class":3422},"        \u002F\u002F injector.getInstance(DataSource.class).close();\n",[3271,7954,7955],{"class":3273,"line":3891},[3271,7956,5197],{"class":3288},[3271,7958,7959],{"class":3273,"line":3901},[3271,7960,3416],{"class":3288},[3271,7962,7963,7965,7968,7970,7973,7975,7977,7980,7983],{"class":3273,"line":3911},[3271,7964,5172],{"class":4739},[3271,7966,7967],{"class":4739}," static",[3271,7969,5175],{"class":4790},[3271,7971,7972],{"class":5178}," main",[3271,7974,5213],{"class":3288},[3271,7976,4854],{"class":4790},[3271,7978,7979],{"class":3288},"[] ",[3271,7981,7982],{"class":4831},"args",[3271,7984,5220],{"class":3288},[3271,7986,7987,7990],{"class":3273,"line":3916},[3271,7988,7989],{"class":5178},"        launch",[3271,7991,7992],{"class":3288},"(args);\n",[3271,7994,7995],{"class":3273,"line":3922},[3271,7996,5197],{"class":3288},[3271,7998,7999],{"class":3273,"line":3939},[3271,8000,4703],{"class":3288},[3150,8002,8003],{},[3162,8004,8005],{},"Розбір коду:",[3150,8007,8008,8014,8015,8017,8018,8021,8022,8025],{},[3162,8009,8010,8011,4548],{},"Рядки 14-17: ",[3154,8012,8013],{},"init()"," Метод ",[3154,8016,8013],{}," викликається перед ",[3154,8019,8020],{},"start()"," — тут створюємо Guice Injector. ",[3154,8023,8024],{},"AudiobookModule"," — це Guice Module, що конфігурує всі Bindings (Repository, Service, ViewModel).",[3150,8027,8028,8031,8032,8034,8035,8037,8038,8041,8042,4548],{},[3162,8029,8030],{},"Рядки 21-24: Завантаження FXML."," Створюємо ",[3154,8033,7195],{}," та встановлюємо ",[3154,8036,7365],{}," через ",[3154,8039,8040],{},"injector::getInstance",". Це method reference — еквівалент ",[3154,8043,8044],{},"clazz -> injector.getInstance(clazz)",[3150,8046,8047,8053,8054,8056,8057,8059,8060,8062,8063,8065,8066,8068,8069,4548],{},[3162,8048,8049,8050,4548],{},"Рядок 26: ",[3154,8051,8052],{},"loader.load()"," Тут відбувається магія: ",[3154,8055,7195],{}," парсить FXML, знаходить ",[3154,8058,4510],{},", викликає ",[3154,8061,7349],{},", Guice створює Controller з усіма залежностями (ViewModel, Navigator), потім ",[3154,8064,7195],{}," ін'єктує ",[3154,8067,4547],{}," поля (UI-елементи) та викликає ",[3154,8070,5486],{},[3150,8072,8073,8076],{},[3162,8074,8075],{},"Рядки 29-31: Підключення CSS."," Завантажуємо CSS-файл для стилізації UI.",[3150,8078,8079,8014,8085,8087],{},[3162,8080,8081,8082,4548],{},"Рядки 38-41: ",[3154,8083,8084],{},"stop()",[3154,8086,8084],{}," викликається при закритті додатку — тут закриваємо ресурси (Connection Pool, файли).",[3250,8089,8091],{"id":8090},"guice-module-конфігурація-залежностей","Guice Module: Конфігурація залежностей",[3150,8093,8094,8095,8097],{},"Створимо ",[3154,8096,8024],{}," — Guice Module, що конфігурує всі Bindings:",[3262,8099,8101],{"className":4730,"code":8100,"language":4732,"meta":3267,"style":3267},"package dev.kostyl.audiobook.infrastructure;\n\nimport com.google.inject.AbstractModule;\nimport com.google.inject.Singleton;\nimport dev.kostyl.audiobook.repository.AudiobookRepository;\nimport dev.kostyl.audiobook.repository.jdbc.JdbcAudiobookRepository;\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 DataSource (Connection Pool)\n        bind(DataSource.class)\n            .toProvider(HikariDataSourceProvider.class)\n            .in(Singleton.class);\n        \n        \u002F\u002F Repositories\n        bind(AudiobookRepository.class)\n            .to(JdbcAudiobookRepository.class);\n        \n        \u002F\u002F ===== Service Layer =====\n        \n        bind(AudiobookService.class)\n            .in(Singleton.class);\n        \n        \u002F\u002F ===== Presentation Layer =====\n        \n        \u002F\u002F ViewModels (Singleton — один екземпляр на весь додаток)\n        bind(AudiobookListViewModel.class)\n            .in(Singleton.class);\n        \n        \u002F\u002F Controllers створюються FXMLLoader через ControllerFactory\n        \u002F\u002F Не потрібно bind() для Controllers — вони створюються автоматично\n    }\n}\n",[3154,8102,8103,8110,8114,8121,8128,8135,8142,8149,8155,8159,8166,8170,8185,8189,8195,8207,8212,8216,8221,8238,8257,8275,8279,8284,8299,8317,8321,8326,8330,8345,8361,8365,8370,8374,8379,8393,8409,8413,8418,8423,8427],{"__ignoreMap":3267},[3271,8104,8105,8107],{"class":3273,"line":3274},[3271,8106,4740],{"class":4739},[3271,8108,8109],{"class":3288}," dev.kostyl.audiobook.infrastructure;\n",[3271,8111,8112],{"class":3273,"line":3307},[3271,8113,3311],{"emptyLinePlaceholder":3310},[3271,8115,8116,8118],{"class":3273,"line":3314},[3271,8117,3319],{"class":4739},[3271,8119,8120],{"class":3288}," com.google.inject.AbstractModule;\n",[3271,8122,8123,8125],{"class":3273,"line":3330},[3271,8124,3319],{"class":4739},[3271,8126,8127],{"class":3288}," com.google.inject.Singleton;\n",[3271,8129,8130,8132],{"class":3273,"line":3344},[3271,8131,3319],{"class":4739},[3271,8133,8134],{"class":3288}," dev.kostyl.audiobook.repository.AudiobookRepository;\n",[3271,8136,8137,8139],{"class":3273,"line":3358},[3271,8138,3319],{"class":4739},[3271,8140,8141],{"class":3288}," dev.kostyl.audiobook.repository.jdbc.JdbcAudiobookRepository;\n",[3271,8143,8144,8146],{"class":3273,"line":3363},[3271,8145,3319],{"class":4739},[3271,8147,8148],{"class":3288}," dev.kostyl.audiobook.service.AudiobookService;\n",[3271,8150,8151,8153],{"class":3273,"line":3380},[3271,8152,3319],{"class":4739},[3271,8154,4754],{"class":3288},[3271,8156,8157],{"class":3273,"line":3391},[3271,8158,3311],{"emptyLinePlaceholder":3310},[3271,8160,8161,8163],{"class":3273,"line":3413},[3271,8162,3319],{"class":4739},[3271,8164,8165],{"class":3288}," javax.sql.DataSource;\n",[3271,8167,8168],{"class":3273,"line":3419},[3271,8169,3311],{"emptyLinePlaceholder":3310},[3271,8171,8172,8174,8176,8178,8180,8183],{"class":3273,"line":3426},[3271,8173,4784],{"class":4739},[3271,8175,4787],{"class":4739},[3271,8177,7672],{"class":4790},[3271,8179,7604],{"class":4739},[3271,8181,8182],{"class":4790}," AbstractModule",[3271,8184,4666],{"class":3288},[3271,8186,8187],{"class":3273,"line":3437},[3271,8188,3416],{"class":3288},[3271,8190,8191,8193],{"class":3273,"line":3464},[3271,8192,4811],{"class":3288},[3271,8194,7636],{"class":4790},[3271,8196,8197,8200,8202,8205],{"class":3273,"line":3470},[3271,8198,8199],{"class":4739},"    protected",[3271,8201,5175],{"class":4790},[3271,8203,8204],{"class":5178}," configure",[3271,8206,5182],{"class":3288},[3271,8208,8209],{"class":3273,"line":3476},[3271,8210,8211],{"class":3422},"        \u002F\u002F ===== Data Access Layer =====\n",[3271,8213,8214],{"class":3273,"line":3501},[3271,8215,7778],{"class":3288},[3271,8217,8218],{"class":3273,"line":3529},[3271,8219,8220],{"class":3422},"        \u002F\u002F DataSource (Connection Pool)\n",[3271,8222,8223,8226,8228,8231,8233,8236],{"class":3273,"line":3543},[3271,8224,8225],{"class":5178},"        bind",[3271,8227,5213],{"class":3288},[3271,8229,8230],{"class":4831},"DataSource",[3271,8232,4548],{"class":3288},[3271,8234,8235],{"class":4831},"class",[3271,8237,5977],{"class":3288},[3271,8239,8240,8243,8246,8248,8251,8253,8255],{"class":3273,"line":3566},[3271,8241,8242],{"class":3288},"            .",[3271,8244,8245],{"class":5178},"toProvider",[3271,8247,5213],{"class":3288},[3271,8249,8250],{"class":4831},"HikariDataSourceProvider",[3271,8252,4548],{"class":3288},[3271,8254,8235],{"class":4831},[3271,8256,5977],{"class":3288},[3271,8258,8259,8261,8264,8266,8269,8271,8273],{"class":3273,"line":3578},[3271,8260,8242],{"class":3288},[3271,8262,8263],{"class":5178},"in",[3271,8265,5213],{"class":3288},[3271,8267,8268],{"class":4831},"Singleton",[3271,8270,4548],{"class":3288},[3271,8272,8235],{"class":4831},[3271,8274,6666],{"class":3288},[3271,8276,8277],{"class":3273,"line":3601},[3271,8278,7778],{"class":3288},[3271,8280,8281],{"class":3273,"line":3613},[3271,8282,8283],{"class":3422},"        \u002F\u002F Repositories\n",[3271,8285,8286,8288,8290,8293,8295,8297],{"class":3273,"line":3619},[3271,8287,8225],{"class":5178},[3271,8289,5213],{"class":3288},[3271,8291,8292],{"class":4831},"AudiobookRepository",[3271,8294,4548],{"class":3288},[3271,8296,8235],{"class":4831},[3271,8298,5977],{"class":3288},[3271,8300,8301,8303,8306,8308,8311,8313,8315],{"class":3273,"line":3641},[3271,8302,8242],{"class":3288},[3271,8304,8305],{"class":5178},"to",[3271,8307,5213],{"class":3288},[3271,8309,8310],{"class":4831},"JdbcAudiobookRepository",[3271,8312,4548],{"class":3288},[3271,8314,8235],{"class":4831},[3271,8316,6666],{"class":3288},[3271,8318,8319],{"class":3273,"line":3646},[3271,8320,7778],{"class":3288},[3271,8322,8323],{"class":3273,"line":3669},[3271,8324,8325],{"class":3422},"        \u002F\u002F ===== Service Layer =====\n",[3271,8327,8328],{"class":3273,"line":3680},[3271,8329,7778],{"class":3288},[3271,8331,8332,8334,8336,8339,8341,8343],{"class":3273,"line":3690},[3271,8333,8225],{"class":5178},[3271,8335,5213],{"class":3288},[3271,8337,8338],{"class":4831},"AudiobookService",[3271,8340,4548],{"class":3288},[3271,8342,8235],{"class":4831},[3271,8344,5977],{"class":3288},[3271,8346,8347,8349,8351,8353,8355,8357,8359],{"class":3273,"line":3695},[3271,8348,8242],{"class":3288},[3271,8350,8263],{"class":5178},[3271,8352,5213],{"class":3288},[3271,8354,8268],{"class":4831},[3271,8356,4548],{"class":3288},[3271,8358,8235],{"class":4831},[3271,8360,6666],{"class":3288},[3271,8362,8363],{"class":3273,"line":3701},[3271,8364,7778],{"class":3288},[3271,8366,8367],{"class":3273,"line":3722},[3271,8368,8369],{"class":3422},"        \u002F\u002F ===== Presentation Layer =====\n",[3271,8371,8372],{"class":3273,"line":3739},[3271,8373,7778],{"class":3288},[3271,8375,8376],{"class":3273,"line":3756},[3271,8377,8378],{"class":3422},"        \u002F\u002F ViewModels (Singleton — один екземпляр на весь додаток)\n",[3271,8380,8381,8383,8385,8387,8389,8391],{"class":3273,"line":3769},[3271,8382,8225],{"class":5178},[3271,8384,5213],{"class":3288},[3271,8386,3156],{"class":4831},[3271,8388,4548],{"class":3288},[3271,8390,8235],{"class":4831},[3271,8392,5977],{"class":3288},[3271,8394,8395,8397,8399,8401,8403,8405,8407],{"class":3273,"line":3782},[3271,8396,8242],{"class":3288},[3271,8398,8263],{"class":5178},[3271,8400,5213],{"class":3288},[3271,8402,8268],{"class":4831},[3271,8404,4548],{"class":3288},[3271,8406,8235],{"class":4831},[3271,8408,6666],{"class":3288},[3271,8410,8411],{"class":3273,"line":3787},[3271,8412,7778],{"class":3288},[3271,8414,8415],{"class":3273,"line":3803},[3271,8416,8417],{"class":3422},"        \u002F\u002F Controllers створюються FXMLLoader через ControllerFactory\n",[3271,8419,8420],{"class":3273,"line":3820},[3271,8421,8422],{"class":3422},"        \u002F\u002F Не потрібно bind() для Controllers — вони створюються автоматично\n",[3271,8424,8425],{"class":3273,"line":3833},[3271,8426,5197],{"class":3288},[3271,8428,8429],{"class":3273,"line":3846},[3271,8430,4703],{"class":3288},[3150,8432,8433],{},[3162,8434,8435],{},"Розбір конфігурації:",[3150,8437,8438,8441,8442,8444,8445,8447],{},[3162,8439,8440],{},"Рядки 18-20: DataSource."," Використовуємо ",[3154,8443,8250],{}," (Provider Pattern) для створення Connection Pool. ",[3154,8446,8268],{}," — один екземпляр на весь додаток.",[3150,8449,8450,8453,8454,8456,8457,8459,8460,8462,8463,4548],{},[3162,8451,8452],{},"Рядки 23-24: Repository."," Прив'язуємо інтерфейс ",[3154,8455,8292],{}," до реалізації ",[3154,8458,8310],{},". Guice автоматично ін'єктує ",[3154,8461,8230],{}," у конструктор ",[3154,8464,8310],{},[3150,8466,8467,4491,8470,8472,8473,8475],{},[3162,8468,8469],{},"Рядки 28-29: Service.",[3154,8471,8338],{}," — Singleton, бо він не має стану (stateless). Guice ін'єктує ",[3154,8474,8292],{}," у конструктор.",[3150,8477,8478,4491,8481,8483,8484,8487],{},[3162,8479,8480],{},"Рядки 34-35: ViewModel.",[3154,8482,3156],{}," — Singleton. Це означає, що всі Controller, які залежать від цього ViewModel, отримають ",[3162,8485,8486],{},"один і той самий екземпляр",". Це важливо для збереження стану (вибраний елемент, фільтри) між переходами між екранами.",[3150,8489,8490,8493,8494,8497,8498,8037,8500,8502],{},[3162,8491,8492],{},"Рядки 37-38: Controllers."," Controllers ",[3162,8495,8496],{},"не потрібно"," прив'язувати у Guice Module — вони створюються ",[3154,8499,7195],{},[3154,8501,7365],{},". Guice лише ін'єктує їхні залежності (ViewModel, Navigator).",[3229,8504,8505,8508,8509,8511,8512,8515,8516,8519],{},[3162,8506,8507],{},"Singleton ViewModel — це нормально?"," Так, якщо ViewModel не містить стану, специфічного для конкретного екрану. Наприклад, ",[3154,8510,3156],{}," зберігає список аудіокниг, фільтри, вибраний елемент — це глобальний стан, що має зберігатися між переходами. Але якщо ViewModel містить стан форми (наприклад, ",[3154,8513,8514],{},"AudiobookFormViewModel","), він має бути ",[3162,8517,8518],{},"не-Singleton"," (новий екземпляр для кожного відкриття форми).",[3236,8521],{},[3145,8523,8525],{"id":8524},"fxmlloader-життєвий-цикл-завантаження","FXMLLoader: Життєвий цикл завантаження",[3150,8527,8528,8529,6594],{},"Розглянемо детально, що відбувається під час виклику ",[3154,8530,8052],{},[8532,8533,8534,8538,8543,8547,8559,8563,8573,8577,8585,8589],"steps",{},[3250,8535,8537],{"id":8536},"крок-1-парсинг-fxml","Крок 1: Парсинг FXML",[3150,8539,8540,8542],{},[3154,8541,7195],{}," читає XML-файл та будує дерево об'єктів JavaFX (Node, Layout, Control).",[3250,8544,8546],{"id":8545},"крок-2-створення-controller","Крок 2: Створення Controller",[3150,8548,8549,8551,8552,8554,8555,8558],{},[3154,8550,7195],{}," знаходить атрибут ",[3154,8553,4510],{}," та викликає ",[3154,8556,8557],{},"ControllerFactory.call(AudiobookListController.class)",". Guice створює Controller через Constructor Injection, ін'єктуючи всі залежності (ViewModel).",[3250,8560,8562],{"id":8561},"крок-3-інєкція-fxml-полів","Крок 3: Ін'єкція @FXML полів",[3150,8564,8565,8567,8568,8570,8571,7865],{},[3154,8566,7195],{}," знаходить всі поля з ",[3154,8569,4547],{}," у Controller та ін'єктує відповідні UI-елементи з FXML (за ",[3154,8572,4539],{},[3250,8574,8576],{"id":8575},"крок-4-виклик-initialize","Крок 4: Виклик initialize()",[3150,8578,8579,8581,8582,8584],{},[3154,8580,7195],{}," викликає метод ",[3154,8583,5486],{}," у Controller (якщо він існує). Тут ініціалізуємо Bindings.",[3250,8586,8588],{"id":8587},"крок-5-повернення-root-node","Крок 5: Повернення Root Node",[3150,8590,8591,8593,8594,8596],{},[3154,8592,8052],{}," повертає кореневий елемент FXML (у нашому випадку — ",[3154,8595,3369],{},"), готовий до відображення у Scene.",[3250,8598,8600],{"id":8599},"діаграма-життєвого-циклу","Діаграма життєвого циклу",[8602,8603,8604,8607],"mermaid",{},[3150,8605,8606],{},"sequenceDiagram\nparticipant App as AudiobookApp\nparticipant Loader as FXMLLoader\nparticipant Guice as Guice Injector\nparticipant Ctrl as AudiobookListController\nparticipant VM as AudiobookListViewModel",[3262,8608,8612],{"className":8609,"code":8611,"language":4598},[8610],"language-text","App->>Loader: new FXMLLoader(fxml)\nApp->>Loader: setControllerFactory(injector::getInstance)\nApp->>Loader: load()\n\nLoader->>Loader: Parse FXML\nLoader->>Loader: Find fx:controller attribute\nLoader->>Guice: getInstance(AudiobookListController.class)\n\nGuice->>Guice: Resolve dependencies\nGuice->>VM: new AudiobookListViewModel(service)\nGuice->>Ctrl: new AudiobookListController(viewModel)\nGuice-->>Loader: Return Controller instance\n\nLoader->>Ctrl: Inject @FXML fields (UI elements)\nLoader->>Ctrl: Call initialize()\nCtrl->>Ctrl: setupBindings()\nCtrl->>VM: initialize() (load data)\n\nLoader-->>App: Return Root Node (BorderPane)\nApp->>App: new Scene(root)\nApp->>App: stage.setScene(scene)\n",[3154,8613,8611],{"__ignoreMap":3267},[3150,8615,8616],{},[3162,8617,8618],{},"Пояснення діаграми:",[8620,8621,8622,8632,8639,8647,8660,8667,8674,8682],"ol",{},[3195,8623,8624,8626,8627,8629,8630,4548],{},[3154,8625,7515],{}," створює ",[3154,8628,7195],{}," та встановлює ",[3154,8631,7365],{},[3195,8633,8634,8636,8637,4548],{},[3154,8635,7195],{}," парсить FXML та знаходить ",[3154,8638,4510],{},[3195,8640,8641,8643,8644,4548],{},[3154,8642,7195],{}," викликає ",[3154,8645,8646],{},"Guice.getInstance(AudiobookListController.class)",[3195,8648,8649,8650,8652,8653,8655,8656,8652,8658,7865],{},"Guice резолвить залежності: створює ",[3154,8651,3156],{}," (з ",[3154,8654,8338],{},"), потім створює ",[3154,8657,4718],{},[3154,8659,3156],{},[3195,8661,8662,8065,8664,8666],{},[3154,8663,7195],{},[3154,8665,4547],{}," поля (UI-елементи) у Controller.",[3195,8668,8669,8643,8671,8673],{},[3154,8670,7195],{},[3154,8672,5486],{}," у Controller.",[3195,8675,8676,8677,8679,8680,5502],{},"Controller ініціалізує Bindings (",[3154,8678,5510],{},") та викликає ",[3154,8681,5501],{},[3195,8683,8684,8686],{},[3154,8685,7195],{}," повертає Root Node, готовий до відображення.",[3236,8688],{},[3145,8690,8692],{"id":8691},"альтернативні-підходи-до-інєкції-viewmodel","Альтернативні підходи до ін'єкції ViewModel",[3150,8694,8695],{},"Ми розглянули Constructor Injection через Guice. Але є інші підходи:",[3250,8697,8699],{"id":8698},"підхід-1-setter-injection-без-guice","Підхід 1: Setter Injection (без Guice)",[3262,8701,8703],{"className":4730,"code":8702,"language":4732,"meta":3267,"style":3267},"public class AudiobookListController {\n    private AudiobookListViewModel viewModel;\n    \n    public void setViewModel(AudiobookListViewModel viewModel) {\n        this.viewModel = viewModel;\n        setupBindings();\n    }\n}\n",[3154,8704,8705,8715,8725,8729,8745,8755,8761,8765],{"__ignoreMap":3267},[3271,8706,8707,8709,8711,8713],{"class":3273,"line":3274},[3271,8708,4784],{"class":4739},[3271,8710,4787],{"class":4739},[3271,8712,4791],{"class":4790},[3271,8714,4666],{"class":3288},[3271,8716,8717,8719,8721,8723],{"class":3273,"line":3307},[3271,8718,5139],{"class":4739},[3271,8720,5142],{"class":4790},[3271,8722,5145],{"class":4831},[3271,8724,4678],{"class":3288},[3271,8726,8727],{"class":3273,"line":3314},[3271,8728,3416],{"class":3288},[3271,8730,8731,8733,8735,8737,8739,8741,8743],{"class":3273,"line":3330},[3271,8732,5172],{"class":4739},[3271,8734,5175],{"class":4790},[3271,8736,5210],{"class":5178},[3271,8738,5213],{"class":3288},[3271,8740,3156],{"class":4790},[3271,8742,5145],{"class":4831},[3271,8744,5220],{"class":3288},[3271,8746,8747,8749,8751,8753],{"class":3273,"line":3344},[3271,8748,5225],{"class":4739},[3271,8750,4548],{"class":3288},[3271,8752,5230],{"class":4831},[3271,8754,5233],{"class":3288},[3271,8756,8757,8759],{"class":3273,"line":3358},[3271,8758,5238],{"class":5178},[3271,8760,5241],{"class":3288},[3271,8762,8763],{"class":3273,"line":3363},[3271,8764,5197],{"class":3288},[3271,8766,8767],{"class":3273,"line":3380},[3271,8768,4703],{"class":3288},[3150,8770,8771,8772,6594],{},"Після завантаження FXML вручну викликаємо ",[3154,8773,5477],{},[3262,8775,8777],{"className":4730,"code":8776,"language":4732,"meta":3267,"style":3267},"FXMLLoader loader = new FXMLLoader(getClass().getResource(\"audiobook-list-view.fxml\"));\nParent root = loader.load();\nAudiobookListController controller = loader.getController();\ncontroller.setViewModel(new AudiobookListViewModel(service));\n",[3154,8778,8779,8805,8821,8839],{"__ignoreMap":3267},[3271,8780,8781,8783,8785,8787,8789,8791,8793,8795,8797,8799,8801,8803],{"class":3273,"line":3274},[3271,8782,7195],{"class":4790},[3271,8784,7215],{"class":4831},[3271,8786,6740],{"class":3288},[3271,8788,6743],{"class":6214},[3271,8790,7222],{"class":5178},[3271,8792,5213],{"class":3288},[3271,8794,7227],{"class":5178},[3271,8796,5596],{"class":3288},[3271,8798,7232],{"class":5178},[3271,8800,5213],{"class":3288},[3271,8802,7237],{"class":5934},[3271,8804,7240],{"class":3288},[3271,8806,8807,8809,8811,8813,8815,8817,8819],{"class":3273,"line":3307},[3271,8808,7245],{"class":4790},[3271,8810,7248],{"class":4831},[3271,8812,6740],{"class":3288},[3271,8814,7253],{"class":4831},[3271,8816,4548],{"class":3288},[3271,8818,7258],{"class":5178},[3271,8820,5241],{"class":3288},[3271,8822,8823,8825,8828,8830,8832,8834,8837],{"class":3273,"line":3314},[3271,8824,4718],{"class":4790},[3271,8826,8827],{"class":4831}," controller",[3271,8829,6740],{"class":3288},[3271,8831,7253],{"class":4831},[3271,8833,4548],{"class":3288},[3271,8835,8836],{"class":5178},"getController",[3271,8838,5241],{"class":3288},[3271,8840,8841,8844,8846,8849,8851,8853,8855],{"class":3273,"line":3330},[3271,8842,8843],{"class":4831},"controller",[3271,8845,4548],{"class":3288},[3271,8847,8848],{"class":5178},"setViewModel",[3271,8850,5213],{"class":3288},[3271,8852,6743],{"class":6214},[3271,8854,5142],{"class":5178},[3271,8856,8857],{"class":3288},"(service));\n",[3150,8859,8860,8863,8864,8867,8870],{},[3162,8861,8862],{},"Переваги:"," Простота, не потрібен Guice.",[8865,8866],"br",{},[3162,8868,8869],{},"Недоліки:"," Ручне створення залежностей, не масштабується.",[3250,8872,8874],{"id":8873},"підхід-2-field-injection-через-guice","Підхід 2: Field Injection через Guice",[3262,8876,8878],{"className":4730,"code":8877,"language":4732,"meta":3267,"style":3267},"public class AudiobookListController {\n    @Inject\n    private AudiobookListViewModel viewModel;\n    \n    @FXML\n    public void initialize() {\n        setupBindings();\n    }\n}\n",[3154,8879,8880,8890,8896,8906,8910,8916,8926,8932,8936],{"__ignoreMap":3267},[3271,8881,8882,8884,8886,8888],{"class":3273,"line":3274},[3271,8883,4784],{"class":4739},[3271,8885,4787],{"class":4739},[3271,8887,4791],{"class":4790},[3271,8889,4666],{"class":3288},[3271,8891,8892,8894],{"class":3273,"line":3307},[3271,8893,4811],{"class":3288},[3271,8895,7308],{"class":4790},[3271,8897,8898,8900,8902,8904],{"class":3273,"line":3314},[3271,8899,5139],{"class":4739},[3271,8901,5142],{"class":4790},[3271,8903,5145],{"class":4831},[3271,8905,4678],{"class":3288},[3271,8907,8908],{"class":3273,"line":3330},[3271,8909,3416],{"class":3288},[3271,8911,8912,8914],{"class":3273,"line":3344},[3271,8913,4811],{"class":3288},[3271,8915,5167],{"class":4790},[3271,8917,8918,8920,8922,8924],{"class":3273,"line":3358},[3271,8919,5172],{"class":4739},[3271,8921,5175],{"class":4790},[3271,8923,5179],{"class":5178},[3271,8925,5182],{"class":3288},[3271,8927,8928,8930],{"class":3273,"line":3363},[3271,8929,5238],{"class":5178},[3271,8931,5241],{"class":3288},[3271,8933,8934],{"class":3273,"line":3380},[3271,8935,5197],{"class":3288},[3271,8937,8938],{"class":3273,"line":3391},[3271,8939,4703],{"class":3288},[3150,8941,8942,8943,8945],{},"Guice ін'єктує ",[3154,8944,5230],{}," через поле (Field Injection).",[3150,8947,8948,8950,8951,8953,8955],{},[3162,8949,8862],{}," Менше коду (немає конструктора).",[8865,8952],{},[3162,8954,8869],{}," Field Injection вважається anti-pattern (складніше тестувати, неявні залежності).",[3250,8957,8959],{"id":8958},"підхід-3-constructor-injection-через-guice-рекомендований","Підхід 3: Constructor Injection через Guice (рекомендований)",[3262,8961,8963],{"className":4730,"code":8962,"language":4732,"meta":3267,"style":3267},"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        setupBindings();\n        viewModel.initialize();\n    }\n}\n",[3154,8964,8965,8975,8987,8991,8997,9011,9021,9025,9029,9035,9045,9051,9061,9065],{"__ignoreMap":3267},[3271,8966,8967,8969,8971,8973],{"class":3273,"line":3274},[3271,8968,4784],{"class":4739},[3271,8970,4787],{"class":4739},[3271,8972,4791],{"class":4790},[3271,8974,4666],{"class":3288},[3271,8976,8977,8979,8981,8983,8985],{"class":3273,"line":3307},[3271,8978,5139],{"class":4739},[3271,8980,7291],{"class":4739},[3271,8982,5142],{"class":4790},[3271,8984,5145],{"class":4831},[3271,8986,4678],{"class":3288},[3271,8988,8989],{"class":3273,"line":3314},[3271,8990,3416],{"class":3288},[3271,8992,8993,8995],{"class":3273,"line":3330},[3271,8994,4811],{"class":3288},[3271,8996,7308],{"class":4790},[3271,8998,8999,9001,9003,9005,9007,9009],{"class":3273,"line":3344},[3271,9000,5172],{"class":4739},[3271,9002,4791],{"class":5178},[3271,9004,5213],{"class":3288},[3271,9006,3156],{"class":4790},[3271,9008,5145],{"class":4831},[3271,9010,5220],{"class":3288},[3271,9012,9013,9015,9017,9019],{"class":3273,"line":3358},[3271,9014,5225],{"class":4739},[3271,9016,4548],{"class":3288},[3271,9018,5230],{"class":4831},[3271,9020,5233],{"class":3288},[3271,9022,9023],{"class":3273,"line":3363},[3271,9024,5197],{"class":3288},[3271,9026,9027],{"class":3273,"line":3380},[3271,9028,3416],{"class":3288},[3271,9030,9031,9033],{"class":3273,"line":3391},[3271,9032,4811],{"class":3288},[3271,9034,5167],{"class":4790},[3271,9036,9037,9039,9041,9043],{"class":3273,"line":3413},[3271,9038,5172],{"class":4739},[3271,9040,5175],{"class":4790},[3271,9042,5179],{"class":5178},[3271,9044,5182],{"class":3288},[3271,9046,9047,9049],{"class":3273,"line":3419},[3271,9048,5238],{"class":5178},[3271,9050,5241],{"class":3288},[3271,9052,9053,9055,9057,9059],{"class":3273,"line":3426},[3271,9054,5246],{"class":4831},[3271,9056,4548],{"class":3288},[3271,9058,5251],{"class":5178},[3271,9060,5241],{"class":3288},[3271,9062,9063],{"class":3273,"line":3437},[3271,9064,5197],{"class":3288},[3271,9066,9067],{"class":3273,"line":3464},[3271,9068,4703],{"class":3288},[3150,9070,9071,9073,9074,7865,9077,9079,9081],{},[3162,9072,8862],{}," Явні залежності, легко тестувати (можна передати mock у конструктор), immutability (",[3154,9075,9076],{},"final",[8865,9078],{},[3162,9080,8869],{}," Трохи більше коду.",[4639,9083,9084,9086],{},[3162,9085,6580],{}," Використовуйте Constructor Injection через Guice для production-додатків. Setter Injection — для прототипів та навчальних проєктів.",[3236,9088],{},[3145,9090,9092],{"id":9091},"порівняння-controller-у-mvc-mvp-та-mvvm","Порівняння: Controller у MVC, MVP та MVVM",[3150,9094,9095],{},"Підсумуємо, як змінюється роль Controller у різних архітектурних патернах:",[9097,9098,9099,9105,9122,9128,9135,9152,9157,9165,9171,9187,9193,9204],"tabs",{},[3150,9100,9101,9102,9104],{},"== MVC (Model-View-Controller)\n",[3162,9103,3168],{}," — це \"мозок\" додатку. Він:",[3192,9106,9107,9110,9113],{},[3195,9108,9109],{},"Отримує події від View (кліки кнопок).",[3195,9111,9112],{},"Викликає методи Model (бізнес-логіка, доступ до даних).",[3195,9114,9115,9116,3223,9119,7865],{},"Оновлює View вручну (",[3154,9117,9118],{},"view.setTitle()",[3154,9120,9121],{},"view.showError()",[3150,9123,9124,9127],{},[3162,9125,9126],{},"Проблема:"," Controller знає про View та Model — сильна зв'язаність. Складно тестувати (потрібен UI).",[3150,9129,9130,9131,9134],{},"== MVP (Model-View-Presenter)\n",[3162,9132,9133],{},"Presenter"," — це \"розумний\" посередник. Він:",[3192,9136,9137,9143,9146],{},[3195,9138,9139,9140,7865],{},"Отримує події від View через інтерфейс (",[3154,9141,9142],{},"view.onAddClicked()",[3195,9144,9145],{},"Викликає методи Model.",[3195,9147,9148,9149,7865],{},"Оновлює View через інтерфейс (",[3154,9150,9151],{},"view.displayAudiobooks(list)",[3150,9153,9154,9156],{},[3162,9155,8862],{}," View — це інтерфейс, легко тестувати (mock View). Presenter не залежить від JavaFX.",[3150,9158,9159,9161,9162,9164],{},[3162,9160,8869],{}," Багато boilerplate-коду (методи ",[3154,9163,6439],{}," для кожного UI-елемента).",[3150,9166,9167,9168,9170],{},"== MVVM (Model-View-ViewModel)\n",[3162,9169,3168],{}," — це \"дурний\" адаптер. Він:",[3192,9172,9173,9176,9181],{},[3195,9174,9175],{},"Ініціалізує Bindings між View та ViewModel.",[3195,9177,9178,9179,7865],{},"Делегує події до ViewModel (",[3154,9180,7065],{},[3195,9182,9183,9186],{},[3162,9184,9185],{},"Не містить логіки"," — лише з'єднує View з ViewModel.",[3150,9188,9189,9192],{},[3162,9190,9191],{},"ViewModel"," — це \"розумний\" компонент. Він:",[3192,9194,9195,9198,9201],{},[3195,9196,9197],{},"Містить всю логіку (валідація, фільтрація, асинхронність).",[3195,9199,9200],{},"Експонує Properties для Bindings.",[3195,9202,9203],{},"Не знає про View (не залежить від JavaFX UI-класів).",[3150,9205,9206,9208],{},[3162,9207,8862],{}," Автоматична синхронізація через Bindings, легко тестувати ViewModel (POJO), мінімальний код у Controller.",[3236,9210],{},[3145,9212,9214],{"id":9213},"практичні-завдання","Практичні завдання",[8532,9216,9217,9221,9231,9243,9255,9259,9272,9278,9288,9292,9310,9325],{},[3250,9218,9220],{"id":9219},"рівень-1-базове-звязування","Рівень 1: Базове зв'язування",[3150,9222,9223,9226,9227,9230],{},[3162,9224,9225],{},"Завдання 1.1:"," Створіть FXML-файл ",[3154,9228,9229],{},"author-list-view.fxml"," з таблицею авторів (колонки: First Name, Last Name, Birth Year) та кнопками Add, Edit, Delete.",[3150,9232,9233,9236,9237,5466,9240,9242],{},[3162,9234,9235],{},"Завдання 1.2:"," Створіть ",[3154,9238,9239],{},"AuthorListController",[3154,9241,4547],{}," полями для всіх UI-елементів.",[3150,9244,9245,9248,9249,9251,9252,4548],{},[3162,9246,9247],{},"Завдання 1.3:"," Реалізуйте метод ",[3154,9250,5510],{},", що прив'язує колонки таблиці до Properties ",[3154,9253,9254],{},"AuthorViewModel",[3250,9256,9258],{"id":9257},"рівень-2-bindings-та-обробка-подій","Рівень 2: Bindings та обробка подій",[3150,9260,9261,9264,9265,9267,9268,9271],{},[3162,9262,9263],{},"Завдання 2.1:"," Додайте у ",[3154,9266,3259],{}," CheckBox \"Show only favorites\". Прив'яжіть його до ",[3154,9269,9270],{},"BooleanProperty showOnlyFavorites"," у ViewModel через Bidirectional Binding.",[3150,9273,9274,9277],{},[3162,9275,9276],{},"Завдання 2.2:"," Реалізуйте обробник подвійного кліку на рядку таблиці, що відкриває діалог з деталями аудіокниги.",[3150,9279,9280,9283,9284,9287],{},[3162,9281,9282],{},"Завдання 2.3:"," Додайте ProgressBar, що показує прогрес завантаження даних. Прив'яжіть його до ",[3154,9285,9286],{},"DoubleProperty loadingProgress"," у ViewModel.",[3250,9289,9291],{"id":9290},"рівень-3-guice-integration","Рівень 3: Guice Integration",[3150,9293,9294,9236,9297,9300,9301,3223,9304,3223,9307,4548],{},[3162,9295,9296],{},"Завдання 3.1:",[3154,9298,9299],{},"AuthorModule"," (Guice Module) з Bindings для ",[3154,9302,9303],{},"AuthorRepository",[3154,9305,9306],{},"AuthorService",[3154,9308,9309],{},"AuthorListViewModel",[3150,9311,9312,9315,9316,9318,9319,8037,9321,5466,9323,4548],{},[3162,9313,9314],{},"Завдання 3.2:"," Модифікуйте ",[3154,9317,7515],{},", щоб він завантажував ",[3154,9320,9229],{},[3154,9322,7195],{},[3154,9324,7365],{},[3150,9326,9327,9330,9331,4515,9334,9337,9338,9341,9342,9345],{},[3162,9328,9329],{},"Завдання 3.3:"," Реалізуйте навігацію між ",[3154,9332,9333],{},"AudiobookListView",[3154,9335,9336],{},"AuthorListView"," через кнопку \"Manage Authors\". Створіть простий ",[3154,9339,9340],{},"Navigator"," (клас з методом ",[3154,9343,9344],{},"navigateTo(String viewName)","), що змінює Root Node у Scene.",[3236,9347],{},[3145,9349,9351],{"id":9350},"підсумок","Підсумок",[3150,9353,9354],{},"У цій статті ми з'єднали ViewModel з UI через View (FXML) та Controller. Ключові висновки:",[3150,9356,9357,9360],{},[3162,9358,9359],{},"Controller у MVVM — це мінімальний адаптер."," Його єдина відповідальність — ініціалізувати Bindings між UI-елементами та ViewModel Properties. Якщо Controller перевищує 100-150 рядків, це сигнал, що логіка просочилася з ViewModel.",[3150,9362,9363,9366,9367,9369,9370,9372],{},[3162,9364,9365],{},"FXML — це декларативний опис UI."," Він описує структуру (що відображається), а не логіку (як воно працює). Атрибут ",[3154,9368,4539],{}," з'єднує FXML-елементи з ",[3154,9371,4547],{}," полями у Controller.",[3150,9374,9375,9378,9379,9382,9383,9385,9386,3223,9388,9391],{},[3162,9376,9377],{},"Bindings — це серце MVVM."," Unidirectional Bindings (",[3154,9380,9381],{},"bind()",") для відображення даних з ViewModel у View. Bidirectional Bindings (",[3154,9384,6401],{},") для синхронізації введення користувача з ViewModel. Computed Bindings (",[3154,9387,6359],{},[3154,9389,9390],{},"Bindings.when()",") для складних залежностей.",[3150,9393,9394,9397,9398,9400],{},[3162,9395,9396],{},"Event Handlers делегують виклики до ViewModel."," Controller не містить логіки обробки подій — лише виклик ",[3154,9399,5530],{},". Виняток — UI-логіка (діалоги підтвердження).",[3150,9402,9403,4491,9406,9409],{},[3162,9404,9405],{},"Guice інтегрується через ControllerFactory.",[3154,9407,9408],{},"FXMLLoader.setControllerFactory(injector::getInstance)"," дозволяє Guice створювати Controllers з автоматичною ін'єкцією залежностей (ViewModel, Navigator).",[3150,9411,9412,4491,9415,9417,9418,9420,9421,9423],{},[3162,9413,9414],{},"Життєвий цикл завантаження:",[3154,9416,7195],{}," парсить FXML → створює Controller через ControllerFactory → ін'єктує ",[3154,9419,4547],{}," поля → викликає ",[3154,9422,5486],{}," → повертає Root Node.",[3150,9425,9426],{},"У наступній статті ми розглянемо повну інтеграцію MVVM з Guice: як організувати модулі, як керувати життєвим циклом ViewModel (Singleton vs Prototype), як передавати параметри між екранами, та як тестувати Controller з mock ViewModel.",[9428,9429,9430],"style",{},"html pre.shiki code .s0P7L, html code.shiki .s0P7L{--shiki-light:#800000;--shiki-default:#808080;--shiki-dark:#808080}html pre.shiki code .sKtos, html code.shiki .sKtos{--shiki-light:#800000;--shiki-default:#569CD6;--shiki-dark:#569CD6}html pre.shiki code .sa4r_, html code.shiki .sa4r_{--shiki-light:#E50000;--shiki-default:#9CDCFE;--shiki-dark:#9CDCFE}html pre.shiki code .sHH4Y, html code.shiki .sHH4Y{--shiki-light:#000000;--shiki-default:#D4D4D4;--shiki-dark:#D4D4D4}html pre.shiki code .su9tN, html code.shiki .su9tN{--shiki-light:#0000FF;--shiki-default:#CE9178;--shiki-dark:#CE9178}html pre.shiki code .spJ8K, html code.shiki .spJ8K{--shiki-light:#008000;--shiki-default:#6A9955;--shiki-dark:#6A9955}html .light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html.light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html.dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html pre.shiki code .sqdDX, html code.shiki .sqdDX{--shiki-light:#800000;--shiki-default:#D7BA7D;--shiki-dark:#D7BA7D}html pre.shiki code .sDUd3, html code.shiki .sDUd3{--shiki-light:#0451A5;--shiki-default:#CE9178;--shiki-dark:#CE9178}html pre.shiki code .su1O8, html code.shiki .su1O8{--shiki-light:#0000FF;--shiki-default:#569CD6;--shiki-dark:#569CD6}html pre.shiki code .sN1BT, html code.shiki .sN1BT{--shiki-light:#267F99;--shiki-default:#4EC9B0;--shiki-dark:#4EC9B0}html pre.shiki code .siwwj, html code.shiki .siwwj{--shiki-light:#001080;--shiki-default:#9CDCFE;--shiki-dark:#9CDCFE}html pre.shiki code .s8Opu, html code.shiki .s8Opu{--shiki-light:#795E26;--shiki-default:#DCDCAA;--shiki-dark:#DCDCAA}html pre.shiki code .sbdoH, html code.shiki .sbdoH{--shiki-light:#A31515;--shiki-default:#CE9178;--shiki-dark:#CE9178}html pre.shiki code .s8xlr, html code.shiki .s8xlr{--shiki-light:#AF00DB;--shiki-default:#C586C0;--shiki-dark:#C586C0}html pre.shiki code .sJj4R, html code.shiki .sJj4R{--shiki-light:#098658;--shiki-default:#B5CEA8;--shiki-dark:#B5CEA8}",{"title":3267,"searchDepth":3307,"depth":3307,"links":9432},[9433,9434,9438,9442,9447,9453,9461,9466,9467,9472],{"id":3147,"depth":3307,"text":3148},{"id":3240,"depth":3307,"text":3241,"children":9435},[9436,9437],{"id":3252,"depth":3314,"text":3253},{"id":4484,"depth":3314,"text":4485},{"id":4711,"depth":3307,"text":4712,"children":9439},[9440,9441],{"id":4726,"depth":3314,"text":4727},{"id":5533,"depth":3314,"text":5534},{"id":6445,"depth":3307,"text":6446,"children":9443},[9444,9445,9446],{"id":6458,"depth":3314,"text":6459},{"id":6587,"depth":3314,"text":6588},{"id":7100,"depth":3314,"text":7101},{"id":7168,"depth":3307,"text":7169,"children":9448},[9449,9450,9451,9452],{"id":7188,"depth":3314,"text":7189},{"id":7356,"depth":3314,"text":7357},{"id":7508,"depth":3314,"text":7509},{"id":8090,"depth":3314,"text":8091},{"id":8524,"depth":3307,"text":8525,"children":9454},[9455,9456,9457,9458,9459,9460],{"id":8536,"depth":3314,"text":8537},{"id":8545,"depth":3314,"text":8546},{"id":8561,"depth":3314,"text":8562},{"id":8575,"depth":3314,"text":8576},{"id":8587,"depth":3314,"text":8588},{"id":8599,"depth":3314,"text":8600},{"id":8691,"depth":3307,"text":8692,"children":9462},[9463,9464,9465],{"id":8698,"depth":3314,"text":8699},{"id":8873,"depth":3314,"text":8874},{"id":8958,"depth":3314,"text":8959},{"id":9091,"depth":3307,"text":9092},{"id":9213,"depth":3307,"text":9214,"children":9468},[9469,9470,9471],{"id":9219,"depth":3314,"text":9220},{"id":9257,"depth":3314,"text":9258},{"id":9290,"depth":3314,"text":9291},{"id":9350,"depth":3307,"text":9351},"Від ViewModel до UI: FXML як декларативний опис інтерфейсу, мінімальний Controller з Bindings, ін'єкція ViewModel, FXMLLoader та ControllerFactory, обробка подій та навігація.","md",null,{},{"title":2430,"description":9473},"hSAywlD52kqI0K7qzQiv3ONz6qWAHuPz23FFTpMILHI",[9480,9482],{"title":2426,"path":2427,"stem":2428,"description":9481,"children":-1},"Від теорії до коду: анатомія ViewModel, Wrapper Pattern для Domain Model, Properties та Commands, асинхронність через Task, lifecycle management та тестування без UI.",{"title":2434,"path":2435,"stem":2436,"description":9483,"children":-1},"Від ручного створення об'єктів до Dependency Injection: Guice Module, ControllerFactory, Constructor Injection, Scopes (Singleton vs Prototype), AssistedInject для параметризованих ViewModel.",1778998390296]