mirror of
https://github.com/mealie-recipes/mealie.git
synced 2026-06-13 20:40:18 -04:00
Compare commits
215 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
a41ad8c6ed | ||
|
|
9c38c89c44 | ||
|
|
998440d064 | ||
|
|
b01d12c377 | ||
|
|
1dee574a08 | ||
|
|
257c4461a3 | ||
|
|
01f4257190 | ||
|
|
d7b7dd6c83 | ||
|
|
23c2eab682 | ||
|
|
def346d16d | ||
|
|
cc324b29ae | ||
|
|
9d58f9b266 | ||
|
|
30b2776f3c | ||
|
|
60d23d0686 | ||
|
|
edf649dea6 | ||
|
|
29b4a3cd22 | ||
|
|
f3a5148628 | ||
|
|
33abd777e0 | ||
|
|
739055caf6 | ||
|
|
8c29bd3439 | ||
|
|
2c4d0b692b | ||
|
|
946b79b77a | ||
|
|
cd154d09b2 | ||
|
|
236c930b54 | ||
|
|
980c847e36 | ||
|
|
91700771e6 | ||
|
|
abb6ad5fd0 | ||
|
|
ac7af02f77 | ||
|
|
525b398687 | ||
|
|
fafc836ccc | ||
|
|
c617b829e5 | ||
|
|
5b1e827d45 | ||
|
|
e33b62be2a | ||
|
|
60c33b499c | ||
|
|
c205dff523 | ||
|
|
ce69899c4b | ||
|
|
a4183e3453 | ||
|
|
ab39408a24 | ||
|
|
8c6c98483c | ||
|
|
ae095ab572 | ||
|
|
65356bc21a | ||
|
|
3aed5de3fc | ||
|
|
a4e9e54dae | ||
|
|
8f698e437e | ||
|
|
ab0d36825a | ||
|
|
4e2f6c57f1 | ||
|
|
de4cb8ba83 | ||
|
|
375f43c596 | ||
|
|
3034945e7e | ||
|
|
8e5effa532 | ||
|
|
3b81d3b18a | ||
|
|
d0f8b5773d | ||
|
|
14910162dc | ||
|
|
035f780d27 | ||
|
|
f10161ee92 | ||
|
|
b1a100a8c5 | ||
|
|
7db39d32d1 | ||
|
|
10921f9a64 | ||
|
|
ba1c44172e | ||
|
|
fd2dc15a15 | ||
|
|
47124488bb | ||
|
|
6e680c972a | ||
|
|
1fd2eb37ae | ||
|
|
923a59791a | ||
|
|
1fcc2c755a | ||
|
|
d5f7a883df | ||
|
|
17f9eef551 | ||
|
|
ca1ab33291 | ||
|
|
6e6ae80c46 | ||
|
|
aa6e109162 | ||
|
|
a6e4b778c1 | ||
|
|
31c7cb7906 | ||
|
|
d954b5cf48 | ||
|
|
e5c2f5570f | ||
|
|
e344f3f1e6 | ||
|
|
6a7f0edbc6 | ||
|
|
d2fd4e0843 | ||
|
|
d639d168fa | ||
|
|
7931e383b2 | ||
|
|
7bfb8c78a6 | ||
|
|
9a2b2a59a2 | ||
|
|
0c0ff8f19d | ||
|
|
6bbd9a162e | ||
|
|
4d6fc9a4c9 | ||
|
|
53c89b340a | ||
|
|
dc64484b8e | ||
|
|
aabab73310 | ||
|
|
542d0e5218 | ||
|
|
a3a21eb533 | ||
|
|
b0cc7c4c25 | ||
|
|
e80ba7dff3 | ||
|
|
bdac51bae2 | ||
|
|
f4827abc1d | ||
|
|
63a180ef2c | ||
|
|
a062a4beaa | ||
|
|
4831adb0f3 | ||
|
|
97550899d0 | ||
|
|
da11204cd7 | ||
|
|
9795b4c553 | ||
|
|
3e1adfa65d | ||
|
|
88214cd61f | ||
|
|
f1b53483da | ||
|
|
5ccac83d08 | ||
|
|
5594b303bc | ||
|
|
58100d0515 | ||
|
|
e2033b2d67 | ||
|
|
5e0f8a4bf7 | ||
|
|
477899fce3 | ||
|
|
04d481fcbf | ||
|
|
5572e51933 | ||
|
|
44915ace12 | ||
|
|
906a143363 | ||
|
|
c41bfbab1e | ||
|
|
69fca013d8 | ||
|
|
24a17e4001 | ||
|
|
6cbd004fb9 | ||
|
|
5d8210d570 | ||
|
|
8d6dc1c6ee | ||
|
|
0e520ba43c | ||
|
|
a9b40cd862 | ||
|
|
ba48e9414c | ||
|
|
72aa1b2514 | ||
|
|
b4f07f9d62 | ||
|
|
eb36912e5c | ||
|
|
d923b4c7fa | ||
|
|
4ecf88379c | ||
|
|
857c8d42e2 | ||
|
|
982802c427 | ||
|
|
4d1381c055 | ||
|
|
adab596683 | ||
|
|
e08fc4e25e | ||
|
|
e8ee37fd43 | ||
|
|
bfe249dc42 | ||
|
|
15989e0c93 | ||
|
|
9af06ce442 | ||
|
|
22fc29742a | ||
|
|
20b1b3de35 | ||
|
|
82d930e645 | ||
|
|
d96c36333b | ||
|
|
b220cd6431 | ||
|
|
598b0f3707 | ||
|
|
c18b9d3184 | ||
|
|
e64d070603 | ||
|
|
d843370c07 | ||
|
|
269be953ce | ||
|
|
9f7d74aecf | ||
|
|
4a0a8e8a5e | ||
|
|
75895cab79 | ||
|
|
be0cdee8b7 | ||
|
|
6024c8bc05 | ||
|
|
b3241d3e8b | ||
|
|
01b9987812 | ||
|
|
4e613e15f0 | ||
|
|
5298bdc90f | ||
|
|
2a6bb7d444 | ||
|
|
21f1d46b6d | ||
|
|
df15a9e74e | ||
|
|
f53cae7c7b | ||
|
|
219138fce1 | ||
|
|
4634ad5666 | ||
|
|
eab7c0d9e5 | ||
|
|
4afb767375 | ||
|
|
a49c32e663 | ||
|
|
5d55e4b4ff | ||
|
|
98c5d142eb | ||
|
|
c0db6ff3d1 | ||
|
|
b71310cdaf | ||
|
|
91fb750768 | ||
|
|
7f1139618d | ||
|
|
42b2bc7c15 | ||
|
|
583bd742fb | ||
|
|
af86ed2028 | ||
|
|
80dc1dcaad | ||
|
|
90e184c0fc | ||
|
|
393103662f | ||
|
|
6e2957fb1e | ||
|
|
e7a668e64e | ||
|
|
88577b696b | ||
|
|
289038ba17 | ||
|
|
4aec294c26 | ||
|
|
99a13bd0c4 | ||
|
|
3bd4353685 | ||
|
|
109ec651cc | ||
|
|
166582acf4 | ||
|
|
4bc88e653f | ||
|
|
94e91d3602 | ||
|
|
d6ce607a4e | ||
|
|
9e8822fabe | ||
|
|
0734ec9ce8 | ||
|
|
f4f2b863e0 | ||
|
|
a7fcb6c84d | ||
|
|
abcee51d0c | ||
|
|
8b61f95c8c | ||
|
|
e01bb60aab | ||
|
|
bf5340b902 | ||
|
|
985041e61f | ||
|
|
70edf36073 | ||
|
|
15a0d25caa | ||
|
|
445754c5d8 | ||
|
|
831cd9c543 | ||
|
|
abaf6062c6 | ||
|
|
125c3914b3 | ||
|
|
93cb6bf341 | ||
|
|
46fb2b2c5a | ||
|
|
ac2a77b3b0 | ||
|
|
66a6426d15 | ||
|
|
508ae30133 | ||
|
|
e3c642debf | ||
|
|
af9e0f27a3 | ||
|
|
e07467df57 | ||
|
|
55b91bf847 | ||
|
|
58fc46af9f | ||
|
|
8e4fe55df1 | ||
|
|
f150c3f41e | ||
|
|
36c7ae5b4b |
@@ -38,3 +38,6 @@ RUN apt-get update \
|
|||||||
libwebp-dev \
|
libwebp-dev \
|
||||||
libsasl2-dev libldap2-dev libssl-dev \
|
libsasl2-dev libldap2-dev libssl-dev \
|
||||||
gnupg gnupg2 gnupg1
|
gnupg gnupg2 gnupg1
|
||||||
|
|
||||||
|
# create directory used for Docker Secrets
|
||||||
|
RUN mkdir -p /run/secrets
|
||||||
|
|||||||
@@ -27,6 +27,7 @@
|
|||||||
"python.formatting.yapfPath": "/usr/local/py-utils/bin/yapf",
|
"python.formatting.yapfPath": "/usr/local/py-utils/bin/yapf",
|
||||||
},
|
},
|
||||||
"extensions": [
|
"extensions": [
|
||||||
|
"charliermarsh.ruff",
|
||||||
"dbaeumer.vscode-eslint",
|
"dbaeumer.vscode-eslint",
|
||||||
"matangover.mypy",
|
"matangover.mypy",
|
||||||
"ms-python.black-formatter",
|
"ms-python.black-formatter",
|
||||||
|
|||||||
12
.github/pull_request_template.md
vendored
12
.github/pull_request_template.md
vendored
@@ -1,6 +1,12 @@
|
|||||||
<!--
|
<!--
|
||||||
This template provides some ideas of things to include in your PR description.
|
This template provides some ideas of things to include in your PR description.
|
||||||
To start, try providing a short summary of your changes in the Title above.
|
|
||||||
|
To start, try providing a short summary of your changes in the Title above. We follow Conventional Commits syntax, please ensure your title is prefixed with one of:
|
||||||
|
- `feat: `
|
||||||
|
- `fix: `
|
||||||
|
- `docs: `
|
||||||
|
- `chore: `
|
||||||
|
|
||||||
If a section of the PR template does not apply to this PR, then delete that section.
|
If a section of the PR template does not apply to this PR, then delete that section.
|
||||||
|
|
||||||
PLEASE READ:
|
PLEASE READ:
|
||||||
@@ -36,6 +42,8 @@ _(REQUIRED)_
|
|||||||
Briefly explain any decisions you made with respect to the changes.
|
Briefly explain any decisions you made with respect to the changes.
|
||||||
Include anything here that you didn't include in *Release Notes*
|
Include anything here that you didn't include in *Release Notes*
|
||||||
above, such as changes to CI or changes to internal methods.
|
above, such as changes to CI or changes to internal methods.
|
||||||
|
|
||||||
|
If there is a UI component to the change, please include before/after images.
|
||||||
-->
|
-->
|
||||||
|
|
||||||
## Which issue(s) this PR fixes:
|
## Which issue(s) this PR fixes:
|
||||||
@@ -44,7 +52,7 @@ _(REQUIRED)_
|
|||||||
|
|
||||||
<!--
|
<!--
|
||||||
If this PR fixes one of more issues, list them here.
|
If this PR fixes one of more issues, list them here.
|
||||||
One line each, like so:
|
One per line, like so:
|
||||||
Fixes #123
|
Fixes #123
|
||||||
Fixes #39
|
Fixes #39
|
||||||
-->
|
-->
|
||||||
|
|||||||
6
.github/workflows/codeql.yml
vendored
6
.github/workflows/codeql.yml
vendored
@@ -49,7 +49,7 @@ jobs:
|
|||||||
|
|
||||||
# Initializes the CodeQL tools for scanning.
|
# Initializes the CodeQL tools for scanning.
|
||||||
- name: Initialize CodeQL
|
- name: Initialize CodeQL
|
||||||
uses: github/codeql-action/init@v2
|
uses: github/codeql-action/init@v3
|
||||||
with:
|
with:
|
||||||
languages: ${{ matrix.language }}
|
languages: ${{ matrix.language }}
|
||||||
# If you wish to specify custom queries, you can do so here or in a config file.
|
# If you wish to specify custom queries, you can do so here or in a config file.
|
||||||
@@ -63,7 +63,7 @@ jobs:
|
|||||||
# Autobuild attempts to build any compiled languages (C/C++, C#, Go, Java, or Swift).
|
# Autobuild attempts to build any compiled languages (C/C++, C#, Go, Java, or Swift).
|
||||||
# If this step fails, then you should remove it and run the build manually (see below)
|
# If this step fails, then you should remove it and run the build manually (see below)
|
||||||
- name: Autobuild
|
- name: Autobuild
|
||||||
uses: github/codeql-action/autobuild@v2
|
uses: github/codeql-action/autobuild@v3
|
||||||
|
|
||||||
# ℹ️ Command-line programs to run using the OS shell.
|
# ℹ️ Command-line programs to run using the OS shell.
|
||||||
# 📚 See https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions#jobsjob_idstepsrun
|
# 📚 See https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions#jobsjob_idstepsrun
|
||||||
@@ -76,6 +76,6 @@ jobs:
|
|||||||
# ./location_of_script_within_repo/buildscript.sh
|
# ./location_of_script_within_repo/buildscript.sh
|
||||||
|
|
||||||
- name: Perform CodeQL Analysis
|
- name: Perform CodeQL Analysis
|
||||||
uses: github/codeql-action/analyze@v2
|
uses: github/codeql-action/analyze@v3
|
||||||
with:
|
with:
|
||||||
category: "/language:${{matrix.language}}"
|
category: "/language:${{matrix.language}}"
|
||||||
|
|||||||
41
.github/workflows/pull-request-lint.yml
vendored
Normal file
41
.github/workflows/pull-request-lint.yml
vendored
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
name: Pull Request Linter
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_call:
|
||||||
|
pull_request:
|
||||||
|
types: [edited] # This captures the PR title changing
|
||||||
|
branches:
|
||||||
|
- mealie-next
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
validate-title:
|
||||||
|
name: Validate PR title
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
# https://github.com/amannn/action-semantic-pull-request
|
||||||
|
- uses: amannn/action-semantic-pull-request@v5
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
with:
|
||||||
|
# Configure which types are allowed (newline-delimited).
|
||||||
|
# Default: https://github.com/commitizen/conventional-commit-types
|
||||||
|
types: |
|
||||||
|
fix
|
||||||
|
feat
|
||||||
|
docs
|
||||||
|
chore
|
||||||
|
# Configure which scopes are allowed (newline-delimited).
|
||||||
|
# These are regex patterns auto-wrapped in `^ $`.
|
||||||
|
scopes: |
|
||||||
|
deps
|
||||||
|
auto
|
||||||
|
l10n
|
||||||
|
# Configure that a scope must always be provided.
|
||||||
|
requireScope: false
|
||||||
|
# If the PR contains one of these newline-delimited labels, the
|
||||||
|
# validation is skipped. If you want to rerun the validation when
|
||||||
|
# labels change, you might want to use the `labeled` and `unlabeled`
|
||||||
|
# event triggers in your workflow.
|
||||||
|
ignoreLabels: |
|
||||||
|
bot
|
||||||
|
ignore-semantic-pull-request
|
||||||
4
.github/workflows/pull-requests.yml
vendored
4
.github/workflows/pull-requests.yml
vendored
@@ -6,6 +6,10 @@ on:
|
|||||||
- mealie-next
|
- mealie-next
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
pull-request-lint:
|
||||||
|
name: "Lint PR"
|
||||||
|
uses: ./.github/workflows/pull-request-lint.yml
|
||||||
|
|
||||||
backend-tests:
|
backend-tests:
|
||||||
name: "Backend Server Tests"
|
name: "Backend Server Tests"
|
||||||
uses: ./.github/workflows/partial-backend.yml
|
uses: ./.github/workflows/partial-backend.yml
|
||||||
|
|||||||
9
.github/workflows/release.yml
vendored
9
.github/workflows/release.yml
vendored
@@ -61,10 +61,15 @@ jobs:
|
|||||||
- name: Checkout 🛎
|
- name: Checkout 🛎
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Extract Version From Tag Name
|
||||||
|
run: echo "VERSION_NUM=$(echo ${{ github.event.release.tag_name }} | sed 's/^v//')" >> $GITHUB_ENV
|
||||||
|
|
||||||
- name: Modify version strings
|
- name: Modify version strings
|
||||||
run: |
|
run: |
|
||||||
sed -i 's/:v[0-9]*.[0-9]*.[0-9]*/:${{ github.event.release.tag_name }}/' docs/docs/documentation/getting-started/installation/sqlite.md
|
sed -i 's/:v[0-9]*.[0-9]*.[0-9]*/:v${{ env.VERSION_NUM }}/' docs/docs/documentation/getting-started/installation/sqlite.md
|
||||||
sed -i 's/:v[0-9]*.[0-9]*.[0-9]*/:${{ github.event.release.tag_name }}/' docs/docs/documentation/getting-started/installation/postgres.md
|
sed -i 's/:v[0-9]*.[0-9]*.[0-9]*/:v${{ env.VERSION_NUM }}/' docs/docs/documentation/getting-started/installation/postgres.md
|
||||||
|
sed -i 's/^version = "[^"]*"/version = "${{ env.VERSION_NUM }}"/' pyproject.toml
|
||||||
|
sed -i 's/^\s*"version": "[^"]*"/"version": "${{ env.VERSION_NUM }}"/' frontend/package.json
|
||||||
|
|
||||||
- name: Create Pull Request
|
- name: Create Pull Request
|
||||||
uses: peter-evans/create-pull-request@v6
|
uses: peter-evans/create-pull-request@v6
|
||||||
|
|||||||
32
.github/workflows/scheduled-checks.yml
vendored
Normal file
32
.github/workflows/scheduled-checks.yml
vendored
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
name: Scheduled Checks
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
# Every monday at 7 AM
|
||||||
|
- cron: 0 7 * * 1
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
update:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
pull-requests: write
|
||||||
|
steps:
|
||||||
|
- name: Checkout 🛎
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Update pre-commit Hooks
|
||||||
|
uses: vrslev/pre-commit-autoupdate@v1.0.0
|
||||||
|
|
||||||
|
- name: Create Pull Request
|
||||||
|
uses: peter-evans/create-pull-request@v6
|
||||||
|
# This doesn't currently work for us because it creates the PR but the workflows don't run.
|
||||||
|
# TODO: Provide a personal access token as a parameter here, that solves that problem.
|
||||||
|
# https://github.com/peter-evans/create-pull-request
|
||||||
|
with:
|
||||||
|
commit-message: "Update pre-commit hooks"
|
||||||
|
branch: "fix/update-pre-commit-hooks"
|
||||||
|
delete-branch: true
|
||||||
|
base: mealie-next
|
||||||
|
title: "fix(auto): Update pre-commit hooks"
|
||||||
|
body: "Auto-generated by `.github/workflows/scheduled-checks.yml`"
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
repos:
|
repos:
|
||||||
- repo: https://github.com/pre-commit/pre-commit-hooks
|
- repo: https://github.com/pre-commit/pre-commit-hooks
|
||||||
rev: v2.3.0
|
rev: v4.6.0
|
||||||
hooks:
|
hooks:
|
||||||
- id: check-yaml
|
- id: check-yaml
|
||||||
exclude: "mkdocs.yml"
|
exclude: "mkdocs.yml"
|
||||||
@@ -12,6 +12,6 @@ repos:
|
|||||||
exclude: ^tests/data/
|
exclude: ^tests/data/
|
||||||
- repo: https://github.com/astral-sh/ruff-pre-commit
|
- repo: https://github.com/astral-sh/ruff-pre-commit
|
||||||
# Ruff version.
|
# Ruff version.
|
||||||
rev: v0.4.3
|
rev: v0.5.5
|
||||||
hooks:
|
hooks:
|
||||||
- id: ruff-format
|
- id: ruff-format
|
||||||
|
|||||||
31
README.md
31
README.md
@@ -1,10 +1,10 @@
|
|||||||
[![Latest Release][latest-release-shield]][latest-release-url]
|
[![Latest Release][latest-release-shield]][latest-release-url]
|
||||||
[![Contributors][contributors-shield]][contributors-url]
|
[![Contributors][contributors-shield]][contributors-url]
|
||||||
[![Forks][forks-shield]][forks-url]
|
|
||||||
[![Stargazers][stars-shield]][stars-url]
|
[![Stargazers][stars-shield]][stars-url]
|
||||||
[![Issues][issues-shield]][issues-url]
|
[![Issues][issues-shield]][issues-url]
|
||||||
[![AGPL License][license-shield]][license-url]
|
[![AGPL License][license-shield]][license-url]
|
||||||
[![Docker Pulls][docker-pull]][docker-pull]
|
[![Docker Pulls][docker-pull]][docker-url]
|
||||||
|
[![GHCR Pulls][ghcr-pulls]][ghcr-url]
|
||||||
|
|
||||||
<!-- PROJECT LOGO -->
|
<!-- PROJECT LOGO -->
|
||||||
<br />
|
<br />
|
||||||
@@ -18,9 +18,9 @@
|
|||||||
<h3 align="center">Mealie</h3>
|
<h3 align="center">Mealie</h3>
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
A Place for All Your Recipes
|
A Place For All Your Recipes
|
||||||
<br />
|
<br />
|
||||||
<a href="https://nightly.mealie.io"><strong>Explore the docs »</strong></a>
|
<a href="https://docs.mealie.io/"><strong>Explore the docs »</strong></a>
|
||||||
<a href="https://github.com/mealie-recipes/mealie">
|
<a href="https://github.com/mealie-recipes/mealie">
|
||||||
</a>
|
</a>
|
||||||
<br />
|
<br />
|
||||||
@@ -38,12 +38,20 @@
|
|||||||
|
|
||||||
# About The Project
|
# About The Project
|
||||||
|
|
||||||
Mealie is a self hosted recipe manager and meal planner with a RestAPI backend and a reactive frontend application built in Vue for a pleasant user experience for the whole family. Easily add recipes into your database by providing the URL and Mealie will automatically import the relevant data, or add a family recipe with the UI editor. Mealie also provides an API for interactions from 3rd party applications.
|
Mealie is a self hosted recipe manager, meal planner and shopping list with a RestAPI backend and a reactive frontend built in Vue for a pleasant user experience for the whole family. Easily add recipes into your database by providing the URL and Mealie will automatically import the relevant data, or add a family recipe with the UI editor. Mealie also provides an API for interactions from 3rd party applications.
|
||||||
|
|
||||||
- [Remember to join the Discord](https://discord.gg/QuStdQGSGK)!
|
- [Remember to join the Discord](https://discord.gg/QuStdQGSGK)!
|
||||||
- [Documentation](https://nightly.mealie.io)
|
- [Documentation](https://docs.mealie.io/)
|
||||||
|
|
||||||
|
|
||||||
|
## Key Features
|
||||||
|
- Recipe imports: Create recipes, by **importing from a URL** or entering data manually
|
||||||
|
- Meal Planner: Use the **Meal Planner** to plan your what you'll cook for the next week
|
||||||
|
- Shopping List: Put the necessary ingredients on your **Shopping List**, organised into sections of your local supermarket
|
||||||
|
- Cookbooks: Group recipes into **Cookbooks** based on your own criteria
|
||||||
|
- Docker: Easy **Docker** deployment
|
||||||
|
- Localisation: **Translations** for 35+ languages
|
||||||
|
|
||||||
<!-- CONTRIBUTING -->
|
<!-- CONTRIBUTING -->
|
||||||
## Contributing
|
## Contributing
|
||||||
|
|
||||||
@@ -58,7 +66,7 @@ If you are not a coder, you can still contribute financially. Financial contribu
|
|||||||
|
|
||||||
### Translations
|
### Translations
|
||||||
|
|
||||||
Translations can be a great way for **non-coders** to contribute to project. We use [Crowdin](https://crowdin.com/project/mealie) to allow several contributors to work on translating Mealie. You can simply help by voting for your preferred translations, or even by completely translating Mealie into a new language.
|
Translations can be a great way for **non-coders** to contribute to the project. We use [Crowdin](https://crowdin.com/project/mealie) to allow several contributors to work on translating Mealie. You can simply help by voting for your preferred translations, or even by completely translating Mealie into a new language.
|
||||||
|
|
||||||
For more information, check out the translation page on the [contributor's guide](https://nightly.mealie.io/contributors/translating/).
|
For more information, check out the translation page on the [contributor's guide](https://nightly.mealie.io/contributors/translating/).
|
||||||
|
|
||||||
@@ -80,16 +88,17 @@ Thanks to Depot for providing build instances for our Docker image builds.
|
|||||||
<!-- MARKDOWN LINKS & IMAGES -->
|
<!-- MARKDOWN LINKS & IMAGES -->
|
||||||
<!-- https://www.markdownguide.org/basic-syntax/#reference-style-links -->
|
<!-- https://www.markdownguide.org/basic-syntax/#reference-style-links -->
|
||||||
[contributors-shield]: https://img.shields.io/github/contributors/mealie-recipes/mealie.svg?style=flat-square
|
[contributors-shield]: https://img.shields.io/github/contributors/mealie-recipes/mealie.svg?style=flat-square
|
||||||
[docker-pull]: https://img.shields.io/docker/pulls/hkotel/mealie
|
[docker-pull]: https://img.shields.io/docker/pulls/hkotel/mealie?style=flat-square
|
||||||
|
[docker-url]: https://hub.docker.com/r/hkotel/mealie
|
||||||
|
[ghcr-pulls]: https://img.shields.io/badge/dynamic/json?url=https%3A%2F%2Fraw.githubusercontent.com%2Fipitio%2Fghcr-pulls%2Fmaster%2Findex.json&query=%24%5B%3F(%40.owner%3D%3D%22mealie-recipes%22%20%26%26%20%40.repo%3D%3D%22mealie%22%20%26%26%20%40.image%3D%3D%22mealie%22)%5D.pulls&style=flat-square&label=ghcr%20pulls
|
||||||
|
[ghcr-url]: https://github.com/mealie-recipes/mealie/pkgs/container/mealie
|
||||||
[contributors-url]: https://github.com/mealie-recipes/mealie/graphs/contributors
|
[contributors-url]: https://github.com/mealie-recipes/mealie/graphs/contributors
|
||||||
[forks-shield]: https://img.shields.io/github/forks/mealie-recipes/mealie.svg?style=flat-square
|
|
||||||
[forks-url]: https://github.com/mealie-recipes/mealie/network/members
|
|
||||||
[stars-shield]: https://img.shields.io/github/stars/mealie-recipes/mealie.svg?style=flat-square
|
[stars-shield]: https://img.shields.io/github/stars/mealie-recipes/mealie.svg?style=flat-square
|
||||||
[stars-url]: https://github.com/mealie-recipes/mealie/stargazers
|
[stars-url]: https://github.com/mealie-recipes/mealie/stargazers
|
||||||
[issues-shield]: https://img.shields.io/github/issues/mealie-recipes/mealie.svg?style=flat-square
|
[issues-shield]: https://img.shields.io/github/issues/mealie-recipes/mealie.svg?style=flat-square
|
||||||
[issues-url]: https://github.com/mealie-recipes/mealie/issues
|
[issues-url]: https://github.com/mealie-recipes/mealie/issues
|
||||||
[latest-release-shield]: https://img.shields.io/github/v/release/mealie-recipes/mealie?style=flat-square&label=latest%20release
|
[latest-release-shield]: https://img.shields.io/github/v/release/mealie-recipes/mealie?style=flat-square&label=latest%20release
|
||||||
[latest-release-url]: https://img.shields.io/github/v/release/mealie-recipes/mealie
|
[latest-release-url]: https://github.com/mealie-recipes/mealie/releases
|
||||||
[license-shield]: https://img.shields.io/github/license/mealie-recipes/mealie.svg?style=flat-square
|
[license-shield]: https://img.shields.io/github/license/mealie-recipes/mealie.svg?style=flat-square
|
||||||
[license-url]: https://github.com/mealie-recipes/mealie/blob/mealie-next/LICENSE
|
[license-url]: https://github.com/mealie-recipes/mealie/blob/mealie-next/LICENSE
|
||||||
[linkedin-shield]: https://img.shields.io/badge/-LinkedIn-black.svg?style=flat-square&logo=linkedin&colorB=555
|
[linkedin-shield]: https://img.shields.io/badge/-LinkedIn-black.svg?style=flat-square&logo=linkedin&colorB=555
|
||||||
|
|||||||
@@ -148,7 +148,7 @@ tasks:
|
|||||||
- poetry run python mealie/app.py
|
- poetry run python mealie/app.py
|
||||||
|
|
||||||
py:migrate:
|
py:migrate:
|
||||||
desc: generates a new migration file e.g. task py:migrate:generate "add new column"
|
desc: generates a new database migration file e.g. task py:migrate -- "add new column"
|
||||||
cmds:
|
cmds:
|
||||||
- poetry run alembic revision --autogenerate -m "{{ .CLI_ARGS }}"
|
- poetry run alembic revision --autogenerate -m "{{ .CLI_ARGS }}"
|
||||||
- task: py:format
|
- task: py:format
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ from text_unidecode import unidecode
|
|||||||
|
|
||||||
import mealie.db.migration_types
|
import mealie.db.migration_types
|
||||||
from alembic import op
|
from alembic import op
|
||||||
from mealie.db.models._model_utils import GUID
|
from mealie.db.models._model_utils.guid import GUID
|
||||||
|
|
||||||
# revision identifiers, used by Alembic.
|
# revision identifiers, used by Alembic.
|
||||||
revision = "5ab195a474eb"
|
revision = "5ab195a474eb"
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ Create Date: 2024-03-18 02:28:15.896959
|
|||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from datetime import datetime
|
from datetime import datetime, timezone
|
||||||
from textwrap import dedent
|
from textwrap import dedent
|
||||||
from typing import Any
|
from typing import Any
|
||||||
from uuid import uuid4
|
from uuid import uuid4
|
||||||
@@ -34,7 +34,7 @@ def new_user_rating(user_id: Any, recipe_id: Any, rating: float | None = None, i
|
|||||||
else:
|
else:
|
||||||
id = "%.32x" % uuid4().int
|
id = "%.32x" % uuid4().int
|
||||||
|
|
||||||
now = datetime.now().isoformat()
|
now = datetime.now(timezone.utc).isoformat()
|
||||||
return {
|
return {
|
||||||
"id": id,
|
"id": id,
|
||||||
"user_id": user_id,
|
"user_id": user_id,
|
||||||
|
|||||||
@@ -0,0 +1,48 @@
|
|||||||
|
"""Add staple flag to foods
|
||||||
|
|
||||||
|
Revision ID: 32d69327997b
|
||||||
|
Revises: 7788478a0338
|
||||||
|
Create Date: 2024-06-22 10:17:03.323966
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
import sqlalchemy as sa
|
||||||
|
from sqlalchemy import orm
|
||||||
|
|
||||||
|
from alembic import op
|
||||||
|
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = "32d69327997b"
|
||||||
|
down_revision = "7788478a0338"
|
||||||
|
branch_labels = None
|
||||||
|
depends_on = None
|
||||||
|
|
||||||
|
|
||||||
|
def is_postgres():
|
||||||
|
return op.get_context().dialect.name == "postgresql"
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade():
|
||||||
|
with op.batch_alter_table("ingredient_foods") as batch_op:
|
||||||
|
batch_op.add_column(sa.Column("on_hand", sa.Boolean(), nullable=True, default=False))
|
||||||
|
|
||||||
|
bind = op.get_bind()
|
||||||
|
session = orm.Session(bind=bind)
|
||||||
|
|
||||||
|
with session:
|
||||||
|
if is_postgres():
|
||||||
|
stmt = "UPDATE ingredient_foods SET on_hand = FALSE;"
|
||||||
|
else:
|
||||||
|
stmt = "UPDATE ingredient_foods SET on_hand = 0;"
|
||||||
|
|
||||||
|
session.execute(sa.text(stmt))
|
||||||
|
|
||||||
|
# forbid nulls after migration
|
||||||
|
with op.batch_alter_table("ingredient_foods") as batch_op:
|
||||||
|
batch_op.alter_column("on_hand", nullable=False)
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade():
|
||||||
|
with op.batch_alter_table("ingredient_foods") as batch_op:
|
||||||
|
batch_op.drop_column("on_hand")
|
||||||
@@ -1,4 +1,8 @@
|
|||||||
preserve_hierarchy: false
|
preserve_hierarchy: false
|
||||||
|
pull_request_title: "chore(l10n): New Crowdin updates"
|
||||||
|
pull_request_labels: [
|
||||||
|
"l10n"
|
||||||
|
]
|
||||||
files:
|
files:
|
||||||
- source: /frontend/lang/messages/en-US.json
|
- source: /frontend/lang/messages/en-US.json
|
||||||
translation: /frontend/lang/messages/%locale%.json
|
translation: /frontend/lang/messages/%locale%.json
|
||||||
|
|||||||
@@ -92,6 +92,9 @@ RUN apt-get update \
|
|||||||
libldap-2.5 \
|
libldap-2.5 \
|
||||||
&& rm -rf /var/lib/apt/lists/*
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
|
# create directory used for Docker Secrets
|
||||||
|
RUN mkdir -p /run/secrets
|
||||||
|
|
||||||
# copying poetry and venv into image
|
# copying poetry and venv into image
|
||||||
COPY --from=builder-base $POETRY_HOME $POETRY_HOME
|
COPY --from=builder-base $POETRY_HOME $POETRY_HOME
|
||||||
COPY --from=builder-base $PYSETUP_PATH $PYSETUP_PATH
|
COPY --from=builder-base $PYSETUP_PATH $PYSETUP_PATH
|
||||||
@@ -104,7 +107,6 @@ COPY --from=crfpp /usr/local/bin/crf_test /usr/local/bin/crf_test
|
|||||||
# copy backend
|
# copy backend
|
||||||
COPY ./mealie $MEALIE_HOME/mealie
|
COPY ./mealie $MEALIE_HOME/mealie
|
||||||
COPY ./poetry.lock ./pyproject.toml $MEALIE_HOME/
|
COPY ./poetry.lock ./pyproject.toml $MEALIE_HOME/
|
||||||
COPY ./gunicorn_conf.py $MEALIE_HOME
|
|
||||||
|
|
||||||
# Alembic
|
# Alembic
|
||||||
COPY ./alembic $MEALIE_HOME/alembic
|
COPY ./alembic $MEALIE_HOME/alembic
|
||||||
|
|||||||
@@ -25,12 +25,6 @@ services:
|
|||||||
POSTGRES_PORT: 5432
|
POSTGRES_PORT: 5432
|
||||||
POSTGRES_DB: mealie
|
POSTGRES_DB: mealie
|
||||||
|
|
||||||
# =====================================
|
|
||||||
# Web Concurrency
|
|
||||||
WEB_GUNICORN: "false"
|
|
||||||
WORKERS_PER_CORE: 0.5
|
|
||||||
MAX_WORKERS: 1
|
|
||||||
WEB_CONCURRENCY: 1
|
|
||||||
# =====================================
|
# =====================================
|
||||||
# Email Configuration
|
# Email Configuration
|
||||||
# SMTP_HOST=
|
# SMTP_HOST=
|
||||||
|
|||||||
@@ -37,14 +37,8 @@ init() {
|
|||||||
|
|
||||||
change_user
|
change_user
|
||||||
init
|
init
|
||||||
GUNICORN_PORT=${API_PORT:-9000}
|
|
||||||
|
|
||||||
# Start API
|
# Start API
|
||||||
HOST_IP=`/sbin/ip route|awk '/default/ { print $3 }'`
|
HOST_IP=`/sbin/ip route|awk '/default/ { print $3 }'`
|
||||||
|
|
||||||
if [ "$WEB_GUNICORN" = 'true' ]; then
|
exec python /app/mealie/main.py
|
||||||
echo "Starting Gunicorn"
|
|
||||||
exec gunicorn mealie.app:app -b 0.0.0.0:$GUNICORN_PORT --forwarded-allow-ips=$HOST_IP -k uvicorn.workers.UvicornWorker -c /app/gunicorn_conf.py --preload
|
|
||||||
else
|
|
||||||
exec python /app/mealie/main.py
|
|
||||||
fi
|
|
||||||
|
|||||||
BIN
docs/docs/assets/img/n8n/n8n-cred-app.png
Normal file
BIN
docs/docs/assets/img/n8n/n8n-cred-app.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 16 KiB |
BIN
docs/docs/assets/img/n8n/n8n-cred-connection.png
Normal file
BIN
docs/docs/assets/img/n8n/n8n-cred-connection.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 26 KiB |
BIN
docs/docs/assets/img/n8n/n8n-mealie-backup.png
Normal file
BIN
docs/docs/assets/img/n8n/n8n-mealie-backup.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 105 KiB |
BIN
docs/docs/assets/img/n8n/n8n-workflow-auth.png
Normal file
BIN
docs/docs/assets/img/n8n/n8n-workflow-auth.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 13 KiB |
BIN
docs/docs/assets/img/n8n/n8n-workflow-import.png
Normal file
BIN
docs/docs/assets/img/n8n/n8n-workflow-import.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 24 KiB |
386
docs/docs/assets/other/n8n/n8n-mealie-backup.json
Normal file
386
docs/docs/assets/other/n8n/n8n-mealie-backup.json
Normal file
@@ -0,0 +1,386 @@
|
|||||||
|
{
|
||||||
|
"name": "Mealie Backup",
|
||||||
|
"nodes": [
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"rule": {
|
||||||
|
"interval": [
|
||||||
|
{}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"id": "2ec440b4-0668-4bc0-aa66-4023d6379f28",
|
||||||
|
"name": "Schedule Trigger",
|
||||||
|
"type": "n8n-nodes-base.scheduleTrigger",
|
||||||
|
"typeVersion": 1.1,
|
||||||
|
"position": [
|
||||||
|
240,
|
||||||
|
660
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"method": "POST",
|
||||||
|
"url": "https://mealie.example/api/admin/backups",
|
||||||
|
"authentication": "genericCredentialType",
|
||||||
|
"genericAuthType": "httpHeaderAuth",
|
||||||
|
"options": {}
|
||||||
|
},
|
||||||
|
"id": "235f26f7-0f45-479e-a7e3-bf8cda7c8426",
|
||||||
|
"name": "Run Backup ",
|
||||||
|
"type": "n8n-nodes-base.httpRequest",
|
||||||
|
"typeVersion": 4.1,
|
||||||
|
"position": [
|
||||||
|
520,
|
||||||
|
520
|
||||||
|
],
|
||||||
|
"notesInFlow": false,
|
||||||
|
"credentials": {
|
||||||
|
"httpHeaderAuth": {
|
||||||
|
"id": "GSL12tNi3MPvTZux",
|
||||||
|
"name": "Mealie API"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"notes": "Send an API call to run the backup"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"method": "POST",
|
||||||
|
"url": "https://ntfy.example/backups",
|
||||||
|
"sendHeaders": true,
|
||||||
|
"headerParameters": {
|
||||||
|
"parameters": [
|
||||||
|
{
|
||||||
|
"name": "Title",
|
||||||
|
"value": "Meale Backup Failure"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Priority",
|
||||||
|
"value": "urgent"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Tags",
|
||||||
|
"value": "warning"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Actions",
|
||||||
|
"value": "view, Open Mealie, https://mealie.example/admin/backups; view, Open n8n, https://n8n.example"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"sendBody": true,
|
||||||
|
"contentType": "raw",
|
||||||
|
"body": "\"Full Panic!\"",
|
||||||
|
"options": {}
|
||||||
|
},
|
||||||
|
"id": "40ba81a5-5741-4b15-98af-1a9e6b34f997",
|
||||||
|
"name": "Ntfy Warning",
|
||||||
|
"type": "n8n-nodes-base.httpRequest",
|
||||||
|
"typeVersion": 4.1,
|
||||||
|
"position": [
|
||||||
|
1000,
|
||||||
|
520
|
||||||
|
],
|
||||||
|
"onError": "continueRegularOutput"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"url": "https://mealie.example/api/admin/backups",
|
||||||
|
"authentication": "genericCredentialType",
|
||||||
|
"genericAuthType": "httpHeaderAuth",
|
||||||
|
"options": {}
|
||||||
|
},
|
||||||
|
"id": "b75571d0-d926-440c-897f-55b89c6a5080",
|
||||||
|
"name": "Get all backups",
|
||||||
|
"type": "n8n-nodes-base.httpRequest",
|
||||||
|
"typeVersion": 4.1,
|
||||||
|
"position": [
|
||||||
|
520,
|
||||||
|
820
|
||||||
|
],
|
||||||
|
"credentials": {
|
||||||
|
"httpHeaderAuth": {
|
||||||
|
"id": "GSL12tNi3MPvTZux",
|
||||||
|
"name": "Mealie API"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"fieldToSplitOut": "imports",
|
||||||
|
"options": {}
|
||||||
|
},
|
||||||
|
"id": "943d0e83-682b-4500-9faf-53284cfb02c6",
|
||||||
|
"name": "Split Out",
|
||||||
|
"type": "n8n-nodes-base.splitOut",
|
||||||
|
"typeVersion": 1,
|
||||||
|
"position": [
|
||||||
|
720,
|
||||||
|
820
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"jsCode": "// Get input data\nconst inputData = items.map(item => item.json);\n\n// Sort the data based on the 'date' field in descending order\ninputData.sort((a, b) => new Date(b.date) - new Date(a.date));\n\n// Get all records except the latest 7\nconst allExceptLatest7 = inputData.slice(7);\n\n// Map the output data back to the required format\nreturn allExceptLatest7.map(record => ({ json: record }));\n"
|
||||||
|
},
|
||||||
|
"id": "64eae81d-fdb6-44f7-9a2d-eff8d1763281",
|
||||||
|
"name": "Code",
|
||||||
|
"type": "n8n-nodes-base.code",
|
||||||
|
"typeVersion": 2,
|
||||||
|
"position": [
|
||||||
|
860,
|
||||||
|
820
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"method": "DELETE",
|
||||||
|
"url": "=https://mealie.example/api/admin/backups/{{ $json.name }}",
|
||||||
|
"authentication": "genericCredentialType",
|
||||||
|
"genericAuthType": "httpHeaderAuth",
|
||||||
|
"options": {}
|
||||||
|
},
|
||||||
|
"id": "1148eeb8-4860-46df-8f61-0e85ea1e0e89",
|
||||||
|
"name": "Delete Oldies",
|
||||||
|
"type": "n8n-nodes-base.httpRequest",
|
||||||
|
"typeVersion": 4.1,
|
||||||
|
"position": [
|
||||||
|
1040,
|
||||||
|
820
|
||||||
|
],
|
||||||
|
"credentials": {
|
||||||
|
"httpHeaderAuth": {
|
||||||
|
"id": "GSL12tNi3MPvTZux",
|
||||||
|
"name": "Mealie API"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"content": "Sends API Call to run backup",
|
||||||
|
"height": 225,
|
||||||
|
"width": 226,
|
||||||
|
"color": 4
|
||||||
|
},
|
||||||
|
"id": "cd2cb5db-87c1-40d8-a746-e61ace231987",
|
||||||
|
"name": "Sticky Note",
|
||||||
|
"type": "n8n-nodes-base.stickyNote",
|
||||||
|
"typeVersion": 1,
|
||||||
|
"position": [
|
||||||
|
460,
|
||||||
|
460
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"content": "Is there an error?",
|
||||||
|
"height": 225,
|
||||||
|
"width": 231,
|
||||||
|
"color": 3
|
||||||
|
},
|
||||||
|
"id": "0bebecbe-903e-4a69-bb1a-35619e68b540",
|
||||||
|
"name": "Sticky Note1",
|
||||||
|
"type": "n8n-nodes-base.stickyNote",
|
||||||
|
"typeVersion": 1,
|
||||||
|
"position": [
|
||||||
|
700,
|
||||||
|
460
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"content": "Send alert to NTFY",
|
||||||
|
"height": 225,
|
||||||
|
"width": 229
|
||||||
|
},
|
||||||
|
"id": "0b732adb-8a84-456d-b26d-5fc5ee5a4cae",
|
||||||
|
"name": "Sticky Note2",
|
||||||
|
"type": "n8n-nodes-base.stickyNote",
|
||||||
|
"typeVersion": 1,
|
||||||
|
"position": [
|
||||||
|
940,
|
||||||
|
460
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"content": "Gets all backups in Mealie",
|
||||||
|
"height": 225,
|
||||||
|
"width": 226,
|
||||||
|
"color": 4
|
||||||
|
},
|
||||||
|
"id": "99c6886b-6a07-4b51-b395-d4bbcbde7d18",
|
||||||
|
"name": "Sticky Note3",
|
||||||
|
"type": "n8n-nodes-base.stickyNote",
|
||||||
|
"typeVersion": 1,
|
||||||
|
"position": [
|
||||||
|
460,
|
||||||
|
760
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"content": "Splits the data, and parses the output",
|
||||||
|
"height": 225,
|
||||||
|
"width": 281
|
||||||
|
},
|
||||||
|
"id": "549555f8-0aed-42c0-9693-9c0d93902796",
|
||||||
|
"name": "Sticky Note4",
|
||||||
|
"type": "n8n-nodes-base.stickyNote",
|
||||||
|
"typeVersion": 1,
|
||||||
|
"position": [
|
||||||
|
700,
|
||||||
|
760
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"content": "Deletes all but the last 7 backups",
|
||||||
|
"height": 225,
|
||||||
|
"width": 229
|
||||||
|
},
|
||||||
|
"id": "bcc5f0ba-73e9-42d7-b01b-c32f9f69f2f7",
|
||||||
|
"name": "Sticky Note5",
|
||||||
|
"type": "n8n-nodes-base.stickyNote",
|
||||||
|
"typeVersion": 1,
|
||||||
|
"position": [
|
||||||
|
1000,
|
||||||
|
760
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"content": "Run every day a 01:00",
|
||||||
|
"height": 225,
|
||||||
|
"width": 226,
|
||||||
|
"color": 4
|
||||||
|
},
|
||||||
|
"id": "ce797062-d727-43e3-a27f-e29b13ad3c9a",
|
||||||
|
"name": "Sticky Note6",
|
||||||
|
"type": "n8n-nodes-base.stickyNote",
|
||||||
|
"typeVersion": 1,
|
||||||
|
"position": [
|
||||||
|
180,
|
||||||
|
600
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"parameters": {
|
||||||
|
"conditions": {
|
||||||
|
"options": {
|
||||||
|
"caseSensitive": true,
|
||||||
|
"leftValue": "",
|
||||||
|
"typeValidation": "strict"
|
||||||
|
},
|
||||||
|
"conditions": [
|
||||||
|
{
|
||||||
|
"id": "8b00bb85-827f-4f2f-813e-db0d25e927d3",
|
||||||
|
"leftValue": "={{ $json.error }}",
|
||||||
|
"rightValue": "",
|
||||||
|
"operator": {
|
||||||
|
"type": "boolean",
|
||||||
|
"operation": "true",
|
||||||
|
"singleValue": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"combinator": "and"
|
||||||
|
},
|
||||||
|
"options": {}
|
||||||
|
},
|
||||||
|
"id": "fefd3e8b-9b71-490a-82e3-25e5468a4135",
|
||||||
|
"name": "Error?",
|
||||||
|
"type": "n8n-nodes-base.filter",
|
||||||
|
"typeVersion": 2,
|
||||||
|
"position": [
|
||||||
|
760,
|
||||||
|
520
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"pinData": {},
|
||||||
|
"connections": {
|
||||||
|
"Schedule Trigger": {
|
||||||
|
"main": [
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"node": "Run Backup ",
|
||||||
|
"type": "main",
|
||||||
|
"index": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"node": "Get all backups",
|
||||||
|
"type": "main",
|
||||||
|
"index": 0
|
||||||
|
}
|
||||||
|
]
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"Run Backup ": {
|
||||||
|
"main": [
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"node": "Error?",
|
||||||
|
"type": "main",
|
||||||
|
"index": 0
|
||||||
|
}
|
||||||
|
]
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"Get all backups": {
|
||||||
|
"main": [
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"node": "Split Out",
|
||||||
|
"type": "main",
|
||||||
|
"index": 0
|
||||||
|
}
|
||||||
|
]
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"Split Out": {
|
||||||
|
"main": [
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"node": "Code",
|
||||||
|
"type": "main",
|
||||||
|
"index": 0
|
||||||
|
}
|
||||||
|
]
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"Code": {
|
||||||
|
"main": [
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"node": "Delete Oldies",
|
||||||
|
"type": "main",
|
||||||
|
"index": 0
|
||||||
|
}
|
||||||
|
]
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"Error?": {
|
||||||
|
"main": [
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"node": "Ntfy Warning",
|
||||||
|
"type": "main",
|
||||||
|
"index": 0
|
||||||
|
}
|
||||||
|
]
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"active": false,
|
||||||
|
"settings": {
|
||||||
|
"executionOrder": "v1"
|
||||||
|
},
|
||||||
|
"versionId": "68e3e469-3ddb-4838-b09d-3c69fdd851f5",
|
||||||
|
"meta": {
|
||||||
|
"templateCredsSetupCompleted": true,
|
||||||
|
"instanceId": "630eefaa8c490b9c5221d83a182af6450c2c3efaf4b580b8ac348631abfe1aeb"
|
||||||
|
},
|
||||||
|
"id": "whloxeXkdBWWi2Uj",
|
||||||
|
"tags": []
|
||||||
|
}
|
||||||
@@ -1,88 +0,0 @@
|
|||||||
# v0.1.0 - Initial Beta
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
- Fixed Can't delete recipe after changing name - Closes Closes #67
|
|
||||||
- Fixed No image when added by URL, and can't add an image - Closes Closes #66
|
|
||||||
- Fixed Images saved with no way to delete when add recipe via URL fails - Closes Closes #43
|
|
||||||
|
|
||||||
### Features
|
|
||||||
- Additional Language Support
|
|
||||||
- Improved deployment documentation
|
|
||||||
- Additional database! SQlite is now supported! - Closes #48
|
|
||||||
- All site data is now backed up.
|
|
||||||
- Support for Prep Time, Total Time, and Cook Time field - Closes #63
|
|
||||||
- New backup import process with support for themes and site settings
|
|
||||||
- **BETA** ARM support! - Closes #69
|
|
||||||
|
|
||||||
### Code / Developer Improvements
|
|
||||||
- Unified Database Access Layers
|
|
||||||
- Poetry / pyproject.toml support over requirements.txt
|
|
||||||
- Local development without database is now possible!
|
|
||||||
- Local mkdocs server added to docker-compose.dev.yml
|
|
||||||
- Major code refactoring to support new database layer
|
|
||||||
- Global variable refactor
|
|
||||||
|
|
||||||
### Breaking Changes
|
|
||||||
|
|
||||||
- Internal docker port is now 80 instead of 9000. You MUST remap the internal port to connect to the UI.
|
|
||||||
|
|
||||||
!!! error "Breaking Changes"
|
|
||||||
As I've adopted the SQL database model I find that using 2 different types of databases will inevitably hinder development. As such after release v0.1.0 support for mongoDB will no longer be available. Prior to upgrading to v0.2.0 you will need to export your site and import after updating. This should be a painless process and require minimal intervention on the users part. Moving forward we will do our best to minimize changes that require user intervention like this and make updates a smooth process.
|
|
||||||
|
|
||||||
|
|
||||||
## v0.0.2 - Pre-release Second Patch
|
|
||||||
A quality update with major props to [zackbcom](https://github.com/zackbcom) for working hard on making the theming just that much better!
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
- Fixed empty backup failure without markdown template
|
|
||||||
- Fixed opacity issues with marked steps - [mtoohey31](https://github.com/mtoohey31)
|
|
||||||
- Fixed hot-reloading development environment - [grssmnn](https://github.com/grssmnn)
|
|
||||||
- Fixed recipe not saving without image - Closes #7 + Closes #54
|
|
||||||
- Fixed parsing error on image property null - Closes #43
|
|
||||||
|
|
||||||
### General Improvements
|
|
||||||
- Added Confirmation component to deleting recipes - [zackbcom](https://github.com/zackbcom)
|
|
||||||
- Updated Theme backend - [zackbcom](https://github.com/zackbcom)
|
|
||||||
- Added Persistent storage to vuex - [zackbcom](https://github.com/zackbcom)
|
|
||||||
- General Color/Theme Improvements
|
|
||||||
- More consistent UI
|
|
||||||
- More minimalist coloring
|
|
||||||
- Added API key extras to Recipe Data - [See Documentation](/api/api-usage/)
|
|
||||||
- Users can now add custom json key/value pairs to all recipes via the editor for access in 3rd part applications. For example users can add a "message" field in the extras that can be accessed on API calls to play a message over google home.
|
|
||||||
- Improved image rendering (nearly x2 speed)
|
|
||||||
- Improved documentation + API Documentation
|
|
||||||
- Improved recipe parsing - Closes #51
|
|
||||||
- User feedback on backup importing
|
|
||||||
|
|
||||||
## v0.0.1 - Pre-release Patch
|
|
||||||
### General
|
|
||||||
- Updated Favicon
|
|
||||||
- Renamed Frontend Window
|
|
||||||
- Added Debug folder to dump scraper data prior to processing.
|
|
||||||
|
|
||||||
### Recipes
|
|
||||||
- Added user feedback on bad URL
|
|
||||||
- Better backend data validation for updating recipes, avoid small syntax errors corrupting database entry. [Closes #8](https://github.com/mealie-recipes/mealie/issues/8)
|
|
||||||
- Fixed spacing Closes while editing new recipes in JSON
|
|
||||||
|
|
||||||
## v0.0.0 - Initial Pre-release
|
|
||||||
The initial pre-release. It should be semi-functional but does not include a lot of user feedback You may notice errors that have no user feedback and have no idea what went wrong.
|
|
||||||
|
|
||||||
### Recipes
|
|
||||||
- Automatic web scrapping for common recipe platforms
|
|
||||||
- Interactive API Documentation thanks to [FastAPI](https://fastapi.tiangolo.com/) and [Swagger](https://petstore.swagger.io/)
|
|
||||||
- UI Recipe Editor
|
|
||||||
- JSON Recipe Editor in browser
|
|
||||||
- Custom tags and categories
|
|
||||||
- Rate recipes
|
|
||||||
- Add notes to recipes
|
|
||||||
- Migration From Other Platforms
|
|
||||||
- Chowdown
|
|
||||||
### Meal Planner
|
|
||||||
- Random Meal plan generation based off categories
|
|
||||||
- Expose notes in the API to allow external applications to access relevant information for meal plans
|
|
||||||
|
|
||||||
### Database Import / Export
|
|
||||||
- Easily Import / Export your recipes from the UI
|
|
||||||
- Export recipes in markdown format for universal access
|
|
||||||
- Use the default or a custom jinja2 template
|
|
||||||
@@ -1,72 +0,0 @@
|
|||||||
# v0.2.1 - Hot Fixes!
|
|
||||||
|
|
||||||
### Features and Improvements
|
|
||||||
- Fixes upload image error when no photo was scrapped
|
|
||||||
- Fixes no last_recipe.json not updating
|
|
||||||
- Added markdown rendering for notes
|
|
||||||
- New notifications
|
|
||||||
- Minor UI improvements
|
|
||||||
- Recipe editor refactor
|
|
||||||
- Settings/Theme models refactor
|
|
||||||
|
|
||||||
### Development / Misc
|
|
||||||
- Added async file response for images, downloading files.
|
|
||||||
- Breakup recipe view component
|
|
||||||
|
|
||||||
# v0.2.0 - Now with Test!
|
|
||||||
This is, what I think, is a big release! Tons of new features and some great quality of life improvements with some additional features. You may find that I made promises to include some fixes/features in v0.2.0. The short of is I greatly underestimated the work needed to refactor the database to a usable state and integrate categories in a way that is useful for users. This shouldn't be taken as a sign that I'm dropping those feature requests or ignoring them. I felt it was better to push a release in the current state rather than drag on development to try and fulfil all of the promises I made.
|
|
||||||
|
|
||||||
!!! warning "Upgrade Process"
|
|
||||||
Database Breaks! I have not yet implemented a database migration service. As such, upgrades cannot be done by simply pulling the image. You must first export your recipes, update your deployment, and then import your recipes. This pattern is likely to be how upgrades take place prior to v1.0. After v1.0 migrations will be done automatically.
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
- Remove ability to save recipe with no name
|
|
||||||
- Fixed data validation error on missing parameters
|
|
||||||
- Fixed failed database initialization at startup - Closes #98
|
|
||||||
- Fixed misaligned text on various cards
|
|
||||||
- Fixed bug that blocked opening links in new tabs - Closes #122
|
|
||||||
- Fixed router link bugs - Closes #122
|
|
||||||
- Fixed navigation on keyboard selection - Closes #139
|
|
||||||
|
|
||||||
### Features and Improvements
|
|
||||||
- 🐳 Dockerfile now 1/5 of the size!
|
|
||||||
- 🌎 UI Language Selection + Additional Supported Language
|
|
||||||
- **Home Page**
|
|
||||||
- By default your homepage will display only the recently added recipes. You can configured sections to show on the home-screen based of categories on the settings page.
|
|
||||||
- A new sidebar is now shown on the main page that lists all the categories in the database. Clicking on them navigates into a page that shows only recipes.
|
|
||||||
- Basic Sort functionality has been added. More options are on the way!
|
|
||||||
- **Meal Planner**
|
|
||||||
- Improved Search (Fuzzy Search)
|
|
||||||
- New Scheduled card support
|
|
||||||
- **Recipe Editor**
|
|
||||||
- Ingredients are now sortable via drag-and-drop
|
|
||||||
- Known categories now show up in the dropdown - Closes 83
|
|
||||||
- Initial code for data validation to prevent errors
|
|
||||||
- **Migrations**
|
|
||||||
- Card based redesign
|
|
||||||
- Upload from the UI
|
|
||||||
- Unified Chowdown / Nextcloud import process. (Removed Git as a dependency)
|
|
||||||
- **API**
|
|
||||||
- Category and Tag endpoints added
|
|
||||||
- Major Endpoint refactor
|
|
||||||
- Improved API documentation
|
|
||||||
- Link to your Local API is now on your `/settings/site`. You can use it to explore your API.
|
|
||||||
|
|
||||||
- **Style**
|
|
||||||
- Continued work on button/style unification
|
|
||||||
- Adding icons to buttons
|
|
||||||
- New Color Theme Picker UI
|
|
||||||
|
|
||||||
### Development
|
|
||||||
- Fixed Vetur config file. Autocomplete in VSCode works!
|
|
||||||
- File/Folder restructuring
|
|
||||||
- Added Prettier config
|
|
||||||
- Fixed incorrect layout code
|
|
||||||
- FastAPI Route tests for major operations - WIP (shallow testing)
|
|
||||||
|
|
||||||
### Breaking Changes
|
|
||||||
|
|
||||||
!!! error "Breaking Changes"
|
|
||||||
- API endpoints have been refactored to adhere to a more consistent standard. This is a WIP and more changes are likely to occur.
|
|
||||||
- Officially Dropped MongoDB Support
|
|
||||||
- Database Breaks! We have not yet implemented a database migration service. As such, upgrades cannot be done by simply pulling the image. You must first export your recipes, update your deployment, and then import your recipes. This pattern is likely to be how upgrades take place prior to v1.0. After v1.0 migrations will be done automatically.
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
# v0.3.0
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
- Fixed open search on `/` when in input. - Closes #174
|
|
||||||
- Error when importing recipe: KeyError: '@type' - Closes #145
|
|
||||||
- Fixed Import Issue - bhg.com - Closes #138
|
|
||||||
- Scraper not working with recipe containing HowToSection - Closes #73
|
|
||||||
|
|
||||||
### Features and Improvements
|
|
||||||
- Improved Nextcloud Imports
|
|
||||||
- Improved Recipe Parser!
|
|
||||||
- Open search with `/` hotkey!
|
|
||||||
- Database and App version are now split
|
|
||||||
- Unified and improved snackbar notifications
|
|
||||||
- New Category/Tag endpoints to filter all recipes by Category or Tag
|
|
||||||
- Category sidebar now has show/hide behavior on mobile
|
|
||||||
- Settings menu on mobile is improved
|
|
||||||
- **Meal Planner**
|
|
||||||
- You can now restrict recipe categories used for random meal-plan creation in the settings menu
|
|
||||||
- Recipe picker dialog will now display recipes when the search bar is empty
|
|
||||||
- Minor UI improvements
|
|
||||||
- **Shopping lists!** Shopping list can now be generated from a meal plan. Currently ingredients are split by recipes or there is a beta feature that attempts to sort them by similarity.
|
|
||||||
- **Recipe Viewer**
|
|
||||||
- Categories, Tags, and Notes will now be displayed below the steps on smaller screens
|
|
||||||
- **Recipe Editor**
|
|
||||||
- Text areas now auto grow to fit content
|
|
||||||
- Description, Steps, and Notes support Markdown! This includes inline html in Markdown.
|
|
||||||
- **Imports**
|
|
||||||
- A revamped dialog has been created to provide more information on restoring backups. Exceptions on the backend are now sent to the frontend and are easily viewable to see what went wrong when you restored a backup. This functionality will be ported over to the migrations in a future release.
|
|
||||||
@@ -1,86 +0,0 @@
|
|||||||
# v0.4.0 Whoa, What a Release!
|
|
||||||
|
|
||||||
**App Version: v0.4.0**
|
|
||||||
|
|
||||||
**Database Version: v0.4.0**
|
|
||||||
|
|
||||||
## Breaking Changes
|
|
||||||
|
|
||||||
!!! error "Breaking Changes"
|
|
||||||
|
|
||||||
#### Database
|
|
||||||
A new database will be created. You must export your data and then import it after upgrading.
|
|
||||||
|
|
||||||
#### Site Settings
|
|
||||||
With the addition of group settings and a re-write of the database layer of the application backend, there is no migration path for your current site settings. Webhooks Settings, Meal Plan Categories are now managed by groups. Site settings, mainly homepage settings, are now site specific and managed by administrators. When upgrading be sure to uncheck the settings when importing your old data.
|
|
||||||
|
|
||||||
#### ENV Variables
|
|
||||||
Names have been changed to be more consistent with industry standards. See the [Installation Page](/mealie/getting-started/install/) for new parameters.
|
|
||||||
|
|
||||||
## Bug Fixes
|
|
||||||
- Fixed Search Results Limited to 100 - #198
|
|
||||||
- Fixed recipes from marmiton.org not fully scrapped - #196
|
|
||||||
- Fixed Unable to get a page to load - #194
|
|
||||||
- Fixed Recipe's from Epicurious don't upload. - #193
|
|
||||||
- Fixed Edited blank recipe in meal plan is not saved - #184
|
|
||||||
- Fixed Create a new meal plan allows selection of an end date that is prior to the start date - #183
|
|
||||||
- Fixed Original URL not saved to imported recipe in 0.3.0-dev - #183
|
|
||||||
- Fixed "IndexError: list index out of range" when viewing shopping list for meal plan containing days without a recipe selected - #178
|
|
||||||
|
|
||||||
## Features and Improvements
|
|
||||||
|
|
||||||
### General
|
|
||||||
- Documentation Rewrite
|
|
||||||
- [New Demo Site!](https://mealie-demo.hay-kot.dev/)
|
|
||||||
- New Documentation
|
|
||||||
- Landing Page
|
|
||||||
- Custom Caddy Configuration
|
|
||||||
- User Management
|
|
||||||
- Introduction
|
|
||||||
- Updated Documentation
|
|
||||||
- Everything!
|
|
||||||
- The API Reference is now better embedded inside of the docs
|
|
||||||
- New default external port in documentation (Port 9000 -> 9925). This is only the port exposed by the host to the docker image. It doesn't change any existing functionality.
|
|
||||||
|
|
||||||
### User Authentication
|
|
||||||
- Authentication! Tons of stuff went into creating a flexible authentication platform for a lot of different use cases. Review the documentation for more information on how to use the authentication, and how everything works together. More complex management of recipes and user restrictions are in the works, but this is a great start! Some key features include
|
|
||||||
- Sign Up Links
|
|
||||||
- Admin and User Roles
|
|
||||||
- Password Change
|
|
||||||
- Group Management
|
|
||||||
- Create/Edit/Delete Restrictions
|
|
||||||
|
|
||||||
### Custom Pages
|
|
||||||
- You can now create custom pages that are displayed on the homepage sidebar to organize categories of recipes into pages. For example, if you have several categories that encompass "Entrée" you can group all those categories together under the "Entrée" page. See [Building Pages](/mealie/site-administration/building-pages/) for more information.
|
|
||||||
!!! tip
|
|
||||||
Note that this replaces the behavior of automatically adding categories to the sidebar.
|
|
||||||
|
|
||||||
### UI Improvements
|
|
||||||
- Completed Redesign of the Admin Panel
|
|
||||||
- Profile Pages
|
|
||||||
- Side Panel Menu
|
|
||||||
- Improved UI for Recipe Search
|
|
||||||
- Language selector is now displayed on all pages and does not require an account
|
|
||||||
|
|
||||||
### Recipe Data
|
|
||||||
- Recipe Database Refactoring. Tons of new information is now stored for recipes in the database. Not all is accessible via the UI, but it's coming.
|
|
||||||
- Nutrition Information
|
|
||||||
- calories
|
|
||||||
- fatContent
|
|
||||||
- fiberContent
|
|
||||||
- proteinContent
|
|
||||||
- sodiumContent
|
|
||||||
- sugarContent
|
|
||||||
- recipeCuisine has been added
|
|
||||||
- "categories" has been migrated to "recipeCategory" to adhere closer to the standard schema
|
|
||||||
- "tool" - a list of tools used for the recipe
|
|
||||||
|
|
||||||
### Behind the Scenes
|
|
||||||
- Removed CDN dependencies
|
|
||||||
- Database Model Refactoring
|
|
||||||
- Import/Export refactoring
|
|
||||||
- File/Folder Name Refactoring
|
|
||||||
- Development is now easier with a makefile
|
|
||||||
- Mealie is now a proper package using poetry
|
|
||||||
- Test refactoring
|
|
||||||
- Test Coverage 83% up from 75%!
|
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
# v0.4.1
|
|
||||||
|
|
||||||
**App Version: v0.4.1**
|
|
||||||
|
|
||||||
**Database Version: v0.4.0**
|
|
||||||
|
|
||||||
!!! error "Breaking Changes"
|
|
||||||
|
|
||||||
#### Recipe Images
|
|
||||||
While it *shouldn't* be a breaking change, I feel it is important to note that you may experience issues with the new image migration. Recipe images are now minified, this is done on start-up, import, migration, and when a new recipe is created. The initial boot or load may be a bit slow if you have lots of recipes but you likely won't notice. What you may notice is that if your recipe slug and the image name do not match, you will encounter issues with your images showing up. This can be resolved by finding the image directory and rename it to the appropriate slug. I did fix multiple edge cases, but it is likely more exists. As always make a backup before you update!
|
|
||||||
|
|
||||||
On the plus side, this comes with a huge performance increase! 🎉
|
|
||||||
|
|
||||||
- Add markdown support for ingredients - Resolves #32
|
|
||||||
- Ingredients editor improvements
|
|
||||||
- Fix Tags/Categories render problems on recipes
|
|
||||||
- Tags redirect to new tag pages
|
|
||||||
- Categories redirect to category pages
|
|
||||||
- Fix Backup download blocked by authentication
|
|
||||||
- Random meal-planner will no longer duplicate recipes unless no other options
|
|
||||||
- New Quick Week button to generate next 5 day week of recipe slots.
|
|
||||||
- Minor UI tweaks
|
|
||||||
- Recipe Cards now display 2 recipe tags
|
|
||||||
- Recipe images are now minified. This comes with a serious performance improvement. On initial startup you may experience some delays. Images are migrated to the new structure on startup, depending on the size of your database this can take some time.
|
|
||||||
- Note that original images are still kept for large displays like on the individual recipe pages.
|
|
||||||
- A smaller image is used for recipe cards
|
|
||||||
- A 'tiny' image is used for search images.
|
|
||||||
- Advanced Search Page. You can now use the search page to filter recipes to include/exclude tags and categories as well as select And/Or matching criteria.
|
|
||||||
- Added link to advanced search on quick search
|
|
||||||
- Better support for NextCloud imports
|
|
||||||
- Translate keywords to tags
|
|
||||||
- Fix rollback on failure
|
|
||||||
- Recipe Tag/Category Input components have been unified and now share a single way to interact. To add a new category in the recipe editor you need to click to '+' icon next to the input and fill out the form. This is the same for adding a Tag.
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
# v0.4.2
|
|
||||||
|
|
||||||
**App Version: v0.4.2**
|
|
||||||
|
|
||||||
**Database Version: v0.4.0**
|
|
||||||
|
|
||||||
!!! error "Breaking Changes"
|
|
||||||
1. With a recent refactor some users been experiencing issues with an environmental variable not being set correct. If you are experiencing issues, please provide your comments [Here](https://github.com/mealie-recipes/mealie/issues/281).
|
|
||||||
|
|
||||||
2. If you are a developer, you may experience issues with development as a new environmental variable has been introduced. Setting `PRODUCTION=false` will allow you to develop as normal.
|
|
||||||
|
|
||||||
## Bug Fixes
|
|
||||||
- Fixed Initialization script (v0.4.1a Hot Fix) - Closes #274
|
|
||||||
- Fixed nested list error on recipe scrape - Closes #306
|
|
||||||
- Fixed ingredient checkboxes - Closes #304
|
|
||||||
- Removed link on recent - Closes #297
|
|
||||||
- Categories sidebar is auto generated if no pages are created - Closes #291
|
|
||||||
- Fix tag issues on creating custom pages - Closes #290
|
|
||||||
- Validate paths on export - Closes #275
|
|
||||||
- Walk Nextcloud import directory - Closes #254
|
|
||||||
|
|
||||||
## General Improvements
|
|
||||||
- Improved Nextcloud Migration. Mealie will now walk the directories in a zip file looking for directories that match the pattern of a Nextcloud Recipe. Closes #254
|
|
||||||
- Rewrite Keywords to Tag Fields
|
|
||||||
- Rewrite url to orgURL
|
|
||||||
- Improved Chowdown Migration
|
|
||||||
- Migration report is now similar to the Backup report
|
|
||||||
- Tags/Categories are now title cased on import "dinner" -> "Dinner"
|
|
||||||
- Depreciate `ENV` variable to `PRODUCTION`
|
|
||||||
- Set `PRODUCTION` env variable to default to true
|
|
||||||
- Unify Logger across the backend
|
|
||||||
- mealie.log and last_recipe.json are now downloadable from the frontend from the /admin/about
|
|
||||||
- New download schema where you request a token and then use that token to hit a single endpoint to download a file. This is a notable change if you are using the API to download backups.
|
|
||||||
- Recipe images can now be added directly from a URL - [See #117 for details](https://github.com/mealie-recipes/mealie/issues/117)
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
# v0.4.3
|
|
||||||
|
|
||||||
**App Version: v0.4.3**
|
|
||||||
|
|
||||||
**Database Version: v0.4.0**
|
|
||||||
|
|
||||||
## Bug Fixes
|
|
||||||
- Fix Upload error for Migrations
|
|
||||||
- Fixes #315 - Cannot select another language
|
|
||||||
- Fixes #314 - case-sensitive emails
|
|
||||||
- Fixes #312 - Profile Image Reload
|
|
||||||
|
|
||||||
## Improvements
|
|
||||||
- New TOKEN_TIME and DEFAULT_EMAIL env variables
|
|
||||||
@@ -1,129 +0,0 @@
|
|||||||
# v0.5.0 Too Many Changes!
|
|
||||||
|
|
||||||
**App Version: v0.5.0**
|
|
||||||
|
|
||||||
**Database Version: v0.5.0**
|
|
||||||
|
|
||||||
## Breaking Changes
|
|
||||||
|
|
||||||
!!! error "Breaking Changes"
|
|
||||||
|
|
||||||
#### Database
|
|
||||||
Database version has been bumped from v0.4.x -> v0.5.0. You will need to export and import your data. Moving forward, we will be using database migrations (BETA) to do this automatically. Note that you still must backup your data. If you don't, it's entirely possible something may go wrong and you could lose your data on upgrade.
|
|
||||||
|
|
||||||
#### Image Directory
|
|
||||||
the /data/img directory has been depreciated. All images are now stored in the /recipes/{slug}/image directory. Images should be migrated automatically, but you may experience issues related to this change.
|
|
||||||
|
|
||||||
#### API Usage
|
|
||||||
If you have been using the API directly, many of the routes and status codes have changed. You may experience issues with directly consuming the API.
|
|
||||||
|
|
||||||
#### Arm/v7 Support
|
|
||||||
Mealie will no longer build in CI/CD due to a issue with the rust compiler on 32 bit devices. You can reference [this issue on the matrix-org/synapse](https://github.com/matrix-org/synapse/issues/9403) Github page that are facing a similar issue. You may still be able to build the docker image you-self.
|
|
||||||
|
|
||||||
!!! warning "Potential Data Loss"
|
|
||||||
With this release comes a major rework of how files are stored on disk and where things belong. Migration of files should be done automatically. We have tested extensively with many different backups and user bases and have found that no one experienced data-loss. HOWEVER, with all the major changes that have occurred, it is vital that to prevent any data-loss you must create a backup and store that backup outside of your mealie instance. If you do not do this, you may lose your data.
|
|
||||||
|
|
||||||
## Bug Fixes
|
|
||||||
- Fixed #25 - Allow changing rating without going into edit
|
|
||||||
- Fixed #475 - trim whitespace on login
|
|
||||||
- Fixes #435 - Better Email Regex
|
|
||||||
- Fixed #428 - Meal Planner now works on iOS devices
|
|
||||||
- Fixed #419 - Typos
|
|
||||||
- Fixed #418 - You can now "export" shopping lists
|
|
||||||
- Fixed #356 - Shopping List items are now grouped
|
|
||||||
- Fixed #329 - Fixed profile image not loading
|
|
||||||
- Fixed #461 - Proper JSON serialization on webhooks
|
|
||||||
- Fixed #332 - Language settings are saved for one browser
|
|
||||||
- Fixes #281 - Slow Handling of Large Sets of Recipes
|
|
||||||
- Fixed #356 - Shopping lists generate duplicate items
|
|
||||||
- Fixed #271 - Slow handling of larger data sets
|
|
||||||
- Fixed #472, #469, #458, #456 - Improve Recipe Parser
|
|
||||||
|
|
||||||
## Features and Improvements
|
|
||||||
|
|
||||||
### Highlights
|
|
||||||
- Recipe Parser
|
|
||||||
- Recipes can now be imported with a bookmarklet!
|
|
||||||
- Significant improvement in supported sites with the new [Recipe Scraper Library](https://github.com/hhursev/recipe-scrapers)
|
|
||||||
- UI Debugging now available at `/recipes/debugger`
|
|
||||||
- Better error messages on failure
|
|
||||||
- ⚠️ last_recipe.json is now depreciated
|
|
||||||
- Beta Support for Postgres! 🎉 See the getting started page for details
|
|
||||||
- Recipe Features
|
|
||||||
- New button bar for editors with improved accessibility and performance
|
|
||||||
- Step Sections now supported
|
|
||||||
- Recipe Assets
|
|
||||||
- Add Asset image to recipe step
|
|
||||||
- Additional View Settings.
|
|
||||||
- Better print support
|
|
||||||
- New Toolbox Page!
|
|
||||||
- Bulk assign categories and tags by keyword search
|
|
||||||
- Title case all Categories or Tags with 1 click
|
|
||||||
- Create/Rename/Delete Operations for Tags/Categories
|
|
||||||
- Remove Unused Categories or Tags with 1 click
|
|
||||||
- Recipe Cards now have a menu button for quick actions!
|
|
||||||
- Edit
|
|
||||||
- Delete
|
|
||||||
- Integrated Share with supported OS/Browsers
|
|
||||||
- Print
|
|
||||||
- New Profile Dashboard!
|
|
||||||
- Edit Your Profile
|
|
||||||
- Create/Edit Themes
|
|
||||||
- View other users in your Group
|
|
||||||
- See what's for dinner
|
|
||||||
- Manage Long Live API Tokens (New)
|
|
||||||
- New Admin Dashboard! 🎉
|
|
||||||
- Now you can get some insight on your application with application statistics and events.
|
|
||||||
- See uncategorized/untagged recipes and organize them!
|
|
||||||
- Backup/Restore right from your dashboard
|
|
||||||
- See server side events. Now you can know who deleted your favorite recipe!
|
|
||||||
- New Event Notifications through the Apprise Library
|
|
||||||
- Get notified when specific server side events occur
|
|
||||||
|
|
||||||
### Meal Planner
|
|
||||||
- Multiple Recipes per day
|
|
||||||
- Supports meals without recipes (Enter title and description)
|
|
||||||
- Generate share-link from created meal-planners
|
|
||||||
- Shopping lists can be directly generated from the meal plan
|
|
||||||
|
|
||||||
### General
|
|
||||||
- User can now favorite recipes
|
|
||||||
- New 'Dark' Color Theme Packaged with Mealie
|
|
||||||
- Updated Recipe Card Sections Toolbar
|
|
||||||
- New Sort Options (They work this time!)
|
|
||||||
- Alphabetical
|
|
||||||
- Rating
|
|
||||||
- Created Date
|
|
||||||
- Updated Date
|
|
||||||
- Shuffle (Random Sort)
|
|
||||||
- New 'Random' Recipe button on recipe sections. Random recipes are selected from the filtered results below. For example, on the "Cakes" category page, you will only get recipes in the "Cakes" category.
|
|
||||||
- Rating can be updated without entering the editor - Closes #25
|
|
||||||
- Updated recipe editor styles and moved notes to below the steps.
|
|
||||||
- Redesigned search bar
|
|
||||||
- 'Dinner this week' shows a warning when no meal is planned yet
|
|
||||||
- 'Dinner today' shows a warning when no meal is planned yet
|
|
||||||
- More localization
|
|
||||||
- Start date for Week is now selectable
|
|
||||||
- Languages are now managed through Crowdin
|
|
||||||
- Application Bar was Rewritten
|
|
||||||
- Sidebar can now be toggled everywhere.
|
|
||||||
- New and improved mobile friendly bottom bar
|
|
||||||
- Improved styling for search bar in desktop
|
|
||||||
- Improved search layout on mobile
|
|
||||||
- Profile image now shown on all sidebars
|
|
||||||
- Switched from Flash Messages to Snackbar (Removed dependency)
|
|
||||||
|
|
||||||
### Performance
|
|
||||||
- Images are now served up by the Caddy increase performance and offloading some loads from the API server
|
|
||||||
- Requesting all recipes from the server has been rewritten to refresh less often and manage client side data better.
|
|
||||||
- All images are now converted to .webp for better compression
|
|
||||||
|
|
||||||
### Behind the Scenes
|
|
||||||
- The database layer has been added for future recipe scaling.
|
|
||||||
- Black and Flake8 now run as CI/CD checks
|
|
||||||
- New debian based docker image
|
|
||||||
- Unified Sidebar Components
|
|
||||||
- Refactor UI components to fit Vue best practices (WIP)
|
|
||||||
- The API returns more consistent status codes
|
|
||||||
- The API returns error code instead of error text when appropriate
|
|
||||||
- ⚠️ May cause side-effects if you were directly consuming the API
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
# v0.5.1
|
|
||||||
|
|
||||||
**App Version: v0.5.1**
|
|
||||||
|
|
||||||
**Database Version: v0.5.0**
|
|
||||||
|
|
||||||
|
|
||||||
## Bug Fixes
|
|
||||||
- Fixed #538 - Missing Ingredients on Editor
|
|
||||||
- Fixed error on webhooks for new groups
|
|
||||||
- Fixed various icons references
|
|
||||||
@@ -1,65 +0,0 @@
|
|||||||
# v0.5.2 - DRAFT
|
|
||||||
|
|
||||||
**App Version: v0.5.2**
|
|
||||||
|
|
||||||
**Database Version: v0.5.0**
|
|
||||||
|
|
||||||
## Bug Fixes
|
|
||||||
- Fixed #617 - Section behavior when adding a step
|
|
||||||
- Fixed #615 - Recipe Settings are not available when creating new recipe
|
|
||||||
- Fixed #625 - API of today's image returns strange characters
|
|
||||||
- Fixed [#590](https://github.com/mealie-recipes/mealie/issues/590) - Duplicate Events when using Gunicorn Workers
|
|
||||||
|
|
||||||
## Features and Improvements
|
|
||||||
|
|
||||||
### General
|
|
||||||
- Recipe Instructions now collapse when checked
|
|
||||||
- Default recipe settings can be set through ENV variables
|
|
||||||
- Recipe Ingredient lists can now container ingredient sections.
|
|
||||||
- You can now download and upload individual recipes
|
|
||||||
|
|
||||||
|
|
||||||
### Localization
|
|
||||||
|
|
||||||
Huge thanks to [@sephrat](https://github.com/sephrat) for all his work on the project. He's very consistent in his contributions to the project and nearly every release has had some of his code in it! Here's some highlights from this release
|
|
||||||
|
|
||||||
- Lazy Load Translations (Huge Performance Increase!)
|
|
||||||
- Tons of localization additions all around the site.
|
|
||||||
- All of the work that goes into managing and making Crowdin a great feature the application
|
|
||||||
|
|
||||||
#### Here a list of contributors on Crowding who make Mealie possible in different locals
|
|
||||||
|
|
||||||
| Name | Languages | Translated (Words) | Target Words |
|
|
||||||
| ---------------------------- | ------------------ | :----------------: | :----------: |
|
|
||||||
| retmas-gh | Polish | 550 | 625 |
|
|
||||||
| startos | Italian | 310 | 322 |
|
|
||||||
| CMBoii | Spanish | 256 | 291 |
|
|
||||||
| sephrat | French | 255 | 296 |
|
|
||||||
| Daniel Tildeman (tildeman) | Swedish | 233 | 228 |
|
|
||||||
| Rourke | Dutch | 216 | 214 |
|
|
||||||
| Andreas Waschinski (Wascha) | German | 207 | 202 |
|
|
||||||
| wengtad | Chinese Simplified | 176 | 343 |
|
|
||||||
| Matthias Borremans (MrBorri) | Dutch | 96 | 89 |
|
|
||||||
| Adam Syndoman (pypckompsite) | Polish | 68 | 65 |
|
|
||||||
| JonasSchubert | German | 22 | 23 |
|
|
||||||
| ThrawnJL | Danish | 7 | 7 |
|
|
||||||
| NicholasBrody | Dutch | 7 | 7 |
|
|
||||||
| Giel Janssens (gieljnssns) | Dutch | 4 | 4 |
|
|
||||||
| kentora | Danish | 3 | 2 |
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
### Docker
|
|
||||||
|
|
||||||
#### Huge thanks to [@wengtad](https://github.com/wengtad) for all his work on improving the deployment with docker.
|
|
||||||
|
|
||||||
- Optimize Docker Dev Size (Frontend: from ~1.52GB to ~429MB | API: from ~657MB to ~380MB)
|
|
||||||
- Optimize Docker Prod Size (from ~542MB to ~373MB)
|
|
||||||
- Add Gunicorn
|
|
||||||
- Add Gunicorn and Webworkers to Dockerfile #550
|
|
||||||
- Add Docs for Gunicorn
|
|
||||||
- Add PUID/PGID to Docker. Fixes Initialization scripts fail to run when not executing as root user inside container #350,
|
|
||||||
- Not able to run correctly in docker if user permissions are specified #429
|
|
||||||
- Merge Dockerfile.dev into Dockerfile (dev shared same base together with prod)
|
|
||||||
- Add Docs for PUID/PGID
|
|
||||||
- Add Docker healthcheck (for this is not necessary, I could remove if you want)
|
|
||||||
@@ -1,178 +0,0 @@
|
|||||||
## v1.0.0b - 2022-05-22
|
|
||||||
|
|
||||||
- Bump Dependencies
|
|
||||||
- Recipe Scrapers to 13.28
|
|
||||||
- Jinja2 to 3.1.2
|
|
||||||
- FastAPI to 0.78.0
|
|
||||||
|
|
||||||
- Recipe Ingredient Editor
|
|
||||||
- [#1140](https://github.com/mealie-recipes/mealie/issues/1140) - Error in processing the quantity of ingredients #1140 - UI Now prevents entering not-allowed characters in quantity field
|
|
||||||
- UI now allows no value to be set in addition to a zero (0) value.
|
|
||||||
- [#1237](https://github.com/mealie-recipes/mealie/issues/1237) - UI: Saving a 0 quantity ingredient displays 0 until the page is refreshed #1237 - UI Now properly reacts to changes in the quantity field.
|
|
||||||
|
|
||||||
- Fix Mealie v0.5.x migration issue [#1183](https://github.com/mealie-recipes/mealie/issues/1183)
|
|
||||||
- Consolidated Frontend Types thanks to [@PFischbeck](https://github.com/Fischbeck)
|
|
||||||
- Added support for SSL/No Auth Email [@nkringle](https://github.com/nkringle)
|
|
||||||
- [Implement several notifications for server actions ](https://github.com/mealie-recipes/mealie/pull/1234)[@miroito](https://github.com/Miroito)
|
|
||||||
- Fix display issue for shared recipe rendering on server [@PFischbeck](https://github.com/Fischbeck)
|
|
||||||
|
|
||||||
## v1.0.0b - 2022-05-09
|
|
||||||
|
|
||||||
- Change MIT license to AGPLv3
|
|
||||||
|
|
||||||
## v1.0.0b - 2022-05-08
|
|
||||||
|
|
||||||
- Rewrote the registration flow for new users.
|
|
||||||
- Added support for seed data at anytime through the user interface.
|
|
||||||
- Improved security for sanitizing HTML inputs for user input.
|
|
||||||
- Added support for importing keywords as tags during scraping - [@miroito](https://github.com/Miroito)
|
|
||||||
- Changed default recipe settings to "disable_amount=True" for new groups.
|
|
||||||
- Add support for merging food, and units.
|
|
||||||
- Allow tags, category, and tool creation - [@miroito](https://github.com/Miroito)
|
|
||||||
- Added additional and more comprehensive filter options for cookbooks
|
|
||||||
- Fixed bookmarklets error
|
|
||||||
|
|
||||||
## v1.0.0b - 2022-03-29
|
|
||||||
|
|
||||||
- Mealie now stores the original text from parsed ingredients, with the ability to peak at the original text from a recipe. [@miroito](https://github.com/Miroito)
|
|
||||||
- Added some management / utility functions for administrators to manage data and cleanup artifacts from the file system.
|
|
||||||
- Fix clear url action in recipe creation [#1101](https://github.com/mealie-recipes/mealie/pull/1101) [@miroito](https://github.com/Miroito)
|
|
||||||
- Add group statistics calculations and data storage measurements
|
|
||||||
- No hard limits are currently imposed on groups - though this may be implemented in the future.
|
|
||||||
|
|
||||||
## v1.0.0b - 2022-03-25
|
|
||||||
|
|
||||||
- Mealie now packages the last git commit as the build ID
|
|
||||||
- Admin section now has a "Maintenance" page where you can check some health metrics like data directory size, logs file size, and if there are some non compliant directories or images. You can also perform clean-up operations to resolve some of these issues.
|
|
||||||
- Dropped 2 dependencies and moved to using our own base model within the project
|
|
||||||
- Removed lots of dead backup code
|
|
||||||
- Recipe names will now be auto-incremented when a conflict is found. So if you're adding a recipe named "Tomato Soup" and that recipe name already exists in your database one will be created with the name "Tomato Soup (1)". Currently this is done in a loop until a suitable name is found, however it will error out after 10 attempts so it's best to find a more descriptive name for your recipe.
|
|
||||||
- Fixed broken PWA where it wouldn't render any content
|
|
||||||
- Added database connection retry loop to ensure that the database is available prior to starting
|
|
||||||
- Reorganized group routes to be more consistent with the rest of the application
|
|
||||||
|
|
||||||
## v1.0.0b Beta Release!
|
|
||||||
|
|
||||||
!!! error "Breaking Changes"
|
|
||||||
As you may have guessed, this release comes with some breaking changes. If you are/were consuming the API you will need to validate all endpoints as nearly all of them have changed.
|
|
||||||
|
|
||||||
To import your data into Mealie v1 from the most recent previous release, you'll need to export and import your data using the built in method. **Note that only your recipes will be usable in the migration**.
|
|
||||||
|
|
||||||
### ✨ What's New (What isn't?!?!)
|
|
||||||
|
|
||||||
#### 🥳 General
|
|
||||||
|
|
||||||
- Mealie will by default only be accessible to users. Future plans are to create spaces for non-users to access a specific group.
|
|
||||||
- Mealie has gone through a big redesign and has tried to standardize it's look a feel a bit more across the board.
|
|
||||||
- User/Group settings are now completely separated from the Administration page.
|
|
||||||
- All settings and configurations pages now have some sort of self-documenting help text. Additional text or descriptions are welcome from PRs
|
|
||||||
- New experimental banner for the site to give users a sense of what features are still "in development" and provide a link to a github issue that provides additional context.
|
|
||||||
- Groups now offer full multi-tenant support so you can all groups have their own set of data.
|
|
||||||
|
|
||||||
##### ⚙️ Site Settings Page
|
|
||||||
|
|
||||||
- Site Settings has been completely revamped. All site-wide settings at defined on the server as ENV variables. The site settings page now only shows you the non-secret values for reference. It also has some helpers to let you know if something isn't configured correctly.
|
|
||||||
- Server Side Bare URL will let you know if the BASE_URL env variable has been set
|
|
||||||
- Secure Site let's you know if you're serving via HTTPS or accessing by localhost. Accessing without a secure site will render some of the features unusable.
|
|
||||||
- Email Configuration Status will let you know if all the email settings have been provided and offer a way to send test emails.
|
|
||||||
|
|
||||||
#### 👨👩👧👦 Users and Groups
|
|
||||||
|
|
||||||
- All members of a group can generate invitation tokens for other users to join their group
|
|
||||||
- Users now a have "Advanced" setting to enable/disable features like Webhooks and API tokens. This will also apply to future features that are deemed as advanced.
|
|
||||||
- "Pages" have been dropped in favor of Cookbooks which are now group specific so each group can have it's own set of cookbooks
|
|
||||||
- Default recipe settings can now be set by the group instead of environmental variables.
|
|
||||||
- Password reset via email
|
|
||||||
- Invitation to group by email
|
|
||||||
- Manage group member permissions
|
|
||||||
|
|
||||||
#### 📦 Data Migrations
|
|
||||||
|
|
||||||
- Migrations have been moved from the Admin Page to a Group Migration page. Migrations from applications (or previous versions of Mealie) can now be imported into Mealie via the Group Migration pages where all recipes will be imported into the group.
|
|
||||||
- **Supported Migrations**
|
|
||||||
- Mealie Pre v1.0.0
|
|
||||||
- Nextcloud Recipes
|
|
||||||
- Chowdown
|
|
||||||
|
|
||||||
#### 🛒 Shopping Lists
|
|
||||||
|
|
||||||
- Shopping Lists has been completely revamped to be more flexible and user friendly.
|
|
||||||
- Add recipe ingredients to a shopping list
|
|
||||||
- Manually add item/ingredient to shopping list
|
|
||||||
- Copy as markdown or plain text
|
|
||||||
- Sort by food/item Labels
|
|
||||||
- Checked items are now hidden
|
|
||||||
- Uncheck all Items
|
|
||||||
- Delete all checked items
|
|
||||||
|
|
||||||
#### 📢 Apprise Integration
|
|
||||||
|
|
||||||
- Server based Apprise notifications have been deprecated. An effort has been made to improve logging overall in the application and make it easier to monitor/debug through the logs.
|
|
||||||
- The Apprise integration has been updated to the latest version and is now used asynchronously.
|
|
||||||
- Notifiers now support a wider variety of events.
|
|
||||||
- Notifiers can now be managed by-group instead of by the server.
|
|
||||||
|
|
||||||
#### 🗓 Meal Plans
|
|
||||||
|
|
||||||
- Meal plans have been completely redesigned to use a calendar approach so you'll be able to see what meals you have planned in a more traditional view
|
|
||||||
- Drag and Drop meals between days
|
|
||||||
- Add Recipes or Notes to a specific day
|
|
||||||
- New context menu action for recipes to add a recipe to a specific day on the meal-plan
|
|
||||||
- New rule based meal plan generator/selector. You can now create rules to restrict the addition of recipes for specific days or meal types (breakfast, lunch, dinner, side). You can also create rules that match against "all" days or "all" meal types to create global rules based around tags and categories. This gives you the most flexibility in creating meal plans.
|
|
||||||
|
|
||||||
#### 🥙 Recipes
|
|
||||||
|
|
||||||
##### 🔍 Search
|
|
||||||
|
|
||||||
- Search now includes the ability to fuzzy search ingredients
|
|
||||||
- Search now includes an additional filter for "Foods" which will filter (Include/Exclude) matches based on your selection.
|
|
||||||
|
|
||||||
##### 🍴 Recipe General
|
|
||||||
|
|
||||||
- Recipe Pages now implement a screen lock on supported devices to keep the screen from going to sleep.
|
|
||||||
- Recipes are now only viewable by group members
|
|
||||||
- Recipes can be shared with share links
|
|
||||||
- Shared recipes can now render previews for the recipe on sites like Twitter, Facebook, and Discord.
|
|
||||||
- Recipes now have a `tools` attribute that contains a list of required tools/equipment for the recipe. Tools can be set with a state to determine if you have that tool or not. If it's marked as on hand it will show checked by default.
|
|
||||||
- Recipe Extras now only show when advanced mode is toggled on
|
|
||||||
- You can now import multiple URLs at a time pre-tagged using the bulk importer. This task runs in the background so no need to wait for it to finish.
|
|
||||||
- Foods/Units for Ingredients are now supported (toggle inside your recipe settings)
|
|
||||||
- Common Food and Units come pre-packaged with Mealie
|
|
||||||
- Landscape and Portrait views are now available
|
|
||||||
- Users with the advanced flag turned on will now be able to manage recipe data in bulk and perform the following actions:
|
|
||||||
- Set Categories
|
|
||||||
- Set Tags
|
|
||||||
- Delete Recipes
|
|
||||||
- Export Recipes
|
|
||||||
- Recipes now have a `/cook` page for a simple view of the recipe where you can click through each step of a recipe and it's associated ingredients.
|
|
||||||
- The Bulk Importer has received many additional upgrades.
|
|
||||||
- Trim Whitespace: automatically removes leading and trailing whitespace
|
|
||||||
- Trim Prefix: Removes the first character of each line. Useful for when you paste in a list of ingredients or instructions that have 1. or 2. in front of them.
|
|
||||||
- Split By Numbered Line: Attempts to split a paragraph into multiple lines based on the patterns matching '1.', '1:' or '1)'.
|
|
||||||
|
|
||||||
##### 🍞 Recipe Ingredients
|
|
||||||
|
|
||||||
- Recipe ingredients can now be scaled when the food/unit is defined
|
|
||||||
- Recipe ingredients can now be copied as markdown lists
|
|
||||||
- example `- [ ] 1 cup of flour`
|
|
||||||
- You can now use Natural Language Processing (NLP) to process ingredients and attempt to parse them into amounts, units, and foods. There is an additional "Brute Force" processor that can be used as pattern matching parser to try and determine ingredients. **Note** if you are processing a Non-English language you will have terrible results with the NLP and will likely need to use the Bruce Force processor.
|
|
||||||
|
|
||||||
##### 📜 Recipe Instructions
|
|
||||||
|
|
||||||
- Can now be merged with the above step automatically through the action menu
|
|
||||||
- Recipe Ingredients can be linked directly to recipe instructions for improved display
|
|
||||||
- There is an option in the linking dialog to automatically link ingredients. This works by using a key-word matching algorithm to find the ingredients. It's not perfect so you'll need to verify the links after use, additionally you will find that it doesn't work for non-english languages.
|
|
||||||
- Recipe Instructions now have a preview tab to show the rendered markdown before saving.
|
|
||||||
|
|
||||||
#### ⚠️ Other things to know...
|
|
||||||
|
|
||||||
- Themes have been deprecated for specific users. You can still set specific themes for your site through ENV variables. This approach should yield much better results for performance and some weirdness users have experienced.
|
|
||||||
- If you've experienced slowness in the past, you may notice a significant improvement in the "All Recipes" and "Search" pages, or wherever large payloads of recipes are being displayed. This is due to not validating responses from the database, as such if you are consuming these API's you may get extra values that are unexpected.
|
|
||||||
|
|
||||||
#### 👨💻 Backend/Development Goodies
|
|
||||||
|
|
||||||
- Codebase is significantly more organized both Frontend and Backend
|
|
||||||
- We've moved to Nuxt for SSR and Typescript for better type safety and less bugs 🎉
|
|
||||||
- Backend now using a Class based architecture to maximize code reuse
|
|
||||||
- Tons of performance improvements across the board
|
|
||||||
- Significant work was done by [@PFischbeck](https://github.com/PFischbeck) to improve type safety on the frontend server and fix many type related errors/bugs!
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
### Bug Fixes
|
|
||||||
|
|
||||||
- Bump isomorphic-dompurify from 0.18.0 to 0.19.0 in /frontend ([#1257](https://github.com/mealie-recipes/mealie/issues/1257))
|
|
||||||
- Bump @nuxtjs/auth-next in /frontend ([#1265](https://github.com/mealie-recipes/mealie/issues/1265))
|
|
||||||
- Bad dev dependency ([#1281](https://github.com/mealie-recipes/mealie/issues/1281))
|
|
||||||
- Add touch support for mealplanner delete ([#1298](https://github.com/mealie-recipes/mealie/issues/1298))
|
|
||||||
|
|
||||||
### Documentation
|
|
||||||
|
|
||||||
- Add references for VSCode dev containers ([#1299](https://github.com/mealie-recipes/mealie/issues/1299))
|
|
||||||
- Docker-compose.dev.yml is currently not functional ([#1300](https://github.com/mealie-recipes/mealie/issues/1300))
|
|
||||||
|
|
||||||
### Features
|
|
||||||
|
|
||||||
- Add reports to bulk recipe import (url) ([#1294](https://github.com/mealie-recipes/mealie/issues/1294))
|
|
||||||
- Rewrite print implementation to support new ing ([#1305](https://github.com/mealie-recipes/mealie/issues/1305))
|
|
||||||
|
|
||||||
### Miscellaneous Tasks
|
|
||||||
|
|
||||||
- Github stalebot changes ([#1271](https://github.com/mealie-recipes/mealie/issues/1271))
|
|
||||||
- Bump eslint-plugin-nuxt in /frontend ([#1258](https://github.com/mealie-recipes/mealie/issues/1258))
|
|
||||||
- Bump @vue/runtime-dom in /frontend ([#1259](https://github.com/mealie-recipes/mealie/issues/1259))
|
|
||||||
- Bump nuxt-vite from 0.1.3 to 0.3.5 in /frontend ([#1260](https://github.com/mealie-recipes/mealie/issues/1260))
|
|
||||||
- Bump vue2-script-setup-transform in /frontend ([#1263](https://github.com/mealie-recipes/mealie/issues/1263))
|
|
||||||
- Update dev dependencies ([#1282](https://github.com/mealie-recipes/mealie/issues/1282))
|
|
||||||
|
|
||||||
### Refactor
|
|
||||||
|
|
||||||
- Split up recipe create page ([#1283](https://github.com/mealie-recipes/mealie/issues/1283))
|
|
||||||
@@ -1,36 +0,0 @@
|
|||||||
### Bug Fixes
|
|
||||||
|
|
||||||
- Update issue links in v1.0.0beta-2 changelog ([#1312](https://github.com/mealie-recipes/mealie/issues/1312))
|
|
||||||
- Bad import path ([#1313](https://github.com/mealie-recipes/mealie/issues/1313))
|
|
||||||
- Printer page refs ([#1314](https://github.com/mealie-recipes/mealie/issues/1314))
|
|
||||||
- Consolidate stores to fix mismatched state
|
|
||||||
- Bump @vue/composition-api from 1.6.1 to 1.6.2 in /frontend ([#1275](https://github.com/mealie-recipes/mealie/issues/1275))
|
|
||||||
- Shopping list label editor ([#1333](https://github.com/mealie-recipes/mealie/issues/1333))
|
|
||||||
|
|
||||||
### Features
|
|
||||||
|
|
||||||
- Default unit fractions to True
|
|
||||||
- Add unit abbreviation support ([#1332](https://github.com/mealie-recipes/mealie/issues/1332))
|
|
||||||
- Attached images by drag and drop for recipe steps ([#1341](https://github.com/mealie-recipes/mealie/issues/1341))
|
|
||||||
|
|
||||||
### Docs
|
|
||||||
|
|
||||||
- Render homepage social media link images at 32x32 size ([#1310](https://github.com/mealie-recipes/mealie/issues/1310))
|
|
||||||
|
|
||||||
### Miscellaneous Tasks
|
|
||||||
|
|
||||||
- Init git-cliff config
|
|
||||||
- Bump @types/sortablejs in /frontend ([#1287](https://github.com/mealie-recipes/mealie/issues/1287))
|
|
||||||
- Bump @babel/eslint-parser in /frontend ([#1290](https://github.com/mealie-recipes/mealie/issues/1290))
|
|
||||||
|
|
||||||
### Refactor
|
|
||||||
|
|
||||||
- Unify recipe-organizer components ([#1340](https://github.com/mealie-recipes/mealie/issues/1340))
|
|
||||||
|
|
||||||
### Security
|
|
||||||
|
|
||||||
- Delay server response whenever username is non existing ([#1338](https://github.com/mealie-recipes/mealie/issues/1338))
|
|
||||||
|
|
||||||
### Wip
|
|
||||||
|
|
||||||
- Pagination-repository ([#1316](https://github.com/mealie-recipes/mealie/issues/1316))
|
|
||||||
@@ -1,126 +0,0 @@
|
|||||||
### Security
|
|
||||||
|
|
||||||
#### v1.0.0beta-3 and Under - Recipe Scraper: Server Side Request Forgery Lead To Denial Of Service
|
|
||||||
|
|
||||||
!!! error "CWE-918: Server-Side Request Forgery (SSRF)"
|
|
||||||
In this case if a attacker try to load a huge file then server will try to load the file and eventually server use its all memory which will dos the server
|
|
||||||
|
|
||||||
##### Mitigation
|
|
||||||
|
|
||||||
HTML is now scraped via a Stream and canceled after a 15 second timeout to prevent arbitrary data from being loaded into the server.
|
|
||||||
|
|
||||||
#### v1.0.0beta-3 and Under - Recipe Assets: Remote Code Execution
|
|
||||||
|
|
||||||
!!! error "CWE-1336: Improper Neutralization of Special Elements Used in a Template Engine"
|
|
||||||
As a low privileged user, Create a new recipe and click on the "+" to add a New Asset.
|
|
||||||
Select a file, then proxy the request that will create the asset.
|
|
||||||
|
|
||||||
Since mealie/routes/recipe/recipe_crud_routes.py:306 is calling slugify on the name POST parameter, we use $ which slugify() will remove completely.
|
|
||||||
|
|
||||||
Since mealie/routes/recipe/recipe_crud_routes.py:306 is concatenating raw user input from the extension POST parameter into the variable file_name, which ultimately gets used when writing to disk, we can use a directory traversal attack in the extension (e.g. ./../../../tmp/pwn.txt) to write the file to arbitrary location on the server.
|
|
||||||
|
|
||||||
As an attacker, now that we have a strong attack primitive, we can start getting creative to get RCE. Since the files were being created by root, we could add an entry to /etc/passwd, create a crontab, etc. but since there was templating functionality in the application that peaked my interest. The PoC in the HTTP request above creates a Jinja2 template at /app/data/template/pwn.html. Since Jinja2 templates execute Python code when rendered, all we have to do now to get code execution is render the malicious template. This was easy enough.
|
|
||||||
|
|
||||||
##### Mitigation
|
|
||||||
|
|
||||||
We've added proper path sanitization to ensure that the user is not allowed to write to arbitrary locations on the server.
|
|
||||||
|
|
||||||
!!! warning "Breaking Change Incoming"
|
|
||||||
As this has shown a significant area of exposure in the templates that Mealie was provided for exporting recipes, we'll be removing this feature in the next Beta release and will instead rely on the community to provide tooling around transforming recipes using templates. This will significantly limit the possible exposure of users injecting malicious templates into the application. The template functionality will be completely removed in the next beta release v1.0.0beta-5
|
|
||||||
|
|
||||||
#### All version Markdown Editor: Cross Site Scripting
|
|
||||||
|
|
||||||
!!! error "CWE-79: Cross-site Scripting (XSS) - Stored"
|
|
||||||
A low privilege user can insert malicious JavaScript code into the Recipe Instructions which will execute in another person's browser that visits the recipe.
|
|
||||||
|
|
||||||
`<img src=x onerror=alert(document.domain)>`
|
|
||||||
|
|
||||||
##### Mitigation
|
|
||||||
|
|
||||||
This issues is present on all pages that allow markdown input. This error has been mitigated by wrapping the 3rd Party Markdown component and using the `domPurify` library to strip out the dangerous HTML.
|
|
||||||
|
|
||||||
#### v1.0.0beta-3 and Under - Image Scraper: Server-Side Request Forgery
|
|
||||||
|
|
||||||
!!! error "CWE-918: Server-Side Request Forgery (SSRF)"
|
|
||||||
In the recipe edit page, is possible to upload an image directly or via an URL provided by the user. The function that handles the fetching and saving of the image via the URL doesn't have any URL verification, which allows to fetch internal services.
|
|
||||||
|
|
||||||
Furthermore, after the resource is fetch, there is no MIME type validation, which would ensure that the resource is indeed an image. After this, because there is no extension in the provided URL, the application will fallback to jpg, and original for the image name.
|
|
||||||
|
|
||||||
Then the result is saved to disk with the original.jpg name, that can be retrieved from the following URL: http://<domain>/api/media/recipes/<recipe-uid>/images/original.jpg. This file will contain the full response of the provided URL.
|
|
||||||
|
|
||||||
**Impact**
|
|
||||||
|
|
||||||
An attacker can get sensitive information of any internal-only services running. For example, if the application is hosted on Amazon Web Services (AWS) platform, its possible to fetch the AWS API endpoint, https://169.254.169.254, which returns API keys and other sensitive metadata.
|
|
||||||
|
|
||||||
##### Mitigation
|
|
||||||
|
|
||||||
Two actions were taken to reduce exposure to SSRF in this case.
|
|
||||||
|
|
||||||
1. The application will not prevent requests being made to local resources by checking for localhost or 127.0.0.1 domain names.
|
|
||||||
2. The mime-type of the response is now checked prior to writing to disk.
|
|
||||||
|
|
||||||
If either of the above actions prevent the user from uploading images, the application will alert the user of what error occurred.
|
|
||||||
|
|
||||||
### Bug Fixes
|
|
||||||
|
|
||||||
- For erroneously-translated datetime config ([#1362](https://github.com/mealie-recipes/mealie/issues/1362))
|
|
||||||
- Fixed text color on RecipeCard in RecipePrintView and implemented ingredient sections ([#1351](https://github.com/mealie-recipes/mealie/issues/1351))
|
|
||||||
- Ingredient sections lost after parsing ([#1368](https://github.com/mealie-recipes/mealie/issues/1368))
|
|
||||||
- Increased float rounding precision for CRF parser ([#1369](https://github.com/mealie-recipes/mealie/issues/1369))
|
|
||||||
- Infinite scroll bug on all recipes page ([#1393](https://github.com/mealie-recipes/mealie/issues/1393))
|
|
||||||
- Fast fail of bulk importer ([#1394](https://github.com/mealie-recipes/mealie/issues/1394))
|
|
||||||
- Bump @mdi/js from 5.9.55 to 6.7.96 in /frontend ([#1279](https://github.com/mealie-recipes/mealie/issues/1279))
|
|
||||||
- Bump @nuxtjs/i18n from 7.0.3 to 7.2.2 in /frontend ([#1288](https://github.com/mealie-recipes/mealie/issues/1288))
|
|
||||||
- Bump date-fns from 2.23.0 to 2.28.0 in /frontend ([#1293](https://github.com/mealie-recipes/mealie/issues/1293))
|
|
||||||
- Bump fuse.js from 6.5.3 to 6.6.2 in /frontend ([#1325](https://github.com/mealie-recipes/mealie/issues/1325))
|
|
||||||
- Bump core-js from 3.17.2 to 3.23.1 in /frontend ([#1383](https://github.com/mealie-recipes/mealie/issues/1383))
|
|
||||||
- All-recipes page now sorts alphabetically ([#1405](https://github.com/mealie-recipes/mealie/issues/1405))
|
|
||||||
- Sort recent recipes by created_at instead of date_added ([#1417](https://github.com/mealie-recipes/mealie/issues/1417))
|
|
||||||
- Only show scaler when ingredients amounts enabled ([#1426](https://github.com/mealie-recipes/mealie/issues/1426))
|
|
||||||
- Add missing types for API token deletion ([#1428](https://github.com/mealie-recipes/mealie/issues/1428))
|
|
||||||
- Entry nutrition checker ([#1448](https://github.com/mealie-recipes/mealie/issues/1448))
|
|
||||||
- Use == operator instead of is_ for sql queries ([#1453](https://github.com/mealie-recipes/mealie/issues/1453))
|
|
||||||
- Use `mtime` instead of `ctime` for backup dates ([#1461](https://github.com/mealie-recipes/mealie/issues/1461))
|
|
||||||
- Mealplan pagination ([#1464](https://github.com/mealie-recipes/mealie/issues/1464))
|
|
||||||
- Properly use pagination for group event notifies ([#1512](https://github.com/mealie-recipes/mealie/pull/1512))
|
|
||||||
|
|
||||||
### Documentation
|
|
||||||
|
|
||||||
- Add go bulk import example ([#1388](https://github.com/mealie-recipes/mealie/issues/1388))
|
|
||||||
- Fix old link
|
|
||||||
- Pagination and filtering, and fixed a few broken links ([#1488](https://github.com/mealie-recipes/mealie/issues/1488))
|
|
||||||
|
|
||||||
### Features
|
|
||||||
|
|
||||||
- Toggle display of ingredient references in recipe instructions ([#1268](https://github.com/mealie-recipes/mealie/issues/1268))
|
|
||||||
- Add custom scaling option ([#1345](https://github.com/mealie-recipes/mealie/issues/1345))
|
|
||||||
- Implemented "order by" API parameters for recipe, food, and unit queries ([#1356](https://github.com/mealie-recipes/mealie/issues/1356))
|
|
||||||
- Implement user favorites page ([#1376](https://github.com/mealie-recipes/mealie/issues/1376))
|
|
||||||
- Extend Apprise JSON notification functionality with programmatic data ([#1355](https://github.com/mealie-recipes/mealie/issues/1355))
|
|
||||||
- Mealplan-webhooks ([#1403](https://github.com/mealie-recipes/mealie/issues/1403))
|
|
||||||
- Added "last-modified" header to supported record types ([#1379](https://github.com/mealie-recipes/mealie/issues/1379))
|
|
||||||
- Re-write get all routes to use pagination ([#1424](https://github.com/mealie-recipes/mealie/issues/1424))
|
|
||||||
- Advanced filtering API ([#1468](https://github.com/mealie-recipes/mealie/issues/1468))
|
|
||||||
- Restore frontend sorting for all recipes ([#1497](https://github.com/mealie-recipes/mealie/issues/1497))
|
|
||||||
- Implemented local storage for sorting and dynamic sort icons on the new recipe sort card ([1506](https://github.com/mealie-recipes/mealie/pull/1506))
|
|
||||||
- create new foods and units from their Data Management pages ([#1511](https://github.com/mealie-recipes/mealie/pull/1511))
|
|
||||||
|
|
||||||
### Miscellaneous Tasks
|
|
||||||
|
|
||||||
- Bump dev deps ([#1418](https://github.com/mealie-recipes/mealie/issues/1418))
|
|
||||||
- Bump @vue/runtime-dom in /frontend ([#1423](https://github.com/mealie-recipes/mealie/issues/1423))
|
|
||||||
- Backend page_all route cleanup ([#1483](https://github.com/mealie-recipes/mealie/issues/1483))
|
|
||||||
|
|
||||||
### Refactor
|
|
||||||
|
|
||||||
- Remove depreciated repo call ([#1370](https://github.com/mealie-recipes/mealie/issues/1370))
|
|
||||||
|
|
||||||
### Hotfix
|
|
||||||
|
|
||||||
- Tame typescript beast
|
|
||||||
|
|
||||||
### UI
|
|
||||||
|
|
||||||
- Improve parser ui text display ([#1437](https://github.com/mealie-recipes/mealie/issues/1437))
|
|
||||||
|
|
||||||
<!-- generated by git-cliff -->
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
### NOTICE:
|
|
||||||
|
|
||||||
Release changelogs are now published on github releases. This file is kept for historical purposes.
|
|
||||||
@@ -2,11 +2,11 @@
|
|||||||
|
|
||||||
[Please Join the Discord](https://discord.gg/QuStdQGSGK). We are building a community of developers working on the project.
|
[Please Join the Discord](https://discord.gg/QuStdQGSGK). We are building a community of developers working on the project.
|
||||||
|
|
||||||
## We Develop with Github
|
## We Develop with GitHub
|
||||||
We use github to host code, to track issues and feature requests, as well as accept pull requests.
|
We use GitHub to host code, to track issues and feature requests, as well as accept pull requests.
|
||||||
|
|
||||||
## We Use [Github Flow](https://docs.github.com/en/get-started/using-github/github-flow), So All Code Changes Happen Through Pull Requests
|
## We Use [GitHub Flow](https://docs.github.com/en/get-started/using-github/github-flow), So All Code Changes Happen Through Pull Requests
|
||||||
Pull requests are the best way to propose changes to the codebase (we use [Github Flow](https://docs.github.com/en/get-started/using-github/github-flow)). We actively welcome your pull requests:
|
Pull requests are the best way to propose changes to the codebase (we use [GitHub Flow](https://docs.github.com/en/get-started/using-github/github-flow)). We actively welcome your pull requests:
|
||||||
|
|
||||||
1. Fork the repo and create your branch from `mealie-next`.
|
1. Fork the repo and create your branch from `mealie-next`.
|
||||||
2. Checkout the Discord, the PRs page, or the Projects page to get an idea of what's already being worked on.
|
2. Checkout the Discord, the PRs page, or the Projects page to get an idea of what's already being worked on.
|
||||||
@@ -14,13 +14,13 @@ Pull requests are the best way to propose changes to the codebase (we use [Githu
|
|||||||
4. Once you've got an idea of what changes you want to make, create a draft PR as soon as you can to let us know what you're working on and how we can help!
|
4. Once you've got an idea of what changes you want to make, create a draft PR as soon as you can to let us know what you're working on and how we can help!
|
||||||
5. If you've changed APIs, update the documentation.
|
5. If you've changed APIs, update the documentation.
|
||||||
6. Run tests, including `task py:check`.
|
6. Run tests, including `task py:check`.
|
||||||
6. Issue that pull request! First make a draft PR, make sure that the automated github tests all pass, then mark as ready for review.
|
7. Issue that pull request! First make a draft PR, make sure that the automated GitHub tests all pass, then mark as ready for review. We follow Conventional Commits syntax; please title your PR as described in the PR template.
|
||||||
7. Be sure to add release notes to the pull request.
|
8. Be sure to add release notes to the pull request.
|
||||||
|
|
||||||
## Any contributions you make will be under the AGPL Software License
|
## Any contributions you make will be under the AGPL Software License
|
||||||
In short, when you submit code changes, your submissions are understood to be under the same [AGPL License](https://choosealicense.com/licenses/agpl-3.0/) that covers the project. Feel free to contact the maintainers if that's a concern.
|
In short, when you submit code changes, your submissions are understood to be under the same [AGPL License](https://choosealicense.com/licenses/agpl-3.0/) that covers the project. Feel free to contact the maintainers if that's a concern.
|
||||||
|
|
||||||
## Report bugs using Github's [issues](https://github.com/mealie-recipes/mealie/issues)
|
## Report bugs using GitHub's [issues](https://github.com/mealie-recipes/mealie/issues)
|
||||||
We use GitHub issues to track public bugs. Report a bug by [opening a new issue](https://github.com/mealie-recipes/mealie/issues/new); it's that easy!
|
We use GitHub issues to track public bugs. Report a bug by [opening a new issue](https://github.com/mealie-recipes/mealie/issues/new); it's that easy!
|
||||||
|
|
||||||
## Write bug reports with detail, background, and sample code
|
## Write bug reports with detail, background, and sample code
|
||||||
|
|||||||
85
docs/docs/contributors/developers-guide/database-changes.md
Normal file
85
docs/docs/contributors/developers-guide/database-changes.md
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
# Development: Database Changes
|
||||||
|
|
||||||
|
This document is open to improvement; please share any insights you have/develop.
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
|
||||||
|
When modifying the database, you will most likely need to change the files under `/mealie/db/models/`.
|
||||||
|
How exactly you need to modify it is of course highly contextual to the change you're making.
|
||||||
|
|
||||||
|
## Using Alembic to generate upgrade script
|
||||||
|
|
||||||
|
In your dev container you can run something like (change the message) `task py:migrate -- "Add creation tag to group preferences"` to have Alembic generate an upgrade script for you.
|
||||||
|
|
||||||
|
The script Alembic generates, will be limited! (Perhaps there's a way to resolve that? Haven't looked into it yet)
|
||||||
|
For example, Alembic generated a script _similar_ to this (it has been modified already to have accurate foreign key names, for instance):
|
||||||
|
|
||||||
|
```Python
|
||||||
|
"""Add creation tag to group preferences
|
||||||
|
|
||||||
|
Revision ID: 0ea6eb8eaa44
|
||||||
|
Revises: ba1e4a6cfe99
|
||||||
|
Create Date: 2024-01-04 12:40:03.062671
|
||||||
|
|
||||||
|
"""
|
||||||
|
import sqlalchemy as sa
|
||||||
|
|
||||||
|
import mealie.db.migration_types
|
||||||
|
from alembic import op
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = "0ea6eb8eaa44"
|
||||||
|
down_revision = "ba1e4a6cfe99"
|
||||||
|
branch_labels = None
|
||||||
|
depends_on = None
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade():
|
||||||
|
### commands auto generated by Alembic - please adjust! ###
|
||||||
|
op.add_column(
|
||||||
|
"group_preferences", sa.Column("recipe_creation_tag", mealie.db.migration_types.GUID(), nullable=True)
|
||||||
|
)
|
||||||
|
op.create_foreign_key("fk_groupprefs_tags", "group_preferences", "tags", ["recipe_creation_tag"], ["id"])
|
||||||
|
### end Alembic commands ###
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade():
|
||||||
|
### commands auto generated by Alembic - please adjust! ###
|
||||||
|
op.drop_constraint("fk_groupprefs_tags", "group_preferences", type_="foreignkey")
|
||||||
|
op.drop_column("group_preferences", "recipe_creation_tag")
|
||||||
|
### end Alembic commands ###
|
||||||
|
```
|
||||||
|
|
||||||
|
But when trying to actually use that upgrade script, it becomes clear that our SQLite database doesn't like them. The minor modification needed looks like:
|
||||||
|
|
||||||
|
```Python
|
||||||
|
"""Add creation tag to group preferences
|
||||||
|
|
||||||
|
Revision ID: 0ea6eb8eaa44
|
||||||
|
Revises: ba1e4a6cfe99
|
||||||
|
Create Date: 2024-01-04 12:40:03.062671
|
||||||
|
|
||||||
|
"""
|
||||||
|
import sqlalchemy as sa
|
||||||
|
|
||||||
|
import mealie.db.migration_types
|
||||||
|
from alembic import op
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = "0ea6eb8eaa44"
|
||||||
|
down_revision = "ba1e4a6cfe99"
|
||||||
|
branch_labels = None
|
||||||
|
depends_on = None
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade():
|
||||||
|
with op.batch_alter_table("group_preferences", schema=None) as batch_op:
|
||||||
|
batch_op.add_column(sa.Column("recipe_creation_tag", mealie.db.migration_types.GUID(), nullable=True))
|
||||||
|
batch_op.create_foreign_key("fk_groupprefs_tags", "tags", ["recipe_creation_tag"], ["id"])
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade():
|
||||||
|
with op.batch_alter_table("group_preferences", schema=None) as batch_op:
|
||||||
|
batch_op.drop_constraint("fk_groupprefs_tags", type_="foreignkey")
|
||||||
|
batch_op.drop_column("recipe_creation_tag")
|
||||||
|
```
|
||||||
@@ -17,4 +17,3 @@ Alternatively, you can register a new parser by fulfilling the `ABCIngredientPar
|
|||||||
## Links
|
## Links
|
||||||
- [Pretrained Model](https://github.com/mealie-recipes/mealie-nlp-model)
|
- [Pretrained Model](https://github.com/mealie-recipes/mealie-nlp-model)
|
||||||
- [CRF++ (Forked)](https://github.com/hay-kot/crfpp)
|
- [CRF++ (Forked)](https://github.com/hay-kot/crfpp)
|
||||||
|
|
||||||
|
|||||||
@@ -93,4 +93,3 @@ mealie_url="http://localhost:9000"
|
|||||||
token = authentication(mail, password, mealie_url)
|
token = authentication(mail, password, mealie_url)
|
||||||
import_from_file(input_file, token, mealie_url)
|
import_from_file(input_file, token, mealie_url)
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,80 @@
|
|||||||
|
# Automating Backups with n8n
|
||||||
|
|
||||||
|
!!! info
|
||||||
|
This guide was submitted by a community member. Find something wrong? Submit a PR to get it fixed!
|
||||||
|
|
||||||
|
> [n8n](https://github.com/n8n-io/n8n) is a free and source-available fair-code licensed workflow automation tool. Alternative to Zapier or Make, allowing you to use a UI to create automated workflows.
|
||||||
|
|
||||||
|
This example workflow:
|
||||||
|
|
||||||
|
1. Backups Mealie every morning via an API call
|
||||||
|
2. Deletes all but the last 7 backups
|
||||||
|
|
||||||
|
> [!CAUTION]
|
||||||
|
> This only automates the backup function, this does not backup your data to anywhere except your local instance. Please make sure you are backing up your data to an external source.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
# Setup
|
||||||
|
|
||||||
|
## Deploying n8n
|
||||||
|
|
||||||
|
Follow the relevant guide in the [n8n Documentation](https://docs.n8n.io/)
|
||||||
|
|
||||||
|
## Importing n8n workflow
|
||||||
|
|
||||||
|
1. In n8n, add a new workflow
|
||||||
|
2. In the top right hit the 3 dot menu and select 'Import from URL...'
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
3. Paste `https://github.com/mealie-recipes/mealie/blob/mealie-next/docs/docs/assets/other/n8n/n8n-mealie-backup.json` and click Import
|
||||||
|
4. Click through the nodes and update the URLs for your environment
|
||||||
|
|
||||||
|
## API Credentials
|
||||||
|
|
||||||
|
#### Generate Mealie API Token
|
||||||
|
|
||||||
|
1. Head to https://mealie.example.com/user/profile/api-tokens
|
||||||
|
> If you dont see this screen make sure that "Show advanced features" is checked under https://mealie.example.com/user/profile/edit
|
||||||
|
2. Under token name, enter the name of the token i.e. 'n8n' and hit Generate
|
||||||
|
3. Copy and keep this API Token somewhere safe, this is like your password!
|
||||||
|
|
||||||
|
> You can use your normal user for this, but assuming you're an admin you could also choose to create a user named n8n and generate the API key against that user.
|
||||||
|
|
||||||
|
#### Setup Credentials in n8n
|
||||||
|
|
||||||
|
> [n8n Docs](https://docs.n8n.io/credentials/add-edit-credentials/)
|
||||||
|
|
||||||
|
1. Create a new "Header Auth" Credential
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
2. In the connection screen set - Name as `Authorization` - Value as `Bearer {INSERT MEALIE API KEY}`
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
3. In the workflow you created, for the "Run Backup", "Get All backups", and "Delete Oldies" nodes, update:
|
||||||
|
- Authentication to `Generic Credential Type`
|
||||||
|
- Generic Auth Type to `Header Auth`
|
||||||
|
- Header Auth to `Mealie API` or whatever you named your credentials
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## Notification Node
|
||||||
|
|
||||||
|
> Please use error notifications of some kind. It's very easy to set and forget an automation, then have the worst happen and lose data.
|
||||||
|
|
||||||
|
[ntfy](https://github.com/binwiederhier/ntfy) is a great open source, self-hostable tool for sending notifications.
|
||||||
|
|
||||||
|
If you want to use ntfy, you will need to install it on your environment, or sign up for their service, and configure it with the webhook URL.
|
||||||
|
|
||||||
|
If you want to use another notification service, you can create a new node in n8n that sends the notification using whatever method you like.
|
||||||
|
|
||||||
|
- For example, if you want to send a push notification via [Pushover](https:/pushover.net/) you could create a new node that uses the Pushover API and sends the notification.
|
||||||
|
- You can use the [Send Email](https://docs.n8n.io/integrations/builtincore-nodes/n8n-nodes-base.sendemail/) node in n8n as an example of how to create your own custom node.
|
||||||
|
- You can send it off to InfluxDB, Slack, Discord etc. Go nuts.
|
||||||
|
|
||||||
|
If you're using another method for backups we'd love to hear about it. Pop in [Discord](https://discord.gg/QuStdQGSGK) and say hi!
|
||||||
@@ -48,7 +48,7 @@ services:
|
|||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
```
|
```
|
||||||
|
|
||||||
Don't forget to change the <code>mydomain.duckns</code> into your personal domain and the <code>duckdnstoken</code> into your token and remove the brackets.
|
Don't forget to change the <code>mydomain.duckdns</code> into your personal domain and the <code>duckdnstoken</code> into your token and remove the brackets.
|
||||||
|
|
||||||
## Step 3: Change the config files
|
## Step 3: Change the config files
|
||||||
|
|
||||||
|
|||||||
@@ -2,14 +2,14 @@
|
|||||||
|
|
||||||
## How do I enable "smart" ingredient handling?
|
## How do I enable "smart" ingredient handling?
|
||||||
|
|
||||||
You might have noticed that scaling up a recipe or making a shopping list doesn't by default handle the ingredients in a way you might expect. Depending on your settings, scaling up might yield things like `2 1 cup broth` instead of `2 cup broth`. And making shopping lists from reciepes that have shared ingredients can yield multiple lines of the same ingredient. **But**, mealie has a mechanism to intelligently handle ingredients and make your day better. How?
|
You might have noticed that scaling up a recipe or making a shopping list doesn't by default handle the ingredients in a way you might expect. Depending on your settings, scaling up might yield things like `2 1 cup broth` instead of `2 cup broth`. And, making shopping lists from recipes that have shared ingredients can yield multiple lines of the same ingredient. **But**, Mealie has a mechanism to intelligently handle ingredients and make your day better. How?
|
||||||
### Set up your Foods and Units
|
### Set up your Foods and Units
|
||||||
Do the following just **once**. Doing this applies to your whole group, so be careful.
|
Do the following just **once**. Doing this applies to your whole group, so be careful.
|
||||||
|
|
||||||
1. Click on your name in the upper left corner to get to your settings
|
1. Click on your name in the upper left corner to get to your settings
|
||||||
2. In the bottom right, select `Manage Data`
|
2. In the bottom right, select `Manage Data`
|
||||||
3. In the Management page, make sure that a little orange button says `Foods`
|
3. In the Management page, make sure that a little orange button says `Foods`
|
||||||
4. If your Foods database is empty, click `Seed` and choose your language. You should end up with a list of foods. (Wait bit for seeding to happen, and try not to seed more than once or you will have duplicates)
|
4. If your Foods database is empty, click `Seed` and choose your language. You should end up with a list of foods. (Wait a bit for seeding to happen, and try not to seed more than once or you will have duplicates)
|
||||||
5. Click the little orange `Foods` button and now choose `Units`.
|
5. Click the little orange `Foods` button and now choose `Units`.
|
||||||
6. Click `Seed` and choose your language. You should end up with a list of units (e.g. `tablespoon`)
|
6. Click `Seed` and choose your language. You should end up with a list of units (e.g. `tablespoon`)
|
||||||
|
|
||||||
@@ -33,9 +33,9 @@ Do the following for each recipe you want to intelligently handle ingredients.
|
|||||||
|
|
||||||
Scaling up this recipe or adding it to a Shopping List will now smartly take care of ingredient amounts and duplicate combinations.
|
Scaling up this recipe or adding it to a Shopping List will now smartly take care of ingredient amounts and duplicate combinations.
|
||||||
|
|
||||||
## Is it Safe to Upgrade Mealie?
|
## Is it safe to upgrade Mealie?
|
||||||
|
|
||||||
Yes. If you are using the v1 branches (including beta), you can upgrade to the latest version of Mealie without performing a site Export/Restore. This process was required in previous versions of Mealie, however we've automated the database migration process to make it easier to upgrade. Not that if you were using the v0.5.x version, you CANNOT upgrade to the latest version automatically. You must follow the migration instructions in the documentation.
|
Yes. If you are using the v1 branches (including beta), you can upgrade to the latest version of Mealie without performing a site Export/Restore. This process was required in previous versions of Mealie, however we've automated the database migration process to make it easier to upgrade. Note that if you were using the v0.5.x version, you CANNOT upgrade to the latest version automatically. You must follow the migration instructions in the documentation.
|
||||||
|
|
||||||
- [Migration From v0.5.x](./migrating-to-mealie-v1.md)
|
- [Migration From v0.5.x](./migrating-to-mealie-v1.md)
|
||||||
|
|
||||||
@@ -45,7 +45,7 @@ You can change the theme by settings the environment variables.
|
|||||||
|
|
||||||
- [Backend Config - Themeing](./installation/backend-config.md#themeing)
|
- [Backend Config - Themeing](./installation/backend-config.md#themeing)
|
||||||
|
|
||||||
## How can I change the Login Session Timeout?
|
## How can I change the login session timeout?
|
||||||
|
|
||||||
Login session can be configured by setting the `TOKEN_TIME` variable on the backend container.
|
Login session can be configured by setting the `TOKEN_TIME` variable on the backend container.
|
||||||
|
|
||||||
@@ -53,7 +53,7 @@ Login session can be configured by setting the `TOKEN_TIME` variable on the back
|
|||||||
|
|
||||||
## Can I serve Mealie on a subpath?
|
## Can I serve Mealie on a subpath?
|
||||||
|
|
||||||
No. Due to limitations from the Javascript Framework, mealie doesn't support serving Mealie on a subpath.
|
No. Due to limitations from the JavaScript Framework, Mealie doesn't support serving Mealie on a subpath.
|
||||||
|
|
||||||
## Can I install Mealie without docker?
|
## Can I install Mealie without docker?
|
||||||
|
|
||||||
@@ -130,8 +130,8 @@ stateDiagram-v2
|
|||||||
|
|
||||||
For more information, check out the [Permissions and Public Access guide](./usage/permissions-and-public-access.md).
|
For more information, check out the [Permissions and Public Access guide](./usage/permissions-and-public-access.md).
|
||||||
|
|
||||||
## Can I use fail2ban with mealie?
|
## Can I use fail2ban with Mealie?
|
||||||
Yes, mealie is configured to properly forward external IP addresses into the `mealie.log` logfile. Note that due to restrictions in docker, IP address forwarding only works on Linux.
|
Yes, Mealie is configured to properly forward external IP addresses into the `mealie.log` logfile. Note that due to restrictions in docker, IP address forwarding only works on Linux.
|
||||||
|
|
||||||
Your fail2ban usage should look like the following:
|
Your fail2ban usage should look like the following:
|
||||||
```
|
```
|
||||||
@@ -139,11 +139,11 @@ Use datepattern : %d-%b-%y %H:%M:%S : Day-MON-Year2 24hour:Minute:Second
|
|||||||
Use failregex line : ^ERROR:\s+Incorrect username or password from <HOST>
|
Use failregex line : ^ERROR:\s+Incorrect username or password from <HOST>
|
||||||
```
|
```
|
||||||
|
|
||||||
## Why An API?
|
## Why an API?
|
||||||
An API allows integration into applications like [Home Assistant](https://www.home-assistant.io/) that can act as notification engines to provide custom notifications based on Meal Plan data to remind you to defrost the chicken, marinade the steak, or start the CrockPot. Additionally, you can access nearly any backend service via the API giving you total control to extend the application. To explore the API spin up your server and navigate to http://yourserver.com/docs for interactive API documentation.
|
An API allows integration into applications like [Home Assistant](https://www.home-assistant.io/) that can act as notification engines to provide custom notifications based on Meal Plan data to remind you to defrost the chicken, marinate the steak, or start the CrockPot. Additionally, you can access nearly any backend service via the API giving you total control to extend the application. To explore the API spin up your server and navigate to http://yourserver.com/docs for interactive API documentation.
|
||||||
|
|
||||||
## Why a Database?
|
## Why a database?
|
||||||
Some users of static-site generator applications like ChowDown have expressed concerns about their data being stuck in a database. Considering this is a new project, it is a valid concern to be worried about your data. Mealie specifically addresses this concern by provided automatic daily backups that export your data in json, plain-text markdown files, and/or custom Jinja2 templates. **This puts you in control of how your data is represented** when exported from Mealie, which means you can easily migrate to any other service provided Mealie doesn't work for you.
|
Some users of static-site generator applications like ChowDown have expressed concerns about their data being stuck in a database. Considering this is a new project, it is a valid concern to be worried about your data. Mealie specifically addresses this concern by providing automatic daily backups that export your data in json, plain-text markdown files, and/or custom Jinja2 templates. **This puts you in control of how your data is represented** when exported from Mealie, which means you can easily migrate to any other service provided Mealie doesn't work for you.
|
||||||
|
|
||||||
As to why we need a database?
|
As to why we need a database?
|
||||||
|
|
||||||
|
|||||||
@@ -14,10 +14,14 @@ Mealie offers two main ways to create recipes. You can use the integrated recipe
|
|||||||
|
|
||||||
Mealie supports importing recipes from a few other sources besides websites. Currently the following sources are supported:
|
Mealie supports importing recipes from a few other sources besides websites. Currently the following sources are supported:
|
||||||
|
|
||||||
- Mealie Pre v1
|
- Tandoor
|
||||||
- Nextcloud Cookbooks
|
- Nextcloud Cookbooks
|
||||||
- Paprika
|
- Paprika
|
||||||
- Chowdown
|
- Chowdown
|
||||||
|
- Plan to Eat
|
||||||
|
- Recipe Keeper
|
||||||
|
- Copy Me That
|
||||||
|
- My Recipe Box
|
||||||
|
|
||||||
You can access these options on your installation at the `/group/migrations` page on your installation. If you'd like to see another source added, feel free to request so on Github.
|
You can access these options on your installation at the `/group/migrations` page on your installation. If you'd like to see another source added, feel free to request so on Github.
|
||||||
|
|
||||||
|
|||||||
@@ -4,21 +4,22 @@
|
|||||||
|
|
||||||
### General
|
### General
|
||||||
|
|
||||||
| Variables | Default | Description |
|
| Variables | Default | Description |
|
||||||
| ----------------------------- | :-------------------: | ----------------------------------------------------------------------------------- |
|
| ----------------------------- | :-------------------: | --------------------------------------------------------------------------------------------------------- |
|
||||||
| PUID | 911 | UserID permissions between host OS and container |
|
| PUID | 911 | UserID permissions between host OS and container |
|
||||||
| PGID | 911 | GroupID permissions between host OS and container |
|
| PGID | 911 | GroupID permissions between host OS and container |
|
||||||
| DEFAULT_GROUP | Home | The default group for users |
|
| DEFAULT_GROUP | Home | The default group for users |
|
||||||
| BASE_URL | http://localhost:8080 | Used for Notifications |
|
| BASE_URL | http://localhost:8080 | Used for Notifications |
|
||||||
| TOKEN_TIME | 48 | The time in hours that a login/auth token is valid |
|
| TOKEN_TIME | 48 | The time in hours that a login/auth token is valid |
|
||||||
| API_PORT | 9000 | The port exposed by backend API. **Do not change this if you're running in Docker** |
|
| API_PORT | 9000 | The port exposed by backend API. **Do not change this if you're running in Docker** |
|
||||||
| API_DOCS | True | Turns on/off access to the API documentation locally. |
|
| API_DOCS | True | Turns on/off access to the API documentation locally |
|
||||||
| TZ | UTC | Must be set to get correct date/time on the server |
|
| TZ | UTC | Must be set to get correct date/time on the server |
|
||||||
| ALLOW_SIGNUP<super>\*</super> | false | Allow user sign-up without token |
|
| ALLOW_SIGNUP<super>\*</super> | false | Allow user sign-up without token |
|
||||||
| LOG_CONFIG_OVERRIDE | | Override the config for logging with a custom path |
|
| LOG_CONFIG_OVERRIDE | | Override the config for logging with a custom path |
|
||||||
| LOG_LEVEL | info | logging level configured |
|
| LOG_LEVEL | info | Logging level (e.g. critical, error, warning, info, debug, trace) |
|
||||||
|
| DAILY_SCHEDULE_TIME | 23:45 | The time of day to run daily server tasks, in HH:MM format. Use the server's local time, *not* UTC |
|
||||||
|
|
||||||
<super>\*</super> Starting in v1.4.0 this was changed to default to `false` as apart of a security review of the application.
|
<super>\*</super> Starting in v1.4.0 this was changed to default to `false` as part of a security review of the application.
|
||||||
|
|
||||||
### Security
|
### Security
|
||||||
|
|
||||||
@@ -55,12 +56,9 @@
|
|||||||
|
|
||||||
Changing the webworker settings may cause unforeseen memory leak issues with Mealie. It's best to leave these at the defaults unless you begin to experience issues with multiple users. Exercise caution when changing these settings
|
Changing the webworker settings may cause unforeseen memory leak issues with Mealie. It's best to leave these at the defaults unless you begin to experience issues with multiple users. Exercise caution when changing these settings
|
||||||
|
|
||||||
| Variables | Default | Description |
|
| Variables | Default | Description |
|
||||||
| ---------------- | :-----: | --------------------------------------------------------------------------------------------------------------------------------- |
|
| --------------- | :-----: | ----------------------------------------------------------------------------- |
|
||||||
| WEB_GUNICORN | false | Enables Gunicorn to manage Uvicorn web for multiple works |
|
| UVICORN_WORKERS | 1 | Sets the number of works for the web server [more info here][unicorn_workers] |
|
||||||
| WORKERS_PER_CORE | 1 | Set the number of workers to the number of CPU cores multiplied by this value (Value \* CPUs). More info [here][workers_per_core] |
|
|
||||||
| MAX_WORKERS | None | Set the maximum number of workers to use. Default is not set meaning unlimited. More info [here][max_workers] |
|
|
||||||
| WEB_CONCURRENCY | 2 | Override the automatic definition of number of workers. More info [here][web_concurrency] |
|
|
||||||
|
|
||||||
### LDAP
|
### LDAP
|
||||||
|
|
||||||
@@ -86,21 +84,21 @@ Changing the webworker settings may cause unforeseen memory leak issues with Mea
|
|||||||
|
|
||||||
For usage, see [Usage - OpenID Connect](../authentication/oidc.md)
|
For usage, see [Usage - OpenID Connect](../authentication/oidc.md)
|
||||||
|
|
||||||
| Variables | Default | Description |
|
| Variables | Default | Description |
|
||||||
| ---------------------- | :-----: | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
|
| ---------------------- | :-----: | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
|
||||||
| OIDC_AUTH_ENABLED | False | Enables authentication via OpenID Connect |
|
| OIDC_AUTH_ENABLED | False | Enables authentication via OpenID Connect |
|
||||||
| OIDC_SIGNUP_ENABLED | True | Enables new users to be created when signing in for the first time with OIDC |
|
| OIDC_SIGNUP_ENABLED | True | Enables new users to be created when signing in for the first time with OIDC |
|
||||||
| OIDC_CONFIGURATION_URL | None | The URL to the OIDC configuration of your provider. This is usually something like https://auth.example.com/.well-known/openid-configuration |
|
| OIDC_CONFIGURATION_URL | None | The URL to the OIDC configuration of your provider. This is usually something like https://auth.example.com/.well-known/openid-configuration |
|
||||||
| OIDC_CLIENT_ID | None | The client id of your configured client in your provider |
|
| OIDC_CLIENT_ID | None | The client id of your configured client in your provider |
|
||||||
| OIDC_USER_GROUP | None | If specified, only users belonging to this group will be able to successfully authenticate, regardless of the `OIDC_ADMIN_GROUP`. For more information see [this page](../authentication/oidc.md#groups) |
|
| OIDC_USER_GROUP | None | If specified, only users belonging to this group will be able to successfully authenticate, regardless of the `OIDC_ADMIN_GROUP`. For more information see [this page](../authentication/oidc.md#groups) |
|
||||||
| OIDC_ADMIN_GROUP | None | If specified, users belonging to this group will be made an admin. For more information see [this page](../authentication/oidc.md#groups) |
|
| OIDC_ADMIN_GROUP | None | If specified, users belonging to this group will be made an admin. For more information see [this page](../authentication/oidc.md#groups) |
|
||||||
| OIDC_AUTO_REDIRECT | False | If `True`, then the login page will be bypassed an you will be sent directly to your Identity Provider. You can still get to the login page by adding `?direct=1` to the login URL |
|
| OIDC_AUTO_REDIRECT | False | If `True`, then the login page will be bypassed an you will be sent directly to your Identity Provider. You can still get to the login page by adding `?direct=1` to the login URL |
|
||||||
| OIDC_PROVIDER_NAME | OAuth | The provider name is shown in SSO login button. "Login with <OIDC_PROVIDER_NAME\>" |
|
| OIDC_PROVIDER_NAME | OAuth | The provider name is shown in SSO login button. "Login with <OIDC_PROVIDER_NAME\>" |
|
||||||
| OIDC_REMEMBER_ME | False | Because redirects bypass the login screen, you cant extend your session by clicking the "Remember Me" checkbox. By setting this value to true, a session will be extended as if "Remember Me" was checked |
|
| OIDC_REMEMBER_ME | False | Because redirects bypass the login screen, you cant extend your session by clicking the "Remember Me" checkbox. By setting this value to true, a session will be extended as if "Remember Me" was checked |
|
||||||
| OIDC_SIGNING_ALGORITHM | RS256 | The algorithm used to sign the id token (examples: RS256, HS256) |
|
| OIDC_SIGNING_ALGORITHM | RS256 | The algorithm used to sign the id token (examples: RS256, HS256) |
|
||||||
| OIDC_USER_CLAIM | email | This is the claim which Mealie will use to look up an existing user by (e.g. "email", "preferred_username") |
|
| OIDC_USER_CLAIM | email | This is the claim which Mealie will use to look up an existing user by (e.g. "email", "preferred_username") |
|
||||||
| OIDC_GROUPS_CLAIM | groups | Optional if not using `OIDC_USER_GROUP` or `OIDC_ADMIN_GROUP`. This is the claim Mealie will request from your IdP and will use to compare to `OIDC_USER_GROUP` or `OIDC_ADMIN_GROUP` to allow the user to log in to Mealie or is set as an admin. **Your IdP must be configured to grant this claim**|
|
| OIDC_GROUPS_CLAIM | groups | Optional if not using `OIDC_USER_GROUP` or `OIDC_ADMIN_GROUP`. This is the claim Mealie will request from your IdP and will use to compare to `OIDC_USER_GROUP` or `OIDC_ADMIN_GROUP` to allow the user to log in to Mealie or is set as an admin. **Your IdP must be configured to grant this claim** |
|
||||||
| OIDC_TLS_CACERTFILE | None | File path to Certificate Authority used to verify server certificate (e.g. `/path/to/ca.crt`) |
|
| OIDC_TLS_CACERTFILE | None | File path to Certificate Authority used to verify server certificate (e.g. `/path/to/ca.crt`) |
|
||||||
|
|
||||||
### OpenAI
|
### OpenAI
|
||||||
|
|
||||||
@@ -108,13 +106,14 @@ For usage, see [Usage - OpenID Connect](../authentication/oidc.md)
|
|||||||
|
|
||||||
Mealie supports various integrations using OpenAI. To enable OpenAI, [you must provide your OpenAI API key](https://platform.openai.com/api-keys). You can tweak how OpenAI is used using these backend settings. Please note that while OpenAI usage is optimized to reduce API costs, you're unlikely to be able to use OpenAI features with the free tier limits.
|
Mealie supports various integrations using OpenAI. To enable OpenAI, [you must provide your OpenAI API key](https://platform.openai.com/api-keys). You can tweak how OpenAI is used using these backend settings. Please note that while OpenAI usage is optimized to reduce API costs, you're unlikely to be able to use OpenAI features with the free tier limits.
|
||||||
|
|
||||||
| Variables | Default | Description |
|
| Variables | Default | Description |
|
||||||
| ------------------------- | :------: | ------------------------------------------------------------------------------------------------------------------------------ |
|
| ------------------------- | :-----: | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
|
||||||
| OPENAI_BASE_URL | None | The base URL for the OpenAI API. If you're not sure, leave this empty to use the standard OpenAI platform |
|
| OPENAI_BASE_URL | None | The base URL for the OpenAI API. If you're not sure, leave this empty to use the standard OpenAI platform |
|
||||||
| OPENAI_API_KEY | None | Your OpenAI API Key. Enables OpenAI-related features |
|
| OPENAI_API_KEY | None | Your OpenAI API Key. Enables OpenAI-related features |
|
||||||
| OPENAI_MODEL | gpt-4o | Which OpenAI model to use. If you're not sure, leave this empty |
|
| OPENAI_MODEL | gpt-4o | Which OpenAI model to use. If you're not sure, leave this empty |
|
||||||
| OPENAI_WORKERS | 2 | Number of OpenAI workers per request. Higher values may increase processing speed, but will incur additional API costs |
|
| OPENAI_WORKERS | 2 | Number of OpenAI workers per request. Higher values may increase processing speed, but will incur additional API costs |
|
||||||
| OPENAI_SEND_DATABASE_DATA | True | Whether to send Mealie data to OpenAI to improve request accuracy. This will incur additional API costs |
|
| OPENAI_SEND_DATABASE_DATA | True | Whether to send Mealie data to OpenAI to improve request accuracy. This will incur additional API costs |
|
||||||
|
| OPENAI_REQUEST_TIMEOUT | 10 | The number of seconds to wait for an OpenAI request to complete before cancelling the request. Leave this empty unless you're running into timeout issues on slower hardware |
|
||||||
|
|
||||||
### Themeing
|
### Themeing
|
||||||
|
|
||||||
@@ -137,6 +136,26 @@ Setting the following environmental variables will change the theme of the front
|
|||||||
| THEME_DARK_WARNING | #FF6D00 | Dark Theme Config Variable |
|
| THEME_DARK_WARNING | #FF6D00 | Dark Theme Config Variable |
|
||||||
| THEME_DARK_ERROR | #EF5350 | Dark Theme Config Variable |
|
| THEME_DARK_ERROR | #EF5350 | Dark Theme Config Variable |
|
||||||
|
|
||||||
[workers_per_core]: https://github.com/tiangolo/uvicorn-gunicorn-docker/blob/2daa3e3873c837d5781feb4ff6a40a89f791f81b/README.md#workers_per_core
|
### Docker Secrets
|
||||||
[max_workers]: https://github.com/tiangolo/uvicorn-gunicorn-docker/blob/2daa3e3873c837d5781feb4ff6a40a89f791f81b/README.md#max_workers
|
|
||||||
[web_concurrency]: https://github.com/tiangolo/uvicorn-gunicorn-docker/blob/2daa3e3873c837d5781feb4ff6a40a89f791f81b/README.md#web_concurrency
|
Setting a credential can be done using secrets when running in a Docker container.
|
||||||
|
This can be used to avoid leaking passwords through compose files, environment variables, or command-line history.
|
||||||
|
For example, to configure the Postgres database password in Docker compose, create a file on the host that contains only the password, and expose that file to the Mealie service as a secret with the correct name.
|
||||||
|
Note that environment variables take priority over secrets, so any previously defined environment variables should be removed when migrating to secrets.
|
||||||
|
|
||||||
|
```
|
||||||
|
services:
|
||||||
|
mealie:
|
||||||
|
...
|
||||||
|
environment:
|
||||||
|
...
|
||||||
|
POSTGRES_USER: postgres
|
||||||
|
secrets:
|
||||||
|
- POSTGRES_PASSWORD
|
||||||
|
|
||||||
|
secrets:
|
||||||
|
POSTGRES_PASSWORD:
|
||||||
|
file: postgrespassword.txt
|
||||||
|
```
|
||||||
|
|
||||||
|
[unicorn_workers]: https://www.uvicorn.org/deployment/#built-in
|
||||||
|
|||||||
@@ -58,7 +58,7 @@ The following steps were tested on a Ubuntu 20.04 server, but should work for mo
|
|||||||
4. Create a docker-compose.yaml file in the mealie directory: `touch docker-compose.yaml`
|
4. Create a docker-compose.yaml file in the mealie directory: `touch docker-compose.yaml`
|
||||||
5. Use the text editor of your choice to edit the file and copy the contents of the docker-compose template for the deployment type you want to use: `nano docker-compose.yaml` or `vi docker-compose.yaml`
|
5. Use the text editor of your choice to edit the file and copy the contents of the docker-compose template for the deployment type you want to use: `nano docker-compose.yaml` or `vi docker-compose.yaml`
|
||||||
|
|
||||||
## Step 2: Customizing The `docker-compose.yaml` files.
|
## Step 3: Customizing The `docker-compose.yaml` files.
|
||||||
|
|
||||||
After you've decided setup the files it's important to set a few ENV variables to ensure that you can use all the features of Mealie. I recommend that you verify and check that:
|
After you've decided setup the files it's important to set a few ENV variables to ensure that you can use all the features of Mealie. I recommend that you verify and check that:
|
||||||
|
|
||||||
@@ -67,7 +67,7 @@ After you've decided setup the files it's important to set a few ENV variables t
|
|||||||
- [x] You've set the [`BASE_URL`](./backend-config.md#general) variable.
|
- [x] You've set the [`BASE_URL`](./backend-config.md#general) variable.
|
||||||
- [x] You've set the `DEFAULT_EMAIL` and `DEFAULT_GROUP` variable.
|
- [x] You've set the `DEFAULT_EMAIL` and `DEFAULT_GROUP` variable.
|
||||||
|
|
||||||
## Step 3: Startup
|
## Step 4: Startup
|
||||||
|
|
||||||
After you've configured your database and updated the `docker-compose.yaml` files, you can start Mealie by running the following command in the directory where you've added your `docker-compose.yaml`.
|
After you've configured your database and updated the `docker-compose.yaml` files, you can start Mealie by running the following command in the directory where you've added your `docker-compose.yaml`.
|
||||||
|
|
||||||
@@ -87,11 +87,11 @@ You should see the containers start up without error. You should now be able to
|
|||||||
|
|
||||||
**Password:** MyPassword
|
**Password:** MyPassword
|
||||||
|
|
||||||
## Step 4: Validate Installation
|
## Step 5: Validate Installation
|
||||||
|
|
||||||
After the startup is complete, you should see a login screen. Use the default credentials above to log in and navigate to `/admin/site-settings`. Here, you'll find a summary of your configuration details and their respective status. Before proceeding, you should validate that the configuration is correct. For any warnings or errors the page will display an error and notify you of what you need to verify.
|
After the startup is complete, you should see a login screen. Use the default credentials above to log in and navigate to `/admin/site-settings`. Here, you'll find a summary of your configuration details and their respective status. Before proceeding, you should validate that the configuration is correct. For any warnings or errors the page will display an error and notify you of what you need to verify.
|
||||||
|
|
||||||
## Step 5: Backup
|
## Step 6: Backup
|
||||||
|
|
||||||
While v1.0.0 is a great step to data-stability and security, it's not a backup. Mealie provides a full site data backup mechanism through the UI.
|
While v1.0.0 is a great step to data-stability and security, it's not a backup. Mealie provides a full site data backup mechanism through the UI.
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
:octicons-tag-24: v1.5.0
|
:octicons-tag-24: v1.5.0
|
||||||
|
|
||||||
## Highlighs
|
## Highlights
|
||||||
|
|
||||||
- Logs are written to `/app/data/mealie.log` by default in the container.
|
- Logs are written to `/app/data/mealie.log` by default in the container.
|
||||||
- Logs are also written to stdout and stderr.
|
- Logs are also written to stdout and stderr.
|
||||||
@@ -13,4 +13,4 @@
|
|||||||
Starting in v1.5.0 logging is now highly configurable. Using the `LOG_CONFIG_OVERRIDE` you can provide the application with a custom configuration to log however you'd like. This configuration file is based off the [Python Logging Config](https://docs.python.org/3/library/logging.config.html#logging.config.fileConfig). It can be difficult to understand the configuration at first, so here are some resources to help get started.
|
Starting in v1.5.0 logging is now highly configurable. Using the `LOG_CONFIG_OVERRIDE` you can provide the application with a custom configuration to log however you'd like. This configuration file is based off the [Python Logging Config](https://docs.python.org/3/library/logging.config.html#logging.config.fileConfig). It can be difficult to understand the configuration at first, so here are some resources to help get started.
|
||||||
|
|
||||||
- This [YouTube Video](https://www.youtube.com/watch?v=9L77QExPmI0) for a great walkthrough on the logging file format.
|
- This [YouTube Video](https://www.youtube.com/watch?v=9L77QExPmI0) for a great walkthrough on the logging file format.
|
||||||
- Our [Logging Config](https://github.com/mealie-recipes/mealie/blob/mealie-next/mealie/core/logger/logconf.prod.json)
|
- Our [Logging Config](https://github.com/mealie-recipes/mealie/blob/mealie-next/mealie/core/logger/logconf.prod.json).
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ PostgreSQL might be considered if you need to support many concurrent users. In
|
|||||||
```yaml
|
```yaml
|
||||||
services:
|
services:
|
||||||
mealie:
|
mealie:
|
||||||
image: ghcr.io/mealie-recipes/mealie:v1.6.0 # (3)
|
image: ghcr.io/mealie-recipes/mealie:v1.10.2 # (3)
|
||||||
container_name: mealie
|
container_name: mealie
|
||||||
restart: always
|
restart: always
|
||||||
ports:
|
ports:
|
||||||
@@ -20,7 +20,7 @@ services:
|
|||||||
- mealie-data:/app/data/
|
- mealie-data:/app/data/
|
||||||
environment:
|
environment:
|
||||||
# Set Backend ENV Variables Here
|
# Set Backend ENV Variables Here
|
||||||
ALLOW_SIGNUP: true
|
ALLOW_SIGNUP: false
|
||||||
PUID: 1000
|
PUID: 1000
|
||||||
PGID: 1000
|
PGID: 1000
|
||||||
TZ: America/Anchorage
|
TZ: America/Anchorage
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ SQLite is a popular, open source, self-contained, zero-configuration database th
|
|||||||
```yaml
|
```yaml
|
||||||
services:
|
services:
|
||||||
mealie:
|
mealie:
|
||||||
image: ghcr.io/mealie-recipes/mealie:v1.6.0 # (3)
|
image: ghcr.io/mealie-recipes/mealie:v1.10.2 # (3)
|
||||||
container_name: mealie
|
container_name: mealie
|
||||||
restart: always
|
restart: always
|
||||||
ports:
|
ports:
|
||||||
@@ -24,7 +24,7 @@ services:
|
|||||||
- mealie-data:/app/data/
|
- mealie-data:/app/data/
|
||||||
environment:
|
environment:
|
||||||
# Set Backend ENV Variables Here
|
# Set Backend ENV Variables Here
|
||||||
ALLOW_SIGNUP: true
|
ALLOW_SIGNUP: false
|
||||||
PUID: 1000
|
PUID: 1000
|
||||||
PGID: 1000
|
PGID: 1000
|
||||||
TZ: America/Anchorage
|
TZ: America/Anchorage
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
@@ -79,8 +79,8 @@ nav:
|
|||||||
- Permissions and Public Access: "documentation/getting-started/usage/permissions-and-public-access.md"
|
- Permissions and Public Access: "documentation/getting-started/usage/permissions-and-public-access.md"
|
||||||
|
|
||||||
- Authentication:
|
- Authentication:
|
||||||
- LDAP: "documentation/getting-started/authentication/ldap.md"
|
- LDAP: "documentation/getting-started/authentication/ldap.md"
|
||||||
- OpenID Connect: "documentation/getting-started/authentication/oidc.md"
|
- OpenID Connect: "documentation/getting-started/authentication/oidc.md"
|
||||||
|
|
||||||
- Community Guides:
|
- Community Guides:
|
||||||
- iOS Shortcuts: "documentation/community-guide/ios.md"
|
- iOS Shortcuts: "documentation/community-guide/ios.md"
|
||||||
@@ -88,6 +88,7 @@ nav:
|
|||||||
- Home Assistant: "documentation/community-guide/home-assistant.md"
|
- Home Assistant: "documentation/community-guide/home-assistant.md"
|
||||||
- Bulk Url Import: "documentation/community-guide/bulk-url-import.md"
|
- Bulk Url Import: "documentation/community-guide/bulk-url-import.md"
|
||||||
- Import Bookmarklet: "documentation/community-guide/import-recipe-bookmarklet.md"
|
- Import Bookmarklet: "documentation/community-guide/import-recipe-bookmarklet.md"
|
||||||
|
- Automate Backups with n8n: "documentation/community-guide/n8n-backup-automation.md"
|
||||||
|
|
||||||
- API Reference: "api/redoc.md"
|
- API Reference: "api/redoc.md"
|
||||||
|
|
||||||
@@ -97,23 +98,7 @@ nav:
|
|||||||
- Developers Guide:
|
- Developers Guide:
|
||||||
- Code Contributions: "contributors/developers-guide/code-contributions.md"
|
- Code Contributions: "contributors/developers-guide/code-contributions.md"
|
||||||
- Dev Getting Started: "contributors/developers-guide/starting-dev-server.md"
|
- Dev Getting Started: "contributors/developers-guide/starting-dev-server.md"
|
||||||
|
- Database Changes: "contributors/developers-guide/database-changes.md"
|
||||||
- Maintainers Guide: "contributors/developers-guide/maintainers.md"
|
- Maintainers Guide: "contributors/developers-guide/maintainers.md"
|
||||||
- Guides:
|
- Guides:
|
||||||
- Improving Ingredient Parser: "contributors/guides/ingredient-parser.md"
|
- Improving Ingredient Parser: "contributors/guides/ingredient-parser.md"
|
||||||
|
|
||||||
- Change Log:
|
|
||||||
- v1.0.0beta-5: "changelog/v1.0.0beta-5.md"
|
|
||||||
- v1.0.0beta-4: "changelog/v1.0.0beta-4.md"
|
|
||||||
- v1.0.0beta-3: "changelog/v1.0.0beta-3.md"
|
|
||||||
- v1.0.0beta-2: "changelog/v1.0.0beta-2.md"
|
|
||||||
- v1.0.0 Beta: "changelog/v1.0.0.md"
|
|
||||||
- v0.5.2 Misc Updates: "changelog/v0.5.2.md"
|
|
||||||
- v0.5.1 Bug Fixes: "changelog/v0.5.1.md"
|
|
||||||
- v0.5.0 General Upgrades: "changelog/v0.5.0.md"
|
|
||||||
- v0.4.3 Hot Fix: "changelog/v0.4.3.md"
|
|
||||||
- v0.4.2 Backend/Migrations: "changelog/v0.4.2.md"
|
|
||||||
- v0.4.1 Frontend/UI: "changelog/v0.4.1.md"
|
|
||||||
- v0.4.0 Authentication: "changelog/v0.4.0.md"
|
|
||||||
- v0.3.0 Improvements: "changelog/v0.3.0.md"
|
|
||||||
- v0.2.0 Now With Tests!: "changelog/v0.2.0.md"
|
|
||||||
- v0.1.0 Beta: "changelog/v0.1.0.md"
|
|
||||||
|
|||||||
@@ -18,8 +18,6 @@
|
|||||||
icon: $globals.icons.testTube,
|
icon: $globals.icons.testTube,
|
||||||
text: $tc('general.test'),
|
text: $tc('general.test'),
|
||||||
event: 'test',
|
event: 'test',
|
||||||
// TODO: There is no functionality hooked up to this. Enable it when there is
|
|
||||||
disabled: true,
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
icon: $globals.icons.save,
|
icon: $globals.icons.save,
|
||||||
|
|||||||
@@ -36,6 +36,9 @@
|
|||||||
<v-list-item-subtitle>
|
<v-list-item-subtitle>
|
||||||
<SafeMarkdown :source="description" />
|
<SafeMarkdown :source="description" />
|
||||||
</v-list-item-subtitle>
|
</v-list-item-subtitle>
|
||||||
|
<div class="d-flex flex-wrap justify-start">
|
||||||
|
<RecipeChips :truncate="true" :items="tags" :title="false" :limit="2" :small="true" url-prefix="tags" />
|
||||||
|
</div>
|
||||||
<div class="d-flex flex-wrap justify-end align-center">
|
<div class="d-flex flex-wrap justify-end align-center">
|
||||||
<slot name="actions">
|
<slot name="actions">
|
||||||
<RecipeFavoriteBadge v-if="isOwnGroup && showRecipeContent" :recipe-id="recipeId" show-always />
|
<RecipeFavoriteBadge v-if="isOwnGroup && showRecipeContent" :recipe-id="recipeId" show-always />
|
||||||
@@ -83,6 +86,7 @@ import RecipeFavoriteBadge from "./RecipeFavoriteBadge.vue";
|
|||||||
import RecipeContextMenu from "./RecipeContextMenu.vue";
|
import RecipeContextMenu from "./RecipeContextMenu.vue";
|
||||||
import RecipeCardImage from "./RecipeCardImage.vue";
|
import RecipeCardImage from "./RecipeCardImage.vue";
|
||||||
import RecipeRating from "./RecipeRating.vue";
|
import RecipeRating from "./RecipeRating.vue";
|
||||||
|
import RecipeChips from "./RecipeChips.vue";
|
||||||
import { useLoggedInState } from "~/composables/use-logged-in-state";
|
import { useLoggedInState } from "~/composables/use-logged-in-state";
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
@@ -91,6 +95,7 @@ export default defineComponent({
|
|||||||
RecipeContextMenu,
|
RecipeContextMenu,
|
||||||
RecipeRating,
|
RecipeRating,
|
||||||
RecipeCardImage,
|
RecipeCardImage,
|
||||||
|
RecipeChips,
|
||||||
},
|
},
|
||||||
props: {
|
props: {
|
||||||
name: {
|
name: {
|
||||||
@@ -114,6 +119,10 @@ export default defineComponent({
|
|||||||
required: false,
|
required: false,
|
||||||
default: "abc123",
|
default: "abc123",
|
||||||
},
|
},
|
||||||
|
tags: {
|
||||||
|
type: Array,
|
||||||
|
default: () => [],
|
||||||
|
},
|
||||||
recipeId: {
|
recipeId: {
|
||||||
type: String,
|
type: String,
|
||||||
required: true,
|
required: true,
|
||||||
|
|||||||
@@ -231,36 +231,46 @@ export default defineComponent({
|
|||||||
|
|
||||||
const shoppingListIngredients: ShoppingListIngredient[] = recipe.recipeIngredient.map((ing) => {
|
const shoppingListIngredients: ShoppingListIngredient[] = recipe.recipeIngredient.map((ing) => {
|
||||||
return {
|
return {
|
||||||
checked: true,
|
checked: !ing.food?.onHand,
|
||||||
ingredient: ing,
|
ingredient: ing,
|
||||||
disableAmount: recipe.settings?.disableAmount || false,
|
disableAmount: recipe.settings?.disableAmount || false,
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
let currentTitle = "";
|
||||||
|
const onHandIngs: ShoppingListIngredient[] = [];
|
||||||
const shoppingListIngredientSections = shoppingListIngredients.reduce((sections, ing) => {
|
const shoppingListIngredientSections = shoppingListIngredients.reduce((sections, ing) => {
|
||||||
// if title append new section to the end of the array
|
|
||||||
if (ing.ingredient.title) {
|
if (ing.ingredient.title) {
|
||||||
|
currentTitle = ing.ingredient.title;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If this is the first item in the section, create a new section
|
||||||
|
if (sections.length === 0 || currentTitle !== sections[sections.length - 1].sectionName) {
|
||||||
|
if (sections.length) {
|
||||||
|
// Add the on-hand ingredients to the previous section
|
||||||
|
sections[sections.length - 1].ingredients.push(...onHandIngs);
|
||||||
|
onHandIngs.length = 0;
|
||||||
|
}
|
||||||
sections.push({
|
sections.push({
|
||||||
sectionName: ing.ingredient.title,
|
sectionName: currentTitle,
|
||||||
ingredients: [ing],
|
ingredients: [],
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Store the on-hand ingredients for later
|
||||||
|
if (ing.ingredient.food?.onHand) {
|
||||||
|
onHandIngs.push(ing);
|
||||||
return sections;
|
return sections;
|
||||||
}
|
}
|
||||||
|
|
||||||
// append new section if first
|
// Add the ingredient to previous section
|
||||||
if (sections.length === 0) {
|
|
||||||
sections.push({
|
|
||||||
sectionName: "",
|
|
||||||
ingredients: [ing],
|
|
||||||
});
|
|
||||||
return sections;
|
|
||||||
}
|
|
||||||
|
|
||||||
// otherwise add ingredient to last section in the array
|
|
||||||
sections[sections.length - 1].ingredients.push(ing);
|
sections[sections.length - 1].ingredients.push(ing);
|
||||||
return sections;
|
return sections;
|
||||||
}, [] as ShoppingListIngredientSection[]);
|
}, [] as ShoppingListIngredientSection[]);
|
||||||
|
|
||||||
|
// Add remaining on-hand ingredients to the previous section
|
||||||
|
shoppingListIngredientSections[shoppingListIngredientSections.length - 1].ingredients.push(...onHandIngs);
|
||||||
|
|
||||||
recipeSectionMap.set(recipe.slug, {
|
recipeSectionMap.set(recipe.slug, {
|
||||||
recipeId: recipe.id,
|
recipeId: recipe.id,
|
||||||
recipeName: recipe.name,
|
recipeName: recipe.name,
|
||||||
|
|||||||
@@ -31,6 +31,13 @@
|
|||||||
<v-switch v-model="preferences.showNotes" hide-details :label="$tc('recipe.notes')" />
|
<v-switch v-model="preferences.showNotes" hide-details :label="$tc('recipe.notes')" />
|
||||||
</v-row>
|
</v-row>
|
||||||
</v-col>
|
</v-col>
|
||||||
|
<v-col cols="auto" align-self="start">
|
||||||
|
<v-row no-gutters>
|
||||||
|
<v-switch v-model="preferences.showNutrition" hide-details :label="$tc('recipe.nutrition')" />
|
||||||
|
</v-row>
|
||||||
|
<v-row no-gutters>
|
||||||
|
</v-row>
|
||||||
|
</v-col>
|
||||||
</v-row>
|
</v-row>
|
||||||
</v-container>
|
</v-container>
|
||||||
<v-card
|
<v-card
|
||||||
|
|||||||
@@ -126,8 +126,8 @@
|
|||||||
<RecipeCardSection
|
<RecipeCardSection
|
||||||
v-if="state.ready"
|
v-if="state.ready"
|
||||||
class="mt-n5"
|
class="mt-n5"
|
||||||
:icon="$globals.icons.search"
|
:icon="$globals.icons.silverwareForkKnife"
|
||||||
:title="$tc('search.results')"
|
:title="$tc('general.recipes')"
|
||||||
:recipes="recipes"
|
:recipes="recipes"
|
||||||
:query="passedQueryWithSeed"
|
:query="passedQueryWithSeed"
|
||||||
@replaceRecipes="replaceRecipes"
|
@replaceRecipes="replaceRecipes"
|
||||||
|
|||||||
@@ -104,6 +104,8 @@
|
|||||||
:buttons="btns"
|
:buttons="btns"
|
||||||
@toggle-section="toggleTitle"
|
@toggle-section="toggleTitle"
|
||||||
@toggle-original="toggleOriginalText"
|
@toggle-original="toggleOriginalText"
|
||||||
|
@insert-above="$emit('insert-above')"
|
||||||
|
@insert-below="$emit('insert-below')"
|
||||||
@insert-ingredient="$emit('insert-ingredient')"
|
@insert-ingredient="$emit('insert-ingredient')"
|
||||||
@delete="$emit('delete')"
|
@delete="$emit('delete')"
|
||||||
/>
|
/>
|
||||||
@@ -148,6 +150,14 @@ export default defineComponent({
|
|||||||
text: i18n.tc("recipe.toggle-section"),
|
text: i18n.tc("recipe.toggle-section"),
|
||||||
event: "toggle-section",
|
event: "toggle-section",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
text: i18n.tc("recipe.insert-above"),
|
||||||
|
event: "insert-above",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: i18n.tc("recipe.insert-below"),
|
||||||
|
event: "insert-below",
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
if (props.allowInsertIngredient) {
|
if (props.allowInsertIngredient) {
|
||||||
|
|||||||
@@ -45,6 +45,22 @@ export default defineComponent({
|
|||||||
.d-inline {
|
.d-inline {
|
||||||
& > p {
|
& > p {
|
||||||
display: inline;
|
display: inline;
|
||||||
|
&:has(>sub)>sup {
|
||||||
|
letter-spacing: -0.05rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&:has(sub) {
|
||||||
|
&:after {
|
||||||
|
letter-spacing: -0.2rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
sup {
|
||||||
|
&+span{
|
||||||
|
letter-spacing: -0.05rem;
|
||||||
|
}
|
||||||
|
&:before {
|
||||||
|
letter-spacing: 0rem;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -102,7 +102,7 @@
|
|||||||
<v-icon left>
|
<v-icon left>
|
||||||
{{ $globals.icons.calendar }}
|
{{ $globals.icons.calendar }}
|
||||||
</v-icon>
|
</v-icon>
|
||||||
{{ $t('recipe.last-made-date', { date: value ? new Date(value+"Z").toLocaleDateString($i18n.locale) : $t("general.never") } ) }}
|
{{ $t('recipe.last-made-date', { date: value ? new Date(value).toLocaleDateString($i18n.locale) : $t("general.never") } ) }}
|
||||||
</v-chip>
|
</v-chip>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -199,11 +199,7 @@ export default defineComponent({
|
|||||||
await userApi.recipes.updateLastMade(props.recipe.slug, newTimelineEvent.value.timestamp);
|
await userApi.recipes.updateLastMade(props.recipe.slug, newTimelineEvent.value.timestamp);
|
||||||
|
|
||||||
// update recipe in parent so the user can see it
|
// update recipe in parent so the user can see it
|
||||||
// we remove the trailing "Z" since this is how the API returns it
|
context.emit("input", newTimelineEvent.value.timestamp);
|
||||||
context.emit(
|
|
||||||
"input", newTimelineEvent.value.timestamp
|
|
||||||
.substring(0, newTimelineEvent.value.timestamp.length - 1)
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// update the image, if provided
|
// update the image, if provided
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
:class="attrs.class.sheet"
|
:class="attrs.class.sheet"
|
||||||
:style="tile ? 'max-width: 100%; width: fit-content;' : 'width: 100%;'"
|
:style="tile ? 'max-width: 100%; width: fit-content;' : 'width: 100%;'"
|
||||||
>
|
>
|
||||||
<v-list-item :to="'/g/' + groupSlug + '/r/' + recipe.slug" :class="attrs.class.listItem">
|
<v-list-item :to="disabled ? '' : '/g/' + groupSlug + '/r/' + recipe.slug" :class="attrs.class.listItem">
|
||||||
<v-list-item-avatar :class="attrs.class.avatar">
|
<v-list-item-avatar :class="attrs.class.avatar">
|
||||||
<v-icon :class="attrs.class.icon" dark :small="small"> {{ $globals.icons.primary }} </v-icon>
|
<v-icon :class="attrs.class.icon" dark :small="small"> {{ $globals.icons.primary }} </v-icon>
|
||||||
</v-list-item-avatar>
|
</v-list-item-avatar>
|
||||||
@@ -56,6 +56,10 @@ export default defineComponent({
|
|||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
|
disabled: {
|
||||||
|
type: Boolean,
|
||||||
|
default: false,
|
||||||
|
}
|
||||||
},
|
},
|
||||||
setup(props) {
|
setup(props) {
|
||||||
const { $auth } = useContext();
|
const { $auth } = useContext();
|
||||||
|
|||||||
@@ -8,14 +8,8 @@
|
|||||||
<v-card-text v-if="edit">
|
<v-card-text v-if="edit">
|
||||||
<div v-for="(item, key, index) in value" :key="index">
|
<div v-for="(item, key, index) in value" :key="index">
|
||||||
<v-text-field
|
<v-text-field
|
||||||
dense
|
dense :value="value[key]" :label="labels[key].label" :suffix="labels[key].suffix" type="number"
|
||||||
:value="value[key]"
|
autocomplete="off" @input="updateValue(key, $event)"></v-text-field>
|
||||||
:label="labels[key].label"
|
|
||||||
:suffix="labels[key].suffix"
|
|
||||||
type="number"
|
|
||||||
autocomplete="off"
|
|
||||||
@input="updateValue(key, $event)"
|
|
||||||
></v-text-field>
|
|
||||||
</div>
|
</div>
|
||||||
</v-card-text>
|
</v-card-text>
|
||||||
<v-list v-if="showViewer" dense class="mt-0 pt-0">
|
<v-list v-if="showViewer" dense class="mt-0 pt-0">
|
||||||
@@ -34,17 +28,10 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { computed, defineComponent, useContext } from "@nuxtjs/composition-api";
|
import { computed, defineComponent } from "@nuxtjs/composition-api";
|
||||||
|
import { useNutritionLabels } from "~/composables/recipes";
|
||||||
import { Nutrition } from "~/lib/api/types/recipe";
|
import { Nutrition } from "~/lib/api/types/recipe";
|
||||||
|
import { NutritionLabelType } from "~/composables/recipes/use-recipe-nutrition";
|
||||||
type NutritionLabelType = {
|
|
||||||
[key: string]: {
|
|
||||||
label: string;
|
|
||||||
suffix: string;
|
|
||||||
value?: string;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
props: {
|
props: {
|
||||||
value: {
|
value: {
|
||||||
@@ -57,37 +44,7 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
setup(props, context) {
|
setup(props, context) {
|
||||||
const { i18n } = useContext();
|
const { labels } = useNutritionLabels();
|
||||||
const labels = <NutritionLabelType>{
|
|
||||||
calories: {
|
|
||||||
label: i18n.tc("recipe.calories"),
|
|
||||||
suffix: i18n.tc("recipe.calories-suffix"),
|
|
||||||
},
|
|
||||||
fatContent: {
|
|
||||||
label: i18n.tc("recipe.fat-content"),
|
|
||||||
suffix: i18n.tc("recipe.grams"),
|
|
||||||
},
|
|
||||||
fiberContent: {
|
|
||||||
label: i18n.tc("recipe.fiber-content"),
|
|
||||||
suffix: i18n.tc("recipe.grams"),
|
|
||||||
},
|
|
||||||
proteinContent: {
|
|
||||||
label: i18n.tc("recipe.protein-content"),
|
|
||||||
suffix: i18n.tc("recipe.grams"),
|
|
||||||
},
|
|
||||||
sodiumContent: {
|
|
||||||
label: i18n.tc("recipe.sodium-content"),
|
|
||||||
suffix: i18n.tc("recipe.milligrams"),
|
|
||||||
},
|
|
||||||
sugarContent: {
|
|
||||||
label: i18n.tc("recipe.sugar-content"),
|
|
||||||
suffix: i18n.tc("recipe.grams"),
|
|
||||||
},
|
|
||||||
carbohydrateContent: {
|
|
||||||
label: i18n.tc("recipe.carbohydrate-content"),
|
|
||||||
suffix: i18n.tc("recipe.grams"),
|
|
||||||
},
|
|
||||||
};
|
|
||||||
const valueNotNull = computed(() => {
|
const valueNotNull = computed(() => {
|
||||||
let key: keyof Nutrition;
|
let key: keyof Nutrition;
|
||||||
for (key in props.value) {
|
for (key in props.value) {
|
||||||
|
|||||||
@@ -114,9 +114,9 @@ export default defineComponent({
|
|||||||
options: {
|
options: {
|
||||||
ignoreLocation: true,
|
ignoreLocation: true,
|
||||||
shouldSort: true,
|
shouldSort: true,
|
||||||
threshold: 0.6,
|
threshold: 0.2,
|
||||||
location: 0,
|
location: 0,
|
||||||
distance: 100,
|
distance: 20,
|
||||||
findAllMatches: true,
|
findAllMatches: true,
|
||||||
maxPatternLength: 32,
|
maxPatternLength: 32,
|
||||||
minMatchCharLength: 1,
|
minMatchCharLength: 1,
|
||||||
|
|||||||
@@ -72,7 +72,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<RecipePageComments
|
<RecipePageComments
|
||||||
v-if="isOwnGroup && !recipe.settings.disableComments && !isEditForm && !isCookMode"
|
v-if="!recipe.settings.disableComments && !isEditForm && !isCookMode"
|
||||||
:recipe="recipe"
|
:recipe="recipe"
|
||||||
class="px-1 my-4 d-print-none"
|
class="px-1 my-4 d-print-none"
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -30,7 +30,7 @@
|
|||||||
</BaseButton>
|
</BaseButton>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div v-for="comment in comments" :key="comment.id" class="d-flex my-2" style="gap: 10px">
|
<div v-for="comment in recipe.comments" :key="comment.id" class="d-flex my-2" style="gap: 10px">
|
||||||
<UserAvatar size="40" :user-id="comment.userId" />
|
<UserAvatar size="40" :user-id="comment.userId" />
|
||||||
<v-card outlined class="flex-grow-1">
|
<v-card outlined class="flex-grow-1">
|
||||||
<v-card-text class="pa-3 pb-0">
|
<v-card-text class="pa-3 pb-0">
|
||||||
@@ -54,9 +54,9 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { defineComponent, ref, toRefs, onMounted, reactive } from "@nuxtjs/composition-api";
|
import { defineComponent, toRefs, reactive } from "@nuxtjs/composition-api";
|
||||||
import { useUserApi } from "~/composables/api";
|
import { useUserApi } from "~/composables/api";
|
||||||
import { Recipe, RecipeCommentOut } from "~/lib/api/types/recipe";
|
import { Recipe } from "~/lib/api/types/recipe";
|
||||||
import UserAvatar from "~/components/Domain/User/UserAvatar.vue";
|
import UserAvatar from "~/components/Domain/User/UserAvatar.vue";
|
||||||
import { NoUndefinedField } from "~/lib/api/types/non-generated";
|
import { NoUndefinedField } from "~/lib/api/types/non-generated";
|
||||||
import { usePageUser } from "~/composables/recipe-page/shared-state";
|
import { usePageUser } from "~/composables/recipe-page/shared-state";
|
||||||
@@ -76,22 +76,12 @@ export default defineComponent({
|
|||||||
setup(props) {
|
setup(props) {
|
||||||
const api = useUserApi();
|
const api = useUserApi();
|
||||||
|
|
||||||
const comments = ref<RecipeCommentOut[]>([]);
|
|
||||||
|
|
||||||
const { user } = usePageUser();
|
const { user } = usePageUser();
|
||||||
|
|
||||||
const state = reactive({
|
const state = reactive({
|
||||||
comment: "",
|
comment: "",
|
||||||
});
|
});
|
||||||
|
|
||||||
onMounted(async () => {
|
|
||||||
const { data } = await api.recipes.comments.byRecipe(props.recipe.slug);
|
|
||||||
|
|
||||||
if (data) {
|
|
||||||
comments.value = data;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
async function submitComment() {
|
async function submitComment() {
|
||||||
const { data } = await api.recipes.comments.createOne({
|
const { data } = await api.recipes.comments.createOne({
|
||||||
recipeId: props.recipe.id,
|
recipeId: props.recipe.id,
|
||||||
@@ -99,7 +89,8 @@ export default defineComponent({
|
|||||||
});
|
});
|
||||||
|
|
||||||
if (data) {
|
if (data) {
|
||||||
comments.value.push(data);
|
// @ts-ignore username is always populated here
|
||||||
|
props.recipe.comments.push(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
state.comment = "";
|
state.comment = "";
|
||||||
@@ -109,11 +100,11 @@ export default defineComponent({
|
|||||||
const { response } = await api.recipes.comments.deleteOne(id);
|
const { response } = await api.recipes.comments.deleteOne(id);
|
||||||
|
|
||||||
if (response?.status === 200) {
|
if (response?.status === 200) {
|
||||||
comments.value = comments.value.filter((comment) => comment.id !== id);
|
props.recipe.comments = props.recipe.comments.filter((comment) => comment.id !== id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return { api, comments, ...toRefs(state), submitComment, deleteComment, user };
|
return { api, ...toRefs(state), submitComment, deleteComment, user };
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
handle=".handle"
|
handle=".handle"
|
||||||
v-bind="{
|
v-bind="{
|
||||||
animation: 200,
|
animation: 200,
|
||||||
group: 'description',
|
group: 'recipe-ingredients',
|
||||||
disabled: false,
|
disabled: false,
|
||||||
ghostClass: 'ghost',
|
ghostClass: 'ghost',
|
||||||
}"
|
}"
|
||||||
@@ -22,6 +22,8 @@
|
|||||||
class="list-group-item"
|
class="list-group-item"
|
||||||
:disable-amount="recipe.settings.disableAmount"
|
:disable-amount="recipe.settings.disableAmount"
|
||||||
@delete="recipe.recipeIngredient.splice(index, 1)"
|
@delete="recipe.recipeIngredient.splice(index, 1)"
|
||||||
|
@insert-above="insertNewIngredient(index)"
|
||||||
|
@insert-below="insertNewIngredient(index+1)"
|
||||||
/>
|
/>
|
||||||
</TransitionGroup>
|
</TransitionGroup>
|
||||||
</draggable>
|
</draggable>
|
||||||
@@ -140,6 +142,20 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function insertNewIngredient(dest: number) {
|
||||||
|
props.recipe.recipeIngredient.splice(dest, 0, {
|
||||||
|
referenceId: uuid4(),
|
||||||
|
title: "",
|
||||||
|
note: "",
|
||||||
|
// @ts-expect-error - prop can be null-type by NoUndefinedField type forces it to be set
|
||||||
|
unit: undefined,
|
||||||
|
// @ts-expect-error - prop can be null-type by NoUndefinedField type forces it to be set
|
||||||
|
food: undefined,
|
||||||
|
disableAmount: true,
|
||||||
|
quantity: 1,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
user,
|
user,
|
||||||
groupSlug,
|
groupSlug,
|
||||||
@@ -148,6 +164,7 @@ export default defineComponent({
|
|||||||
hasFoodOrUnit,
|
hasFoodOrUnit,
|
||||||
imageKey,
|
imageKey,
|
||||||
drag,
|
drag,
|
||||||
|
insertNewIngredient,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -79,7 +79,7 @@
|
|||||||
handle=".handle"
|
handle=".handle"
|
||||||
v-bind="{
|
v-bind="{
|
||||||
animation: 200,
|
animation: 200,
|
||||||
group: 'description',
|
group: 'recipe-instructions',
|
||||||
ghostClass: 'ghost',
|
ghostClass: 'ghost',
|
||||||
}"
|
}"
|
||||||
@input="updateIndex"
|
@input="updateIndex"
|
||||||
@@ -170,12 +170,22 @@
|
|||||||
text: $tc('recipe.move-to-bottom'),
|
text: $tc('recipe.move-to-bottom'),
|
||||||
event: 'move-to-bottom',
|
event: 'move-to-bottom',
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
text: $tc('recipe.insert-above'),
|
||||||
|
event: 'insert-above'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: $tc('recipe.insert-below'),
|
||||||
|
event: 'insert-below'
|
||||||
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
]"
|
]"
|
||||||
@merge-above="mergeAbove(index - 1, index)"
|
@merge-above="mergeAbove(index - 1, index)"
|
||||||
@move-to-top="moveTo('top', index)"
|
@move-to-top="moveTo('top', index)"
|
||||||
@move-to-bottom="moveTo('bottom', index)"
|
@move-to-bottom="moveTo('bottom', index)"
|
||||||
|
@insert-above="insert(index)"
|
||||||
|
@insert-below="insert(index+1)"
|
||||||
@toggle-section="toggleShowTitle(step.id)"
|
@toggle-section="toggleShowTitle(step.id)"
|
||||||
@link-ingredients="openDialog(index, step.text, step.ingredientReferences)"
|
@link-ingredients="openDialog(index, step.text, step.ingredientReferences)"
|
||||||
@preview-step="togglePreviewState(index)"
|
@preview-step="togglePreviewState(index)"
|
||||||
@@ -550,6 +560,10 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function insert(dest: number) {
|
||||||
|
props.value.splice(dest, 0, { id: uuid4(), text: "", title: "", ingredientReferences: [] });
|
||||||
|
}
|
||||||
|
|
||||||
const previewStates = ref<boolean[]>([]);
|
const previewStates = ref<boolean[]>([]);
|
||||||
|
|
||||||
function togglePreviewState(index: number) {
|
function togglePreviewState(index: number) {
|
||||||
@@ -681,6 +695,7 @@ export default defineComponent({
|
|||||||
showCookMode,
|
showCookMode,
|
||||||
isCookMode,
|
isCookMode,
|
||||||
isEditForm,
|
isEditForm,
|
||||||
|
insert,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -30,6 +30,9 @@
|
|||||||
<!-- Ingredients -->
|
<!-- Ingredients -->
|
||||||
<section>
|
<section>
|
||||||
<v-card-title class="headline pl-0"> {{ $t("recipe.ingredients") }} </v-card-title>
|
<v-card-title class="headline pl-0"> {{ $t("recipe.ingredients") }} </v-card-title>
|
||||||
|
<div class="font-italic px-0 py-0">
|
||||||
|
<SafeMarkdown :source="recipe.recipeYield" />
|
||||||
|
</div>
|
||||||
<div
|
<div
|
||||||
v-for="(ingredientSection, sectionIndex) in ingredientSections"
|
v-for="(ingredientSection, sectionIndex) in ingredientSections"
|
||||||
:key="`ingredient-section-${sectionIndex}`"
|
:key="`ingredient-section-${sectionIndex}`"
|
||||||
@@ -83,19 +86,41 @@
|
|||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<!-- Nutrition -->
|
||||||
|
<div v-if="preferences.showNutrition">
|
||||||
|
<v-card-title class="headline pl-0"> {{ $t("recipe.nutrition") }} </v-card-title>
|
||||||
|
|
||||||
|
<section>
|
||||||
|
<div class="print-section">
|
||||||
|
<table class="nutrition-table">
|
||||||
|
<tbody>
|
||||||
|
<tr v-for="(value, key) in recipe.nutrition" :key="key">
|
||||||
|
<template v-if="value">
|
||||||
|
<td>{{ labels[key].label }}</td>
|
||||||
|
<td>{{ value || '-' }}</td>
|
||||||
|
</template>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { defineComponent, computed } from "@nuxtjs/composition-api";
|
import { computed, defineComponent } from "@nuxtjs/composition-api";
|
||||||
import RecipeTimeCard from "~/components/Domain/Recipe/RecipeTimeCard.vue";
|
import RecipeTimeCard from "~/components/Domain/Recipe/RecipeTimeCard.vue";
|
||||||
import { useStaticRoutes } from "~/composables/api";
|
import { useStaticRoutes } from "~/composables/api";
|
||||||
import { Recipe, RecipeIngredient, RecipeStep } from "~/lib/api/types/recipe";
|
import { Recipe, RecipeIngredient, RecipeStep} from "~/lib/api/types/recipe";
|
||||||
import { NoUndefinedField } from "~/lib/api/types/non-generated";
|
import { NoUndefinedField } from "~/lib/api/types/non-generated";
|
||||||
import { ImagePosition, useUserPrintPreferences } from "~/composables/use-users/preferences";
|
import { ImagePosition, useUserPrintPreferences } from "~/composables/use-users/preferences";
|
||||||
import { parseIngredientText } from "~/composables/recipes";
|
import { parseIngredientText, useNutritionLabels } from "~/composables/recipes";
|
||||||
import { usePageState } from "~/composables/recipe-page/shared-state";
|
import { usePageState } from "~/composables/recipe-page/shared-state";
|
||||||
|
|
||||||
|
|
||||||
type IngredientSection = {
|
type IngredientSection = {
|
||||||
sectionName: string;
|
sectionName: string;
|
||||||
ingredients: RecipeIngredient[];
|
ingredients: RecipeIngredient[];
|
||||||
@@ -129,6 +154,10 @@ export default defineComponent({
|
|||||||
const preferences = useUserPrintPreferences();
|
const preferences = useUserPrintPreferences();
|
||||||
const { recipeImage } = useStaticRoutes();
|
const { recipeImage } = useStaticRoutes();
|
||||||
const { imageKey } = usePageState(props.recipe.slug);
|
const { imageKey } = usePageState(props.recipe.slug);
|
||||||
|
const {labels} = useNutritionLabels();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
const recipeImageUrl = computed(() => {
|
const recipeImageUrl = computed(() => {
|
||||||
return recipeImage(props.recipe.id, props.recipe.image, imageKey.value);
|
return recipeImage(props.recipe.id, props.recipe.image, imageKey.value);
|
||||||
@@ -221,6 +250,7 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
labels,
|
||||||
hasNotes,
|
hasNotes,
|
||||||
imageKey,
|
imageKey,
|
||||||
ImagePosition,
|
ImagePosition,
|
||||||
@@ -290,4 +320,16 @@ li {
|
|||||||
list-style-type: none;
|
list-style-type: none;
|
||||||
margin-bottom: 0.25rem;
|
margin-bottom: 0.25rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.nutrition-table {
|
||||||
|
width: 25%;
|
||||||
|
border-collapse: collapse;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nutrition-table td {
|
||||||
|
padding: 2px;
|
||||||
|
text-align: left;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -8,7 +8,7 @@
|
|||||||
<template v-if="!useMobileFormat" #opposite>
|
<template v-if="!useMobileFormat" #opposite>
|
||||||
<v-chip v-if="event.timestamp" label large>
|
<v-chip v-if="event.timestamp" label large>
|
||||||
<v-icon class="mr-1"> {{ $globals.icons.calendar }} </v-icon>
|
<v-icon class="mr-1"> {{ $globals.icons.calendar }} </v-icon>
|
||||||
{{ new Date(event.timestamp+"Z").toLocaleDateString($i18n.locale) }}
|
{{ new Date(event.timestamp).toLocaleDateString($i18n.locale) }}
|
||||||
</v-chip>
|
</v-chip>
|
||||||
</template>
|
</template>
|
||||||
<v-card
|
<v-card
|
||||||
@@ -25,7 +25,7 @@
|
|||||||
<v-col v-if="useMobileFormat" align-self="center" class="pr-0">
|
<v-col v-if="useMobileFormat" align-self="center" class="pr-0">
|
||||||
<v-chip label>
|
<v-chip label>
|
||||||
<v-icon> {{ $globals.icons.calendar }} </v-icon>
|
<v-icon> {{ $globals.icons.calendar }} </v-icon>
|
||||||
{{ new Date(event.timestamp+"Z").toLocaleDateString($i18n.locale) }}
|
{{ new Date(event.timestamp || "").toLocaleDateString($i18n.locale) }}
|
||||||
</v-chip>
|
</v-chip>
|
||||||
</v-col>
|
</v-col>
|
||||||
<v-col v-else cols="9" style="margin: auto; text-align: center;">
|
<v-col v-else cols="9" style="margin: auto; text-align: center;">
|
||||||
|
|||||||
@@ -69,13 +69,13 @@
|
|||||||
</v-row>
|
</v-row>
|
||||||
<v-row v-if="!listItem.checked && recipeList && recipeList.length && displayRecipeRefs" no-gutters class="mb-2">
|
<v-row v-if="!listItem.checked && recipeList && recipeList.length && displayRecipeRefs" no-gutters class="mb-2">
|
||||||
<v-col cols="auto" style="width: 100%;">
|
<v-col cols="auto" style="width: 100%;">
|
||||||
<RecipeList :recipes="recipeList" :list-item="listItem" small tile />
|
<RecipeList :recipes="recipeList" :list-item="listItem" :disabled="$nuxt.isOffline" small tile />
|
||||||
</v-col>
|
</v-col>
|
||||||
</v-row>
|
</v-row>
|
||||||
<v-row v-if="listItem.checked" no-gutters class="mb-2">
|
<v-row v-if="listItem.checked" no-gutters class="mb-2">
|
||||||
<v-col cols="auto">
|
<v-col cols="auto">
|
||||||
<div class="text-caption font-weight-light font-italic">
|
<div class="text-caption font-weight-light font-italic">
|
||||||
{{ $t("shopping-list.completed-on", {date: new Date(listItem.updateAt+"Z").toLocaleDateString($i18n.locale)}) }}
|
{{ $t("shopping-list.completed-on", {date: new Date(listItem.updateAt || "").toLocaleDateString($i18n.locale)}) }}
|
||||||
</div>
|
</div>
|
||||||
</v-col>
|
</v-col>
|
||||||
</v-row>
|
</v-row>
|
||||||
@@ -135,7 +135,7 @@ export default defineComponent({
|
|||||||
recipes: {
|
recipes: {
|
||||||
type: Map<string, RecipeSummary>,
|
type: Map<string, RecipeSummary>,
|
||||||
default: undefined,
|
default: undefined,
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
setup(props, context) {
|
setup(props, context) {
|
||||||
const { i18n } = useContext();
|
const { i18n } = useContext();
|
||||||
|
|||||||
@@ -53,4 +53,3 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|||||||
@@ -157,9 +157,9 @@
|
|||||||
|
|
||||||
const topLinks = computed<SidebarLinks>(() => [
|
const topLinks = computed<SidebarLinks>(() => [
|
||||||
{
|
{
|
||||||
icon: $globals.icons.search,
|
icon: $globals.icons.silverwareForkKnife,
|
||||||
to: `/g/${groupSlug.value}`,
|
to: `/g/${groupSlug.value}`,
|
||||||
title: i18n.tc("sidebar.search"),
|
title: i18n.tc("general.recipes"),
|
||||||
restricted: true,
|
restricted: true,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -32,4 +32,3 @@ The Page folder is dedicated to 'single-use' component to break up large amounts
|
|||||||
- These are *last resort* components. Only to be used when the page becomes unmanageable.
|
- These are *last resort* components. Only to be used when the page becomes unmanageable.
|
||||||
- Page components should be prefixed with their page name
|
- Page components should be prefixed with their page name
|
||||||
- Examples: HomeAbout, HomeContact, ClientProfile
|
- Examples: HomeAbout, HomeContact, ClientProfile
|
||||||
|
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
<template>
|
<template>
|
||||||
<v-alert border="left" colored-border type="warning" elevation="2" :icon="$globals.icons.alert">
|
<BannerWarning
|
||||||
<b>{{ $t("banner-experimental.title") }}</b>
|
:title="$tc('banner-experimental.title')"
|
||||||
<div>{{ $t("banner-experimental.description") }}</div>
|
:description="$tc('banner-experimental.description')"
|
||||||
<div v-if="issue != ''" class="py-2">
|
>
|
||||||
|
<template v-if="issue" #default>
|
||||||
<a :href="issue" target="_blank">{{ $t("banner-experimental.issue-link-text") }}</a>
|
<a :href="issue" target="_blank">{{ $t("banner-experimental.issue-link-text") }}</a>
|
||||||
</div>
|
</template>
|
||||||
</v-alert>
|
</BannerWarning>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
|
|||||||
26
frontend/components/global/BannerWarning.vue
Normal file
26
frontend/components/global/BannerWarning.vue
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
<template>
|
||||||
|
<v-alert border="left" colored-border type="warning" elevation="2" :icon="$globals.icons.alert">
|
||||||
|
<b v-if="title">{{ title }}</b>
|
||||||
|
<div v-if="description">{{ description }}</div>
|
||||||
|
<div v-if="$slots.default" class="py-2">
|
||||||
|
<slot></slot>
|
||||||
|
</div>
|
||||||
|
</v-alert>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
title: {
|
||||||
|
type: String,
|
||||||
|
required: false,
|
||||||
|
default: "",
|
||||||
|
},
|
||||||
|
description: {
|
||||||
|
type: String,
|
||||||
|
required: false,
|
||||||
|
default: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
||||||
@@ -38,7 +38,7 @@
|
|||||||
</v-list-item-icon>
|
</v-list-item-icon>
|
||||||
<v-list-item-title>{{ item.text }}</v-list-item-title>
|
<v-list-item-title>{{ item.text }}</v-list-item-title>
|
||||||
</v-list-item>
|
</v-list-item>
|
||||||
<v-divider v-if="item.divider" :key="`divider-${index}`" class="my-1" ></v-divider>\
|
<v-divider v-if="item.divider" :key="`divider-${index}`" class="my-1" ></v-divider>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</v-list-item-group>
|
</v-list-item-group>
|
||||||
@@ -80,7 +80,7 @@ export interface MenuItem {
|
|||||||
value?: string;
|
value?: string;
|
||||||
event?: string;
|
event?: string;
|
||||||
divider?: boolean;
|
divider?: boolean;
|
||||||
hide?:boolean;
|
hide?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
|
|||||||
@@ -2,4 +2,5 @@ export { useFraction } from "./use-fraction";
|
|||||||
export { useRecipe } from "./use-recipe";
|
export { useRecipe } from "./use-recipe";
|
||||||
export { useRecipes, recentRecipes, allRecipes, useLazyRecipes } from "./use-recipes";
|
export { useRecipes, recentRecipes, allRecipes, useLazyRecipes } from "./use-recipes";
|
||||||
export { parseIngredientText, useParsedIngredientText } from "./use-recipe-ingredients";
|
export { parseIngredientText, useParsedIngredientText } from "./use-recipe-ingredients";
|
||||||
|
export { useNutritionLabels } from "./use-recipe-nutrition";
|
||||||
export { useTools } from "./use-recipe-tools";
|
export { useTools } from "./use-recipe-tools";
|
||||||
|
|||||||
@@ -31,13 +31,13 @@ describe(parseIngredientText.name, () => {
|
|||||||
test("ingredient text with fraction", () => {
|
test("ingredient text with fraction", () => {
|
||||||
const ingredient = createRecipeIngredient({ quantity: 1.5, unit: { fraction: true, id: "1", name: "cup" } });
|
const ingredient = createRecipeIngredient({ quantity: 1.5, unit: { fraction: true, id: "1", name: "cup" } });
|
||||||
|
|
||||||
expect(parseIngredientText(ingredient, false, 1, true)).contain("1 <sup>1</sup>").and.to.contain("<sub>2</sub>");
|
expect(parseIngredientText(ingredient, false, 1, true)).contain("1<sup>1</sup>").and.to.contain("<sub>2</sub>");
|
||||||
});
|
});
|
||||||
|
|
||||||
test("ingredient text with fraction when unit is null", () => {
|
test("ingredient text with fraction when unit is null", () => {
|
||||||
const ingredient = createRecipeIngredient({ quantity: 1.5, unit: undefined });
|
const ingredient = createRecipeIngredient({ quantity: 1.5, unit: undefined });
|
||||||
|
|
||||||
expect(parseIngredientText(ingredient, false, 1, true)).contain("1 <sup>1</sup>").and.to.contain("<sub>2</sub>");
|
expect(parseIngredientText(ingredient, false, 1, true)).contain("1<sup>1</sup>").and.to.contain("<sub>2</sub>");
|
||||||
});
|
});
|
||||||
|
|
||||||
test("ingredient text with fraction no formatting", () => {
|
test("ingredient text with fraction no formatting", () => {
|
||||||
|
|||||||
@@ -63,7 +63,7 @@ export function useParsedIngredientText(ingredient: RecipeIngredient, disableAmo
|
|||||||
|
|
||||||
if (fraction[1] > 0) {
|
if (fraction[1] > 0) {
|
||||||
returnQty += includeFormating ?
|
returnQty += includeFormating ?
|
||||||
` <sup>${fraction[1]}</sup>⁄<sub>${fraction[2]}</sub>` :
|
`<sup>${fraction[1]}</sup><span>⁄</span><sub>${fraction[2]}</sub>` :
|
||||||
` ${fraction[1]}/${fraction[2]}`;
|
` ${fraction[1]}/${fraction[2]}`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
47
frontend/composables/recipes/use-recipe-nutrition.ts
Normal file
47
frontend/composables/recipes/use-recipe-nutrition.ts
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
import { useContext } from "@nuxtjs/composition-api";
|
||||||
|
|
||||||
|
|
||||||
|
export interface NutritionLabelType {
|
||||||
|
[key: string]: {
|
||||||
|
label: string;
|
||||||
|
suffix: string;
|
||||||
|
value?: string;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
export function useNutritionLabels() {
|
||||||
|
const { i18n } = useContext();
|
||||||
|
const labels = <NutritionLabelType>{
|
||||||
|
calories: {
|
||||||
|
label: i18n.tc("recipe.calories"),
|
||||||
|
suffix: i18n.tc("recipe.calories-suffix"),
|
||||||
|
},
|
||||||
|
fatContent: {
|
||||||
|
label: i18n.tc("recipe.fat-content"),
|
||||||
|
suffix: i18n.tc("recipe.grams"),
|
||||||
|
},
|
||||||
|
fiberContent: {
|
||||||
|
label: i18n.tc("recipe.fiber-content"),
|
||||||
|
suffix: i18n.tc("recipe.grams"),
|
||||||
|
},
|
||||||
|
proteinContent: {
|
||||||
|
label: i18n.tc("recipe.protein-content"),
|
||||||
|
suffix: i18n.tc("recipe.grams"),
|
||||||
|
},
|
||||||
|
sodiumContent: {
|
||||||
|
label: i18n.tc("recipe.sodium-content"),
|
||||||
|
suffix: i18n.tc("recipe.milligrams"),
|
||||||
|
},
|
||||||
|
sugarContent: {
|
||||||
|
label: i18n.tc("recipe.sugar-content"),
|
||||||
|
suffix: i18n.tc("recipe.grams"),
|
||||||
|
},
|
||||||
|
carbohydrateContent: {
|
||||||
|
label: i18n.tc("recipe.carbohydrate-content"),
|
||||||
|
suffix: i18n.tc("recipe.grams"),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
return { labels }
|
||||||
|
}
|
||||||
@@ -19,6 +19,7 @@ export const useFoodData = function () {
|
|||||||
name: "",
|
name: "",
|
||||||
description: "",
|
description: "",
|
||||||
labelId: undefined,
|
labelId: undefined,
|
||||||
|
onHand: false,
|
||||||
});
|
});
|
||||||
|
|
||||||
function reset() {
|
function reset() {
|
||||||
@@ -26,6 +27,7 @@ export const useFoodData = function () {
|
|||||||
data.name = "";
|
data.name = "";
|
||||||
data.description = "";
|
data.description = "";
|
||||||
data.labelId = undefined;
|
data.labelId = undefined;
|
||||||
|
data.onHand = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|||||||
@@ -64,7 +64,6 @@ export const useGroupWebhooks = function () {
|
|||||||
newDt.setMinutes(Number(minutes));
|
newDt.setMinutes(Number(minutes));
|
||||||
|
|
||||||
updateData.scheduledTime = `${pad(newDt.getUTCHours(), 2)}:${pad(newDt.getUTCMinutes(), 2)}`;
|
updateData.scheduledTime = `${pad(newDt.getUTCHours(), 2)}:${pad(newDt.getUTCMinutes(), 2)}`;
|
||||||
console.log(updateData.scheduledTime);
|
|
||||||
|
|
||||||
const payload = {
|
const payload = {
|
||||||
...updateData,
|
...updateData,
|
||||||
@@ -85,7 +84,14 @@ export const useGroupWebhooks = function () {
|
|||||||
if (data) {
|
if (data) {
|
||||||
this.refreshAll();
|
this.refreshAll();
|
||||||
}
|
}
|
||||||
|
loading.value = false;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
async testOne(id: string | number) {
|
||||||
|
loading.value = true;
|
||||||
|
await api.groupWebhooks.testOne(id);
|
||||||
|
loading.value = false;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const webhooks = actions.getAll();
|
const webhooks = actions.getAll();
|
||||||
|
|||||||
223
frontend/composables/use-shopping-list-item-actions.ts
Normal file
223
frontend/composables/use-shopping-list-item-actions.ts
Normal file
@@ -0,0 +1,223 @@
|
|||||||
|
import { computed, reactive, watch } from "@nuxtjs/composition-api";
|
||||||
|
import { useLocalStorage } from "@vueuse/core";
|
||||||
|
import { useUserApi } from "~/composables/api";
|
||||||
|
import { ShoppingListItemOut, ShoppingListOut } from "~/lib/api/types/group";
|
||||||
|
import { RequestResponse } from "~/lib/api/types/non-generated";
|
||||||
|
|
||||||
|
const localStorageKey = "shopping-list-queue";
|
||||||
|
const queueTimeout = 5 * 60 * 1000; // 5 minutes
|
||||||
|
|
||||||
|
type ItemQueueType = "create" | "update" | "delete";
|
||||||
|
|
||||||
|
interface ShoppingListQueue {
|
||||||
|
create: ShoppingListItemOut[];
|
||||||
|
update: ShoppingListItemOut[];
|
||||||
|
delete: ShoppingListItemOut[];
|
||||||
|
|
||||||
|
lastUpdate: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface Storage {
|
||||||
|
[key: string]: ShoppingListQueue;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useShoppingListItemActions(shoppingListId: string) {
|
||||||
|
const api = useUserApi();
|
||||||
|
const storage = useLocalStorage(localStorageKey, {} as Storage, { deep: true });
|
||||||
|
const queue = reactive(getQueue());
|
||||||
|
const queueEmpty = computed(() => !queue.create.length && !queue.update.length && !queue.delete.length);
|
||||||
|
if (queueEmpty.value) {
|
||||||
|
queue.lastUpdate = Date.now();
|
||||||
|
}
|
||||||
|
|
||||||
|
storage.value[shoppingListId] = { ...queue }
|
||||||
|
watch(
|
||||||
|
() => queue,
|
||||||
|
(value) => {
|
||||||
|
storage.value[shoppingListId] = { ...value }
|
||||||
|
},
|
||||||
|
{
|
||||||
|
deep: true,
|
||||||
|
immediate: true,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
function isValidQueueObject(obj: any): obj is ShoppingListQueue {
|
||||||
|
if (typeof obj !== "object" || obj === null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const hasRequiredProps = "create" in obj && "update" in obj && "delete" in obj && "lastUpdate" in obj;
|
||||||
|
if (!hasRequiredProps) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const arraysValid = Array.isArray(obj.create) && Array.isArray(obj.update) && Array.isArray(obj.delete);
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-argument
|
||||||
|
const lastUpdateValid = typeof obj.lastUpdate === "number" && !isNaN(new Date(obj.lastUpdate).getTime());
|
||||||
|
|
||||||
|
return arraysValid && lastUpdateValid;
|
||||||
|
}
|
||||||
|
|
||||||
|
function createEmptyQueue(): ShoppingListQueue {
|
||||||
|
const newQueue = { create: [], update: [], delete: [], lastUpdate: Date.now() };
|
||||||
|
return newQueue;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getQueue(): ShoppingListQueue {
|
||||||
|
try {
|
||||||
|
const fetchedQueue = storage.value[shoppingListId];
|
||||||
|
if (!isValidQueueObject(fetchedQueue)) {
|
||||||
|
console.log("Invalid queue object in local storage; resetting queue.");
|
||||||
|
return createEmptyQueue();
|
||||||
|
} else {
|
||||||
|
return fetchedQueue;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.log("Error validating queue object in local storage; resetting queue.", error);
|
||||||
|
return createEmptyQueue();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeFromQueue(itemQueue: ShoppingListItemOut[], item: ShoppingListItemOut): boolean {
|
||||||
|
const index = itemQueue.findIndex(i => i.id === item.id);
|
||||||
|
if (index === -1) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
itemQueue.splice(index, 1);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getList() {
|
||||||
|
const response = await api.shopping.lists.getOne(shoppingListId);
|
||||||
|
return response.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
function createItem(item: ShoppingListItemOut) {
|
||||||
|
removeFromQueue(queue.create, item);
|
||||||
|
queue.create.push(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateItem(item: ShoppingListItemOut) {
|
||||||
|
const removedFromCreate = removeFromQueue(queue.create, item);
|
||||||
|
if (removedFromCreate) {
|
||||||
|
// this item hasn't been created yet, so we don't need to update it
|
||||||
|
queue.create.push(item);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
removeFromQueue(queue.update, item);
|
||||||
|
queue.update.push(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
function deleteItem(item: ShoppingListItemOut) {
|
||||||
|
const removedFromCreate = removeFromQueue(queue.create, item);
|
||||||
|
if (removedFromCreate) {
|
||||||
|
// this item hasn't been created yet, so we don't need to delete it
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
removeFromQueue(queue.update, item);
|
||||||
|
removeFromQueue(queue.delete, item);
|
||||||
|
queue.delete.push(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getQueueItems(itemQueueType: ItemQueueType) {
|
||||||
|
return queue[itemQueueType];
|
||||||
|
}
|
||||||
|
|
||||||
|
function clearQueueItems(itemQueueType: ItemQueueType | "all", itemIds: string[] | null = null) {
|
||||||
|
if (itemQueueType === "create" || itemQueueType === "all") {
|
||||||
|
queue.create = itemIds ? queue.create.filter(item => !itemIds.includes(item.id)) : [];
|
||||||
|
}
|
||||||
|
if (itemQueueType === "update" || itemQueueType === "all") {
|
||||||
|
queue.update = itemIds ? queue.update.filter(item => !itemIds.includes(item.id)) : [];
|
||||||
|
}
|
||||||
|
if (itemQueueType === "delete" || itemQueueType === "all") {
|
||||||
|
queue.delete = itemIds ? queue.delete.filter(item => !itemIds.includes(item.id)) : [];
|
||||||
|
}
|
||||||
|
if (queueEmpty.value) {
|
||||||
|
queue.lastUpdate = Date.now();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function checkUpdateState(list: ShoppingListOut) {
|
||||||
|
const cutoffDate = new Date(queue.lastUpdate + queueTimeout).toISOString();
|
||||||
|
if (list.updateAt && list.updateAt > cutoffDate) {
|
||||||
|
// If the queue is too far behind the shopping list to reliably do updates, we clear the queue
|
||||||
|
console.log("Out of sync with server; clearing queue");
|
||||||
|
clearQueueItems("all");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Processes the queue items and returns whether the processing was successful.
|
||||||
|
*/
|
||||||
|
async function processQueueItems(
|
||||||
|
action: (items: ShoppingListItemOut[]) => Promise<RequestResponse<any>>,
|
||||||
|
itemQueueType: ItemQueueType,
|
||||||
|
): Promise<boolean> {
|
||||||
|
let queueItems: ShoppingListItemOut[];
|
||||||
|
try {
|
||||||
|
queueItems = getQueueItems(itemQueueType);
|
||||||
|
if (!queueItems.length) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.log(`Error fetching queue items of type ${itemQueueType}:`, error);
|
||||||
|
clearQueueItems(itemQueueType);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const itemsToProcess = [...queueItems];
|
||||||
|
await action(itemsToProcess)
|
||||||
|
.then(() => {
|
||||||
|
if (window.$nuxt.isOnline) {
|
||||||
|
clearQueueItems(itemQueueType, itemsToProcess.map(item => item.id));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.log(`Error processing queue items of type ${itemQueueType}:`, error);
|
||||||
|
clearQueueItems(itemQueueType);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function process() {
|
||||||
|
if(queueEmpty.value) {
|
||||||
|
queue.lastUpdate = Date.now();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await getList();
|
||||||
|
if (!data) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
checkUpdateState(data);
|
||||||
|
|
||||||
|
// We send each bulk request one at a time, since the backend may merge items
|
||||||
|
// "failures" here refers to an actual error, rather than failing to reach the backend
|
||||||
|
let failures = 0;
|
||||||
|
if (!(await processQueueItems((items) => api.shopping.items.deleteMany(items), "delete"))) failures++;
|
||||||
|
if (!(await processQueueItems((items) => api.shopping.items.updateMany(items), "update"))) failures++;
|
||||||
|
if (!(await processQueueItems((items) => api.shopping.items.createMany(items), "create"))) failures++;
|
||||||
|
|
||||||
|
// If we're online, or the queue is empty, the queue is fully processed, so we're up to date
|
||||||
|
// Otherwise, if all three queue processes failed, we've already reset the queue, so we need to reset the date
|
||||||
|
if (window.$nuxt.isOnline || queueEmpty.value || failures === 3) {
|
||||||
|
queue.lastUpdate = Date.now();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
getList,
|
||||||
|
createItem,
|
||||||
|
updateItem,
|
||||||
|
deleteItem,
|
||||||
|
process,
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -6,6 +6,7 @@ export interface UserPrintPreferences {
|
|||||||
imagePosition: string;
|
imagePosition: string;
|
||||||
showDescription: boolean;
|
showDescription: boolean;
|
||||||
showNotes: boolean;
|
showNotes: boolean;
|
||||||
|
showNutrition: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface UserSearchQuery {
|
export interface UserSearchQuery {
|
||||||
@@ -18,6 +19,10 @@ export enum ImagePosition {
|
|||||||
right = "right",
|
right = "right",
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface UserMealPlanPreferences {
|
||||||
|
numberOfDays: number;
|
||||||
|
}
|
||||||
|
|
||||||
export interface UserRecipePreferences {
|
export interface UserRecipePreferences {
|
||||||
orderBy: string;
|
orderBy: string;
|
||||||
orderDirection: string;
|
orderDirection: string;
|
||||||
@@ -40,6 +45,20 @@ export interface UserParsingPreferences {
|
|||||||
parser: RegisteredParser;
|
parser: RegisteredParser;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function useUserMealPlanPreferences(): Ref<UserMealPlanPreferences> {
|
||||||
|
const fromStorage = useLocalStorage(
|
||||||
|
"meal-planner-preferences",
|
||||||
|
{
|
||||||
|
numberOfDays: 7,
|
||||||
|
},
|
||||||
|
{ mergeDefaults: true }
|
||||||
|
// we cast to a Ref because by default it will return an optional type ref
|
||||||
|
// but since we pass defaults we know all properties are set.
|
||||||
|
) as unknown as Ref<UserMealPlanPreferences>;
|
||||||
|
|
||||||
|
return fromStorage;
|
||||||
|
}
|
||||||
|
|
||||||
export function useUserPrintPreferences(): Ref<UserPrintPreferences> {
|
export function useUserPrintPreferences(): Ref<UserPrintPreferences> {
|
||||||
const fromStorage = useLocalStorage(
|
const fromStorage = useLocalStorage(
|
||||||
"recipe-print-preferences",
|
"recipe-print-preferences",
|
||||||
|
|||||||
@@ -210,7 +210,8 @@
|
|||||||
"unsaved-changes": "You have unsaved changes. Do you want to save before leaving? Okay to save, Cancel to discard changes.",
|
"unsaved-changes": "You have unsaved changes. Do you want to save before leaving? Okay to save, Cancel to discard changes.",
|
||||||
"clipboard-copy-failure": "Failed to copy to the clipboard.",
|
"clipboard-copy-failure": "Failed to copy to the clipboard.",
|
||||||
"confirm-delete-generic-items": "Are you sure you want to delete the following items?",
|
"confirm-delete-generic-items": "Are you sure you want to delete the following items?",
|
||||||
"organizers": "Organizers"
|
"organizers": "Organizers",
|
||||||
|
"caution": "Caution"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "Is jy seker jy wil <b>{groupName}<b/> uitvee?",
|
"are-you-sure-you-want-to-delete-the-group": "Is jy seker jy wil <b>{groupName}<b/> uitvee?",
|
||||||
@@ -291,6 +292,8 @@
|
|||||||
"mealplan-updated": "Maaltydplan opgedateer",
|
"mealplan-updated": "Maaltydplan opgedateer",
|
||||||
"no-meal-plan-defined-yet": "Nog geen maaltydplan opgestel nie",
|
"no-meal-plan-defined-yet": "Nog geen maaltydplan opgestel nie",
|
||||||
"no-meal-planned-for-today": "Geen maaltyd beplan vir vandag nie",
|
"no-meal-planned-for-today": "Geen maaltyd beplan vir vandag nie",
|
||||||
|
"numberOfDays-hint": "Number of days on page load",
|
||||||
|
"numberOfDays-label": "Default Days",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Slegs resepte met hierdie kategorieë sal in maaltydplanne gebruik word",
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Slegs resepte met hierdie kategorieë sal in maaltydplanne gebruik word",
|
||||||
"planner": "Beplanner",
|
"planner": "Beplanner",
|
||||||
"quick-week": "Vinnige week",
|
"quick-week": "Vinnige week",
|
||||||
@@ -377,6 +380,10 @@
|
|||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "My Recipe Box",
|
"title": "My Recipe Box",
|
||||||
"description-long": "Mealie can import recipes from My Recipe Box. Export your recipes in CSV format, then upload the .csv file below."
|
"description-long": "Mealie can import recipes from My Recipe Box. Export your recipes in CSV format, then upload the .csv file below."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Mealie can import recipes from Recipe Keeper. Export your recipes in zip format, then upload the .zip file below."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "Bestanddele",
|
"ingredients": "Bestanddele",
|
||||||
"insert-ingredient": "Voeg bestanddeel in",
|
"insert-ingredient": "Voeg bestanddeel in",
|
||||||
"insert-section": "Voeg bestanddeel in",
|
"insert-section": "Voeg bestanddeel in",
|
||||||
|
"insert-above": "Insert Above",
|
||||||
|
"insert-below": "Insert Below",
|
||||||
"instructions": "Instruksies",
|
"instructions": "Instruksies",
|
||||||
"key-name-required": "Sleutelnaam word vereis",
|
"key-name-required": "Sleutelnaam word vereis",
|
||||||
"landscape-view-coming-soon": "Landscape View (Coming Soon)",
|
"landscape-view-coming-soon": "Landscape View (Coming Soon)",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "Kon nie verslag uitvee nie",
|
"report-deletion-failed": "Kon nie verslag uitvee nie",
|
||||||
"recipe-debugger": "Resep debugger",
|
"recipe-debugger": "Resep debugger",
|
||||||
"recipe-debugger-description": "Gryp die URL van die resep wat jy wil debug en plak dit hier. Die URL sal deur die resepskraper geskraap word en die resultate sal vertoon word. As jy nie enige data terugstuur sien nie, word die webwerf wat jy probeer skraap nie deur Mealie of sy skraperbiblioteek ondersteun nie.",
|
"recipe-debugger-description": "Gryp die URL van die resep wat jy wil debug en plak dit hier. Die URL sal deur die resepskraper geskraap word en die resultate sal vertoon word. As jy nie enige data terugstuur sien nie, word die webwerf wat jy probeer skraap nie deur Mealie of sy skraperbiblioteek ondersteun nie.",
|
||||||
|
"use-openai": "Use OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Use OpenAI to parse the results instead of relying on the scraper library. When creating a recipe via URL, this is done automatically if the scraper library fails, but you may test it manually here.",
|
||||||
"debug": "Debug",
|
"debug": "Debug",
|
||||||
"tree-view": "Boomstruktuur",
|
"tree-view": "Boomstruktuur",
|
||||||
"recipe-yield": "Resep opbrengs",
|
"recipe-yield": "Resep opbrengs",
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "Back-up gemaak op {path}",
|
"backup-created-at-response-export_path": "Back-up gemaak op {path}",
|
||||||
"backup-deleted": "Back-up verwyder",
|
"backup-deleted": "Back-up verwyder",
|
||||||
"restore-success": "Restore successful",
|
"restore-success": "Restore successful",
|
||||||
|
"restore-fail": "Restore failed. Check your server logs for more details",
|
||||||
"backup-tag": "Back-up merker",
|
"backup-tag": "Back-up merker",
|
||||||
"create-heading": "Maak 'n back-up",
|
"create-heading": "Maak 'n back-up",
|
||||||
"delete-backup": "Verwyder back-up",
|
"delete-backup": "Verwyder back-up",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "Geen gekoppelde resepte|Een gekoppelde resep|{count} gekoppelde resepte",
|
"linked-recipes-count": "Geen gekoppelde resepte|Een gekoppelde resep|{count} gekoppelde resepte",
|
||||||
"items-checked-count": "Geen items gemerk|Een item gemerk|{count} items gemerk",
|
"items-checked-count": "Geen items gemerk|Een item gemerk|{count} items gemerk",
|
||||||
"no-label": "Geen etiket nie",
|
"no-label": "Geen etiket nie",
|
||||||
"completed-on": "Voltooi op {date}"
|
"completed-on": "Voltooi op {date}",
|
||||||
|
"you-are-offline": "You are offline",
|
||||||
|
"you-are-offline-description": "Not all features are available while offline. You can still add, modify, and remove items, but you will not be able to sync your changes to the server until you are back online.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "Are you sure you want to check all items?",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "Are you sure you want to uncheck all items?",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "Are you sure you want to delete all checked items?"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "Alle resepte",
|
"all-recipes": "Alle resepte",
|
||||||
@@ -941,7 +958,7 @@
|
|||||||
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie",
|
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie",
|
||||||
"permissions": "Permissies",
|
"permissions": "Permissies",
|
||||||
"administrator": "Administrateur",
|
"administrator": "Administrateur",
|
||||||
"user-can-invite-other-to-group": "Gebruiker kan ander na groep nooi",
|
"user-can-invite-other-to-group": "User can invite others to group",
|
||||||
"user-can-manage-group": "Gebruiker kan groep bestuur",
|
"user-can-manage-group": "Gebruiker kan groep bestuur",
|
||||||
"user-can-organize-group-data": "Gebruiker kan groepdata organiseer",
|
"user-can-organize-group-data": "Gebruiker kan groepdata organiseer",
|
||||||
"enable-advanced-features": "Aktiveer gevorderde funksies",
|
"enable-advanced-features": "Aktiveer gevorderde funksies",
|
||||||
@@ -972,7 +989,9 @@
|
|||||||
"edit-food": "Wysig kos",
|
"edit-food": "Wysig kos",
|
||||||
"food-data": "Voedseldata",
|
"food-data": "Voedseldata",
|
||||||
"example-food-singular": "ex: Onion",
|
"example-food-singular": "ex: Onion",
|
||||||
"example-food-plural": "ex: Onions"
|
"example-food-plural": "ex: Onions",
|
||||||
|
"label-overwrite-warning": "This will assign the chosen label to all selected foods and potentially overwrite your existing labels.",
|
||||||
|
"on-hand-checkbox-label": "Setting this flag will make this food unchecked by default when adding a recipe to a shopping list."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Saai die databasis met algemene eenhede gebaseer op jou plaaslike taal.",
|
"seed-dialog-text": "Saai die databasis met algemene eenhede gebaseer op jou plaaslike taal.",
|
||||||
@@ -1000,7 +1019,8 @@
|
|||||||
"seed-dialog-text": "Vul die databasis met algemene etikette gebaseer op jou plaaslike taal.",
|
"seed-dialog-text": "Vul die databasis met algemene etikette gebaseer op jou plaaslike taal.",
|
||||||
"edit-label": "Wysig etiket",
|
"edit-label": "Wysig etiket",
|
||||||
"new-label": "Nuwe etiket",
|
"new-label": "Nuwe etiket",
|
||||||
"labels": "Etikette"
|
"labels": "Etikette",
|
||||||
|
"assign-label": "Assign Label"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Verwyder uitvoerlêers",
|
"purge-exports": "Verwyder uitvoerlêers",
|
||||||
|
|||||||
@@ -210,7 +210,8 @@
|
|||||||
"unsaved-changes": "You have unsaved changes. Do you want to save before leaving? Okay to save, Cancel to discard changes.",
|
"unsaved-changes": "You have unsaved changes. Do you want to save before leaving? Okay to save, Cancel to discard changes.",
|
||||||
"clipboard-copy-failure": "Failed to copy to the clipboard.",
|
"clipboard-copy-failure": "Failed to copy to the clipboard.",
|
||||||
"confirm-delete-generic-items": "Are you sure you want to delete the following items?",
|
"confirm-delete-generic-items": "Are you sure you want to delete the following items?",
|
||||||
"organizers": "Organizers"
|
"organizers": "Organizers",
|
||||||
|
"caution": "Caution"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "هل انت متأكد من رغبتك في حذف <b>{groupName}<b/>؟",
|
"are-you-sure-you-want-to-delete-the-group": "هل انت متأكد من رغبتك في حذف <b>{groupName}<b/>؟",
|
||||||
@@ -291,6 +292,8 @@
|
|||||||
"mealplan-updated": "تم تحديث خطة الوجبات",
|
"mealplan-updated": "تم تحديث خطة الوجبات",
|
||||||
"no-meal-plan-defined-yet": "لم يتم تحديد خطة بعد",
|
"no-meal-plan-defined-yet": "لم يتم تحديد خطة بعد",
|
||||||
"no-meal-planned-for-today": "لم يتم تخطيط وجبة لهذا اليوم",
|
"no-meal-planned-for-today": "لم يتم تخطيط وجبة لهذا اليوم",
|
||||||
|
"numberOfDays-hint": "Number of days on page load",
|
||||||
|
"numberOfDays-label": "Default Days",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "فقط الوجبات التي تحتوي على التصنيفات التالية سوف تستخدم لإنشاء خطتك",
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "فقط الوجبات التي تحتوي على التصنيفات التالية سوف تستخدم لإنشاء خطتك",
|
||||||
"planner": "المخطط",
|
"planner": "المخطط",
|
||||||
"quick-week": "Quick Week",
|
"quick-week": "Quick Week",
|
||||||
@@ -377,6 +380,10 @@
|
|||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "My Recipe Box",
|
"title": "My Recipe Box",
|
||||||
"description-long": "Mealie can import recipes from My Recipe Box. Export your recipes in CSV format, then upload the .csv file below."
|
"description-long": "Mealie can import recipes from My Recipe Box. Export your recipes in CSV format, then upload the .csv file below."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Mealie can import recipes from Recipe Keeper. Export your recipes in zip format, then upload the .zip file below."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "المكونات",
|
"ingredients": "المكونات",
|
||||||
"insert-ingredient": "Insert Ingredient",
|
"insert-ingredient": "Insert Ingredient",
|
||||||
"insert-section": "ادراج قسم",
|
"insert-section": "ادراج قسم",
|
||||||
|
"insert-above": "Insert Above",
|
||||||
|
"insert-below": "Insert Below",
|
||||||
"instructions": "التعليمات",
|
"instructions": "التعليمات",
|
||||||
"key-name-required": "Key Name Required",
|
"key-name-required": "Key Name Required",
|
||||||
"landscape-view-coming-soon": "Landscape View (Coming Soon)",
|
"landscape-view-coming-soon": "Landscape View (Coming Soon)",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "Report deletion failed",
|
"report-deletion-failed": "Report deletion failed",
|
||||||
"recipe-debugger": "Recipe Debugger",
|
"recipe-debugger": "Recipe Debugger",
|
||||||
"recipe-debugger-description": "Grab the URL of the recipe you want to debug and paste it here. The URL will be scraped by the recipe scraper and the results will be displayed. If you don't see any data returned, the site you are trying to scrape is not supported by Mealie or its scraper library.",
|
"recipe-debugger-description": "Grab the URL of the recipe you want to debug and paste it here. The URL will be scraped by the recipe scraper and the results will be displayed. If you don't see any data returned, the site you are trying to scrape is not supported by Mealie or its scraper library.",
|
||||||
|
"use-openai": "Use OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Use OpenAI to parse the results instead of relying on the scraper library. When creating a recipe via URL, this is done automatically if the scraper library fails, but you may test it manually here.",
|
||||||
"debug": "Debug",
|
"debug": "Debug",
|
||||||
"tree-view": "Tree View",
|
"tree-view": "Tree View",
|
||||||
"recipe-yield": "Recipe Yield",
|
"recipe-yield": "Recipe Yield",
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "Backup Created at {path}",
|
"backup-created-at-response-export_path": "Backup Created at {path}",
|
||||||
"backup-deleted": "Backup deleted",
|
"backup-deleted": "Backup deleted",
|
||||||
"restore-success": "Restore successful",
|
"restore-success": "Restore successful",
|
||||||
|
"restore-fail": "Restore failed. Check your server logs for more details",
|
||||||
"backup-tag": "Backup Tag",
|
"backup-tag": "Backup Tag",
|
||||||
"create-heading": "Create a Backup",
|
"create-heading": "Create a Backup",
|
||||||
"delete-backup": "Delete Backup",
|
"delete-backup": "Delete Backup",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "No Linked Recipes|One Linked Recipe|{count} Linked Recipes",
|
"linked-recipes-count": "No Linked Recipes|One Linked Recipe|{count} Linked Recipes",
|
||||||
"items-checked-count": "No items checked|One item checked|{count} items checked",
|
"items-checked-count": "No items checked|One item checked|{count} items checked",
|
||||||
"no-label": "No Label",
|
"no-label": "No Label",
|
||||||
"completed-on": "Completed on {date}"
|
"completed-on": "Completed on {date}",
|
||||||
|
"you-are-offline": "You are offline",
|
||||||
|
"you-are-offline-description": "Not all features are available while offline. You can still add, modify, and remove items, but you will not be able to sync your changes to the server until you are back online.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "Are you sure you want to check all items?",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "Are you sure you want to uncheck all items?",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "Are you sure you want to delete all checked items?"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "All Recipes",
|
"all-recipes": "All Recipes",
|
||||||
@@ -941,7 +958,7 @@
|
|||||||
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie",
|
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie",
|
||||||
"permissions": "Permissions",
|
"permissions": "Permissions",
|
||||||
"administrator": "Administrator",
|
"administrator": "Administrator",
|
||||||
"user-can-invite-other-to-group": "User can invite other to group",
|
"user-can-invite-other-to-group": "User can invite others to group",
|
||||||
"user-can-manage-group": "User can manage group",
|
"user-can-manage-group": "User can manage group",
|
||||||
"user-can-organize-group-data": "User can organize group data",
|
"user-can-organize-group-data": "User can organize group data",
|
||||||
"enable-advanced-features": "Enable advanced features",
|
"enable-advanced-features": "Enable advanced features",
|
||||||
@@ -972,7 +989,9 @@
|
|||||||
"edit-food": "Edit Food",
|
"edit-food": "Edit Food",
|
||||||
"food-data": "Food Data",
|
"food-data": "Food Data",
|
||||||
"example-food-singular": "ex: Onion",
|
"example-food-singular": "ex: Onion",
|
||||||
"example-food-plural": "ex: Onions"
|
"example-food-plural": "ex: Onions",
|
||||||
|
"label-overwrite-warning": "This will assign the chosen label to all selected foods and potentially overwrite your existing labels.",
|
||||||
|
"on-hand-checkbox-label": "Setting this flag will make this food unchecked by default when adding a recipe to a shopping list."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Seed the database with common units based on your local language.",
|
"seed-dialog-text": "Seed the database with common units based on your local language.",
|
||||||
@@ -1000,7 +1019,8 @@
|
|||||||
"seed-dialog-text": "Seed the database with common labels based on your local language.",
|
"seed-dialog-text": "Seed the database with common labels based on your local language.",
|
||||||
"edit-label": "Edit Label",
|
"edit-label": "Edit Label",
|
||||||
"new-label": "New Label",
|
"new-label": "New Label",
|
||||||
"labels": "Labels"
|
"labels": "Labels",
|
||||||
|
"assign-label": "Assign Label"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Purge Exports",
|
"purge-exports": "Purge Exports",
|
||||||
|
|||||||
@@ -210,7 +210,8 @@
|
|||||||
"unsaved-changes": "Имате незапазени промени. Желаете ли да ги запазите преди да излезете? Натиснете Ок за запазване и Отказ за отхвърляне на промените.",
|
"unsaved-changes": "Имате незапазени промени. Желаете ли да ги запазите преди да излезете? Натиснете Ок за запазване и Отказ за отхвърляне на промените.",
|
||||||
"clipboard-copy-failure": "Линкът към рецептата е копиран в клипборда.",
|
"clipboard-copy-failure": "Линкът към рецептата е копиран в клипборда.",
|
||||||
"confirm-delete-generic-items": "Сигурни ли сте, че желаете да изтриете следните елементи?",
|
"confirm-delete-generic-items": "Сигурни ли сте, че желаете да изтриете следните елементи?",
|
||||||
"organizers": "Органайзер"
|
"organizers": "Органайзер",
|
||||||
|
"caution": "Caution"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "Сигурни ли сте, че искате да изтриете <b>{groupName}<b/>?",
|
"are-you-sure-you-want-to-delete-the-group": "Сигурни ли сте, че искате да изтриете <b>{groupName}<b/>?",
|
||||||
@@ -291,6 +292,8 @@
|
|||||||
"mealplan-updated": "Седмичното меню бе обновено",
|
"mealplan-updated": "Седмичното меню бе обновено",
|
||||||
"no-meal-plan-defined-yet": "Все още няма създадено седмично меню",
|
"no-meal-plan-defined-yet": "Все още няма създадено седмично меню",
|
||||||
"no-meal-planned-for-today": "За днес няма планирано меню",
|
"no-meal-planned-for-today": "За днес няма планирано меню",
|
||||||
|
"numberOfDays-hint": "Number of days on page load",
|
||||||
|
"numberOfDays-label": "Default Days",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Само рецептите от тези категории ще бъдат използвани в хранителните планове",
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Само рецептите от тези категории ще бъдат използвани в хранителните планове",
|
||||||
"planner": "Планьор",
|
"planner": "Планьор",
|
||||||
"quick-week": "Бърза седмица",
|
"quick-week": "Бърза седмица",
|
||||||
@@ -377,6 +380,10 @@
|
|||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "Моята кутия с рецепти",
|
"title": "Моята кутия с рецепти",
|
||||||
"description-long": "Mealie може да импортира рецепти от Моята кутия за рецепти. Експортирайте рецептите си в CSV формат, после качете .csv файлът по-долу."
|
"description-long": "Mealie може да импортира рецепти от Моята кутия за рецепти. Експортирайте рецептите си в CSV формат, после качете .csv файлът по-долу."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Mealie can import recipes from Recipe Keeper. Export your recipes in zip format, then upload the .zip file below."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "Съставки",
|
"ingredients": "Съставки",
|
||||||
"insert-ingredient": "Въведете съставка",
|
"insert-ingredient": "Въведете съставка",
|
||||||
"insert-section": "Въведете раздел",
|
"insert-section": "Въведете раздел",
|
||||||
|
"insert-above": "Insert Above",
|
||||||
|
"insert-below": "Insert Below",
|
||||||
"instructions": "Инструкции",
|
"instructions": "Инструкции",
|
||||||
"key-name-required": "Ключовото име е задължително",
|
"key-name-required": "Ключовото име е задължително",
|
||||||
"landscape-view-coming-soon": "Пейзажен изглед",
|
"landscape-view-coming-soon": "Пейзажен изглед",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "Неуспешно изтриване на доклад",
|
"report-deletion-failed": "Неуспешно изтриване на доклад",
|
||||||
"recipe-debugger": "Debugger на рецепти",
|
"recipe-debugger": "Debugger на рецепти",
|
||||||
"recipe-debugger-description": "Вземете URL на рецептата, която желаете да проверите за грешки и го поставете тук. URL ще бъде обходен и резултатите ще бъдат визуализирани. Ако не виждате върнати данни, сайтът който се опитвате да обходите не се поддържа от Mealie или библиотеката за обхождане.",
|
"recipe-debugger-description": "Вземете URL на рецептата, която желаете да проверите за грешки и го поставете тук. URL ще бъде обходен и резултатите ще бъдат визуализирани. Ако не виждате върнати данни, сайтът който се опитвате да обходите не се поддържа от Mealie или библиотеката за обхождане.",
|
||||||
|
"use-openai": "Use OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Use OpenAI to parse the results instead of relying on the scraper library. When creating a recipe via URL, this is done automatically if the scraper library fails, but you may test it manually here.",
|
||||||
"debug": "Отстраняване на грешки",
|
"debug": "Отстраняване на грешки",
|
||||||
"tree-view": "Дървовиден изглед",
|
"tree-view": "Дървовиден изглед",
|
||||||
"recipe-yield": "Добиване от рецепта",
|
"recipe-yield": "Добиване от рецепта",
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "Резервно копие е създадено на {path}",
|
"backup-created-at-response-export_path": "Резервно копие е създадено на {path}",
|
||||||
"backup-deleted": "Резервното копие е изтрито",
|
"backup-deleted": "Резервното копие е изтрито",
|
||||||
"restore-success": "Успешно възстановяване",
|
"restore-success": "Успешно възстановяване",
|
||||||
|
"restore-fail": "Restore failed. Check your server logs for more details",
|
||||||
"backup-tag": "Етикет на резервното копие",
|
"backup-tag": "Етикет на резервното копие",
|
||||||
"create-heading": "Създай резервно копие",
|
"create-heading": "Създай резервно копие",
|
||||||
"delete-backup": "Изтрий резервно копие",
|
"delete-backup": "Изтрий резервно копие",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "Няма свързани рецепти|Една свързана рецепта|{count} свързани рецепти",
|
"linked-recipes-count": "Няма свързани рецепти|Една свързана рецепта|{count} свързани рецепти",
|
||||||
"items-checked-count": "Няма отбелязани етикети|Един елемент е отбелязан|{count} елементи са отбелязани",
|
"items-checked-count": "Няма отбелязани етикети|Един елемент е отбелязан|{count} елементи са отбелязани",
|
||||||
"no-label": "Няма етикет",
|
"no-label": "Няма етикет",
|
||||||
"completed-on": "Приключена на {date}"
|
"completed-on": "Приключена на {date}",
|
||||||
|
"you-are-offline": "You are offline",
|
||||||
|
"you-are-offline-description": "Not all features are available while offline. You can still add, modify, and remove items, but you will not be able to sync your changes to the server until you are back online.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "Are you sure you want to check all items?",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "Are you sure you want to uncheck all items?",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "Are you sure you want to delete all checked items?"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "Всички рецепти",
|
"all-recipes": "Всички рецепти",
|
||||||
@@ -941,7 +958,7 @@
|
|||||||
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie",
|
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie",
|
||||||
"permissions": "Права",
|
"permissions": "Права",
|
||||||
"administrator": "Администратор",
|
"administrator": "Администратор",
|
||||||
"user-can-invite-other-to-group": "Потребителя може да добавя други в групата",
|
"user-can-invite-other-to-group": "User can invite others to group",
|
||||||
"user-can-manage-group": "Потребителя може да управлява групата",
|
"user-can-manage-group": "Потребителя може да управлява групата",
|
||||||
"user-can-organize-group-data": "Потребителя може да организира данните на групата",
|
"user-can-organize-group-data": "Потребителя може да организира данните на групата",
|
||||||
"enable-advanced-features": "Включване на разширени функции",
|
"enable-advanced-features": "Включване на разширени функции",
|
||||||
@@ -972,7 +989,9 @@
|
|||||||
"edit-food": "Редактирай храна",
|
"edit-food": "Редактирай храна",
|
||||||
"food-data": "Данни за храните",
|
"food-data": "Данни за храните",
|
||||||
"example-food-singular": "пример: Домат",
|
"example-food-singular": "пример: Домат",
|
||||||
"example-food-plural": "пример: Домати"
|
"example-food-plural": "пример: Домати",
|
||||||
|
"label-overwrite-warning": "This will assign the chosen label to all selected foods and potentially overwrite your existing labels.",
|
||||||
|
"on-hand-checkbox-label": "Setting this flag will make this food unchecked by default when adding a recipe to a shopping list."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Заредете базата данни с мерни единици на Вашия местен език.",
|
"seed-dialog-text": "Заредете базата данни с мерни единици на Вашия местен език.",
|
||||||
@@ -1000,7 +1019,8 @@
|
|||||||
"seed-dialog-text": "Заредете базата данни с етикети на Вашия местен език.",
|
"seed-dialog-text": "Заредете базата данни с етикети на Вашия местен език.",
|
||||||
"edit-label": "Редактиране на етикет",
|
"edit-label": "Редактиране на етикет",
|
||||||
"new-label": "Нов етикет",
|
"new-label": "Нов етикет",
|
||||||
"labels": "Етикети"
|
"labels": "Етикети",
|
||||||
|
"assign-label": "Assign Label"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Изчистване на експортите",
|
"purge-exports": "Изчистване на експортите",
|
||||||
|
|||||||
@@ -64,7 +64,7 @@
|
|||||||
"something-went-wrong": "Alguna cosa ha anat malament!",
|
"something-went-wrong": "Alguna cosa ha anat malament!",
|
||||||
"subscribed-events": "Esdeveniments subscrits",
|
"subscribed-events": "Esdeveniments subscrits",
|
||||||
"test-message-sent": "S'ha enviat el missatge",
|
"test-message-sent": "S'ha enviat el missatge",
|
||||||
"message-sent": "Message Sent",
|
"message-sent": "Missatge enviat",
|
||||||
"new-notification": "Nova notificació",
|
"new-notification": "Nova notificació",
|
||||||
"event-notifiers": "Notificacions d'esdeveniments",
|
"event-notifiers": "Notificacions d'esdeveniments",
|
||||||
"apprise-url-skipped-if-blank": "Apprise URL (si es deixa buit, s'ignorarà)",
|
"apprise-url-skipped-if-blank": "Apprise URL (si es deixa buit, s'ignorarà)",
|
||||||
@@ -81,7 +81,7 @@
|
|||||||
"recipe-events": "Esdeveniments de receptes"
|
"recipe-events": "Esdeveniments de receptes"
|
||||||
},
|
},
|
||||||
"general": {
|
"general": {
|
||||||
"add": "Add",
|
"add": "Afegeix",
|
||||||
"cancel": "Anuŀla",
|
"cancel": "Anuŀla",
|
||||||
"clear": "Neteja",
|
"clear": "Neteja",
|
||||||
"close": "Tanca",
|
"close": "Tanca",
|
||||||
@@ -145,23 +145,23 @@
|
|||||||
"save": "Desa",
|
"save": "Desa",
|
||||||
"settings": "Configuració",
|
"settings": "Configuració",
|
||||||
"share": "Compartiu",
|
"share": "Compartiu",
|
||||||
"show-all": "Show All",
|
"show-all": "Mostra-ho tot",
|
||||||
"shuffle": "Barreja",
|
"shuffle": "Barreja",
|
||||||
"sort": "Ordena",
|
"sort": "Ordena",
|
||||||
"sort-ascending": "Sort Ascending",
|
"sort-ascending": "Ordre ascendent",
|
||||||
"sort-descending": "Sort Descending",
|
"sort-descending": "Ordre descendent",
|
||||||
"sort-alphabetically": "Alfabèticament",
|
"sort-alphabetically": "Alfabèticament",
|
||||||
"status": "Estat",
|
"status": "Estat",
|
||||||
"subject": "Assumpte",
|
"subject": "Assumpte",
|
||||||
"submit": "Envia",
|
"submit": "Envia",
|
||||||
"success-count": "Amb èxit: {count}",
|
"success-count": "Amb èxit: {count}",
|
||||||
"sunday": "Diumenge",
|
"sunday": "Diumenge",
|
||||||
"system": "System",
|
"system": "Sistema",
|
||||||
"templates": "Plantilles:",
|
"templates": "Plantilles:",
|
||||||
"test": "Prova",
|
"test": "Prova",
|
||||||
"themes": "Temes",
|
"themes": "Temes",
|
||||||
"thursday": "Dijous",
|
"thursday": "Dijous",
|
||||||
"title": "Title",
|
"title": "Títol",
|
||||||
"token": "Token",
|
"token": "Token",
|
||||||
"tuesday": "Dimarts",
|
"tuesday": "Dimarts",
|
||||||
"type": "Tipus",
|
"type": "Tipus",
|
||||||
@@ -176,7 +176,7 @@
|
|||||||
"units": "Unitats",
|
"units": "Unitats",
|
||||||
"back": "Torna",
|
"back": "Torna",
|
||||||
"next": "Següent",
|
"next": "Següent",
|
||||||
"start": "Start",
|
"start": "Comença",
|
||||||
"toggle-view": "Commuta la visualització",
|
"toggle-view": "Commuta la visualització",
|
||||||
"date": "Data",
|
"date": "Data",
|
||||||
"id": "Id",
|
"id": "Id",
|
||||||
@@ -210,7 +210,8 @@
|
|||||||
"unsaved-changes": "Tens canvis que no estan guardats. Vols guardar-los abans de sortir? Clica d'acord per guardar-los o cancel·lar per descartar els canvis.",
|
"unsaved-changes": "Tens canvis que no estan guardats. Vols guardar-los abans de sortir? Clica d'acord per guardar-los o cancel·lar per descartar els canvis.",
|
||||||
"clipboard-copy-failure": "No s'ha pogut copiar al porta-retalls.",
|
"clipboard-copy-failure": "No s'ha pogut copiar al porta-retalls.",
|
||||||
"confirm-delete-generic-items": "Are you sure you want to delete the following items?",
|
"confirm-delete-generic-items": "Are you sure you want to delete the following items?",
|
||||||
"organizers": "Organizers"
|
"organizers": "Organitzadors",
|
||||||
|
"caution": "Precaució"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "Esteu segur de voler suprimir el grup <b>{groupName}<b/>?",
|
"are-you-sure-you-want-to-delete-the-group": "Esteu segur de voler suprimir el grup <b>{groupName}<b/>?",
|
||||||
@@ -246,7 +247,7 @@
|
|||||||
"group-preferences": "Preferències per grup",
|
"group-preferences": "Preferències per grup",
|
||||||
"private-group": "Grup privat",
|
"private-group": "Grup privat",
|
||||||
"private-group-description": "Marcar el grup com a privat afectarà a totes les opcions de visualització pública per defecte. Podeu canviar-les individualment en les opcions de visualització de cada recepta.",
|
"private-group-description": "Marcar el grup com a privat afectarà a totes les opcions de visualització pública per defecte. Podeu canviar-les individualment en les opcions de visualització de cada recepta.",
|
||||||
"enable-public-access": "Enable Public Access",
|
"enable-public-access": "Permetre l'accés públic",
|
||||||
"enable-public-access-description": "Make group recipes public by default, and allow visitors to view recipes without logging-in",
|
"enable-public-access-description": "Make group recipes public by default, and allow visitors to view recipes without logging-in",
|
||||||
"allow-users-outside-of-your-group-to-see-your-recipes": "Permeteu als usuaris d'altres grups, visualitzar les vostres receptes",
|
"allow-users-outside-of-your-group-to-see-your-recipes": "Permeteu als usuaris d'altres grups, visualitzar les vostres receptes",
|
||||||
"allow-users-outside-of-your-group-to-see-your-recipes-description": "Si ho habiliteu, podreu compartir enllaços públics de receptes específiques sense autoritzar l'usuari. Si està deshabilitat, només podreu compartir amb usuaris del vostre grup o generant enllaços privats",
|
"allow-users-outside-of-your-group-to-see-your-recipes-description": "Si ho habiliteu, podreu compartir enllaços públics de receptes específiques sense autoritzar l'usuari. Si està deshabilitat, només podreu compartir amb usuaris del vostre grup o generant enllaços privats",
|
||||||
@@ -291,6 +292,8 @@
|
|||||||
"mealplan-updated": "S'ha actualitzat el menú",
|
"mealplan-updated": "S'ha actualitzat el menú",
|
||||||
"no-meal-plan-defined-yet": "No hi ha cap menú planificat",
|
"no-meal-plan-defined-yet": "No hi ha cap menú planificat",
|
||||||
"no-meal-planned-for-today": "No hi han cap menú per a hui",
|
"no-meal-planned-for-today": "No hi han cap menú per a hui",
|
||||||
|
"numberOfDays-hint": "Number of days on page load",
|
||||||
|
"numberOfDays-label": "Dies per defecte",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Només s'utilitzaran aquestes categories per als menús",
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Només s'utilitzaran aquestes categories per als menús",
|
||||||
"planner": "Planificador",
|
"planner": "Planificador",
|
||||||
"quick-week": "Pla ràpid",
|
"quick-week": "Pla ràpid",
|
||||||
@@ -365,7 +368,7 @@
|
|||||||
"choose-migration-type": "Elegeix un tipus de migració",
|
"choose-migration-type": "Elegeix un tipus de migració",
|
||||||
"tag-all-recipes": "Etiqueta totes les receptes amb {tag-name}",
|
"tag-all-recipes": "Etiqueta totes les receptes amb {tag-name}",
|
||||||
"nextcloud-text": "Les receptes de Nextcloud poden ser importades d'un fitxer ZIP que contingui les dades emmagatzemades en Nextcloud. Segueix l'exemple d'estructura de directori de sota per assegurar que les receptes podran ser importades.",
|
"nextcloud-text": "Les receptes de Nextcloud poden ser importades d'un fitxer ZIP que contingui les dades emmagatzemades en Nextcloud. Segueix l'exemple d'estructura de directori de sota per assegurar que les receptes podran ser importades.",
|
||||||
"chowdown-text": "Mealie natively supports the chowdown repository format. Download the code repository as a .zip file and upload it below.",
|
"chowdown-text": "Mealie suporta de forma nativa el format de Chowdown. Descarrega el codi del repositori com a .zip i puja'l a sota.",
|
||||||
"recipe-1": "Recepta 1",
|
"recipe-1": "Recepta 1",
|
||||||
"recipe-2": "Recepta 2",
|
"recipe-2": "Recepta 2",
|
||||||
"paprika-text": "Mealie pot importar receptes des de l'aplicació Paprika. Exporta les teves receptes de Paprika, reanomena l'extensió de l'arxiu a .zip i penja'l aquí sota.",
|
"paprika-text": "Mealie pot importar receptes des de l'aplicació Paprika. Exporta les teves receptes de Paprika, reanomena l'extensió de l'arxiu a .zip i penja'l aquí sota.",
|
||||||
@@ -375,8 +378,12 @@
|
|||||||
"description-long": "Mealie pot importar receptes de Plan to Eat."
|
"description-long": "Mealie pot importar receptes de Plan to Eat."
|
||||||
},
|
},
|
||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "My Recipe Box",
|
"title": "La Meva Caixa de Receptes",
|
||||||
"description-long": "Mealie can import recipes from My Recipe Box. Export your recipes in CSV format, then upload the .csv file below."
|
"description-long": "Mealie can import recipes from My Recipe Box. Export your recipes in CSV format, then upload the .csv file below."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Mealie can import recipes from Recipe Keeper. Export your recipes in zip format, then upload the .zip file below."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "Ingredients",
|
"ingredients": "Ingredients",
|
||||||
"insert-ingredient": "Afegiu ingredient",
|
"insert-ingredient": "Afegiu ingredient",
|
||||||
"insert-section": "Insereix una secció",
|
"insert-section": "Insereix una secció",
|
||||||
|
"insert-above": "Insert Above",
|
||||||
|
"insert-below": "Insert Below",
|
||||||
"instructions": "Instruccions",
|
"instructions": "Instruccions",
|
||||||
"key-name-required": "Es requereix un nom de clau",
|
"key-name-required": "Es requereix un nom de clau",
|
||||||
"landscape-view-coming-soon": "Vista apaïsada (aviat)",
|
"landscape-view-coming-soon": "Vista apaïsada (aviat)",
|
||||||
@@ -511,8 +520,8 @@
|
|||||||
"cook-mode": "Mode \"cuinant\"",
|
"cook-mode": "Mode \"cuinant\"",
|
||||||
"link-ingredients": "Enllaça amb els ingredients",
|
"link-ingredients": "Enllaça amb els ingredients",
|
||||||
"merge-above": "Fusiona amb el de dalt",
|
"merge-above": "Fusiona amb el de dalt",
|
||||||
"move-to-bottom": "Move To Bottom",
|
"move-to-bottom": "Moure al Final",
|
||||||
"move-to-top": "Move To Top",
|
"move-to-top": "Moure al Principi",
|
||||||
"reset-scale": "Reinicialitza",
|
"reset-scale": "Reinicialitza",
|
||||||
"decrease-scale-label": "Divideix",
|
"decrease-scale-label": "Divideix",
|
||||||
"increase-scale-label": "Multiplica",
|
"increase-scale-label": "Multiplica",
|
||||||
@@ -549,8 +558,8 @@
|
|||||||
"looking-for-migrations": "Estàs buscant migracions?",
|
"looking-for-migrations": "Estàs buscant migracions?",
|
||||||
"import-with-url": "Importar amb l'URL",
|
"import-with-url": "Importar amb l'URL",
|
||||||
"create-recipe": "Crea la recepta",
|
"create-recipe": "Crea la recepta",
|
||||||
"create-recipe-description": "Create a new recipe from scratch.",
|
"create-recipe-description": "Crea una nova recepta des de zero.",
|
||||||
"create-recipes": "Create Recipes",
|
"create-recipes": "Crea Receptes",
|
||||||
"import-with-zip": "Importar amb un .zip",
|
"import-with-zip": "Importar amb un .zip",
|
||||||
"create-recipe-from-an-image": "Crea la recepta a partir d'una imatge",
|
"create-recipe-from-an-image": "Crea la recepta a partir d'una imatge",
|
||||||
"bulk-url-import": "Importació d'URL en massa",
|
"bulk-url-import": "Importació d'URL en massa",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "No s'ha pogut suprimir l'informe",
|
"report-deletion-failed": "No s'ha pogut suprimir l'informe",
|
||||||
"recipe-debugger": "Depuradora de receptes",
|
"recipe-debugger": "Depuradora de receptes",
|
||||||
"recipe-debugger-description": "Agafa l'URL de la recepta que vols depurar i enganxa-la aquí. L'URL serà reastrejada pel rastrejador de receptes i es mostraran els resultats. Si no veieu cap dada retornada, el lloc que esteu provant de rastrejar no és compatible amb Mealie ni la seva biblioteca de rastreig.",
|
"recipe-debugger-description": "Agafa l'URL de la recepta que vols depurar i enganxa-la aquí. L'URL serà reastrejada pel rastrejador de receptes i es mostraran els resultats. Si no veieu cap dada retornada, el lloc que esteu provant de rastrejar no és compatible amb Mealie ni la seva biblioteca de rastreig.",
|
||||||
|
"use-openai": "Use OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Use OpenAI to parse the results instead of relying on the scraper library. When creating a recipe via URL, this is done automatically if the scraper library fails, but you may test it manually here.",
|
||||||
"debug": "Depuració",
|
"debug": "Depuració",
|
||||||
"tree-view": "Vista en arbre",
|
"tree-view": "Vista en arbre",
|
||||||
"recipe-yield": "Rendiment de la recepta",
|
"recipe-yield": "Rendiment de la recepta",
|
||||||
@@ -584,8 +595,8 @@
|
|||||||
"upload-image": "Puja una imatge",
|
"upload-image": "Puja una imatge",
|
||||||
"screen-awake": "Mantenir la pantalla encesa",
|
"screen-awake": "Mantenir la pantalla encesa",
|
||||||
"remove-image": "Esborrar la imatge",
|
"remove-image": "Esborrar la imatge",
|
||||||
"nextStep": "Next step",
|
"nextStep": "Següent pas",
|
||||||
"recipe-actions": "Recipe Actions",
|
"recipe-actions": "Accions de la Recepta",
|
||||||
"parser": {
|
"parser": {
|
||||||
"experimental-alert-text": "Mealie uses natural language processing to parse and create units and food items for your recipe ingredients. This feature is experimental and may not always work as expected. If you prefer not to use the parsed results, you can select 'Cancel' and your changes will not be saved.",
|
"experimental-alert-text": "Mealie uses natural language processing to parse and create units and food items for your recipe ingredients. This feature is experimental and may not always work as expected. If you prefer not to use the parsed results, you can select 'Cancel' and your changes will not be saved.",
|
||||||
"ingredient-parser": "Ingredient Parser",
|
"ingredient-parser": "Ingredient Parser",
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "S'ha creat una còpia de seguretat a {path}",
|
"backup-created-at-response-export_path": "S'ha creat una còpia de seguretat a {path}",
|
||||||
"backup-deleted": "Còpia de seguretat suprimida",
|
"backup-deleted": "Còpia de seguretat suprimida",
|
||||||
"restore-success": "La restauració s'ha dut a terme correctament",
|
"restore-success": "La restauració s'ha dut a terme correctament",
|
||||||
|
"restore-fail": "Restore failed. Check your server logs for more details",
|
||||||
"backup-tag": "Etiqueta de la còpia de seguretat",
|
"backup-tag": "Etiqueta de la còpia de seguretat",
|
||||||
"create-heading": "Crea una còpia de seguretat",
|
"create-heading": "Crea una còpia de seguretat",
|
||||||
"delete-backup": "Esborra la còpia de seguretat",
|
"delete-backup": "Esborra la còpia de seguretat",
|
||||||
@@ -731,11 +743,11 @@
|
|||||||
"webhook-name": "Nom del Webhook",
|
"webhook-name": "Nom del Webhook",
|
||||||
"description": "Els webhooks definits a sota s'executaran quan es defineixi un menú pel dia. En un temps estipulat, els webhooks s'enviaran amb les dades de la recepta que estigui programada pel dia. L'execució dels webhooks no és exacta. Els webhooks s'executen en un interval de 5 minuts, és a dir, que s'executaran en un interval de +/- 5 minuts del temps estipulat."
|
"description": "Els webhooks definits a sota s'executaran quan es defineixi un menú pel dia. En un temps estipulat, els webhooks s'enviaran amb les dades de la recepta que estigui programada pel dia. L'execució dels webhooks no és exacta. Els webhooks s'executen en un interval de 5 minuts, és a dir, que s'executaran en un interval de +/- 5 minuts del temps estipulat."
|
||||||
},
|
},
|
||||||
"bug-report": "Bug Report",
|
"bug-report": "Informe d'errors",
|
||||||
"bug-report-information": "Use this information to report a bug. Providing details of your instance to developers is the best way to get your issues resolved quickly.",
|
"bug-report-information": "Use this information to report a bug. Providing details of your instance to developers is the best way to get your issues resolved quickly.",
|
||||||
"tracker": "Tracker",
|
"tracker": "Tracker",
|
||||||
"configuration": "Configuration",
|
"configuration": "Configuració",
|
||||||
"docker-volume": "Docker Volume",
|
"docker-volume": "Volum de Docker",
|
||||||
"docker-volume-help": "Mealie requires that the frontend container and the backend share the same docker volume or storage. This ensures that the frontend container can properly access the images and assets stored on disk.",
|
"docker-volume-help": "Mealie requires that the frontend container and the backend share the same docker volume or storage. This ensures that the frontend container can properly access the images and assets stored on disk.",
|
||||||
"volumes-are-misconfigured": "Volumes are misconfigured.",
|
"volumes-are-misconfigured": "Volumes are misconfigured.",
|
||||||
"volumes-are-configured-correctly": "Volumes are configured correctly.",
|
"volumes-are-configured-correctly": "Volumes are configured correctly.",
|
||||||
@@ -744,8 +756,8 @@
|
|||||||
"email-configuration-status": "Estat de la configuració del correu electrònic",
|
"email-configuration-status": "Estat de la configuració del correu electrònic",
|
||||||
"email-configured": "Correu electrònic configurat",
|
"email-configured": "Correu electrònic configurat",
|
||||||
"email-test-results": "Email Test Results",
|
"email-test-results": "Email Test Results",
|
||||||
"ready": "Ready",
|
"ready": "Llest",
|
||||||
"not-ready": "Not Ready - Check Environmental Variables",
|
"not-ready": "No està llest - Comprova les variables d'entorn",
|
||||||
"succeeded": "Va tenir èxit",
|
"succeeded": "Va tenir èxit",
|
||||||
"failed": "Ha fallat",
|
"failed": "Ha fallat",
|
||||||
"general-about": "Informació General",
|
"general-about": "Informació General",
|
||||||
@@ -760,12 +772,12 @@
|
|||||||
"server-side-base-url-success-text": "Server Side URL does not match the default",
|
"server-side-base-url-success-text": "Server Side URL does not match the default",
|
||||||
"ldap-ready": "LDAP Ready",
|
"ldap-ready": "LDAP Ready",
|
||||||
"ldap-ready-error-text": "Not all LDAP Values are configured. This can be ignored if you are not using LDAP Authentication.",
|
"ldap-ready-error-text": "Not all LDAP Values are configured. This can be ignored if you are not using LDAP Authentication.",
|
||||||
"ldap-ready-success-text": "Required LDAP variables are all set.",
|
"ldap-ready-success-text": "Les variables requerides per LDAP estan establertes.",
|
||||||
"build": "Build",
|
"build": "Versió de compilació",
|
||||||
"recipe-scraper-version": "Recipe Scraper Version",
|
"recipe-scraper-version": "Recipe Scraper Version",
|
||||||
"oidc-ready": "OIDC Ready",
|
"oidc-ready": "OIDC Llest",
|
||||||
"oidc-ready-error-text": "Not all OIDC Values are configured. This can be ignored if you are not using OIDC Authentication.",
|
"oidc-ready-error-text": "No tots els valors d'OIDC estan configurats. Es pot ignorar si no s'està utilitzant l'autenticació d'OIDC.",
|
||||||
"oidc-ready-success-text": "Required OIDC variables are all set.",
|
"oidc-ready-success-text": "Les variables requerides per OICD estan establertes.",
|
||||||
"openai-ready": "Llest per OpenAI",
|
"openai-ready": "Llest per OpenAI",
|
||||||
"openai-ready-error-text": "No tots els valors d'OpenAI estan configurats. Es pot ignorar si no s'estan utilitzant les funcionalitats d'OpenAI.",
|
"openai-ready-error-text": "No tots els valors d'OpenAI estan configurats. Es pot ignorar si no s'estan utilitzant les funcionalitats d'OpenAI.",
|
||||||
"openai-ready-success-text": "Les variables requerides per OpenAI estan establertes."
|
"openai-ready-success-text": "Les variables requerides per OpenAI estan establertes."
|
||||||
@@ -782,7 +794,7 @@
|
|||||||
"food": "Aliments",
|
"food": "Aliments",
|
||||||
"note": "Nota",
|
"note": "Nota",
|
||||||
"label": "Etiqueta",
|
"label": "Etiqueta",
|
||||||
"save-label": "Save Label",
|
"save-label": "Guarda l'etiqueta",
|
||||||
"linked-item-warning": "Aquest element està enllaçat amb una o més receptes. Modificar les unitats o els aliments pot provocar resultats inesperats en afegir o elimina la recepta del llistat.",
|
"linked-item-warning": "Aquest element està enllaçat amb una o més receptes. Modificar les unitats o els aliments pot provocar resultats inesperats en afegir o elimina la recepta del llistat.",
|
||||||
"toggle-food": "Mostra el nom de l'aliment",
|
"toggle-food": "Mostra el nom de l'aliment",
|
||||||
"manage-labels": "Gestiona etiquetes",
|
"manage-labels": "Gestiona etiquetes",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "No Linked Recipes|One Linked Recipe|{count} Linked Recipes",
|
"linked-recipes-count": "No Linked Recipes|One Linked Recipe|{count} Linked Recipes",
|
||||||
"items-checked-count": "No items checked|One item checked|{count} items checked",
|
"items-checked-count": "No items checked|One item checked|{count} items checked",
|
||||||
"no-label": "Sense etiqueta",
|
"no-label": "Sense etiqueta",
|
||||||
"completed-on": "Completat el {date}"
|
"completed-on": "Completat el {date}",
|
||||||
|
"you-are-offline": "You are offline",
|
||||||
|
"you-are-offline-description": "Not all features are available while offline. You can still add, modify, and remove items, but you will not be able to sync your changes to the server until you are back online.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "Are you sure you want to check all items?",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "Are you sure you want to uncheck all items?",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "Are you sure you want to delete all checked items?"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "Receptes",
|
"all-recipes": "Receptes",
|
||||||
@@ -842,7 +859,7 @@
|
|||||||
"untagged-count": "{count} sense etiquetar",
|
"untagged-count": "{count} sense etiquetar",
|
||||||
"create-a-tag": "Crea una etiqueta",
|
"create-a-tag": "Crea una etiqueta",
|
||||||
"tag-name": "Nom de l'etiqueta",
|
"tag-name": "Nom de l'etiqueta",
|
||||||
"tag": "Tag"
|
"tag": "Etiqueta"
|
||||||
},
|
},
|
||||||
"tool": {
|
"tool": {
|
||||||
"tools": "Estris",
|
"tools": "Estris",
|
||||||
@@ -852,7 +869,7 @@
|
|||||||
"create-new-tool": "Crea un nou estri",
|
"create-new-tool": "Crea un nou estri",
|
||||||
"on-hand-checkbox-label": "Mostra com a disponible (marcat)",
|
"on-hand-checkbox-label": "Mostra com a disponible (marcat)",
|
||||||
"required-tools": "Eines necessàries",
|
"required-tools": "Eines necessàries",
|
||||||
"tool": "Tool"
|
"tool": "Eina"
|
||||||
},
|
},
|
||||||
"user": {
|
"user": {
|
||||||
"admin": "Administrador/a",
|
"admin": "Administrador/a",
|
||||||
@@ -877,8 +894,8 @@
|
|||||||
"link-id": "Id de l'enllaç",
|
"link-id": "Id de l'enllaç",
|
||||||
"link-name": "Nom de l'enllaç",
|
"link-name": "Nom de l'enllaç",
|
||||||
"login": "Inicieu sessió",
|
"login": "Inicieu sessió",
|
||||||
"login-oidc": "Login with",
|
"login-oidc": "Inicia sessió amb",
|
||||||
"or": "or",
|
"or": "o",
|
||||||
"logout": "Tanca la sessió",
|
"logout": "Tanca la sessió",
|
||||||
"manage-users": "Gestionar usuaris",
|
"manage-users": "Gestionar usuaris",
|
||||||
"manage-users-description": "Create and manage users.",
|
"manage-users-description": "Create and manage users.",
|
||||||
@@ -939,9 +956,9 @@
|
|||||||
"user-name": "Nom de l'usuari",
|
"user-name": "Nom de l'usuari",
|
||||||
"authentication-method": "Authentication Method",
|
"authentication-method": "Authentication Method",
|
||||||
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie",
|
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie",
|
||||||
"permissions": "Permissions",
|
"permissions": "Permisos",
|
||||||
"administrator": "Administrator",
|
"administrator": "Administrador",
|
||||||
"user-can-invite-other-to-group": "User can invite other to group",
|
"user-can-invite-other-to-group": "User can invite others to group",
|
||||||
"user-can-manage-group": "User can manage group",
|
"user-can-manage-group": "User can manage group",
|
||||||
"user-can-organize-group-data": "User can organize group data",
|
"user-can-organize-group-data": "User can organize group data",
|
||||||
"enable-advanced-features": "Enable advanced features",
|
"enable-advanced-features": "Enable advanced features",
|
||||||
@@ -949,7 +966,7 @@
|
|||||||
"dont-want-to-see-this-anymore-be-sure-to-change-your-email": "Don't want to see this anymore? Be sure to change your email in your user settings!",
|
"dont-want-to-see-this-anymore-be-sure-to-change-your-email": "Don't want to see this anymore? Be sure to change your email in your user settings!",
|
||||||
"forgot-password": "Contrasenya oblidada",
|
"forgot-password": "Contrasenya oblidada",
|
||||||
"forgot-password-text": "Please enter your email address and we will send you a link to reset your password.",
|
"forgot-password-text": "Please enter your email address and we will send you a link to reset your password.",
|
||||||
"changes-reflected-immediately": "Changes to this user will be reflected immediately."
|
"changes-reflected-immediately": "Els canvis en aquest usuari s'actualitzaran immediatament."
|
||||||
},
|
},
|
||||||
"language-dialog": {
|
"language-dialog": {
|
||||||
"translated": "traduït",
|
"translated": "traduït",
|
||||||
@@ -964,30 +981,32 @@
|
|||||||
"merge-food-example": "Combinant {food1} i {food2}",
|
"merge-food-example": "Combinant {food1} i {food2}",
|
||||||
"seed-dialog-text": "Afegeix a la base de dades els noms dels aliments en el vostre idioma. Açò crearà més de 200 aliments comuns per a què pugueu organitzar la vostra base de dades. Els noms dels aliments han estat traduïts gràcies a l'esforç de la comunitat.",
|
"seed-dialog-text": "Afegeix a la base de dades els noms dels aliments en el vostre idioma. Açò crearà més de 200 aliments comuns per a què pugueu organitzar la vostra base de dades. Els noms dels aliments han estat traduïts gràcies a l'esforç de la comunitat.",
|
||||||
"seed-dialog-warning": "Ja teniu algunes dades a la vostra base. Aquesta acció no tindrà en compte duplicats i haureu d'eliminar-los manualment.",
|
"seed-dialog-warning": "Ja teniu algunes dades a la vostra base. Aquesta acció no tindrà en compte duplicats i haureu d'eliminar-los manualment.",
|
||||||
"combine-food": "Combine Food",
|
"combine-food": "Combinar Aliment",
|
||||||
"source-food": "Source Food",
|
"source-food": "Aliment d'Origen",
|
||||||
"target-food": "Target Food",
|
"target-food": "Aliment de Destí",
|
||||||
"create-food": "Create Food",
|
"create-food": "Crear Aliment",
|
||||||
"food-label": "Food Label",
|
"food-label": "Food Label",
|
||||||
"edit-food": "Edit Food",
|
"edit-food": "Editar Aliment",
|
||||||
"food-data": "Food Data",
|
"food-data": "Food Data",
|
||||||
"example-food-singular": "ex: Onion",
|
"example-food-singular": "p. ex.: Ceba",
|
||||||
"example-food-plural": "ex: Onions"
|
"example-food-plural": "p. ex.: Cebes",
|
||||||
|
"label-overwrite-warning": "This will assign the chosen label to all selected foods and potentially overwrite your existing labels.",
|
||||||
|
"on-hand-checkbox-label": "Setting this flag will make this food unchecked by default when adding a recipe to a shopping list."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Afegeix a la base de dades les unitats més comunes en el vostre idioma.",
|
"seed-dialog-text": "Afegeix a la base de dades les unitats més comunes en el vostre idioma.",
|
||||||
"combine-unit-description": "Combining the selected units will merge the Source Unit and Target Unit into a single unit. The {source-unit-will-be-deleted} and all of the references to the Source Unit will be updated to point to the Target Unit.",
|
"combine-unit-description": "Combining the selected units will merge the Source Unit and Target Unit into a single unit. The {source-unit-will-be-deleted} and all of the references to the Source Unit will be updated to point to the Target Unit.",
|
||||||
"combine-unit": "Combine Unit",
|
"combine-unit": "Combina Unitats",
|
||||||
"source-unit": "Source Unit",
|
"source-unit": "Unitat Origen",
|
||||||
"target-unit": "Target Unit",
|
"target-unit": "Unitat Destí",
|
||||||
"merging-unit-into-unit": "Merging {0} into {1}",
|
"merging-unit-into-unit": "Merging {0} into {1}",
|
||||||
"create-unit": "Crea la unitat",
|
"create-unit": "Crea la unitat",
|
||||||
"abbreviation": "Abbreviation",
|
"abbreviation": "Abreviatura",
|
||||||
"plural-abbreviation": "Plural Abbreviation",
|
"plural-abbreviation": "Abreviatura del Plural",
|
||||||
"description": "Description",
|
"description": "Descripció",
|
||||||
"display-as-fraction": "Display as Fraction",
|
"display-as-fraction": "Mostra com una Fracció",
|
||||||
"use-abbreviation": "Use Abbreviation",
|
"use-abbreviation": "Utilitza Abreviatura",
|
||||||
"edit-unit": "Edit Unit",
|
"edit-unit": "Editar Unitat",
|
||||||
"unit-data": "Unit Data",
|
"unit-data": "Unit Data",
|
||||||
"use-abbv": "Use Abbv.",
|
"use-abbv": "Use Abbv.",
|
||||||
"fraction": "Fraction",
|
"fraction": "Fraction",
|
||||||
@@ -1000,7 +1019,8 @@
|
|||||||
"seed-dialog-text": "Afegeix a la base de dades etiquetes comunes en el vostre idioma.",
|
"seed-dialog-text": "Afegeix a la base de dades etiquetes comunes en el vostre idioma.",
|
||||||
"edit-label": "Edit Label",
|
"edit-label": "Edit Label",
|
||||||
"new-label": "New Label",
|
"new-label": "New Label",
|
||||||
"labels": "Labels"
|
"labels": "Labels",
|
||||||
|
"assign-label": "Assigna L'etiqueta"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Purge Exports",
|
"purge-exports": "Purge Exports",
|
||||||
|
|||||||
@@ -64,7 +64,7 @@
|
|||||||
"something-went-wrong": "Něco se nepovedlo!",
|
"something-went-wrong": "Něco se nepovedlo!",
|
||||||
"subscribed-events": "Odebírané události",
|
"subscribed-events": "Odebírané události",
|
||||||
"test-message-sent": "Testovací zpráva odeslána",
|
"test-message-sent": "Testovací zpráva odeslána",
|
||||||
"message-sent": "Message Sent",
|
"message-sent": "Zpráva odeslána",
|
||||||
"new-notification": "Nové oznámení",
|
"new-notification": "Nové oznámení",
|
||||||
"event-notifiers": "Notifikace událostí",
|
"event-notifiers": "Notifikace událostí",
|
||||||
"apprise-url-skipped-if-blank": "Apprise URL (přeskočeno pokud je prázdné)",
|
"apprise-url-skipped-if-blank": "Apprise URL (přeskočeno pokud je prázdné)",
|
||||||
@@ -81,7 +81,7 @@
|
|||||||
"recipe-events": "Recipe Events"
|
"recipe-events": "Recipe Events"
|
||||||
},
|
},
|
||||||
"general": {
|
"general": {
|
||||||
"add": "Add",
|
"add": "Přidat",
|
||||||
"cancel": "Zrušit",
|
"cancel": "Zrušit",
|
||||||
"clear": "Vymazat",
|
"clear": "Vymazat",
|
||||||
"close": "Zavřít",
|
"close": "Zavřít",
|
||||||
@@ -145,11 +145,11 @@
|
|||||||
"save": "Uložit",
|
"save": "Uložit",
|
||||||
"settings": "Nastavení",
|
"settings": "Nastavení",
|
||||||
"share": "Sdílet",
|
"share": "Sdílet",
|
||||||
"show-all": "Show All",
|
"show-all": "Zobrazit vše",
|
||||||
"shuffle": "Náhodně",
|
"shuffle": "Náhodně",
|
||||||
"sort": "Seřadit",
|
"sort": "Seřadit",
|
||||||
"sort-ascending": "Sort Ascending",
|
"sort-ascending": "Řadit vzestupně",
|
||||||
"sort-descending": "Sort Descending",
|
"sort-descending": "Řadit sestupně",
|
||||||
"sort-alphabetically": "Abecedně",
|
"sort-alphabetically": "Abecedně",
|
||||||
"status": "Stav",
|
"status": "Stav",
|
||||||
"subject": "Předmět",
|
"subject": "Předmět",
|
||||||
@@ -161,7 +161,7 @@
|
|||||||
"test": "Test",
|
"test": "Test",
|
||||||
"themes": "Motivy",
|
"themes": "Motivy",
|
||||||
"thursday": "Čtvrtek",
|
"thursday": "Čtvrtek",
|
||||||
"title": "Title",
|
"title": "Název",
|
||||||
"token": "Token",
|
"token": "Token",
|
||||||
"tuesday": "Úterý",
|
"tuesday": "Úterý",
|
||||||
"type": "Typ",
|
"type": "Typ",
|
||||||
@@ -208,9 +208,10 @@
|
|||||||
"upload-file": "Nahrát soubor",
|
"upload-file": "Nahrát soubor",
|
||||||
"created-on-date": "Vytvořeno dne: {0}",
|
"created-on-date": "Vytvořeno dne: {0}",
|
||||||
"unsaved-changes": "You have unsaved changes. Do you want to save before leaving? Okay to save, Cancel to discard changes.",
|
"unsaved-changes": "You have unsaved changes. Do you want to save before leaving? Okay to save, Cancel to discard changes.",
|
||||||
"clipboard-copy-failure": "Failed to copy to the clipboard.",
|
"clipboard-copy-failure": "Zkopírování do schránky se nezdařilo.",
|
||||||
"confirm-delete-generic-items": "Are you sure you want to delete the following items?",
|
"confirm-delete-generic-items": "Are you sure you want to delete the following items?",
|
||||||
"organizers": "Organizers"
|
"organizers": "Organizers",
|
||||||
|
"caution": "Caution"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "Jste si jisti, že chcete smazat <b>{groupName}<b/>?",
|
"are-you-sure-you-want-to-delete-the-group": "Jste si jisti, že chcete smazat <b>{groupName}<b/>?",
|
||||||
@@ -246,7 +247,7 @@
|
|||||||
"group-preferences": "Nastavení skupiny",
|
"group-preferences": "Nastavení skupiny",
|
||||||
"private-group": "Soukromá skupina",
|
"private-group": "Soukromá skupina",
|
||||||
"private-group-description": "Setting your group to private will default all public view options to default. This overrides an individual recipes public view settings.",
|
"private-group-description": "Setting your group to private will default all public view options to default. This overrides an individual recipes public view settings.",
|
||||||
"enable-public-access": "Enable Public Access",
|
"enable-public-access": "Povolit veřejný přístup",
|
||||||
"enable-public-access-description": "Make group recipes public by default, and allow visitors to view recipes without logging-in",
|
"enable-public-access-description": "Make group recipes public by default, and allow visitors to view recipes without logging-in",
|
||||||
"allow-users-outside-of-your-group-to-see-your-recipes": "Povolit uživatelům mimo vaši skupinu vidět vaše recepty",
|
"allow-users-outside-of-your-group-to-see-your-recipes": "Povolit uživatelům mimo vaši skupinu vidět vaše recepty",
|
||||||
"allow-users-outside-of-your-group-to-see-your-recipes-description": "When enabled you can use a public share link to share specific recipes without authorizing the user. When disabled, you can only share recipes with users who are in your group or with a pre-generated private link",
|
"allow-users-outside-of-your-group-to-see-your-recipes-description": "When enabled you can use a public share link to share specific recipes without authorizing the user. When disabled, you can only share recipes with users who are in your group or with a pre-generated private link",
|
||||||
@@ -291,6 +292,8 @@
|
|||||||
"mealplan-updated": "Jídelníček byl aktualizován",
|
"mealplan-updated": "Jídelníček byl aktualizován",
|
||||||
"no-meal-plan-defined-yet": "Dosud nebyl definován žádný jídelníček",
|
"no-meal-plan-defined-yet": "Dosud nebyl definován žádný jídelníček",
|
||||||
"no-meal-planned-for-today": "Pro dnešek není naplánováno žádné jídlo",
|
"no-meal-planned-for-today": "Pro dnešek není naplánováno žádné jídlo",
|
||||||
|
"numberOfDays-hint": "Number of days on page load",
|
||||||
|
"numberOfDays-label": "Default Days",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Pouze recepty z těchto kategorií budou použity v jídelníčku",
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Pouze recepty z těchto kategorií budou použity v jídelníčku",
|
||||||
"planner": "Plánovač",
|
"planner": "Plánovač",
|
||||||
"quick-week": "Rychlý plán týdne",
|
"quick-week": "Rychlý plán týdne",
|
||||||
@@ -377,6 +380,10 @@
|
|||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "My Recipe Box",
|
"title": "My Recipe Box",
|
||||||
"description-long": "Mealie can import recipes from My Recipe Box. Export your recipes in CSV format, then upload the .csv file below."
|
"description-long": "Mealie can import recipes from My Recipe Box. Export your recipes in CSV format, then upload the .csv file below."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Mealie can import recipes from Recipe Keeper. Export your recipes in zip format, then upload the .zip file below."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "Ingredience",
|
"ingredients": "Ingredience",
|
||||||
"insert-ingredient": "Vložte ingredience",
|
"insert-ingredient": "Vložte ingredience",
|
||||||
"insert-section": "Vložit sekci",
|
"insert-section": "Vložit sekci",
|
||||||
|
"insert-above": "Insert Above",
|
||||||
|
"insert-below": "Insert Below",
|
||||||
"instructions": "Postup",
|
"instructions": "Postup",
|
||||||
"key-name-required": "Je vyžadován název klíče",
|
"key-name-required": "Je vyžadován název klíče",
|
||||||
"landscape-view-coming-soon": "Landscape View (Coming Soon)",
|
"landscape-view-coming-soon": "Landscape View (Coming Soon)",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "Report deletion failed",
|
"report-deletion-failed": "Report deletion failed",
|
||||||
"recipe-debugger": "Recipe Debugger",
|
"recipe-debugger": "Recipe Debugger",
|
||||||
"recipe-debugger-description": "Grab the URL of the recipe you want to debug and paste it here. The URL will be scraped by the recipe scraper and the results will be displayed. If you don't see any data returned, the site you are trying to scrape is not supported by Mealie or its scraper library.",
|
"recipe-debugger-description": "Grab the URL of the recipe you want to debug and paste it here. The URL will be scraped by the recipe scraper and the results will be displayed. If you don't see any data returned, the site you are trying to scrape is not supported by Mealie or its scraper library.",
|
||||||
|
"use-openai": "Use OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Use OpenAI to parse the results instead of relying on the scraper library. When creating a recipe via URL, this is done automatically if the scraper library fails, but you may test it manually here.",
|
||||||
"debug": "Debug",
|
"debug": "Debug",
|
||||||
"tree-view": "Stromové zobrazení",
|
"tree-view": "Stromové zobrazení",
|
||||||
"recipe-yield": "Recipe Yield",
|
"recipe-yield": "Recipe Yield",
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "Záloha vytvořena v {path}",
|
"backup-created-at-response-export_path": "Záloha vytvořena v {path}",
|
||||||
"backup-deleted": "Záloha smazána",
|
"backup-deleted": "Záloha smazána",
|
||||||
"restore-success": "Restore successful",
|
"restore-success": "Restore successful",
|
||||||
|
"restore-fail": "Restore failed. Check your server logs for more details",
|
||||||
"backup-tag": "Štítek zálohy",
|
"backup-tag": "Štítek zálohy",
|
||||||
"create-heading": "Create a Backup",
|
"create-heading": "Create a Backup",
|
||||||
"delete-backup": "Smazat zálohu",
|
"delete-backup": "Smazat zálohu",
|
||||||
@@ -782,10 +794,10 @@
|
|||||||
"food": "Jídlo",
|
"food": "Jídlo",
|
||||||
"note": "Poznámka",
|
"note": "Poznámka",
|
||||||
"label": "Popisek",
|
"label": "Popisek",
|
||||||
"save-label": "Save Label",
|
"save-label": "Uložit štítek",
|
||||||
"linked-item-warning": "Tato položka je propojena s jedním nebo více recepty. Úprava jednotky nebo jídla bude mít neočekávané důsledky při přidání nebo odebrání receptu z tohoto seznamu.",
|
"linked-item-warning": "Tato položka je propojena s jedním nebo více recepty. Úprava jednotky nebo jídla bude mít neočekávané důsledky při přidání nebo odebrání receptu z tohoto seznamu.",
|
||||||
"toggle-food": "Přepnout typ položky",
|
"toggle-food": "Přepnout typ položky",
|
||||||
"manage-labels": "Manage Labels",
|
"manage-labels": "Spravovat štítky",
|
||||||
"are-you-sure-you-want-to-delete-this-item": "Are you sure you want to delete this item?",
|
"are-you-sure-you-want-to-delete-this-item": "Are you sure you want to delete this item?",
|
||||||
"copy-as-text": "Copy as Text",
|
"copy-as-text": "Copy as Text",
|
||||||
"copy-as-markdown": "Copy as Markdown",
|
"copy-as-markdown": "Copy as Markdown",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "No Linked Recipes|One Linked Recipe|{count} Linked Recipes",
|
"linked-recipes-count": "No Linked Recipes|One Linked Recipe|{count} Linked Recipes",
|
||||||
"items-checked-count": "No items checked|One item checked|{count} items checked",
|
"items-checked-count": "No items checked|One item checked|{count} items checked",
|
||||||
"no-label": "Bez štítku",
|
"no-label": "Bez štítku",
|
||||||
"completed-on": "Dokončeno dne {date}"
|
"completed-on": "Dokončeno dne {date}",
|
||||||
|
"you-are-offline": "You are offline",
|
||||||
|
"you-are-offline-description": "Not all features are available while offline. You can still add, modify, and remove items, but you will not be able to sync your changes to the server until you are back online.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "Are you sure you want to check all items?",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "Are you sure you want to uncheck all items?",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "Are you sure you want to delete all checked items?"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "Všechny recepty",
|
"all-recipes": "Všechny recepty",
|
||||||
@@ -941,7 +958,7 @@
|
|||||||
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie",
|
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie",
|
||||||
"permissions": "Permissions",
|
"permissions": "Permissions",
|
||||||
"administrator": "Administrator",
|
"administrator": "Administrator",
|
||||||
"user-can-invite-other-to-group": "User can invite other to group",
|
"user-can-invite-other-to-group": "User can invite others to group",
|
||||||
"user-can-manage-group": "User can manage group",
|
"user-can-manage-group": "User can manage group",
|
||||||
"user-can-organize-group-data": "User can organize group data",
|
"user-can-organize-group-data": "User can organize group data",
|
||||||
"enable-advanced-features": "Enable advanced features",
|
"enable-advanced-features": "Enable advanced features",
|
||||||
@@ -972,7 +989,9 @@
|
|||||||
"edit-food": "Edit Food",
|
"edit-food": "Edit Food",
|
||||||
"food-data": "Food Data",
|
"food-data": "Food Data",
|
||||||
"example-food-singular": "ex: Onion",
|
"example-food-singular": "ex: Onion",
|
||||||
"example-food-plural": "ex: Onions"
|
"example-food-plural": "ex: Onions",
|
||||||
|
"label-overwrite-warning": "This will assign the chosen label to all selected foods and potentially overwrite your existing labels.",
|
||||||
|
"on-hand-checkbox-label": "Setting this flag will make this food unchecked by default when adding a recipe to a shopping list."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Naplnit databázi s běžnými jednotkami používanými ve vašem jazyce.",
|
"seed-dialog-text": "Naplnit databázi s běžnými jednotkami používanými ve vašem jazyce.",
|
||||||
@@ -999,8 +1018,9 @@
|
|||||||
"labels": {
|
"labels": {
|
||||||
"seed-dialog-text": "Naplnit databázi s běžnými popisky používanými ve vašem jazyce.",
|
"seed-dialog-text": "Naplnit databázi s běžnými popisky používanými ve vašem jazyce.",
|
||||||
"edit-label": "Upravit štítek",
|
"edit-label": "Upravit štítek",
|
||||||
"new-label": "New Label",
|
"new-label": "Nový štítek",
|
||||||
"labels": "Štítky"
|
"labels": "Štítky",
|
||||||
|
"assign-label": "Přiřadit štítek"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Purge Exports",
|
"purge-exports": "Purge Exports",
|
||||||
|
|||||||
@@ -64,7 +64,7 @@
|
|||||||
"something-went-wrong": "Noget gik galt!",
|
"something-went-wrong": "Noget gik galt!",
|
||||||
"subscribed-events": "Abonnerede begivenheder",
|
"subscribed-events": "Abonnerede begivenheder",
|
||||||
"test-message-sent": "Testbesked sendt",
|
"test-message-sent": "Testbesked sendt",
|
||||||
"message-sent": "Message Sent",
|
"message-sent": "Besked afsendt",
|
||||||
"new-notification": "Ny notifikation",
|
"new-notification": "Ny notifikation",
|
||||||
"event-notifiers": "Notifikation om begivenheder",
|
"event-notifiers": "Notifikation om begivenheder",
|
||||||
"apprise-url-skipped-if-blank": "Informations link (sprunget over hvis ladet være tomt)",
|
"apprise-url-skipped-if-blank": "Informations link (sprunget over hvis ladet være tomt)",
|
||||||
@@ -86,7 +86,7 @@
|
|||||||
"clear": "Ryd",
|
"clear": "Ryd",
|
||||||
"close": "Luk",
|
"close": "Luk",
|
||||||
"confirm": "Bekræft",
|
"confirm": "Bekræft",
|
||||||
"confirm-how-does-everything-look": "How does everything look?",
|
"confirm-how-does-everything-look": "Hvordan ser alting ud?",
|
||||||
"confirm-delete-generic": "Er du sikker på, du vil slette dette?",
|
"confirm-delete-generic": "Er du sikker på, du vil slette dette?",
|
||||||
"copied_message": "Kopieret!",
|
"copied_message": "Kopieret!",
|
||||||
"create": "Opret",
|
"create": "Opret",
|
||||||
@@ -161,7 +161,7 @@
|
|||||||
"test": "Afprøv",
|
"test": "Afprøv",
|
||||||
"themes": "Temaer",
|
"themes": "Temaer",
|
||||||
"thursday": "Torsdag",
|
"thursday": "Torsdag",
|
||||||
"title": "Title",
|
"title": "Titel",
|
||||||
"token": "Token",
|
"token": "Token",
|
||||||
"tuesday": "Tirsdag",
|
"tuesday": "Tirsdag",
|
||||||
"type": "Type",
|
"type": "Type",
|
||||||
@@ -210,7 +210,8 @@
|
|||||||
"unsaved-changes": "Du har ændringer som ikke er gemt. Vil du gemme før du forlader? Vælg \"Okay\" for at gemme, eller \"Annullér\" for at kassere ændringer.",
|
"unsaved-changes": "Du har ændringer som ikke er gemt. Vil du gemme før du forlader? Vælg \"Okay\" for at gemme, eller \"Annullér\" for at kassere ændringer.",
|
||||||
"clipboard-copy-failure": "Kopiering til udklipsholderen mislykkedes.",
|
"clipboard-copy-failure": "Kopiering til udklipsholderen mislykkedes.",
|
||||||
"confirm-delete-generic-items": "Er du sikker på at du ønsker at slette de valgte emner?",
|
"confirm-delete-generic-items": "Er du sikker på at du ønsker at slette de valgte emner?",
|
||||||
"organizers": "Organizers"
|
"organizers": "Organisatorer",
|
||||||
|
"caution": "Bemærk"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "Er du sikker på, du vil slette <b>{groupName}<b/>?",
|
"are-you-sure-you-want-to-delete-the-group": "Er du sikker på, du vil slette <b>{groupName}<b/>?",
|
||||||
@@ -291,6 +292,8 @@
|
|||||||
"mealplan-updated": "Madplanen blev ændret",
|
"mealplan-updated": "Madplanen blev ændret",
|
||||||
"no-meal-plan-defined-yet": "Ingen madplan er defineret",
|
"no-meal-plan-defined-yet": "Ingen madplan er defineret",
|
||||||
"no-meal-planned-for-today": "Ingen ret er planlagt til i dag",
|
"no-meal-planned-for-today": "Ingen ret er planlagt til i dag",
|
||||||
|
"numberOfDays-hint": "Antal dage ved sideindlæsning",
|
||||||
|
"numberOfDays-label": "Standarddage",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Kun opskrifter med disse kategorier vil blive brugt i madplaner",
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Kun opskrifter med disse kategorier vil blive brugt i madplaner",
|
||||||
"planner": "Planlæg madplan",
|
"planner": "Planlæg madplan",
|
||||||
"quick-week": "Hurtig uge",
|
"quick-week": "Hurtig uge",
|
||||||
@@ -365,7 +368,7 @@
|
|||||||
"choose-migration-type": "Vælg Migreringstype",
|
"choose-migration-type": "Vælg Migreringstype",
|
||||||
"tag-all-recipes": "Tag alle opskrifter med {tag-name} tag",
|
"tag-all-recipes": "Tag alle opskrifter med {tag-name} tag",
|
||||||
"nextcloud-text": "Nextcloud opskrifter kan importeres fra en zip-fil, der indeholder data lagret i Nextcloud. Se eksempelmappestrukturen nedenfor for at sikre, at dine opskrifter kan importeres.",
|
"nextcloud-text": "Nextcloud opskrifter kan importeres fra en zip-fil, der indeholder data lagret i Nextcloud. Se eksempelmappestrukturen nedenfor for at sikre, at dine opskrifter kan importeres.",
|
||||||
"chowdown-text": "Mealie natively supports the chowdown repository format. Download the code repository as a .zip file and upload it below.",
|
"chowdown-text": "Mealie understøtter chowdown repository fil formater. Download repositoriet som en .zip-fil og upload den nedenfor.",
|
||||||
"recipe-1": "Opskrift 1",
|
"recipe-1": "Opskrift 1",
|
||||||
"recipe-2": "Opskrift 2",
|
"recipe-2": "Opskrift 2",
|
||||||
"paprika-text": "Mealie kan importere opskrifter fra Paprika applikationen. Eksporter dine opskrifter fra paprika, omdøbe eksportudvidelsen til .zip og uploade den nedenfor.",
|
"paprika-text": "Mealie kan importere opskrifter fra Paprika applikationen. Eksporter dine opskrifter fra paprika, omdøbe eksportudvidelsen til .zip og uploade den nedenfor.",
|
||||||
@@ -375,8 +378,12 @@
|
|||||||
"description-long": "Mealie kan importere opskrifter, der er markeret som \"Planlæg at spise\"."
|
"description-long": "Mealie kan importere opskrifter, der er markeret som \"Planlæg at spise\"."
|
||||||
},
|
},
|
||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "My Recipe Box",
|
"title": "Mine opskrifter",
|
||||||
"description-long": "Mealie can import recipes from My Recipe Box. Export your recipes in CSV format, then upload the .csv file below."
|
"description-long": "Mealie kan importere opskrifter fra My Recipe Box. Eksporter dine opskrifter i CSV-format, og upload derefter .csv-filen nedenfor."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Mealie kan importere opskrifter fra Recipe Keeper. Eksportér dine opskrifter i zip-format, og upload derefter .zip-filen nedenfor."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "Ingredienser",
|
"ingredients": "Ingredienser",
|
||||||
"insert-ingredient": "Indsæt Ingrediens",
|
"insert-ingredient": "Indsæt Ingrediens",
|
||||||
"insert-section": "Indsæt sektion",
|
"insert-section": "Indsæt sektion",
|
||||||
|
"insert-above": "Indsæt ovenover",
|
||||||
|
"insert-below": "Indsæt nedenunder",
|
||||||
"instructions": "Instruktioner",
|
"instructions": "Instruktioner",
|
||||||
"key-name-required": "Nøglenavn påkrævet",
|
"key-name-required": "Nøglenavn påkrævet",
|
||||||
"landscape-view-coming-soon": "Liggende visning (Kommer snart)",
|
"landscape-view-coming-soon": "Liggende visning (Kommer snart)",
|
||||||
@@ -528,7 +537,7 @@
|
|||||||
"edit-timeline-event": "Rediger tidslinjebegivenhed",
|
"edit-timeline-event": "Rediger tidslinjebegivenhed",
|
||||||
"timeline": "Tidslinje",
|
"timeline": "Tidslinje",
|
||||||
"timeline-is-empty": "Intet på tidslinjen endnu. Prøv at lave denne opskrift!",
|
"timeline-is-empty": "Intet på tidslinjen endnu. Prøv at lave denne opskrift!",
|
||||||
"timeline-no-events-found-try-adjusting-filters": "No events found. Try adjusting your search filters.",
|
"timeline-no-events-found-try-adjusting-filters": "Ingen begivenheder fundet. Prøv at justere dine søgefiltre.",
|
||||||
"group-global-timeline": "{groupName} Global Tidslinje",
|
"group-global-timeline": "{groupName} Global Tidslinje",
|
||||||
"open-timeline": "Åbn tidslinje",
|
"open-timeline": "Åbn tidslinje",
|
||||||
"made-this": "Jeg har lavet denne",
|
"made-this": "Jeg har lavet denne",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "Sletning af rapport mislykkedes",
|
"report-deletion-failed": "Sletning af rapport mislykkedes",
|
||||||
"recipe-debugger": "Fejlsøgning af opskrifter",
|
"recipe-debugger": "Fejlsøgning af opskrifter",
|
||||||
"recipe-debugger-description": "Indsæt URL'en på hjemmesiden, der indeholder den opskrift, du vil fejlsøge. URL-adressen vil blive læst og resultaterne vil blive vist. Hvis ingen data bliver vist, er indhentning af opskrifter fra hjemmesiden endnu ikke understøttet af Mealie.",
|
"recipe-debugger-description": "Indsæt URL'en på hjemmesiden, der indeholder den opskrift, du vil fejlsøge. URL-adressen vil blive læst og resultaterne vil blive vist. Hvis ingen data bliver vist, er indhentning af opskrifter fra hjemmesiden endnu ikke understøttet af Mealie.",
|
||||||
|
"use-openai": "Brug OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Brug OpenAI til at fortolke resultaterne i stedet for at stole på scraper biblioteket. Når du opretter en opskrift via URL, gøres dette automatisk, hvis skraberbiblioteket fejler, men du kan teste det manuelt her.",
|
||||||
"debug": "Fejlsøgning",
|
"debug": "Fejlsøgning",
|
||||||
"tree-view": "Træ visning",
|
"tree-view": "Træ visning",
|
||||||
"recipe-yield": "Udbytte af opskrift",
|
"recipe-yield": "Udbytte af opskrift",
|
||||||
@@ -585,21 +596,21 @@
|
|||||||
"screen-awake": "Hold skærmen tændt",
|
"screen-awake": "Hold skærmen tændt",
|
||||||
"remove-image": "Fjern billede",
|
"remove-image": "Fjern billede",
|
||||||
"nextStep": "Næste trin",
|
"nextStep": "Næste trin",
|
||||||
"recipe-actions": "Recipe Actions",
|
"recipe-actions": "Opskriftshandlinger",
|
||||||
"parser": {
|
"parser": {
|
||||||
"experimental-alert-text": "Mealie uses natural language processing to parse and create units and food items for your recipe ingredients. This feature is experimental and may not always work as expected. If you prefer not to use the parsed results, you can select 'Cancel' and your changes will not be saved.",
|
"experimental-alert-text": "Mealie bruger natural language processing til at behandle og oprette relevante måleenheder og fødrerevarer som ingredienser i dine opskrifter. Denne funktion er eksperimentel og fungerer måske ikke altid som forventet. Hvis du foretrækker ikke at bruge de fortolkede resultater, kan du vælge 'Annullér' og dine ændringer vil ikke blive gemt.",
|
||||||
"ingredient-parser": "Ingredient Parser",
|
"ingredient-parser": "Ingrediens- Parser",
|
||||||
"explanation": "To use the ingredient parser, click the 'Parse All' button to start the process. Once the processed ingredients are available, you can review the items and verify that they were parsed correctly. The model's confidence score is displayed on the right of the item title. This score is an average of all the individual scores and may not always be completely accurate.",
|
"explanation": "For at bruge ingrediensparseren, skal du klikke på knappen 'Parse Alle' for at starte behandlingen. Når relevante ingredienser er identificeret, kan du gennemgå sem og kontrollere, at de blev korrekt identificeret. Modellens konfidensscore vises til højre for ingrediensens titel. Denne score er et gennemsnit af alle de enkelte scorer og er måske ikke altid helt præcis.",
|
||||||
"alerts-explainer": "Alerts will be displayed if a matching foods or unit is found but does not exists in the database.",
|
"alerts-explainer": "En advarsel vil blive vist, hvis en identificeret fødevare eller måleenhed ikke findes i databasen.",
|
||||||
"select-parser": "Select Parser",
|
"select-parser": "Vælg Parser",
|
||||||
"natural-language-processor": "Natural Language Processor",
|
"natural-language-processor": "Natural Language Processor",
|
||||||
"brute-parser": "Brute Parser",
|
"brute-parser": "Brute Parser",
|
||||||
"openai-parser": "OpenAI Parser",
|
"openai-parser": "OpenAI Parser",
|
||||||
"parse-all": "Parse All",
|
"parse-all": "Fortolk alt",
|
||||||
"no-unit": "No unit",
|
"no-unit": "Ingen enhed",
|
||||||
"missing-unit": "Create missing unit: {unit}",
|
"missing-unit": "Opret manglende måleenhed: {unit}",
|
||||||
"missing-food": "Create missing food: {food}",
|
"missing-food": "Opret manglende fødevare: {food}",
|
||||||
"no-food": "No Food"
|
"no-food": "Ingen fødevarer"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "Backup oprettet ved {path}",
|
"backup-created-at-response-export_path": "Backup oprettet ved {path}",
|
||||||
"backup-deleted": "Backup slettet",
|
"backup-deleted": "Backup slettet",
|
||||||
"restore-success": "Gendannelse lykkedes",
|
"restore-success": "Gendannelse lykkedes",
|
||||||
|
"restore-fail": "Gendannelse mislykkedes. Tjek dine serverlogs for flere detaljer",
|
||||||
"backup-tag": "Backupnavn",
|
"backup-tag": "Backupnavn",
|
||||||
"create-heading": "Opret en backup",
|
"create-heading": "Opret en backup",
|
||||||
"delete-backup": "Slet backup",
|
"delete-backup": "Slet backup",
|
||||||
@@ -766,9 +778,9 @@
|
|||||||
"oidc-ready": "OIDC er Klar",
|
"oidc-ready": "OIDC er Klar",
|
||||||
"oidc-ready-error-text": "Ikke alle OIDC værdier er konfigureret. Dette kan ignoreres, hvis du ikke bruger OIDC godkendelse.",
|
"oidc-ready-error-text": "Ikke alle OIDC værdier er konfigureret. Dette kan ignoreres, hvis du ikke bruger OIDC godkendelse.",
|
||||||
"oidc-ready-success-text": "Alle påkrævede OIDC værdier er angivet.",
|
"oidc-ready-success-text": "Alle påkrævede OIDC værdier er angivet.",
|
||||||
"openai-ready": "OpenAI Ready",
|
"openai-ready": "OpenAI Klar",
|
||||||
"openai-ready-error-text": "Not all OpenAI Values are configured. This can be ignored if you are not using OpenAI features.",
|
"openai-ready-error-text": "Ikke alle OpenAI værdier er konfigureret. Dette kan ignoreres, hvis du ikke bruger OpenAI funktioner.",
|
||||||
"openai-ready-success-text": "Required OpenAI variables are all set."
|
"openai-ready-success-text": "Påkrævede OpenAI variabler er alle indstillet."
|
||||||
},
|
},
|
||||||
"shopping-list": {
|
"shopping-list": {
|
||||||
"all-lists": "Alle lister",
|
"all-lists": "Alle lister",
|
||||||
@@ -782,7 +794,7 @@
|
|||||||
"food": "Fødevarer",
|
"food": "Fødevarer",
|
||||||
"note": "Note",
|
"note": "Note",
|
||||||
"label": "Etiket",
|
"label": "Etiket",
|
||||||
"save-label": "Save Label",
|
"save-label": "Gem etiket",
|
||||||
"linked-item-warning": "Dette element er tilknyttet en eller flere opskrifter. Justering af enheder eller fødevarer vil give uventede resultater, når du tilføjer eller fjerner opskriften fra denne liste.",
|
"linked-item-warning": "Dette element er tilknyttet en eller flere opskrifter. Justering af enheder eller fødevarer vil give uventede resultater, når du tilføjer eller fjerner opskriften fra denne liste.",
|
||||||
"toggle-food": "Slå fødevarer til/fra",
|
"toggle-food": "Slå fødevarer til/fra",
|
||||||
"manage-labels": "Håndter etiketter",
|
"manage-labels": "Håndter etiketter",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "Ingen Sammenkædede Opskrifter: En Sammenkædet Opskrift.{count} Sammenkædede Opskrifter",
|
"linked-recipes-count": "Ingen Sammenkædede Opskrifter: En Sammenkædet Opskrift.{count} Sammenkædede Opskrifter",
|
||||||
"items-checked-count": "Ingen elementer markeret|Et element markeret|{count} elementer er markeret",
|
"items-checked-count": "Ingen elementer markeret|Et element markeret|{count} elementer er markeret",
|
||||||
"no-label": "Ingen etiket",
|
"no-label": "Ingen etiket",
|
||||||
"completed-on": "Afsluttet den {date}"
|
"completed-on": "Afsluttet den {date}",
|
||||||
|
"you-are-offline": "Du er offline",
|
||||||
|
"you-are-offline-description": "Ikke alle funktioner er tilgængelige mens offline. Du kan stadig tilføje, modificere, og fjerne elementer, men du vil ikke kunne synkronisere dine ændringer til serveren, før du er online igen.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "Er du sikker på, at du vil markere alle elementer?",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "Er du sikker på, at du vil fjerne markeringen af alle elementer?",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "Er du sikker på, at du vil sletter de valgte elementer?"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "Alle opskr.",
|
"all-recipes": "Alle opskr.",
|
||||||
@@ -842,7 +859,7 @@
|
|||||||
"untagged-count": "Ikke-tagget: {count}",
|
"untagged-count": "Ikke-tagget: {count}",
|
||||||
"create-a-tag": "Opret tag",
|
"create-a-tag": "Opret tag",
|
||||||
"tag-name": "Tag navn",
|
"tag-name": "Tag navn",
|
||||||
"tag": "Tag"
|
"tag": "Mærker"
|
||||||
},
|
},
|
||||||
"tool": {
|
"tool": {
|
||||||
"tools": "Værktøjer",
|
"tools": "Værktøjer",
|
||||||
@@ -938,7 +955,7 @@
|
|||||||
"user-details": "Brugerdetaljer",
|
"user-details": "Brugerdetaljer",
|
||||||
"user-name": "Brugernavn",
|
"user-name": "Brugernavn",
|
||||||
"authentication-method": "Godkendelsesmetode",
|
"authentication-method": "Godkendelsesmetode",
|
||||||
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie",
|
"authentication-method-hint": "Dette angiver, hvordan en bruger vil logge ind på Mealie. Hvis du ikke er sikker, vælg 'Mealie'",
|
||||||
"permissions": "Rettigheder",
|
"permissions": "Rettigheder",
|
||||||
"administrator": "Administrator",
|
"administrator": "Administrator",
|
||||||
"user-can-invite-other-to-group": "Bruger kan invitere andre til gruppen",
|
"user-can-invite-other-to-group": "Bruger kan invitere andre til gruppen",
|
||||||
@@ -972,7 +989,9 @@
|
|||||||
"edit-food": "Redigér fødevare",
|
"edit-food": "Redigér fødevare",
|
||||||
"food-data": "Oplysninger om fødevare",
|
"food-data": "Oplysninger om fødevare",
|
||||||
"example-food-singular": "fx.: grøntsag",
|
"example-food-singular": "fx.: grøntsag",
|
||||||
"example-food-plural": "fx.: grøntsager"
|
"example-food-plural": "fx.: grøntsager",
|
||||||
|
"label-overwrite-warning": "Dette vil tildele den valgte etiket til alle udvalgte fødevarer og potentielt overskrive dine eksisterende etiketter.",
|
||||||
|
"on-hand-checkbox-label": "Ændring af dette flag ændrer markeringen, så denne fødevare ikke er markeret som standard, når du tilføjer en opskrift til en indkøbsliste."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Opret standard enheder i dit sprog.",
|
"seed-dialog-text": "Opret standard enheder i dit sprog.",
|
||||||
@@ -1000,7 +1019,8 @@
|
|||||||
"seed-dialog-text": "Opret standard etiketter på dit sprog.",
|
"seed-dialog-text": "Opret standard etiketter på dit sprog.",
|
||||||
"edit-label": "Redigér etiket",
|
"edit-label": "Redigér etiket",
|
||||||
"new-label": "Ny etiket",
|
"new-label": "Ny etiket",
|
||||||
"labels": "Etiketter"
|
"labels": "Etiketter",
|
||||||
|
"assign-label": "Tildel etiket"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Tøm Eksport",
|
"purge-exports": "Tøm Eksport",
|
||||||
@@ -1024,10 +1044,10 @@
|
|||||||
"source-unit-will-be-deleted": "Kildeenhed vil blive slettet"
|
"source-unit-will-be-deleted": "Kildeenhed vil blive slettet"
|
||||||
},
|
},
|
||||||
"recipe-actions": {
|
"recipe-actions": {
|
||||||
"recipe-actions-data": "Recipe Actions Data",
|
"recipe-actions-data": "Data for Opskriftshandlinger",
|
||||||
"new-recipe-action": "New Recipe Action",
|
"new-recipe-action": "Ny Opskriftshandling",
|
||||||
"edit-recipe-action": "Edit Recipe Action",
|
"edit-recipe-action": "Rediger Opskriftshandling",
|
||||||
"action-type": "Action Type"
|
"action-type": "Handlingstype"
|
||||||
},
|
},
|
||||||
"create-alias": "Opret alias",
|
"create-alias": "Opret alias",
|
||||||
"manage-aliases": "Administrer Aliaser",
|
"manage-aliases": "Administrer Aliaser",
|
||||||
|
|||||||
@@ -210,7 +210,8 @@
|
|||||||
"unsaved-changes": "Du hast ungespeicherte Änderungen. Möchtest du vor dem Verlassen speichern? OK um zu speichern, Cancel um Änderungen zu verwerfen.",
|
"unsaved-changes": "Du hast ungespeicherte Änderungen. Möchtest du vor dem Verlassen speichern? OK um zu speichern, Cancel um Änderungen zu verwerfen.",
|
||||||
"clipboard-copy-failure": "Fehler beim Kopieren in die Zwischenablage.",
|
"clipboard-copy-failure": "Fehler beim Kopieren in die Zwischenablage.",
|
||||||
"confirm-delete-generic-items": "Bist du dir sicher, dass du die folgenden Einträge löschen möchtest?",
|
"confirm-delete-generic-items": "Bist du dir sicher, dass du die folgenden Einträge löschen möchtest?",
|
||||||
"organizers": "Organisieren"
|
"organizers": "Organisieren",
|
||||||
|
"caution": "Vorsicht"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "Bist du dir sicher, dass du die Gruppe <b>{groupName}<b/> löschen möchtest?",
|
"are-you-sure-you-want-to-delete-the-group": "Bist du dir sicher, dass du die Gruppe <b>{groupName}<b/> löschen möchtest?",
|
||||||
@@ -291,6 +292,8 @@
|
|||||||
"mealplan-updated": "Essensplan aktualisiert",
|
"mealplan-updated": "Essensplan aktualisiert",
|
||||||
"no-meal-plan-defined-yet": "Noch kein Essensplan definiert",
|
"no-meal-plan-defined-yet": "Noch kein Essensplan definiert",
|
||||||
"no-meal-planned-for-today": "Kein Essen für heute geplant",
|
"no-meal-planned-for-today": "Kein Essen für heute geplant",
|
||||||
|
"numberOfDays-hint": "Anzahl der Tage beim Laden der Seite",
|
||||||
|
"numberOfDays-label": "Anzuzeigende Tage",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Nur Rezepte dieser Kategorien werden in Essensplänen verwendet",
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Nur Rezepte dieser Kategorien werden in Essensplänen verwendet",
|
||||||
"planner": "Planer",
|
"planner": "Planer",
|
||||||
"quick-week": "Schnelle Woche",
|
"quick-week": "Schnelle Woche",
|
||||||
@@ -377,6 +380,10 @@
|
|||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "My Recipe Box",
|
"title": "My Recipe Box",
|
||||||
"description-long": "Mealie kann Rezepte von My Recipe Box importieren. Exportiere deine Rezepte im CSV-Format und lade dann unten die .csv Datei hoch."
|
"description-long": "Mealie kann Rezepte von My Recipe Box importieren. Exportiere deine Rezepte im CSV-Format und lade dann unten die .csv Datei hoch."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Mealie kann Rezepte von Recipe Keeper importieren. Exportiere deine Rezepte im ZIP-Format und lade dann unten die .zip Datei hoch.\n"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "Zutaten",
|
"ingredients": "Zutaten",
|
||||||
"insert-ingredient": "Zutat einfügen",
|
"insert-ingredient": "Zutat einfügen",
|
||||||
"insert-section": "Abschnitt einfügen",
|
"insert-section": "Abschnitt einfügen",
|
||||||
|
"insert-above": "Darüber einfügen",
|
||||||
|
"insert-below": "Darunter einfügen",
|
||||||
"instructions": "Zubereitung",
|
"instructions": "Zubereitung",
|
||||||
"key-name-required": "Schlüsselname benötigt",
|
"key-name-required": "Schlüsselname benötigt",
|
||||||
"landscape-view-coming-soon": "Querformat",
|
"landscape-view-coming-soon": "Querformat",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "Bericht löschen fehlgeschlagen",
|
"report-deletion-failed": "Bericht löschen fehlgeschlagen",
|
||||||
"recipe-debugger": "Rezept Debugger",
|
"recipe-debugger": "Rezept Debugger",
|
||||||
"recipe-debugger-description": "Füge die URL des Rezepts, das du debuggen möchtest, hier ein. Die URL wird vom Scraper eingelesen und die Ergebnisse werden angezeigt. Wenn du keine Ausgabedaten sehen solltest, wird das Einlesen dieser Webseite nicht von Mealie oder dessen Scraper-Bibliothek unterstützt.",
|
"recipe-debugger-description": "Füge die URL des Rezepts, das du debuggen möchtest, hier ein. Die URL wird vom Scraper eingelesen und die Ergebnisse werden angezeigt. Wenn du keine Ausgabedaten sehen solltest, wird das Einlesen dieser Webseite nicht von Mealie oder dessen Scraper-Bibliothek unterstützt.",
|
||||||
|
"use-openai": "Verwende OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Verwende OpenAI anstelle der Scraper-Bibliothek, um die Einträge zu parsen. Wenn du ein Rezept über dessen URL erstellst und der Versuch über die Scraper-Bibliothek fehlschlägt, passiert das automatisch. Aber du kannst es hier auch manuell testen.",
|
||||||
"debug": "Debug",
|
"debug": "Debug",
|
||||||
"tree-view": "Strukturierte Ansicht",
|
"tree-view": "Strukturierte Ansicht",
|
||||||
"recipe-yield": "Portionsangabe",
|
"recipe-yield": "Portionsangabe",
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "Sicherung erstellt unter {path}",
|
"backup-created-at-response-export_path": "Sicherung erstellt unter {path}",
|
||||||
"backup-deleted": "Sicherung gelöscht",
|
"backup-deleted": "Sicherung gelöscht",
|
||||||
"restore-success": "Wiederherstellung erfolgreich",
|
"restore-success": "Wiederherstellung erfolgreich",
|
||||||
|
"restore-fail": "Wiederherstellung fehlgeschlagen. Überprüfe deine Serverprotokolle für weitere Informationen",
|
||||||
"backup-tag": "Sicherungsbeschreibung",
|
"backup-tag": "Sicherungsbeschreibung",
|
||||||
"create-heading": "Sicherung erstellen",
|
"create-heading": "Sicherung erstellen",
|
||||||
"delete-backup": "Sicherung löschen",
|
"delete-backup": "Sicherung löschen",
|
||||||
@@ -782,7 +794,7 @@
|
|||||||
"food": "Lebensmittel",
|
"food": "Lebensmittel",
|
||||||
"note": "Notiz",
|
"note": "Notiz",
|
||||||
"label": "Etikett",
|
"label": "Etikett",
|
||||||
"save-label": "Save Label",
|
"save-label": "Etikett speichern",
|
||||||
"linked-item-warning": "Dieser Eintrag ist mit einem oder mehreren Rezepten verknüpft. Das Ändern der Einheiten oder Lebensmittel führt zu unerwarteten Ergebnissen, wenn das Rezept von dieser Einkaufsliste entfernt oder hinzugefügt wird.",
|
"linked-item-warning": "Dieser Eintrag ist mit einem oder mehreren Rezepten verknüpft. Das Ändern der Einheiten oder Lebensmittel führt zu unerwarteten Ergebnissen, wenn das Rezept von dieser Einkaufsliste entfernt oder hinzugefügt wird.",
|
||||||
"toggle-food": "Lebensmittel-Eingabe umschalten",
|
"toggle-food": "Lebensmittel-Eingabe umschalten",
|
||||||
"manage-labels": "Etiketten verwalten",
|
"manage-labels": "Etiketten verwalten",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "Kein verknüpftes Rezept|Ein verknüpftes Rezept|{count} verknüpfte Rezepte",
|
"linked-recipes-count": "Kein verknüpftes Rezept|Ein verknüpftes Rezept|{count} verknüpfte Rezepte",
|
||||||
"items-checked-count": "Kein Eintrag erledigt|Ein Eintrag erledigt|{count} Einträge erledigt",
|
"items-checked-count": "Kein Eintrag erledigt|Ein Eintrag erledigt|{count} Einträge erledigt",
|
||||||
"no-label": "Kein Etikett",
|
"no-label": "Kein Etikett",
|
||||||
"completed-on": "Erledigt am {date}"
|
"completed-on": "Erledigt am {date}",
|
||||||
|
"you-are-offline": "Keine Verbindung zum Internet",
|
||||||
|
"you-are-offline-description": "Nicht alle Funktionen sind offline verfügbar. Du kannst weiterhin Einträge hinzufügen, ändern und entfernen, aber Änderungen werden erst dann mit dem Server synchronisiert, wenn du wieder online bist.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "Bist du sicher, dass du alle Elemente markieren möchtest?",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "Bist du sicher, dass du die Auswahl aller Elemente aufheben möchtest?",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "Bist du sicher, dass du alle ausgewählten Elemente löschen möchtest?"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "Alle Rezepte",
|
"all-recipes": "Alle Rezepte",
|
||||||
@@ -829,7 +846,7 @@
|
|||||||
"sign-up-links": "Registrierungslinks",
|
"sign-up-links": "Registrierungslinks",
|
||||||
"sign-up-token-deleted": "Registrierungs-Token entfernt",
|
"sign-up-token-deleted": "Registrierungs-Token entfernt",
|
||||||
"sign-up-token-deletion-failed": "Entfernen des Registrierungs-Tokens fehlgeschlagen",
|
"sign-up-token-deletion-failed": "Entfernen des Registrierungs-Tokens fehlgeschlagen",
|
||||||
"welcome-to-mealie": "Willkommen bei Mealie! Um ein Benutzer dieser Instanz zu werden musst du einen gültigen Einladungslink haben. Wenn du keine Einladung erhalten hast kannst du dich nicht registrieren. Wende dich an den Administrator, um einen Link zu erhalten."
|
"welcome-to-mealie": "Willkommen bei Mealie! Um ein Benutzer dieser Instanz zu werden, musst du einen gültigen Einladungslink haben. Wenn du keine Einladung erhalten hast, kannst du dich nicht registrieren. Wende dich an den Administrator, um einen Link zu erhalten."
|
||||||
},
|
},
|
||||||
"tag": {
|
"tag": {
|
||||||
"tag-created": "Schlagwort angelegt",
|
"tag-created": "Schlagwort angelegt",
|
||||||
@@ -972,7 +989,9 @@
|
|||||||
"edit-food": "Lebensmittel bearbeiten",
|
"edit-food": "Lebensmittel bearbeiten",
|
||||||
"food-data": "Lebensmitteldaten",
|
"food-data": "Lebensmitteldaten",
|
||||||
"example-food-singular": "z.B. Zwiebel",
|
"example-food-singular": "z.B. Zwiebel",
|
||||||
"example-food-plural": "z.B. Zwiebeln"
|
"example-food-plural": "z.B. Zwiebeln",
|
||||||
|
"label-overwrite-warning": "Hiermit wird das ausgewählte Etikett allen ausgewählten Lebensmitteln zugewiesen und möglicherweise bestehende Etiketten werden überschrieben.",
|
||||||
|
"on-hand-checkbox-label": "Mit dieser Markierung wird dieses Lebensmittel standardmäßig deaktiviert, wenn ein Rezept einer Einkaufsliste hinzugefügt wird."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Füllt die Datenbank mit gängigen Maßeinheiten basierend auf deiner Sprache.",
|
"seed-dialog-text": "Füllt die Datenbank mit gängigen Maßeinheiten basierend auf deiner Sprache.",
|
||||||
@@ -1000,7 +1019,8 @@
|
|||||||
"seed-dialog-text": "Füllt die Datenbank mit gängigen Etiketten basierend auf deiner Sprache.",
|
"seed-dialog-text": "Füllt die Datenbank mit gängigen Etiketten basierend auf deiner Sprache.",
|
||||||
"edit-label": "Etikett bearbeiten",
|
"edit-label": "Etikett bearbeiten",
|
||||||
"new-label": "Neues Etikett",
|
"new-label": "Neues Etikett",
|
||||||
"labels": "Etiketten"
|
"labels": "Etiketten",
|
||||||
|
"assign-label": "Etikett zuweisen"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Exporte bereinigen",
|
"purge-exports": "Exporte bereinigen",
|
||||||
|
|||||||
@@ -67,7 +67,7 @@
|
|||||||
"message-sent": "Το μήνυμα εστάλη",
|
"message-sent": "Το μήνυμα εστάλη",
|
||||||
"new-notification": "Νέα ειδοποίηση",
|
"new-notification": "Νέα ειδοποίηση",
|
||||||
"event-notifiers": "Ειδοποιητές Συμβάντος",
|
"event-notifiers": "Ειδοποιητές Συμβάντος",
|
||||||
"apprise-url-skipped-if-blank": "Apprise URL (skipped if blank)",
|
"apprise-url-skipped-if-blank": "URL για ενημέρωση (παραλείπεται αν είναι κενό)",
|
||||||
"enable-notifier": "Ενεργοποίηση Ειδοποίησης",
|
"enable-notifier": "Ενεργοποίηση Ειδοποίησης",
|
||||||
"what-events": "What events should this notifier subscribe to?",
|
"what-events": "What events should this notifier subscribe to?",
|
||||||
"user-events": "Συμβάντα Χρήστη",
|
"user-events": "Συμβάντα Χρήστη",
|
||||||
@@ -210,7 +210,8 @@
|
|||||||
"unsaved-changes": "Εχετε μη αποθηκευμένες αλλαγές. Θέλετε να κάνετε αποθήκευση πριν από την αποχώρηση; Εντάξει για αποθήκευση, Ακυρο για απόρριψη των αλλαγών.",
|
"unsaved-changes": "Εχετε μη αποθηκευμένες αλλαγές. Θέλετε να κάνετε αποθήκευση πριν από την αποχώρηση; Εντάξει για αποθήκευση, Ακυρο για απόρριψη των αλλαγών.",
|
||||||
"clipboard-copy-failure": "Η αντιγραφή στο πρόχειρο απέτυχε.",
|
"clipboard-copy-failure": "Η αντιγραφή στο πρόχειρο απέτυχε.",
|
||||||
"confirm-delete-generic-items": "Θέλετε σίγουρα να διαγράψετε τα ακόλουθα αντικείμενα;",
|
"confirm-delete-generic-items": "Θέλετε σίγουρα να διαγράψετε τα ακόλουθα αντικείμενα;",
|
||||||
"organizers": "Οργανωτές"
|
"organizers": "Οργανωτές",
|
||||||
|
"caution": "Προσοχή"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "Θέλετε σίγουρα να διαγράψετε αυτό τον ασφαλή σύνδεσμο <b>{groupName}<b/>;",
|
"are-you-sure-you-want-to-delete-the-group": "Θέλετε σίγουρα να διαγράψετε αυτό τον ασφαλή σύνδεσμο <b>{groupName}<b/>;",
|
||||||
@@ -291,6 +292,8 @@
|
|||||||
"mealplan-updated": "Το Γεύμα Ενημερώθηκε",
|
"mealplan-updated": "Το Γεύμα Ενημερώθηκε",
|
||||||
"no-meal-plan-defined-yet": "Δεν έχει οριστεί ακόμα σχέδιο γεύματος",
|
"no-meal-plan-defined-yet": "Δεν έχει οριστεί ακόμα σχέδιο γεύματος",
|
||||||
"no-meal-planned-for-today": "Δεν έχει προγραμματιστεί γεύμα για σήμερα",
|
"no-meal-planned-for-today": "Δεν έχει προγραμματιστεί γεύμα για σήμερα",
|
||||||
|
"numberOfDays-hint": "Αριθμός ημερών κατά την φόρτωση της σελίδας",
|
||||||
|
"numberOfDays-label": "Προεπιλεγμένες Ημέρες",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Μόνο συνταγές με αυτές τις κατηγορίες θα χρησιμοποιηθούν στα σχέδια γεύματος",
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Μόνο συνταγές με αυτές τις κατηγορίες θα χρησιμοποιηθούν στα σχέδια γεύματος",
|
||||||
"planner": "Προγραμματισμός",
|
"planner": "Προγραμματισμός",
|
||||||
"quick-week": "Γρήγορη προβολή",
|
"quick-week": "Γρήγορη προβολή",
|
||||||
@@ -377,6 +380,10 @@
|
|||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "Κιβώτιο Συνταγών",
|
"title": "Κιβώτιο Συνταγών",
|
||||||
"description-long": "Το Mealie μπορεί να εισάγει συνταγές από το Κουτί Συνταγών. Εξάγετε τις συνταγές σας σε μορφή CSV, στη συνέχεια, ανεβάστε το αρχείο .csv παρακάτω."
|
"description-long": "Το Mealie μπορεί να εισάγει συνταγές από το Κουτί Συνταγών. Εξάγετε τις συνταγές σας σε μορφή CSV, στη συνέχεια, ανεβάστε το αρχείο .csv παρακάτω."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Το Mealie μπορεί να εισάγει συνταγές από το Recipe Keeper. Εξάγετε τις συνταγές σας σε μορφή zip, στη συνέχεια, ανεβάστε το αρχείο .zip παρακάτω."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "Συστατικά",
|
"ingredients": "Συστατικά",
|
||||||
"insert-ingredient": "Insert Ingredient",
|
"insert-ingredient": "Insert Ingredient",
|
||||||
"insert-section": "Εισαγωγή ενότητας",
|
"insert-section": "Εισαγωγή ενότητας",
|
||||||
|
"insert-above": "Εισαγωγή από επάνω",
|
||||||
|
"insert-below": "Εισαγωγή από κάτω",
|
||||||
"instructions": "Οδηγίες",
|
"instructions": "Οδηγίες",
|
||||||
"key-name-required": "Απαιτείται Όνομα Κλειδιού",
|
"key-name-required": "Απαιτείται Όνομα Κλειδιού",
|
||||||
"landscape-view-coming-soon": "Οριζόντια Προβολή (Σύντομα)",
|
"landscape-view-coming-soon": "Οριζόντια Προβολή (Σύντομα)",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "Report deletion failed",
|
"report-deletion-failed": "Report deletion failed",
|
||||||
"recipe-debugger": "Recipe Debugger",
|
"recipe-debugger": "Recipe Debugger",
|
||||||
"recipe-debugger-description": "Grab the URL of the recipe you want to debug and paste it here. The URL will be scraped by the recipe scraper and the results will be displayed. If you don't see any data returned, the site you are trying to scrape is not supported by Mealie or its scraper library.",
|
"recipe-debugger-description": "Grab the URL of the recipe you want to debug and paste it here. The URL will be scraped by the recipe scraper and the results will be displayed. If you don't see any data returned, the site you are trying to scrape is not supported by Mealie or its scraper library.",
|
||||||
|
"use-openai": "Χρήση OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Χρησιμοποιήστε το OpenAI για να αναλύσετε τα αποτελέσματα αντί να βασιστείτε στη βιβλιοθήκη του scraper. Κατά τη δημιουργία μιας συνταγής μέσω URL, αυτό γίνεται αυτόματα αν η βιβλιοθήκη του scraper αποτύχει, αλλά μπορείτε να την δοκιμάσετε χειροκίνητα εδώ.",
|
||||||
"debug": "Debug",
|
"debug": "Debug",
|
||||||
"tree-view": "Tree View",
|
"tree-view": "Tree View",
|
||||||
"recipe-yield": "Recipe Yield",
|
"recipe-yield": "Recipe Yield",
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "Δημιουργήθηκε αντίγραφο ασφαλείας στο: {path}",
|
"backup-created-at-response-export_path": "Δημιουργήθηκε αντίγραφο ασφαλείας στο: {path}",
|
||||||
"backup-deleted": "Το αντίγραφο ασφαλείας διαγράφηκε",
|
"backup-deleted": "Το αντίγραφο ασφαλείας διαγράφηκε",
|
||||||
"restore-success": "Restore successful",
|
"restore-success": "Restore successful",
|
||||||
|
"restore-fail": "Η επαναφορά απέτυχε. Ελέγξτε τα αρχεία καταγραφής του διακομιστή σας για περισσότερες λεπτομέρειες",
|
||||||
"backup-tag": "Ετικέτα Αντιγράφου Ασφαλείας",
|
"backup-tag": "Ετικέτα Αντιγράφου Ασφαλείας",
|
||||||
"create-heading": "Δημιουργία αντιγράφου ασφαλείας",
|
"create-heading": "Δημιουργία αντιγράφου ασφαλείας",
|
||||||
"delete-backup": "Διαγραφή Αντιγράφου Ασφαλείας",
|
"delete-backup": "Διαγραφή Αντιγράφου Ασφαλείας",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "No Linked Recipes|One Linked Recipe|{count} Linked Recipes",
|
"linked-recipes-count": "No Linked Recipes|One Linked Recipe|{count} Linked Recipes",
|
||||||
"items-checked-count": "No items checked|One item checked|{count} items checked",
|
"items-checked-count": "No items checked|One item checked|{count} items checked",
|
||||||
"no-label": "No Label",
|
"no-label": "No Label",
|
||||||
"completed-on": "Completed on {date}"
|
"completed-on": "Completed on {date}",
|
||||||
|
"you-are-offline": "Είστε εκτός σύνδεσης",
|
||||||
|
"you-are-offline-description": "Δεν είναι όλες οι λειτουργίες διαθέσιμες όταν είναι εκτός σύνδεσης. Μπορείτε ακόμα να προσθέσετε, να τροποποιήσετε και να αφαιρέσετε αντικείμενα, αλλά δεν θα μπορείτε να συγχρονίσετε τις αλλαγές σας στο διακομιστή μέχρι να επανέλθει η σύνδεση στο διαδίκτυο.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "Θέλετε σίγουρα να επιλέξετε όλα τα στοιχεία;",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "Θέλετε σίγουρα να αποεπιλέξετε όλα τα στοιχεία;",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "Θέλετε σίγουρα να διαγράψετε όλα τα επιλεγμένα στοιχεία;"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "Συνταγές όλες",
|
"all-recipes": "Συνταγές όλες",
|
||||||
@@ -941,7 +958,7 @@
|
|||||||
"authentication-method-hint": "Αυτό καθορίζει τον τρόπο με τον οποίο ένας χρήστης θα ταυτοποιηθεί με το Mealie. Αν δεν είστε σίγουροι, επιλέξτε 'Mealie'",
|
"authentication-method-hint": "Αυτό καθορίζει τον τρόπο με τον οποίο ένας χρήστης θα ταυτοποιηθεί με το Mealie. Αν δεν είστε σίγουροι, επιλέξτε 'Mealie'",
|
||||||
"permissions": "Permissions",
|
"permissions": "Permissions",
|
||||||
"administrator": "Administrator",
|
"administrator": "Administrator",
|
||||||
"user-can-invite-other-to-group": "User can invite other to group",
|
"user-can-invite-other-to-group": "Ο χρήστης μπορεί να προσκαλέσει άλλους στην ομάδα",
|
||||||
"user-can-manage-group": "User can manage group",
|
"user-can-manage-group": "User can manage group",
|
||||||
"user-can-organize-group-data": "User can organize group data",
|
"user-can-organize-group-data": "User can organize group data",
|
||||||
"enable-advanced-features": "Ενεργοποίηση χαρακτηριστικών για προχωρημένους",
|
"enable-advanced-features": "Ενεργοποίηση χαρακτηριστικών για προχωρημένους",
|
||||||
@@ -972,7 +989,9 @@
|
|||||||
"edit-food": "Edit Food",
|
"edit-food": "Edit Food",
|
||||||
"food-data": "Food Data",
|
"food-data": "Food Data",
|
||||||
"example-food-singular": "ex: Onion",
|
"example-food-singular": "ex: Onion",
|
||||||
"example-food-plural": "ex: Onions"
|
"example-food-plural": "ex: Onions",
|
||||||
|
"label-overwrite-warning": "This will assign the chosen label to all selected foods and potentially overwrite your existing labels.",
|
||||||
|
"on-hand-checkbox-label": "Η ρύθμιση αυτής της σημαίας θα κάνει αυτό το φαγητό αποεπιλεγμένο από προεπιλογή κατά την προσθήκη μιας συνταγής σε μια λίστα αγορών."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Seed the database with common units based on your local language.",
|
"seed-dialog-text": "Seed the database with common units based on your local language.",
|
||||||
@@ -1000,7 +1019,8 @@
|
|||||||
"seed-dialog-text": "Seed the database with common labels based on your local language.",
|
"seed-dialog-text": "Seed the database with common labels based on your local language.",
|
||||||
"edit-label": "Edit Label",
|
"edit-label": "Edit Label",
|
||||||
"new-label": "New Label",
|
"new-label": "New Label",
|
||||||
"labels": "Labels"
|
"labels": "Labels",
|
||||||
|
"assign-label": "Ορισμός Ετικέτας"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Purge Exports",
|
"purge-exports": "Purge Exports",
|
||||||
|
|||||||
@@ -189,10 +189,10 @@
|
|||||||
"delete-with-name": "Delete {name}",
|
"delete-with-name": "Delete {name}",
|
||||||
"confirm-delete-generic-with-name": "Are you sure you want to delete this {name}?",
|
"confirm-delete-generic-with-name": "Are you sure you want to delete this {name}?",
|
||||||
"confirm-delete-own-admin-account": "Please note that you are trying to delete your own admin account! This action cannot be undone and will permanently delete your account?",
|
"confirm-delete-own-admin-account": "Please note that you are trying to delete your own admin account! This action cannot be undone and will permanently delete your account?",
|
||||||
"organizer": "Organizer",
|
"organizer": "Organiser",
|
||||||
"transfer": "Transfer",
|
"transfer": "Transfer",
|
||||||
"copy": "Copy",
|
"copy": "Copy",
|
||||||
"color": "Color",
|
"color": "Colour",
|
||||||
"timestamp": "Timestamp",
|
"timestamp": "Timestamp",
|
||||||
"last-made": "Last Made",
|
"last-made": "Last Made",
|
||||||
"learn-more": "Learn More",
|
"learn-more": "Learn More",
|
||||||
@@ -210,7 +210,8 @@
|
|||||||
"unsaved-changes": "You have unsaved changes. Do you want to save before leaving? Okay to save, Cancel to discard changes.",
|
"unsaved-changes": "You have unsaved changes. Do you want to save before leaving? Okay to save, Cancel to discard changes.",
|
||||||
"clipboard-copy-failure": "Failed to copy to the clipboard.",
|
"clipboard-copy-failure": "Failed to copy to the clipboard.",
|
||||||
"confirm-delete-generic-items": "Are you sure you want to delete the following items?",
|
"confirm-delete-generic-items": "Are you sure you want to delete the following items?",
|
||||||
"organizers": "Organisers"
|
"organizers": "Organisers",
|
||||||
|
"caution": "Caution"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "Are you sure you want to delete <b>{groupName}<b/>?",
|
"are-you-sure-you-want-to-delete-the-group": "Are you sure you want to delete <b>{groupName}<b/>?",
|
||||||
@@ -267,7 +268,7 @@
|
|||||||
"group-management": "Group Management",
|
"group-management": "Group Management",
|
||||||
"admin-group-management": "Admin Group Management",
|
"admin-group-management": "Admin Group Management",
|
||||||
"admin-group-management-text": "Changes to this group will be reflected immediately.",
|
"admin-group-management-text": "Changes to this group will be reflected immediately.",
|
||||||
"group-id-value": "Group Id: {0}"
|
"group-id-value": "Group ID: {0}"
|
||||||
},
|
},
|
||||||
"meal-plan": {
|
"meal-plan": {
|
||||||
"create-a-new-meal-plan": "Create a New Meal Plan",
|
"create-a-new-meal-plan": "Create a New Meal Plan",
|
||||||
@@ -282,16 +283,18 @@
|
|||||||
"meal-planner": "Meal Planner",
|
"meal-planner": "Meal Planner",
|
||||||
"meal-plans": "Meal Plans",
|
"meal-plans": "Meal Plans",
|
||||||
"mealplan-categories": "MEALPLAN CATEGORIES",
|
"mealplan-categories": "MEALPLAN CATEGORIES",
|
||||||
"mealplan-created": "Mealplan created",
|
"mealplan-created": "Meal plan created",
|
||||||
"mealplan-creation-failed": "Mealplan creation failed",
|
"mealplan-creation-failed": "Meal plan creation failed",
|
||||||
"mealplan-deleted": "Mealplan deleted",
|
"mealplan-deleted": "Mealplan deleted",
|
||||||
"mealplan-deletion-failed": "Mealplan deletion failed",
|
"mealplan-deletion-failed": "Meal plan deletion failed",
|
||||||
"mealplan-settings": "Mealplan Settings",
|
"mealplan-settings": "Meal plan settings",
|
||||||
"mealplan-update-failed": "Mealplan update failed",
|
"mealplan-update-failed": "Meal plan update failed",
|
||||||
"mealplan-updated": "Mealplan Updated",
|
"mealplan-updated": "Meal plan updated",
|
||||||
"no-meal-plan-defined-yet": "No meal plan defined yet",
|
"no-meal-plan-defined-yet": "No meal plan defined yet",
|
||||||
"no-meal-planned-for-today": "No meal planned for today",
|
"no-meal-planned-for-today": "No meal planned for today",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Only recipes with these categories will be used in Meal Plans",
|
"numberOfDays-hint": "Number of days on page load",
|
||||||
|
"numberOfDays-label": "Default Days",
|
||||||
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Only the recipes with these categories will be used in Meal Plans",
|
||||||
"planner": "Planner",
|
"planner": "Planner",
|
||||||
"quick-week": "Quick Week",
|
"quick-week": "Quick Week",
|
||||||
"side": "Side",
|
"side": "Side",
|
||||||
@@ -377,6 +380,10 @@
|
|||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "My Recipe Box",
|
"title": "My Recipe Box",
|
||||||
"description-long": "Mealie can import recipes from My Recipe Box. Export your recipes in CSV format, then upload the .csv file below."
|
"description-long": "Mealie can import recipes from My Recipe Box. Export your recipes in CSV format, then upload the .csv file below."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Mealie can import recipes from Recipe Keeper. Export your recipes in zip format, then upload the .zip file below."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "Ingredients",
|
"ingredients": "Ingredients",
|
||||||
"insert-ingredient": "Insert Ingredient",
|
"insert-ingredient": "Insert Ingredient",
|
||||||
"insert-section": "Insert Section",
|
"insert-section": "Insert Section",
|
||||||
|
"insert-above": "Insert Above",
|
||||||
|
"insert-below": "Insert Below",
|
||||||
"instructions": "Instructions",
|
"instructions": "Instructions",
|
||||||
"key-name-required": "Key Name Required",
|
"key-name-required": "Key Name Required",
|
||||||
"landscape-view-coming-soon": "Landscape View (Coming Soon)",
|
"landscape-view-coming-soon": "Landscape View (Coming Soon)",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "Report deletion failed",
|
"report-deletion-failed": "Report deletion failed",
|
||||||
"recipe-debugger": "Recipe Debugger",
|
"recipe-debugger": "Recipe Debugger",
|
||||||
"recipe-debugger-description": "Grab the URL of the recipe you want to debug and paste it here. The URL will be scraped by the recipe scraper and the results will be displayed. If you don't see any data returned, the site you are trying to scrape is not supported by Mealie or its scraper library.",
|
"recipe-debugger-description": "Grab the URL of the recipe you want to debug and paste it here. The URL will be scraped by the recipe scraper and the results will be displayed. If you don't see any data returned, the site you are trying to scrape is not supported by Mealie or its scraper library.",
|
||||||
|
"use-openai": "Use OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Use OpenAI to parse the results instead of relying on the scraper library. When creating a recipe via URL, this is done automatically if the scraper library fails, but you may test it manually here.",
|
||||||
"debug": "Debug",
|
"debug": "Debug",
|
||||||
"tree-view": "Tree View",
|
"tree-view": "Tree View",
|
||||||
"recipe-yield": "Recipe Yield",
|
"recipe-yield": "Recipe Yield",
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "Backup Created at {path}",
|
"backup-created-at-response-export_path": "Backup Created at {path}",
|
||||||
"backup-deleted": "Backup deleted",
|
"backup-deleted": "Backup deleted",
|
||||||
"restore-success": "Restore successful",
|
"restore-success": "Restore successful",
|
||||||
|
"restore-fail": "Restore failed. Check your server logs for more details",
|
||||||
"backup-tag": "Backup Tag",
|
"backup-tag": "Backup Tag",
|
||||||
"create-heading": "Create a Backup",
|
"create-heading": "Create a Backup",
|
||||||
"delete-backup": "Delete Backup",
|
"delete-backup": "Delete Backup",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "No Linked Recipes|One Linked Recipe|{count} Linked Recipes",
|
"linked-recipes-count": "No Linked Recipes|One Linked Recipe|{count} Linked Recipes",
|
||||||
"items-checked-count": "No items checked|One item checked|{count} items checked",
|
"items-checked-count": "No items checked|One item checked|{count} items checked",
|
||||||
"no-label": "No Label",
|
"no-label": "No Label",
|
||||||
"completed-on": "Completed on {date}"
|
"completed-on": "Completed on {date}",
|
||||||
|
"you-are-offline": "You are offline",
|
||||||
|
"you-are-offline-description": "Not all features are available while offline. You can still add, modify, and remove items, but you will not be able to sync your changes to the server until you are back online.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "Are you sure you want to check all items?",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "Are you sure you want to uncheck all items?",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "Are you sure you want to delete all checked items?"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "All Recipes",
|
"all-recipes": "All Recipes",
|
||||||
@@ -918,13 +935,13 @@
|
|||||||
"you-are-not-allowed-to-delete-this-user": "You are not allowed to delete this user",
|
"you-are-not-allowed-to-delete-this-user": "You are not allowed to delete this user",
|
||||||
"enable-advanced-content": "Enable Advanced Content",
|
"enable-advanced-content": "Enable Advanced Content",
|
||||||
"enable-advanced-content-description": "Enables advanced features like Recipe Scaling, API keys, Webhooks, and Data Management. Don't worry, you can always change this later",
|
"enable-advanced-content-description": "Enables advanced features like Recipe Scaling, API keys, Webhooks, and Data Management. Don't worry, you can always change this later",
|
||||||
"favorite-recipes": "Favorite Recipes",
|
"favorite-recipes": "Favourite Recipes",
|
||||||
"email-or-username": "Email or Username",
|
"email-or-username": "Email or Username",
|
||||||
"remember-me": "Remember Me",
|
"remember-me": "Remember Me",
|
||||||
"please-enter-your-email-and-password": "Please enter your email and password",
|
"please-enter-your-email-and-password": "Please enter your email and password",
|
||||||
"invalid-credentials": "Invalid Credentials",
|
"invalid-credentials": "Invalid Credentials",
|
||||||
"account-locked-please-try-again-later": "Account Locked. Please try again later",
|
"account-locked-please-try-again-later": "Account Locked. Please try again later",
|
||||||
"user-favorites": "User Favorites",
|
"user-favorites": "User Favourites",
|
||||||
"password-strength-values": {
|
"password-strength-values": {
|
||||||
"weak": "Weak",
|
"weak": "Weak",
|
||||||
"good": "Good",
|
"good": "Good",
|
||||||
@@ -941,7 +958,7 @@
|
|||||||
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie'",
|
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie'",
|
||||||
"permissions": "Permissions",
|
"permissions": "Permissions",
|
||||||
"administrator": "Administrator",
|
"administrator": "Administrator",
|
||||||
"user-can-invite-other-to-group": "User can invite other to group",
|
"user-can-invite-other-to-group": "User can invite others to group",
|
||||||
"user-can-manage-group": "User can manage group",
|
"user-can-manage-group": "User can manage group",
|
||||||
"user-can-organize-group-data": "User can organize group data",
|
"user-can-organize-group-data": "User can organize group data",
|
||||||
"enable-advanced-features": "Enable advanced features",
|
"enable-advanced-features": "Enable advanced features",
|
||||||
@@ -972,7 +989,9 @@
|
|||||||
"edit-food": "Edit Food",
|
"edit-food": "Edit Food",
|
||||||
"food-data": "Food Data",
|
"food-data": "Food Data",
|
||||||
"example-food-singular": "ex: Onion",
|
"example-food-singular": "ex: Onion",
|
||||||
"example-food-plural": "ex: Onions"
|
"example-food-plural": "ex: Onions",
|
||||||
|
"label-overwrite-warning": "This will assign the chosen label to all selected foods and potentially overwrite your existing labels.",
|
||||||
|
"on-hand-checkbox-label": "Setting this flag will make this food unchecked by default when adding a recipe to a shopping list."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Seed the database with common units based on your local language.",
|
"seed-dialog-text": "Seed the database with common units based on your local language.",
|
||||||
@@ -1000,7 +1019,8 @@
|
|||||||
"seed-dialog-text": "Seed the database with common labels based on your local language.",
|
"seed-dialog-text": "Seed the database with common labels based on your local language.",
|
||||||
"edit-label": "Edit Label",
|
"edit-label": "Edit Label",
|
||||||
"new-label": "New Label",
|
"new-label": "New Label",
|
||||||
"labels": "Labels"
|
"labels": "Labels",
|
||||||
|
"assign-label": "Assign Label"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Purge Exports",
|
"purge-exports": "Purge Exports",
|
||||||
|
|||||||
@@ -210,7 +210,8 @@
|
|||||||
"unsaved-changes": "You have unsaved changes. Do you want to save before leaving? Okay to save, Cancel to discard changes.",
|
"unsaved-changes": "You have unsaved changes. Do you want to save before leaving? Okay to save, Cancel to discard changes.",
|
||||||
"clipboard-copy-failure": "Failed to copy to the clipboard.",
|
"clipboard-copy-failure": "Failed to copy to the clipboard.",
|
||||||
"confirm-delete-generic-items": "Are you sure you want to delete the following items?",
|
"confirm-delete-generic-items": "Are you sure you want to delete the following items?",
|
||||||
"organizers": "Organizers"
|
"organizers": "Organizers",
|
||||||
|
"caution": "Caution"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "Are you sure you want to delete <b>{groupName}<b/>?",
|
"are-you-sure-you-want-to-delete-the-group": "Are you sure you want to delete <b>{groupName}<b/>?",
|
||||||
@@ -291,6 +292,8 @@
|
|||||||
"mealplan-updated": "Mealplan Updated",
|
"mealplan-updated": "Mealplan Updated",
|
||||||
"no-meal-plan-defined-yet": "No meal plan defined yet",
|
"no-meal-plan-defined-yet": "No meal plan defined yet",
|
||||||
"no-meal-planned-for-today": "No meal planned for today",
|
"no-meal-planned-for-today": "No meal planned for today",
|
||||||
|
"numberOfDays-hint": "Number of days on page load",
|
||||||
|
"numberOfDays-label": "Default Days",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Only recipes with these categories will be used in Meal Plans",
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Only recipes with these categories will be used in Meal Plans",
|
||||||
"planner": "Planner",
|
"planner": "Planner",
|
||||||
"quick-week": "Quick Week",
|
"quick-week": "Quick Week",
|
||||||
@@ -377,6 +380,10 @@
|
|||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "My Recipe Box",
|
"title": "My Recipe Box",
|
||||||
"description-long": "Mealie can import recipes from My Recipe Box. Export your recipes in CSV format, then upload the .csv file below."
|
"description-long": "Mealie can import recipes from My Recipe Box. Export your recipes in CSV format, then upload the .csv file below."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Mealie can import recipes from Recipe Keeper. Export your recipes in zip format, then upload the .zip file below."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "Ingredients",
|
"ingredients": "Ingredients",
|
||||||
"insert-ingredient": "Insert Ingredient",
|
"insert-ingredient": "Insert Ingredient",
|
||||||
"insert-section": "Insert Section",
|
"insert-section": "Insert Section",
|
||||||
|
"insert-above": "Insert Above",
|
||||||
|
"insert-below": "Insert Below",
|
||||||
"instructions": "Instructions",
|
"instructions": "Instructions",
|
||||||
"key-name-required": "Key Name Required",
|
"key-name-required": "Key Name Required",
|
||||||
"landscape-view-coming-soon": "Landscape View",
|
"landscape-view-coming-soon": "Landscape View",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "Report deletion failed",
|
"report-deletion-failed": "Report deletion failed",
|
||||||
"recipe-debugger": "Recipe Debugger",
|
"recipe-debugger": "Recipe Debugger",
|
||||||
"recipe-debugger-description": "Grab the URL of the recipe you want to debug and paste it here. The URL will be scraped by the recipe scraper and the results will be displayed. If you don't see any data returned, the site you are trying to scrape is not supported by Mealie or its scraper library.",
|
"recipe-debugger-description": "Grab the URL of the recipe you want to debug and paste it here. The URL will be scraped by the recipe scraper and the results will be displayed. If you don't see any data returned, the site you are trying to scrape is not supported by Mealie or its scraper library.",
|
||||||
|
"use-openai": "Use OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Use OpenAI to parse the results instead of relying on the scraper library. When creating a recipe via URL, this is done automatically if the scraper library fails, but you may test it manually here.",
|
||||||
"debug": "Debug",
|
"debug": "Debug",
|
||||||
"tree-view": "Tree View",
|
"tree-view": "Tree View",
|
||||||
"recipe-yield": "Recipe Yield",
|
"recipe-yield": "Recipe Yield",
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "Backup Created at {path}",
|
"backup-created-at-response-export_path": "Backup Created at {path}",
|
||||||
"backup-deleted": "Backup deleted",
|
"backup-deleted": "Backup deleted",
|
||||||
"restore-success": "Restore successful",
|
"restore-success": "Restore successful",
|
||||||
|
"restore-fail": "Restore failed. Check your server logs for more details",
|
||||||
"backup-tag": "Backup Tag",
|
"backup-tag": "Backup Tag",
|
||||||
"create-heading": "Create A Backup",
|
"create-heading": "Create A Backup",
|
||||||
"delete-backup": "Delete Backup",
|
"delete-backup": "Delete Backup",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "No Linked Recipes|One Linked Recipe|{count} Linked Recipes",
|
"linked-recipes-count": "No Linked Recipes|One Linked Recipe|{count} Linked Recipes",
|
||||||
"items-checked-count": "No items checked|One item checked|{count} items checked",
|
"items-checked-count": "No items checked|One item checked|{count} items checked",
|
||||||
"no-label": "No Label",
|
"no-label": "No Label",
|
||||||
"completed-on": "Completed on {date}"
|
"completed-on": "Completed on {date}",
|
||||||
|
"you-are-offline": "You are offline",
|
||||||
|
"you-are-offline-description": "Not all features are available while offline. You can still add, modify, and remove items, but you will not be able to sync your changes to the server until you are back online.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "Are you sure you want to check all items?",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "Are you sure you want to uncheck all items?",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "Are you sure you want to delete all checked items?"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "All Recipes",
|
"all-recipes": "All Recipes",
|
||||||
@@ -941,7 +958,7 @@
|
|||||||
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie'",
|
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie'",
|
||||||
"permissions": "Permissions",
|
"permissions": "Permissions",
|
||||||
"administrator": "Administrator",
|
"administrator": "Administrator",
|
||||||
"user-can-invite-other-to-group": "User can invite other to group",
|
"user-can-invite-other-to-group": "User can invite others to group",
|
||||||
"user-can-manage-group": "User can manage group",
|
"user-can-manage-group": "User can manage group",
|
||||||
"user-can-organize-group-data": "User can organize group data",
|
"user-can-organize-group-data": "User can organize group data",
|
||||||
"enable-advanced-features": "Enable advanced features",
|
"enable-advanced-features": "Enable advanced features",
|
||||||
@@ -972,7 +989,9 @@
|
|||||||
"edit-food": "Edit Food",
|
"edit-food": "Edit Food",
|
||||||
"food-data": "Food Data",
|
"food-data": "Food Data",
|
||||||
"example-food-singular": "ex: Onion",
|
"example-food-singular": "ex: Onion",
|
||||||
"example-food-plural": "ex: Onions"
|
"example-food-plural": "ex: Onions",
|
||||||
|
"label-overwrite-warning": "This will assign the chosen label to all selected foods and potentially overwrite your existing labels.",
|
||||||
|
"on-hand-checkbox-label": "Setting this flag will make this food unchecked by default when adding a recipe to a shopping list."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Seed the database with common units based on your local language.",
|
"seed-dialog-text": "Seed the database with common units based on your local language.",
|
||||||
@@ -1000,7 +1019,8 @@
|
|||||||
"seed-dialog-text": "Seed the database with common labels based on your local language.",
|
"seed-dialog-text": "Seed the database with common labels based on your local language.",
|
||||||
"edit-label": "Edit Label",
|
"edit-label": "Edit Label",
|
||||||
"new-label": "New Label",
|
"new-label": "New Label",
|
||||||
"labels": "Labels"
|
"labels": "Labels",
|
||||||
|
"assign-label": "Assign Label"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Purge Exports",
|
"purge-exports": "Purge Exports",
|
||||||
|
|||||||
@@ -210,7 +210,8 @@
|
|||||||
"unsaved-changes": "Tienes cambios sin guardar. ¿Quieres guardar antes de salir? Aceptar para guardar, Cancelar para descartar cambios.",
|
"unsaved-changes": "Tienes cambios sin guardar. ¿Quieres guardar antes de salir? Aceptar para guardar, Cancelar para descartar cambios.",
|
||||||
"clipboard-copy-failure": "No se pudo copiar al portapapeles.",
|
"clipboard-copy-failure": "No se pudo copiar al portapapeles.",
|
||||||
"confirm-delete-generic-items": "¿Estás seguro que quieres eliminar los siguientes elementos?",
|
"confirm-delete-generic-items": "¿Estás seguro que quieres eliminar los siguientes elementos?",
|
||||||
"organizers": "Organizadores"
|
"organizers": "Organizadores",
|
||||||
|
"caution": "Precaución"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "Por favor, confirma que deseas eliminar <b>{groupName}<b/>",
|
"are-you-sure-you-want-to-delete-the-group": "Por favor, confirma que deseas eliminar <b>{groupName}<b/>",
|
||||||
@@ -291,6 +292,8 @@
|
|||||||
"mealplan-updated": "Menú actualizado",
|
"mealplan-updated": "Menú actualizado",
|
||||||
"no-meal-plan-defined-yet": "Todavía no hay ningún menú",
|
"no-meal-plan-defined-yet": "Todavía no hay ningún menú",
|
||||||
"no-meal-planned-for-today": "No hay ningún menú para hoy",
|
"no-meal-planned-for-today": "No hay ningún menú para hoy",
|
||||||
|
"numberOfDays-hint": "Número de días al cargar la página",
|
||||||
|
"numberOfDays-label": "Días por defecto",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Sólo las recetas con estas categorías se utilizarán en los menús",
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Sólo las recetas con estas categorías se utilizarán en los menús",
|
||||||
"planner": "Planificador",
|
"planner": "Planificador",
|
||||||
"quick-week": "Plan rápido",
|
"quick-week": "Plan rápido",
|
||||||
@@ -372,11 +375,15 @@
|
|||||||
"mealie-text": "Mealie puede importar recetas de la aplicación Mealie desde una versión anterior a v1.0. Exporta tus recetas de tu antigua instancia y sube el archivo zip a continuación. Ten en cuenta que solo se pueden importar recetas de la exportación.",
|
"mealie-text": "Mealie puede importar recetas de la aplicación Mealie desde una versión anterior a v1.0. Exporta tus recetas de tu antigua instancia y sube el archivo zip a continuación. Ten en cuenta que solo se pueden importar recetas de la exportación.",
|
||||||
"plantoeat": {
|
"plantoeat": {
|
||||||
"title": "Plan to Eat",
|
"title": "Plan to Eat",
|
||||||
"description-long": "Mealie puede importar recetas de Plan a Comer."
|
"description-long": "Mealie puede importar recetas de Plan to Eat."
|
||||||
},
|
},
|
||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "My Recipe Box",
|
"title": "My Recipe Box",
|
||||||
"description-long": "Mealie puede importar recetas de My Recipe Box. Exporta tus recetas en formato CSV y sube el archivo a continuación."
|
"description-long": "Mealie puede importar recetas de My Recipe Box. Exporta tus recetas en formato CSV y sube el archivo a continuación."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Mealie puede importar recetas de Recipe Keeper. Exporta tus recetas en formato zip y luego sube el archivo a continuación."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "Ingredientes",
|
"ingredients": "Ingredientes",
|
||||||
"insert-ingredient": "Agregar ingrediente",
|
"insert-ingredient": "Agregar ingrediente",
|
||||||
"insert-section": "Insertar sección",
|
"insert-section": "Insertar sección",
|
||||||
|
"insert-above": "Insertar Arriba",
|
||||||
|
"insert-below": "Insertar Debajo",
|
||||||
"instructions": "Instrucciones",
|
"instructions": "Instrucciones",
|
||||||
"key-name-required": "Nombre de clave requerido",
|
"key-name-required": "Nombre de clave requerido",
|
||||||
"landscape-view-coming-soon": "Vista apaisada (Próximamente)",
|
"landscape-view-coming-soon": "Vista apaisada (Próximamente)",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "Error al eliminar el reporte",
|
"report-deletion-failed": "Error al eliminar el reporte",
|
||||||
"recipe-debugger": "Depurador de recetas",
|
"recipe-debugger": "Depurador de recetas",
|
||||||
"recipe-debugger-description": "Obtenga la URL de la receta que desea depurar y pegala aquí. La URL será analizadada por el extractor de recetas y los resultados se mostrarán. Si no ves ningún dato devuelta, el sitio que estás intentando rascar no está soportado por Mealie o su biblioteca de analizadores.",
|
"recipe-debugger-description": "Obtenga la URL de la receta que desea depurar y pegala aquí. La URL será analizadada por el extractor de recetas y los resultados se mostrarán. Si no ves ningún dato devuelta, el sitio que estás intentando rascar no está soportado por Mealie o su biblioteca de analizadores.",
|
||||||
|
"use-openai": "Usar OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Utilice OpenAI para analizar los resultados en lugar de depender de la biblioteca de analizadores. Cuando se crea una receta a través de la URL, esto se hace automáticamente si la biblioteca del analizador falla, pero puede probarla manualmente aquí.",
|
||||||
"debug": "Depuración",
|
"debug": "Depuración",
|
||||||
"tree-view": "Vista en árbol",
|
"tree-view": "Vista en árbol",
|
||||||
"recipe-yield": "Porciones",
|
"recipe-yield": "Porciones",
|
||||||
@@ -594,7 +605,7 @@
|
|||||||
"select-parser": "Seleccionar Analizador",
|
"select-parser": "Seleccionar Analizador",
|
||||||
"natural-language-processor": "Procesador de Lenguaje Natural",
|
"natural-language-processor": "Procesador de Lenguaje Natural",
|
||||||
"brute-parser": "Analizador Bruto",
|
"brute-parser": "Analizador Bruto",
|
||||||
"openai-parser": "OpenAI Parser",
|
"openai-parser": "Analizador OpenAI",
|
||||||
"parse-all": "Analizar Todo",
|
"parse-all": "Analizar Todo",
|
||||||
"no-unit": "Sin unidad",
|
"no-unit": "Sin unidad",
|
||||||
"missing-unit": "Crear unidad faltante: {unit}",
|
"missing-unit": "Crear unidad faltante: {unit}",
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "Copia de seguridad creada en {path}",
|
"backup-created-at-response-export_path": "Copia de seguridad creada en {path}",
|
||||||
"backup-deleted": "Copia de seguridad eliminada",
|
"backup-deleted": "Copia de seguridad eliminada",
|
||||||
"restore-success": "Restauración exitosa",
|
"restore-success": "Restauración exitosa",
|
||||||
|
"restore-fail": "Restauración fallida. Compruebe los registros de su servidor para más detalles",
|
||||||
"backup-tag": "Etiqueta de la copia de seguridad",
|
"backup-tag": "Etiqueta de la copia de seguridad",
|
||||||
"create-heading": "Crear una copia de seguridad",
|
"create-heading": "Crear una copia de seguridad",
|
||||||
"delete-backup": "Eliminar copia de seguridad",
|
"delete-backup": "Eliminar copia de seguridad",
|
||||||
@@ -748,7 +760,7 @@
|
|||||||
"not-ready": "No Listo - Comprobar variables de ambiente",
|
"not-ready": "No Listo - Comprobar variables de ambiente",
|
||||||
"succeeded": "Logrado",
|
"succeeded": "Logrado",
|
||||||
"failed": "Error",
|
"failed": "Error",
|
||||||
"general-about": "General Acerca de",
|
"general-about": "Información General",
|
||||||
"application-version": "Versión de la aplicación",
|
"application-version": "Versión de la aplicación",
|
||||||
"application-version-error-text": "Su versión actual ({0}) no coincide con la última versión. Considere actualizar a la última versión ({1}).",
|
"application-version-error-text": "Su versión actual ({0}) no coincide con la última versión. Considere actualizar a la última versión ({1}).",
|
||||||
"mealie-is-up-to-date": "Mealie está actualizada",
|
"mealie-is-up-to-date": "Mealie está actualizada",
|
||||||
@@ -766,9 +778,9 @@
|
|||||||
"oidc-ready": "OIDC Listo",
|
"oidc-ready": "OIDC Listo",
|
||||||
"oidc-ready-error-text": "No todos los valores OIDC están configurados. Puedes ignorar esto si no estás usando autenticación OIDC.",
|
"oidc-ready-error-text": "No todos los valores OIDC están configurados. Puedes ignorar esto si no estás usando autenticación OIDC.",
|
||||||
"oidc-ready-success-text": "Todas las variables OIDC requeridas están configuradas.",
|
"oidc-ready-success-text": "Todas las variables OIDC requeridas están configuradas.",
|
||||||
"openai-ready": "OpenAI Ready",
|
"openai-ready": "OpenAI Listo",
|
||||||
"openai-ready-error-text": "Not all OpenAI Values are configured. This can be ignored if you are not using OpenAI features.",
|
"openai-ready-error-text": "No todos los valores OpenAI están configurados. Puedes ignorar esto si no estás usando funciones de OpenAI.",
|
||||||
"openai-ready-success-text": "Required OpenAI variables are all set."
|
"openai-ready-success-text": "Las variables OpenAI requeridas están todas definidas."
|
||||||
},
|
},
|
||||||
"shopping-list": {
|
"shopping-list": {
|
||||||
"all-lists": "Todas las listas",
|
"all-lists": "Todas las listas",
|
||||||
@@ -782,7 +794,7 @@
|
|||||||
"food": "Alimentos",
|
"food": "Alimentos",
|
||||||
"note": "Nota",
|
"note": "Nota",
|
||||||
"label": "Etiqueta",
|
"label": "Etiqueta",
|
||||||
"save-label": "Save Label",
|
"save-label": "Guardar etiqueta",
|
||||||
"linked-item-warning": "Este elemento está vinculado a una o más recetas. Ajustar las unidades o los alimentos producirá resultados inesperados al añadir o quitar la receta de esta lista.",
|
"linked-item-warning": "Este elemento está vinculado a una o más recetas. Ajustar las unidades o los alimentos producirá resultados inesperados al añadir o quitar la receta de esta lista.",
|
||||||
"toggle-food": "Mostrar nombre del alimento",
|
"toggle-food": "Mostrar nombre del alimento",
|
||||||
"manage-labels": "Administrar etiquetas",
|
"manage-labels": "Administrar etiquetas",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "No hay recetas vinculadas|Una receta vinculada|{count} recetas vinculadas",
|
"linked-recipes-count": "No hay recetas vinculadas|Una receta vinculada|{count} recetas vinculadas",
|
||||||
"items-checked-count": "Ningún elemento comprobado|Un elemento comprobado|{count} elementos comprobados",
|
"items-checked-count": "Ningún elemento comprobado|Un elemento comprobado|{count} elementos comprobados",
|
||||||
"no-label": "Sin Etiqueta",
|
"no-label": "Sin Etiqueta",
|
||||||
"completed-on": "Completado el {date}"
|
"completed-on": "Completado el {date}",
|
||||||
|
"you-are-offline": "Estás sin conexión",
|
||||||
|
"you-are-offline-description": "No todas las características están disponibles mientras esté fuera de línea. Todavía puedes añadir, modificar y eliminar elementos, pero no podrá sincronizar sus cambios en el servidor hasta que vuelva a estar en línea.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "¿Seguro que quieres seleccionar todos los elementos?",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "¿Seguro que quieres de-seleccionar todos los elementos?",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "¿Está seguro que deseas eliminar los elementos seleccionados?"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "Recetas",
|
"all-recipes": "Recetas",
|
||||||
@@ -941,7 +958,7 @@
|
|||||||
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie",
|
"authentication-method-hint": "This specifies how a user will authenticate with Mealie. If you're not sure, choose 'Mealie",
|
||||||
"permissions": "Permisos",
|
"permissions": "Permisos",
|
||||||
"administrator": "Administrador",
|
"administrator": "Administrador",
|
||||||
"user-can-invite-other-to-group": "El usuario puede invitar a otros al grupo",
|
"user-can-invite-other-to-group": "User can invite others to group",
|
||||||
"user-can-manage-group": "El usuario puede administrar el grupo",
|
"user-can-manage-group": "El usuario puede administrar el grupo",
|
||||||
"user-can-organize-group-data": "El usuario puede organizar los datos del grupo",
|
"user-can-organize-group-data": "El usuario puede organizar los datos del grupo",
|
||||||
"enable-advanced-features": "Habilitar Características Avanzadas",
|
"enable-advanced-features": "Habilitar Características Avanzadas",
|
||||||
@@ -972,7 +989,9 @@
|
|||||||
"edit-food": "Editar Alimento",
|
"edit-food": "Editar Alimento",
|
||||||
"food-data": "Datos de Alimento",
|
"food-data": "Datos de Alimento",
|
||||||
"example-food-singular": "ej: Cebolla",
|
"example-food-singular": "ej: Cebolla",
|
||||||
"example-food-plural": "ej: Cebollas"
|
"example-food-plural": "ej: Cebollas",
|
||||||
|
"label-overwrite-warning": "Esto asignará la etiqueta elegida a todos los alimentos seleccionados y potencialmente sobrescribirá las etiquetas existentes.",
|
||||||
|
"on-hand-checkbox-label": "Establecer esta bandera hará que este alimento se desmarque por defecto al añadir una receta a una lista de compras."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Añade a la base de datos unidades comunes basadas en su idioma local.",
|
"seed-dialog-text": "Añade a la base de datos unidades comunes basadas en su idioma local.",
|
||||||
@@ -1000,7 +1019,8 @@
|
|||||||
"seed-dialog-text": "Añade a la base de datos etiquetas comunes basadas en su idioma local.",
|
"seed-dialog-text": "Añade a la base de datos etiquetas comunes basadas en su idioma local.",
|
||||||
"edit-label": "Editar etiqueta",
|
"edit-label": "Editar etiqueta",
|
||||||
"new-label": "Nueva etiqueta",
|
"new-label": "Nueva etiqueta",
|
||||||
"labels": "Etiquetas"
|
"labels": "Etiquetas",
|
||||||
|
"assign-label": "Asignar Etiqueta"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Limpiar exportaciones",
|
"purge-exports": "Limpiar exportaciones",
|
||||||
@@ -1137,14 +1157,14 @@
|
|||||||
"admin": {
|
"admin": {
|
||||||
"maintenance": {
|
"maintenance": {
|
||||||
"storage-details": "Detalle del almacenamiento",
|
"storage-details": "Detalle del almacenamiento",
|
||||||
"page-title": "Mantenimiento del sitio",
|
"page-title": "Mantenimiento del Sitio",
|
||||||
"summary-title": "Índice",
|
"summary-title": "Índice",
|
||||||
"button-label-get-summary": "Obtener Resumen",
|
"button-label-get-summary": "Obtener Resumen",
|
||||||
"button-label-open-details": "Detalles",
|
"button-label-open-details": "Detalles",
|
||||||
"info-description-data-dir-size": "Tamaño del directorio de datos",
|
"info-description-data-dir-size": "Tamaño del Directorio de Datos",
|
||||||
"info-description-log-file-size": "Tamaño del archivo de registro",
|
"info-description-log-file-size": "Tamaño del archivo de registro",
|
||||||
"info-description-cleanable-directories": "Directorios eliminables",
|
"info-description-cleanable-directories": "Directorios Eliminables",
|
||||||
"info-description-cleanable-images": "Imágenes eliminables",
|
"info-description-cleanable-images": "Imágenes Eliminables",
|
||||||
"storage": {
|
"storage": {
|
||||||
"title-temporary-directory": "Directorio temporal (.temp)",
|
"title-temporary-directory": "Directorio temporal (.temp)",
|
||||||
"title-backups-directory": "Directorio de Copias de Seguridad (respaldos)",
|
"title-backups-directory": "Directorio de Copias de Seguridad (respaldos)",
|
||||||
|
|||||||
@@ -210,7 +210,8 @@
|
|||||||
"unsaved-changes": "Et ole tallentanut tekemiäsi muutoksia. Tallennetaanko ne? Paina \"ok\" tallentaaksesi ja \"peruuta\", jos et halua tallentaa.",
|
"unsaved-changes": "Et ole tallentanut tekemiäsi muutoksia. Tallennetaanko ne? Paina \"ok\" tallentaaksesi ja \"peruuta\", jos et halua tallentaa.",
|
||||||
"clipboard-copy-failure": "Kopioiminen leikepöydälle epäonnistui.",
|
"clipboard-copy-failure": "Kopioiminen leikepöydälle epäonnistui.",
|
||||||
"confirm-delete-generic-items": "Haluatko varmasti poistaa seuraavat kohteet?",
|
"confirm-delete-generic-items": "Haluatko varmasti poistaa seuraavat kohteet?",
|
||||||
"organizers": "Järjestäjät"
|
"organizers": "Järjestäjät",
|
||||||
|
"caution": "Huomio"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "Haluatko varmasti poistaa ryhmän <b>{groupName}<b/>?",
|
"are-you-sure-you-want-to-delete-the-group": "Haluatko varmasti poistaa ryhmän <b>{groupName}<b/>?",
|
||||||
@@ -291,6 +292,8 @@
|
|||||||
"mealplan-updated": "Ateriasuunnitelma päivitetty",
|
"mealplan-updated": "Ateriasuunnitelma päivitetty",
|
||||||
"no-meal-plan-defined-yet": "Ateriasuunnitelmaa ei ole vielä määritelty",
|
"no-meal-plan-defined-yet": "Ateriasuunnitelmaa ei ole vielä määritelty",
|
||||||
"no-meal-planned-for-today": "Ei ateriasuunnitelmaa tälle päivälle",
|
"no-meal-planned-for-today": "Ei ateriasuunnitelmaa tälle päivälle",
|
||||||
|
"numberOfDays-hint": "Number of days on page load",
|
||||||
|
"numberOfDays-label": "Default Days",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Vain näiden luokkien reseptejä käytetään ateriasuunnitelmissa",
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Vain näiden luokkien reseptejä käytetään ateriasuunnitelmissa",
|
||||||
"planner": "Suunnittelija",
|
"planner": "Suunnittelija",
|
||||||
"quick-week": "Pikaviikkosuunnitelma",
|
"quick-week": "Pikaviikkosuunnitelma",
|
||||||
@@ -377,6 +380,10 @@
|
|||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "Reseptilaatikkoni",
|
"title": "Reseptilaatikkoni",
|
||||||
"description-long": "Voit tuoda Mealieen reseptejä reseptilaatikosta. Tuo reseptisi csv-muodossa ja lataa sitten csv-tiedosto alempaa."
|
"description-long": "Voit tuoda Mealieen reseptejä reseptilaatikosta. Tuo reseptisi csv-muodossa ja lataa sitten csv-tiedosto alempaa."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Mealie can import recipes from Recipe Keeper. Export your recipes in zip format, then upload the .zip file below."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "Ainesosat",
|
"ingredients": "Ainesosat",
|
||||||
"insert-ingredient": "Lisää Ainesosa",
|
"insert-ingredient": "Lisää Ainesosa",
|
||||||
"insert-section": "Lisää osio",
|
"insert-section": "Lisää osio",
|
||||||
|
"insert-above": "Insert Above",
|
||||||
|
"insert-below": "Insert Below",
|
||||||
"instructions": "Ohjeet",
|
"instructions": "Ohjeet",
|
||||||
"key-name-required": "Avaimen nimi vaaditaan",
|
"key-name-required": "Avaimen nimi vaaditaan",
|
||||||
"landscape-view-coming-soon": "Landscape View (Coming Soon)",
|
"landscape-view-coming-soon": "Landscape View (Coming Soon)",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "Raportin poistaminen epäonnistui",
|
"report-deletion-failed": "Raportin poistaminen epäonnistui",
|
||||||
"recipe-debugger": "Reseptin vianhaku",
|
"recipe-debugger": "Reseptin vianhaku",
|
||||||
"recipe-debugger-description": "Tartu sen reseptin URL-osoitteeseen, jonka virheenkorjausta haluat tehdä, ja liitä se tähän. Reseptikaappain hakee URL-osoitteen ja tulokset näytetään. Jos et näe palautettua dataa, Mealie tai sen kappauskirjasto ei tue sivua, jota yrität kaapata.",
|
"recipe-debugger-description": "Tartu sen reseptin URL-osoitteeseen, jonka virheenkorjausta haluat tehdä, ja liitä se tähän. Reseptikaappain hakee URL-osoitteen ja tulokset näytetään. Jos et näe palautettua dataa, Mealie tai sen kappauskirjasto ei tue sivua, jota yrität kaapata.",
|
||||||
|
"use-openai": "Use OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Use OpenAI to parse the results instead of relying on the scraper library. When creating a recipe via URL, this is done automatically if the scraper library fails, but you may test it manually here.",
|
||||||
"debug": "Vianhaku",
|
"debug": "Vianhaku",
|
||||||
"tree-view": "Puunäkymä",
|
"tree-view": "Puunäkymä",
|
||||||
"recipe-yield": "Reseptin tekijä",
|
"recipe-yield": "Reseptin tekijä",
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "Varmuuskopio luotu sijaintiin {path}",
|
"backup-created-at-response-export_path": "Varmuuskopio luotu sijaintiin {path}",
|
||||||
"backup-deleted": "Varmuuskopio poistettu",
|
"backup-deleted": "Varmuuskopio poistettu",
|
||||||
"restore-success": "Palautus onnistui",
|
"restore-success": "Palautus onnistui",
|
||||||
|
"restore-fail": "Restore failed. Check your server logs for more details",
|
||||||
"backup-tag": "Varmuuskopion tunniste",
|
"backup-tag": "Varmuuskopion tunniste",
|
||||||
"create-heading": "Create a Backup",
|
"create-heading": "Create a Backup",
|
||||||
"delete-backup": "Poista varmuuskopio",
|
"delete-backup": "Poista varmuuskopio",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "Ei Linkitettyjä Reseptejä: Yksi Linkitetty Resepti:{count} Linkitettyjä Reseptejä",
|
"linked-recipes-count": "Ei Linkitettyjä Reseptejä: Yksi Linkitetty Resepti:{count} Linkitettyjä Reseptejä",
|
||||||
"items-checked-count": "Ei kohteita valittu: Yksi kohde tarkistettu:{count} kohdetta tarkistettu",
|
"items-checked-count": "Ei kohteita valittu: Yksi kohde tarkistettu:{count} kohdetta tarkistettu",
|
||||||
"no-label": "Ei tunnistetta",
|
"no-label": "Ei tunnistetta",
|
||||||
"completed-on": "Valmistui {date}"
|
"completed-on": "Valmistui {date}",
|
||||||
|
"you-are-offline": "You are offline",
|
||||||
|
"you-are-offline-description": "Not all features are available while offline. You can still add, modify, and remove items, but you will not be able to sync your changes to the server until you are back online.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "Are you sure you want to check all items?",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "Are you sure you want to uncheck all items?",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "Are you sure you want to delete all checked items?"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "Reseptit",
|
"all-recipes": "Reseptit",
|
||||||
@@ -941,7 +958,7 @@
|
|||||||
"authentication-method-hint": "Tämä määrittelee, miten käyttäjä todentaa Mealien. Jos et ole varma, valitse 'Mealie'",
|
"authentication-method-hint": "Tämä määrittelee, miten käyttäjä todentaa Mealien. Jos et ole varma, valitse 'Mealie'",
|
||||||
"permissions": "Käyttöoikeudet",
|
"permissions": "Käyttöoikeudet",
|
||||||
"administrator": "Ylläpitäjä",
|
"administrator": "Ylläpitäjä",
|
||||||
"user-can-invite-other-to-group": "Käyttäjä voi kutsua muita ryhmään",
|
"user-can-invite-other-to-group": "User can invite others to group",
|
||||||
"user-can-manage-group": "Käyttäjä voi hallita ryhmää",
|
"user-can-manage-group": "Käyttäjä voi hallita ryhmää",
|
||||||
"user-can-organize-group-data": "Käyttäjä voi järjestellä ryhmän tietoja",
|
"user-can-organize-group-data": "Käyttäjä voi järjestellä ryhmän tietoja",
|
||||||
"enable-advanced-features": "Salli edistyneemmät ominaisuudet",
|
"enable-advanced-features": "Salli edistyneemmät ominaisuudet",
|
||||||
@@ -972,7 +989,9 @@
|
|||||||
"edit-food": "Muokkaa elintarviketta",
|
"edit-food": "Muokkaa elintarviketta",
|
||||||
"food-data": "Elintarvikkeiden tiedot",
|
"food-data": "Elintarvikkeiden tiedot",
|
||||||
"example-food-singular": "esim. sipuli",
|
"example-food-singular": "esim. sipuli",
|
||||||
"example-food-plural": "esim. sipulit"
|
"example-food-plural": "esim. sipulit",
|
||||||
|
"label-overwrite-warning": "Tämä määrittää valitun tunnisteen kaikille valituille elintarvikkeille ja mahdollisesti korvaa olemassa olevat tunnisteet.",
|
||||||
|
"on-hand-checkbox-label": "Setting this flag will make this food unchecked by default when adding a recipe to a shopping list."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Lisää tietokantaan yksiköt paikallisen kielen perusteella.",
|
"seed-dialog-text": "Lisää tietokantaan yksiköt paikallisen kielen perusteella.",
|
||||||
@@ -1000,7 +1019,8 @@
|
|||||||
"seed-dialog-text": "Lisää tietokantaan yleiset tunnisteet paikallisen kielen perusteella.",
|
"seed-dialog-text": "Lisää tietokantaan yleiset tunnisteet paikallisen kielen perusteella.",
|
||||||
"edit-label": "Muokkaa tunnistetta",
|
"edit-label": "Muokkaa tunnistetta",
|
||||||
"new-label": "Uusi tunniste",
|
"new-label": "Uusi tunniste",
|
||||||
"labels": "Tunnisteet"
|
"labels": "Tunnisteet",
|
||||||
|
"assign-label": "Määritä tunniste"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Tyhjennä Vientitiedostot",
|
"purge-exports": "Tyhjennä Vientitiedostot",
|
||||||
|
|||||||
@@ -210,7 +210,8 @@
|
|||||||
"unsaved-changes": "Vous avez des modifications non enregistrées. Voulez-vous les enregistrer ? Ok pour enregistrer, annuler pour ignorer les modifications.",
|
"unsaved-changes": "Vous avez des modifications non enregistrées. Voulez-vous les enregistrer ? Ok pour enregistrer, annuler pour ignorer les modifications.",
|
||||||
"clipboard-copy-failure": "Échec de la copie vers le presse-papiers.",
|
"clipboard-copy-failure": "Échec de la copie vers le presse-papiers.",
|
||||||
"confirm-delete-generic-items": "Êtes-vous sûr de vouloir supprimer les éléments suivants ?",
|
"confirm-delete-generic-items": "Êtes-vous sûr de vouloir supprimer les éléments suivants ?",
|
||||||
"organizers": "Classification"
|
"organizers": "Classification",
|
||||||
|
"caution": "Avertissement"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "Êtes-vous certain de vouloir supprimer <b>{groupName}<b/>?",
|
"are-you-sure-you-want-to-delete-the-group": "Êtes-vous certain de vouloir supprimer <b>{groupName}<b/>?",
|
||||||
@@ -291,6 +292,8 @@
|
|||||||
"mealplan-updated": "Menu mis à jour",
|
"mealplan-updated": "Menu mis à jour",
|
||||||
"no-meal-plan-defined-yet": "Aucun menu planifié",
|
"no-meal-plan-defined-yet": "Aucun menu planifié",
|
||||||
"no-meal-planned-for-today": "Aucun repas prévu pour aujourd'hui",
|
"no-meal-planned-for-today": "Aucun repas prévu pour aujourd'hui",
|
||||||
|
"numberOfDays-hint": "Nombre de jours lors du chargement de la page",
|
||||||
|
"numberOfDays-label": "Jours par défaut",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Seules les recettes appartenant à ces catégories seront utilisées dans les menus",
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Seules les recettes appartenant à ces catégories seront utilisées dans les menus",
|
||||||
"planner": "Planificateur",
|
"planner": "Planificateur",
|
||||||
"quick-week": "Semaine rapide",
|
"quick-week": "Semaine rapide",
|
||||||
@@ -377,6 +380,10 @@
|
|||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "My Recipe Box",
|
"title": "My Recipe Box",
|
||||||
"description-long": "Mealie peut importer des recettes depuis My Recipe Box. Exportez vos recettes au format CSV, puis téléchargez le fichier CSV ci-dessous."
|
"description-long": "Mealie peut importer des recettes depuis My Recipe Box. Exportez vos recettes au format CSV, puis téléchargez le fichier CSV ci-dessous."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Mealie peut importer des recettes depuis Recipe Keeper. Exportez vos recettes au format Zip, puis téléversez le fichier .zip ci-dessous."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "Ingrédients",
|
"ingredients": "Ingrédients",
|
||||||
"insert-ingredient": "Insérer un ingrédient",
|
"insert-ingredient": "Insérer un ingrédient",
|
||||||
"insert-section": "Insérer une section",
|
"insert-section": "Insérer une section",
|
||||||
|
"insert-above": "Insérer au-dessus",
|
||||||
|
"insert-below": "Insérer en dessous",
|
||||||
"instructions": "Instructions",
|
"instructions": "Instructions",
|
||||||
"key-name-required": "Un nom de clé est requis",
|
"key-name-required": "Un nom de clé est requis",
|
||||||
"landscape-view-coming-soon": "Vue paysage (bientôt disponible)",
|
"landscape-view-coming-soon": "Vue paysage (bientôt disponible)",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "La suppression du rapport a échoué",
|
"report-deletion-failed": "La suppression du rapport a échoué",
|
||||||
"recipe-debugger": "Débogueur de recette",
|
"recipe-debugger": "Débogueur de recette",
|
||||||
"recipe-debugger-description": "Récupérez l'URL de la recette que vous voulez déboguer et collez-la ici. La recette sera analysée et les résultats seront affichés. Si vous ne voyez aucune donnée retournée, le site que vous essayez de récupérer n'est pas pris en charge par Mealie.",
|
"recipe-debugger-description": "Récupérez l'URL de la recette que vous voulez déboguer et collez-la ici. La recette sera analysée et les résultats seront affichés. Si vous ne voyez aucune donnée retournée, le site que vous essayez de récupérer n'est pas pris en charge par Mealie.",
|
||||||
|
"use-openai": "Utiliser OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Utilisez OpenAI pour analyser les résultats au lieu de la bibliothèque d’extraction. Lors de la création d'une recette via une URL, cela se fait automatiquement si la bibliothèque d’extraction échoue, mais vous pouvez le tester manuellement ici.",
|
||||||
"debug": "Déboguer",
|
"debug": "Déboguer",
|
||||||
"tree-view": "Vue en arborescence",
|
"tree-view": "Vue en arborescence",
|
||||||
"recipe-yield": "Nombre de parts",
|
"recipe-yield": "Nombre de parts",
|
||||||
@@ -594,7 +605,7 @@
|
|||||||
"select-parser": "Sélectionner l'analyseur",
|
"select-parser": "Sélectionner l'analyseur",
|
||||||
"natural-language-processor": "Traitement du Langage Naturel",
|
"natural-language-processor": "Traitement du Langage Naturel",
|
||||||
"brute-parser": "Analyseur brut",
|
"brute-parser": "Analyseur brut",
|
||||||
"openai-parser": "OpenAI Parser",
|
"openai-parser": "Parseur OpenAI",
|
||||||
"parse-all": "Tout analyser",
|
"parse-all": "Tout analyser",
|
||||||
"no-unit": "Pas d'unité",
|
"no-unit": "Pas d'unité",
|
||||||
"missing-unit": "Créer une unité manquante : {unit}",
|
"missing-unit": "Créer une unité manquante : {unit}",
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "Sauvegarde créée dans {path}",
|
"backup-created-at-response-export_path": "Sauvegarde créée dans {path}",
|
||||||
"backup-deleted": "Sauvegarde supprimée",
|
"backup-deleted": "Sauvegarde supprimée",
|
||||||
"restore-success": "Restauration réussie",
|
"restore-success": "Restauration réussie",
|
||||||
|
"restore-fail": "Échec de la restauration. Vérifiez les journaux de votre serveur pour plus de détails",
|
||||||
"backup-tag": "Tag de la sauvegarde",
|
"backup-tag": "Tag de la sauvegarde",
|
||||||
"create-heading": "Créer une sauvegarde",
|
"create-heading": "Créer une sauvegarde",
|
||||||
"delete-backup": "Supprimer la sauvegarde",
|
"delete-backup": "Supprimer la sauvegarde",
|
||||||
@@ -766,9 +778,9 @@
|
|||||||
"oidc-ready": "Prêt pour OIDC",
|
"oidc-ready": "Prêt pour OIDC",
|
||||||
"oidc-ready-error-text": "Toutes les valeurs OIDC ne sont pas configurées. Vous pouvez ignorer cet avertissement si vous n’utilisez pas l’authentification OIDC.",
|
"oidc-ready-error-text": "Toutes les valeurs OIDC ne sont pas configurées. Vous pouvez ignorer cet avertissement si vous n’utilisez pas l’authentification OIDC.",
|
||||||
"oidc-ready-success-text": "Les variables OIDC obligatoires sont toutes définies.",
|
"oidc-ready-success-text": "Les variables OIDC obligatoires sont toutes définies.",
|
||||||
"openai-ready": "OpenAI Ready",
|
"openai-ready": "Prêt pour OpenAI",
|
||||||
"openai-ready-error-text": "Not all OpenAI Values are configured. This can be ignored if you are not using OpenAI features.",
|
"openai-ready-error-text": "Toutes les valeurs OpenAI ne sont pas configurées. Vous pouvez ignorer cet avertissement si vous n'utilisez pas les fonctionnalités OpenAI.",
|
||||||
"openai-ready-success-text": "Required OpenAI variables are all set."
|
"openai-ready-success-text": "Les variables OpenAI obligatoires sont toutes définies."
|
||||||
},
|
},
|
||||||
"shopping-list": {
|
"shopping-list": {
|
||||||
"all-lists": "Toutes les listes",
|
"all-lists": "Toutes les listes",
|
||||||
@@ -782,7 +794,7 @@
|
|||||||
"food": "Aliments",
|
"food": "Aliments",
|
||||||
"note": "Note",
|
"note": "Note",
|
||||||
"label": "Étiquette",
|
"label": "Étiquette",
|
||||||
"save-label": "Save Label",
|
"save-label": "Sauvegarder le libellé",
|
||||||
"linked-item-warning": "Cet article est lié à une ou plusieurs recettes. Ajuster les unités ou les aliments donnera des résultats inattendus lors de l’ajout ou de la suppression de la recette de cette liste.",
|
"linked-item-warning": "Cet article est lié à une ou plusieurs recettes. Ajuster les unités ou les aliments donnera des résultats inattendus lors de l’ajout ou de la suppression de la recette de cette liste.",
|
||||||
"toggle-food": "Activer/Désactiver aliment",
|
"toggle-food": "Activer/Désactiver aliment",
|
||||||
"manage-labels": "Gérer les libellés",
|
"manage-labels": "Gérer les libellés",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "Aucune recette liée|Une recette liée|{count} recettes liées",
|
"linked-recipes-count": "Aucune recette liée|Une recette liée|{count} recettes liées",
|
||||||
"items-checked-count": "Aucun élément coché|Un élément coché|{count} éléments cochés",
|
"items-checked-count": "Aucun élément coché|Un élément coché|{count} éléments cochés",
|
||||||
"no-label": "Aucune étiquette",
|
"no-label": "Aucune étiquette",
|
||||||
"completed-on": "Terminé le {date}"
|
"completed-on": "Terminé le {date}",
|
||||||
|
"you-are-offline": "Vous êtes hors-ligne",
|
||||||
|
"you-are-offline-description": "Certaines fonctionnalités ne sont pas disponibles lorsque vous êtes hors-ligne. Vous pouvez toujours ajouter, modifier et supprimer des éléments, mais il ne sera pas possible de synchroniser les changements avec le serveur tant que vous ne serez pas en ligne.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "Voulez-vous vraiment sélectionner tous les éléments ?",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "Voulez-vous vraiment désélectionner tous les éléments ?",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "Voulez-vous vraiment supprimer tous les éléments sélectionnés ?"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "Les recettes",
|
"all-recipes": "Les recettes",
|
||||||
@@ -941,7 +958,7 @@
|
|||||||
"authentication-method-hint": "Ceci infique comment un utilisateur va s'authentifier sur Mealie. Si vous n'êtes pas sûr, choisissez 'Mealie'",
|
"authentication-method-hint": "Ceci infique comment un utilisateur va s'authentifier sur Mealie. Si vous n'êtes pas sûr, choisissez 'Mealie'",
|
||||||
"permissions": "Autorisations",
|
"permissions": "Autorisations",
|
||||||
"administrator": "Administrateur",
|
"administrator": "Administrateur",
|
||||||
"user-can-invite-other-to-group": "L'utilisateur peut inviter quelqu'un au groupe",
|
"user-can-invite-other-to-group": "User can invite others to group",
|
||||||
"user-can-manage-group": "L'utilisateur peut gérer le groupe",
|
"user-can-manage-group": "L'utilisateur peut gérer le groupe",
|
||||||
"user-can-organize-group-data": "L'utilisateur peut organiser des données de groupe",
|
"user-can-organize-group-data": "L'utilisateur peut organiser des données de groupe",
|
||||||
"enable-advanced-features": "Activer les fonctions avancées",
|
"enable-advanced-features": "Activer les fonctions avancées",
|
||||||
@@ -972,7 +989,9 @@
|
|||||||
"edit-food": "Modifier Aliment",
|
"edit-food": "Modifier Aliment",
|
||||||
"food-data": "Données de l'aliment",
|
"food-data": "Données de l'aliment",
|
||||||
"example-food-singular": "ex : Oignon",
|
"example-food-singular": "ex : Oignon",
|
||||||
"example-food-plural": "ex : Oignons"
|
"example-food-plural": "ex : Oignons",
|
||||||
|
"label-overwrite-warning": "Ceci affectera l’étiquette choisie à tous les aliments sélectionnés et remplacera potentiellement vos étiquettes existantes.",
|
||||||
|
"on-hand-checkbox-label": "Appliquer cet attribut décochera cet aliment par défaut lorsqu'il est ajouté à une liste de courses."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Initialisez la base de données avec des unités communes basées sur votre langue locale.",
|
"seed-dialog-text": "Initialisez la base de données avec des unités communes basées sur votre langue locale.",
|
||||||
@@ -1000,7 +1019,8 @@
|
|||||||
"seed-dialog-text": "Initialisez la base de données avec des étiquettes communes basées sur votre langue locale.",
|
"seed-dialog-text": "Initialisez la base de données avec des étiquettes communes basées sur votre langue locale.",
|
||||||
"edit-label": "Modifier l’étiquette",
|
"edit-label": "Modifier l’étiquette",
|
||||||
"new-label": "Nouvelle étiquette",
|
"new-label": "Nouvelle étiquette",
|
||||||
"labels": "Étiquettes"
|
"labels": "Étiquettes",
|
||||||
|
"assign-label": "Affecter une étiquette"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Purger les exports",
|
"purge-exports": "Purger les exports",
|
||||||
|
|||||||
@@ -210,7 +210,8 @@
|
|||||||
"unsaved-changes": "Vous avez des modifications non enregistrées. Voulez-vous enregistrer avant de partir ? OK pour enregistrer, Annuler pour ignorer les modifications.",
|
"unsaved-changes": "Vous avez des modifications non enregistrées. Voulez-vous enregistrer avant de partir ? OK pour enregistrer, Annuler pour ignorer les modifications.",
|
||||||
"clipboard-copy-failure": "Échec de la copie dans le presse-papiers.",
|
"clipboard-copy-failure": "Échec de la copie dans le presse-papiers.",
|
||||||
"confirm-delete-generic-items": "Êtes-vous sûr de vouloir supprimer les éléments suivants ?",
|
"confirm-delete-generic-items": "Êtes-vous sûr de vouloir supprimer les éléments suivants ?",
|
||||||
"organizers": "Classification"
|
"organizers": "Classification",
|
||||||
|
"caution": "Avertissement"
|
||||||
},
|
},
|
||||||
"group": {
|
"group": {
|
||||||
"are-you-sure-you-want-to-delete-the-group": "Voulez-vous vraiment supprimer <b>{groupName}<b/> ?",
|
"are-you-sure-you-want-to-delete-the-group": "Voulez-vous vraiment supprimer <b>{groupName}<b/> ?",
|
||||||
@@ -291,6 +292,8 @@
|
|||||||
"mealplan-updated": "Menu mis à jour",
|
"mealplan-updated": "Menu mis à jour",
|
||||||
"no-meal-plan-defined-yet": "Aucun menu planifié",
|
"no-meal-plan-defined-yet": "Aucun menu planifié",
|
||||||
"no-meal-planned-for-today": "Aucun repas prévu pour aujourd’hui",
|
"no-meal-planned-for-today": "Aucun repas prévu pour aujourd’hui",
|
||||||
|
"numberOfDays-hint": "Nombre de jours lors du chargement de la page",
|
||||||
|
"numberOfDays-label": "Jours par défaut",
|
||||||
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Seules les recettes appartenant à ces catégories seront utilisées dans les menus",
|
"only-recipes-with-these-categories-will-be-used-in-meal-plans": "Seules les recettes appartenant à ces catégories seront utilisées dans les menus",
|
||||||
"planner": "Menu",
|
"planner": "Menu",
|
||||||
"quick-week": "Semaine rapide",
|
"quick-week": "Semaine rapide",
|
||||||
@@ -377,6 +380,10 @@
|
|||||||
"myrecipebox": {
|
"myrecipebox": {
|
||||||
"title": "My Recipe Box",
|
"title": "My Recipe Box",
|
||||||
"description-long": "Mealie peut importer des recettes depuis My Recipe Box. Exportez vos recettes au format CSV, puis téléchargez le fichier CSV ci-dessous."
|
"description-long": "Mealie peut importer des recettes depuis My Recipe Box. Exportez vos recettes au format CSV, puis téléchargez le fichier CSV ci-dessous."
|
||||||
|
},
|
||||||
|
"recipekeeper": {
|
||||||
|
"title": "Recipe Keeper",
|
||||||
|
"description-long": "Mealie peut importer des recettes depuis Recipe Keeper. Exportez vos recettes au format Zip, puis téléversez le fichier .zip ci-dessous."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"new-recipe": {
|
"new-recipe": {
|
||||||
@@ -442,6 +449,8 @@
|
|||||||
"ingredients": "Ingrédients",
|
"ingredients": "Ingrédients",
|
||||||
"insert-ingredient": "Insérer un ingrédient",
|
"insert-ingredient": "Insérer un ingrédient",
|
||||||
"insert-section": "Insérer une section",
|
"insert-section": "Insérer une section",
|
||||||
|
"insert-above": "Insérer au-dessus",
|
||||||
|
"insert-below": "Insérer en dessous",
|
||||||
"instructions": "Instructions",
|
"instructions": "Instructions",
|
||||||
"key-name-required": "Un nom de clé est requis",
|
"key-name-required": "Un nom de clé est requis",
|
||||||
"landscape-view-coming-soon": "Vue paysage",
|
"landscape-view-coming-soon": "Vue paysage",
|
||||||
@@ -577,6 +586,8 @@
|
|||||||
"report-deletion-failed": "La suppression du rapport a échoué",
|
"report-deletion-failed": "La suppression du rapport a échoué",
|
||||||
"recipe-debugger": "Débogueur de recette",
|
"recipe-debugger": "Débogueur de recette",
|
||||||
"recipe-debugger-description": "Récupérez l'URL de la recette que vous voulez déboguer et collez-la ici. La recette sera analysée et les résultats seront affichés. Si vous ne voyez aucune donnée retournée, le site que vous essayez de récupérer n'est pas pris en charge par Mealie.",
|
"recipe-debugger-description": "Récupérez l'URL de la recette que vous voulez déboguer et collez-la ici. La recette sera analysée et les résultats seront affichés. Si vous ne voyez aucune donnée retournée, le site que vous essayez de récupérer n'est pas pris en charge par Mealie.",
|
||||||
|
"use-openai": "Utiliser OpenAI",
|
||||||
|
"recipe-debugger-use-openai-description": "Utilisez OpenAI pour analyser les résultats au lieu de la bibliothèque d’extraction. Lors de la création d'une recette via une URL, cela se fait automatiquement si la bibliothèque d’extraction échoue, mais vous pouvez le tester manuellement ici.",
|
||||||
"debug": "Déboguer",
|
"debug": "Déboguer",
|
||||||
"tree-view": "Vue en arborescence",
|
"tree-view": "Vue en arborescence",
|
||||||
"recipe-yield": "Nombre de parts",
|
"recipe-yield": "Nombre de parts",
|
||||||
@@ -594,7 +605,7 @@
|
|||||||
"select-parser": "Sélectionner l'analyseur",
|
"select-parser": "Sélectionner l'analyseur",
|
||||||
"natural-language-processor": "Traitement du Langage Naturel",
|
"natural-language-processor": "Traitement du Langage Naturel",
|
||||||
"brute-parser": "Analyseur brut",
|
"brute-parser": "Analyseur brut",
|
||||||
"openai-parser": "OpenAI Parser",
|
"openai-parser": "Parseur OpenAI",
|
||||||
"parse-all": "Tout analyser",
|
"parse-all": "Tout analyser",
|
||||||
"no-unit": "Pas d'unité",
|
"no-unit": "Pas d'unité",
|
||||||
"missing-unit": "Créer une unité manquante : {unit}",
|
"missing-unit": "Créer une unité manquante : {unit}",
|
||||||
@@ -629,6 +640,7 @@
|
|||||||
"backup-created-at-response-export_path": "Sauvegarde créée dans {path}",
|
"backup-created-at-response-export_path": "Sauvegarde créée dans {path}",
|
||||||
"backup-deleted": "Sauvegarde supprimée",
|
"backup-deleted": "Sauvegarde supprimée",
|
||||||
"restore-success": "Restauration réussie",
|
"restore-success": "Restauration réussie",
|
||||||
|
"restore-fail": "Échec de la restauration. Vérifiez les journaux de votre serveur pour plus de détails",
|
||||||
"backup-tag": "Tag de la sauvegarde",
|
"backup-tag": "Tag de la sauvegarde",
|
||||||
"create-heading": "Créer une sauvegarde",
|
"create-heading": "Créer une sauvegarde",
|
||||||
"delete-backup": "Supprimer la sauvegarde",
|
"delete-backup": "Supprimer la sauvegarde",
|
||||||
@@ -766,9 +778,9 @@
|
|||||||
"oidc-ready": "Prêt pour OIDC",
|
"oidc-ready": "Prêt pour OIDC",
|
||||||
"oidc-ready-error-text": "Toutes les valeurs OIDC ne sont pas configurées. Vous pouvez ignorer cet avertissement si vous n’utilisez pas l’authentification OIDC.",
|
"oidc-ready-error-text": "Toutes les valeurs OIDC ne sont pas configurées. Vous pouvez ignorer cet avertissement si vous n’utilisez pas l’authentification OIDC.",
|
||||||
"oidc-ready-success-text": "Les variables OIDC obligatoires sont toutes définies.",
|
"oidc-ready-success-text": "Les variables OIDC obligatoires sont toutes définies.",
|
||||||
"openai-ready": "OpenAI Ready",
|
"openai-ready": "Prêt pour OpenAI",
|
||||||
"openai-ready-error-text": "Not all OpenAI Values are configured. This can be ignored if you are not using OpenAI features.",
|
"openai-ready-error-text": "Toutes les valeurs OpenAI ne sont pas configurées. Vous pouvez ignorer cet avertissement si vous n'utilisez pas les fonctionnalités OpenAI.",
|
||||||
"openai-ready-success-text": "Required OpenAI variables are all set."
|
"openai-ready-success-text": "Les variables OpenAI obligatoires sont toutes définies."
|
||||||
},
|
},
|
||||||
"shopping-list": {
|
"shopping-list": {
|
||||||
"all-lists": "Toutes les listes",
|
"all-lists": "Toutes les listes",
|
||||||
@@ -782,7 +794,7 @@
|
|||||||
"food": "Aliment",
|
"food": "Aliment",
|
||||||
"note": "Note",
|
"note": "Note",
|
||||||
"label": "Étiquette",
|
"label": "Étiquette",
|
||||||
"save-label": "Save Label",
|
"save-label": "Sauvegarder le libellé",
|
||||||
"linked-item-warning": "Cet article est lié à une ou plusieurs recettes. Ajuster les unités ou les aliments donnera des résultats inattendus lors de l’ajout ou de la suppression de la recette de cette liste.",
|
"linked-item-warning": "Cet article est lié à une ou plusieurs recettes. Ajuster les unités ou les aliments donnera des résultats inattendus lors de l’ajout ou de la suppression de la recette de cette liste.",
|
||||||
"toggle-food": "Activer/Désactiver aliment",
|
"toggle-food": "Activer/Désactiver aliment",
|
||||||
"manage-labels": "Gérer les libellés",
|
"manage-labels": "Gérer les libellés",
|
||||||
@@ -797,7 +809,12 @@
|
|||||||
"linked-recipes-count": "Aucune recette liée|Une recette liée|{count} recettes liées",
|
"linked-recipes-count": "Aucune recette liée|Une recette liée|{count} recettes liées",
|
||||||
"items-checked-count": "Aucun élément coché|Un élément coché|{count} éléments cochés",
|
"items-checked-count": "Aucun élément coché|Un élément coché|{count} éléments cochés",
|
||||||
"no-label": "Aucune étiquette",
|
"no-label": "Aucune étiquette",
|
||||||
"completed-on": "Terminé le {date}"
|
"completed-on": "Terminé le {date}",
|
||||||
|
"you-are-offline": "Vous êtes hors-ligne",
|
||||||
|
"you-are-offline-description": "Certaines fonctionnalités ne sont pas disponibles lorsque vous êtes hors-ligne. Vous pouvez toujours ajouter, modifier et supprimer des éléments, mais il ne sera pas possible de synchroniser les changements avec le serveur tant que vous ne serez pas en ligne.",
|
||||||
|
"are-you-sure-you-want-to-check-all-items": "Voulez-vous vraiment sélectionner tous les éléments ?",
|
||||||
|
"are-you-sure-you-want-to-uncheck-all-items": "Voulez-vous vraiment désélectionner tous les éléments ?",
|
||||||
|
"are-you-sure-you-want-to-delete-checked-items": "Voulez-vous vraiment supprimer tous les éléments sélectionnés ?"
|
||||||
},
|
},
|
||||||
"sidebar": {
|
"sidebar": {
|
||||||
"all-recipes": "Recettes",
|
"all-recipes": "Recettes",
|
||||||
@@ -941,7 +958,7 @@
|
|||||||
"authentication-method-hint": "Ceci infique comment un utilisateur va s'authentifier sur Mealie. Si vous n'êtes pas sûr, choisissez 'Mealie'",
|
"authentication-method-hint": "Ceci infique comment un utilisateur va s'authentifier sur Mealie. Si vous n'êtes pas sûr, choisissez 'Mealie'",
|
||||||
"permissions": "Autorisations",
|
"permissions": "Autorisations",
|
||||||
"administrator": "Administrateur",
|
"administrator": "Administrateur",
|
||||||
"user-can-invite-other-to-group": "L'utilisateur peut inviter quelqu'un au groupe",
|
"user-can-invite-other-to-group": "L’utilisateur peut inviter d’autres personnes dans le groupe",
|
||||||
"user-can-manage-group": "L'utilisateur peut gérer le groupe",
|
"user-can-manage-group": "L'utilisateur peut gérer le groupe",
|
||||||
"user-can-organize-group-data": "L'utilisateur peut organiser des données de groupe",
|
"user-can-organize-group-data": "L'utilisateur peut organiser des données de groupe",
|
||||||
"enable-advanced-features": "Activer les fonctions avancées",
|
"enable-advanced-features": "Activer les fonctions avancées",
|
||||||
@@ -972,7 +989,9 @@
|
|||||||
"edit-food": "Modifier Aliment",
|
"edit-food": "Modifier Aliment",
|
||||||
"food-data": "Données de l'aliment",
|
"food-data": "Données de l'aliment",
|
||||||
"example-food-singular": "ex : Oignon",
|
"example-food-singular": "ex : Oignon",
|
||||||
"example-food-plural": "ex : Oignons"
|
"example-food-plural": "ex : Oignons",
|
||||||
|
"label-overwrite-warning": "Ceci affectera l’étiquette choisie à tous les aliments sélectionnés et remplacera potentiellement vos étiquettes existantes.",
|
||||||
|
"on-hand-checkbox-label": "Appliquer cet attribut décochera cet aliment par défaut lorsqu'il est ajouté à une liste de courses."
|
||||||
},
|
},
|
||||||
"units": {
|
"units": {
|
||||||
"seed-dialog-text": "Initialisez la base de données avec des unités communes basées sur votre langue locale.",
|
"seed-dialog-text": "Initialisez la base de données avec des unités communes basées sur votre langue locale.",
|
||||||
@@ -1000,7 +1019,8 @@
|
|||||||
"seed-dialog-text": "Initialisez la base de données avec des étiquettes communes basées sur votre langue locale.",
|
"seed-dialog-text": "Initialisez la base de données avec des étiquettes communes basées sur votre langue locale.",
|
||||||
"edit-label": "Modifier l’étiquette",
|
"edit-label": "Modifier l’étiquette",
|
||||||
"new-label": "Nouvelle étiquette",
|
"new-label": "Nouvelle étiquette",
|
||||||
"labels": "Étiquettes"
|
"labels": "Étiquettes",
|
||||||
|
"assign-label": "Affecter une étiquette"
|
||||||
},
|
},
|
||||||
"recipes": {
|
"recipes": {
|
||||||
"purge-exports": "Purger les exports",
|
"purge-exports": "Purger les exports",
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user