From acfc1ab9a60ab810feb86088f84bb2ff7528803d Mon Sep 17 00:00:00 2001 From: malakelbanna Date: Thu, 5 Dec 2024 01:53:32 +0200 Subject: [PATCH 1/2] pattern:implemented the Template View pattern (#1320) --- pom.xml | 1 + templateview/README.md | 144 ++++++++++++++++++ templateview/etc/template-view.urm.puml | 33 ++++ templateview/etc/template_view_urm.png | Bin 0 -> 44852 bytes templateview/pom.xml | 67 ++++++++ .../java/com/iluwater/templateview/App.java | 52 +++++++ .../templateview/ContactPageView.java | 42 +++++ .../iluwater/templateview/HomePageView.java | 41 +++++ .../iluwater/templateview/TemplateView.java | 63 ++++++++ .../com/iluwater/templateview/AppTest.java | 40 +++++ .../templateview/ContactPageViewTest.java | 43 ++++++ .../templateview/HomePageViewTest.java | 43 ++++++ .../templateview/TemplateViewTest.java | 59 +++++++ 13 files changed, 628 insertions(+) create mode 100644 templateview/README.md create mode 100644 templateview/etc/template-view.urm.puml create mode 100644 templateview/etc/template_view_urm.png create mode 100644 templateview/pom.xml create mode 100644 templateview/src/main/java/com/iluwater/templateview/App.java create mode 100644 templateview/src/main/java/com/iluwater/templateview/ContactPageView.java create mode 100644 templateview/src/main/java/com/iluwater/templateview/HomePageView.java create mode 100644 templateview/src/main/java/com/iluwater/templateview/TemplateView.java create mode 100644 templateview/src/test/java/com/iluwater/templateview/AppTest.java create mode 100644 templateview/src/test/java/com/iluwater/templateview/ContactPageViewTest.java create mode 100644 templateview/src/test/java/com/iluwater/templateview/HomePageViewTest.java create mode 100644 templateview/src/test/java/com/iluwater/templateview/TemplateViewTest.java diff --git a/pom.xml b/pom.xml index ef3a39265d53..51ca5fb52459 100644 --- a/pom.xml +++ b/pom.xml @@ -218,6 +218,7 @@ function-composition microservices-distributed-tracing microservices-idempotent-consumer + templateview diff --git a/templateview/README.md b/templateview/README.md new file mode 100644 index 000000000000..b0282bb857c4 --- /dev/null +++ b/templateview/README.md @@ -0,0 +1,144 @@ +--- +title: "Template View Pattern in Java: Streamlining Dynamic Webpage Rendering" +shortTitle: Template View +description: "Learn about the Template View design pattern in Java, which simplifies webpage rendering by separating static and dynamic content. Ideal for developers building reusable and maintainable UI components." +category: Behavioral +language: en +tag: + - Abstraction + - Code simplification + - Decoupling + - Extensibility + - Gang of Four + - Inheritance + - Polymorphism + - Reusability +--- + +## Intent of Template View Design Pattern + +Separate the structure and static parts of a webpage (or view) from its dynamic content. Template View ensures a consistent layout while allowing flexibility for different types of views. + +## Detailed Explanation of Template View Pattern with Real-World Examples + +### Real-World Example + +> Think of a blog website where each post page follows the same layout with a header, footer, and main content area. While the header and footer remain consistent, the main content differs for each blog post. The Template View pattern encapsulates the shared layout (header and footer) in a base class while delegating the rendering of the main content to subclasses. + +### In Plain Words + +> The Template View pattern provides a way to define a consistent layout in a base class while letting subclasses implement the specific, dynamic content for different views. + +### Wikipedia Says + +> While not a classic Gang of Four pattern, Template View aligns closely with the Template Method pattern, applied specifically to rendering webpages or views. It defines a skeleton for rendering, delegating dynamic parts to subclasses while keeping the structure consistent. + +## Programmatic Example of Template View Pattern in Java + +Our example involves rendering different types of views (`HomePageView` and `ContactPageView`) with a common structure consisting of a header, dynamic content, and a footer. + +### The Abstract Base Class: TemplateView + +The `TemplateView` class defines the skeleton for rendering a view. Subclasses provide implementations for rendering dynamic content. + +```java +@Slf4j +public abstract class TemplateView { + + public final void render() { + printHeader(); + renderDynamicContent(); + printFooter(); + } + + protected void printHeader() { + LOGGER.info("Rendering header..."); + } + + protected abstract void renderDynamicContent(); + + protected void printFooter() { + LOGGER.info("Rendering footer..."); + } +} +``` +### Concrete Class: HomePageView +```java +@Slf4j +public class HomePageView extends TemplateView { + + @Override + protected void renderDynamicContent() { + LOGGER.info("Welcome to the Home Page!"); + } +} +``` +### Concrete Class: ContactPageView +```java +@Slf4j +public class ContactPageView extends TemplateView { + + @Override + protected void renderDynamicContent() { + LOGGER.info("Contact us at: contact@example.com"); + } +} +``` +### Application Class: App +The `App` class demonstrates rendering different views using the Template View pattern. +```java +@Slf4j +public class App { + + public static void main(String[] args) { + TemplateView homePage = new HomePageView(); + LOGGER.info("Rendering HomePage:"); + homePage.render(); + + TemplateView contactPage = new ContactPageView(); + LOGGER.info("\nRendering ContactPage:"); + contactPage.render(); + } +} +``` +## Output of the Program +```lessRendering HomePage: +Rendering header... +Welcome to the Home Page! +Rendering footer... + +Rendering ContactPage: +Rendering header... +Contact us at: contact@example.com +Rendering footer... +``` +## When to Use the Template View Pattern in Java +- When you want to enforce a consistent structure for rendering views while allowing flexibility in dynamic content. +- When you need to separate the static layout (header, footer) from the dynamic parts of a view (main content). +- To enhance code reusability and reduce duplication in rendering logic. + +## Benefits and Trade-offs of Template View Pattern +**Benefits:** +- Code Reusability: Centralizes shared layout logic in the base class. +- Maintainability: Reduces duplication, making updates easier. +- Flexibility: Allows subclasses to customize dynamic content. + +**Trade-offs:** +- Increased Number of Classes: Requires creating separate classes for each type of view. +- Design Overhead: Might be overkill for simple applications with few views. + +## Related Java Design Patterns +- **Template Method:** A similar pattern focusing on defining a skeleton algorithm, allowing subclasses to implement specific steps. +- **Strategy Pattern:** Offers flexibility in choosing dynamic behaviors at runtime instead of hardcoding them in subclasses. +- **Decorator Pattern:** Can complement Template View for dynamically adding responsibilities to views. + +## Real World Applications of Template View Pattern +- Web frameworks like Spring MVC and Django use this concept to render views consistently. +- CMS platforms like WordPress follow this pattern for theme templates, separating layout from content. + +## References and Credits +- [Effective Java](https://amzn.to/4cGk2Jz) +- [Head First Design Patterns: Building Extensible and Maintainable Object-Oriented Software](https://amzn.to/49NGldq) +- [Refactoring to Patterns](https://amzn.to/3VOO4F5) +- [Template Method Pattern](https://refactoring.guru/design-patterns/template-method) +- [Basics of Django: Model-View-Template (MVT) Architecture](https://angelogentileiii.medium.com/basics-of-django-model-view-template-mvt-architecture-8585aecffbf6) diff --git a/templateview/etc/template-view.urm.puml b/templateview/etc/template-view.urm.puml new file mode 100644 index 000000000000..503737b51686 --- /dev/null +++ b/templateview/etc/template-view.urm.puml @@ -0,0 +1,33 @@ +@startuml +package com.iluwater.templateview { + class App { + + App() + + main(args : String[]) {static} + } + + abstract class TemplateView { + - LOGGER : Logger {static} + + TemplateView() + + render() : void {final} + # printHeader() : void + # renderDynamicContent() : void {abstract} + # printFooter() : void + } + + class HomePageView { + - LOGGER : Logger {static} + + HomePageView() + + renderDynamicContent() : void + } + + class ContactPageView { + - LOGGER : Logger {static} + + ContactPageView() + + renderDynamicContent() : void + } +} + +App --> TemplateView +TemplateView <|-- HomePageView +TemplateView <|-- ContactPageView +@enduml diff --git a/templateview/etc/template_view_urm.png b/templateview/etc/template_view_urm.png new file mode 100644 index 0000000000000000000000000000000000000000..0c3e54132fd77aa0f7c71d491cea06a313e540c5 GIT binary patch literal 44852 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 literal 0 HcmV?d00001 diff --git a/templateview/pom.xml b/templateview/pom.xml new file mode 100644 index 000000000000..ce5869a3174b --- /dev/null +++ b/templateview/pom.xml @@ -0,0 +1,67 @@ + + + + 4.0.0 + + com.iluwatar + java-design-patterns + 1.26.0-SNAPSHOT + + templateview + + + org.junit.jupiter + junit-jupiter-engine + test + + + org.mockito + mockito-inline + test + + + + + + org.apache.maven.plugins + maven-assembly-plugin + + + + + + com.iluwatar.templateview.App + + + + + + + + + diff --git a/templateview/src/main/java/com/iluwater/templateview/App.java b/templateview/src/main/java/com/iluwater/templateview/App.java new file mode 100644 index 000000000000..5ffcb6cda1d1 --- /dev/null +++ b/templateview/src/main/java/com/iluwater/templateview/App.java @@ -0,0 +1,52 @@ +/* + * This project is licensed under the MIT license. Module model-view-viewmodel is using ZK framework licensed under LGPL (see lgpl-3.0.txt). + * + * The MIT License + * Copyright © 2014-2022 Ilkka Seppälä + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package com.iluwater.templateview; + +import lombok.extern.slf4j.Slf4j; + +/** + * The App class demonstrates the Template View pattern. + * In this example, it renders different views such as the HomePage and ContactPage. + */ +@Slf4j +public class App { + + /** + * Program entry point. + * + * @param args command line args + */ + public static void main(String[] args) { + // Create and render the HomePageView + TemplateView homePage = new HomePageView(); + LOGGER.info("Rendering HomePage:"); + homePage.render(); + + // Create and render the ContactPageView + TemplateView contactPage = new ContactPageView(); + LOGGER.info("\nRendering ContactPage:"); + contactPage.render(); + } +} diff --git a/templateview/src/main/java/com/iluwater/templateview/ContactPageView.java b/templateview/src/main/java/com/iluwater/templateview/ContactPageView.java new file mode 100644 index 000000000000..c55ce8642c07 --- /dev/null +++ b/templateview/src/main/java/com/iluwater/templateview/ContactPageView.java @@ -0,0 +1,42 @@ +/* + * This project is licensed under the MIT license. Module model-view-viewmodel is using ZK framework licensed under LGPL (see lgpl-3.0.txt). + * + * The MIT License + * Copyright © 2014-2022 Ilkka Seppälä + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package com.iluwater.templateview; + +import lombok.extern.slf4j.Slf4j; + +/** + * ContactPageView implements the TemplateView and provides dynamic content specific to the contact page. + */ +@Slf4j +public class ContactPageView extends TemplateView { + + /** + * Renders dynamic content for the contact page. + */ + @Override + protected void renderDynamicContent() { + LOGGER.info("Contact us at: contact@example.com"); + } +} diff --git a/templateview/src/main/java/com/iluwater/templateview/HomePageView.java b/templateview/src/main/java/com/iluwater/templateview/HomePageView.java new file mode 100644 index 000000000000..3c83c25a7f72 --- /dev/null +++ b/templateview/src/main/java/com/iluwater/templateview/HomePageView.java @@ -0,0 +1,41 @@ +/* + * This project is licensed under the MIT license. Module model-view-viewmodel is using ZK framework licensed under LGPL (see lgpl-3.0.txt). + * + * The MIT License + * Copyright © 2014-2022 Ilkka Seppälä + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package com.iluwater.templateview; + +import lombok.extern.slf4j.Slf4j; + +/** + * HomePageView implements the TemplateView and provides dynamic content specific to the homepage. + */ +@Slf4j +public class HomePageView extends TemplateView { + /** + * Renders dynamic content for the homepage. + */ + @Override + protected void renderDynamicContent() { + LOGGER.info("Welcome to the Home Page!"); + } +} diff --git a/templateview/src/main/java/com/iluwater/templateview/TemplateView.java b/templateview/src/main/java/com/iluwater/templateview/TemplateView.java new file mode 100644 index 000000000000..81e2bb80ed9b --- /dev/null +++ b/templateview/src/main/java/com/iluwater/templateview/TemplateView.java @@ -0,0 +1,63 @@ +/* + * This project is licensed under the MIT license. Module model-view-viewmodel is using ZK framework licensed under LGPL (see lgpl-3.0.txt). + * + * The MIT License + * Copyright © 2014-2022 Ilkka Seppälä + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package com.iluwater.templateview; + +import lombok.extern.slf4j.Slf4j; + +/** + * TemplateView defines the skeleton for rendering views. + * Concrete subclasses will provide the dynamic content for specific views. + */ +@Slf4j +public abstract class TemplateView { + + /** + * Render the common structure of the view, delegating dynamic content to subclasses. + */ + public final void render() { + printHeader(); + renderDynamicContent(); + printFooter(); + } + + /** + * Prints the common header of the view. + */ + protected void printHeader() { + LOGGER.info("Rendering header..."); + } + + /** + * Subclasses must provide the implementation for rendering dynamic content. + */ + protected abstract void renderDynamicContent(); + + /** + * Prints the common footer of the view. + */ + protected void printFooter() { + LOGGER.info("Rendering footer..."); + } +} diff --git a/templateview/src/test/java/com/iluwater/templateview/AppTest.java b/templateview/src/test/java/com/iluwater/templateview/AppTest.java new file mode 100644 index 000000000000..fb73abfc6c46 --- /dev/null +++ b/templateview/src/test/java/com/iluwater/templateview/AppTest.java @@ -0,0 +1,40 @@ +/* + * This project is licensed under the MIT license. Module model-view-viewmodel is using ZK framework licensed under LGPL (see lgpl-3.0.txt). + * + * The MIT License + * Copyright © 2014-2022 Ilkka Seppälä + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package com.iluwater.templateview; + +import org.junit.jupiter.api.Test; +import static org.junit.jupiter.api.Assertions.assertDoesNotThrow; + +/** + * Application test + */ +class AppTest { + + @Test + void shouldExecuteWithoutException() { + // Verify that main() method executes without throwing exceptions + assertDoesNotThrow(() -> App.main(new String[]{})); + } +} diff --git a/templateview/src/test/java/com/iluwater/templateview/ContactPageViewTest.java b/templateview/src/test/java/com/iluwater/templateview/ContactPageViewTest.java new file mode 100644 index 000000000000..9af715c35ca7 --- /dev/null +++ b/templateview/src/test/java/com/iluwater/templateview/ContactPageViewTest.java @@ -0,0 +1,43 @@ +/* + * This project is licensed under the MIT license. Module model-view-viewmodel is using ZK framework licensed under LGPL (see lgpl-3.0.txt). + * + * The MIT License + * Copyright © 2014-2022 Ilkka Seppälä + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package com.iluwater.templateview; + +import org.junit.jupiter.api.Test; +import static org.mockito.Mockito.*; + +class ContactPageViewTest { + + @Test + void testRenderDynamicContent() { + // Create a spy for ContactPageView + ContactPageView contactPage = spy(ContactPageView.class); + + // Render dynamic content for ContactPageView + contactPage.renderDynamicContent(); + + // Verify that the correct message is logged + verify(contactPage).renderDynamicContent(); + } +} diff --git a/templateview/src/test/java/com/iluwater/templateview/HomePageViewTest.java b/templateview/src/test/java/com/iluwater/templateview/HomePageViewTest.java new file mode 100644 index 000000000000..2d9abfe5926e --- /dev/null +++ b/templateview/src/test/java/com/iluwater/templateview/HomePageViewTest.java @@ -0,0 +1,43 @@ +/* + * This project is licensed under the MIT license. Module model-view-viewmodel is using ZK framework licensed under LGPL (see lgpl-3.0.txt). + * + * The MIT License + * Copyright © 2014-2022 Ilkka Seppälä + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package com.iluwater.templateview; + +import org.junit.jupiter.api.Test; +import static org.mockito.Mockito.*; + +class HomePageViewTest { + + @Test + void testRenderDynamicContent() { + // Create a spy for HomePageView + HomePageView homePage = spy(HomePageView.class); + + // Render dynamic content for HomePageView + homePage.renderDynamicContent(); + + // Verify that the correct message is logged + verify(homePage).renderDynamicContent(); + } +} diff --git a/templateview/src/test/java/com/iluwater/templateview/TemplateViewTest.java b/templateview/src/test/java/com/iluwater/templateview/TemplateViewTest.java new file mode 100644 index 000000000000..2051c2d722c9 --- /dev/null +++ b/templateview/src/test/java/com/iluwater/templateview/TemplateViewTest.java @@ -0,0 +1,59 @@ +/* + * This project is licensed under the MIT license. Module model-view-viewmodel is using ZK framework licensed under LGPL (see lgpl-3.0.txt). + * + * The MIT License + * Copyright © 2014-2022 Ilkka Seppälä + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package com.iluwater.templateview; + +import org.junit.jupiter.api.Test; +import static org.mockito.Mockito.*; + +class TemplateViewTest { + + @Test + void testRenderHomePage() { + // Create a spy for HomePageView + TemplateView homePage = spy(HomePageView.class); + + // Call the render method + homePage.render(); + + // Verify that the steps of rendering are executed in the correct order + verify(homePage).printHeader(); // Header is printed + verify(homePage).renderDynamicContent(); // Dynamic content specific to home page + verify(homePage).printFooter(); // Footer is printed + } + + @Test + void testRenderContactPage() { + // Create a spy for ContactPageView + TemplateView contactPage = spy(ContactPageView.class); + + // Call the render method + contactPage.render(); + + // Verify that the steps of rendering are executed in the correct order + verify(contactPage).printHeader(); // Header is printed + verify(contactPage).renderDynamicContent(); // Dynamic content specific to contact page + verify(contactPage).printFooter(); // Footer is printed + } +} From 8b80235943cabdb2fe864766682db6f80f043299 Mon Sep 17 00:00:00 2001 From: malakelbanna Date: Tue, 10 Dec 2024 21:06:45 +0200 Subject: [PATCH 2/2] fix:added links in README and updated package name (#1320) --- templateview/README.md | 6 +++--- .../{iluwater => iluwatar}/templateview/App.java | 13 ++++++++++--- .../templateview/ContactPageView.java | 2 +- .../templateview/HomePageView.java | 2 +- .../templateview/TemplateView.java | 2 +- .../templateview/AppTest.java | 2 +- .../templateview/ContactPageViewTest.java | 2 +- .../templateview/HomePageViewTest.java | 2 +- .../templateview/TemplateViewTest.java | 2 +- 9 files changed, 20 insertions(+), 13 deletions(-) rename templateview/src/main/java/com/{iluwater => iluwatar}/templateview/App.java (75%) rename templateview/src/main/java/com/{iluwater => iluwatar}/templateview/ContactPageView.java (97%) rename templateview/src/main/java/com/{iluwater => iluwatar}/templateview/HomePageView.java (97%) rename templateview/src/main/java/com/{iluwater => iluwatar}/templateview/TemplateView.java (98%) rename templateview/src/test/java/com/{iluwater => iluwatar}/templateview/AppTest.java (97%) rename templateview/src/test/java/com/{iluwater => iluwatar}/templateview/ContactPageViewTest.java (98%) rename templateview/src/test/java/com/{iluwater => iluwatar}/templateview/HomePageViewTest.java (97%) rename templateview/src/test/java/com/{iluwater => iluwatar}/templateview/TemplateViewTest.java (98%) diff --git a/templateview/README.md b/templateview/README.md index b0282bb857c4..3a53447c2cff 100644 --- a/templateview/README.md +++ b/templateview/README.md @@ -128,9 +128,9 @@ Rendering footer... - Design Overhead: Might be overkill for simple applications with few views. ## Related Java Design Patterns -- **Template Method:** A similar pattern focusing on defining a skeleton algorithm, allowing subclasses to implement specific steps. -- **Strategy Pattern:** Offers flexibility in choosing dynamic behaviors at runtime instead of hardcoding them in subclasses. -- **Decorator Pattern:** Can complement Template View for dynamically adding responsibilities to views. +- [Template Method](https://java-design-patterns.com/patterns/template-method/): A similar pattern focusing on defining a skeleton algorithm, allowing subclasses to implement specific steps. +- [Strategy Pattern](https://java-design-patterns.com/patterns/strategy/): Offers flexibility in choosing dynamic behaviors at runtime instead of hardcoding them in subclasses. +- [Decorator Pattern](https://java-design-patterns.com/patterns/decorator/): Can complement Template View for dynamically adding responsibilities to views. ## Real World Applications of Template View Pattern - Web frameworks like Spring MVC and Django use this concept to render views consistently. diff --git a/templateview/src/main/java/com/iluwater/templateview/App.java b/templateview/src/main/java/com/iluwatar/templateview/App.java similarity index 75% rename from templateview/src/main/java/com/iluwater/templateview/App.java rename to templateview/src/main/java/com/iluwatar/templateview/App.java index 5ffcb6cda1d1..55ec2b7728c2 100644 --- a/templateview/src/main/java/com/iluwater/templateview/App.java +++ b/templateview/src/main/java/com/iluwatar/templateview/App.java @@ -22,13 +22,20 @@ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ -package com.iluwater.templateview; +package com.iluwatar.templateview; import lombok.extern.slf4j.Slf4j; /** - * The App class demonstrates the Template View pattern. - * In this example, it renders different views such as the HomePage and ContactPage. + * Template View defines a consistent layout for rendering views, delegating dynamic content + * rendering to subclasses. + * + *

In this example, the {@link TemplateView} class provides the skeleton for rendering views + * with a header, dynamic content, and a footer. Subclasses {@link HomePageView} and + * {@link ContactPageView} define the specific dynamic content for their respective views. + * + *

The {@link App} class demonstrates the usage of the Template View Pattern by rendering + * instances of {@link HomePageView} and {@link ContactPageView}. */ @Slf4j public class App { diff --git a/templateview/src/main/java/com/iluwater/templateview/ContactPageView.java b/templateview/src/main/java/com/iluwatar/templateview/ContactPageView.java similarity index 97% rename from templateview/src/main/java/com/iluwater/templateview/ContactPageView.java rename to templateview/src/main/java/com/iluwatar/templateview/ContactPageView.java index c55ce8642c07..980ed48fa85d 100644 --- a/templateview/src/main/java/com/iluwater/templateview/ContactPageView.java +++ b/templateview/src/main/java/com/iluwatar/templateview/ContactPageView.java @@ -22,7 +22,7 @@ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ -package com.iluwater.templateview; +package com.iluwatar.templateview; import lombok.extern.slf4j.Slf4j; diff --git a/templateview/src/main/java/com/iluwater/templateview/HomePageView.java b/templateview/src/main/java/com/iluwatar/templateview/HomePageView.java similarity index 97% rename from templateview/src/main/java/com/iluwater/templateview/HomePageView.java rename to templateview/src/main/java/com/iluwatar/templateview/HomePageView.java index 3c83c25a7f72..704f7b58f5c2 100644 --- a/templateview/src/main/java/com/iluwater/templateview/HomePageView.java +++ b/templateview/src/main/java/com/iluwatar/templateview/HomePageView.java @@ -22,7 +22,7 @@ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ -package com.iluwater.templateview; +package com.iluwatar.templateview; import lombok.extern.slf4j.Slf4j; diff --git a/templateview/src/main/java/com/iluwater/templateview/TemplateView.java b/templateview/src/main/java/com/iluwatar/templateview/TemplateView.java similarity index 98% rename from templateview/src/main/java/com/iluwater/templateview/TemplateView.java rename to templateview/src/main/java/com/iluwatar/templateview/TemplateView.java index 81e2bb80ed9b..5d0fd36c5d00 100644 --- a/templateview/src/main/java/com/iluwater/templateview/TemplateView.java +++ b/templateview/src/main/java/com/iluwatar/templateview/TemplateView.java @@ -22,7 +22,7 @@ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ -package com.iluwater.templateview; +package com.iluwatar.templateview; import lombok.extern.slf4j.Slf4j; diff --git a/templateview/src/test/java/com/iluwater/templateview/AppTest.java b/templateview/src/test/java/com/iluwatar/templateview/AppTest.java similarity index 97% rename from templateview/src/test/java/com/iluwater/templateview/AppTest.java rename to templateview/src/test/java/com/iluwatar/templateview/AppTest.java index fb73abfc6c46..79bd38125bc6 100644 --- a/templateview/src/test/java/com/iluwater/templateview/AppTest.java +++ b/templateview/src/test/java/com/iluwatar/templateview/AppTest.java @@ -22,7 +22,7 @@ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ -package com.iluwater.templateview; +package com.iluwatar.templateview; import org.junit.jupiter.api.Test; import static org.junit.jupiter.api.Assertions.assertDoesNotThrow; diff --git a/templateview/src/test/java/com/iluwater/templateview/ContactPageViewTest.java b/templateview/src/test/java/com/iluwatar/templateview/ContactPageViewTest.java similarity index 98% rename from templateview/src/test/java/com/iluwater/templateview/ContactPageViewTest.java rename to templateview/src/test/java/com/iluwatar/templateview/ContactPageViewTest.java index 9af715c35ca7..203974a48608 100644 --- a/templateview/src/test/java/com/iluwater/templateview/ContactPageViewTest.java +++ b/templateview/src/test/java/com/iluwatar/templateview/ContactPageViewTest.java @@ -22,7 +22,7 @@ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ -package com.iluwater.templateview; +package com.iluwatar.templateview; import org.junit.jupiter.api.Test; import static org.mockito.Mockito.*; diff --git a/templateview/src/test/java/com/iluwater/templateview/HomePageViewTest.java b/templateview/src/test/java/com/iluwatar/templateview/HomePageViewTest.java similarity index 97% rename from templateview/src/test/java/com/iluwater/templateview/HomePageViewTest.java rename to templateview/src/test/java/com/iluwatar/templateview/HomePageViewTest.java index 2d9abfe5926e..33285a50107b 100644 --- a/templateview/src/test/java/com/iluwater/templateview/HomePageViewTest.java +++ b/templateview/src/test/java/com/iluwatar/templateview/HomePageViewTest.java @@ -22,7 +22,7 @@ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ -package com.iluwater.templateview; +package com.iluwatar.templateview; import org.junit.jupiter.api.Test; import static org.mockito.Mockito.*; diff --git a/templateview/src/test/java/com/iluwater/templateview/TemplateViewTest.java b/templateview/src/test/java/com/iluwatar/templateview/TemplateViewTest.java similarity index 98% rename from templateview/src/test/java/com/iluwater/templateview/TemplateViewTest.java rename to templateview/src/test/java/com/iluwatar/templateview/TemplateViewTest.java index 2051c2d722c9..ff53711583b0 100644 --- a/templateview/src/test/java/com/iluwater/templateview/TemplateViewTest.java +++ b/templateview/src/test/java/com/iluwatar/templateview/TemplateViewTest.java @@ -22,7 +22,7 @@ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ -package com.iluwater.templateview; +package com.iluwatar.templateview; import org.junit.jupiter.api.Test; import static org.mockito.Mockito.*;