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