Compare commits
3 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 1034ea72a8 | |||
| d3c58c5dd2 | |||
| 43fdd7081c |
+15
-3
@@ -52,13 +52,25 @@ def export_backup_archive(
|
|||||||
payload["tables"][table_name] = [dict(row) for row in rows]
|
payload["tables"][table_name] = [dict(row) for row in rows]
|
||||||
|
|
||||||
uploads_root = Path(upload_folder)
|
uploads_root = Path(upload_folder)
|
||||||
with zipfile.ZipFile(archive_path, "w", compression=zipfile.ZIP_DEFLATED) as archive:
|
uploads_snapshot_dir = Path(tempfile.mkdtemp(prefix="nouri-backup-uploads-"))
|
||||||
archive.writestr("backup.json", json.dumps(payload, ensure_ascii=False, indent=2))
|
try:
|
||||||
if uploads_root.exists():
|
if uploads_root.exists():
|
||||||
for file_path in uploads_root.rglob("*"):
|
for file_path in uploads_root.rglob("*"):
|
||||||
if file_path.is_file():
|
if not file_path.is_file():
|
||||||
|
continue
|
||||||
relative_path = file_path.relative_to(uploads_root)
|
relative_path = file_path.relative_to(uploads_root)
|
||||||
|
snapshot_path = uploads_snapshot_dir / relative_path
|
||||||
|
snapshot_path.parent.mkdir(parents=True, exist_ok=True)
|
||||||
|
shutil.copy2(file_path, snapshot_path)
|
||||||
|
|
||||||
|
with zipfile.ZipFile(archive_path, "w", compression=zipfile.ZIP_DEFLATED) as archive:
|
||||||
|
archive.writestr("backup.json", json.dumps(payload, ensure_ascii=False, indent=2))
|
||||||
|
for file_path in uploads_snapshot_dir.rglob("*"):
|
||||||
|
if file_path.is_file():
|
||||||
|
relative_path = file_path.relative_to(uploads_snapshot_dir)
|
||||||
archive.write(file_path, f"uploads/{relative_path.as_posix()}")
|
archive.write(file_path, f"uploads/{relative_path.as_posix()}")
|
||||||
|
finally:
|
||||||
|
shutil.rmtree(uploads_snapshot_dir, ignore_errors=True)
|
||||||
|
|
||||||
return archive_path, backup_name
|
return archive_path, backup_name
|
||||||
|
|
||||||
|
|||||||
+11
@@ -487,6 +487,7 @@ def bootstrap_legacy_schema(database: sqlite3.Connection) -> None:
|
|||||||
add_column_if_missing(database, "shopping_entries", "household_id INTEGER")
|
add_column_if_missing(database, "shopping_entries", "household_id INTEGER")
|
||||||
add_column_if_missing(database, "shopping_entries", "owner_user_id INTEGER")
|
add_column_if_missing(database, "shopping_entries", "owner_user_id INTEGER")
|
||||||
add_column_if_missing(database, "shopping_entries", "visibility TEXT NOT NULL DEFAULT 'shared'")
|
add_column_if_missing(database, "shopping_entries", "visibility TEXT NOT NULL DEFAULT 'shared'")
|
||||||
|
add_column_if_missing(database, "shopping_entries", "shopping_note TEXT NOT NULL DEFAULT ''")
|
||||||
add_column_if_missing(database, "shopping_entries", "needed_for_date TEXT")
|
add_column_if_missing(database, "shopping_entries", "needed_for_date TEXT")
|
||||||
add_column_if_missing(database, "shopping_entries", "needed_for_daypart_id INTEGER")
|
add_column_if_missing(database, "shopping_entries", "needed_for_daypart_id INTEGER")
|
||||||
|
|
||||||
@@ -740,6 +741,7 @@ def ensure_schema_upgrades(database: sqlite3.Connection) -> None:
|
|||||||
add_column_if_missing(database, "items", "is_quick_added INTEGER NOT NULL DEFAULT 0")
|
add_column_if_missing(database, "items", "is_quick_added INTEGER NOT NULL DEFAULT 0")
|
||||||
add_column_if_missing(database, "shopping_entries", "needed_for_date TEXT")
|
add_column_if_missing(database, "shopping_entries", "needed_for_date TEXT")
|
||||||
add_column_if_missing(database, "shopping_entries", "needed_for_daypart_id INTEGER")
|
add_column_if_missing(database, "shopping_entries", "needed_for_daypart_id INTEGER")
|
||||||
|
add_column_if_missing(database, "shopping_entries", "shopping_note TEXT NOT NULL DEFAULT ''")
|
||||||
add_column_if_missing(database, "user_settings", "suggestion_style TEXT NOT NULL DEFAULT 'balanced'")
|
add_column_if_missing(database, "user_settings", "suggestion_style TEXT NOT NULL DEFAULT 'balanced'")
|
||||||
add_column_if_missing(database, "user_settings", "energy_preference TEXT NOT NULL DEFAULT 'neutral'")
|
add_column_if_missing(database, "user_settings", "energy_preference TEXT NOT NULL DEFAULT 'neutral'")
|
||||||
add_column_if_missing(database, "user_settings", "protein_preference TEXT NOT NULL DEFAULT 'mixed'")
|
add_column_if_missing(database, "user_settings", "protein_preference TEXT NOT NULL DEFAULT 'mixed'")
|
||||||
@@ -803,6 +805,7 @@ def ensure_schema_upgrades(database: sqlite3.Connection) -> None:
|
|||||||
database.execute("UPDATE items SET meal_tags = '' WHERE meal_tags IS NULL")
|
database.execute("UPDATE items SET meal_tags = '' WHERE meal_tags IS NULL")
|
||||||
database.execute("UPDATE items SET is_archived = 0 WHERE is_archived IS NULL")
|
database.execute("UPDATE items SET is_archived = 0 WHERE is_archived IS NULL")
|
||||||
database.execute("UPDATE items SET is_quick_added = 0 WHERE is_quick_added IS NULL")
|
database.execute("UPDATE items SET is_quick_added = 0 WHERE is_quick_added IS NULL")
|
||||||
|
database.execute("UPDATE shopping_entries SET shopping_note = '' WHERE shopping_note IS NULL")
|
||||||
database.execute("UPDATE user_settings SET suggestion_style = 'balanced' WHERE suggestion_style IS NULL OR suggestion_style = ''")
|
database.execute("UPDATE user_settings SET suggestion_style = 'balanced' WHERE suggestion_style IS NULL OR suggestion_style = ''")
|
||||||
database.execute("UPDATE user_settings SET energy_preference = 'neutral' WHERE energy_preference IS NULL OR energy_preference = ''")
|
database.execute("UPDATE user_settings SET energy_preference = 'neutral' WHERE energy_preference IS NULL OR energy_preference = ''")
|
||||||
database.execute("UPDATE user_settings SET protein_preference = 'mixed' WHERE protein_preference IS NULL OR protein_preference = ''")
|
database.execute("UPDATE user_settings SET protein_preference = 'mixed' WHERE protein_preference IS NULL OR protein_preference = ''")
|
||||||
@@ -848,6 +851,14 @@ def ensure_schema_upgrades(database: sqlite3.Connection) -> None:
|
|||||||
ON shopping_entries (household_id, visibility, is_checked)
|
ON shopping_entries (household_id, visibility, is_checked)
|
||||||
"""
|
"""
|
||||||
)
|
)
|
||||||
|
database.execute("DROP INDEX IF EXISTS idx_shopping_entries_open_item")
|
||||||
|
database.execute(
|
||||||
|
"""
|
||||||
|
CREATE UNIQUE INDEX IF NOT EXISTS idx_shopping_entries_open_item
|
||||||
|
ON shopping_entries (item_id, COALESCE(shopping_note, ''))
|
||||||
|
WHERE is_checked = 0
|
||||||
|
"""
|
||||||
|
)
|
||||||
database.execute(
|
database.execute(
|
||||||
"""
|
"""
|
||||||
CREATE INDEX IF NOT EXISTS idx_shopping_needs_household_activation
|
CREATE INDEX IF NOT EXISTS idx_shopping_needs_household_activation
|
||||||
|
|||||||
+120
-20
@@ -9,7 +9,6 @@ import sqlite3
|
|||||||
|
|
||||||
from flask import (
|
from flask import (
|
||||||
Blueprint,
|
Blueprint,
|
||||||
after_this_request,
|
|
||||||
current_app,
|
current_app,
|
||||||
flash,
|
flash,
|
||||||
g,
|
g,
|
||||||
@@ -1002,6 +1001,10 @@ def should_activate_shopping_need(needed_date: date, today: date | None = None)
|
|||||||
return (today or date.today()) >= shopping_activation_date_for(needed_date)
|
return (today or date.today()) >= shopping_activation_date_for(needed_date)
|
||||||
|
|
||||||
|
|
||||||
|
def normalize_shopping_note(value: str | None) -> str:
|
||||||
|
return " ".join((value or "").strip().split())[:80]
|
||||||
|
|
||||||
|
|
||||||
def schedule_shopping_need(
|
def schedule_shopping_need(
|
||||||
*,
|
*,
|
||||||
item_id: int,
|
item_id: int,
|
||||||
@@ -1083,14 +1086,16 @@ def add_to_shopping_list(
|
|||||||
visibility_override: str | None = None,
|
visibility_override: str | None = None,
|
||||||
needed_for_date: str | None = None,
|
needed_for_date: str | None = None,
|
||||||
needed_for_daypart_id: int | None = None,
|
needed_for_daypart_id: int | None = None,
|
||||||
|
shopping_note: str | None = None,
|
||||||
) -> bool:
|
) -> bool:
|
||||||
item = get_item(item_id)
|
item = get_item(item_id)
|
||||||
|
normalized_note = normalize_shopping_note(shopping_note)
|
||||||
existing = get_db().execute(
|
existing = get_db().execute(
|
||||||
"""
|
"""
|
||||||
SELECT id FROM shopping_entries
|
SELECT id FROM shopping_entries
|
||||||
WHERE item_id = ? AND is_checked = 0
|
WHERE item_id = ? AND shopping_note = ? AND is_checked = 0
|
||||||
""",
|
""",
|
||||||
(item_id,),
|
(item_id, normalized_note),
|
||||||
).fetchone()
|
).fetchone()
|
||||||
if existing:
|
if existing:
|
||||||
return False
|
return False
|
||||||
@@ -1100,15 +1105,16 @@ def add_to_shopping_list(
|
|||||||
get_db().execute(
|
get_db().execute(
|
||||||
"""
|
"""
|
||||||
INSERT INTO shopping_entries (
|
INSERT INTO shopping_entries (
|
||||||
household_id, owner_user_id, visibility, item_id, added_by, needed_for_date, needed_for_daypart_id
|
household_id, owner_user_id, visibility, item_id, shopping_note, added_by, needed_for_date, needed_for_daypart_id
|
||||||
)
|
)
|
||||||
VALUES (?, ?, ?, ?, ?, ?, ?)
|
VALUES (?, ?, ?, ?, ?, ?, ?, ?)
|
||||||
""",
|
""",
|
||||||
(
|
(
|
||||||
current_household_id(),
|
current_household_id(),
|
||||||
owner_user_id,
|
owner_user_id,
|
||||||
visibility,
|
visibility,
|
||||||
item_id,
|
item_id,
|
||||||
|
normalized_note,
|
||||||
user_id,
|
user_id,
|
||||||
needed_for_date,
|
needed_for_date,
|
||||||
needed_for_daypart_id,
|
needed_for_daypart_id,
|
||||||
@@ -1151,6 +1157,7 @@ def ensure_item_or_missing_components_are_shopped(
|
|||||||
needed_for_date: str | None = None,
|
needed_for_date: str | None = None,
|
||||||
needed_for_daypart_id: int | None = None,
|
needed_for_daypart_id: int | None = None,
|
||||||
source_item_id: int | None = None,
|
source_item_id: int | None = None,
|
||||||
|
shopping_note: str | None = None,
|
||||||
) -> dict:
|
) -> dict:
|
||||||
item = get_item(item_id)
|
item = get_item(item_id)
|
||||||
if item["kind"] == "meal":
|
if item["kind"] == "meal":
|
||||||
@@ -1220,6 +1227,7 @@ def ensure_item_or_missing_components_are_shopped(
|
|||||||
visibility_override=visibility,
|
visibility_override=visibility,
|
||||||
needed_for_date=needed_for_date,
|
needed_for_date=needed_for_date,
|
||||||
needed_for_daypart_id=needed_for_daypart_id,
|
needed_for_daypart_id=needed_for_daypart_id,
|
||||||
|
shopping_note=shopping_note,
|
||||||
)
|
)
|
||||||
return {
|
return {
|
||||||
"added": added,
|
"added": added,
|
||||||
@@ -1290,6 +1298,39 @@ def fetch_items_by_ids(item_ids: list[int]) -> list[dict]:
|
|||||||
return [items_by_id[item_id] for item_id in normalized_ids if item_id in items_by_id]
|
return [items_by_id[item_id] for item_id in normalized_ids if item_id in items_by_id]
|
||||||
|
|
||||||
|
|
||||||
|
def find_shopping_food_by_name(name: str) -> dict | None:
|
||||||
|
normalized_name = name.strip().lower()
|
||||||
|
if not normalized_name:
|
||||||
|
return None
|
||||||
|
row = get_db().execute(
|
||||||
|
f"""
|
||||||
|
SELECT items.*,
|
||||||
|
owner.display_name AS owner_display_name,
|
||||||
|
owner.username AS owner_username,
|
||||||
|
target.display_name AS target_display_name,
|
||||||
|
target.username AS target_username,
|
||||||
|
EXISTS(
|
||||||
|
SELECT 1
|
||||||
|
FROM shopping_entries
|
||||||
|
WHERE shopping_entries.item_id = items.id AND shopping_entries.is_checked = 0
|
||||||
|
) AS is_on_shopping_list
|
||||||
|
FROM items
|
||||||
|
LEFT JOIN users AS owner ON owner.id = items.owner_user_id
|
||||||
|
LEFT JOIN users AS target ON target.id = items.target_user_id
|
||||||
|
WHERE items.kind = 'food'
|
||||||
|
AND items.is_archived = 0
|
||||||
|
AND LOWER(items.name) = ?
|
||||||
|
AND {visible_clause('items')}
|
||||||
|
ORDER BY LOWER(items.name), items.id
|
||||||
|
LIMIT 1
|
||||||
|
""",
|
||||||
|
[normalized_name, *visible_params()],
|
||||||
|
).fetchone()
|
||||||
|
if row is None:
|
||||||
|
return None
|
||||||
|
return attach_builder_keys(attach_dayparts(describe_records([row])))[0]
|
||||||
|
|
||||||
|
|
||||||
def fetch_shopping_entries():
|
def fetch_shopping_entries():
|
||||||
rows = get_db().execute(
|
rows = get_db().execute(
|
||||||
f"""
|
f"""
|
||||||
@@ -3710,18 +3751,17 @@ def backup_export():
|
|||||||
current_app.config["APP_VERSION"],
|
current_app.config["APP_VERSION"],
|
||||||
)
|
)
|
||||||
|
|
||||||
@after_this_request
|
archive_size = Path(archive_path).stat().st_size
|
||||||
def cleanup_backup(response):
|
response = send_file(
|
||||||
Path(archive_path).unlink(missing_ok=True)
|
|
||||||
return response
|
|
||||||
|
|
||||||
return send_file(
|
|
||||||
archive_path,
|
archive_path,
|
||||||
as_attachment=True,
|
as_attachment=True,
|
||||||
download_name=download_name,
|
download_name=download_name,
|
||||||
mimetype="application/zip",
|
mimetype="application/zip",
|
||||||
max_age=0,
|
max_age=0,
|
||||||
)
|
)
|
||||||
|
response.content_length = archive_size
|
||||||
|
response.call_on_close(lambda: Path(archive_path).unlink(missing_ok=True))
|
||||||
|
return response
|
||||||
|
|
||||||
|
|
||||||
@main_bp.post("/settings/backup/restore")
|
@main_bp.post("/settings/backup/restore")
|
||||||
@@ -4387,6 +4427,56 @@ def remove_shopping_entry(entry_id: int) -> None:
|
|||||||
get_db().commit()
|
get_db().commit()
|
||||||
|
|
||||||
|
|
||||||
|
@main_bp.post("/shopping/<int:entry_id>/note")
|
||||||
|
@login_required
|
||||||
|
def shopping_update_note(entry_id: int):
|
||||||
|
entry = get_db().execute(
|
||||||
|
f"""
|
||||||
|
SELECT shopping_entries.*,
|
||||||
|
owner.display_name AS owner_display_name,
|
||||||
|
owner.username AS owner_username
|
||||||
|
FROM shopping_entries
|
||||||
|
LEFT JOIN users AS owner ON owner.id = shopping_entries.owner_user_id
|
||||||
|
WHERE shopping_entries.id = ? AND {visible_clause('shopping_entries')}
|
||||||
|
""",
|
||||||
|
[entry_id, *visible_params()],
|
||||||
|
).fetchone()
|
||||||
|
if entry is None:
|
||||||
|
flash("Der Einkaufseintrag wurde nicht gefunden.", "error")
|
||||||
|
return redirect(url_with_scroll_position(url_for("main.shopping_list")))
|
||||||
|
|
||||||
|
try:
|
||||||
|
ensure_can_edit(describe_record(dict(entry)), "Diesen Einkaufseintrag kannst du gerade nicht ändern.")
|
||||||
|
except PermissionError as exc:
|
||||||
|
flash(str(exc), "error")
|
||||||
|
return redirect(url_with_scroll_position(url_for("main.shopping_list")))
|
||||||
|
|
||||||
|
shopping_note = normalize_shopping_note(request.form.get("shopping_note"))
|
||||||
|
duplicate = get_db().execute(
|
||||||
|
"""
|
||||||
|
SELECT id
|
||||||
|
FROM shopping_entries
|
||||||
|
WHERE item_id = ?
|
||||||
|
AND shopping_note = ?
|
||||||
|
AND is_checked = 0
|
||||||
|
AND id != ?
|
||||||
|
LIMIT 1
|
||||||
|
""",
|
||||||
|
(entry["item_id"], shopping_note, entry_id),
|
||||||
|
).fetchone()
|
||||||
|
if duplicate:
|
||||||
|
flash("Dieser Hinweis steht für das Lebensmittel schon auf der Einkaufsliste.", "info")
|
||||||
|
return redirect(url_with_scroll_position(url_for("main.shopping_list")))
|
||||||
|
|
||||||
|
get_db().execute(
|
||||||
|
"UPDATE shopping_entries SET shopping_note = ? WHERE id = ?",
|
||||||
|
(shopping_note, entry_id),
|
||||||
|
)
|
||||||
|
get_db().commit()
|
||||||
|
flash("Der Einkaufshinweis wurde gespeichert.", "success")
|
||||||
|
return redirect(url_with_scroll_position(url_for("main.shopping_list")))
|
||||||
|
|
||||||
|
|
||||||
@main_bp.post("/items/<int:item_id>/shopping/bought")
|
@main_bp.post("/items/<int:item_id>/shopping/bought")
|
||||||
@login_required
|
@login_required
|
||||||
def item_mark_bought(item_id: int):
|
def item_mark_bought(item_id: int):
|
||||||
@@ -4438,30 +4528,40 @@ def item_remove_from_shopping(item_id: int):
|
|||||||
def shopping_list():
|
def shopping_list():
|
||||||
if request.method == "POST":
|
if request.method == "POST":
|
||||||
selected_item_id = request.form.get("item_id", "").strip()
|
selected_item_id = request.form.get("item_id", "").strip()
|
||||||
if not selected_item_id.isdigit():
|
item_search = request.form.get("item_search", "").strip()
|
||||||
flash("Bitte zuerst etwas auswählen.", "error")
|
shopping_note = normalize_shopping_note(request.form.get("shopping_note"))
|
||||||
else:
|
item = None
|
||||||
|
if selected_item_id.isdigit():
|
||||||
try:
|
try:
|
||||||
item = get_item(int(selected_item_id))
|
item = get_item(int(selected_item_id))
|
||||||
|
except ValueError as exc:
|
||||||
|
flash(str(exc), "error")
|
||||||
|
elif item_search:
|
||||||
|
item = find_shopping_food_by_name(item_search)
|
||||||
|
if item is None:
|
||||||
|
flash("Bitte ein Lebensmittel aus der Suche auswählen.", "error")
|
||||||
|
else:
|
||||||
|
flash("Bitte zuerst etwas auswählen.", "error")
|
||||||
|
|
||||||
|
if item is not None:
|
||||||
result = ensure_item_or_missing_components_are_shopped(
|
result = ensure_item_or_missing_components_are_shopped(
|
||||||
item["id"],
|
item["id"],
|
||||||
g.user["id"],
|
g.user["id"],
|
||||||
item["visibility"],
|
item["visibility"],
|
||||||
|
shopping_note=shopping_note,
|
||||||
)
|
)
|
||||||
if result["count"]:
|
if result["count"]:
|
||||||
flash(f"Die Einkaufsliste wurde ergänzt: {', '.join(result['names'][:4])}.", "success")
|
note_suffix = f" ({shopping_note})" if shopping_note else ""
|
||||||
|
flash(f"Die Einkaufsliste wurde ergänzt: {', '.join(result['names'][:4])}{note_suffix}.", "success")
|
||||||
elif result["scheduled_count"]:
|
elif result["scheduled_count"]:
|
||||||
flash("Ein paar Dinge sind für einen späteren Einkauf vorgemerkt.", "info")
|
flash("Ein paar Dinge sind für einen späteren Einkauf vorgemerkt.", "info")
|
||||||
else:
|
else:
|
||||||
flash("Dafür ist gerade nichts zusätzlich nötig.", "info")
|
flash("Dieser Einkaufseintrag steht so schon auf der Liste.", "info")
|
||||||
except ValueError as exc:
|
|
||||||
flash(str(exc), "error")
|
|
||||||
return redirect(url_with_scroll_position(url_for("main.shopping_list")))
|
return redirect(url_with_scroll_position(url_for("main.shopping_list")))
|
||||||
|
|
||||||
entries = fetch_shopping_entries()
|
entries = fetch_shopping_entries()
|
||||||
upcoming_entries = fetch_upcoming_shopping_needs()
|
upcoming_entries = fetch_upcoming_shopping_needs()
|
||||||
addable_items = fetch_items(include_archived=True, include_quick_added=True)
|
addable_items = fetch_items(kind="food", include_archived=False, include_quick_added=True)
|
||||||
addable_items = [item for item in addable_items if item["kind"] == "food" and not item["is_on_shopping_list"]]
|
|
||||||
household_settings = get_household_settings()
|
household_settings = get_household_settings()
|
||||||
shopping_weekday_label = dict(WEEKDAY_OPTIONS).get(household_settings["shopping_weekday"], "gesetzt")
|
shopping_weekday_label = dict(WEEKDAY_OPTIONS).get(household_settings["shopping_weekday"], "gesetzt")
|
||||||
return render_template(
|
return render_template(
|
||||||
|
|||||||
+2
-1
@@ -167,6 +167,7 @@ CREATE TABLE IF NOT EXISTS shopping_entries (
|
|||||||
owner_user_id INTEGER,
|
owner_user_id INTEGER,
|
||||||
visibility TEXT NOT NULL DEFAULT 'shared',
|
visibility TEXT NOT NULL DEFAULT 'shared',
|
||||||
item_id INTEGER NOT NULL,
|
item_id INTEGER NOT NULL,
|
||||||
|
shopping_note TEXT NOT NULL DEFAULT '',
|
||||||
added_by INTEGER,
|
added_by INTEGER,
|
||||||
checked_by INTEGER,
|
checked_by INTEGER,
|
||||||
needed_for_date TEXT,
|
needed_for_date TEXT,
|
||||||
@@ -183,7 +184,7 @@ CREATE TABLE IF NOT EXISTS shopping_entries (
|
|||||||
);
|
);
|
||||||
|
|
||||||
CREATE UNIQUE INDEX IF NOT EXISTS idx_shopping_entries_open_item
|
CREATE UNIQUE INDEX IF NOT EXISTS idx_shopping_entries_open_item
|
||||||
ON shopping_entries (item_id)
|
ON shopping_entries (item_id, COALESCE(shopping_note, ''))
|
||||||
WHERE is_checked = 0;
|
WHERE is_checked = 0;
|
||||||
|
|
||||||
CREATE TABLE IF NOT EXISTS shopping_needs (
|
CREATE TABLE IF NOT EXISTS shopping_needs (
|
||||||
|
|||||||
+143
-6
@@ -1219,6 +1219,13 @@ h3 {
|
|||||||
grid-column: 1 / -1;
|
grid-column: 1 / -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.shopping-add-form {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: minmax(0, 1.5fr) minmax(11rem, 0.8fr) auto;
|
||||||
|
gap: 0.8rem;
|
||||||
|
align-items: end;
|
||||||
|
}
|
||||||
|
|
||||||
.shopping-add-grid {
|
.shopping-add-grid {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: repeat(auto-fit, minmax(190px, 1fr));
|
grid-template-columns: repeat(auto-fit, minmax(190px, 1fr));
|
||||||
@@ -1244,6 +1251,9 @@ h3 {
|
|||||||
padding: 0.85rem 0.95rem;
|
padding: 0.85rem 0.95rem;
|
||||||
border-radius: 22px;
|
border-radius: 22px;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
|
background: color-mix(in srgb, var(--surface-strong) 86%, var(--accent-soft) 14%);
|
||||||
|
color: var(--text);
|
||||||
|
border-color: color-mix(in srgb, var(--line) 72%, var(--accent) 28%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.shopping-add-card-visual,
|
.shopping-add-card-visual,
|
||||||
@@ -1342,6 +1352,20 @@ h3 {
|
|||||||
margin: 0;
|
margin: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.shopping-entry-note {
|
||||||
|
width: fit-content;
|
||||||
|
max-width: 100%;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0.12rem 0.5rem;
|
||||||
|
border-radius: 999px;
|
||||||
|
background: color-mix(in srgb, var(--accent-soft) 56%, transparent 44%);
|
||||||
|
color: color-mix(in srgb, var(--accent-strong) 70%, var(--text) 30%);
|
||||||
|
font-size: 0.82rem;
|
||||||
|
font-weight: 700;
|
||||||
|
line-height: 1.35;
|
||||||
|
overflow-wrap: anywhere;
|
||||||
|
}
|
||||||
|
|
||||||
.shopping-entry-actions {
|
.shopping-entry-actions {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@@ -1356,6 +1380,18 @@ h3 {
|
|||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.shopping-entry-check-button {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.shopping-entry-check-mark {
|
||||||
|
font-size: 1rem;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
|
||||||
.shopping-entry-close-form {
|
.shopping-entry-close-form {
|
||||||
flex: 0 0 auto;
|
flex: 0 0 auto;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
@@ -1414,25 +1450,81 @@ h3 {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@media (max-width: 680px) {
|
@media (max-width: 680px) {
|
||||||
|
.shopping-add-form {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
|
||||||
|
.shopping-add-form button {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
.shopping-entry-row {
|
.shopping-entry-row {
|
||||||
align-items: stretch;
|
display: grid;
|
||||||
flex-direction: column;
|
grid-template-columns: minmax(0, 1fr) auto auto;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.7rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.shopping-entry-open {
|
.shopping-entry-open {
|
||||||
width: 100%;
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.shopping-entry-main {
|
||||||
|
grid-template-columns: 56px minmax(0, 1fr);
|
||||||
|
gap: 0.8rem;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.shopping-entry-visual,
|
||||||
|
.shopping-entry-fallback {
|
||||||
|
width: 56px;
|
||||||
|
height: 56px;
|
||||||
|
border-radius: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.shopping-entry-copy {
|
||||||
|
gap: 0.12rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.shopping-entry-copy strong {
|
||||||
|
font-size: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.shopping-entry-copy .muted {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.shopping-entry-note {
|
||||||
|
font-size: 0.78rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.shopping-entry-actions,
|
.shopping-entry-actions,
|
||||||
.shopping-entry-actions form,
|
.shopping-entry-actions form,
|
||||||
.shopping-entry-actions button,
|
.shopping-entry-actions button,
|
||||||
.shopping-entry-close-form {
|
.shopping-entry-close-form {
|
||||||
width: 100%;
|
width: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.shopping-entry-check-button {
|
||||||
|
min-width: 0;
|
||||||
|
padding: 0.75rem 0.9rem;
|
||||||
|
border-radius: 16px;
|
||||||
|
gap: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.shopping-entry-check-label {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.shopping-entry-check-mark {
|
||||||
|
font-size: 1.05rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.shopping-entry-close {
|
.shopping-entry-close {
|
||||||
width: 100%;
|
width: 2.75rem;
|
||||||
border-radius: 18px;
|
height: 2.75rem;
|
||||||
|
min-width: 2.75rem;
|
||||||
|
border-radius: 16px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1448,6 +1540,31 @@ h3 {
|
|||||||
border-color: color-mix(in srgb, var(--line) 58%, rgba(243, 177, 125, 0.18) 42%);
|
border-color: color-mix(in srgb, var(--line) 58%, rgba(243, 177, 125, 0.18) 42%);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[data-theme="dark"] button.shopping-add-card,
|
||||||
|
[data-theme="dark"] .shopping-add-card {
|
||||||
|
background: linear-gradient(
|
||||||
|
180deg,
|
||||||
|
color-mix(in srgb, var(--surface-soft) 70%, #4a3e3a 30%),
|
||||||
|
color-mix(in srgb, var(--surface) 92%, #241f1d 8%)
|
||||||
|
);
|
||||||
|
color: var(--text);
|
||||||
|
border-color: color-mix(in srgb, var(--line) 58%, rgba(243, 177, 125, 0.16) 42%);
|
||||||
|
box-shadow: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
[data-theme="dark"] button.shopping-add-card:hover,
|
||||||
|
[data-theme="dark"] .shopping-add-card:hover {
|
||||||
|
background: linear-gradient(
|
||||||
|
180deg,
|
||||||
|
color-mix(in srgb, var(--surface-soft) 62%, #5a4840 38%),
|
||||||
|
color-mix(in srgb, var(--surface) 88%, #2f2724 12%)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
[data-theme="dark"] .shopping-add-card-copy small {
|
||||||
|
color: color-mix(in srgb, var(--muted) 86%, white 14%);
|
||||||
|
}
|
||||||
|
|
||||||
[data-theme="dark"] .shopping-entry-card {
|
[data-theme="dark"] .shopping-entry-card {
|
||||||
background: linear-gradient(
|
background: linear-gradient(
|
||||||
180deg,
|
180deg,
|
||||||
@@ -1457,6 +1574,11 @@ h3 {
|
|||||||
border-color: color-mix(in srgb, var(--line) 58%, rgba(243, 177, 125, 0.14) 42%);
|
border-color: color-mix(in srgb, var(--line) 58%, rgba(243, 177, 125, 0.14) 42%);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[data-theme="dark"] .shopping-entry-note {
|
||||||
|
background: color-mix(in srgb, var(--accent-soft) 54%, rgba(32, 27, 25, 0.46) 46%);
|
||||||
|
color: color-mix(in srgb, var(--accent-strong) 82%, white 18%);
|
||||||
|
}
|
||||||
|
|
||||||
.auth-shell {
|
.auth-shell {
|
||||||
min-height: calc(100vh - 10rem);
|
min-height: calc(100vh - 10rem);
|
||||||
display: grid;
|
display: grid;
|
||||||
@@ -1783,6 +1905,7 @@ legend {
|
|||||||
border-radius: 18px;
|
border-radius: 18px;
|
||||||
border: 1px solid var(--line);
|
border: 1px solid var(--line);
|
||||||
background: color-mix(in srgb, var(--surface-strong) 82%, #fff 18%);
|
background: color-mix(in srgb, var(--surface-strong) 82%, #fff 18%);
|
||||||
|
color: var(--text);
|
||||||
}
|
}
|
||||||
|
|
||||||
.quick-select-card strong,
|
.quick-select-card strong,
|
||||||
@@ -1800,6 +1923,20 @@ legend {
|
|||||||
color: var(--muted);
|
color: var(--muted);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[data-theme="dark"] .quick-select-card {
|
||||||
|
background: linear-gradient(
|
||||||
|
180deg,
|
||||||
|
color-mix(in srgb, var(--surface-soft) 70%, #4a3e3a 30%),
|
||||||
|
color-mix(in srgb, var(--surface) 92%, #241f1d 8%)
|
||||||
|
);
|
||||||
|
border-color: color-mix(in srgb, var(--line) 58%, rgba(243, 177, 125, 0.16) 42%);
|
||||||
|
color: var(--text);
|
||||||
|
}
|
||||||
|
|
||||||
|
[data-theme="dark"] .quick-select-card small {
|
||||||
|
color: color-mix(in srgb, var(--muted) 86%, white 14%);
|
||||||
|
}
|
||||||
|
|
||||||
.inline-photo img {
|
.inline-photo img {
|
||||||
width: min(220px, 100%);
|
width: min(220px, 100%);
|
||||||
border-radius: 18px;
|
border-radius: 18px;
|
||||||
|
|||||||
@@ -10,68 +10,34 @@
|
|||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section class="panel compact-form-panel">
|
<section class="panel compact-form-panel">
|
||||||
<div class="stack-sections">
|
<form method="post" class="shopping-add-form">
|
||||||
|
{{ csrf_input() }}
|
||||||
<label>
|
<label>
|
||||||
Lebensmittel suchen
|
Lebensmittel suchen
|
||||||
<input
|
<input
|
||||||
type="text"
|
type="text"
|
||||||
|
name="item_search"
|
||||||
|
list="shopping-food-options"
|
||||||
placeholder="Nach Lebensmitteln suchen"
|
placeholder="Nach Lebensmitteln suchen"
|
||||||
data-filter-input
|
autocomplete="off"
|
||||||
data-filter-target="#shopping-add-list"
|
|
||||||
data-filter-limit="8"
|
|
||||||
>
|
>
|
||||||
</label>
|
</label>
|
||||||
<div class="shopping-add-grid" id="shopping-add-list">
|
<label>
|
||||||
|
Einkaufshinweis
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
name="shopping_note"
|
||||||
|
maxlength="80"
|
||||||
|
placeholder="z. B. TK, Dose, frisch"
|
||||||
|
>
|
||||||
|
</label>
|
||||||
|
<datalist id="shopping-food-options">
|
||||||
{% for item in addable_items %}
|
{% for item in addable_items %}
|
||||||
{% set item_icon_class = {
|
<option value="{{ item.name }}"></option>
|
||||||
'protein': 'icon-component-protein',
|
|
||||||
'carb': 'icon-component-carb',
|
|
||||||
'veg': 'icon-component-veg',
|
|
||||||
'fruit': 'icon-component-fruit',
|
|
||||||
'dairy': 'icon-component-dairy',
|
|
||||||
'nuts': 'icon-component-nuts',
|
|
||||||
'seeds': 'icon-component-seeds',
|
|
||||||
'neutral': 'icon-component-neutral',
|
|
||||||
}.get(item.primary_builder_key or item.base_type, 'icon-component-neutral') %}
|
|
||||||
<form method="post" data-filter-label="{{ item.name|lower }} {{ item.base_type_label|lower }} {{ item.for_label|lower }}">
|
|
||||||
{{ csrf_input() }}
|
|
||||||
<input type="hidden" name="item_id" value="{{ item.id }}">
|
|
||||||
<button class="shopping-add-card" type="submit">
|
|
||||||
<span class="shopping-add-card-visual">
|
|
||||||
{% if item.photo_filename %}
|
|
||||||
<img
|
|
||||||
src="{{ image_url(item.photo_filename, 'md') }}"
|
|
||||||
srcset="{{ image_srcset(item.photo_filename) }}"
|
|
||||||
sizes="{{ image_sizes('grid') }}"
|
|
||||||
alt=""
|
|
||||||
loading="lazy">
|
|
||||||
{% else %}
|
|
||||||
<span class="shopping-add-card-fallback">
|
|
||||||
<span class="ui-icon {{ item_icon_class }}"></span>
|
|
||||||
</span>
|
|
||||||
{% endif %}
|
|
||||||
</span>
|
|
||||||
<span class="shopping-add-card-copy">
|
|
||||||
<strong>{{ item.name }}</strong>
|
|
||||||
<small>
|
|
||||||
{% if item.is_archived %}
|
|
||||||
Archiviert
|
|
||||||
{% elif item.is_quick_added %}
|
|
||||||
Unsortiert
|
|
||||||
{% elif item.is_home %}
|
|
||||||
Zuhause · trotzdem ergänzen
|
|
||||||
{% else %}
|
|
||||||
Gerade nicht da
|
|
||||||
{% endif %}
|
|
||||||
</small>
|
|
||||||
</span>
|
|
||||||
</button>
|
|
||||||
</form>
|
|
||||||
{% else %}
|
|
||||||
<p class="shopping-add-empty muted">Gerade ist nichts zusätzlich offen.</p>
|
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</div>
|
</datalist>
|
||||||
</div>
|
<button type="submit">Auf die Liste</button>
|
||||||
|
</form>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
{% if entries %}
|
{% if entries %}
|
||||||
@@ -119,6 +85,9 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="shopping-entry-copy">
|
<div class="shopping-entry-copy">
|
||||||
<strong>{{ entry.item_name }}</strong>
|
<strong>{{ entry.item_name }}</strong>
|
||||||
|
{% if entry.shopping_note %}
|
||||||
|
<p class="shopping-entry-note">{{ entry.shopping_note }}</p>
|
||||||
|
{% endif %}
|
||||||
{% if entry.needed_for_label %}
|
{% if entry.needed_for_label %}
|
||||||
<p class="muted">
|
<p class="muted">
|
||||||
Für {{ entry.needed_for_label }}
|
Für {{ entry.needed_for_label }}
|
||||||
@@ -131,7 +100,10 @@
|
|||||||
<div class="shopping-entry-actions">
|
<div class="shopping-entry-actions">
|
||||||
<form method="post" action="{{ url_for('main.shopping_check', entry_id=entry.id) }}">
|
<form method="post" action="{{ url_for('main.shopping_check', entry_id=entry.id) }}">
|
||||||
{{ csrf_input() }}
|
{{ csrf_input() }}
|
||||||
<button type="submit">Eingekauft</button>
|
<button type="submit" class="shopping-entry-check-button">
|
||||||
|
<span class="shopping-entry-check-mark" aria-hidden="true">✔</span>
|
||||||
|
<span class="shopping-entry-check-label">Eingekauft</span>
|
||||||
|
</button>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
{% if entry.can_edit %}
|
{% if entry.can_edit %}
|
||||||
@@ -150,7 +122,9 @@
|
|||||||
<div>
|
<div>
|
||||||
<h3>{{ entry.item_name }}</h3>
|
<h3>{{ entry.item_name }}</h3>
|
||||||
<p>
|
<p>
|
||||||
{% if entry.needed_for_label %}
|
{% if entry.shopping_note %}
|
||||||
|
{{ entry.shopping_note }}
|
||||||
|
{% elif entry.needed_for_label %}
|
||||||
Für {{ entry.needed_for_label }}
|
Für {{ entry.needed_for_label }}
|
||||||
{% if entry.needed_daypart_name %} · {{ entry.needed_daypart_name }}{% endif %}
|
{% if entry.needed_daypart_name %} · {{ entry.needed_daypart_name }}{% endif %}
|
||||||
{% elif entry.is_home %}
|
{% elif entry.is_home %}
|
||||||
@@ -171,6 +145,20 @@
|
|||||||
<button type="submit">Eingekauft</button>
|
<button type="submit">Eingekauft</button>
|
||||||
</form>
|
</form>
|
||||||
{% if entry.can_edit %}
|
{% if entry.can_edit %}
|
||||||
|
<form method="post" action="{{ url_for('main.shopping_update_note', entry_id=entry.id) }}">
|
||||||
|
{{ csrf_input() }}
|
||||||
|
<label>
|
||||||
|
Einkaufshinweis
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
name="shopping_note"
|
||||||
|
maxlength="80"
|
||||||
|
value="{{ entry.shopping_note }}"
|
||||||
|
placeholder="z. B. TK, Dose, frisch"
|
||||||
|
>
|
||||||
|
</label>
|
||||||
|
<button class="ghost-button" type="submit">Hinweis speichern</button>
|
||||||
|
</form>
|
||||||
<form method="post" action="{{ url_for('main.shopping_remove', entry_id=entry.id) }}">
|
<form method="post" action="{{ url_for('main.shopping_remove', entry_id=entry.id) }}">
|
||||||
{{ csrf_input() }}
|
{{ csrf_input() }}
|
||||||
<button class="ghost-button" type="submit">Von Einkaufsliste nehmen</button>
|
<button class="ghost-button" type="submit">Von Einkaufsliste nehmen</button>
|
||||||
|
|||||||
Reference in New Issue
Block a user