[{"data":1,"prerenderedAt":11333},["ShallowReactive",2],{"navigation_docs":3,"-java-pr2-mvvm-navigation-screen-management":3135,"-java-pr2-mvvm-navigation-screen-management-surround":11328},[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":2442,"body":3137,"description":11322,"extension":11323,"links":11324,"meta":11325,"navigation":3992,"path":2443,"seo":11326,"stem":2444,"__hash__":11327},"docs\u002F04.java\u002Fpr2\u002F32.mvvm-navigation-screen-management.md",{"type":3138,"value":3139,"toc":11283},"minimark",[3140,3144,3149,3153,3161,3164,3420,3425,3431,3437,3446,3456,3462,3469,3472,3494,3501,3504,3508,3511,3516,3530,3578,3583,3594,3599,3607,3613,3617,3625,3690,3694,3709,3713,3721,3726,3730,3735,3802,3806,3814,3818,3826,3831,3835,3948,3962,3964,3968,3971,3975,4445,4448,4452,6104,6108,6119,6134,6147,6157,6175,6181,6195,6206,6216,6218,6222,6232,6236,7045,7050,7085,7089,7485,7495,7497,7501,7507,7511,7526,7707,7712,7723,7727,7742,8028,8033,8042,8046,8054,8287,8292,8297,8316,8318,8322,8325,8329,8462,8465,8469,8476,8762,8765,8918,8921,9231,9239,9244,9248,9251,9598,9603,9612,9619,9621,9625,9628,9632,9722,9726,9731,10043,10048,10426,10431,10522,10526,10531,10941,10945,11038,11052,11054,11058,11194,11196,11200,11203,11214,11220,11230,11236,11248,11270,11276,11279],[3141,3142,2442],"h1",{"id":3143},"навігація-та-управління-екранами-у-javafx-mvvm",[3145,3146,3148],"h2",{"id":3147},"вступ-проблема-переходів-між-екранами","Вступ: Проблема переходів між екранами",[3150,3151,3152],"p",{},"Користувач переглядає список аудіокниг. Натискає \"Add Audiobook\" → відкривається форма додавання. Заповнює форму, натискає \"Save\" → повертається до списку, де бачить нову аудіокнигу. Натискає на аудіокнигу → відкривається екран деталей. Натискає \"Edit\" → відкривається форма редагування з даними цієї аудіокниги.",[3150,3154,3155,3156,3160],{},"Це типовий сценарій навігації у desktop-додатку. Але ",[3157,3158,3159],"strong",{},"хто керує цими переходами?"," Controller? ViewModel? Application?",[3150,3162,3163],{},"У наївному підході Controller містить код навігації:",[3165,3166,3171],"pre",{"className":3167,"code":3168,"language":3169,"meta":3170,"style":3170},"language-java shiki shiki-themes light-plus dark-plus dark-plus","@FXML\nprivate void onAddClicked() {\n    try {\n        FXMLLoader loader = new FXMLLoader(getClass().getResource(\"audiobook-form-view.fxml\"));\n        Parent root = loader.load();\n        \n        AudiobookFormController controller = loader.getController();\n        controller.setViewModel(new AudiobookFormViewModel(service));\n        \n        Stage stage = (Stage) addButton.getScene().getWindow();\n        stage.getScene().setRoot(root);\n    } catch (IOException e) {\n        e.printStackTrace();\n    }\n}\n","java","",[3172,3173,3174,3187,3204,3214,3254,3277,3283,3303,3324,3329,3356,3374,3395,3408,3414],"code",{"__ignoreMap":3170},[3175,3176,3179,3183],"span",{"class":3177,"line":3178},"line",1,[3175,3180,3182],{"class":3181},"sHH4Y","@",[3175,3184,3186],{"class":3185},"sN1BT","FXML\n",[3175,3188,3190,3194,3197,3201],{"class":3177,"line":3189},2,[3175,3191,3193],{"class":3192},"su1O8","private",[3175,3195,3196],{"class":3185}," void",[3175,3198,3200],{"class":3199},"s8Opu"," onAddClicked",[3175,3202,3203],{"class":3181},"() {\n",[3175,3205,3207,3211],{"class":3177,"line":3206},3,[3175,3208,3210],{"class":3209},"s8xlr","    try",[3175,3212,3213],{"class":3181}," {\n",[3175,3215,3217,3220,3224,3227,3230,3233,3236,3239,3242,3245,3247,3251],{"class":3177,"line":3216},4,[3175,3218,3219],{"class":3185},"        FXMLLoader",[3175,3221,3223],{"class":3222},"siwwj"," loader",[3175,3225,3226],{"class":3181}," = ",[3175,3228,3229],{"class":3209},"new",[3175,3231,3232],{"class":3199}," FXMLLoader",[3175,3234,3235],{"class":3181},"(",[3175,3237,3238],{"class":3199},"getClass",[3175,3240,3241],{"class":3181},"().",[3175,3243,3244],{"class":3199},"getResource",[3175,3246,3235],{"class":3181},[3175,3248,3250],{"class":3249},"sbdoH","\"audiobook-form-view.fxml\"",[3175,3252,3253],{"class":3181},"));\n",[3175,3255,3257,3260,3263,3265,3268,3271,3274],{"class":3177,"line":3256},5,[3175,3258,3259],{"class":3185},"        Parent",[3175,3261,3262],{"class":3222}," root",[3175,3264,3226],{"class":3181},[3175,3266,3267],{"class":3222},"loader",[3175,3269,3270],{"class":3181},".",[3175,3272,3273],{"class":3199},"load",[3175,3275,3276],{"class":3181},"();\n",[3175,3278,3280],{"class":3177,"line":3279},6,[3175,3281,3282],{"class":3181},"        \n",[3175,3284,3286,3289,3292,3294,3296,3298,3301],{"class":3177,"line":3285},7,[3175,3287,3288],{"class":3185},"        AudiobookFormController",[3175,3290,3291],{"class":3222}," controller",[3175,3293,3226],{"class":3181},[3175,3295,3267],{"class":3222},[3175,3297,3270],{"class":3181},[3175,3299,3300],{"class":3199},"getController",[3175,3302,3276],{"class":3181},[3175,3304,3306,3309,3311,3314,3316,3318,3321],{"class":3177,"line":3305},8,[3175,3307,3308],{"class":3222},"        controller",[3175,3310,3270],{"class":3181},[3175,3312,3313],{"class":3199},"setViewModel",[3175,3315,3235],{"class":3181},[3175,3317,3229],{"class":3209},[3175,3319,3320],{"class":3199}," AudiobookFormViewModel",[3175,3322,3323],{"class":3181},"(service));\n",[3175,3325,3327],{"class":3177,"line":3326},9,[3175,3328,3282],{"class":3181},[3175,3330,3332,3335,3338,3341,3344,3346,3349,3351,3354],{"class":3177,"line":3331},10,[3175,3333,3334],{"class":3185},"        Stage",[3175,3336,3337],{"class":3222}," stage",[3175,3339,3340],{"class":3181}," = (Stage) ",[3175,3342,3343],{"class":3222},"addButton",[3175,3345,3270],{"class":3181},[3175,3347,3348],{"class":3199},"getScene",[3175,3350,3241],{"class":3181},[3175,3352,3353],{"class":3199},"getWindow",[3175,3355,3276],{"class":3181},[3175,3357,3359,3362,3364,3366,3368,3371],{"class":3177,"line":3358},11,[3175,3360,3361],{"class":3222},"        stage",[3175,3363,3270],{"class":3181},[3175,3365,3348],{"class":3199},[3175,3367,3241],{"class":3181},[3175,3369,3370],{"class":3199},"setRoot",[3175,3372,3373],{"class":3181},"(root);\n",[3175,3375,3377,3380,3383,3386,3389,3392],{"class":3177,"line":3376},12,[3175,3378,3379],{"class":3181},"    } ",[3175,3381,3382],{"class":3209},"catch",[3175,3384,3385],{"class":3181}," (",[3175,3387,3388],{"class":3185},"IOException",[3175,3390,3391],{"class":3222}," e",[3175,3393,3394],{"class":3181},") {\n",[3175,3396,3398,3401,3403,3406],{"class":3177,"line":3397},13,[3175,3399,3400],{"class":3222},"        e",[3175,3402,3270],{"class":3181},[3175,3404,3405],{"class":3199},"printStackTrace",[3175,3407,3276],{"class":3181},[3175,3409,3411],{"class":3177,"line":3410},14,[3175,3412,3413],{"class":3181},"    }\n",[3175,3415,3417],{"class":3177,"line":3416},15,[3175,3418,3419],{"class":3181},"}\n",[3150,3421,3422],{},[3157,3423,3424],{},"Проблеми цього підходу:",[3150,3426,3427,3430],{},[3157,3428,3429],{},"Проблема 1: Дублювання коду."," Кожен Controller, що виконує навігацію, містить однаковий код завантаження FXML, створення ViewModel, зміни Scene. Якщо у додатку 10 екранів, це 10 копій цього коду.",[3150,3432,3433,3436],{},[3157,3434,3435],{},"Проблема 2: Порушення Single Responsibility."," Controller не повинен знати, як завантажувати FXML, як створювати ViewModel, як отримувати Stage. Його відповідальність — з'єднувати View з ViewModel.",[3150,3438,3439,3442,3443,3445],{},[3157,3440,3441],{},"Проблема 3: Жорстке кодування шляхів."," ",[3172,3444,3250],{}," — це magic string. Якщо перейменувати файл, доведеться шукати всі місця, де він використовується.",[3150,3447,3448,3451,3452,3455],{},[3157,3449,3450],{},"Проблема 4: Складність передачі параметрів."," Як передати ",[3172,3453,3454],{},"Audiobook"," з екрану списку у форму редагування? Через Singleton ViewModel? Через глобальну змінну? Це призводить до сильної зв'язаності.",[3150,3457,3458,3461],{},[3157,3459,3460],{},"Проблема 5: Відсутність історії навігації."," Як реалізувати кнопку \"Back\"? Потрібно зберігати стек попередніх екранів, але де?",[3150,3463,3464,3465,3468],{},"Рішення всіх цих проблем — ",[3157,3466,3467],{},"Navigator Pattern",": централізований компонент, що керує навігацією. Він знає, як завантажувати екрани, як передавати параметри, як зберігати історію, як відкривати діалоги.",[3150,3470,3471],{},"У цій статті ми побудуємо повноцінну систему навігації для MVVM-додатку. Ми розглянемо:",[3473,3474,3475,3479,3482,3485,3488,3491],"ul",{},[3476,3477,3478],"li",{},"Navigator Pattern для централізованої навігації.",[3476,3480,3481],{},"ScreenRegistry для реєстрації екранів (замість magic strings).",[3476,3483,3484],{},"Передачу параметрів між екранами (через ViewModel, через Navigator, через AssistedInject).",[3476,3486,3487],{},"Navigation Stack для історії переходів (кнопка \"Back\").",[3476,3489,3490],{},"Модальні діалоги з поверненням результату.",[3476,3492,3493],{},"Інтеграцію Navigator з ViewModel (як ViewModel ініціює навігацію без порушення MVVM).",[3495,3496,3497,3500],"note",{},[3157,3498,3499],{},"Navigator у MVVM — це інфраструктурний компонент."," Він не є частиною Model, View або ViewModel. Це окремий шар, що забезпечує навігацію. ViewModel не знає про Navigator безпосередньо — вони комунікують через Events або Callbacks.",[3502,3503],"hr",{},[3145,3505,3507],{"id":3506},"патерни-навігації-у-desktop-додатках","Патерни навігації у desktop-додатках",[3150,3509,3510],{},"Перш ніж писати код, розглянемо три основні патерни навігації у JavaFX-додатках:",[3512,3513,3515],"h3",{"id":3514},"_1-single-window-navigation-заміна-вмісту-одного-вікна","1. Single Window Navigation: Заміна вмісту одного вікна",[3150,3517,3518,3521,3522,3525,3526,3529],{},[3157,3519,3520],{},"Концепція:"," Один ",[3172,3523,3524],{},"Stage"," (вікно), одна ",[3172,3527,3528],{},"Scene",", але Root Node змінюється при навігації.",[3165,3531,3533],{"className":3167,"code":3532,"language":3169,"meta":3170,"style":3170},"\u002F\u002F Перехід з екрану A на екран B\nParent screenB = loadFxml(\"screen-b.fxml\");\nstage.getScene().setRoot(screenB);\n",[3172,3534,3535,3541,3562],{"__ignoreMap":3170},[3175,3536,3537],{"class":3177,"line":3178},[3175,3538,3540],{"class":3539},"spJ8K","\u002F\u002F Перехід з екрану A на екран B\n",[3175,3542,3543,3546,3549,3551,3554,3556,3559],{"class":3177,"line":3189},[3175,3544,3545],{"class":3185},"Parent",[3175,3547,3548],{"class":3222}," screenB",[3175,3550,3226],{"class":3181},[3175,3552,3553],{"class":3199},"loadFxml",[3175,3555,3235],{"class":3181},[3175,3557,3558],{"class":3249},"\"screen-b.fxml\"",[3175,3560,3561],{"class":3181},");\n",[3175,3563,3564,3567,3569,3571,3573,3575],{"class":3177,"line":3206},[3175,3565,3566],{"class":3222},"stage",[3175,3568,3270],{"class":3181},[3175,3570,3348],{"class":3199},[3175,3572,3241],{"class":3181},[3175,3574,3370],{"class":3199},[3175,3576,3577],{"class":3181},"(screenB);\n",[3150,3579,3580],{},[3157,3581,3582],{},"Переваги:",[3473,3584,3585,3588,3591],{},[3476,3586,3587],{},"Просто реалізувати.",[3476,3589,3590],{},"Один екран на весь екран — немає накладання вікон.",[3476,3592,3593],{},"Легко керувати розміром вікна (один Stage).",[3150,3595,3596],{},[3157,3597,3598],{},"Недоліки:",[3473,3600,3601,3604],{},[3476,3602,3603],{},"Неможливо показати два екрани одночасно (наприклад, список + деталі).",[3476,3605,3606],{},"Складно реалізувати модальні діалоги (потрібен окремий Stage).",[3150,3608,3609,3612],{},[3157,3610,3611],{},"Коли використовувати:"," Додатки з лінійною навігацією (wizard, форми), мобільні додатки (один екран на весь екран).",[3512,3614,3616],{"id":3615},"_2-multiple-windows-відкриття-нових-stage","2. Multiple Windows: Відкриття нових Stage",[3150,3618,3619,3621,3622,3624],{},[3157,3620,3520],{}," Кожен екран — це окремий ",[3172,3623,3524],{}," (вікно). Навігація = відкриття нового вікна.",[3165,3626,3628],{"className":3167,"code":3627,"language":3169,"meta":3170,"style":3170},"\u002F\u002F Відкриття нового вікна\nStage newStage = new Stage();\nnewStage.setScene(new Scene(loadFxml(\"screen-b.fxml\")));\nnewStage.show();\n",[3172,3629,3630,3635,3651,3679],{"__ignoreMap":3170},[3175,3631,3632],{"class":3177,"line":3178},[3175,3633,3634],{"class":3539},"\u002F\u002F Відкриття нового вікна\n",[3175,3636,3637,3639,3642,3644,3646,3649],{"class":3177,"line":3189},[3175,3638,3524],{"class":3185},[3175,3640,3641],{"class":3222}," newStage",[3175,3643,3226],{"class":3181},[3175,3645,3229],{"class":3209},[3175,3647,3648],{"class":3199}," Stage",[3175,3650,3276],{"class":3181},[3175,3652,3653,3656,3658,3661,3663,3665,3668,3670,3672,3674,3676],{"class":3177,"line":3206},[3175,3654,3655],{"class":3222},"newStage",[3175,3657,3270],{"class":3181},[3175,3659,3660],{"class":3199},"setScene",[3175,3662,3235],{"class":3181},[3175,3664,3229],{"class":3209},[3175,3666,3667],{"class":3199}," Scene",[3175,3669,3235],{"class":3181},[3175,3671,3553],{"class":3199},[3175,3673,3235],{"class":3181},[3175,3675,3558],{"class":3249},[3175,3677,3678],{"class":3181},")));\n",[3175,3680,3681,3683,3685,3688],{"class":3177,"line":3216},[3175,3682,3655],{"class":3222},[3175,3684,3270],{"class":3181},[3175,3686,3687],{"class":3199},"show",[3175,3689,3276],{"class":3181},[3150,3691,3692],{},[3157,3693,3582],{},[3473,3695,3696,3699,3706],{},[3476,3697,3698],{},"Можна показати кілька екранів одночасно.",[3476,3700,3701,3702,3705],{},"Природна підтримка модальних діалогів (",[3172,3703,3704],{},"stage.initModality(Modality.APPLICATION_MODAL)",").",[3476,3707,3708],{},"Користувач може розташовувати вікна як хоче.",[3150,3710,3711],{},[3157,3712,3598],{},[3473,3714,3715,3718],{},[3476,3716,3717],{},"Складно керувати життєвим циклом вікон (коли закривати, як передавати дані між вікнами).",[3476,3719,3720],{},"Може бути багато відкритих вікон → плутанина.",[3150,3722,3723,3725],{},[3157,3724,3611],{}," Складні додатки з багатьма незалежними екранами (IDE, графічні редактори), діалоги.",[3512,3727,3729],{"id":3728},"_3-master-detail-розділений-екран","3. Master-Detail: Розділений екран",[3150,3731,3732,3734],{},[3157,3733,3520],{}," Один екран розділений на дві частини: Master (список) та Detail (деталі вибраного елемента).",[3165,3736,3738],{"className":3167,"code":3737,"language":3169,"meta":3170,"style":3170},"\u002F\u002F BorderPane: ліворуч список, праворуч деталі\nBorderPane root = new BorderPane();\nroot.setLeft(loadFxml(\"audiobook-list.fxml\"));\nroot.setRight(loadFxml(\"audiobook-detail.fxml\"));\n",[3172,3739,3740,3745,3761,3782],{"__ignoreMap":3170},[3175,3741,3742],{"class":3177,"line":3178},[3175,3743,3744],{"class":3539},"\u002F\u002F BorderPane: ліворуч список, праворуч деталі\n",[3175,3746,3747,3750,3752,3754,3756,3759],{"class":3177,"line":3189},[3175,3748,3749],{"class":3185},"BorderPane",[3175,3751,3262],{"class":3222},[3175,3753,3226],{"class":3181},[3175,3755,3229],{"class":3209},[3175,3757,3758],{"class":3199}," BorderPane",[3175,3760,3276],{"class":3181},[3175,3762,3763,3766,3768,3771,3773,3775,3777,3780],{"class":3177,"line":3206},[3175,3764,3765],{"class":3222},"root",[3175,3767,3270],{"class":3181},[3175,3769,3770],{"class":3199},"setLeft",[3175,3772,3235],{"class":3181},[3175,3774,3553],{"class":3199},[3175,3776,3235],{"class":3181},[3175,3778,3779],{"class":3249},"\"audiobook-list.fxml\"",[3175,3781,3253],{"class":3181},[3175,3783,3784,3786,3788,3791,3793,3795,3797,3800],{"class":3177,"line":3216},[3175,3785,3765],{"class":3222},[3175,3787,3270],{"class":3181},[3175,3789,3790],{"class":3199},"setRight",[3175,3792,3235],{"class":3181},[3175,3794,3553],{"class":3199},[3175,3796,3235],{"class":3181},[3175,3798,3799],{"class":3249},"\"audiobook-detail.fxml\"",[3175,3801,3253],{"class":3181},[3150,3803,3804],{},[3157,3805,3582],{},[3473,3807,3808,3811],{},[3476,3809,3810],{},"Два екрани одночасно — зручно для перегляду списку та деталей.",[3476,3812,3813],{},"Немає переходів між екранами — все на одному екрані.",[3150,3815,3816],{},[3157,3817,3598],{},[3473,3819,3820,3823],{},[3476,3821,3822],{},"Обмежений простір для кожної частини.",[3476,3824,3825],{},"Складно адаптувати під різні розміри екрану.",[3150,3827,3828,3830],{},[3157,3829,3611],{}," Додатки з чіткою структурою Master-Detail (email-клієнти, файлові менеджери).",[3512,3832,3834],{"id":3833},"порівняння-патернів","Порівняння патернів",[3836,3837,3838,3876,3912],"card-group",{},[3839,3840,3843,3847,3858,3862,3870],"card",{"icon":3841,"title":3842},"i-heroicons-window","Single Window",[3150,3844,3845],{},[3157,3846,3582],{},[3473,3848,3849,3852,3855],{},[3476,3850,3851],{},"Простота реалізації",[3476,3853,3854],{},"Один екран на весь екран",[3476,3856,3857],{},"Легке керування розміром",[3150,3859,3860],{},[3157,3861,3598],{},[3473,3863,3864,3867],{},[3476,3865,3866],{},"Неможливо показати два екрани",[3476,3868,3869],{},"Складні модальні діалоги",[3150,3871,3872,3875],{},[3157,3873,3874],{},"Використання:","\nWizard, форми, мобільні додатки",[3839,3877,3880,3884,3895,3899,3907],{"icon":3878,"title":3879},"i-heroicons-squares-2x2","Multiple Windows",[3150,3881,3882],{},[3157,3883,3582],{},[3473,3885,3886,3889,3892],{},[3476,3887,3888],{},"Кілька екранів одночасно",[3476,3890,3891],{},"Природні модальні діалоги",[3476,3893,3894],{},"Гнучке розташування",[3150,3896,3897],{},[3157,3898,3598],{},[3473,3900,3901,3904],{},[3476,3902,3903],{},"Складне керування життєвим циклом",[3476,3905,3906],{},"Можлива плутанина з вікнами",[3150,3908,3909,3911],{},[3157,3910,3874],{},"\nIDE, графічні редактори, складні додатки",[3839,3913,3916,3920,3931,3935,3943],{"icon":3914,"title":3915},"i-heroicons-view-columns","Master-Detail",[3150,3917,3918],{},[3157,3919,3582],{},[3473,3921,3922,3925,3928],{},[3476,3923,3924],{},"Два екрани одночасно",[3476,3926,3927],{},"Немає переходів",[3476,3929,3930],{},"Зручний перегляд",[3150,3932,3933],{},[3157,3934,3598],{},[3473,3936,3937,3940],{},[3476,3938,3939],{},"Обмежений простір",[3476,3941,3942],{},"Складна адаптація",[3150,3944,3945,3947],{},[3157,3946,3874],{},"\nEmail-клієнти, файлові менеджери",[3150,3949,3950,3953,3954,3957,3958,3961],{},[3157,3951,3952],{},"Наш вибір:"," Ми реалізуємо ",[3157,3955,3956],{},"Single Window Navigation"," з підтримкою ",[3157,3959,3960],{},"Multiple Windows для діалогів",". Це найпоширеніший підхід для desktop-додатків: основна навігація через заміну Root Node, діалоги через окремі Stage.",[3502,3963],{},[3145,3965,3967],{"id":3966},"navigator-pattern-централізоване-управління-навігацією","Navigator Pattern: Централізоване управління навігацією",[3150,3969,3970],{},"Navigator — це клас, що інкапсулює всю логіку навігації: завантаження FXML, створення Controllers через Guice, зміну Root Node, відкриття діалогів.",[3512,3972,3974],{"id":3973},"інтерфейс-navigator","Інтерфейс Navigator",[3165,3976,3978],{"className":3167,"code":3977,"language":3169,"meta":3170,"style":3170},"package dev.kostyl.audiobook.infrastructure.navigation;\n\nimport javafx.scene.Parent;\n\nimport java.util.Map;\nimport java.util.Optional;\n\npublic interface Navigator {\n    \n    \u002F**\n     * Перехід на екран за його ідентифікатором.\n     * \n     * @param screenId ідентифікатор екрану (наприклад, \"audiobook-list\")\n     *\u002F\n    void navigateTo(String screenId);\n    \n    \u002F**\n     * Перехід на екран з параметрами.\n     * \n     * @param screenId ідентифікатор екрану\n     * @param params параметри для передачі у ViewModel\n     *\u002F\n    void navigateTo(String screenId, Map\u003CString, Object> params);\n    \n    \u002F**\n     * Відкриття модального діалогу.\n     * \n     * @param screenId ідентифікатор діалогу\n     * @return результат діалогу (Optional.empty() якщо скасовано)\n     *\u002F\n    \u003CT> Optional\u003CT> openDialog(String screenId);\n    \n    \u002F**\n     * Відкриття модального діалогу з параметрами.\n     * \n     * @param screenId ідентифікатор діалогу\n     * @param params параметри для передачі у ViewModel\n     * @return результат діалогу\n     *\u002F\n    \u003CT> Optional\u003CT> openDialog(String screenId, Map\u003CString, Object> params);\n    \n    \u002F**\n     * Повернення на попередній екран (Back button).\n     *\u002F\n    void goBack();\n    \n    \u002F**\n     * Перевірка, чи можна повернутися назад.\n     * \n     * @return true якщо є попередній екран у стеку\n     *\u002F\n    boolean canGoBack();\n}\n",[3172,3979,3980,3988,3994,4002,4006,4013,4020,4024,4037,4042,4047,4052,4057,4071,4076,4093,4098,4103,4109,4114,4126,4139,4144,4181,4186,4191,4197,4202,4214,4225,4230,4261,4266,4271,4277,4282,4293,4304,4314,4319,4362,4367,4372,4378,4383,4393,4398,4403,4409,4414,4424,4429,4440],{"__ignoreMap":3170},[3175,3981,3982,3985],{"class":3177,"line":3178},[3175,3983,3984],{"class":3192},"package",[3175,3986,3987],{"class":3181}," dev.kostyl.audiobook.infrastructure.navigation;\n",[3175,3989,3990],{"class":3177,"line":3189},[3175,3991,3993],{"emptyLinePlaceholder":3992},true,"\n",[3175,3995,3996,3999],{"class":3177,"line":3206},[3175,3997,3998],{"class":3192},"import",[3175,4000,4001],{"class":3181}," javafx.scene.Parent;\n",[3175,4003,4004],{"class":3177,"line":3216},[3175,4005,3993],{"emptyLinePlaceholder":3992},[3175,4007,4008,4010],{"class":3177,"line":3256},[3175,4009,3998],{"class":3192},[3175,4011,4012],{"class":3181}," java.util.Map;\n",[3175,4014,4015,4017],{"class":3177,"line":3279},[3175,4016,3998],{"class":3192},[3175,4018,4019],{"class":3181}," java.util.Optional;\n",[3175,4021,4022],{"class":3177,"line":3285},[3175,4023,3993],{"emptyLinePlaceholder":3992},[3175,4025,4026,4029,4032,4035],{"class":3177,"line":3305},[3175,4027,4028],{"class":3192},"public",[3175,4030,4031],{"class":3192}," interface",[3175,4033,4034],{"class":3185}," Navigator",[3175,4036,3213],{"class":3181},[3175,4038,4039],{"class":3177,"line":3326},[3175,4040,4041],{"class":3181},"    \n",[3175,4043,4044],{"class":3177,"line":3331},[3175,4045,4046],{"class":3539},"    \u002F**\n",[3175,4048,4049],{"class":3177,"line":3358},[3175,4050,4051],{"class":3539},"     * Перехід на екран за його ідентифікатором.\n",[3175,4053,4054],{"class":3177,"line":3376},[3175,4055,4056],{"class":3539},"     * \n",[3175,4058,4059,4062,4065,4068],{"class":3177,"line":3397},[3175,4060,4061],{"class":3539},"     * ",[3175,4063,4064],{"class":3192},"@param",[3175,4066,4067],{"class":3222}," screenId",[3175,4069,4070],{"class":3539}," ідентифікатор екрану (наприклад, \"audiobook-list\")\n",[3175,4072,4073],{"class":3177,"line":3410},[3175,4074,4075],{"class":3539},"     *\u002F\n",[3175,4077,4078,4081,4084,4086,4089,4091],{"class":3177,"line":3416},[3175,4079,4080],{"class":3185},"    void",[3175,4082,4083],{"class":3199}," navigateTo",[3175,4085,3235],{"class":3181},[3175,4087,4088],{"class":3185},"String",[3175,4090,4067],{"class":3222},[3175,4092,3561],{"class":3181},[3175,4094,4096],{"class":3177,"line":4095},16,[3175,4097,4041],{"class":3181},[3175,4099,4101],{"class":3177,"line":4100},17,[3175,4102,4046],{"class":3539},[3175,4104,4106],{"class":3177,"line":4105},18,[3175,4107,4108],{"class":3539},"     * Перехід на екран з параметрами.\n",[3175,4110,4112],{"class":3177,"line":4111},19,[3175,4113,4056],{"class":3539},[3175,4115,4117,4119,4121,4123],{"class":3177,"line":4116},20,[3175,4118,4061],{"class":3539},[3175,4120,4064],{"class":3192},[3175,4122,4067],{"class":3222},[3175,4124,4125],{"class":3539}," ідентифікатор екрану\n",[3175,4127,4129,4131,4133,4136],{"class":3177,"line":4128},21,[3175,4130,4061],{"class":3539},[3175,4132,4064],{"class":3192},[3175,4134,4135],{"class":3222}," params",[3175,4137,4138],{"class":3539}," параметри для передачі у ViewModel\n",[3175,4140,4142],{"class":3177,"line":4141},22,[3175,4143,4075],{"class":3539},[3175,4145,4147,4149,4151,4153,4155,4157,4160,4163,4166,4168,4170,4173,4176,4179],{"class":3177,"line":4146},23,[3175,4148,4080],{"class":3185},[3175,4150,4083],{"class":3199},[3175,4152,3235],{"class":3181},[3175,4154,4088],{"class":3185},[3175,4156,4067],{"class":3222},[3175,4158,4159],{"class":3181},", ",[3175,4161,4162],{"class":3185},"Map",[3175,4164,4165],{"class":3181},"\u003C",[3175,4167,4088],{"class":3185},[3175,4169,4159],{"class":3181},[3175,4171,4172],{"class":3185},"Object",[3175,4174,4175],{"class":3181},"> ",[3175,4177,4178],{"class":3222},"params",[3175,4180,3561],{"class":3181},[3175,4182,4184],{"class":3177,"line":4183},24,[3175,4185,4041],{"class":3181},[3175,4187,4189],{"class":3177,"line":4188},25,[3175,4190,4046],{"class":3539},[3175,4192,4194],{"class":3177,"line":4193},26,[3175,4195,4196],{"class":3539},"     * Відкриття модального діалогу.\n",[3175,4198,4200],{"class":3177,"line":4199},27,[3175,4201,4056],{"class":3539},[3175,4203,4205,4207,4209,4211],{"class":3177,"line":4204},28,[3175,4206,4061],{"class":3539},[3175,4208,4064],{"class":3192},[3175,4210,4067],{"class":3222},[3175,4212,4213],{"class":3539}," ідентифікатор діалогу\n",[3175,4215,4217,4219,4222],{"class":3177,"line":4216},29,[3175,4218,4061],{"class":3539},[3175,4220,4221],{"class":3192},"@return",[3175,4223,4224],{"class":3539}," результат діалогу (Optional.empty() якщо скасовано)\n",[3175,4226,4228],{"class":3177,"line":4227},30,[3175,4229,4075],{"class":3539},[3175,4231,4233,4236,4239,4241,4244,4246,4248,4250,4253,4255,4257,4259],{"class":3177,"line":4232},31,[3175,4234,4235],{"class":3181},"    \u003C",[3175,4237,4238],{"class":3185},"T",[3175,4240,4175],{"class":3181},[3175,4242,4243],{"class":3185},"Optional",[3175,4245,4165],{"class":3181},[3175,4247,4238],{"class":3185},[3175,4249,4175],{"class":3181},[3175,4251,4252],{"class":3199},"openDialog",[3175,4254,3235],{"class":3181},[3175,4256,4088],{"class":3185},[3175,4258,4067],{"class":3222},[3175,4260,3561],{"class":3181},[3175,4262,4264],{"class":3177,"line":4263},32,[3175,4265,4041],{"class":3181},[3175,4267,4269],{"class":3177,"line":4268},33,[3175,4270,4046],{"class":3539},[3175,4272,4274],{"class":3177,"line":4273},34,[3175,4275,4276],{"class":3539},"     * Відкриття модального діалогу з параметрами.\n",[3175,4278,4280],{"class":3177,"line":4279},35,[3175,4281,4056],{"class":3539},[3175,4283,4285,4287,4289,4291],{"class":3177,"line":4284},36,[3175,4286,4061],{"class":3539},[3175,4288,4064],{"class":3192},[3175,4290,4067],{"class":3222},[3175,4292,4213],{"class":3539},[3175,4294,4296,4298,4300,4302],{"class":3177,"line":4295},37,[3175,4297,4061],{"class":3539},[3175,4299,4064],{"class":3192},[3175,4301,4135],{"class":3222},[3175,4303,4138],{"class":3539},[3175,4305,4307,4309,4311],{"class":3177,"line":4306},38,[3175,4308,4061],{"class":3539},[3175,4310,4221],{"class":3192},[3175,4312,4313],{"class":3539}," результат діалогу\n",[3175,4315,4317],{"class":3177,"line":4316},39,[3175,4318,4075],{"class":3539},[3175,4320,4322,4324,4326,4328,4330,4332,4334,4336,4338,4340,4342,4344,4346,4348,4350,4352,4354,4356,4358,4360],{"class":3177,"line":4321},40,[3175,4323,4235],{"class":3181},[3175,4325,4238],{"class":3185},[3175,4327,4175],{"class":3181},[3175,4329,4243],{"class":3185},[3175,4331,4165],{"class":3181},[3175,4333,4238],{"class":3185},[3175,4335,4175],{"class":3181},[3175,4337,4252],{"class":3199},[3175,4339,3235],{"class":3181},[3175,4341,4088],{"class":3185},[3175,4343,4067],{"class":3222},[3175,4345,4159],{"class":3181},[3175,4347,4162],{"class":3185},[3175,4349,4165],{"class":3181},[3175,4351,4088],{"class":3185},[3175,4353,4159],{"class":3181},[3175,4355,4172],{"class":3185},[3175,4357,4175],{"class":3181},[3175,4359,4178],{"class":3222},[3175,4361,3561],{"class":3181},[3175,4363,4365],{"class":3177,"line":4364},41,[3175,4366,4041],{"class":3181},[3175,4368,4370],{"class":3177,"line":4369},42,[3175,4371,4046],{"class":3539},[3175,4373,4375],{"class":3177,"line":4374},43,[3175,4376,4377],{"class":3539},"     * Повернення на попередній екран (Back button).\n",[3175,4379,4381],{"class":3177,"line":4380},44,[3175,4382,4075],{"class":3539},[3175,4384,4386,4388,4391],{"class":3177,"line":4385},45,[3175,4387,4080],{"class":3185},[3175,4389,4390],{"class":3199}," goBack",[3175,4392,3276],{"class":3181},[3175,4394,4396],{"class":3177,"line":4395},46,[3175,4397,4041],{"class":3181},[3175,4399,4401],{"class":3177,"line":4400},47,[3175,4402,4046],{"class":3539},[3175,4404,4406],{"class":3177,"line":4405},48,[3175,4407,4408],{"class":3539},"     * Перевірка, чи можна повернутися назад.\n",[3175,4410,4412],{"class":3177,"line":4411},49,[3175,4413,4056],{"class":3539},[3175,4415,4417,4419,4421],{"class":3177,"line":4416},50,[3175,4418,4061],{"class":3539},[3175,4420,4221],{"class":3192},[3175,4422,4423],{"class":3539}," true якщо є попередній екран у стеку\n",[3175,4425,4427],{"class":3177,"line":4426},51,[3175,4428,4075],{"class":3539},[3175,4430,4432,4435,4438],{"class":3177,"line":4431},52,[3175,4433,4434],{"class":3185},"    boolean",[3175,4436,4437],{"class":3199}," canGoBack",[3175,4439,3276],{"class":3181},[3175,4441,4443],{"class":3177,"line":4442},53,[3175,4444,3419],{"class":3181},[3150,4446,4447],{},"Цей інтерфейс визначає контракт Navigator: які операції він підтримує. Тепер створимо реалізацію.",[3512,4449,4451],{"id":4450},"реалізація-fxmlnavigator","Реалізація FxmlNavigator",[3165,4453,4455],{"className":3167,"code":4454,"language":3169,"meta":3170,"style":3170},"package dev.kostyl.audiobook.infrastructure.navigation;\n\nimport com.google.inject.Inject;\nimport com.google.inject.Injector;\nimport com.google.inject.Singleton;\nimport javafx.fxml.FXMLLoader;\nimport javafx.scene.Parent;\nimport javafx.scene.Scene;\nimport javafx.stage.Modality;\nimport javafx.stage.Stage;\n\nimport java.io.IOException;\nimport java.util.*;\n\n@Singleton\npublic class FxmlNavigator implements Navigator {\n    \n    private final Injector injector;\n    private final ScreenRegistry screenRegistry;\n    private final Stack\u003CNavigationEntry> navigationStack;\n    \n    private Stage primaryStage;\n    \n    @Inject\n    public FxmlNavigator(Injector injector, ScreenRegistry screenRegistry) {\n        this.injector = injector;\n        this.screenRegistry = screenRegistry;\n        this.navigationStack = new Stack\u003C>();\n    }\n    \n    \u002F**\n     * Встановлення primary Stage (викликається з Application.start()).\n     *\u002F\n    public void setPrimaryStage(Stage primaryStage) {\n        this.primaryStage = primaryStage;\n    }\n    \n    @Override\n    public void navigateTo(String screenId) {\n        navigateTo(screenId, Collections.emptyMap());\n    }\n    \n    @Override\n    public void navigateTo(String screenId, Map\u003CString, Object> params) {\n        try {\n            \u002F\u002F Завантаження FXML\n            String fxmlPath = screenRegistry.getFxmlPath(screenId);\n            Parent root = loadFxml(fxmlPath);\n            \n            \u002F\u002F Збереження поточного екрану у стек\n            if (primaryStage.getScene() != null && primaryStage.getScene().getRoot() != null) {\n                navigationStack.push(new NavigationEntry(\n                    getCurrentScreenId(),\n                    primaryStage.getScene().getRoot()\n                ));\n            }\n            \n            \u002F\u002F Зміна Root Node\n            if (primaryStage.getScene() == null) {\n                primaryStage.setScene(new Scene(root));\n            } else {\n                primaryStage.getScene().setRoot(root);\n            }\n            \n            \u002F\u002F Передача параметрів у ViewModel (якщо потрібно)\n            \u002F\u002F TODO: реалізувати передачу параметрів\n            \n        } catch (IOException e) {\n            throw new NavigationException(\"Failed to navigate to screen: \" + screenId, e);\n        }\n    }\n    \n    @Override\n    public \u003CT> Optional\u003CT> openDialog(String screenId) {\n        return openDialog(screenId, Collections.emptyMap());\n    }\n    \n    @Override\n    public \u003CT> Optional\u003CT> openDialog(String screenId, Map\u003CString, Object> params) {\n        try {\n            \u002F\u002F Завантаження FXML\n            String fxmlPath = screenRegistry.getFxmlPath(screenId);\n            FXMLLoader loader = new FXMLLoader(getClass().getResource(fxmlPath));\n            loader.setControllerFactory(injector::getInstance);\n            Parent root = loader.load();\n            \n            \u002F\u002F Створення модального Stage\n            Stage dialogStage = new Stage();\n            dialogStage.initModality(Modality.APPLICATION_MODAL);\n            dialogStage.initOwner(primaryStage);\n            dialogStage.setScene(new Scene(root));\n            \n            \u002F\u002F Показ діалогу (блокує виконання до закриття)\n            dialogStage.showAndWait();\n            \n            \u002F\u002F Отримання результату з Controller\n            Object controller = loader.getController();\n            if (controller instanceof DialogController) {\n                return Optional.ofNullable(((DialogController\u003CT>) controller).getResult());\n            }\n            \n            return Optional.empty();\n            \n        } catch (IOException e) {\n            throw new NavigationException(\"Failed to open dialog: \" + screenId, e);\n        }\n    }\n    \n    @Override\n    public void goBack() {\n        if (!canGoBack()) {\n            throw new IllegalStateException(\"Cannot go back: navigation stack is empty\");\n        }\n        \n        NavigationEntry previousEntry = navigationStack.pop();\n        primaryStage.getScene().setRoot(previousEntry.getRoot());\n    }\n    \n    @Override\n    public boolean canGoBack() {\n        return !navigationStack.isEmpty();\n    }\n    \n    private Parent loadFxml(String fxmlPath) throws IOException {\n        FXMLLoader loader = new FXMLLoader(getClass().getResource(fxmlPath));\n        loader.setControllerFactory(injector::getInstance);\n        return loader.load();\n    }\n    \n    private String getCurrentScreenId() {\n        \u002F\u002F TODO: зберігати поточний screenId при навігації\n        return \"unknown\";\n    }\n    \n    \u002F**\n     * Запис у стеку навігації.\n     *\u002F\n    private static class NavigationEntry {\n        private final String screenId;\n        private final Parent root;\n        \n        public NavigationEntry(String screenId, Parent root) {\n            this.screenId = screenId;\n            this.root = root;\n        }\n        \n        public String getScreenId() {\n            return screenId;\n        }\n        \n        public Parent getRoot() {\n            return root;\n        }\n    }\n}\n",[3172,4456,4457,4463,4467,4474,4481,4488,4495,4501,4508,4515,4522,4526,4533,4540,4544,4551,4568,4572,4589,4603,4624,4628,4639,4643,4651,4674,4687,4699,4716,4720,4724,4728,4733,4737,4754,4766,4770,4774,4781,4797,4816,4820,4824,4830,4862,4869,4874,4894,4908,4913,4918,4957,4977,4985,5002,5008,5014,5019,5025,5045,5064,5075,5090,5095,5100,5106,5112,5117,5133,5153,5159,5164,5169,5176,5206,5225,5230,5235,5242,5287,5294,5299,5316,5341,5361,5378,5383,5389,5406,5429,5442,5459,5464,5470,5482,5487,5493,5511,5525,5553,5558,5563,5578,5583,5598,5614,5619,5624,5629,5636,5647,5662,5679,5684,5689,5709,5734,5739,5744,5751,5763,5780,5785,5790,5818,5841,5857,5870,5875,5880,5893,5899,5909,5914,5919,5924,5930,5935,5949,5963,5976,5981,6003,6017,6029,6034,6039,6051,6059,6064,6069,6081,6089,6094,6099],{"__ignoreMap":3170},[3175,4458,4459,4461],{"class":3177,"line":3178},[3175,4460,3984],{"class":3192},[3175,4462,3987],{"class":3181},[3175,4464,4465],{"class":3177,"line":3189},[3175,4466,3993],{"emptyLinePlaceholder":3992},[3175,4468,4469,4471],{"class":3177,"line":3206},[3175,4470,3998],{"class":3192},[3175,4472,4473],{"class":3181}," com.google.inject.Inject;\n",[3175,4475,4476,4478],{"class":3177,"line":3216},[3175,4477,3998],{"class":3192},[3175,4479,4480],{"class":3181}," com.google.inject.Injector;\n",[3175,4482,4483,4485],{"class":3177,"line":3256},[3175,4484,3998],{"class":3192},[3175,4486,4487],{"class":3181}," com.google.inject.Singleton;\n",[3175,4489,4490,4492],{"class":3177,"line":3279},[3175,4491,3998],{"class":3192},[3175,4493,4494],{"class":3181}," javafx.fxml.FXMLLoader;\n",[3175,4496,4497,4499],{"class":3177,"line":3285},[3175,4498,3998],{"class":3192},[3175,4500,4001],{"class":3181},[3175,4502,4503,4505],{"class":3177,"line":3305},[3175,4504,3998],{"class":3192},[3175,4506,4507],{"class":3181}," javafx.scene.Scene;\n",[3175,4509,4510,4512],{"class":3177,"line":3326},[3175,4511,3998],{"class":3192},[3175,4513,4514],{"class":3181}," javafx.stage.Modality;\n",[3175,4516,4517,4519],{"class":3177,"line":3331},[3175,4518,3998],{"class":3192},[3175,4520,4521],{"class":3181}," javafx.stage.Stage;\n",[3175,4523,4524],{"class":3177,"line":3358},[3175,4525,3993],{"emptyLinePlaceholder":3992},[3175,4527,4528,4530],{"class":3177,"line":3376},[3175,4529,3998],{"class":3192},[3175,4531,4532],{"class":3181}," java.io.IOException;\n",[3175,4534,4535,4537],{"class":3177,"line":3397},[3175,4536,3998],{"class":3192},[3175,4538,4539],{"class":3181}," java.util.*;\n",[3175,4541,4542],{"class":3177,"line":3410},[3175,4543,3993],{"emptyLinePlaceholder":3992},[3175,4545,4546,4548],{"class":3177,"line":3416},[3175,4547,3182],{"class":3181},[3175,4549,4550],{"class":3185},"Singleton\n",[3175,4552,4553,4555,4558,4561,4564,4566],{"class":3177,"line":4095},[3175,4554,4028],{"class":3192},[3175,4556,4557],{"class":3192}," class",[3175,4559,4560],{"class":3185}," FxmlNavigator",[3175,4562,4563],{"class":3192}," implements",[3175,4565,4034],{"class":3185},[3175,4567,3213],{"class":3181},[3175,4569,4570],{"class":3177,"line":4100},[3175,4571,4041],{"class":3181},[3175,4573,4574,4577,4580,4583,4586],{"class":3177,"line":4105},[3175,4575,4576],{"class":3192},"    private",[3175,4578,4579],{"class":3192}," final",[3175,4581,4582],{"class":3185}," Injector",[3175,4584,4585],{"class":3222}," injector",[3175,4587,4588],{"class":3181},";\n",[3175,4590,4591,4593,4595,4598,4601],{"class":3177,"line":4111},[3175,4592,4576],{"class":3192},[3175,4594,4579],{"class":3192},[3175,4596,4597],{"class":3185}," ScreenRegistry",[3175,4599,4600],{"class":3222}," screenRegistry",[3175,4602,4588],{"class":3181},[3175,4604,4605,4607,4609,4612,4614,4617,4619,4622],{"class":3177,"line":4116},[3175,4606,4576],{"class":3192},[3175,4608,4579],{"class":3192},[3175,4610,4611],{"class":3185}," Stack",[3175,4613,4165],{"class":3181},[3175,4615,4616],{"class":3185},"NavigationEntry",[3175,4618,4175],{"class":3181},[3175,4620,4621],{"class":3222},"navigationStack",[3175,4623,4588],{"class":3181},[3175,4625,4626],{"class":3177,"line":4128},[3175,4627,4041],{"class":3181},[3175,4629,4630,4632,4634,4637],{"class":3177,"line":4141},[3175,4631,4576],{"class":3192},[3175,4633,3648],{"class":3185},[3175,4635,4636],{"class":3222}," primaryStage",[3175,4638,4588],{"class":3181},[3175,4640,4641],{"class":3177,"line":4146},[3175,4642,4041],{"class":3181},[3175,4644,4645,4648],{"class":3177,"line":4183},[3175,4646,4647],{"class":3181},"    @",[3175,4649,4650],{"class":3185},"Inject\n",[3175,4652,4653,4656,4658,4660,4663,4665,4667,4670,4672],{"class":3177,"line":4188},[3175,4654,4655],{"class":3192},"    public",[3175,4657,4560],{"class":3199},[3175,4659,3235],{"class":3181},[3175,4661,4662],{"class":3185},"Injector",[3175,4664,4585],{"class":3222},[3175,4666,4159],{"class":3181},[3175,4668,4669],{"class":3185},"ScreenRegistry",[3175,4671,4600],{"class":3222},[3175,4673,3394],{"class":3181},[3175,4675,4676,4679,4681,4684],{"class":3177,"line":4193},[3175,4677,4678],{"class":3192},"        this",[3175,4680,3270],{"class":3181},[3175,4682,4683],{"class":3222},"injector",[3175,4685,4686],{"class":3181}," = injector;\n",[3175,4688,4689,4691,4693,4696],{"class":3177,"line":4199},[3175,4690,4678],{"class":3192},[3175,4692,3270],{"class":3181},[3175,4694,4695],{"class":3222},"screenRegistry",[3175,4697,4698],{"class":3181}," = screenRegistry;\n",[3175,4700,4701,4703,4705,4707,4709,4711,4713],{"class":3177,"line":4204},[3175,4702,4678],{"class":3192},[3175,4704,3270],{"class":3181},[3175,4706,4621],{"class":3222},[3175,4708,3226],{"class":3181},[3175,4710,3229],{"class":3209},[3175,4712,4611],{"class":3185},[3175,4714,4715],{"class":3181},"\u003C>();\n",[3175,4717,4718],{"class":3177,"line":4216},[3175,4719,3413],{"class":3181},[3175,4721,4722],{"class":3177,"line":4227},[3175,4723,4041],{"class":3181},[3175,4725,4726],{"class":3177,"line":4232},[3175,4727,4046],{"class":3539},[3175,4729,4730],{"class":3177,"line":4263},[3175,4731,4732],{"class":3539},"     * Встановлення primary Stage (викликається з Application.start()).\n",[3175,4734,4735],{"class":3177,"line":4268},[3175,4736,4075],{"class":3539},[3175,4738,4739,4741,4743,4746,4748,4750,4752],{"class":3177,"line":4273},[3175,4740,4655],{"class":3192},[3175,4742,3196],{"class":3185},[3175,4744,4745],{"class":3199}," setPrimaryStage",[3175,4747,3235],{"class":3181},[3175,4749,3524],{"class":3185},[3175,4751,4636],{"class":3222},[3175,4753,3394],{"class":3181},[3175,4755,4756,4758,4760,4763],{"class":3177,"line":4279},[3175,4757,4678],{"class":3192},[3175,4759,3270],{"class":3181},[3175,4761,4762],{"class":3222},"primaryStage",[3175,4764,4765],{"class":3181}," = primaryStage;\n",[3175,4767,4768],{"class":3177,"line":4284},[3175,4769,3413],{"class":3181},[3175,4771,4772],{"class":3177,"line":4295},[3175,4773,4041],{"class":3181},[3175,4775,4776,4778],{"class":3177,"line":4306},[3175,4777,4647],{"class":3181},[3175,4779,4780],{"class":3185},"Override\n",[3175,4782,4783,4785,4787,4789,4791,4793,4795],{"class":3177,"line":4316},[3175,4784,4655],{"class":3192},[3175,4786,3196],{"class":3185},[3175,4788,4083],{"class":3199},[3175,4790,3235],{"class":3181},[3175,4792,4088],{"class":3185},[3175,4794,4067],{"class":3222},[3175,4796,3394],{"class":3181},[3175,4798,4799,4802,4805,4808,4810,4813],{"class":3177,"line":4321},[3175,4800,4801],{"class":3199},"        navigateTo",[3175,4803,4804],{"class":3181},"(screenId, ",[3175,4806,4807],{"class":3222},"Collections",[3175,4809,3270],{"class":3181},[3175,4811,4812],{"class":3199},"emptyMap",[3175,4814,4815],{"class":3181},"());\n",[3175,4817,4818],{"class":3177,"line":4364},[3175,4819,3413],{"class":3181},[3175,4821,4822],{"class":3177,"line":4369},[3175,4823,4041],{"class":3181},[3175,4825,4826,4828],{"class":3177,"line":4374},[3175,4827,4647],{"class":3181},[3175,4829,4780],{"class":3185},[3175,4831,4832,4834,4836,4838,4840,4842,4844,4846,4848,4850,4852,4854,4856,4858,4860],{"class":3177,"line":4380},[3175,4833,4655],{"class":3192},[3175,4835,3196],{"class":3185},[3175,4837,4083],{"class":3199},[3175,4839,3235],{"class":3181},[3175,4841,4088],{"class":3185},[3175,4843,4067],{"class":3222},[3175,4845,4159],{"class":3181},[3175,4847,4162],{"class":3185},[3175,4849,4165],{"class":3181},[3175,4851,4088],{"class":3185},[3175,4853,4159],{"class":3181},[3175,4855,4172],{"class":3185},[3175,4857,4175],{"class":3181},[3175,4859,4178],{"class":3222},[3175,4861,3394],{"class":3181},[3175,4863,4864,4867],{"class":3177,"line":4385},[3175,4865,4866],{"class":3209},"        try",[3175,4868,3213],{"class":3181},[3175,4870,4871],{"class":3177,"line":4395},[3175,4872,4873],{"class":3539},"            \u002F\u002F Завантаження FXML\n",[3175,4875,4876,4879,4882,4884,4886,4888,4891],{"class":3177,"line":4400},[3175,4877,4878],{"class":3185},"            String",[3175,4880,4881],{"class":3222}," fxmlPath",[3175,4883,3226],{"class":3181},[3175,4885,4695],{"class":3222},[3175,4887,3270],{"class":3181},[3175,4889,4890],{"class":3199},"getFxmlPath",[3175,4892,4893],{"class":3181},"(screenId);\n",[3175,4895,4896,4899,4901,4903,4905],{"class":3177,"line":4405},[3175,4897,4898],{"class":3185},"            Parent",[3175,4900,3262],{"class":3222},[3175,4902,3226],{"class":3181},[3175,4904,3553],{"class":3199},[3175,4906,4907],{"class":3181},"(fxmlPath);\n",[3175,4909,4910],{"class":3177,"line":4411},[3175,4911,4912],{"class":3181},"            \n",[3175,4914,4915],{"class":3177,"line":4416},[3175,4916,4917],{"class":3539},"            \u002F\u002F Збереження поточного екрану у стек\n",[3175,4919,4920,4923,4925,4927,4929,4931,4934,4937,4940,4942,4944,4946,4948,4951,4953,4955],{"class":3177,"line":4426},[3175,4921,4922],{"class":3209},"            if",[3175,4924,3385],{"class":3181},[3175,4926,4762],{"class":3222},[3175,4928,3270],{"class":3181},[3175,4930,3348],{"class":3199},[3175,4932,4933],{"class":3181},"() != ",[3175,4935,4936],{"class":3192},"null",[3175,4938,4939],{"class":3181}," && ",[3175,4941,4762],{"class":3222},[3175,4943,3270],{"class":3181},[3175,4945,3348],{"class":3199},[3175,4947,3241],{"class":3181},[3175,4949,4950],{"class":3199},"getRoot",[3175,4952,4933],{"class":3181},[3175,4954,4936],{"class":3192},[3175,4956,3394],{"class":3181},[3175,4958,4959,4962,4964,4967,4969,4971,4974],{"class":3177,"line":4431},[3175,4960,4961],{"class":3222},"                navigationStack",[3175,4963,3270],{"class":3181},[3175,4965,4966],{"class":3199},"push",[3175,4968,3235],{"class":3181},[3175,4970,3229],{"class":3209},[3175,4972,4973],{"class":3199}," NavigationEntry",[3175,4975,4976],{"class":3181},"(\n",[3175,4978,4979,4982],{"class":3177,"line":4442},[3175,4980,4981],{"class":3199},"                    getCurrentScreenId",[3175,4983,4984],{"class":3181},"(),\n",[3175,4986,4988,4991,4993,4995,4997,4999],{"class":3177,"line":4987},54,[3175,4989,4990],{"class":3222},"                    primaryStage",[3175,4992,3270],{"class":3181},[3175,4994,3348],{"class":3199},[3175,4996,3241],{"class":3181},[3175,4998,4950],{"class":3199},[3175,5000,5001],{"class":3181},"()\n",[3175,5003,5005],{"class":3177,"line":5004},55,[3175,5006,5007],{"class":3181},"                ));\n",[3175,5009,5011],{"class":3177,"line":5010},56,[3175,5012,5013],{"class":3181},"            }\n",[3175,5015,5017],{"class":3177,"line":5016},57,[3175,5018,4912],{"class":3181},[3175,5020,5022],{"class":3177,"line":5021},58,[3175,5023,5024],{"class":3539},"            \u002F\u002F Зміна Root Node\n",[3175,5026,5028,5030,5032,5034,5036,5038,5041,5043],{"class":3177,"line":5027},59,[3175,5029,4922],{"class":3209},[3175,5031,3385],{"class":3181},[3175,5033,4762],{"class":3222},[3175,5035,3270],{"class":3181},[3175,5037,3348],{"class":3199},[3175,5039,5040],{"class":3181},"() == ",[3175,5042,4936],{"class":3192},[3175,5044,3394],{"class":3181},[3175,5046,5048,5051,5053,5055,5057,5059,5061],{"class":3177,"line":5047},60,[3175,5049,5050],{"class":3222},"                primaryStage",[3175,5052,3270],{"class":3181},[3175,5054,3660],{"class":3199},[3175,5056,3235],{"class":3181},[3175,5058,3229],{"class":3209},[3175,5060,3667],{"class":3199},[3175,5062,5063],{"class":3181},"(root));\n",[3175,5065,5067,5070,5073],{"class":3177,"line":5066},61,[3175,5068,5069],{"class":3181},"            } ",[3175,5071,5072],{"class":3209},"else",[3175,5074,3213],{"class":3181},[3175,5076,5078,5080,5082,5084,5086,5088],{"class":3177,"line":5077},62,[3175,5079,5050],{"class":3222},[3175,5081,3270],{"class":3181},[3175,5083,3348],{"class":3199},[3175,5085,3241],{"class":3181},[3175,5087,3370],{"class":3199},[3175,5089,3373],{"class":3181},[3175,5091,5093],{"class":3177,"line":5092},63,[3175,5094,5013],{"class":3181},[3175,5096,5098],{"class":3177,"line":5097},64,[3175,5099,4912],{"class":3181},[3175,5101,5103],{"class":3177,"line":5102},65,[3175,5104,5105],{"class":3539},"            \u002F\u002F Передача параметрів у ViewModel (якщо потрібно)\n",[3175,5107,5109],{"class":3177,"line":5108},66,[3175,5110,5111],{"class":3539},"            \u002F\u002F TODO: реалізувати передачу параметрів\n",[3175,5113,5115],{"class":3177,"line":5114},67,[3175,5116,4912],{"class":3181},[3175,5118,5120,5123,5125,5127,5129,5131],{"class":3177,"line":5119},68,[3175,5121,5122],{"class":3181},"        } ",[3175,5124,3382],{"class":3209},[3175,5126,3385],{"class":3181},[3175,5128,3388],{"class":3185},[3175,5130,3391],{"class":3222},[3175,5132,3394],{"class":3181},[3175,5134,5136,5139,5142,5145,5147,5150],{"class":3177,"line":5135},69,[3175,5137,5138],{"class":3209},"            throw",[3175,5140,5141],{"class":3209}," new",[3175,5143,5144],{"class":3199}," NavigationException",[3175,5146,3235],{"class":3181},[3175,5148,5149],{"class":3249},"\"Failed to navigate to screen: \"",[3175,5151,5152],{"class":3181}," + screenId, e);\n",[3175,5154,5156],{"class":3177,"line":5155},70,[3175,5157,5158],{"class":3181},"        }\n",[3175,5160,5162],{"class":3177,"line":5161},71,[3175,5163,3413],{"class":3181},[3175,5165,5167],{"class":3177,"line":5166},72,[3175,5168,4041],{"class":3181},[3175,5170,5172,5174],{"class":3177,"line":5171},73,[3175,5173,4647],{"class":3181},[3175,5175,4780],{"class":3185},[3175,5177,5179,5181,5184,5186,5188,5190,5192,5194,5196,5198,5200,5202,5204],{"class":3177,"line":5178},74,[3175,5180,4655],{"class":3192},[3175,5182,5183],{"class":3181}," \u003C",[3175,5185,4238],{"class":3185},[3175,5187,4175],{"class":3181},[3175,5189,4243],{"class":3185},[3175,5191,4165],{"class":3181},[3175,5193,4238],{"class":3185},[3175,5195,4175],{"class":3181},[3175,5197,4252],{"class":3199},[3175,5199,3235],{"class":3181},[3175,5201,4088],{"class":3185},[3175,5203,4067],{"class":3222},[3175,5205,3394],{"class":3181},[3175,5207,5209,5212,5215,5217,5219,5221,5223],{"class":3177,"line":5208},75,[3175,5210,5211],{"class":3209},"        return",[3175,5213,5214],{"class":3199}," openDialog",[3175,5216,4804],{"class":3181},[3175,5218,4807],{"class":3222},[3175,5220,3270],{"class":3181},[3175,5222,4812],{"class":3199},[3175,5224,4815],{"class":3181},[3175,5226,5228],{"class":3177,"line":5227},76,[3175,5229,3413],{"class":3181},[3175,5231,5233],{"class":3177,"line":5232},77,[3175,5234,4041],{"class":3181},[3175,5236,5238,5240],{"class":3177,"line":5237},78,[3175,5239,4647],{"class":3181},[3175,5241,4780],{"class":3185},[3175,5243,5245,5247,5249,5251,5253,5255,5257,5259,5261,5263,5265,5267,5269,5271,5273,5275,5277,5279,5281,5283,5285],{"class":3177,"line":5244},79,[3175,5246,4655],{"class":3192},[3175,5248,5183],{"class":3181},[3175,5250,4238],{"class":3185},[3175,5252,4175],{"class":3181},[3175,5254,4243],{"class":3185},[3175,5256,4165],{"class":3181},[3175,5258,4238],{"class":3185},[3175,5260,4175],{"class":3181},[3175,5262,4252],{"class":3199},[3175,5264,3235],{"class":3181},[3175,5266,4088],{"class":3185},[3175,5268,4067],{"class":3222},[3175,5270,4159],{"class":3181},[3175,5272,4162],{"class":3185},[3175,5274,4165],{"class":3181},[3175,5276,4088],{"class":3185},[3175,5278,4159],{"class":3181},[3175,5280,4172],{"class":3185},[3175,5282,4175],{"class":3181},[3175,5284,4178],{"class":3222},[3175,5286,3394],{"class":3181},[3175,5288,5290,5292],{"class":3177,"line":5289},80,[3175,5291,4866],{"class":3209},[3175,5293,3213],{"class":3181},[3175,5295,5297],{"class":3177,"line":5296},81,[3175,5298,4873],{"class":3539},[3175,5300,5302,5304,5306,5308,5310,5312,5314],{"class":3177,"line":5301},82,[3175,5303,4878],{"class":3185},[3175,5305,4881],{"class":3222},[3175,5307,3226],{"class":3181},[3175,5309,4695],{"class":3222},[3175,5311,3270],{"class":3181},[3175,5313,4890],{"class":3199},[3175,5315,4893],{"class":3181},[3175,5317,5319,5322,5324,5326,5328,5330,5332,5334,5336,5338],{"class":3177,"line":5318},83,[3175,5320,5321],{"class":3185},"            FXMLLoader",[3175,5323,3223],{"class":3222},[3175,5325,3226],{"class":3181},[3175,5327,3229],{"class":3209},[3175,5329,3232],{"class":3199},[3175,5331,3235],{"class":3181},[3175,5333,3238],{"class":3199},[3175,5335,3241],{"class":3181},[3175,5337,3244],{"class":3199},[3175,5339,5340],{"class":3181},"(fxmlPath));\n",[3175,5342,5344,5347,5349,5352,5355,5358],{"class":3177,"line":5343},84,[3175,5345,5346],{"class":3222},"            loader",[3175,5348,3270],{"class":3181},[3175,5350,5351],{"class":3199},"setControllerFactory",[3175,5353,5354],{"class":3181},"(injector",[3175,5356,5357],{"class":3209},"::",[3175,5359,5360],{"class":3181},"getInstance);\n",[3175,5362,5364,5366,5368,5370,5372,5374,5376],{"class":3177,"line":5363},85,[3175,5365,4898],{"class":3185},[3175,5367,3262],{"class":3222},[3175,5369,3226],{"class":3181},[3175,5371,3267],{"class":3222},[3175,5373,3270],{"class":3181},[3175,5375,3273],{"class":3199},[3175,5377,3276],{"class":3181},[3175,5379,5381],{"class":3177,"line":5380},86,[3175,5382,4912],{"class":3181},[3175,5384,5386],{"class":3177,"line":5385},87,[3175,5387,5388],{"class":3539},"            \u002F\u002F Створення модального Stage\n",[3175,5390,5392,5395,5398,5400,5402,5404],{"class":3177,"line":5391},88,[3175,5393,5394],{"class":3185},"            Stage",[3175,5396,5397],{"class":3222}," dialogStage",[3175,5399,3226],{"class":3181},[3175,5401,3229],{"class":3209},[3175,5403,3648],{"class":3199},[3175,5405,3276],{"class":3181},[3175,5407,5409,5412,5414,5417,5419,5422,5424,5427],{"class":3177,"line":5408},89,[3175,5410,5411],{"class":3222},"            dialogStage",[3175,5413,3270],{"class":3181},[3175,5415,5416],{"class":3199},"initModality",[3175,5418,3235],{"class":3181},[3175,5420,5421],{"class":3222},"Modality",[3175,5423,3270],{"class":3181},[3175,5425,5426],{"class":3222},"APPLICATION_MODAL",[3175,5428,3561],{"class":3181},[3175,5430,5432,5434,5436,5439],{"class":3177,"line":5431},90,[3175,5433,5411],{"class":3222},[3175,5435,3270],{"class":3181},[3175,5437,5438],{"class":3199},"initOwner",[3175,5440,5441],{"class":3181},"(primaryStage);\n",[3175,5443,5445,5447,5449,5451,5453,5455,5457],{"class":3177,"line":5444},91,[3175,5446,5411],{"class":3222},[3175,5448,3270],{"class":3181},[3175,5450,3660],{"class":3199},[3175,5452,3235],{"class":3181},[3175,5454,3229],{"class":3209},[3175,5456,3667],{"class":3199},[3175,5458,5063],{"class":3181},[3175,5460,5462],{"class":3177,"line":5461},92,[3175,5463,4912],{"class":3181},[3175,5465,5467],{"class":3177,"line":5466},93,[3175,5468,5469],{"class":3539},"            \u002F\u002F Показ діалогу (блокує виконання до закриття)\n",[3175,5471,5473,5475,5477,5480],{"class":3177,"line":5472},94,[3175,5474,5411],{"class":3222},[3175,5476,3270],{"class":3181},[3175,5478,5479],{"class":3199},"showAndWait",[3175,5481,3276],{"class":3181},[3175,5483,5485],{"class":3177,"line":5484},95,[3175,5486,4912],{"class":3181},[3175,5488,5490],{"class":3177,"line":5489},96,[3175,5491,5492],{"class":3539},"            \u002F\u002F Отримання результату з Controller\n",[3175,5494,5496,5499,5501,5503,5505,5507,5509],{"class":3177,"line":5495},97,[3175,5497,5498],{"class":3185},"            Object",[3175,5500,3291],{"class":3222},[3175,5502,3226],{"class":3181},[3175,5504,3267],{"class":3222},[3175,5506,3270],{"class":3181},[3175,5508,3300],{"class":3199},[3175,5510,3276],{"class":3181},[3175,5512,5514,5516,5519,5522],{"class":3177,"line":5513},98,[3175,5515,4922],{"class":3209},[3175,5517,5518],{"class":3181}," (controller ",[3175,5520,5521],{"class":3192},"instanceof",[3175,5523,5524],{"class":3181}," DialogController) {\n",[3175,5526,5528,5531,5534,5536,5539,5542,5545,5548,5551],{"class":3177,"line":5527},99,[3175,5529,5530],{"class":3209},"                return",[3175,5532,5533],{"class":3222}," Optional",[3175,5535,3270],{"class":3181},[3175,5537,5538],{"class":3199},"ofNullable",[3175,5540,5541],{"class":3181},"(((",[3175,5543,5544],{"class":3185},"DialogController",[3175,5546,5547],{"class":3181},"\u003CT>) controller).",[3175,5549,5550],{"class":3199},"getResult",[3175,5552,4815],{"class":3181},[3175,5554,5556],{"class":3177,"line":5555},100,[3175,5557,5013],{"class":3181},[3175,5559,5561],{"class":3177,"line":5560},101,[3175,5562,4912],{"class":3181},[3175,5564,5566,5569,5571,5573,5576],{"class":3177,"line":5565},102,[3175,5567,5568],{"class":3209},"            return",[3175,5570,5533],{"class":3222},[3175,5572,3270],{"class":3181},[3175,5574,5575],{"class":3199},"empty",[3175,5577,3276],{"class":3181},[3175,5579,5581],{"class":3177,"line":5580},103,[3175,5582,4912],{"class":3181},[3175,5584,5586,5588,5590,5592,5594,5596],{"class":3177,"line":5585},104,[3175,5587,5122],{"class":3181},[3175,5589,3382],{"class":3209},[3175,5591,3385],{"class":3181},[3175,5593,3388],{"class":3185},[3175,5595,3391],{"class":3222},[3175,5597,3394],{"class":3181},[3175,5599,5601,5603,5605,5607,5609,5612],{"class":3177,"line":5600},105,[3175,5602,5138],{"class":3209},[3175,5604,5141],{"class":3209},[3175,5606,5144],{"class":3199},[3175,5608,3235],{"class":3181},[3175,5610,5611],{"class":3249},"\"Failed to open dialog: \"",[3175,5613,5152],{"class":3181},[3175,5615,5617],{"class":3177,"line":5616},106,[3175,5618,5158],{"class":3181},[3175,5620,5622],{"class":3177,"line":5621},107,[3175,5623,3413],{"class":3181},[3175,5625,5627],{"class":3177,"line":5626},108,[3175,5628,4041],{"class":3181},[3175,5630,5632,5634],{"class":3177,"line":5631},109,[3175,5633,4647],{"class":3181},[3175,5635,4780],{"class":3185},[3175,5637,5639,5641,5643,5645],{"class":3177,"line":5638},110,[3175,5640,4655],{"class":3192},[3175,5642,3196],{"class":3185},[3175,5644,4390],{"class":3199},[3175,5646,3203],{"class":3181},[3175,5648,5650,5653,5656,5659],{"class":3177,"line":5649},111,[3175,5651,5652],{"class":3209},"        if",[3175,5654,5655],{"class":3181}," (!",[3175,5657,5658],{"class":3199},"canGoBack",[3175,5660,5661],{"class":3181},"()) {\n",[3175,5663,5665,5667,5669,5672,5674,5677],{"class":3177,"line":5664},112,[3175,5666,5138],{"class":3209},[3175,5668,5141],{"class":3209},[3175,5670,5671],{"class":3199}," IllegalStateException",[3175,5673,3235],{"class":3181},[3175,5675,5676],{"class":3249},"\"Cannot go back: navigation stack is empty\"",[3175,5678,3561],{"class":3181},[3175,5680,5682],{"class":3177,"line":5681},113,[3175,5683,5158],{"class":3181},[3175,5685,5687],{"class":3177,"line":5686},114,[3175,5688,3282],{"class":3181},[3175,5690,5692,5695,5698,5700,5702,5704,5707],{"class":3177,"line":5691},115,[3175,5693,5694],{"class":3185},"        NavigationEntry",[3175,5696,5697],{"class":3222}," previousEntry",[3175,5699,3226],{"class":3181},[3175,5701,4621],{"class":3222},[3175,5703,3270],{"class":3181},[3175,5705,5706],{"class":3199},"pop",[3175,5708,3276],{"class":3181},[3175,5710,5712,5715,5717,5719,5721,5723,5725,5728,5730,5732],{"class":3177,"line":5711},116,[3175,5713,5714],{"class":3222},"        primaryStage",[3175,5716,3270],{"class":3181},[3175,5718,3348],{"class":3199},[3175,5720,3241],{"class":3181},[3175,5722,3370],{"class":3199},[3175,5724,3235],{"class":3181},[3175,5726,5727],{"class":3222},"previousEntry",[3175,5729,3270],{"class":3181},[3175,5731,4950],{"class":3199},[3175,5733,4815],{"class":3181},[3175,5735,5737],{"class":3177,"line":5736},117,[3175,5738,3413],{"class":3181},[3175,5740,5742],{"class":3177,"line":5741},118,[3175,5743,4041],{"class":3181},[3175,5745,5747,5749],{"class":3177,"line":5746},119,[3175,5748,4647],{"class":3181},[3175,5750,4780],{"class":3185},[3175,5752,5754,5756,5759,5761],{"class":3177,"line":5753},120,[3175,5755,4655],{"class":3192},[3175,5757,5758],{"class":3185}," boolean",[3175,5760,4437],{"class":3199},[3175,5762,3203],{"class":3181},[3175,5764,5766,5768,5771,5773,5775,5778],{"class":3177,"line":5765},121,[3175,5767,5211],{"class":3209},[3175,5769,5770],{"class":3181}," !",[3175,5772,4621],{"class":3222},[3175,5774,3270],{"class":3181},[3175,5776,5777],{"class":3199},"isEmpty",[3175,5779,3276],{"class":3181},[3175,5781,5783],{"class":3177,"line":5782},122,[3175,5784,3413],{"class":3181},[3175,5786,5788],{"class":3177,"line":5787},123,[3175,5789,4041],{"class":3181},[3175,5791,5793,5795,5798,5801,5803,5805,5807,5810,5813,5816],{"class":3177,"line":5792},124,[3175,5794,4576],{"class":3192},[3175,5796,5797],{"class":3185}," Parent",[3175,5799,5800],{"class":3199}," loadFxml",[3175,5802,3235],{"class":3181},[3175,5804,4088],{"class":3185},[3175,5806,4881],{"class":3222},[3175,5808,5809],{"class":3181},") ",[3175,5811,5812],{"class":3192},"throws",[3175,5814,5815],{"class":3185}," IOException",[3175,5817,3213],{"class":3181},[3175,5819,5821,5823,5825,5827,5829,5831,5833,5835,5837,5839],{"class":3177,"line":5820},125,[3175,5822,3219],{"class":3185},[3175,5824,3223],{"class":3222},[3175,5826,3226],{"class":3181},[3175,5828,3229],{"class":3209},[3175,5830,3232],{"class":3199},[3175,5832,3235],{"class":3181},[3175,5834,3238],{"class":3199},[3175,5836,3241],{"class":3181},[3175,5838,3244],{"class":3199},[3175,5840,5340],{"class":3181},[3175,5842,5844,5847,5849,5851,5853,5855],{"class":3177,"line":5843},126,[3175,5845,5846],{"class":3222},"        loader",[3175,5848,3270],{"class":3181},[3175,5850,5351],{"class":3199},[3175,5852,5354],{"class":3181},[3175,5854,5357],{"class":3209},[3175,5856,5360],{"class":3181},[3175,5858,5860,5862,5864,5866,5868],{"class":3177,"line":5859},127,[3175,5861,5211],{"class":3209},[3175,5863,3223],{"class":3222},[3175,5865,3270],{"class":3181},[3175,5867,3273],{"class":3199},[3175,5869,3276],{"class":3181},[3175,5871,5873],{"class":3177,"line":5872},128,[3175,5874,3413],{"class":3181},[3175,5876,5878],{"class":3177,"line":5877},129,[3175,5879,4041],{"class":3181},[3175,5881,5883,5885,5888,5891],{"class":3177,"line":5882},130,[3175,5884,4576],{"class":3192},[3175,5886,5887],{"class":3185}," String",[3175,5889,5890],{"class":3199}," getCurrentScreenId",[3175,5892,3203],{"class":3181},[3175,5894,5896],{"class":3177,"line":5895},131,[3175,5897,5898],{"class":3539},"        \u002F\u002F TODO: зберігати поточний screenId при навігації\n",[3175,5900,5902,5904,5907],{"class":3177,"line":5901},132,[3175,5903,5211],{"class":3209},[3175,5905,5906],{"class":3249}," \"unknown\"",[3175,5908,4588],{"class":3181},[3175,5910,5912],{"class":3177,"line":5911},133,[3175,5913,3413],{"class":3181},[3175,5915,5917],{"class":3177,"line":5916},134,[3175,5918,4041],{"class":3181},[3175,5920,5922],{"class":3177,"line":5921},135,[3175,5923,4046],{"class":3539},[3175,5925,5927],{"class":3177,"line":5926},136,[3175,5928,5929],{"class":3539},"     * Запис у стеку навігації.\n",[3175,5931,5933],{"class":3177,"line":5932},137,[3175,5934,4075],{"class":3539},[3175,5936,5938,5940,5943,5945,5947],{"class":3177,"line":5937},138,[3175,5939,4576],{"class":3192},[3175,5941,5942],{"class":3192}," static",[3175,5944,4557],{"class":3192},[3175,5946,4973],{"class":3185},[3175,5948,3213],{"class":3181},[3175,5950,5952,5955,5957,5959,5961],{"class":3177,"line":5951},139,[3175,5953,5954],{"class":3192},"        private",[3175,5956,4579],{"class":3192},[3175,5958,5887],{"class":3185},[3175,5960,4067],{"class":3222},[3175,5962,4588],{"class":3181},[3175,5964,5966,5968,5970,5972,5974],{"class":3177,"line":5965},140,[3175,5967,5954],{"class":3192},[3175,5969,4579],{"class":3192},[3175,5971,5797],{"class":3185},[3175,5973,3262],{"class":3222},[3175,5975,4588],{"class":3181},[3175,5977,5979],{"class":3177,"line":5978},141,[3175,5980,3282],{"class":3181},[3175,5982,5984,5987,5989,5991,5993,5995,5997,5999,6001],{"class":3177,"line":5983},142,[3175,5985,5986],{"class":3192},"        public",[3175,5988,4973],{"class":3199},[3175,5990,3235],{"class":3181},[3175,5992,4088],{"class":3185},[3175,5994,4067],{"class":3222},[3175,5996,4159],{"class":3181},[3175,5998,3545],{"class":3185},[3175,6000,3262],{"class":3222},[3175,6002,3394],{"class":3181},[3175,6004,6006,6009,6011,6014],{"class":3177,"line":6005},143,[3175,6007,6008],{"class":3192},"            this",[3175,6010,3270],{"class":3181},[3175,6012,6013],{"class":3222},"screenId",[3175,6015,6016],{"class":3181}," = screenId;\n",[3175,6018,6020,6022,6024,6026],{"class":3177,"line":6019},144,[3175,6021,6008],{"class":3192},[3175,6023,3270],{"class":3181},[3175,6025,3765],{"class":3222},[3175,6027,6028],{"class":3181}," = root;\n",[3175,6030,6032],{"class":3177,"line":6031},145,[3175,6033,5158],{"class":3181},[3175,6035,6037],{"class":3177,"line":6036},146,[3175,6038,3282],{"class":3181},[3175,6040,6042,6044,6046,6049],{"class":3177,"line":6041},147,[3175,6043,5986],{"class":3192},[3175,6045,5887],{"class":3185},[3175,6047,6048],{"class":3199}," getScreenId",[3175,6050,3203],{"class":3181},[3175,6052,6054,6056],{"class":3177,"line":6053},148,[3175,6055,5568],{"class":3209},[3175,6057,6058],{"class":3181}," screenId;\n",[3175,6060,6062],{"class":3177,"line":6061},149,[3175,6063,5158],{"class":3181},[3175,6065,6067],{"class":3177,"line":6066},150,[3175,6068,3282],{"class":3181},[3175,6070,6072,6074,6076,6079],{"class":3177,"line":6071},151,[3175,6073,5986],{"class":3192},[3175,6075,5797],{"class":3185},[3175,6077,6078],{"class":3199}," getRoot",[3175,6080,3203],{"class":3181},[3175,6082,6084,6086],{"class":3177,"line":6083},152,[3175,6085,5568],{"class":3209},[3175,6087,6088],{"class":3181}," root;\n",[3175,6090,6092],{"class":3177,"line":6091},153,[3175,6093,5158],{"class":3181},[3175,6095,6097],{"class":3177,"line":6096},154,[3175,6098,3413],{"class":3181},[3175,6100,6102],{"class":3177,"line":6101},155,[3175,6103,3419],{"class":3181},[3512,6105,6107],{"id":6106},"розбір-реалізації","Розбір реалізації",[3150,6109,6110,3442,6113,6115,6116,6118],{},[3157,6111,6112],{},"Рядки 20-21: Залежності.",[3172,6114,4662],{}," для створення Controllers через Guice. ",[3172,6117,4669],{}," для отримання шляхів до FXML за ідентифікатором екрану.",[3150,6120,6121,3442,6124,6127,6128,6130,6131,6133],{},[3157,6122,6123],{},"Рядок 22: Navigation Stack.",[3172,6125,6126],{},"Stack\u003CNavigationEntry>"," зберігає історію навігації. Кожен запис містить ",[3172,6129,6013],{}," та ",[3172,6132,3545],{}," (Root Node екрану).",[3150,6135,6136,6139,6140,6143,6144,3270],{},[3157,6137,6138],{},"Рядок 24: Primary Stage."," Посилання на головне вікно додатку. Встановлюється з ",[3172,6141,6142],{},"Application.start()"," через ",[3172,6145,6146],{},"setPrimaryStage()",[3150,6148,6149,6152,6153,6156],{},[3157,6150,6151],{},"Рядки 44-64: navigateTo()."," Завантажує FXML через ",[3172,6154,6155],{},"loadFxml()",", зберігає поточний екран у стек, змінює Root Node у Scene. Якщо Scene ще не створена (перший запуск), створює нову Scene.",[3150,6158,6159,6162,6163,6165,6166,6168,6169,6172,6173,3270],{},[3157,6160,6161],{},"Рядки 76-103: openDialog()."," Завантажує FXML, створює новий ",[3172,6164,3524],{}," з модальністю ",[3172,6167,5426],{}," (блокує головне вікно), показує діалог через ",[3172,6170,6171],{},"showAndWait()"," (блокує виконання до закриття), отримує результат з Controller через інтерфейс ",[3172,6174,5544],{},[3150,6176,6177,6180],{},[3157,6178,6179],{},"Рядки 107-114: goBack()."," Витягує попередній екран зі стеку та встановлює його як Root Node. Якщо стек порожній, викидає виняток.",[3150,6182,6183,6186,6187,6190,6191,6194],{},[3157,6184,6185],{},"Рядки 382-386: loadFxml()."," Створює ",[3172,6188,6189],{},"FXMLLoader",", встановлює ",[3172,6192,6193],{},"ControllerFactory"," для інтеграції з Guice, завантажує FXML та повертає Root Node.",[3150,6196,6197,6200,6201,6203,6204,6133],{},[3157,6198,6199],{},"Рядки 396-412: NavigationEntry."," Внутрішній клас для зберігання запису у стеку навігації. Містить ",[3172,6202,6013],{}," (для відладки) та ",[3172,6205,3545],{},[3495,6207,6208,6211,6212,6215],{},[3157,6209,6210],{},"Чому зберігати Parent у стеку?"," Коли користувач повертається назад, ми відновлюємо попередній екран ",[3157,6213,6214],{},"зі збереженим станом"," (прокрутка, вибір, введені дані). Якщо б ми перезавантажували FXML, стан втрачався б. Але це має недолік: всі екрани у стеку залишаються у пам'яті. Для додатків з багатьма екранами розгляньте обмеження розміру стеку або очищення старих екранів.",[3502,6217],{},[3145,6219,6221],{"id":6220},"screenregistry-реєстрація-екранів","ScreenRegistry: Реєстрація екранів",[3150,6223,6224,6225,6228,6229,6231],{},"Замість magic strings (",[3172,6226,6227],{},"\"audiobook-list-view.fxml\"",") створимо ",[3172,6230,4669],{}," — централізований реєстр екранів.",[3512,6233,6235],{"id":6234},"реалізація-screenregistry","Реалізація ScreenRegistry",[3165,6237,6239],{"className":3167,"code":6238,"language":3169,"meta":3170,"style":3170},"package dev.kostyl.audiobook.infrastructure.navigation;\n\nimport com.google.inject.Singleton;\n\nimport java.util.HashMap;\nimport java.util.Map;\n\n@Singleton\npublic class ScreenRegistry {\n    \n    private final Map\u003CString, ScreenDefinition> screens = new HashMap\u003C>();\n    \n    public ScreenRegistry() {\n        registerScreens();\n    }\n    \n    private void registerScreens() {\n        \u002F\u002F Основні екрани\n        register(\"audiobook-list\", \"\u002Ffxml\u002Faudiobook-list-view.fxml\", \"Audiobook Platform\");\n        register(\"audiobook-form\", \"\u002Ffxml\u002Faudiobook-form-view.fxml\", \"Add Audiobook\");\n        register(\"audiobook-detail\", \"\u002Ffxml\u002Faudiobook-detail-view.fxml\", \"Audiobook Details\");\n        \n        register(\"author-list\", \"\u002Ffxml\u002Fauthor-list-view.fxml\", \"Authors\");\n        register(\"author-form\", \"\u002Ffxml\u002Fauthor-form-view.fxml\", \"Add Author\");\n        \n        register(\"genre-list\", \"\u002Ffxml\u002Fgenre-list-view.fxml\", \"Genres\");\n        \n        \u002F\u002F Dialogs\n        register(\"confirm-delete\", \"\u002Ffxml\u002Fdialogs\u002Fconfirm-delete-dialog.fxml\", \"Confirm Delete\");\n        register(\"select-author\", \"\u002Ffxml\u002Fdialogs\u002Fselect-author-dialog.fxml\", \"Select Author\");\n    }\n    \n    private void register(String screenId, String fxmlPath, String title) {\n        screens.put(screenId, new ScreenDefinition(screenId, fxmlPath, title));\n    }\n    \n    public String getFxmlPath(String screenId) {\n        ScreenDefinition screen = screens.get(screenId);\n        if (screen == null) {\n            throw new IllegalArgumentException(\"Screen not found: \" + screenId);\n        }\n        return screen.getFxmlPath();\n    }\n    \n    public String getTitle(String screenId) {\n        ScreenDefinition screen = screens.get(screenId);\n        if (screen == null) {\n            throw new IllegalArgumentException(\"Screen not found: \" + screenId);\n        }\n        return screen.getTitle();\n    }\n    \n    public boolean isRegistered(String screenId) {\n        return screens.containsKey(screenId);\n    }\n    \n    \u002F**\n     * Визначення екрану.\n     *\u002F\n    private static class ScreenDefinition {\n        private final String screenId;\n        private final String fxmlPath;\n        private final String title;\n        \n        public ScreenDefinition(String screenId, String fxmlPath, String title) {\n            this.screenId = screenId;\n            this.fxmlPath = fxmlPath;\n            this.title = title;\n        }\n        \n        public String getScreenId() {\n            return screenId;\n        }\n        \n        public String getFxmlPath() {\n            return fxmlPath;\n        }\n        \n        public String getTitle() {\n            return title;\n        }\n    }\n}\n",[3172,6240,6241,6247,6251,6257,6261,6268,6274,6278,6284,6294,6298,6330,6334,6342,6349,6353,6357,6368,6373,6395,6416,6437,6441,6462,6483,6487,6508,6512,6517,6538,6559,6563,6567,6597,6617,6621,6625,6642,6661,6672,6689,6693,6705,6709,6713,6730,6746,6756,6770,6774,6787,6791,6795,6812,6826,6830,6834,6838,6843,6847,6859,6871,6883,6895,6899,6925,6935,6947,6959,6963,6967,6977,6983,6987,6991,7001,7008,7012,7016,7026,7033,7037,7041],{"__ignoreMap":3170},[3175,6242,6243,6245],{"class":3177,"line":3178},[3175,6244,3984],{"class":3192},[3175,6246,3987],{"class":3181},[3175,6248,6249],{"class":3177,"line":3189},[3175,6250,3993],{"emptyLinePlaceholder":3992},[3175,6252,6253,6255],{"class":3177,"line":3206},[3175,6254,3998],{"class":3192},[3175,6256,4487],{"class":3181},[3175,6258,6259],{"class":3177,"line":3216},[3175,6260,3993],{"emptyLinePlaceholder":3992},[3175,6262,6263,6265],{"class":3177,"line":3256},[3175,6264,3998],{"class":3192},[3175,6266,6267],{"class":3181}," java.util.HashMap;\n",[3175,6269,6270,6272],{"class":3177,"line":3279},[3175,6271,3998],{"class":3192},[3175,6273,4012],{"class":3181},[3175,6275,6276],{"class":3177,"line":3285},[3175,6277,3993],{"emptyLinePlaceholder":3992},[3175,6279,6280,6282],{"class":3177,"line":3305},[3175,6281,3182],{"class":3181},[3175,6283,4550],{"class":3185},[3175,6285,6286,6288,6290,6292],{"class":3177,"line":3326},[3175,6287,4028],{"class":3192},[3175,6289,4557],{"class":3192},[3175,6291,4597],{"class":3185},[3175,6293,3213],{"class":3181},[3175,6295,6296],{"class":3177,"line":3331},[3175,6297,4041],{"class":3181},[3175,6299,6300,6302,6304,6307,6309,6311,6313,6316,6318,6321,6323,6325,6328],{"class":3177,"line":3358},[3175,6301,4576],{"class":3192},[3175,6303,4579],{"class":3192},[3175,6305,6306],{"class":3185}," Map",[3175,6308,4165],{"class":3181},[3175,6310,4088],{"class":3185},[3175,6312,4159],{"class":3181},[3175,6314,6315],{"class":3185},"ScreenDefinition",[3175,6317,4175],{"class":3181},[3175,6319,6320],{"class":3222},"screens",[3175,6322,3226],{"class":3181},[3175,6324,3229],{"class":3209},[3175,6326,6327],{"class":3185}," HashMap",[3175,6329,4715],{"class":3181},[3175,6331,6332],{"class":3177,"line":3376},[3175,6333,4041],{"class":3181},[3175,6335,6336,6338,6340],{"class":3177,"line":3397},[3175,6337,4655],{"class":3192},[3175,6339,4597],{"class":3199},[3175,6341,3203],{"class":3181},[3175,6343,6344,6347],{"class":3177,"line":3410},[3175,6345,6346],{"class":3199},"        registerScreens",[3175,6348,3276],{"class":3181},[3175,6350,6351],{"class":3177,"line":3416},[3175,6352,3413],{"class":3181},[3175,6354,6355],{"class":3177,"line":4095},[3175,6356,4041],{"class":3181},[3175,6358,6359,6361,6363,6366],{"class":3177,"line":4100},[3175,6360,4576],{"class":3192},[3175,6362,3196],{"class":3185},[3175,6364,6365],{"class":3199}," registerScreens",[3175,6367,3203],{"class":3181},[3175,6369,6370],{"class":3177,"line":4105},[3175,6371,6372],{"class":3539},"        \u002F\u002F Основні екрани\n",[3175,6374,6375,6378,6380,6383,6385,6388,6390,6393],{"class":3177,"line":4111},[3175,6376,6377],{"class":3199},"        register",[3175,6379,3235],{"class":3181},[3175,6381,6382],{"class":3249},"\"audiobook-list\"",[3175,6384,4159],{"class":3181},[3175,6386,6387],{"class":3249},"\"\u002Ffxml\u002Faudiobook-list-view.fxml\"",[3175,6389,4159],{"class":3181},[3175,6391,6392],{"class":3249},"\"Audiobook Platform\"",[3175,6394,3561],{"class":3181},[3175,6396,6397,6399,6401,6404,6406,6409,6411,6414],{"class":3177,"line":4116},[3175,6398,6377],{"class":3199},[3175,6400,3235],{"class":3181},[3175,6402,6403],{"class":3249},"\"audiobook-form\"",[3175,6405,4159],{"class":3181},[3175,6407,6408],{"class":3249},"\"\u002Ffxml\u002Faudiobook-form-view.fxml\"",[3175,6410,4159],{"class":3181},[3175,6412,6413],{"class":3249},"\"Add Audiobook\"",[3175,6415,3561],{"class":3181},[3175,6417,6418,6420,6422,6425,6427,6430,6432,6435],{"class":3177,"line":4128},[3175,6419,6377],{"class":3199},[3175,6421,3235],{"class":3181},[3175,6423,6424],{"class":3249},"\"audiobook-detail\"",[3175,6426,4159],{"class":3181},[3175,6428,6429],{"class":3249},"\"\u002Ffxml\u002Faudiobook-detail-view.fxml\"",[3175,6431,4159],{"class":3181},[3175,6433,6434],{"class":3249},"\"Audiobook Details\"",[3175,6436,3561],{"class":3181},[3175,6438,6439],{"class":3177,"line":4141},[3175,6440,3282],{"class":3181},[3175,6442,6443,6445,6447,6450,6452,6455,6457,6460],{"class":3177,"line":4146},[3175,6444,6377],{"class":3199},[3175,6446,3235],{"class":3181},[3175,6448,6449],{"class":3249},"\"author-list\"",[3175,6451,4159],{"class":3181},[3175,6453,6454],{"class":3249},"\"\u002Ffxml\u002Fauthor-list-view.fxml\"",[3175,6456,4159],{"class":3181},[3175,6458,6459],{"class":3249},"\"Authors\"",[3175,6461,3561],{"class":3181},[3175,6463,6464,6466,6468,6471,6473,6476,6478,6481],{"class":3177,"line":4183},[3175,6465,6377],{"class":3199},[3175,6467,3235],{"class":3181},[3175,6469,6470],{"class":3249},"\"author-form\"",[3175,6472,4159],{"class":3181},[3175,6474,6475],{"class":3249},"\"\u002Ffxml\u002Fauthor-form-view.fxml\"",[3175,6477,4159],{"class":3181},[3175,6479,6480],{"class":3249},"\"Add Author\"",[3175,6482,3561],{"class":3181},[3175,6484,6485],{"class":3177,"line":4188},[3175,6486,3282],{"class":3181},[3175,6488,6489,6491,6493,6496,6498,6501,6503,6506],{"class":3177,"line":4193},[3175,6490,6377],{"class":3199},[3175,6492,3235],{"class":3181},[3175,6494,6495],{"class":3249},"\"genre-list\"",[3175,6497,4159],{"class":3181},[3175,6499,6500],{"class":3249},"\"\u002Ffxml\u002Fgenre-list-view.fxml\"",[3175,6502,4159],{"class":3181},[3175,6504,6505],{"class":3249},"\"Genres\"",[3175,6507,3561],{"class":3181},[3175,6509,6510],{"class":3177,"line":4199},[3175,6511,3282],{"class":3181},[3175,6513,6514],{"class":3177,"line":4204},[3175,6515,6516],{"class":3539},"        \u002F\u002F Dialogs\n",[3175,6518,6519,6521,6523,6526,6528,6531,6533,6536],{"class":3177,"line":4216},[3175,6520,6377],{"class":3199},[3175,6522,3235],{"class":3181},[3175,6524,6525],{"class":3249},"\"confirm-delete\"",[3175,6527,4159],{"class":3181},[3175,6529,6530],{"class":3249},"\"\u002Ffxml\u002Fdialogs\u002Fconfirm-delete-dialog.fxml\"",[3175,6532,4159],{"class":3181},[3175,6534,6535],{"class":3249},"\"Confirm Delete\"",[3175,6537,3561],{"class":3181},[3175,6539,6540,6542,6544,6547,6549,6552,6554,6557],{"class":3177,"line":4227},[3175,6541,6377],{"class":3199},[3175,6543,3235],{"class":3181},[3175,6545,6546],{"class":3249},"\"select-author\"",[3175,6548,4159],{"class":3181},[3175,6550,6551],{"class":3249},"\"\u002Ffxml\u002Fdialogs\u002Fselect-author-dialog.fxml\"",[3175,6553,4159],{"class":3181},[3175,6555,6556],{"class":3249},"\"Select Author\"",[3175,6558,3561],{"class":3181},[3175,6560,6561],{"class":3177,"line":4232},[3175,6562,3413],{"class":3181},[3175,6564,6565],{"class":3177,"line":4263},[3175,6566,4041],{"class":3181},[3175,6568,6569,6571,6573,6576,6578,6580,6582,6584,6586,6588,6590,6592,6595],{"class":3177,"line":4268},[3175,6570,4576],{"class":3192},[3175,6572,3196],{"class":3185},[3175,6574,6575],{"class":3199}," register",[3175,6577,3235],{"class":3181},[3175,6579,4088],{"class":3185},[3175,6581,4067],{"class":3222},[3175,6583,4159],{"class":3181},[3175,6585,4088],{"class":3185},[3175,6587,4881],{"class":3222},[3175,6589,4159],{"class":3181},[3175,6591,4088],{"class":3185},[3175,6593,6594],{"class":3222}," title",[3175,6596,3394],{"class":3181},[3175,6598,6599,6602,6604,6607,6609,6611,6614],{"class":3177,"line":4273},[3175,6600,6601],{"class":3222},"        screens",[3175,6603,3270],{"class":3181},[3175,6605,6606],{"class":3199},"put",[3175,6608,4804],{"class":3181},[3175,6610,3229],{"class":3209},[3175,6612,6613],{"class":3199}," ScreenDefinition",[3175,6615,6616],{"class":3181},"(screenId, fxmlPath, title));\n",[3175,6618,6619],{"class":3177,"line":4279},[3175,6620,3413],{"class":3181},[3175,6622,6623],{"class":3177,"line":4284},[3175,6624,4041],{"class":3181},[3175,6626,6627,6629,6631,6634,6636,6638,6640],{"class":3177,"line":4295},[3175,6628,4655],{"class":3192},[3175,6630,5887],{"class":3185},[3175,6632,6633],{"class":3199}," getFxmlPath",[3175,6635,3235],{"class":3181},[3175,6637,4088],{"class":3185},[3175,6639,4067],{"class":3222},[3175,6641,3394],{"class":3181},[3175,6643,6644,6647,6650,6652,6654,6656,6659],{"class":3177,"line":4306},[3175,6645,6646],{"class":3185},"        ScreenDefinition",[3175,6648,6649],{"class":3222}," screen",[3175,6651,3226],{"class":3181},[3175,6653,6320],{"class":3222},[3175,6655,3270],{"class":3181},[3175,6657,6658],{"class":3199},"get",[3175,6660,4893],{"class":3181},[3175,6662,6663,6665,6668,6670],{"class":3177,"line":4316},[3175,6664,5652],{"class":3209},[3175,6666,6667],{"class":3181}," (screen == ",[3175,6669,4936],{"class":3192},[3175,6671,3394],{"class":3181},[3175,6673,6674,6676,6678,6681,6683,6686],{"class":3177,"line":4321},[3175,6675,5138],{"class":3209},[3175,6677,5141],{"class":3209},[3175,6679,6680],{"class":3199}," IllegalArgumentException",[3175,6682,3235],{"class":3181},[3175,6684,6685],{"class":3249},"\"Screen not found: \"",[3175,6687,6688],{"class":3181}," + screenId);\n",[3175,6690,6691],{"class":3177,"line":4364},[3175,6692,5158],{"class":3181},[3175,6694,6695,6697,6699,6701,6703],{"class":3177,"line":4369},[3175,6696,5211],{"class":3209},[3175,6698,6649],{"class":3222},[3175,6700,3270],{"class":3181},[3175,6702,4890],{"class":3199},[3175,6704,3276],{"class":3181},[3175,6706,6707],{"class":3177,"line":4374},[3175,6708,3413],{"class":3181},[3175,6710,6711],{"class":3177,"line":4380},[3175,6712,4041],{"class":3181},[3175,6714,6715,6717,6719,6722,6724,6726,6728],{"class":3177,"line":4385},[3175,6716,4655],{"class":3192},[3175,6718,5887],{"class":3185},[3175,6720,6721],{"class":3199}," getTitle",[3175,6723,3235],{"class":3181},[3175,6725,4088],{"class":3185},[3175,6727,4067],{"class":3222},[3175,6729,3394],{"class":3181},[3175,6731,6732,6734,6736,6738,6740,6742,6744],{"class":3177,"line":4395},[3175,6733,6646],{"class":3185},[3175,6735,6649],{"class":3222},[3175,6737,3226],{"class":3181},[3175,6739,6320],{"class":3222},[3175,6741,3270],{"class":3181},[3175,6743,6658],{"class":3199},[3175,6745,4893],{"class":3181},[3175,6747,6748,6750,6752,6754],{"class":3177,"line":4400},[3175,6749,5652],{"class":3209},[3175,6751,6667],{"class":3181},[3175,6753,4936],{"class":3192},[3175,6755,3394],{"class":3181},[3175,6757,6758,6760,6762,6764,6766,6768],{"class":3177,"line":4405},[3175,6759,5138],{"class":3209},[3175,6761,5141],{"class":3209},[3175,6763,6680],{"class":3199},[3175,6765,3235],{"class":3181},[3175,6767,6685],{"class":3249},[3175,6769,6688],{"class":3181},[3175,6771,6772],{"class":3177,"line":4411},[3175,6773,5158],{"class":3181},[3175,6775,6776,6778,6780,6782,6785],{"class":3177,"line":4416},[3175,6777,5211],{"class":3209},[3175,6779,6649],{"class":3222},[3175,6781,3270],{"class":3181},[3175,6783,6784],{"class":3199},"getTitle",[3175,6786,3276],{"class":3181},[3175,6788,6789],{"class":3177,"line":4426},[3175,6790,3413],{"class":3181},[3175,6792,6793],{"class":3177,"line":4431},[3175,6794,4041],{"class":3181},[3175,6796,6797,6799,6801,6804,6806,6808,6810],{"class":3177,"line":4442},[3175,6798,4655],{"class":3192},[3175,6800,5758],{"class":3185},[3175,6802,6803],{"class":3199}," isRegistered",[3175,6805,3235],{"class":3181},[3175,6807,4088],{"class":3185},[3175,6809,4067],{"class":3222},[3175,6811,3394],{"class":3181},[3175,6813,6814,6816,6819,6821,6824],{"class":3177,"line":4987},[3175,6815,5211],{"class":3209},[3175,6817,6818],{"class":3222}," screens",[3175,6820,3270],{"class":3181},[3175,6822,6823],{"class":3199},"containsKey",[3175,6825,4893],{"class":3181},[3175,6827,6828],{"class":3177,"line":5004},[3175,6829,3413],{"class":3181},[3175,6831,6832],{"class":3177,"line":5010},[3175,6833,4041],{"class":3181},[3175,6835,6836],{"class":3177,"line":5016},[3175,6837,4046],{"class":3539},[3175,6839,6840],{"class":3177,"line":5021},[3175,6841,6842],{"class":3539},"     * Визначення екрану.\n",[3175,6844,6845],{"class":3177,"line":5027},[3175,6846,4075],{"class":3539},[3175,6848,6849,6851,6853,6855,6857],{"class":3177,"line":5047},[3175,6850,4576],{"class":3192},[3175,6852,5942],{"class":3192},[3175,6854,4557],{"class":3192},[3175,6856,6613],{"class":3185},[3175,6858,3213],{"class":3181},[3175,6860,6861,6863,6865,6867,6869],{"class":3177,"line":5066},[3175,6862,5954],{"class":3192},[3175,6864,4579],{"class":3192},[3175,6866,5887],{"class":3185},[3175,6868,4067],{"class":3222},[3175,6870,4588],{"class":3181},[3175,6872,6873,6875,6877,6879,6881],{"class":3177,"line":5077},[3175,6874,5954],{"class":3192},[3175,6876,4579],{"class":3192},[3175,6878,5887],{"class":3185},[3175,6880,4881],{"class":3222},[3175,6882,4588],{"class":3181},[3175,6884,6885,6887,6889,6891,6893],{"class":3177,"line":5092},[3175,6886,5954],{"class":3192},[3175,6888,4579],{"class":3192},[3175,6890,5887],{"class":3185},[3175,6892,6594],{"class":3222},[3175,6894,4588],{"class":3181},[3175,6896,6897],{"class":3177,"line":5097},[3175,6898,3282],{"class":3181},[3175,6900,6901,6903,6905,6907,6909,6911,6913,6915,6917,6919,6921,6923],{"class":3177,"line":5102},[3175,6902,5986],{"class":3192},[3175,6904,6613],{"class":3199},[3175,6906,3235],{"class":3181},[3175,6908,4088],{"class":3185},[3175,6910,4067],{"class":3222},[3175,6912,4159],{"class":3181},[3175,6914,4088],{"class":3185},[3175,6916,4881],{"class":3222},[3175,6918,4159],{"class":3181},[3175,6920,4088],{"class":3185},[3175,6922,6594],{"class":3222},[3175,6924,3394],{"class":3181},[3175,6926,6927,6929,6931,6933],{"class":3177,"line":5108},[3175,6928,6008],{"class":3192},[3175,6930,3270],{"class":3181},[3175,6932,6013],{"class":3222},[3175,6934,6016],{"class":3181},[3175,6936,6937,6939,6941,6944],{"class":3177,"line":5114},[3175,6938,6008],{"class":3192},[3175,6940,3270],{"class":3181},[3175,6942,6943],{"class":3222},"fxmlPath",[3175,6945,6946],{"class":3181}," = fxmlPath;\n",[3175,6948,6949,6951,6953,6956],{"class":3177,"line":5119},[3175,6950,6008],{"class":3192},[3175,6952,3270],{"class":3181},[3175,6954,6955],{"class":3222},"title",[3175,6957,6958],{"class":3181}," = title;\n",[3175,6960,6961],{"class":3177,"line":5135},[3175,6962,5158],{"class":3181},[3175,6964,6965],{"class":3177,"line":5155},[3175,6966,3282],{"class":3181},[3175,6968,6969,6971,6973,6975],{"class":3177,"line":5161},[3175,6970,5986],{"class":3192},[3175,6972,5887],{"class":3185},[3175,6974,6048],{"class":3199},[3175,6976,3203],{"class":3181},[3175,6978,6979,6981],{"class":3177,"line":5166},[3175,6980,5568],{"class":3209},[3175,6982,6058],{"class":3181},[3175,6984,6985],{"class":3177,"line":5171},[3175,6986,5158],{"class":3181},[3175,6988,6989],{"class":3177,"line":5178},[3175,6990,3282],{"class":3181},[3175,6992,6993,6995,6997,6999],{"class":3177,"line":5208},[3175,6994,5986],{"class":3192},[3175,6996,5887],{"class":3185},[3175,6998,6633],{"class":3199},[3175,7000,3203],{"class":3181},[3175,7002,7003,7005],{"class":3177,"line":5227},[3175,7004,5568],{"class":3209},[3175,7006,7007],{"class":3181}," fxmlPath;\n",[3175,7009,7010],{"class":3177,"line":5232},[3175,7011,5158],{"class":3181},[3175,7013,7014],{"class":3177,"line":5237},[3175,7015,3282],{"class":3181},[3175,7017,7018,7020,7022,7024],{"class":3177,"line":5244},[3175,7019,5986],{"class":3192},[3175,7021,5887],{"class":3185},[3175,7023,6721],{"class":3199},[3175,7025,3203],{"class":3181},[3175,7027,7028,7030],{"class":3177,"line":5289},[3175,7029,5568],{"class":3209},[3175,7031,7032],{"class":3181}," title;\n",[3175,7034,7035],{"class":3177,"line":5296},[3175,7036,5158],{"class":3181},[3175,7038,7039],{"class":3177,"line":5301},[3175,7040,3413],{"class":3181},[3175,7042,7043],{"class":3177,"line":5318},[3175,7044,3419],{"class":3181},[3150,7046,7047],{},[3157,7048,7049],{},"Переваги ScreenRegistry:",[3473,7051,7052,7058,7070,7076],{},[3476,7053,7054,7057],{},[3157,7055,7056],{},"Централізація:"," Всі екрани зареєстровані в одному місці. Легко побачити, які екрани є у додатку.",[3476,7059,7060,7063,7064,7066,7067,7069],{},[3157,7061,7062],{},"Відсутність magic strings:"," Замість ",[3172,7065,6227],{}," використовуємо ",[3172,7068,6382],{},". Якщо перейменувати файл, змінюємо лише Registry.",[3476,7071,7072,7075],{},[3157,7073,7074],{},"Додаткова інформація:"," Можна зберігати не лише шлях до FXML, а й title (для встановлення заголовку вікна), розмір вікна, іконку.",[3476,7077,7078,3442,7081,7084],{},[3157,7079,7080],{},"Валідація:",[3172,7082,7083],{},"getFxmlPath()"," викидає виняток, якщо екран не зареєстрований. Це виявляє помилки на етапі розробки.",[3512,7086,7088],{"id":7087},"використання-navigator-у-controller","Використання Navigator у Controller",[3165,7090,7092],{"className":3167,"code":7091,"language":3169,"meta":3170,"style":3170},"public class AudiobookListController {\n    \n    private final AudiobookListViewModel viewModel;\n    private final Navigator navigator;\n    \n    @Inject\n    public AudiobookListController(AudiobookListViewModel viewModel, Navigator navigator) {\n        this.viewModel = viewModel;\n        this.navigator = navigator;\n    }\n    \n    @FXML\n    private void onAddClicked() {\n        navigator.navigateTo(\"audiobook-form\");\n    }\n    \n    @FXML\n    private void onEditClicked() {\n        if (viewModel.getSelectedAudiobook() != null) {\n            navigator.navigateTo(\"audiobook-form\", Map.of(\n                \"audiobook\", viewModel.getSelectedAudiobook().getAudiobook()\n            ));\n        }\n    }\n    \n    @FXML\n    private void onDeleteClicked() {\n        Optional\u003CBoolean> result = navigator.openDialog(\"confirm-delete\");\n        if (result.isPresent() && result.get()) {\n            viewModel.deleteSelected();\n        }\n    }\n    \n    @FXML\n    private void onBackClicked() {\n        if (navigator.canGoBack()) {\n            navigator.goBack();\n        }\n    }\n}\n",[3172,7093,7094,7105,7109,7123,7136,7140,7146,7168,7180,7192,7196,7200,7206,7216,7232,7236,7240,7246,7257,7276,7300,7320,7325,7329,7333,7337,7343,7354,7383,7407,7419,7423,7427,7431,7437,7448,7462,7473,7477,7481],{"__ignoreMap":3170},[3175,7095,7096,7098,7100,7103],{"class":3177,"line":3178},[3175,7097,4028],{"class":3192},[3175,7099,4557],{"class":3192},[3175,7101,7102],{"class":3185}," AudiobookListController",[3175,7104,3213],{"class":3181},[3175,7106,7107],{"class":3177,"line":3189},[3175,7108,4041],{"class":3181},[3175,7110,7111,7113,7115,7118,7121],{"class":3177,"line":3206},[3175,7112,4576],{"class":3192},[3175,7114,4579],{"class":3192},[3175,7116,7117],{"class":3185}," AudiobookListViewModel",[3175,7119,7120],{"class":3222}," viewModel",[3175,7122,4588],{"class":3181},[3175,7124,7125,7127,7129,7131,7134],{"class":3177,"line":3216},[3175,7126,4576],{"class":3192},[3175,7128,4579],{"class":3192},[3175,7130,4034],{"class":3185},[3175,7132,7133],{"class":3222}," navigator",[3175,7135,4588],{"class":3181},[3175,7137,7138],{"class":3177,"line":3256},[3175,7139,4041],{"class":3181},[3175,7141,7142,7144],{"class":3177,"line":3279},[3175,7143,4647],{"class":3181},[3175,7145,4650],{"class":3185},[3175,7147,7148,7150,7152,7154,7157,7159,7161,7164,7166],{"class":3177,"line":3285},[3175,7149,4655],{"class":3192},[3175,7151,7102],{"class":3199},[3175,7153,3235],{"class":3181},[3175,7155,7156],{"class":3185},"AudiobookListViewModel",[3175,7158,7120],{"class":3222},[3175,7160,4159],{"class":3181},[3175,7162,7163],{"class":3185},"Navigator",[3175,7165,7133],{"class":3222},[3175,7167,3394],{"class":3181},[3175,7169,7170,7172,7174,7177],{"class":3177,"line":3305},[3175,7171,4678],{"class":3192},[3175,7173,3270],{"class":3181},[3175,7175,7176],{"class":3222},"viewModel",[3175,7178,7179],{"class":3181}," = viewModel;\n",[3175,7181,7182,7184,7186,7189],{"class":3177,"line":3326},[3175,7183,4678],{"class":3192},[3175,7185,3270],{"class":3181},[3175,7187,7188],{"class":3222},"navigator",[3175,7190,7191],{"class":3181}," = navigator;\n",[3175,7193,7194],{"class":3177,"line":3331},[3175,7195,3413],{"class":3181},[3175,7197,7198],{"class":3177,"line":3358},[3175,7199,4041],{"class":3181},[3175,7201,7202,7204],{"class":3177,"line":3376},[3175,7203,4647],{"class":3181},[3175,7205,3186],{"class":3185},[3175,7207,7208,7210,7212,7214],{"class":3177,"line":3397},[3175,7209,4576],{"class":3192},[3175,7211,3196],{"class":3185},[3175,7213,3200],{"class":3199},[3175,7215,3203],{"class":3181},[3175,7217,7218,7221,7223,7226,7228,7230],{"class":3177,"line":3410},[3175,7219,7220],{"class":3222},"        navigator",[3175,7222,3270],{"class":3181},[3175,7224,7225],{"class":3199},"navigateTo",[3175,7227,3235],{"class":3181},[3175,7229,6403],{"class":3249},[3175,7231,3561],{"class":3181},[3175,7233,7234],{"class":3177,"line":3416},[3175,7235,3413],{"class":3181},[3175,7237,7238],{"class":3177,"line":4095},[3175,7239,4041],{"class":3181},[3175,7241,7242,7244],{"class":3177,"line":4100},[3175,7243,4647],{"class":3181},[3175,7245,3186],{"class":3185},[3175,7247,7248,7250,7252,7255],{"class":3177,"line":4105},[3175,7249,4576],{"class":3192},[3175,7251,3196],{"class":3185},[3175,7253,7254],{"class":3199}," onEditClicked",[3175,7256,3203],{"class":3181},[3175,7258,7259,7261,7263,7265,7267,7270,7272,7274],{"class":3177,"line":4111},[3175,7260,5652],{"class":3209},[3175,7262,3385],{"class":3181},[3175,7264,7176],{"class":3222},[3175,7266,3270],{"class":3181},[3175,7268,7269],{"class":3199},"getSelectedAudiobook",[3175,7271,4933],{"class":3181},[3175,7273,4936],{"class":3192},[3175,7275,3394],{"class":3181},[3175,7277,7278,7281,7283,7285,7287,7289,7291,7293,7295,7298],{"class":3177,"line":4116},[3175,7279,7280],{"class":3222},"            navigator",[3175,7282,3270],{"class":3181},[3175,7284,7225],{"class":3199},[3175,7286,3235],{"class":3181},[3175,7288,6403],{"class":3249},[3175,7290,4159],{"class":3181},[3175,7292,4162],{"class":3222},[3175,7294,3270],{"class":3181},[3175,7296,7297],{"class":3199},"of",[3175,7299,4976],{"class":3181},[3175,7301,7302,7305,7307,7309,7311,7313,7315,7318],{"class":3177,"line":4128},[3175,7303,7304],{"class":3249},"                \"audiobook\"",[3175,7306,4159],{"class":3181},[3175,7308,7176],{"class":3222},[3175,7310,3270],{"class":3181},[3175,7312,7269],{"class":3199},[3175,7314,3241],{"class":3181},[3175,7316,7317],{"class":3199},"getAudiobook",[3175,7319,5001],{"class":3181},[3175,7321,7322],{"class":3177,"line":4141},[3175,7323,7324],{"class":3181},"            ));\n",[3175,7326,7327],{"class":3177,"line":4146},[3175,7328,5158],{"class":3181},[3175,7330,7331],{"class":3177,"line":4183},[3175,7332,3413],{"class":3181},[3175,7334,7335],{"class":3177,"line":4188},[3175,7336,4041],{"class":3181},[3175,7338,7339,7341],{"class":3177,"line":4193},[3175,7340,4647],{"class":3181},[3175,7342,3186],{"class":3185},[3175,7344,7345,7347,7349,7352],{"class":3177,"line":4199},[3175,7346,4576],{"class":3192},[3175,7348,3196],{"class":3185},[3175,7350,7351],{"class":3199}," onDeleteClicked",[3175,7353,3203],{"class":3181},[3175,7355,7356,7359,7361,7364,7366,7369,7371,7373,7375,7377,7379,7381],{"class":3177,"line":4204},[3175,7357,7358],{"class":3185},"        Optional",[3175,7360,4165],{"class":3181},[3175,7362,7363],{"class":3185},"Boolean",[3175,7365,4175],{"class":3181},[3175,7367,7368],{"class":3222},"result",[3175,7370,3226],{"class":3181},[3175,7372,7188],{"class":3222},[3175,7374,3270],{"class":3181},[3175,7376,4252],{"class":3199},[3175,7378,3235],{"class":3181},[3175,7380,6525],{"class":3249},[3175,7382,3561],{"class":3181},[3175,7384,7385,7387,7389,7391,7393,7396,7399,7401,7403,7405],{"class":3177,"line":4216},[3175,7386,5652],{"class":3209},[3175,7388,3385],{"class":3181},[3175,7390,7368],{"class":3222},[3175,7392,3270],{"class":3181},[3175,7394,7395],{"class":3199},"isPresent",[3175,7397,7398],{"class":3181},"() && ",[3175,7400,7368],{"class":3222},[3175,7402,3270],{"class":3181},[3175,7404,6658],{"class":3199},[3175,7406,5661],{"class":3181},[3175,7408,7409,7412,7414,7417],{"class":3177,"line":4227},[3175,7410,7411],{"class":3222},"            viewModel",[3175,7413,3270],{"class":3181},[3175,7415,7416],{"class":3199},"deleteSelected",[3175,7418,3276],{"class":3181},[3175,7420,7421],{"class":3177,"line":4232},[3175,7422,5158],{"class":3181},[3175,7424,7425],{"class":3177,"line":4263},[3175,7426,3413],{"class":3181},[3175,7428,7429],{"class":3177,"line":4268},[3175,7430,4041],{"class":3181},[3175,7432,7433,7435],{"class":3177,"line":4273},[3175,7434,4647],{"class":3181},[3175,7436,3186],{"class":3185},[3175,7438,7439,7441,7443,7446],{"class":3177,"line":4279},[3175,7440,4576],{"class":3192},[3175,7442,3196],{"class":3185},[3175,7444,7445],{"class":3199}," onBackClicked",[3175,7447,3203],{"class":3181},[3175,7449,7450,7452,7454,7456,7458,7460],{"class":3177,"line":4284},[3175,7451,5652],{"class":3209},[3175,7453,3385],{"class":3181},[3175,7455,7188],{"class":3222},[3175,7457,3270],{"class":3181},[3175,7459,5658],{"class":3199},[3175,7461,5661],{"class":3181},[3175,7463,7464,7466,7468,7471],{"class":3177,"line":4295},[3175,7465,7280],{"class":3222},[3175,7467,3270],{"class":3181},[3175,7469,7470],{"class":3199},"goBack",[3175,7472,3276],{"class":3181},[3175,7474,7475],{"class":3177,"line":4306},[3175,7476,5158],{"class":3181},[3175,7478,7479],{"class":3177,"line":4316},[3175,7480,3413],{"class":3181},[3175,7482,7483],{"class":3177,"line":4321},[3175,7484,3419],{"class":3181},[3150,7486,7487,7488,6130,7491,7494],{},"Тепер Controller не містить коду завантаження FXML — лише виклики ",[3172,7489,7490],{},"navigator.navigateTo()",[3172,7492,7493],{},"navigator.openDialog()",". Весь складний код інкапсульований у Navigator.",[3502,7496],{},[3145,7498,7500],{"id":7499},"передача-параметрів-між-екранами","Передача параметрів між екранами",[3150,7502,7503,7504,7506],{},"Як передати ",[3172,7505,3454],{}," з екрану списку у форму редагування? Є три підходи:",[3512,7508,7510],{"id":7509},"підхід-1-через-singleton-viewmodel","Підхід 1: Через Singleton ViewModel",[3150,7512,7513,3442,7515,7517,7518,7521,7522,7525],{},[3157,7514,3520],{},[3172,7516,7156],{}," — Singleton. Він зберігає ",[3172,7519,7520],{},"selectedAudiobook",". ",[3172,7523,7524],{},"AudiobookFormViewModel"," отримує його через ін'єкцію.",[3165,7527,7529],{"className":3167,"code":7528,"language":3169,"meta":3170,"style":3170},"@Singleton\npublic class AudiobookListViewModel {\n    private final ObjectProperty\u003CAudiobook> selectedAudiobook = new SimpleObjectProperty\u003C>();\n    \n    \u002F\u002F Getters\n}\n\npublic class AudiobookFormViewModel {\n    private final AudiobookListViewModel listViewModel;\n    \n    @Inject\n    public AudiobookFormViewModel(AudiobookListViewModel listViewModel) {\n        this.listViewModel = listViewModel;\n        \n        \u002F\u002F Завантаження даних з selectedAudiobook\n        Audiobook audiobook = listViewModel.getSelectedAudiobook();\n        if (audiobook != null) {\n            loadAudiobook(audiobook);\n        }\n    }\n}\n",[3172,7530,7531,7537,7547,7573,7577,7582,7586,7590,7600,7613,7617,7623,7637,7649,7653,7658,7676,7687,7695,7699,7703],{"__ignoreMap":3170},[3175,7532,7533,7535],{"class":3177,"line":3178},[3175,7534,3182],{"class":3181},[3175,7536,4550],{"class":3185},[3175,7538,7539,7541,7543,7545],{"class":3177,"line":3189},[3175,7540,4028],{"class":3192},[3175,7542,4557],{"class":3192},[3175,7544,7117],{"class":3185},[3175,7546,3213],{"class":3181},[3175,7548,7549,7551,7553,7556,7558,7560,7562,7564,7566,7568,7571],{"class":3177,"line":3206},[3175,7550,4576],{"class":3192},[3175,7552,4579],{"class":3192},[3175,7554,7555],{"class":3185}," ObjectProperty",[3175,7557,4165],{"class":3181},[3175,7559,3454],{"class":3185},[3175,7561,4175],{"class":3181},[3175,7563,7520],{"class":3222},[3175,7565,3226],{"class":3181},[3175,7567,3229],{"class":3209},[3175,7569,7570],{"class":3185}," SimpleObjectProperty",[3175,7572,4715],{"class":3181},[3175,7574,7575],{"class":3177,"line":3216},[3175,7576,4041],{"class":3181},[3175,7578,7579],{"class":3177,"line":3256},[3175,7580,7581],{"class":3539},"    \u002F\u002F Getters\n",[3175,7583,7584],{"class":3177,"line":3279},[3175,7585,3419],{"class":3181},[3175,7587,7588],{"class":3177,"line":3285},[3175,7589,3993],{"emptyLinePlaceholder":3992},[3175,7591,7592,7594,7596,7598],{"class":3177,"line":3305},[3175,7593,4028],{"class":3192},[3175,7595,4557],{"class":3192},[3175,7597,3320],{"class":3185},[3175,7599,3213],{"class":3181},[3175,7601,7602,7604,7606,7608,7611],{"class":3177,"line":3326},[3175,7603,4576],{"class":3192},[3175,7605,4579],{"class":3192},[3175,7607,7117],{"class":3185},[3175,7609,7610],{"class":3222}," listViewModel",[3175,7612,4588],{"class":3181},[3175,7614,7615],{"class":3177,"line":3331},[3175,7616,4041],{"class":3181},[3175,7618,7619,7621],{"class":3177,"line":3358},[3175,7620,4647],{"class":3181},[3175,7622,4650],{"class":3185},[3175,7624,7625,7627,7629,7631,7633,7635],{"class":3177,"line":3376},[3175,7626,4655],{"class":3192},[3175,7628,3320],{"class":3199},[3175,7630,3235],{"class":3181},[3175,7632,7156],{"class":3185},[3175,7634,7610],{"class":3222},[3175,7636,3394],{"class":3181},[3175,7638,7639,7641,7643,7646],{"class":3177,"line":3397},[3175,7640,4678],{"class":3192},[3175,7642,3270],{"class":3181},[3175,7644,7645],{"class":3222},"listViewModel",[3175,7647,7648],{"class":3181}," = listViewModel;\n",[3175,7650,7651],{"class":3177,"line":3410},[3175,7652,3282],{"class":3181},[3175,7654,7655],{"class":3177,"line":3416},[3175,7656,7657],{"class":3539},"        \u002F\u002F Завантаження даних з selectedAudiobook\n",[3175,7659,7660,7663,7666,7668,7670,7672,7674],{"class":3177,"line":4095},[3175,7661,7662],{"class":3185},"        Audiobook",[3175,7664,7665],{"class":3222}," audiobook",[3175,7667,3226],{"class":3181},[3175,7669,7645],{"class":3222},[3175,7671,3270],{"class":3181},[3175,7673,7269],{"class":3199},[3175,7675,3276],{"class":3181},[3175,7677,7678,7680,7683,7685],{"class":3177,"line":4100},[3175,7679,5652],{"class":3209},[3175,7681,7682],{"class":3181}," (audiobook != ",[3175,7684,4936],{"class":3192},[3175,7686,3394],{"class":3181},[3175,7688,7689,7692],{"class":3177,"line":4105},[3175,7690,7691],{"class":3199},"            loadAudiobook",[3175,7693,7694],{"class":3181},"(audiobook);\n",[3175,7696,7697],{"class":3177,"line":4111},[3175,7698,5158],{"class":3181},[3175,7700,7701],{"class":3177,"line":4116},[3175,7702,3413],{"class":3181},[3175,7704,7705],{"class":3177,"line":4128},[3175,7706,3419],{"class":3181},[3150,7708,7709,7711],{},[3157,7710,3582],{}," Просто реалізувати, не потрібен Navigator для передачі параметрів.",[3150,7713,7714,7716,7717,7719,7720,7722],{},[3157,7715,3598],{}," Сильна зв'язаність між ViewModels. ",[3172,7718,7524],{}," залежить від ",[3172,7721,7156],{},", хоча логічно вони незалежні. Складно тестувати.",[3512,7724,7726],{"id":7725},"підхід-2-через-параметри-navigator","Підхід 2: Через параметри Navigator",[3150,7728,7729,3442,7731,7734,7735,7738,7739,3270],{},[3157,7730,3520],{},[3172,7732,7733],{},"Navigator.navigateTo(screenId, params)"," приймає ",[3172,7736,7737],{},"Map\u003CString, Object>"," з параметрами. Navigator передає їх у ViewModel через setter або метод ",[3172,7740,7741],{},"initialize()",[3165,7743,7745],{"className":3167,"code":7744,"language":3169,"meta":3170,"style":3170},"\u002F\u002F Controller\nnavigator.navigateTo(\"audiobook-form\", Map.of(\"audiobook\", audiobook));\n\n\u002F\u002F Navigator\n@Override\npublic void navigateTo(String screenId, Map\u003CString, Object> params) {\n    Parent root = loadFxml(fxmlPath);\n    \n    \u002F\u002F Отримання Controller з FXMLLoader\n    Object controller = loader.getController();\n    \n    \u002F\u002F Передача параметрів у ViewModel через Controller\n    if (controller instanceof ParameterizedController) {\n        ((ParameterizedController) controller).setParameters(params);\n    }\n    \n    primaryStage.getScene().setRoot(root);\n}\n\n\u002F\u002F Controller\npublic class AudiobookFormController implements ParameterizedController {\n    \n    @Override\n    public void setParameters(Map\u003CString, Object> params) {\n        Audiobook audiobook = (Audiobook) params.get(\"audiobook\");\n        if (audiobook != null) {\n            viewModel.loadAudiobook(audiobook);\n        }\n    }\n}\n",[3172,7746,7747,7752,7780,7784,7789,7795,7815,7828,7832,7837,7854,7858,7863,7875,7886,7890,7894,7909,7913,7917,7921,7937,7941,7947,7974,7995,8005,8016,8020,8024],{"__ignoreMap":3170},[3175,7748,7749],{"class":3177,"line":3178},[3175,7750,7751],{"class":3539},"\u002F\u002F Controller\n",[3175,7753,7754,7756,7758,7760,7762,7764,7766,7768,7770,7772,7774,7777],{"class":3177,"line":3189},[3175,7755,7188],{"class":3222},[3175,7757,3270],{"class":3181},[3175,7759,7225],{"class":3199},[3175,7761,3235],{"class":3181},[3175,7763,6403],{"class":3249},[3175,7765,4159],{"class":3181},[3175,7767,4162],{"class":3222},[3175,7769,3270],{"class":3181},[3175,7771,7297],{"class":3199},[3175,7773,3235],{"class":3181},[3175,7775,7776],{"class":3249},"\"audiobook\"",[3175,7778,7779],{"class":3181},", audiobook));\n",[3175,7781,7782],{"class":3177,"line":3206},[3175,7783,3993],{"emptyLinePlaceholder":3992},[3175,7785,7786],{"class":3177,"line":3216},[3175,7787,7788],{"class":3539},"\u002F\u002F Navigator\n",[3175,7790,7791,7793],{"class":3177,"line":3256},[3175,7792,3182],{"class":3181},[3175,7794,4780],{"class":3185},[3175,7796,7797,7799,7801,7803,7805,7807,7810,7812],{"class":3177,"line":3279},[3175,7798,4028],{"class":3192},[3175,7800,3196],{"class":3185},[3175,7802,4083],{"class":3199},[3175,7804,3235],{"class":3181},[3175,7806,4088],{"class":3185},[3175,7808,7809],{"class":3181}," screenId, ",[3175,7811,4162],{"class":3185},[3175,7813,7814],{"class":3181},"\u003CString, Object> params) {\n",[3175,7816,7817,7820,7822,7824,7826],{"class":3177,"line":3285},[3175,7818,7819],{"class":3185},"    Parent",[3175,7821,3262],{"class":3222},[3175,7823,3226],{"class":3181},[3175,7825,3553],{"class":3199},[3175,7827,4907],{"class":3181},[3175,7829,7830],{"class":3177,"line":3305},[3175,7831,4041],{"class":3181},[3175,7833,7834],{"class":3177,"line":3326},[3175,7835,7836],{"class":3539},"    \u002F\u002F Отримання Controller з FXMLLoader\n",[3175,7838,7839,7842,7844,7846,7848,7850,7852],{"class":3177,"line":3331},[3175,7840,7841],{"class":3185},"    Object",[3175,7843,3291],{"class":3222},[3175,7845,3226],{"class":3181},[3175,7847,3267],{"class":3222},[3175,7849,3270],{"class":3181},[3175,7851,3300],{"class":3199},[3175,7853,3276],{"class":3181},[3175,7855,7856],{"class":3177,"line":3358},[3175,7857,4041],{"class":3181},[3175,7859,7860],{"class":3177,"line":3376},[3175,7861,7862],{"class":3539},"    \u002F\u002F Передача параметрів у ViewModel через Controller\n",[3175,7864,7865,7868,7870,7872],{"class":3177,"line":3397},[3175,7866,7867],{"class":3209},"    if",[3175,7869,5518],{"class":3181},[3175,7871,5521],{"class":3192},[3175,7873,7874],{"class":3181}," ParameterizedController) {\n",[3175,7876,7877,7880,7883],{"class":3177,"line":3410},[3175,7878,7879],{"class":3181},"        ((ParameterizedController) controller).",[3175,7881,7882],{"class":3199},"setParameters",[3175,7884,7885],{"class":3181},"(params);\n",[3175,7887,7888],{"class":3177,"line":3416},[3175,7889,3413],{"class":3181},[3175,7891,7892],{"class":3177,"line":4095},[3175,7893,4041],{"class":3181},[3175,7895,7896,7899,7901,7903,7905,7907],{"class":3177,"line":4100},[3175,7897,7898],{"class":3222},"    primaryStage",[3175,7900,3270],{"class":3181},[3175,7902,3348],{"class":3199},[3175,7904,3241],{"class":3181},[3175,7906,3370],{"class":3199},[3175,7908,3373],{"class":3181},[3175,7910,7911],{"class":3177,"line":4105},[3175,7912,3419],{"class":3181},[3175,7914,7915],{"class":3177,"line":4111},[3175,7916,3993],{"emptyLinePlaceholder":3992},[3175,7918,7919],{"class":3177,"line":4116},[3175,7920,7751],{"class":3539},[3175,7922,7923,7925,7927,7930,7932,7935],{"class":3177,"line":4128},[3175,7924,4028],{"class":3192},[3175,7926,4557],{"class":3192},[3175,7928,7929],{"class":3185}," AudiobookFormController",[3175,7931,4563],{"class":3192},[3175,7933,7934],{"class":3185}," ParameterizedController",[3175,7936,3213],{"class":3181},[3175,7938,7939],{"class":3177,"line":4141},[3175,7940,4041],{"class":3181},[3175,7942,7943,7945],{"class":3177,"line":4146},[3175,7944,4647],{"class":3181},[3175,7946,4780],{"class":3185},[3175,7948,7949,7951,7953,7956,7958,7960,7962,7964,7966,7968,7970,7972],{"class":3177,"line":4183},[3175,7950,4655],{"class":3192},[3175,7952,3196],{"class":3185},[3175,7954,7955],{"class":3199}," setParameters",[3175,7957,3235],{"class":3181},[3175,7959,4162],{"class":3185},[3175,7961,4165],{"class":3181},[3175,7963,4088],{"class":3185},[3175,7965,4159],{"class":3181},[3175,7967,4172],{"class":3185},[3175,7969,4175],{"class":3181},[3175,7971,4178],{"class":3222},[3175,7973,3394],{"class":3181},[3175,7975,7976,7978,7980,7983,7985,7987,7989,7991,7993],{"class":3177,"line":4188},[3175,7977,7662],{"class":3185},[3175,7979,7665],{"class":3222},[3175,7981,7982],{"class":3181}," = (Audiobook) ",[3175,7984,4178],{"class":3222},[3175,7986,3270],{"class":3181},[3175,7988,6658],{"class":3199},[3175,7990,3235],{"class":3181},[3175,7992,7776],{"class":3249},[3175,7994,3561],{"class":3181},[3175,7996,7997,7999,8001,8003],{"class":3177,"line":4193},[3175,7998,5652],{"class":3209},[3175,8000,7682],{"class":3181},[3175,8002,4936],{"class":3192},[3175,8004,3394],{"class":3181},[3175,8006,8007,8009,8011,8014],{"class":3177,"line":4199},[3175,8008,7411],{"class":3222},[3175,8010,3270],{"class":3181},[3175,8012,8013],{"class":3199},"loadAudiobook",[3175,8015,7694],{"class":3181},[3175,8017,8018],{"class":3177,"line":4204},[3175,8019,5158],{"class":3181},[3175,8021,8022],{"class":3177,"line":4216},[3175,8023,3413],{"class":3181},[3175,8025,8026],{"class":3177,"line":4227},[3175,8027,3419],{"class":3181},[3150,8029,8030,8032],{},[3157,8031,3582],{}," Явна передача параметрів, немає зв'язаності між ViewModels.",[3150,8034,8035,8037,8038,8041],{},[3157,8036,3598],{}," Потрібен інтерфейс ",[3172,8039,8040],{},"ParameterizedController",", type-safety відсутня (Map з Object).",[3512,8043,8045],{"id":8044},"підхід-3-через-assistedinject-factory","Підхід 3: Через AssistedInject (Factory)",[3150,8047,8048,8050,8051,3270],{},[3157,8049,3520],{}," ViewModel створюється через Factory з параметром. Navigator викликає Factory замість ",[3172,8052,8053],{},"injector.getInstance()",[3165,8055,8057],{"className":3167,"code":8056,"language":3169,"meta":3170,"style":3170},"\u002F\u002F Factory\npublic interface AudiobookFormViewModelFactory {\n    AudiobookFormViewModel create(@Nullable Audiobook audiobook);\n}\n\n\u002F\u002F ViewModel\npublic class AudiobookFormViewModel {\n    @Inject\n    public AudiobookFormViewModel(\n        @Assisted @Nullable Audiobook audiobook,\n        AudiobookService service\n    ) {\n        if (audiobook != null) {\n            loadAudiobook(audiobook);\n        }\n    }\n}\n\n\u002F\u002F Navigator (модифікований)\npublic void navigateTo(String screenId, Audiobook audiobook) {\n    AudiobookFormViewModelFactory factory = injector.getInstance(AudiobookFormViewModelFactory.class);\n    AudiobookFormViewModel viewModel = factory.create(audiobook);\n    \n    \u002F\u002F Завантаження FXML та передача ViewModel у Controller\n    \u002F\u002F ...\n}\n",[3172,8058,8059,8064,8075,8096,8100,8104,8109,8119,8125,8133,8153,8161,8166,8176,8182,8186,8190,8194,8198,8203,8222,8251,8269,8273,8278,8283],{"__ignoreMap":3170},[3175,8060,8061],{"class":3177,"line":3178},[3175,8062,8063],{"class":3539},"\u002F\u002F Factory\n",[3175,8065,8066,8068,8070,8073],{"class":3177,"line":3189},[3175,8067,4028],{"class":3192},[3175,8069,4031],{"class":3192},[3175,8071,8072],{"class":3185}," AudiobookFormViewModelFactory",[3175,8074,3213],{"class":3181},[3175,8076,8077,8080,8083,8086,8089,8092,8094],{"class":3177,"line":3206},[3175,8078,8079],{"class":3185},"    AudiobookFormViewModel",[3175,8081,8082],{"class":3199}," create",[3175,8084,8085],{"class":3181},"(@",[3175,8087,8088],{"class":3185},"Nullable",[3175,8090,8091],{"class":3185}," Audiobook",[3175,8093,7665],{"class":3222},[3175,8095,3561],{"class":3181},[3175,8097,8098],{"class":3177,"line":3216},[3175,8099,3419],{"class":3181},[3175,8101,8102],{"class":3177,"line":3256},[3175,8103,3993],{"emptyLinePlaceholder":3992},[3175,8105,8106],{"class":3177,"line":3279},[3175,8107,8108],{"class":3539},"\u002F\u002F ViewModel\n",[3175,8110,8111,8113,8115,8117],{"class":3177,"line":3285},[3175,8112,4028],{"class":3192},[3175,8114,4557],{"class":3192},[3175,8116,3320],{"class":3185},[3175,8118,3213],{"class":3181},[3175,8120,8121,8123],{"class":3177,"line":3305},[3175,8122,4647],{"class":3181},[3175,8124,4650],{"class":3185},[3175,8126,8127,8129,8131],{"class":3177,"line":3326},[3175,8128,4655],{"class":3192},[3175,8130,3320],{"class":3199},[3175,8132,4976],{"class":3181},[3175,8134,8135,8138,8141,8144,8146,8148,8150],{"class":3177,"line":3331},[3175,8136,8137],{"class":3181},"        @",[3175,8139,8140],{"class":3185},"Assisted",[3175,8142,8143],{"class":3181}," @",[3175,8145,8088],{"class":3185},[3175,8147,8091],{"class":3185},[3175,8149,7665],{"class":3222},[3175,8151,8152],{"class":3181},",\n",[3175,8154,8155,8158],{"class":3177,"line":3358},[3175,8156,8157],{"class":3185},"        AudiobookService",[3175,8159,8160],{"class":3222}," service\n",[3175,8162,8163],{"class":3177,"line":3376},[3175,8164,8165],{"class":3181},"    ) {\n",[3175,8167,8168,8170,8172,8174],{"class":3177,"line":3397},[3175,8169,5652],{"class":3209},[3175,8171,7682],{"class":3181},[3175,8173,4936],{"class":3192},[3175,8175,3394],{"class":3181},[3175,8177,8178,8180],{"class":3177,"line":3410},[3175,8179,7691],{"class":3199},[3175,8181,7694],{"class":3181},[3175,8183,8184],{"class":3177,"line":3416},[3175,8185,5158],{"class":3181},[3175,8187,8188],{"class":3177,"line":4095},[3175,8189,3413],{"class":3181},[3175,8191,8192],{"class":3177,"line":4100},[3175,8193,3419],{"class":3181},[3175,8195,8196],{"class":3177,"line":4105},[3175,8197,3993],{"emptyLinePlaceholder":3992},[3175,8199,8200],{"class":3177,"line":4111},[3175,8201,8202],{"class":3539},"\u002F\u002F Navigator (модифікований)\n",[3175,8204,8205,8207,8209,8211,8213,8215,8217,8219],{"class":3177,"line":4116},[3175,8206,4028],{"class":3192},[3175,8208,3196],{"class":3185},[3175,8210,4083],{"class":3199},[3175,8212,3235],{"class":3181},[3175,8214,4088],{"class":3185},[3175,8216,7809],{"class":3181},[3175,8218,3454],{"class":3185},[3175,8220,8221],{"class":3181}," audiobook) {\n",[3175,8223,8224,8227,8230,8232,8234,8236,8239,8241,8244,8246,8249],{"class":3177,"line":4128},[3175,8225,8226],{"class":3185},"    AudiobookFormViewModelFactory",[3175,8228,8229],{"class":3222}," factory",[3175,8231,3226],{"class":3181},[3175,8233,4683],{"class":3222},[3175,8235,3270],{"class":3181},[3175,8237,8238],{"class":3199},"getInstance",[3175,8240,3235],{"class":3181},[3175,8242,8243],{"class":3222},"AudiobookFormViewModelFactory",[3175,8245,3270],{"class":3181},[3175,8247,8248],{"class":3222},"class",[3175,8250,3561],{"class":3181},[3175,8252,8253,8255,8257,8259,8262,8264,8267],{"class":3177,"line":4141},[3175,8254,8079],{"class":3185},[3175,8256,7120],{"class":3222},[3175,8258,3226],{"class":3181},[3175,8260,8261],{"class":3222},"factory",[3175,8263,3270],{"class":3181},[3175,8265,8266],{"class":3199},"create",[3175,8268,7694],{"class":3181},[3175,8270,8271],{"class":3177,"line":4146},[3175,8272,4041],{"class":3181},[3175,8274,8275],{"class":3177,"line":4183},[3175,8276,8277],{"class":3539},"    \u002F\u002F Завантаження FXML та передача ViewModel у Controller\n",[3175,8279,8280],{"class":3177,"line":4188},[3175,8281,8282],{"class":3539},"    \u002F\u002F ...\n",[3175,8284,8285],{"class":3177,"line":4193},[3175,8286,3419],{"class":3181},[3150,8288,8289,8291],{},[3157,8290,3582],{}," Type-safety, явні залежності, легко тестувати.",[3150,8293,8294,8296],{},[3157,8295,3598],{}," Складніша реалізація, потрібна інтеграція Factory з Navigator.",[8298,8299,8300,8303,8304,8307,8308,8311,8312,8315],"tip",{},[3157,8301,8302],{},"Рекомендація:"," Використовуйте ",[3157,8305,8306],{},"Підхід 2"," (параметри Navigator) для простих випадків (передача ID, прапорців). Використовуйте ",[3157,8309,8310],{},"Підхід 3"," (AssistedInject) для складних випадків (передача об'єктів Domain Model, кілька параметрів). Уникайте ",[3157,8313,8314],{},"Підходу 1"," (Singleton ViewModel) — він створює сильну зв'язаність.",[3502,8317],{},[3145,8319,8321],{"id":8320},"інтеграція-navigator-з-viewmodel","Інтеграція Navigator з ViewModel",[3150,8323,8324],{},"У MVVM ViewModel не повинен знати про Navigator безпосередньо — це порушує принцип розділення відповідальностей. Але як ViewModel ініціює навігацію (наприклад, після успішного збереження)?",[3512,8326,8328],{"id":8327},"проблема-viewmodel-не-може-викликати-navigator","Проблема: ViewModel не може викликати Navigator",[3165,8330,8332],{"className":3167,"code":8331,"language":3169,"meta":3170,"style":3170},"\u002F\u002F ПОГАНО: ViewModel залежить від Navigator\npublic class AudiobookFormViewModel {\n    private final Navigator navigator;\n    \n    @Inject\n    public AudiobookFormViewModel(Navigator navigator) {\n        this.navigator = navigator;\n    }\n    \n    public void save() {\n        \u002F\u002F Збереження аудіокниги\n        audiobookService.save(audiobook);\n        \n        \u002F\u002F Навігація назад\n        navigator.goBack(); \u002F\u002F Порушення MVVM!\n    }\n}\n",[3172,8333,8334,8339,8349,8361,8365,8371,8385,8395,8399,8403,8414,8419,8431,8435,8440,8454,8458],{"__ignoreMap":3170},[3175,8335,8336],{"class":3177,"line":3178},[3175,8337,8338],{"class":3539},"\u002F\u002F ПОГАНО: ViewModel залежить від Navigator\n",[3175,8340,8341,8343,8345,8347],{"class":3177,"line":3189},[3175,8342,4028],{"class":3192},[3175,8344,4557],{"class":3192},[3175,8346,3320],{"class":3185},[3175,8348,3213],{"class":3181},[3175,8350,8351,8353,8355,8357,8359],{"class":3177,"line":3206},[3175,8352,4576],{"class":3192},[3175,8354,4579],{"class":3192},[3175,8356,4034],{"class":3185},[3175,8358,7133],{"class":3222},[3175,8360,4588],{"class":3181},[3175,8362,8363],{"class":3177,"line":3216},[3175,8364,4041],{"class":3181},[3175,8366,8367,8369],{"class":3177,"line":3256},[3175,8368,4647],{"class":3181},[3175,8370,4650],{"class":3185},[3175,8372,8373,8375,8377,8379,8381,8383],{"class":3177,"line":3279},[3175,8374,4655],{"class":3192},[3175,8376,3320],{"class":3199},[3175,8378,3235],{"class":3181},[3175,8380,7163],{"class":3185},[3175,8382,7133],{"class":3222},[3175,8384,3394],{"class":3181},[3175,8386,8387,8389,8391,8393],{"class":3177,"line":3285},[3175,8388,4678],{"class":3192},[3175,8390,3270],{"class":3181},[3175,8392,7188],{"class":3222},[3175,8394,7191],{"class":3181},[3175,8396,8397],{"class":3177,"line":3305},[3175,8398,3413],{"class":3181},[3175,8400,8401],{"class":3177,"line":3326},[3175,8402,4041],{"class":3181},[3175,8404,8405,8407,8409,8412],{"class":3177,"line":3331},[3175,8406,4655],{"class":3192},[3175,8408,3196],{"class":3185},[3175,8410,8411],{"class":3199}," save",[3175,8413,3203],{"class":3181},[3175,8415,8416],{"class":3177,"line":3358},[3175,8417,8418],{"class":3539},"        \u002F\u002F Збереження аудіокниги\n",[3175,8420,8421,8424,8426,8429],{"class":3177,"line":3376},[3175,8422,8423],{"class":3222},"        audiobookService",[3175,8425,3270],{"class":3181},[3175,8427,8428],{"class":3199},"save",[3175,8430,7694],{"class":3181},[3175,8432,8433],{"class":3177,"line":3397},[3175,8434,3282],{"class":3181},[3175,8436,8437],{"class":3177,"line":3410},[3175,8438,8439],{"class":3539},"        \u002F\u002F Навігація назад\n",[3175,8441,8442,8444,8446,8448,8451],{"class":3177,"line":3416},[3175,8443,7220],{"class":3222},[3175,8445,3270],{"class":3181},[3175,8447,7470],{"class":3199},[3175,8449,8450],{"class":3181},"(); ",[3175,8452,8453],{"class":3539},"\u002F\u002F Порушення MVVM!\n",[3175,8455,8456],{"class":3177,"line":4095},[3175,8457,3413],{"class":3181},[3175,8459,8460],{"class":3177,"line":4100},[3175,8461,3419],{"class":3181},[3150,8463,8464],{},"Це порушує MVVM: ViewModel не повинен знати про UI-логіку (навігацію). Крім того, це ускладнює тестування: потрібно mock Navigator.",[3512,8466,8468],{"id":8467},"рішення-1-events-через-property","Рішення 1: Events через Property",[3150,8470,8471,8472,8475],{},"ViewModel експонує ",[3172,8473,8474],{},"ObjectProperty\u003CNavigationRequest>",", Controller слухає зміни та викликає Navigator.",[3165,8477,8479],{"className":3167,"code":8478,"language":3169,"meta":3170,"style":3170},"\u002F\u002F NavigationRequest — immutable об'єкт з інформацією про навігацію\npublic class NavigationRequest {\n    private final String screenId;\n    private final Map\u003CString, Object> params;\n    private final NavigationType type; \u002F\u002F NAVIGATE, GO_BACK, OPEN_DIALOG\n    \n    public NavigationRequest(String screenId) {\n        this(screenId, Collections.emptyMap(), NavigationType.NAVIGATE);\n    }\n    \n    public NavigationRequest(String screenId, Map\u003CString, Object> params) {\n        this(screenId, params, NavigationType.NAVIGATE);\n    }\n    \n    public NavigationRequest(String screenId, Map\u003CString, Object> params, NavigationType type) {\n        this.screenId = screenId;\n        this.params = params;\n        this.type = type;\n    }\n    \n    \u002F\u002F Getters\n}\n\npublic enum NavigationType {\n    NAVIGATE, GO_BACK, OPEN_DIALOG\n}\n",[3172,8480,8481,8486,8497,8509,8531,8549,8553,8567,8592,8596,8600,8630,8645,8649,8653,8689,8699,8710,8722,8726,8730,8734,8738,8742,8753,8758],{"__ignoreMap":3170},[3175,8482,8483],{"class":3177,"line":3178},[3175,8484,8485],{"class":3539},"\u002F\u002F NavigationRequest — immutable об'єкт з інформацією про навігацію\n",[3175,8487,8488,8490,8492,8495],{"class":3177,"line":3189},[3175,8489,4028],{"class":3192},[3175,8491,4557],{"class":3192},[3175,8493,8494],{"class":3185}," NavigationRequest",[3175,8496,3213],{"class":3181},[3175,8498,8499,8501,8503,8505,8507],{"class":3177,"line":3206},[3175,8500,4576],{"class":3192},[3175,8502,4579],{"class":3192},[3175,8504,5887],{"class":3185},[3175,8506,4067],{"class":3222},[3175,8508,4588],{"class":3181},[3175,8510,8511,8513,8515,8517,8519,8521,8523,8525,8527,8529],{"class":3177,"line":3216},[3175,8512,4576],{"class":3192},[3175,8514,4579],{"class":3192},[3175,8516,6306],{"class":3185},[3175,8518,4165],{"class":3181},[3175,8520,4088],{"class":3185},[3175,8522,4159],{"class":3181},[3175,8524,4172],{"class":3185},[3175,8526,4175],{"class":3181},[3175,8528,4178],{"class":3222},[3175,8530,4588],{"class":3181},[3175,8532,8533,8535,8537,8540,8543,8546],{"class":3177,"line":3256},[3175,8534,4576],{"class":3192},[3175,8536,4579],{"class":3192},[3175,8538,8539],{"class":3185}," NavigationType",[3175,8541,8542],{"class":3222}," type",[3175,8544,8545],{"class":3181},"; ",[3175,8547,8548],{"class":3539},"\u002F\u002F NAVIGATE, GO_BACK, OPEN_DIALOG\n",[3175,8550,8551],{"class":3177,"line":3279},[3175,8552,4041],{"class":3181},[3175,8554,8555,8557,8559,8561,8563,8565],{"class":3177,"line":3285},[3175,8556,4655],{"class":3192},[3175,8558,8494],{"class":3199},[3175,8560,3235],{"class":3181},[3175,8562,4088],{"class":3185},[3175,8564,4067],{"class":3222},[3175,8566,3394],{"class":3181},[3175,8568,8569,8571,8573,8575,8577,8579,8582,8585,8587,8590],{"class":3177,"line":3305},[3175,8570,4678],{"class":3192},[3175,8572,4804],{"class":3181},[3175,8574,4807],{"class":3222},[3175,8576,3270],{"class":3181},[3175,8578,4812],{"class":3199},[3175,8580,8581],{"class":3181},"(), ",[3175,8583,8584],{"class":3222},"NavigationType",[3175,8586,3270],{"class":3181},[3175,8588,8589],{"class":3222},"NAVIGATE",[3175,8591,3561],{"class":3181},[3175,8593,8594],{"class":3177,"line":3326},[3175,8595,3413],{"class":3181},[3175,8597,8598],{"class":3177,"line":3331},[3175,8599,4041],{"class":3181},[3175,8601,8602,8604,8606,8608,8610,8612,8614,8616,8618,8620,8622,8624,8626,8628],{"class":3177,"line":3358},[3175,8603,4655],{"class":3192},[3175,8605,8494],{"class":3199},[3175,8607,3235],{"class":3181},[3175,8609,4088],{"class":3185},[3175,8611,4067],{"class":3222},[3175,8613,4159],{"class":3181},[3175,8615,4162],{"class":3185},[3175,8617,4165],{"class":3181},[3175,8619,4088],{"class":3185},[3175,8621,4159],{"class":3181},[3175,8623,4172],{"class":3185},[3175,8625,4175],{"class":3181},[3175,8627,4178],{"class":3222},[3175,8629,3394],{"class":3181},[3175,8631,8632,8634,8637,8639,8641,8643],{"class":3177,"line":3376},[3175,8633,4678],{"class":3192},[3175,8635,8636],{"class":3181},"(screenId, params, ",[3175,8638,8584],{"class":3222},[3175,8640,3270],{"class":3181},[3175,8642,8589],{"class":3222},[3175,8644,3561],{"class":3181},[3175,8646,8647],{"class":3177,"line":3397},[3175,8648,3413],{"class":3181},[3175,8650,8651],{"class":3177,"line":3410},[3175,8652,4041],{"class":3181},[3175,8654,8655,8657,8659,8661,8663,8665,8667,8669,8671,8673,8675,8677,8679,8681,8683,8685,8687],{"class":3177,"line":3416},[3175,8656,4655],{"class":3192},[3175,8658,8494],{"class":3199},[3175,8660,3235],{"class":3181},[3175,8662,4088],{"class":3185},[3175,8664,4067],{"class":3222},[3175,8666,4159],{"class":3181},[3175,8668,4162],{"class":3185},[3175,8670,4165],{"class":3181},[3175,8672,4088],{"class":3185},[3175,8674,4159],{"class":3181},[3175,8676,4172],{"class":3185},[3175,8678,4175],{"class":3181},[3175,8680,4178],{"class":3222},[3175,8682,4159],{"class":3181},[3175,8684,8584],{"class":3185},[3175,8686,8542],{"class":3222},[3175,8688,3394],{"class":3181},[3175,8690,8691,8693,8695,8697],{"class":3177,"line":4095},[3175,8692,4678],{"class":3192},[3175,8694,3270],{"class":3181},[3175,8696,6013],{"class":3222},[3175,8698,6016],{"class":3181},[3175,8700,8701,8703,8705,8707],{"class":3177,"line":4100},[3175,8702,4678],{"class":3192},[3175,8704,3270],{"class":3181},[3175,8706,4178],{"class":3222},[3175,8708,8709],{"class":3181}," = params;\n",[3175,8711,8712,8714,8716,8719],{"class":3177,"line":4105},[3175,8713,4678],{"class":3192},[3175,8715,3270],{"class":3181},[3175,8717,8718],{"class":3222},"type",[3175,8720,8721],{"class":3181}," = type;\n",[3175,8723,8724],{"class":3177,"line":4111},[3175,8725,3413],{"class":3181},[3175,8727,8728],{"class":3177,"line":4116},[3175,8729,4041],{"class":3181},[3175,8731,8732],{"class":3177,"line":4128},[3175,8733,7581],{"class":3539},[3175,8735,8736],{"class":3177,"line":4141},[3175,8737,3419],{"class":3181},[3175,8739,8740],{"class":3177,"line":4146},[3175,8741,3993],{"emptyLinePlaceholder":3992},[3175,8743,8744,8746,8749,8751],{"class":3177,"line":4183},[3175,8745,4028],{"class":3192},[3175,8747,8748],{"class":3192}," enum",[3175,8750,8539],{"class":3185},[3175,8752,3213],{"class":3181},[3175,8754,8755],{"class":3177,"line":4188},[3175,8756,8757],{"class":3181},"    NAVIGATE, GO_BACK, OPEN_DIALOG\n",[3175,8759,8760],{"class":3177,"line":4193},[3175,8761,3419],{"class":3181},[3150,8763,8764],{},"ViewModel:",[3165,8766,8768],{"className":3167,"code":8767,"language":3169,"meta":3170,"style":3170},"public class AudiobookFormViewModel {\n    private final ObjectProperty\u003CNavigationRequest> navigationRequest = new SimpleObjectProperty\u003C>();\n    \n    public void save() {\n        \u002F\u002F Збереження аудіокниги\n        audiobookService.save(audiobook);\n        \n        \u002F\u002F Запит на навігацію назад\n        navigationRequest.set(new NavigationRequest(null, null, NavigationType.GO_BACK));\n    }\n    \n    public ObjectProperty\u003CNavigationRequest> navigationRequestProperty() {\n        return navigationRequest;\n    }\n}\n",[3172,8769,8770,8780,8806,8810,8820,8824,8834,8838,8843,8878,8882,8886,8903,8910,8914],{"__ignoreMap":3170},[3175,8771,8772,8774,8776,8778],{"class":3177,"line":3178},[3175,8773,4028],{"class":3192},[3175,8775,4557],{"class":3192},[3175,8777,3320],{"class":3185},[3175,8779,3213],{"class":3181},[3175,8781,8782,8784,8786,8788,8790,8793,8795,8798,8800,8802,8804],{"class":3177,"line":3189},[3175,8783,4576],{"class":3192},[3175,8785,4579],{"class":3192},[3175,8787,7555],{"class":3185},[3175,8789,4165],{"class":3181},[3175,8791,8792],{"class":3185},"NavigationRequest",[3175,8794,4175],{"class":3181},[3175,8796,8797],{"class":3222},"navigationRequest",[3175,8799,3226],{"class":3181},[3175,8801,3229],{"class":3209},[3175,8803,7570],{"class":3185},[3175,8805,4715],{"class":3181},[3175,8807,8808],{"class":3177,"line":3206},[3175,8809,4041],{"class":3181},[3175,8811,8812,8814,8816,8818],{"class":3177,"line":3216},[3175,8813,4655],{"class":3192},[3175,8815,3196],{"class":3185},[3175,8817,8411],{"class":3199},[3175,8819,3203],{"class":3181},[3175,8821,8822],{"class":3177,"line":3256},[3175,8823,8418],{"class":3539},[3175,8825,8826,8828,8830,8832],{"class":3177,"line":3279},[3175,8827,8423],{"class":3222},[3175,8829,3270],{"class":3181},[3175,8831,8428],{"class":3199},[3175,8833,7694],{"class":3181},[3175,8835,8836],{"class":3177,"line":3285},[3175,8837,3282],{"class":3181},[3175,8839,8840],{"class":3177,"line":3305},[3175,8841,8842],{"class":3539},"        \u002F\u002F Запит на навігацію назад\n",[3175,8844,8845,8848,8850,8853,8855,8857,8859,8861,8863,8865,8867,8869,8871,8873,8876],{"class":3177,"line":3326},[3175,8846,8847],{"class":3222},"        navigationRequest",[3175,8849,3270],{"class":3181},[3175,8851,8852],{"class":3199},"set",[3175,8854,3235],{"class":3181},[3175,8856,3229],{"class":3209},[3175,8858,8494],{"class":3199},[3175,8860,3235],{"class":3181},[3175,8862,4936],{"class":3192},[3175,8864,4159],{"class":3181},[3175,8866,4936],{"class":3192},[3175,8868,4159],{"class":3181},[3175,8870,8584],{"class":3222},[3175,8872,3270],{"class":3181},[3175,8874,8875],{"class":3222},"GO_BACK",[3175,8877,3253],{"class":3181},[3175,8879,8880],{"class":3177,"line":3331},[3175,8881,3413],{"class":3181},[3175,8883,8884],{"class":3177,"line":3358},[3175,8885,4041],{"class":3181},[3175,8887,8888,8890,8892,8894,8896,8898,8901],{"class":3177,"line":3376},[3175,8889,4655],{"class":3192},[3175,8891,7555],{"class":3185},[3175,8893,4165],{"class":3181},[3175,8895,8792],{"class":3185},[3175,8897,4175],{"class":3181},[3175,8899,8900],{"class":3199},"navigationRequestProperty",[3175,8902,3203],{"class":3181},[3175,8904,8905,8907],{"class":3177,"line":3397},[3175,8906,5211],{"class":3209},[3175,8908,8909],{"class":3181}," navigationRequest;\n",[3175,8911,8912],{"class":3177,"line":3410},[3175,8913,3413],{"class":3181},[3175,8915,8916],{"class":3177,"line":3416},[3175,8917,3419],{"class":3181},[3150,8919,8920],{},"Controller:",[3165,8922,8924],{"className":3167,"code":8923,"language":3169,"meta":3170,"style":3170},"public class AudiobookFormController {\n    \n    @FXML\n    public void initialize() {\n        setupBindings();\n        setupNavigationListener();\n    }\n    \n    private void setupNavigationListener() {\n        viewModel.navigationRequestProperty().addListener((obs, old, request) -> {\n            if (request != null) {\n                handleNavigationRequest(request);\n                viewModel.navigationRequestProperty().set(null); \u002F\u002F Reset\n            }\n        });\n    }\n    \n    private void handleNavigationRequest(NavigationRequest request) {\n        switch (request.getType()) {\n            case NAVIGATE:\n                navigator.navigateTo(request.getScreenId(), request.getParams());\n                break;\n            case GO_BACK:\n                navigator.goBack();\n                break;\n            case OPEN_DIALOG:\n                navigator.openDialog(request.getScreenId(), request.getParams());\n                break;\n        }\n    }\n}\n",[3172,8925,8926,8936,8940,8946,8957,8964,8971,8975,8979,8990,9012,9023,9031,9054,9058,9063,9067,9071,9089,9106,9117,9146,9153,9162,9172,9178,9187,9213,9219,9223,9227],{"__ignoreMap":3170},[3175,8927,8928,8930,8932,8934],{"class":3177,"line":3178},[3175,8929,4028],{"class":3192},[3175,8931,4557],{"class":3192},[3175,8933,7929],{"class":3185},[3175,8935,3213],{"class":3181},[3175,8937,8938],{"class":3177,"line":3189},[3175,8939,4041],{"class":3181},[3175,8941,8942,8944],{"class":3177,"line":3206},[3175,8943,4647],{"class":3181},[3175,8945,3186],{"class":3185},[3175,8947,8948,8950,8952,8955],{"class":3177,"line":3216},[3175,8949,4655],{"class":3192},[3175,8951,3196],{"class":3185},[3175,8953,8954],{"class":3199}," initialize",[3175,8956,3203],{"class":3181},[3175,8958,8959,8962],{"class":3177,"line":3256},[3175,8960,8961],{"class":3199},"        setupBindings",[3175,8963,3276],{"class":3181},[3175,8965,8966,8969],{"class":3177,"line":3279},[3175,8967,8968],{"class":3199},"        setupNavigationListener",[3175,8970,3276],{"class":3181},[3175,8972,8973],{"class":3177,"line":3285},[3175,8974,3413],{"class":3181},[3175,8976,8977],{"class":3177,"line":3305},[3175,8978,4041],{"class":3181},[3175,8980,8981,8983,8985,8988],{"class":3177,"line":3326},[3175,8982,4576],{"class":3192},[3175,8984,3196],{"class":3185},[3175,8986,8987],{"class":3199}," setupNavigationListener",[3175,8989,3203],{"class":3181},[3175,8991,8992,8995,8997,8999,9001,9004,9007,9010],{"class":3177,"line":3331},[3175,8993,8994],{"class":3222},"        viewModel",[3175,8996,3270],{"class":3181},[3175,8998,8900],{"class":3199},[3175,9000,3241],{"class":3181},[3175,9002,9003],{"class":3199},"addListener",[3175,9005,9006],{"class":3181},"((obs, old, request) ",[3175,9008,9009],{"class":3192},"->",[3175,9011,3213],{"class":3181},[3175,9013,9014,9016,9019,9021],{"class":3177,"line":3358},[3175,9015,4922],{"class":3209},[3175,9017,9018],{"class":3181}," (request != ",[3175,9020,4936],{"class":3192},[3175,9022,3394],{"class":3181},[3175,9024,9025,9028],{"class":3177,"line":3376},[3175,9026,9027],{"class":3199},"                handleNavigationRequest",[3175,9029,9030],{"class":3181},"(request);\n",[3175,9032,9033,9036,9038,9040,9042,9044,9046,9048,9051],{"class":3177,"line":3397},[3175,9034,9035],{"class":3222},"                viewModel",[3175,9037,3270],{"class":3181},[3175,9039,8900],{"class":3199},[3175,9041,3241],{"class":3181},[3175,9043,8852],{"class":3199},[3175,9045,3235],{"class":3181},[3175,9047,4936],{"class":3192},[3175,9049,9050],{"class":3181},"); ",[3175,9052,9053],{"class":3539},"\u002F\u002F Reset\n",[3175,9055,9056],{"class":3177,"line":3410},[3175,9057,5013],{"class":3181},[3175,9059,9060],{"class":3177,"line":3416},[3175,9061,9062],{"class":3181},"        });\n",[3175,9064,9065],{"class":3177,"line":4095},[3175,9066,3413],{"class":3181},[3175,9068,9069],{"class":3177,"line":4100},[3175,9070,4041],{"class":3181},[3175,9072,9073,9075,9077,9080,9082,9084,9087],{"class":3177,"line":4105},[3175,9074,4576],{"class":3192},[3175,9076,3196],{"class":3185},[3175,9078,9079],{"class":3199}," handleNavigationRequest",[3175,9081,3235],{"class":3181},[3175,9083,8792],{"class":3185},[3175,9085,9086],{"class":3222}," request",[3175,9088,3394],{"class":3181},[3175,9090,9091,9094,9096,9099,9101,9104],{"class":3177,"line":4111},[3175,9092,9093],{"class":3209},"        switch",[3175,9095,3385],{"class":3181},[3175,9097,9098],{"class":3222},"request",[3175,9100,3270],{"class":3181},[3175,9102,9103],{"class":3199},"getType",[3175,9105,5661],{"class":3181},[3175,9107,9108,9111,9114],{"class":3177,"line":4116},[3175,9109,9110],{"class":3209},"            case",[3175,9112,9113],{"class":3181}," NAVIGATE",[3175,9115,9116],{"class":3209},":\n",[3175,9118,9119,9122,9124,9126,9128,9130,9132,9135,9137,9139,9141,9144],{"class":3177,"line":4128},[3175,9120,9121],{"class":3222},"                navigator",[3175,9123,3270],{"class":3181},[3175,9125,7225],{"class":3199},[3175,9127,3235],{"class":3181},[3175,9129,9098],{"class":3222},[3175,9131,3270],{"class":3181},[3175,9133,9134],{"class":3199},"getScreenId",[3175,9136,8581],{"class":3181},[3175,9138,9098],{"class":3222},[3175,9140,3270],{"class":3181},[3175,9142,9143],{"class":3199},"getParams",[3175,9145,4815],{"class":3181},[3175,9147,9148,9151],{"class":3177,"line":4141},[3175,9149,9150],{"class":3209},"                break",[3175,9152,4588],{"class":3181},[3175,9154,9155,9157,9160],{"class":3177,"line":4146},[3175,9156,9110],{"class":3209},[3175,9158,9159],{"class":3181}," GO_BACK",[3175,9161,9116],{"class":3209},[3175,9163,9164,9166,9168,9170],{"class":3177,"line":4183},[3175,9165,9121],{"class":3222},[3175,9167,3270],{"class":3181},[3175,9169,7470],{"class":3199},[3175,9171,3276],{"class":3181},[3175,9173,9174,9176],{"class":3177,"line":4188},[3175,9175,9150],{"class":3209},[3175,9177,4588],{"class":3181},[3175,9179,9180,9182,9185],{"class":3177,"line":4193},[3175,9181,9110],{"class":3209},[3175,9183,9184],{"class":3181}," OPEN_DIALOG",[3175,9186,9116],{"class":3209},[3175,9188,9189,9191,9193,9195,9197,9199,9201,9203,9205,9207,9209,9211],{"class":3177,"line":4199},[3175,9190,9121],{"class":3222},[3175,9192,3270],{"class":3181},[3175,9194,4252],{"class":3199},[3175,9196,3235],{"class":3181},[3175,9198,9098],{"class":3222},[3175,9200,3270],{"class":3181},[3175,9202,9134],{"class":3199},[3175,9204,8581],{"class":3181},[3175,9206,9098],{"class":3222},[3175,9208,3270],{"class":3181},[3175,9210,9143],{"class":3199},[3175,9212,4815],{"class":3181},[3175,9214,9215,9217],{"class":3177,"line":4204},[3175,9216,9150],{"class":3209},[3175,9218,4588],{"class":3181},[3175,9220,9221],{"class":3177,"line":4216},[3175,9222,5158],{"class":3181},[3175,9224,9225],{"class":3177,"line":4227},[3175,9226,3413],{"class":3181},[3175,9228,9229],{"class":3177,"line":4232},[3175,9230,3419],{"class":3181},[3150,9232,9233,9235,9236,9238],{},[3157,9234,3582],{}," ViewModel не залежить від Navigator, легко тестувати (перевірити, що ",[3172,9237,8797],{}," встановлено).",[3150,9240,9241,9243],{},[3157,9242,3598],{}," Додатковий boilerplate-код (NavigationRequest, listener у Controller).",[3512,9245,9247],{"id":9246},"рішення-2-callback-через-інтерфейс","Рішення 2: Callback через інтерфейс",[3150,9249,9250],{},"ViewModel приймає callback-інтерфейс у конструкторі. Controller передає лямбду, що викликає Navigator.",[3165,9252,9254],{"className":3167,"code":9253,"language":3169,"meta":3170,"style":3170},"\u002F\u002F Інтерфейс для callback\npublic interface NavigationCallback {\n    void navigateTo(String screenId);\n    void goBack();\n}\n\n\u002F\u002F ViewModel\npublic class AudiobookFormViewModel {\n    private final NavigationCallback navigationCallback;\n    \n    @Inject\n    public AudiobookFormViewModel(\n        AudiobookService service,\n        @Assisted NavigationCallback navigationCallback\n    ) {\n        this.audiobookService = service;\n        this.navigationCallback = navigationCallback;\n    }\n    \n    public void save() {\n        audiobookService.save(audiobook);\n        navigationCallback.goBack();\n    }\n}\n\n\u002F\u002F Controller\npublic class AudiobookFormController {\n    \n    @Inject\n    public AudiobookFormController(AudiobookFormViewModelFactory factory, Navigator navigator) {\n        this.viewModel = factory.create(new NavigationCallback() {\n            @Override\n            public void navigateTo(String screenId) {\n                navigator.navigateTo(screenId);\n            }\n            \n            @Override\n            public void goBack() {\n                navigator.goBack();\n            }\n        });\n    }\n}\n",[3172,9255,9256,9261,9272,9286,9294,9298,9302,9306,9316,9329,9333,9339,9347,9356,9367,9371,9383,9395,9399,9403,9413,9423,9434,9438,9442,9446,9450,9460,9464,9470,9490,9514,9521,9538,9548,9552,9556,9562,9572,9582,9586,9590,9594],{"__ignoreMap":3170},[3175,9257,9258],{"class":3177,"line":3178},[3175,9259,9260],{"class":3539},"\u002F\u002F Інтерфейс для callback\n",[3175,9262,9263,9265,9267,9270],{"class":3177,"line":3189},[3175,9264,4028],{"class":3192},[3175,9266,4031],{"class":3192},[3175,9268,9269],{"class":3185}," NavigationCallback",[3175,9271,3213],{"class":3181},[3175,9273,9274,9276,9278,9280,9282,9284],{"class":3177,"line":3206},[3175,9275,4080],{"class":3185},[3175,9277,4083],{"class":3199},[3175,9279,3235],{"class":3181},[3175,9281,4088],{"class":3185},[3175,9283,4067],{"class":3222},[3175,9285,3561],{"class":3181},[3175,9287,9288,9290,9292],{"class":3177,"line":3216},[3175,9289,4080],{"class":3185},[3175,9291,4390],{"class":3199},[3175,9293,3276],{"class":3181},[3175,9295,9296],{"class":3177,"line":3256},[3175,9297,3419],{"class":3181},[3175,9299,9300],{"class":3177,"line":3279},[3175,9301,3993],{"emptyLinePlaceholder":3992},[3175,9303,9304],{"class":3177,"line":3285},[3175,9305,8108],{"class":3539},[3175,9307,9308,9310,9312,9314],{"class":3177,"line":3305},[3175,9309,4028],{"class":3192},[3175,9311,4557],{"class":3192},[3175,9313,3320],{"class":3185},[3175,9315,3213],{"class":3181},[3175,9317,9318,9320,9322,9324,9327],{"class":3177,"line":3326},[3175,9319,4576],{"class":3192},[3175,9321,4579],{"class":3192},[3175,9323,9269],{"class":3185},[3175,9325,9326],{"class":3222}," navigationCallback",[3175,9328,4588],{"class":3181},[3175,9330,9331],{"class":3177,"line":3331},[3175,9332,4041],{"class":3181},[3175,9334,9335,9337],{"class":3177,"line":3358},[3175,9336,4647],{"class":3181},[3175,9338,4650],{"class":3185},[3175,9340,9341,9343,9345],{"class":3177,"line":3376},[3175,9342,4655],{"class":3192},[3175,9344,3320],{"class":3199},[3175,9346,4976],{"class":3181},[3175,9348,9349,9351,9354],{"class":3177,"line":3397},[3175,9350,8157],{"class":3185},[3175,9352,9353],{"class":3222}," service",[3175,9355,8152],{"class":3181},[3175,9357,9358,9360,9362,9364],{"class":3177,"line":3410},[3175,9359,8137],{"class":3181},[3175,9361,8140],{"class":3185},[3175,9363,9269],{"class":3185},[3175,9365,9366],{"class":3222}," navigationCallback\n",[3175,9368,9369],{"class":3177,"line":3416},[3175,9370,8165],{"class":3181},[3175,9372,9373,9375,9377,9380],{"class":3177,"line":4095},[3175,9374,4678],{"class":3192},[3175,9376,3270],{"class":3181},[3175,9378,9379],{"class":3222},"audiobookService",[3175,9381,9382],{"class":3181}," = service;\n",[3175,9384,9385,9387,9389,9392],{"class":3177,"line":4100},[3175,9386,4678],{"class":3192},[3175,9388,3270],{"class":3181},[3175,9390,9391],{"class":3222},"navigationCallback",[3175,9393,9394],{"class":3181}," = navigationCallback;\n",[3175,9396,9397],{"class":3177,"line":4105},[3175,9398,3413],{"class":3181},[3175,9400,9401],{"class":3177,"line":4111},[3175,9402,4041],{"class":3181},[3175,9404,9405,9407,9409,9411],{"class":3177,"line":4116},[3175,9406,4655],{"class":3192},[3175,9408,3196],{"class":3185},[3175,9410,8411],{"class":3199},[3175,9412,3203],{"class":3181},[3175,9414,9415,9417,9419,9421],{"class":3177,"line":4128},[3175,9416,8423],{"class":3222},[3175,9418,3270],{"class":3181},[3175,9420,8428],{"class":3199},[3175,9422,7694],{"class":3181},[3175,9424,9425,9428,9430,9432],{"class":3177,"line":4141},[3175,9426,9427],{"class":3222},"        navigationCallback",[3175,9429,3270],{"class":3181},[3175,9431,7470],{"class":3199},[3175,9433,3276],{"class":3181},[3175,9435,9436],{"class":3177,"line":4146},[3175,9437,3413],{"class":3181},[3175,9439,9440],{"class":3177,"line":4183},[3175,9441,3419],{"class":3181},[3175,9443,9444],{"class":3177,"line":4188},[3175,9445,3993],{"emptyLinePlaceholder":3992},[3175,9447,9448],{"class":3177,"line":4193},[3175,9449,7751],{"class":3539},[3175,9451,9452,9454,9456,9458],{"class":3177,"line":4199},[3175,9453,4028],{"class":3192},[3175,9455,4557],{"class":3192},[3175,9457,7929],{"class":3185},[3175,9459,3213],{"class":3181},[3175,9461,9462],{"class":3177,"line":4204},[3175,9463,4041],{"class":3181},[3175,9465,9466,9468],{"class":3177,"line":4216},[3175,9467,4647],{"class":3181},[3175,9469,4650],{"class":3185},[3175,9471,9472,9474,9476,9478,9480,9482,9484,9486,9488],{"class":3177,"line":4227},[3175,9473,4655],{"class":3192},[3175,9475,7929],{"class":3199},[3175,9477,3235],{"class":3181},[3175,9479,8243],{"class":3185},[3175,9481,8229],{"class":3222},[3175,9483,4159],{"class":3181},[3175,9485,7163],{"class":3185},[3175,9487,7133],{"class":3222},[3175,9489,3394],{"class":3181},[3175,9491,9492,9494,9496,9498,9500,9502,9504,9506,9508,9510,9512],{"class":3177,"line":4232},[3175,9493,4678],{"class":3192},[3175,9495,3270],{"class":3181},[3175,9497,7176],{"class":3222},[3175,9499,3226],{"class":3181},[3175,9501,8261],{"class":3222},[3175,9503,3270],{"class":3181},[3175,9505,8266],{"class":3199},[3175,9507,3235],{"class":3181},[3175,9509,3229],{"class":3209},[3175,9511,9269],{"class":3199},[3175,9513,3203],{"class":3181},[3175,9515,9516,9519],{"class":3177,"line":4263},[3175,9517,9518],{"class":3181},"            @",[3175,9520,4780],{"class":3185},[3175,9522,9523,9526,9528,9530,9532,9534,9536],{"class":3177,"line":4268},[3175,9524,9525],{"class":3192},"            public",[3175,9527,3196],{"class":3185},[3175,9529,4083],{"class":3199},[3175,9531,3235],{"class":3181},[3175,9533,4088],{"class":3185},[3175,9535,4067],{"class":3222},[3175,9537,3394],{"class":3181},[3175,9539,9540,9542,9544,9546],{"class":3177,"line":4273},[3175,9541,9121],{"class":3222},[3175,9543,3270],{"class":3181},[3175,9545,7225],{"class":3199},[3175,9547,4893],{"class":3181},[3175,9549,9550],{"class":3177,"line":4279},[3175,9551,5013],{"class":3181},[3175,9553,9554],{"class":3177,"line":4284},[3175,9555,4912],{"class":3181},[3175,9557,9558,9560],{"class":3177,"line":4295},[3175,9559,9518],{"class":3181},[3175,9561,4780],{"class":3185},[3175,9563,9564,9566,9568,9570],{"class":3177,"line":4306},[3175,9565,9525],{"class":3192},[3175,9567,3196],{"class":3185},[3175,9569,4390],{"class":3199},[3175,9571,3203],{"class":3181},[3175,9573,9574,9576,9578,9580],{"class":3177,"line":4316},[3175,9575,9121],{"class":3222},[3175,9577,3270],{"class":3181},[3175,9579,7470],{"class":3199},[3175,9581,3276],{"class":3181},[3175,9583,9584],{"class":3177,"line":4321},[3175,9585,5013],{"class":3181},[3175,9587,9588],{"class":3177,"line":4364},[3175,9589,9062],{"class":3181},[3175,9591,9592],{"class":3177,"line":4369},[3175,9593,3413],{"class":3181},[3175,9595,9596],{"class":3177,"line":4374},[3175,9597,3419],{"class":3181},[3150,9599,9600,9602],{},[3157,9601,3582],{}," Простіше, ніж Events, явний контракт (інтерфейс).",[3150,9604,9605,9607,9608,9611],{},[3157,9606,3598],{}," ViewModel залежить від ",[3172,9609,9610],{},"NavigationCallback"," (хоча це абстракція, не конкретна реалізація).",[9613,9614,9615,9618],"warning",{},[3157,9616,9617],{},"Не передавайте Navigator безпосередньо у ViewModel."," Це порушує MVVM та ускладнює тестування. Використовуйте Events (Рішення 1) або Callbacks (Рішення 2) для непрямої комунікації.",[3502,9620],{},[3145,9622,9624],{"id":9623},"модальні-діалоги-з-поверненням-результату","Модальні діалоги з поверненням результату",[3150,9626,9627],{},"Діалоги — це окремі екрани, що відкриваються поверх головного вікна та повертають результат (наприклад, вибраний елемент, підтвердження).",[3512,9629,9631],{"id":9630},"інтерфейс-dialogcontroller","Інтерфейс DialogController",[3165,9633,9635],{"className":3167,"code":9634,"language":3169,"meta":3170,"style":3170},"package dev.kostyl.audiobook.infrastructure.navigation;\n\nimport java.util.Optional;\n\npublic interface DialogController\u003CT> {\n    \n    \u002F**\n     * Отримання результату діалогу.\n     * \n     * @return результат (Optional.empty() якщо діалог скасовано)\n     *\u002F\n    Optional\u003CT> getResult();\n}\n",[3172,9636,9637,9643,9647,9653,9657,9673,9677,9681,9686,9690,9699,9703,9718],{"__ignoreMap":3170},[3175,9638,9639,9641],{"class":3177,"line":3178},[3175,9640,3984],{"class":3192},[3175,9642,3987],{"class":3181},[3175,9644,9645],{"class":3177,"line":3189},[3175,9646,3993],{"emptyLinePlaceholder":3992},[3175,9648,9649,9651],{"class":3177,"line":3206},[3175,9650,3998],{"class":3192},[3175,9652,4019],{"class":3181},[3175,9654,9655],{"class":3177,"line":3216},[3175,9656,3993],{"emptyLinePlaceholder":3992},[3175,9658,9659,9661,9663,9666,9668,9670],{"class":3177,"line":3256},[3175,9660,4028],{"class":3192},[3175,9662,4031],{"class":3192},[3175,9664,9665],{"class":3185}," DialogController",[3175,9667,4165],{"class":3181},[3175,9669,4238],{"class":3185},[3175,9671,9672],{"class":3181},"> {\n",[3175,9674,9675],{"class":3177,"line":3279},[3175,9676,4041],{"class":3181},[3175,9678,9679],{"class":3177,"line":3285},[3175,9680,4046],{"class":3539},[3175,9682,9683],{"class":3177,"line":3305},[3175,9684,9685],{"class":3539},"     * Отримання результату діалогу.\n",[3175,9687,9688],{"class":3177,"line":3326},[3175,9689,4056],{"class":3539},[3175,9691,9692,9694,9696],{"class":3177,"line":3331},[3175,9693,4061],{"class":3539},[3175,9695,4221],{"class":3192},[3175,9697,9698],{"class":3539}," результат (Optional.empty() якщо діалог скасовано)\n",[3175,9700,9701],{"class":3177,"line":3358},[3175,9702,4075],{"class":3539},[3175,9704,9705,9708,9710,9712,9714,9716],{"class":3177,"line":3376},[3175,9706,9707],{"class":3185},"    Optional",[3175,9709,4165],{"class":3181},[3175,9711,4238],{"class":3185},[3175,9713,4175],{"class":3181},[3175,9715,5550],{"class":3199},[3175,9717,3276],{"class":3181},[3175,9719,9720],{"class":3177,"line":3397},[3175,9721,3419],{"class":3181},[3512,9723,9725],{"id":9724},"приклад-діалог-підтвердження-видалення","Приклад: Діалог підтвердження видалення",[3150,9727,9728],{},[3157,9729,9730],{},"confirm-delete-dialog.fxml:",[3165,9732,9736],{"className":9733,"code":9734,"language":9735,"meta":3170,"style":3170},"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\n\u003CVBox xmlns:fx=\"http:\u002F\u002Fjavafx.com\u002Ffxml\"\n      fx:controller=\"dev.kostyl.audiobook.controller.ConfirmDeleteDialogController\"\n      spacing=\"20\" padding=\"20\" prefWidth=\"400\">\n    \n    \u003CLabel text=\"Are you sure you want to delete this audiobook?\"\n           style=\"-fx-font-size: 14px;\"\u002F>\n    \n    \u003CLabel fx:id=\"audiobookTitleLabel\"\n           style=\"-fx-font-weight: bold; -fx-font-size: 16px;\"\u002F>\n    \n    \u003CHBox spacing=\"10\" alignment=\"CENTER_RIGHT\">\n        \u003CButton fx:id=\"cancelButton\" text=\"Cancel\" onAction=\"#onCancelClicked\"\u002F>\n        \u003CButton fx:id=\"confirmButton\" text=\"Delete\" \n                styleClass=\"danger-button\" onAction=\"#onConfirmClicked\"\u002F>\n    \u003C\u002FHBox>\n    \n\u003C\u002FVBox>\n","xml",[3172,9737,9738,9769,9773,9787,9800,9804,9819,9829,9857,9861,9876,9889,9893,9907,9918,9922,9947,9979,10002,10021,10030,10034],{"__ignoreMap":3170},[3175,9739,9740,9744,9747,9751,9754,9758,9761,9763,9766],{"class":3177,"line":3178},[3175,9741,9743],{"class":9742},"s0P7L","\u003C?",[3175,9745,9735],{"class":9746},"sKtos",[3175,9748,9750],{"class":9749},"sa4r_"," version",[3175,9752,9753],{"class":3181},"=",[3175,9755,9757],{"class":9756},"su9tN","\"1.0\"",[3175,9759,9760],{"class":9749}," encoding",[3175,9762,9753],{"class":3181},[3175,9764,9765],{"class":9756},"\"UTF-8\"",[3175,9767,9768],{"class":9742},"?>\n",[3175,9770,9771],{"class":3177,"line":3189},[3175,9772,3993],{"emptyLinePlaceholder":3992},[3175,9774,9775,9777,9779,9782,9785],{"class":3177,"line":3206},[3175,9776,9743],{"class":9742},[3175,9778,3998],{"class":9746},[3175,9780,9781],{"class":9749}," javafx",[3175,9783,9784],{"class":3181},".scene.control.*",[3175,9786,9768],{"class":9742},[3175,9788,9789,9791,9793,9795,9798],{"class":3177,"line":3216},[3175,9790,9743],{"class":9742},[3175,9792,3998],{"class":9746},[3175,9794,9781],{"class":9749},[3175,9796,9797],{"class":3181},".scene.layout.*",[3175,9799,9768],{"class":9742},[3175,9801,9802],{"class":3177,"line":3256},[3175,9803,3993],{"emptyLinePlaceholder":3992},[3175,9805,9806,9808,9811,9814,9816],{"class":3177,"line":3279},[3175,9807,4165],{"class":9742},[3175,9809,9810],{"class":9746},"VBox",[3175,9812,9813],{"class":9749}," xmlns:fx",[3175,9815,9753],{"class":3181},[3175,9817,9818],{"class":9756},"\"http:\u002F\u002Fjavafx.com\u002Ffxml\"\n",[3175,9820,9821,9824,9826],{"class":3177,"line":3285},[3175,9822,9823],{"class":9749},"      fx:controller",[3175,9825,9753],{"class":3181},[3175,9827,9828],{"class":9756},"\"dev.kostyl.audiobook.controller.ConfirmDeleteDialogController\"\n",[3175,9830,9831,9834,9836,9839,9842,9844,9846,9849,9851,9854],{"class":3177,"line":3305},[3175,9832,9833],{"class":9749},"      spacing",[3175,9835,9753],{"class":3181},[3175,9837,9838],{"class":9756},"\"20\"",[3175,9840,9841],{"class":9749}," padding",[3175,9843,9753],{"class":3181},[3175,9845,9838],{"class":9756},[3175,9847,9848],{"class":9749}," prefWidth",[3175,9850,9753],{"class":3181},[3175,9852,9853],{"class":9756},"\"400\"",[3175,9855,9856],{"class":9742},">\n",[3175,9858,9859],{"class":3177,"line":3326},[3175,9860,4041],{"class":3181},[3175,9862,9863,9865,9868,9871,9873],{"class":3177,"line":3331},[3175,9864,4235],{"class":9742},[3175,9866,9867],{"class":9746},"Label",[3175,9869,9870],{"class":9749}," text",[3175,9872,9753],{"class":3181},[3175,9874,9875],{"class":9756},"\"Are you sure you want to delete this audiobook?\"\n",[3175,9877,9878,9881,9883,9886],{"class":3177,"line":3358},[3175,9879,9880],{"class":9749},"           style",[3175,9882,9753],{"class":3181},[3175,9884,9885],{"class":9756},"\"-fx-font-size: 14px;\"",[3175,9887,9888],{"class":9742},"\u002F>\n",[3175,9890,9891],{"class":3177,"line":3376},[3175,9892,4041],{"class":3181},[3175,9894,9895,9897,9899,9902,9904],{"class":3177,"line":3397},[3175,9896,4235],{"class":9742},[3175,9898,9867],{"class":9746},[3175,9900,9901],{"class":9749}," fx:id",[3175,9903,9753],{"class":3181},[3175,9905,9906],{"class":9756},"\"audiobookTitleLabel\"\n",[3175,9908,9909,9911,9913,9916],{"class":3177,"line":3410},[3175,9910,9880],{"class":9749},[3175,9912,9753],{"class":3181},[3175,9914,9915],{"class":9756},"\"-fx-font-weight: bold; -fx-font-size: 16px;\"",[3175,9917,9888],{"class":9742},[3175,9919,9920],{"class":3177,"line":3416},[3175,9921,4041],{"class":3181},[3175,9923,9924,9926,9929,9932,9934,9937,9940,9942,9945],{"class":3177,"line":4095},[3175,9925,4235],{"class":9742},[3175,9927,9928],{"class":9746},"HBox",[3175,9930,9931],{"class":9749}," spacing",[3175,9933,9753],{"class":3181},[3175,9935,9936],{"class":9756},"\"10\"",[3175,9938,9939],{"class":9749}," alignment",[3175,9941,9753],{"class":3181},[3175,9943,9944],{"class":9756},"\"CENTER_RIGHT\"",[3175,9946,9856],{"class":9742},[3175,9948,9949,9952,9955,9957,9959,9962,9964,9966,9969,9972,9974,9977],{"class":3177,"line":4100},[3175,9950,9951],{"class":9742},"        \u003C",[3175,9953,9954],{"class":9746},"Button",[3175,9956,9901],{"class":9749},[3175,9958,9753],{"class":3181},[3175,9960,9961],{"class":9756},"\"cancelButton\"",[3175,9963,9870],{"class":9749},[3175,9965,9753],{"class":3181},[3175,9967,9968],{"class":9756},"\"Cancel\"",[3175,9970,9971],{"class":9749}," onAction",[3175,9973,9753],{"class":3181},[3175,9975,9976],{"class":9756},"\"#onCancelClicked\"",[3175,9978,9888],{"class":9742},[3175,9980,9981,9983,9985,9987,9989,9992,9994,9996,9999],{"class":3177,"line":4105},[3175,9982,9951],{"class":9742},[3175,9984,9954],{"class":9746},[3175,9986,9901],{"class":9749},[3175,9988,9753],{"class":3181},[3175,9990,9991],{"class":9756},"\"confirmButton\"",[3175,9993,9870],{"class":9749},[3175,9995,9753],{"class":3181},[3175,9997,9998],{"class":9756},"\"Delete\"",[3175,10000,10001],{"class":3181}," \n",[3175,10003,10004,10007,10009,10012,10014,10016,10019],{"class":3177,"line":4111},[3175,10005,10006],{"class":9749},"                styleClass",[3175,10008,9753],{"class":3181},[3175,10010,10011],{"class":9756},"\"danger-button\"",[3175,10013,9971],{"class":9749},[3175,10015,9753],{"class":3181},[3175,10017,10018],{"class":9756},"\"#onConfirmClicked\"",[3175,10020,9888],{"class":9742},[3175,10022,10023,10026,10028],{"class":3177,"line":4116},[3175,10024,10025],{"class":9742},"    \u003C\u002F",[3175,10027,9928],{"class":9746},[3175,10029,9856],{"class":9742},[3175,10031,10032],{"class":3177,"line":4128},[3175,10033,4041],{"class":3181},[3175,10035,10036,10039,10041],{"class":3177,"line":4141},[3175,10037,10038],{"class":9742},"\u003C\u002F",[3175,10040,9810],{"class":9746},[3175,10042,9856],{"class":9742},[3150,10044,10045],{},[3157,10046,10047],{},"ConfirmDeleteDialogController:",[3165,10049,10051],{"className":3167,"code":10050,"language":3169,"meta":3170,"style":3170},"package dev.kostyl.audiobook.controller;\n\nimport dev.kostyl.audiobook.infrastructure.navigation.DialogController;\nimport javafx.fxml.FXML;\nimport javafx.scene.control.Button;\nimport javafx.scene.control.Label;\nimport javafx.stage.Stage;\n\nimport java.util.Optional;\n\npublic class ConfirmDeleteDialogController implements DialogController\u003CBoolean> {\n    \n    @FXML private Label audiobookTitleLabel;\n    @FXML private Button cancelButton;\n    @FXML private Button confirmButton;\n    \n    private Boolean result;\n    \n    @FXML\n    public void initialize() {\n        \u002F\u002F Можна встановити title через параметри Navigator\n        audiobookTitleLabel.setText(\"Selected audiobook\");\n    }\n    \n    @FXML\n    private void onConfirmClicked() {\n        result = true;\n        closeDialog();\n    }\n    \n    @FXML\n    private void onCancelClicked() {\n        result = false;\n        closeDialog();\n    }\n    \n    private void closeDialog() {\n        Stage stage = (Stage) cancelButton.getScene().getWindow();\n        stage.close();\n    }\n    \n    @Override\n    public Optional\u003CBoolean> getResult() {\n        return Optional.ofNullable(result);\n    }\n}\n",[3172,10052,10053,10060,10064,10071,10078,10085,10092,10098,10102,10108,10112,10131,10135,10153,10169,10184,10188,10200,10204,10210,10220,10225,10242,10246,10250,10256,10267,10277,10284,10288,10292,10298,10309,10318,10324,10328,10332,10343,10364,10375,10379,10383,10389,10405,10418,10422],{"__ignoreMap":3170},[3175,10054,10055,10057],{"class":3177,"line":3178},[3175,10056,3984],{"class":3192},[3175,10058,10059],{"class":3181}," dev.kostyl.audiobook.controller;\n",[3175,10061,10062],{"class":3177,"line":3189},[3175,10063,3993],{"emptyLinePlaceholder":3992},[3175,10065,10066,10068],{"class":3177,"line":3206},[3175,10067,3998],{"class":3192},[3175,10069,10070],{"class":3181}," dev.kostyl.audiobook.infrastructure.navigation.DialogController;\n",[3175,10072,10073,10075],{"class":3177,"line":3216},[3175,10074,3998],{"class":3192},[3175,10076,10077],{"class":3181}," javafx.fxml.FXML;\n",[3175,10079,10080,10082],{"class":3177,"line":3256},[3175,10081,3998],{"class":3192},[3175,10083,10084],{"class":3181}," javafx.scene.control.Button;\n",[3175,10086,10087,10089],{"class":3177,"line":3279},[3175,10088,3998],{"class":3192},[3175,10090,10091],{"class":3181}," javafx.scene.control.Label;\n",[3175,10093,10094,10096],{"class":3177,"line":3285},[3175,10095,3998],{"class":3192},[3175,10097,4521],{"class":3181},[3175,10099,10100],{"class":3177,"line":3305},[3175,10101,3993],{"emptyLinePlaceholder":3992},[3175,10103,10104,10106],{"class":3177,"line":3326},[3175,10105,3998],{"class":3192},[3175,10107,4019],{"class":3181},[3175,10109,10110],{"class":3177,"line":3331},[3175,10111,3993],{"emptyLinePlaceholder":3992},[3175,10113,10114,10116,10118,10121,10123,10125,10127,10129],{"class":3177,"line":3358},[3175,10115,4028],{"class":3192},[3175,10117,4557],{"class":3192},[3175,10119,10120],{"class":3185}," ConfirmDeleteDialogController",[3175,10122,4563],{"class":3192},[3175,10124,9665],{"class":3185},[3175,10126,4165],{"class":3181},[3175,10128,7363],{"class":3185},[3175,10130,9672],{"class":3181},[3175,10132,10133],{"class":3177,"line":3376},[3175,10134,4041],{"class":3181},[3175,10136,10137,10139,10142,10145,10148,10151],{"class":3177,"line":3397},[3175,10138,4647],{"class":3181},[3175,10140,10141],{"class":3185},"FXML",[3175,10143,10144],{"class":3192}," private",[3175,10146,10147],{"class":3185}," Label",[3175,10149,10150],{"class":3222}," audiobookTitleLabel",[3175,10152,4588],{"class":3181},[3175,10154,10155,10157,10159,10161,10164,10167],{"class":3177,"line":3410},[3175,10156,4647],{"class":3181},[3175,10158,10141],{"class":3185},[3175,10160,10144],{"class":3192},[3175,10162,10163],{"class":3185}," Button",[3175,10165,10166],{"class":3222}," cancelButton",[3175,10168,4588],{"class":3181},[3175,10170,10171,10173,10175,10177,10179,10182],{"class":3177,"line":3416},[3175,10172,4647],{"class":3181},[3175,10174,10141],{"class":3185},[3175,10176,10144],{"class":3192},[3175,10178,10163],{"class":3185},[3175,10180,10181],{"class":3222}," confirmButton",[3175,10183,4588],{"class":3181},[3175,10185,10186],{"class":3177,"line":4095},[3175,10187,4041],{"class":3181},[3175,10189,10190,10192,10195,10198],{"class":3177,"line":4100},[3175,10191,4576],{"class":3192},[3175,10193,10194],{"class":3185}," Boolean",[3175,10196,10197],{"class":3222}," result",[3175,10199,4588],{"class":3181},[3175,10201,10202],{"class":3177,"line":4105},[3175,10203,4041],{"class":3181},[3175,10205,10206,10208],{"class":3177,"line":4111},[3175,10207,4647],{"class":3181},[3175,10209,3186],{"class":3185},[3175,10211,10212,10214,10216,10218],{"class":3177,"line":4116},[3175,10213,4655],{"class":3192},[3175,10215,3196],{"class":3185},[3175,10217,8954],{"class":3199},[3175,10219,3203],{"class":3181},[3175,10221,10222],{"class":3177,"line":4128},[3175,10223,10224],{"class":3539},"        \u002F\u002F Можна встановити title через параметри Navigator\n",[3175,10226,10227,10230,10232,10235,10237,10240],{"class":3177,"line":4141},[3175,10228,10229],{"class":3222},"        audiobookTitleLabel",[3175,10231,3270],{"class":3181},[3175,10233,10234],{"class":3199},"setText",[3175,10236,3235],{"class":3181},[3175,10238,10239],{"class":3249},"\"Selected audiobook\"",[3175,10241,3561],{"class":3181},[3175,10243,10244],{"class":3177,"line":4146},[3175,10245,3413],{"class":3181},[3175,10247,10248],{"class":3177,"line":4183},[3175,10249,4041],{"class":3181},[3175,10251,10252,10254],{"class":3177,"line":4188},[3175,10253,4647],{"class":3181},[3175,10255,3186],{"class":3185},[3175,10257,10258,10260,10262,10265],{"class":3177,"line":4193},[3175,10259,4576],{"class":3192},[3175,10261,3196],{"class":3185},[3175,10263,10264],{"class":3199}," onConfirmClicked",[3175,10266,3203],{"class":3181},[3175,10268,10269,10272,10275],{"class":3177,"line":4199},[3175,10270,10271],{"class":3181},"        result = ",[3175,10273,10274],{"class":3192},"true",[3175,10276,4588],{"class":3181},[3175,10278,10279,10282],{"class":3177,"line":4204},[3175,10280,10281],{"class":3199},"        closeDialog",[3175,10283,3276],{"class":3181},[3175,10285,10286],{"class":3177,"line":4216},[3175,10287,3413],{"class":3181},[3175,10289,10290],{"class":3177,"line":4227},[3175,10291,4041],{"class":3181},[3175,10293,10294,10296],{"class":3177,"line":4232},[3175,10295,4647],{"class":3181},[3175,10297,3186],{"class":3185},[3175,10299,10300,10302,10304,10307],{"class":3177,"line":4263},[3175,10301,4576],{"class":3192},[3175,10303,3196],{"class":3185},[3175,10305,10306],{"class":3199}," onCancelClicked",[3175,10308,3203],{"class":3181},[3175,10310,10311,10313,10316],{"class":3177,"line":4268},[3175,10312,10271],{"class":3181},[3175,10314,10315],{"class":3192},"false",[3175,10317,4588],{"class":3181},[3175,10319,10320,10322],{"class":3177,"line":4273},[3175,10321,10281],{"class":3199},[3175,10323,3276],{"class":3181},[3175,10325,10326],{"class":3177,"line":4279},[3175,10327,3413],{"class":3181},[3175,10329,10330],{"class":3177,"line":4284},[3175,10331,4041],{"class":3181},[3175,10333,10334,10336,10338,10341],{"class":3177,"line":4295},[3175,10335,4576],{"class":3192},[3175,10337,3196],{"class":3185},[3175,10339,10340],{"class":3199}," closeDialog",[3175,10342,3203],{"class":3181},[3175,10344,10345,10347,10349,10351,10354,10356,10358,10360,10362],{"class":3177,"line":4306},[3175,10346,3334],{"class":3185},[3175,10348,3337],{"class":3222},[3175,10350,3340],{"class":3181},[3175,10352,10353],{"class":3222},"cancelButton",[3175,10355,3270],{"class":3181},[3175,10357,3348],{"class":3199},[3175,10359,3241],{"class":3181},[3175,10361,3353],{"class":3199},[3175,10363,3276],{"class":3181},[3175,10365,10366,10368,10370,10373],{"class":3177,"line":4316},[3175,10367,3361],{"class":3222},[3175,10369,3270],{"class":3181},[3175,10371,10372],{"class":3199},"close",[3175,10374,3276],{"class":3181},[3175,10376,10377],{"class":3177,"line":4321},[3175,10378,3413],{"class":3181},[3175,10380,10381],{"class":3177,"line":4364},[3175,10382,4041],{"class":3181},[3175,10384,10385,10387],{"class":3177,"line":4369},[3175,10386,4647],{"class":3181},[3175,10388,4780],{"class":3185},[3175,10390,10391,10393,10395,10397,10399,10401,10403],{"class":3177,"line":4374},[3175,10392,4655],{"class":3192},[3175,10394,5533],{"class":3185},[3175,10396,4165],{"class":3181},[3175,10398,7363],{"class":3185},[3175,10400,4175],{"class":3181},[3175,10402,5550],{"class":3199},[3175,10404,3203],{"class":3181},[3175,10406,10407,10409,10411,10413,10415],{"class":3177,"line":4380},[3175,10408,5211],{"class":3209},[3175,10410,5533],{"class":3222},[3175,10412,3270],{"class":3181},[3175,10414,5538],{"class":3199},[3175,10416,10417],{"class":3181},"(result);\n",[3175,10419,10420],{"class":3177,"line":4385},[3175,10421,3413],{"class":3181},[3175,10423,10424],{"class":3177,"line":4395},[3175,10425,3419],{"class":3181},[3150,10427,10428],{},[3157,10429,10430],{},"Використання у Controller:",[3165,10432,10434],{"className":3167,"code":10433,"language":3169,"meta":3170,"style":3170},"@FXML\nprivate void onDeleteClicked() {\n    Optional\u003CBoolean> result = navigator.openDialog(\"confirm-delete\");\n    \n    if (result.isPresent() && result.get()) {\n        viewModel.deleteSelected();\n    }\n}\n",[3172,10435,10436,10442,10452,10478,10482,10504,10514,10518],{"__ignoreMap":3170},[3175,10437,10438,10440],{"class":3177,"line":3178},[3175,10439,3182],{"class":3181},[3175,10441,3186],{"class":3185},[3175,10443,10444,10446,10448,10450],{"class":3177,"line":3189},[3175,10445,3193],{"class":3192},[3175,10447,3196],{"class":3185},[3175,10449,7351],{"class":3199},[3175,10451,3203],{"class":3181},[3175,10453,10454,10456,10458,10460,10462,10464,10466,10468,10470,10472,10474,10476],{"class":3177,"line":3206},[3175,10455,9707],{"class":3185},[3175,10457,4165],{"class":3181},[3175,10459,7363],{"class":3185},[3175,10461,4175],{"class":3181},[3175,10463,7368],{"class":3222},[3175,10465,3226],{"class":3181},[3175,10467,7188],{"class":3222},[3175,10469,3270],{"class":3181},[3175,10471,4252],{"class":3199},[3175,10473,3235],{"class":3181},[3175,10475,6525],{"class":3249},[3175,10477,3561],{"class":3181},[3175,10479,10480],{"class":3177,"line":3216},[3175,10481,4041],{"class":3181},[3175,10483,10484,10486,10488,10490,10492,10494,10496,10498,10500,10502],{"class":3177,"line":3256},[3175,10485,7867],{"class":3209},[3175,10487,3385],{"class":3181},[3175,10489,7368],{"class":3222},[3175,10491,3270],{"class":3181},[3175,10493,7395],{"class":3199},[3175,10495,7398],{"class":3181},[3175,10497,7368],{"class":3222},[3175,10499,3270],{"class":3181},[3175,10501,6658],{"class":3199},[3175,10503,5661],{"class":3181},[3175,10505,10506,10508,10510,10512],{"class":3177,"line":3279},[3175,10507,8994],{"class":3222},[3175,10509,3270],{"class":3181},[3175,10511,7416],{"class":3199},[3175,10513,3276],{"class":3181},[3175,10515,10516],{"class":3177,"line":3285},[3175,10517,3413],{"class":3181},[3175,10519,10520],{"class":3177,"line":3305},[3175,10521,3419],{"class":3181},[3512,10523,10525],{"id":10524},"приклад-діалог-вибору-автора","Приклад: Діалог вибору автора",[3150,10527,10528],{},[3157,10529,10530],{},"SelectAuthorDialogController:",[3165,10532,10534],{"className":3167,"code":10533,"language":3169,"meta":3170,"style":3170},"public class SelectAuthorDialogController implements DialogController\u003CAuthor> {\n    \n    @FXML private TableView\u003CAuthor> authorTable;\n    @FXML private Button selectButton;\n    @FXML private Button cancelButton;\n    \n    private Author selectedAuthor;\n    \n    @Inject\n    private AuthorRepository authorRepository;\n    \n    @FXML\n    public void initialize() {\n        \u002F\u002F Завантаження авторів\n        authorTable.getItems().addAll(authorRepository.findAll());\n        \n        \u002F\u002F Активація кнопки Select лише коли щось обрано\n        selectButton.disableProperty().bind(\n            authorTable.getSelectionModel().selectedItemProperty().isNull()\n        );\n    }\n    \n    @FXML\n    private void onSelectClicked() {\n        selectedAuthor = authorTable.getSelectionModel().getSelectedItem();\n        closeDialog();\n    }\n    \n    @FXML\n    private void onCancelClicked() {\n        selectedAuthor = null;\n        closeDialog();\n    }\n    \n    private void closeDialog() {\n        Stage stage = (Stage) selectButton.getScene().getWindow();\n        stage.close();\n    }\n    \n    @Override\n    public Optional\u003CAuthor> getResult() {\n        return Optional.ofNullable(selectedAuthor);\n    }\n}\n",[3172,10535,10536,10556,10560,10582,10597,10611,10615,10627,10631,10637,10649,10653,10659,10669,10674,10701,10705,10710,10727,10749,10754,10758,10762,10768,10779,10797,10803,10807,10811,10817,10827,10835,10841,10845,10849,10859,10880,10890,10894,10898,10904,10920,10933,10937],{"__ignoreMap":3170},[3175,10537,10538,10540,10542,10545,10547,10549,10551,10554],{"class":3177,"line":3178},[3175,10539,4028],{"class":3192},[3175,10541,4557],{"class":3192},[3175,10543,10544],{"class":3185}," SelectAuthorDialogController",[3175,10546,4563],{"class":3192},[3175,10548,9665],{"class":3185},[3175,10550,4165],{"class":3181},[3175,10552,10553],{"class":3185},"Author",[3175,10555,9672],{"class":3181},[3175,10557,10558],{"class":3177,"line":3189},[3175,10559,4041],{"class":3181},[3175,10561,10562,10564,10566,10568,10571,10573,10575,10577,10580],{"class":3177,"line":3206},[3175,10563,4647],{"class":3181},[3175,10565,10141],{"class":3185},[3175,10567,10144],{"class":3192},[3175,10569,10570],{"class":3185}," TableView",[3175,10572,4165],{"class":3181},[3175,10574,10553],{"class":3185},[3175,10576,4175],{"class":3181},[3175,10578,10579],{"class":3222},"authorTable",[3175,10581,4588],{"class":3181},[3175,10583,10584,10586,10588,10590,10592,10595],{"class":3177,"line":3216},[3175,10585,4647],{"class":3181},[3175,10587,10141],{"class":3185},[3175,10589,10144],{"class":3192},[3175,10591,10163],{"class":3185},[3175,10593,10594],{"class":3222}," selectButton",[3175,10596,4588],{"class":3181},[3175,10598,10599,10601,10603,10605,10607,10609],{"class":3177,"line":3256},[3175,10600,4647],{"class":3181},[3175,10602,10141],{"class":3185},[3175,10604,10144],{"class":3192},[3175,10606,10163],{"class":3185},[3175,10608,10166],{"class":3222},[3175,10610,4588],{"class":3181},[3175,10612,10613],{"class":3177,"line":3279},[3175,10614,4041],{"class":3181},[3175,10616,10617,10619,10622,10625],{"class":3177,"line":3285},[3175,10618,4576],{"class":3192},[3175,10620,10621],{"class":3185}," Author",[3175,10623,10624],{"class":3222}," selectedAuthor",[3175,10626,4588],{"class":3181},[3175,10628,10629],{"class":3177,"line":3305},[3175,10630,4041],{"class":3181},[3175,10632,10633,10635],{"class":3177,"line":3326},[3175,10634,4647],{"class":3181},[3175,10636,4650],{"class":3185},[3175,10638,10639,10641,10644,10647],{"class":3177,"line":3331},[3175,10640,4576],{"class":3192},[3175,10642,10643],{"class":3185}," AuthorRepository",[3175,10645,10646],{"class":3222}," authorRepository",[3175,10648,4588],{"class":3181},[3175,10650,10651],{"class":3177,"line":3358},[3175,10652,4041],{"class":3181},[3175,10654,10655,10657],{"class":3177,"line":3376},[3175,10656,4647],{"class":3181},[3175,10658,3186],{"class":3185},[3175,10660,10661,10663,10665,10667],{"class":3177,"line":3397},[3175,10662,4655],{"class":3192},[3175,10664,3196],{"class":3185},[3175,10666,8954],{"class":3199},[3175,10668,3203],{"class":3181},[3175,10670,10671],{"class":3177,"line":3410},[3175,10672,10673],{"class":3539},"        \u002F\u002F Завантаження авторів\n",[3175,10675,10676,10679,10681,10684,10686,10689,10691,10694,10696,10699],{"class":3177,"line":3416},[3175,10677,10678],{"class":3222},"        authorTable",[3175,10680,3270],{"class":3181},[3175,10682,10683],{"class":3199},"getItems",[3175,10685,3241],{"class":3181},[3175,10687,10688],{"class":3199},"addAll",[3175,10690,3235],{"class":3181},[3175,10692,10693],{"class":3222},"authorRepository",[3175,10695,3270],{"class":3181},[3175,10697,10698],{"class":3199},"findAll",[3175,10700,4815],{"class":3181},[3175,10702,10703],{"class":3177,"line":4095},[3175,10704,3282],{"class":3181},[3175,10706,10707],{"class":3177,"line":4100},[3175,10708,10709],{"class":3539},"        \u002F\u002F Активація кнопки Select лише коли щось обрано\n",[3175,10711,10712,10715,10717,10720,10722,10725],{"class":3177,"line":4105},[3175,10713,10714],{"class":3222},"        selectButton",[3175,10716,3270],{"class":3181},[3175,10718,10719],{"class":3199},"disableProperty",[3175,10721,3241],{"class":3181},[3175,10723,10724],{"class":3199},"bind",[3175,10726,4976],{"class":3181},[3175,10728,10729,10732,10734,10737,10739,10742,10744,10747],{"class":3177,"line":4111},[3175,10730,10731],{"class":3222},"            authorTable",[3175,10733,3270],{"class":3181},[3175,10735,10736],{"class":3199},"getSelectionModel",[3175,10738,3241],{"class":3181},[3175,10740,10741],{"class":3199},"selectedItemProperty",[3175,10743,3241],{"class":3181},[3175,10745,10746],{"class":3199},"isNull",[3175,10748,5001],{"class":3181},[3175,10750,10751],{"class":3177,"line":4116},[3175,10752,10753],{"class":3181},"        );\n",[3175,10755,10756],{"class":3177,"line":4128},[3175,10757,3413],{"class":3181},[3175,10759,10760],{"class":3177,"line":4141},[3175,10761,4041],{"class":3181},[3175,10763,10764,10766],{"class":3177,"line":4146},[3175,10765,4647],{"class":3181},[3175,10767,3186],{"class":3185},[3175,10769,10770,10772,10774,10777],{"class":3177,"line":4183},[3175,10771,4576],{"class":3192},[3175,10773,3196],{"class":3185},[3175,10775,10776],{"class":3199}," onSelectClicked",[3175,10778,3203],{"class":3181},[3175,10780,10781,10784,10786,10788,10790,10792,10795],{"class":3177,"line":4188},[3175,10782,10783],{"class":3181},"        selectedAuthor = ",[3175,10785,10579],{"class":3222},[3175,10787,3270],{"class":3181},[3175,10789,10736],{"class":3199},[3175,10791,3241],{"class":3181},[3175,10793,10794],{"class":3199},"getSelectedItem",[3175,10796,3276],{"class":3181},[3175,10798,10799,10801],{"class":3177,"line":4193},[3175,10800,10281],{"class":3199},[3175,10802,3276],{"class":3181},[3175,10804,10805],{"class":3177,"line":4199},[3175,10806,3413],{"class":3181},[3175,10808,10809],{"class":3177,"line":4204},[3175,10810,4041],{"class":3181},[3175,10812,10813,10815],{"class":3177,"line":4216},[3175,10814,4647],{"class":3181},[3175,10816,3186],{"class":3185},[3175,10818,10819,10821,10823,10825],{"class":3177,"line":4227},[3175,10820,4576],{"class":3192},[3175,10822,3196],{"class":3185},[3175,10824,10306],{"class":3199},[3175,10826,3203],{"class":3181},[3175,10828,10829,10831,10833],{"class":3177,"line":4232},[3175,10830,10783],{"class":3181},[3175,10832,4936],{"class":3192},[3175,10834,4588],{"class":3181},[3175,10836,10837,10839],{"class":3177,"line":4263},[3175,10838,10281],{"class":3199},[3175,10840,3276],{"class":3181},[3175,10842,10843],{"class":3177,"line":4268},[3175,10844,3413],{"class":3181},[3175,10846,10847],{"class":3177,"line":4273},[3175,10848,4041],{"class":3181},[3175,10850,10851,10853,10855,10857],{"class":3177,"line":4279},[3175,10852,4576],{"class":3192},[3175,10854,3196],{"class":3185},[3175,10856,10340],{"class":3199},[3175,10858,3203],{"class":3181},[3175,10860,10861,10863,10865,10867,10870,10872,10874,10876,10878],{"class":3177,"line":4284},[3175,10862,3334],{"class":3185},[3175,10864,3337],{"class":3222},[3175,10866,3340],{"class":3181},[3175,10868,10869],{"class":3222},"selectButton",[3175,10871,3270],{"class":3181},[3175,10873,3348],{"class":3199},[3175,10875,3241],{"class":3181},[3175,10877,3353],{"class":3199},[3175,10879,3276],{"class":3181},[3175,10881,10882,10884,10886,10888],{"class":3177,"line":4295},[3175,10883,3361],{"class":3222},[3175,10885,3270],{"class":3181},[3175,10887,10372],{"class":3199},[3175,10889,3276],{"class":3181},[3175,10891,10892],{"class":3177,"line":4306},[3175,10893,3413],{"class":3181},[3175,10895,10896],{"class":3177,"line":4316},[3175,10897,4041],{"class":3181},[3175,10899,10900,10902],{"class":3177,"line":4321},[3175,10901,4647],{"class":3181},[3175,10903,4780],{"class":3185},[3175,10905,10906,10908,10910,10912,10914,10916,10918],{"class":3177,"line":4364},[3175,10907,4655],{"class":3192},[3175,10909,5533],{"class":3185},[3175,10911,4165],{"class":3181},[3175,10913,10553],{"class":3185},[3175,10915,4175],{"class":3181},[3175,10917,5550],{"class":3199},[3175,10919,3203],{"class":3181},[3175,10921,10922,10924,10926,10928,10930],{"class":3177,"line":4369},[3175,10923,5211],{"class":3209},[3175,10925,5533],{"class":3222},[3175,10927,3270],{"class":3181},[3175,10929,5538],{"class":3199},[3175,10931,10932],{"class":3181},"(selectedAuthor);\n",[3175,10934,10935],{"class":3177,"line":4374},[3175,10936,3413],{"class":3181},[3175,10938,10939],{"class":3177,"line":4380},[3175,10940,3419],{"class":3181},[3150,10942,10943],{},[3157,10944,3874],{},[3165,10946,10948],{"className":3167,"code":10947,"language":3169,"meta":3170,"style":3170},"@FXML\nprivate void onSelectAuthorClicked() {\n    Optional\u003CAuthor> result = navigator.openDialog(\"select-author\");\n    \n    if (result.isPresent()) {\n        viewModel.setSelectedAuthor(result.get());\n    }\n}\n",[3172,10949,10950,10956,10967,10993,10997,11011,11030,11034],{"__ignoreMap":3170},[3175,10951,10952,10954],{"class":3177,"line":3178},[3175,10953,3182],{"class":3181},[3175,10955,3186],{"class":3185},[3175,10957,10958,10960,10962,10965],{"class":3177,"line":3189},[3175,10959,3193],{"class":3192},[3175,10961,3196],{"class":3185},[3175,10963,10964],{"class":3199}," onSelectAuthorClicked",[3175,10966,3203],{"class":3181},[3175,10968,10969,10971,10973,10975,10977,10979,10981,10983,10985,10987,10989,10991],{"class":3177,"line":3206},[3175,10970,9707],{"class":3185},[3175,10972,4165],{"class":3181},[3175,10974,10553],{"class":3185},[3175,10976,4175],{"class":3181},[3175,10978,7368],{"class":3222},[3175,10980,3226],{"class":3181},[3175,10982,7188],{"class":3222},[3175,10984,3270],{"class":3181},[3175,10986,4252],{"class":3199},[3175,10988,3235],{"class":3181},[3175,10990,6546],{"class":3249},[3175,10992,3561],{"class":3181},[3175,10994,10995],{"class":3177,"line":3216},[3175,10996,4041],{"class":3181},[3175,10998,10999,11001,11003,11005,11007,11009],{"class":3177,"line":3256},[3175,11000,7867],{"class":3209},[3175,11002,3385],{"class":3181},[3175,11004,7368],{"class":3222},[3175,11006,3270],{"class":3181},[3175,11008,7395],{"class":3199},[3175,11010,5661],{"class":3181},[3175,11012,11013,11015,11017,11020,11022,11024,11026,11028],{"class":3177,"line":3279},[3175,11014,8994],{"class":3222},[3175,11016,3270],{"class":3181},[3175,11018,11019],{"class":3199},"setSelectedAuthor",[3175,11021,3235],{"class":3181},[3175,11023,7368],{"class":3222},[3175,11025,3270],{"class":3181},[3175,11027,6658],{"class":3199},[3175,11029,4815],{"class":3181},[3175,11031,11032],{"class":3177,"line":3285},[3175,11033,3413],{"class":3181},[3175,11035,11036],{"class":3177,"line":3305},[3175,11037,3419],{"class":3181},[8298,11039,11040,11043,11044,11047,11048,11051],{},[3157,11041,11042],{},"Діалоги повертають Optional:"," Якщо користувач натиснув \"Cancel\" або закрив вікно, ",[3172,11045,11046],{},"getResult()"," повертає ",[3172,11049,11050],{},"Optional.empty()",". Це явно вказує, що результату немає, і змушує викликаючий код обробити цей випадок.",[3502,11053],{},[3145,11055,11057],{"id":11056},"практичні-завдання","Практичні завдання",[11059,11060,11061,11065,11081,11093,11109,11113,11132,11144,11153,11157,11178,11184],"steps",{},[3512,11062,11064],{"id":11063},"рівень-1-базова-навігація","Рівень 1: Базова навігація",[3150,11066,11067,11070,11071,11073,11074,11077,11078,3270],{},[3157,11068,11069],{},"Завдання 1.1:"," Створіть ",[3172,11072,4669],{}," з реєстрацією трьох екранів: \"home\", \"settings\", \"about\". Створіть ",[3172,11075,11076],{},"FxmlNavigator"," та реалізуйте метод ",[3172,11079,11080],{},"navigateTo()",[3150,11082,11083,11070,11086,11089,11090,11092],{},[3157,11084,11085],{},"Завдання 1.2:",[3172,11087,11088],{},"MainController"," з трьома кнопками: \"Home\", \"Settings\", \"About\". При натисканні кожної кнопки викликайте ",[3172,11091,7490],{}," з відповідним screenId.",[3150,11094,11095,11098,11099,11101,11102,11105,11106,3270],{},[3157,11096,11097],{},"Завдання 1.3:"," Додайте кнопку \"Back\" у кожен екран. Прив'яжіть її ",[3172,11100,10719],{}," до ",[3172,11103,11104],{},"navigator.canGoBack().not()",". При натисканні викликайте ",[3172,11107,11108],{},"navigator.goBack()",[3512,11110,11112],{"id":11111},"рівень-2-передача-параметрів-та-діалоги","Рівень 2: Передача параметрів та діалоги",[3150,11114,11115,11118,11119,11121,11122,11124,11125,11128,11129,11131],{},[3157,11116,11117],{},"Завдання 2.1:"," Реалізуйте передачу параметрів через ",[3172,11120,7733],{},". Створіть інтерфейс ",[3172,11123,8040],{}," з методом ",[3172,11126,11127],{},"setParameters(Map\u003CString, Object>)",". Модифікуйте ",[3172,11130,11076],{},", щоб він викликав цей метод після завантаження FXML.",[3150,11133,11134,11137,11138,11141,11142,3270],{},[3157,11135,11136],{},"Завдання 2.2:"," Створіть діалог \"Select Genre\" з TableView жанрів. Реалізуйте ",[3172,11139,11140],{},"DialogController\u003CGenre>",". При натисканні \"Select\" поверніть обраний жанр. При натисканні \"Cancel\" поверніть ",[3172,11143,11050],{},[3150,11145,11146,11149,11150,11152],{},[3157,11147,11148],{},"Завдання 2.3:"," Реалізуйте навігацію з параметром: при натисканні на аудіокнигу у списку відкривайте екран деталей з передачею ",[3172,11151,3454],{}," через параметри Navigator.",[3512,11154,11156],{"id":11155},"рівень-3-інтеграція-з-viewmodel-та-складні-сценарії","Рівень 3: Інтеграція з ViewModel та складні сценарії",[3150,11158,11159,11162,11163,11165,11166,11169,11170,11172,11173,11175,11176,3270],{},[3157,11160,11161],{},"Завдання 3.1:"," Реалізуйте ",[3172,11164,8792],{}," та інтеграцію з ViewModel. ",[3172,11167,11168],{},"AudiobookFormViewModel.save()"," має встановлювати ",[3172,11171,8797],{}," у ",[3172,11174,8875],{},". Controller слухає зміни та викликає ",[3172,11177,11108],{},[3150,11179,11180,11183],{},[3157,11181,11182],{},"Завдання 3.2:"," Створіть Wizard Pattern: форма з трьома кроками (Step 1: Basic Info, Step 2: Details, Step 3: Confirmation). Кнопки \"Next\", \"Previous\", \"Finish\". Navigator керує переходами між кроками. Дані зберігаються у Singleton ViewModel, доступному всім крокам.",[3150,11185,11186,11189,11190,11193],{},[3157,11187,11188],{},"Завдання 3.3:"," Реалізуйте обмеження розміру Navigation Stack (максимум 10 екранів). При перевищенні видаляйте найстаріший екран зі стеку. Додайте метод ",[3172,11191,11192],{},"navigator.clearHistory()"," для очищення стеку.",[3502,11195],{},[3145,11197,11199],{"id":11198},"підсумок","Підсумок",[3150,11201,11202],{},"У цій статті ми побудували повноцінну систему навігації для JavaFX MVVM-додатку. Ключові висновки:",[3150,11204,11205,11208,11209,11211,11212,3270],{},[3157,11206,11207],{},"Navigator Pattern централізує управління навігацією."," Замість дублювання коду завантаження FXML у кожному Controller, весь код інкапсульований у Navigator. Controller лише викликає ",[3172,11210,7490],{}," або ",[3172,11213,7493],{},[3150,11215,11216,11219],{},[3157,11217,11218],{},"ScreenRegistry замінює magic strings."," Всі екрани зареєстровані у централізованому реєстрі з ідентифікаторами (\"audiobook-list\", \"audiobook-form\"). Це дозволяє легко змінювати шляхи до FXML, додавати метадані (title, розмір вікна), валідувати існування екранів.",[3150,11221,11222,11225,11226,11229],{},[3157,11223,11224],{},"Navigation Stack зберігає історію переходів."," Кожен перехід зберігає поточний екран у стек. Метод ",[3172,11227,11228],{},"goBack()"," витягує попередній екран зі стеку та відновлює його зі збереженим станом (прокрутка, вибір, введені дані).",[3150,11231,11232,11235],{},[3157,11233,11234],{},"Три підходи до передачі параметрів:"," Singleton ViewModel (проста реалізація, сильна зв'язаність), параметри Navigator (явна передача, відсутність type-safety), AssistedInject Factory (type-safety, складніша реалізація). Вибір залежить від складності параметрів.",[3150,11237,11238,11241,11242,11244,11245,11247],{},[3157,11239,11240],{},"ViewModel не залежить від Navigator."," Інтеграція через Events (",[3172,11243,8474],{},") або Callbacks (",[3172,11246,9610],{}," інтерфейс). Controller слухає Events або передає Callback у ViewModel. Це зберігає розділення відповідальностей MVVM.",[3150,11249,11250,3442,11253,11256,11257,11259,11260,11262,11263,11266,11267,3270],{},[3157,11251,11252],{},"Модальні діалоги через окремі Stage.",[3172,11254,11255],{},"Navigator.openDialog()"," створює новий Stage з модальністю ",[3172,11258,5426],{},", показує діалог через ",[3172,11261,6171],{},", отримує результат через інтерфейс ",[3172,11264,11265],{},"DialogController\u003CT>",". Результат повертається як ",[3172,11268,11269],{},"Optional\u003CT>",[3150,11271,11272,11275],{},[3157,11273,11274],{},"Single Window Navigation для основних екранів, Multiple Windows для діалогів."," Основна навігація через заміну Root Node у Scene (один екран на весь екран). Діалоги через окремі Stage (модальні вікна поверх головного).",[3150,11277,11278],{},"У наступній статті ми розглянемо тестування JavaFX MVVM-додатків: unit-тестування ViewModel з mock залежностями, інтеграційне тестування ViewModel + Repository з H2 database, UI-тестування з TestFX (автоматизоване тестування JavaFX інтерфейсу), Test Doubles (Mock vs Stub vs Fake), та організацію тестів для різних шарів додатку.",[11280,11281,11282],"style",{},"html pre.shiki code .sHH4Y, html code.shiki .sHH4Y{--shiki-light:#000000;--shiki-default:#D4D4D4;--shiki-dark:#D4D4D4}html pre.shiki code .sN1BT, html code.shiki .sN1BT{--shiki-light:#267F99;--shiki-default:#4EC9B0;--shiki-dark:#4EC9B0}html pre.shiki code .su1O8, html code.shiki .su1O8{--shiki-light:#0000FF;--shiki-default:#569CD6;--shiki-dark:#569CD6}html pre.shiki code .s8Opu, html code.shiki .s8Opu{--shiki-light:#795E26;--shiki-default:#DCDCAA;--shiki-dark:#DCDCAA}html pre.shiki code .s8xlr, html code.shiki .s8xlr{--shiki-light:#AF00DB;--shiki-default:#C586C0;--shiki-dark:#C586C0}html pre.shiki code .siwwj, html code.shiki .siwwj{--shiki-light:#001080;--shiki-default:#9CDCFE;--shiki-dark:#9CDCFE}html pre.shiki code .sbdoH, html code.shiki .sbdoH{--shiki-light:#A31515;--shiki-default:#CE9178;--shiki-dark:#CE9178}html .light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html.light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html.dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html pre.shiki code .spJ8K, html code.shiki .spJ8K{--shiki-light:#008000;--shiki-default:#6A9955;--shiki-dark:#6A9955}html pre.shiki code .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 .su9tN, html code.shiki .su9tN{--shiki-light:#0000FF;--shiki-default:#CE9178;--shiki-dark:#CE9178}",{"title":3170,"searchDepth":3189,"depth":3189,"links":11284},[11285,11286,11292,11297,11301,11306,11311,11316,11321],{"id":3147,"depth":3189,"text":3148},{"id":3506,"depth":3189,"text":3507,"children":11287},[11288,11289,11290,11291],{"id":3514,"depth":3206,"text":3515},{"id":3615,"depth":3206,"text":3616},{"id":3728,"depth":3206,"text":3729},{"id":3833,"depth":3206,"text":3834},{"id":3966,"depth":3189,"text":3967,"children":11293},[11294,11295,11296],{"id":3973,"depth":3206,"text":3974},{"id":4450,"depth":3206,"text":4451},{"id":6106,"depth":3206,"text":6107},{"id":6220,"depth":3189,"text":6221,"children":11298},[11299,11300],{"id":6234,"depth":3206,"text":6235},{"id":7087,"depth":3206,"text":7088},{"id":7499,"depth":3189,"text":7500,"children":11302},[11303,11304,11305],{"id":7509,"depth":3206,"text":7510},{"id":7725,"depth":3206,"text":7726},{"id":8044,"depth":3206,"text":8045},{"id":8320,"depth":3189,"text":8321,"children":11307},[11308,11309,11310],{"id":8327,"depth":3206,"text":8328},{"id":8467,"depth":3206,"text":8468},{"id":9246,"depth":3206,"text":9247},{"id":9623,"depth":3189,"text":9624,"children":11312},[11313,11314,11315],{"id":9630,"depth":3206,"text":9631},{"id":9724,"depth":3206,"text":9725},{"id":10524,"depth":3206,"text":10525},{"id":11056,"depth":3189,"text":11057,"children":11317},[11318,11319,11320],{"id":11063,"depth":3206,"text":11064},{"id":11111,"depth":3206,"text":11112},{"id":11155,"depth":3206,"text":11156},{"id":11198,"depth":3189,"text":11199},"Від хаотичних переходів до централізованої навігації: Navigator Pattern, ScreenRegistry, передача параметрів між екранами, Navigation Stack для історії переходів, модальні діалоги з поверненням результату, інтеграція з ViewModel через Events.","md",null,{},{"title":2442,"description":11322},"CMLMLOkzcVTKLVx2qOul5jm_mQZy7t1uqCoa2SClDno",[11329,11331],{"title":2438,"path":2439,"stem":2440,"description":11330,"children":-1},"Від реактивної валідації Properties до централізованої системи Validators: валідація на рівні ViewModel, Validator Pattern, CompositeValidator, асинхронна валідація унікальності, обробка помилок Repository та Service, відображення помилок у View.",{"title":2446,"path":2447,"stem":2448,"description":11332,"children":-1},"Від unit-тестів ViewModel до UI-автоматизації: тестування з Mockito, інтеграційні тести з H2 database, TestFX для автоматизованого тестування JavaFX UI, Test Doubles (Mock vs Stub vs Fake), організація тестів для різних шарів.",1778998390823]