add links to add a new X to: game, edition, purchase, session, device, platform
This commit is contained in:
		| @ -1,8 +1,9 @@ | ||||
| from random import choices | ||||
| from string import ascii_lowercase | ||||
| from typing import Any | ||||
| from typing import Any, Callable | ||||
|  | ||||
| from django.template.loader import render_to_string | ||||
| from django.urls import NoReverseMatch, reverse | ||||
| from django.utils.safestring import mark_safe | ||||
|  | ||||
|  | ||||
| @ -31,16 +32,68 @@ HTMLAttribute = tuple[str, str] | ||||
| HTMLTag = str | ||||
|  | ||||
|  | ||||
| def A(attributes: list[HTMLAttribute], children: list[HTMLTag] | HTMLTag) -> HTMLTag: | ||||
| def Component( | ||||
|     attributes: list[HTMLAttribute] = [], | ||||
|     children: list[HTMLTag] | HTMLTag = [], | ||||
|     template: str = "", | ||||
|     tag_name: str = "", | ||||
| ) -> HTMLTag: | ||||
|     if not tag_name and not template: | ||||
|         raise ValueError("One of template or tag_name is required.") | ||||
|     if isinstance(children, str): | ||||
|         children = [children] | ||||
|     childrenBlob = "\n".join(children) | ||||
|     attributesList = [f'{name} = "{value}"' for name, value in attributes] | ||||
|     attributesBlob = " ".join(attributesList) | ||||
|     tag: str = f"<a {attributesBlob}>{childrenBlob}</a>" | ||||
|     tag: str = "" | ||||
|     if tag_name != "": | ||||
|         tag = f"<a {attributesBlob}>{childrenBlob}</a>" | ||||
|     elif template != "": | ||||
|         tag = render_to_string( | ||||
|             template, | ||||
|             {name: value for name, value in attributes} | {"slot": "\n".join(children)}, | ||||
|         ) | ||||
|     return mark_safe(tag) | ||||
|  | ||||
|  | ||||
| def A( | ||||
|     attributes: list[HTMLAttribute] = [], | ||||
|     children: list[HTMLTag] | HTMLTag = [], | ||||
|     url: str | Callable[..., Any] = "", | ||||
| ): | ||||
|     """ | ||||
|     Returns the HTML tag "a". | ||||
|     "url" can either be: | ||||
|         - URL (string) | ||||
|         - path name passed to reverse() (string) | ||||
|         - function | ||||
|     """ | ||||
|     additional_attributes = [] | ||||
|     if url: | ||||
|         if type(url) is str: | ||||
|             try: | ||||
|                 url_result = reverse(url) | ||||
|             except NoReverseMatch: | ||||
|                 url_result = url | ||||
|         elif callable(url): | ||||
|             url_result = url() | ||||
|         else: | ||||
|             raise TypeError("'url' is neither str nor function.") | ||||
|         additional_attributes = [("href", url_result)] | ||||
|     return Component( | ||||
|         tag_name="a", attributes=attributes + additional_attributes, children=children | ||||
|     ) | ||||
|  | ||||
|  | ||||
| def Button( | ||||
|     attributes: list[HTMLAttribute] = [], | ||||
|     children: list[HTMLTag] | HTMLTag = [], | ||||
| ): | ||||
|     return Component( | ||||
|         template="cotton/button.html", attributes=attributes, children=children | ||||
|     ) | ||||
|  | ||||
|  | ||||
| def safe_division(numerator: int | float, denominator: int | float) -> int | float: | ||||
|     """ | ||||
|     Divides without triggering division by zero exception. | ||||
|  | ||||
| @ -1,4 +1,5 @@ | ||||
| <button type="button" | ||||
|         class="text-white bg-blue-700 hover:bg-blue-800 focus:ring-4 focus:ring-blue-300 font-medium rounded-lg text-sm px-5 py-2.5 me-2 mb-2 mt-2 dark:bg-blue-600 dark:hover:bg-blue-700 focus:outline-none dark:focus:ring-blue-800"> | ||||
|     {{ text }} | ||||
|     {{ slot }} | ||||
| </button> | ||||
|  | ||||
| @ -2,6 +2,11 @@ | ||||
| <div class="shadow-md sm:rounded-lg" hx-boost="false"> | ||||
|     <div class="relative overflow-x-auto sm:rounded-lg"> | ||||
|         <table class="w-full text-sm text-left rtl:text-right text-gray-500 dark:text-gray-400"> | ||||
|             {% if header_action %} | ||||
|                 <c-table-header> | ||||
|                 {{ header_action }} | ||||
|                 </c-table-header> | ||||
|             {% endif %} | ||||
|             <thead class="text-xs text-gray-700 uppercase bg-gray-50 dark:bg-gray-700 dark:text-gray-400"> | ||||
|                 <tr> | ||||
|                     {% for column in columns %}<th scope="col" class="px-6 py-3">{{ column }}</th>{% endfor %} | ||||
|  | ||||
							
								
								
									
										3
									
								
								games/templates/cotton/table_header.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								games/templates/cotton/table_header.html
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,3 @@ | ||||
| <caption class="p-2 text-lg font-semibold rtl:text-left text-right text-gray-900 bg-white dark:text-white dark:bg-gray-900"> | ||||
|     {{ slot }} | ||||
| </caption> | ||||
| @ -5,6 +5,6 @@ | ||||
| {% endblock title %} | ||||
| {% block content %} | ||||
|     <div class="2xl:max-w-screen-2xl md:max-w-screen-md sm:max-w-screen-sm self-center"> | ||||
|         <c-simple-table :columns=data.columns :rows=data.rows :page_obj=page_obj :elided_page_range=elided_page_range /> | ||||
|         <c-simple-table :columns=data.columns :rows=data.rows :page_obj=page_obj :elided_page_range=elided_page_range :header_action=data.header_action /> | ||||
|     </div> | ||||
| {% endblock content %} | ||||
|  | ||||
| @ -7,6 +7,7 @@ from django.shortcuts import get_object_or_404, redirect, render | ||||
| from django.template.loader import render_to_string | ||||
| from django.urls import reverse | ||||
|  | ||||
| from common.utils import A, Button | ||||
| from games.forms import DeviceForm | ||||
| from games.models import Device | ||||
| from games.views.general import dateformat | ||||
| @ -35,6 +36,7 @@ def list_devices(request: HttpRequest) -> HttpResponse: | ||||
|             else None | ||||
|         ), | ||||
|         "data": { | ||||
|             "header_action": A([], Button([], "Add device"), url="add_device"), | ||||
|             "columns": [ | ||||
|                 "Name", | ||||
|                 "Type", | ||||
|  | ||||
| @ -7,7 +7,7 @@ from django.shortcuts import get_object_or_404, redirect, render | ||||
| from django.template.loader import render_to_string | ||||
| from django.urls import reverse | ||||
|  | ||||
| from common.utils import A, truncate_with_popover | ||||
| from common.utils import A, Button, truncate_with_popover | ||||
| from games.forms import EditionForm | ||||
| from games.models import Edition, Game | ||||
| from games.views.general import dateformat | ||||
| @ -36,6 +36,7 @@ def list_editions(request: HttpRequest) -> HttpResponse: | ||||
|             else None | ||||
|         ), | ||||
|         "data": { | ||||
|             "header_action": A([], Button([], "Add edition"), url="add_edition"), | ||||
|             "columns": [ | ||||
|                 "Game", | ||||
|                 "Name", | ||||
|  | ||||
| @ -9,7 +9,7 @@ from django.template.loader import render_to_string | ||||
| from django.urls import reverse | ||||
|  | ||||
| from common.time import format_duration | ||||
| from common.utils import A, safe_division, truncate_with_popover | ||||
| from common.utils import A, Button, safe_division, truncate_with_popover | ||||
| from games.forms import GameForm | ||||
| from games.models import Edition, Game, Purchase, Session | ||||
| from games.views.general import ( | ||||
| @ -45,6 +45,7 @@ def list_games(request: HttpRequest) -> HttpResponse: | ||||
|             else None | ||||
|         ), | ||||
|         "data": { | ||||
|             "header_action": A([], Button([], "Add game"), url="add_game"), | ||||
|             "columns": [ | ||||
|                 "Name", | ||||
|                 "Sort Name", | ||||
|  | ||||
| @ -7,6 +7,7 @@ from django.shortcuts import get_object_or_404, redirect, render | ||||
| from django.template.loader import render_to_string | ||||
| from django.urls import reverse | ||||
|  | ||||
| from common.utils import A, Button | ||||
| from games.forms import PlatformForm | ||||
| from games.models import Platform | ||||
| from games.views.general import dateformat, use_custom_redirect | ||||
| @ -35,6 +36,7 @@ def list_platforms(request: HttpRequest) -> HttpResponse: | ||||
|             else None | ||||
|         ), | ||||
|         "data": { | ||||
|             "header_action": A([], Button([], "Add platform"), url="add_platform"), | ||||
|             "columns": [ | ||||
|                 "Name", | ||||
|                 "Group", | ||||
|  | ||||
| @ -13,7 +13,7 @@ from django.template.loader import render_to_string | ||||
| from django.urls import reverse | ||||
| from django.utils import timezone | ||||
|  | ||||
| from common.utils import A, truncate_with_popover | ||||
| from common.utils import A, Button, truncate_with_popover | ||||
| from games.forms import PurchaseForm | ||||
| from games.models import Edition, Purchase | ||||
| from games.views.general import dateformat, use_custom_redirect | ||||
| @ -42,6 +42,7 @@ def list_purchases(request: HttpRequest) -> HttpResponse: | ||||
|             else None | ||||
|         ), | ||||
|         "data": { | ||||
|             "header_action": A([], Button([], "Add purchase"), url="add_purchase"), | ||||
|             "columns": [ | ||||
|                 "Name", | ||||
|                 "Platform", | ||||
|  | ||||
| @ -9,7 +9,7 @@ from django.urls import reverse | ||||
| from django.utils import timezone | ||||
|  | ||||
| from common.time import format_duration | ||||
| from common.utils import A, truncate_with_popover | ||||
| from common.utils import A, Button, truncate_with_popover | ||||
| from games.forms import SessionForm | ||||
| from games.models import Purchase, Session | ||||
| from games.views.general import ( | ||||
| @ -45,6 +45,7 @@ def list_sessions(request: HttpRequest) -> HttpResponse: | ||||
|             else None | ||||
|         ), | ||||
|         "data": { | ||||
|             "header_action": A([], Button([], "Add session"), url="add_session"), | ||||
|             "columns": [ | ||||
|                 "Name", | ||||
|                 "Date", | ||||
| @ -57,16 +58,11 @@ def list_sessions(request: HttpRequest) -> HttpResponse: | ||||
|             "rows": [ | ||||
|                 [ | ||||
|                     A( | ||||
|                         [ | ||||
|                             ( | ||||
|                                 "href", | ||||
|                                 reverse( | ||||
|                                     "view_game", | ||||
|                                     args=[session.purchase.edition.game.pk], | ||||
|                                 ), | ||||
|                             ) | ||||
|                         ], | ||||
|                         truncate_with_popover(session.purchase.edition.name), | ||||
|                         children=truncate_with_popover(session.purchase.edition.name), | ||||
|                         url=reverse( | ||||
|                             "view_game", | ||||
|                             args=[session.purchase.edition.game.pk], | ||||
|                         ), | ||||
|                     ), | ||||
|                     f"{session.timestamp_start.strftime(datetimeformat)}{f" — {session.timestamp_end.strftime(timeformat)}" if session.timestamp_end else ""}", | ||||
|                     ( | ||||
|  | ||||
		Reference in New Issue
	
	Block a user