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