Add remaining files (with AI taint)
This commit is contained in:
parent
a611362ad8
commit
b8660d3200
27
karaokatalog/ui/static/components/pieces/ClearSearchLink.js
Normal file
27
karaokatalog/ui/static/components/pieces/ClearSearchLink.js
Normal file
@ -0,0 +1,27 @@
|
||||
/* TODO: TAINT: AI-generated. This file was AI-generated (Gemini and/or Cursor), and I haven't manually reviewed it yet.
|
||||
I should do this at some point in the future and change things I don't like. */
|
||||
import search from "../../model/search.js"
|
||||
|
||||
export default {
|
||||
view: function (vnode) {
|
||||
return ["Möchtest du ", m(
|
||||
"a.clear-search-link",
|
||||
{
|
||||
onclick: e => {
|
||||
e.preventDefault()
|
||||
search.query = null
|
||||
},
|
||||
href: "#",
|
||||
role: "button",
|
||||
tabindex: 0,
|
||||
onkeypress: e => {
|
||||
if ((e.key === "Enter" || e.key === " ")) {
|
||||
e.preventDefault()
|
||||
search.query = null
|
||||
}
|
||||
},
|
||||
},
|
||||
`alle ${vnode.attrs.totalFavoriteCount} Favoriten anzeigen`,
|
||||
), "?"]
|
||||
},
|
||||
}
|
||||
122
karaokatalog/ui/static/components/pieces/PaginationControls.js
Normal file
122
karaokatalog/ui/static/components/pieces/PaginationControls.js
Normal file
@ -0,0 +1,122 @@
|
||||
/* TODO: TAINT: AI-generated. This file was AI-generated (Gemini and/or Cursor), and I haven't manually reviewed it yet.
|
||||
I should do this at some point in the future and change things I don't like. */
|
||||
const PaginationControls = {
|
||||
view: function (vnode) {
|
||||
const { currentPage, totalPages, onPageChange } = vnode.attrs
|
||||
|
||||
if (totalPages <= 1) {
|
||||
return null // Keine Steuerelemente anzeigen, wenn nicht benötigt
|
||||
}
|
||||
|
||||
const pages = []
|
||||
const delta = 2 // Anzahl der Seiten, die um die aktuelle Seite herum angezeigt werden
|
||||
|
||||
// Hilfsfunktion zum Hinzufügen einer Seitenzahl
|
||||
const addPage = pageNumber => {
|
||||
pages.push(
|
||||
m(
|
||||
"span.page-number",
|
||||
{
|
||||
class:
|
||||
pageNumber === currentPage ? "active" : "clickable",
|
||||
onclick: () =>
|
||||
pageNumber !== currentPage
|
||||
? onPageChange(pageNumber)
|
||||
: null,
|
||||
role: "button",
|
||||
tabindex: 0, // Für Tastaturnavigation
|
||||
onkeypress: e => {
|
||||
// Für Tastaturnavigation (Enter/Space)
|
||||
if (
|
||||
(e.key === "Enter" || e.key === " ") &&
|
||||
pageNumber !== currentPage
|
||||
) {
|
||||
e.preventDefault()
|
||||
onPageChange(pageNumber)
|
||||
}
|
||||
},
|
||||
"aria-label": `Gehe zu Seite ${pageNumber}`,
|
||||
"aria-current":
|
||||
pageNumber === currentPage ? "page" : undefined,
|
||||
},
|
||||
pageNumber,
|
||||
),
|
||||
)
|
||||
}
|
||||
|
||||
// "Zurück"-Button
|
||||
const prevButton = m(
|
||||
"button.pagination-button",
|
||||
{
|
||||
onclick: () => onPageChange(currentPage - 1),
|
||||
disabled: currentPage === 1,
|
||||
"aria-label": "Vorherige Seite",
|
||||
},
|
||||
"Zurück",
|
||||
)
|
||||
|
||||
// "Weiter"-Button
|
||||
const nextButton = m(
|
||||
"button.pagination-button",
|
||||
{
|
||||
onclick: () => onPageChange(currentPage + 1),
|
||||
disabled: currentPage === totalPages,
|
||||
"aria-label": "Nächste Seite",
|
||||
},
|
||||
"Weiter",
|
||||
)
|
||||
|
||||
// Logik zur Seitenzahlanzeige
|
||||
if (totalPages <= 7) {
|
||||
// Alle Seiten anzeigen, wenn es 7 oder weniger sind
|
||||
for (let i = 1; i <= totalPages; i++) {
|
||||
addPage(i)
|
||||
}
|
||||
} else {
|
||||
addPage(1) // Immer die erste Seite anzeigen
|
||||
|
||||
if (currentPage > delta + 2) {
|
||||
// Auslassungspunkte anzeigen, wenn die aktuelle Seite weit vom Anfang entfernt ist
|
||||
pages.push(m("span.dots", "..."))
|
||||
}
|
||||
|
||||
// Bereich der Seiten um die aktuelle Seite bestimmen
|
||||
let start = Math.max(2, currentPage - delta)
|
||||
let end = Math.min(totalPages - 1, currentPage + delta)
|
||||
|
||||
if (currentPage <= delta + 1) {
|
||||
// Aktuelle Seite ist nahe am Anfang
|
||||
start = 2
|
||||
end = Math.min(totalPages - 1, 2 * delta + 1)
|
||||
} else if (currentPage >= totalPages - delta) {
|
||||
// Aktuelle Seite ist nahe am Ende
|
||||
start = Math.max(2, totalPages - 2 * delta)
|
||||
end = totalPages - 1
|
||||
}
|
||||
|
||||
// Korrektur für den Fall, dass start und end sich überschneiden oder ungültig werden
|
||||
if (start > end && currentPage < totalPages / 2) {
|
||||
// Wenn start > end und wir sind eher am Anfang
|
||||
end = start // Zeige zumindest die Startseite
|
||||
} else if (start > end && currentPage > totalPages / 2) {
|
||||
// Wenn start > end und wir sind eher am Ende
|
||||
start = end // Zeige zumindest die Endseite
|
||||
}
|
||||
|
||||
for (let i = start; i <= end; i++) {
|
||||
if (i > 0) addPage(i) // Sicherstellen, dass i positiv ist
|
||||
}
|
||||
|
||||
if (currentPage < totalPages - delta - 1 && end < totalPages - 1) {
|
||||
// Auslassungspunkte anzeigen, wenn die aktuelle Seite weit vom Ende entfernt ist
|
||||
pages.push(m("span.dots", "..."))
|
||||
}
|
||||
|
||||
if (totalPages > 1) addPage(totalPages) // Immer die letzte Seite anzeigen (wenn nicht 1)
|
||||
}
|
||||
|
||||
return m("div.pagination-controls", [prevButton, ...pages, nextButton])
|
||||
},
|
||||
}
|
||||
|
||||
export default PaginationControls
|
||||
16
karaokatalog/ui/static/components/pieces/Tab.js
Normal file
16
karaokatalog/ui/static/components/pieces/Tab.js
Normal file
@ -0,0 +1,16 @@
|
||||
/* TODO: TAINT: AI-generated. This file was AI-generated (Gemini and/or Cursor), and I haven't manually reviewed it yet.
|
||||
I should do this at some point in the future and change things I don't like. */
|
||||
export const Tab = {
|
||||
view(vnode) {
|
||||
return m(
|
||||
m.route.Link,
|
||||
{
|
||||
class: m.route.get() === vnode.attrs.href ? "active" : "",
|
||||
href: vnode.attrs.href,
|
||||
role: "button", // Für Barrierefreiheit
|
||||
tabindex: 0, // Für Tastaturnavigation
|
||||
},
|
||||
vnode.attrs.label,
|
||||
)
|
||||
},
|
||||
}
|
||||
275
karaokatalog/ui/static/style.css
Normal file
275
karaokatalog/ui/static/style.css
Normal file
@ -0,0 +1,275 @@
|
||||
/* TODO: TAINT: AI-generated. This file was AI-generated (Gemini and/or Cursor), and I haven't manually reviewed it yet.
|
||||
I should do this at some point in the future and change things I don't like. */
|
||||
|
||||
/* Globale Resets und Basis-Styling (wie vorher) */
|
||||
body,
|
||||
html {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
font-family: "Roboto", sans-serif;
|
||||
background-color: #f4f4f9;
|
||||
color: #333;
|
||||
line-height: 1.6;
|
||||
}
|
||||
|
||||
main > p {
|
||||
text-align: center;
|
||||
padding: 1em;
|
||||
font-size: 1.1em;
|
||||
color: #757575;
|
||||
}
|
||||
|
||||
/* Suchleiste (Höhe ca. 60px) */
|
||||
search {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
background-color: #3f51b5; /* Indigo - Material Primary */
|
||||
padding: 12px 16px;
|
||||
z-index: 1000;
|
||||
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.2);
|
||||
box-sizing: border-box; /* Stellt sicher, dass Padding die Höhe nicht sprengt */
|
||||
height: 60px; /* Feste Höhe für genaue Positionierung */
|
||||
}
|
||||
|
||||
input[type="search"] {
|
||||
width: 100%;
|
||||
padding: 10px 12px;
|
||||
border-radius: 4px;
|
||||
border: none;
|
||||
font-size: 16px;
|
||||
box-sizing: border-box;
|
||||
background-color: #fff;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
input[type="search"]::placeholder {
|
||||
color: #888;
|
||||
}
|
||||
|
||||
/* Tab Bar (Höhe ca. 48px) */
|
||||
nav {
|
||||
display: flex;
|
||||
position: fixed;
|
||||
top: 60px; /* Direkt unter der Suchleiste */
|
||||
left: 0;
|
||||
right: 0;
|
||||
background-color: #3f51b5; /* Gleiche Farbe wie Suchleiste für einen einheitlichen Header-Block */
|
||||
z-index: 999; /* Unter der Suchleiste, falls sie überlappen könnten */
|
||||
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.15); /* Dezenter Schatten */
|
||||
height: 48px; /* Feste Höhe */
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
nav a {
|
||||
flex-grow: 1;
|
||||
display: flex; /* Für vertikales Zentrieren des Texts */
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
padding: 0 8px; /* Horizontal Padding */
|
||||
text-align: center;
|
||||
color: rgba(
|
||||
255,
|
||||
255,
|
||||
255,
|
||||
0.75
|
||||
); /* Etwas helleres, aber gedämpftes Weiß für inaktive Tabs */
|
||||
cursor: pointer;
|
||||
font-weight: 500;
|
||||
font-size: 0.9em; /* Leicht kleiner als Haupttext */
|
||||
text-transform: uppercase;
|
||||
border-bottom: 3px solid transparent; /* Platzhalter für den aktiven Indikator */
|
||||
transition:
|
||||
color 0.2s ease-in-out,
|
||||
border-bottom-color 0.2s ease-in-out;
|
||||
user-select: none;
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
nav a:hover {
|
||||
background-color: rgba(255, 255, 255, 0.1); /* Leichter Hover-Effekt */
|
||||
}
|
||||
|
||||
nav a.active {
|
||||
color: #ffffff; /* Reines Weiß für aktiven Tab */
|
||||
border-bottom-color: #ff4081; /* Material Accent Pink für den Indikator */
|
||||
}
|
||||
|
||||
/* song .Liste Container */
|
||||
main {
|
||||
padding-top: calc(60px + 48px); /* Höhe Suchleiste + Höhe TabBar */
|
||||
padding-bottom: 20px; /* Platz für Paginierung, falls am Ende */
|
||||
}
|
||||
|
||||
/* Link zum Zurücksetzen der Suche in der main > p */
|
||||
main > p a.clear-search-link,
|
||||
main > p .clear-search-link {
|
||||
/* Zweite Regel für den Fall, dass Mithril das 'a' nicht direkt rendert */
|
||||
color: #3f51b5; /* Material Indigo, passend zum Header */
|
||||
text-decoration: underline;
|
||||
cursor: pointer;
|
||||
font-weight: 500; /* Etwas hervorheben */
|
||||
background: none; /* Sicherstellen, dass keine Button-Styles übernommen werden */
|
||||
border: none;
|
||||
padding: 0;
|
||||
display: inline; /* Damit es sich wie ein normaler Link im Text verhält */
|
||||
}
|
||||
|
||||
main > p a.clear-search-link:hover,
|
||||
main > p a.clear-search-link:focus,
|
||||
main > p .clear-search-link:hover,
|
||||
main > p .clear-search-link:focus {
|
||||
color: #303f9f; /* Dunkleres Indigo für Hover/Focus */
|
||||
text-decoration: none; /* Optional: Unterstreichung bei Hover entfernen */
|
||||
}
|
||||
|
||||
/* Restliche Styles für Artist-Group, song .Item etc. bleiben wie zuvor */
|
||||
section.artist {
|
||||
margin: 12px 8px;
|
||||
background-color: #ffffff;
|
||||
border-radius: 4px;
|
||||
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
section.artist > h2 {
|
||||
margin: 0;
|
||||
padding: 12px 16px;
|
||||
background-color: #e8eaf6;
|
||||
color: #303f9f;
|
||||
font-size: 1.1em;
|
||||
font-weight: 500;
|
||||
border-bottom: 1px solid #c5cae9;
|
||||
}
|
||||
|
||||
.song {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
padding: 12px 16px;
|
||||
border-bottom: 1px solid #f0f0f0;
|
||||
transition: background-color 0.2s ease-in-out;
|
||||
}
|
||||
|
||||
.song:last-child {
|
||||
border-bottom: none;
|
||||
}
|
||||
|
||||
.song:hover {
|
||||
background-color: #f9f9f9;
|
||||
}
|
||||
|
||||
.song .info {
|
||||
flex-grow: 1;
|
||||
margin-right: 8px; /* Abstand zum Icon */
|
||||
overflow: hidden; /* Verhindert Textüberlauf */
|
||||
}
|
||||
|
||||
.song .title {
|
||||
font-size: 1em;
|
||||
font-weight: 400;
|
||||
color: #212121;
|
||||
white-space: nowrap;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis; /* Text abschneiden bei Überlauf */
|
||||
}
|
||||
|
||||
.song .artist {
|
||||
font-size: 0.85em;
|
||||
color: #757575;
|
||||
white-space: nowrap;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
}
|
||||
|
||||
.song .actions {
|
||||
margin-left: auto;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.heart-icon {
|
||||
cursor: pointer;
|
||||
color: #bdbdbd;
|
||||
font-size: 24px;
|
||||
user-select: none;
|
||||
transition:
|
||||
color 0.2s ease-in-out,
|
||||
transform 0.1s ease-out;
|
||||
vertical-align: middle;
|
||||
}
|
||||
|
||||
.heart-icon:hover {
|
||||
transform: scale(1.1);
|
||||
}
|
||||
|
||||
.heart-icon.is-favorite {
|
||||
color: #ff4081;
|
||||
font-variation-settings: "FILL" 1;
|
||||
}
|
||||
|
||||
.material-symbols-outlined {
|
||||
font-variation-settings:
|
||||
"FILL" 0,
|
||||
"wght" 400,
|
||||
"GRAD" 0,
|
||||
"opsz" 24;
|
||||
}
|
||||
|
||||
/* NEU: Pagination Controls Styles */
|
||||
.pagination-controls {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
padding: 20px 0; /* Mehr Padding für bessere Sichtbarkeit */
|
||||
margin-top: 10px;
|
||||
user-select: none;
|
||||
flex-wrap: wrap; /* Erlaubt Umbruch bei vielen Seitenzahlen auf kleinen Bildschirmen */
|
||||
}
|
||||
|
||||
.pagination-controls button.pagination-button, /* Klasse für Buttons hinzugefügt */
|
||||
.pagination-controls span.page-number {
|
||||
margin: 4px; /* Etwas mehr Margin für Umbruch */
|
||||
padding: 8px 12px;
|
||||
border: 1px solid #ddd;
|
||||
background-color: #fff;
|
||||
color: #3f51b5; /* Material Indigo */
|
||||
cursor: pointer;
|
||||
border-radius: 4px;
|
||||
transition:
|
||||
background-color 0.2s,
|
||||
color 0.2s,
|
||||
border-color 0.2s;
|
||||
font-size: 0.9em;
|
||||
min-width: 36px; /* Mindestbreite für bessere Klickbarkeit */
|
||||
text-align: center;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
.pagination-controls button.pagination-button:hover:not(:disabled),
|
||||
.pagination-controls span.page-number.clickable:hover {
|
||||
background-color: #e8eaf6; /* Heller Indigo-Ton für Hover */
|
||||
border-color: #c5cae9;
|
||||
}
|
||||
|
||||
.pagination-controls button.pagination-button:disabled {
|
||||
color: #aaa;
|
||||
cursor: not-allowed;
|
||||
background-color: #f9f9f9;
|
||||
border-color: #eee;
|
||||
}
|
||||
|
||||
.pagination-controls span.page-number.active {
|
||||
background-color: #3f51b5;
|
||||
color: #fff;
|
||||
border-color: #3f51b5;
|
||||
font-weight: bold;
|
||||
cursor: default;
|
||||
}
|
||||
|
||||
.pagination-controls span.dots {
|
||||
padding: 8px 6px;
|
||||
color: #777;
|
||||
margin: 4px;
|
||||
font-size: 0.9em;
|
||||
}
|
||||
Loading…
x
Reference in New Issue
Block a user