2022-01-08 22:24:34 -09:00
|
|
|
<template>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-container
|
|
|
|
v-if="shoppingList"
|
|
|
|
class="md-container"
|
|
|
|
>
|
|
|
|
<BaseDialog
|
|
|
|
v-model="checkAllDialog"
|
|
|
|
:title="$t('general.confirm')"
|
|
|
|
can-confirm
|
|
|
|
@confirm="checkAll"
|
|
|
|
>
|
|
|
|
<v-card-text>
|
|
|
|
{{ $t('shopping-list.are-you-sure-you-want-to-check-all-items') }}
|
|
|
|
</v-card-text>
|
2024-06-28 10:37:21 +01:00
|
|
|
</BaseDialog>
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
<BaseDialog
|
|
|
|
v-model="uncheckAllDialog"
|
|
|
|
:title="$t('general.confirm')"
|
|
|
|
can-confirm
|
|
|
|
@confirm="uncheckAll"
|
|
|
|
>
|
|
|
|
<v-card-text>
|
|
|
|
{{ $t('shopping-list.are-you-sure-you-want-to-uncheck-all-items') }}
|
|
|
|
</v-card-text>
|
2024-06-28 10:37:21 +01:00
|
|
|
</BaseDialog>
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
<BaseDialog
|
|
|
|
v-model="deleteCheckedDialog"
|
|
|
|
:title="$t('general.confirm')"
|
|
|
|
can-confirm
|
|
|
|
@confirm="deleteChecked"
|
|
|
|
>
|
|
|
|
<v-card-text>
|
|
|
|
{{ $t('shopping-list.are-you-sure-you-want-to-delete-checked-items') }}
|
|
|
|
</v-card-text>
|
2024-06-28 10:37:21 +01:00
|
|
|
</BaseDialog>
|
|
|
|
|
2022-01-08 22:24:34 -09:00
|
|
|
<BasePageTitle divider>
|
|
|
|
<template #header>
|
2024-06-21 22:17:39 +10:00
|
|
|
<v-container>
|
|
|
|
<v-row>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-col
|
|
|
|
class="text-left"
|
|
|
|
>
|
|
|
|
<ButtonLink
|
|
|
|
:to="`/shopping-lists?disableRedirect=true`"
|
|
|
|
:text="$t('shopping-list.all-lists')"
|
|
|
|
:icon="$globals.icons.backArrow"
|
|
|
|
/>
|
2024-06-21 22:17:39 +10:00
|
|
|
</v-col>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-col
|
2025-06-20 11:59:13 +02:00
|
|
|
v-if="mdAndUp"
|
|
|
|
cols="6"
|
2025-06-20 04:42:12 -05:00
|
|
|
class="d-none d-sm-flex justify-center"
|
2025-06-20 00:09:12 +07:00
|
|
|
>
|
|
|
|
<v-img
|
|
|
|
max-height="100"
|
|
|
|
max-width="100"
|
|
|
|
:src="require('~/static/svgs/shopping-cart.svg')"
|
|
|
|
/>
|
2024-06-21 22:17:39 +10:00
|
|
|
</v-col>
|
2025-01-26 08:04:40 -06:00
|
|
|
<v-col class="d-flex justify-end">
|
|
|
|
<BaseButtonGroup
|
|
|
|
:buttons="[
|
|
|
|
{
|
|
|
|
icon: $globals.icons.contentCopy,
|
|
|
|
text: '',
|
|
|
|
event: 'edit',
|
|
|
|
children: [
|
|
|
|
{
|
|
|
|
icon: $globals.icons.contentCopy,
|
2025-06-20 00:09:12 +07:00
|
|
|
text: $t('shopping-list.copy-as-text'),
|
2025-01-26 08:04:40 -06:00
|
|
|
event: 'copy-plain',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
icon: $globals.icons.contentCopy,
|
2025-06-20 00:09:12 +07:00
|
|
|
text: $t('shopping-list.copy-as-markdown'),
|
2025-01-26 08:04:40 -06:00
|
|
|
event: 'copy-markdown',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
icon: $globals.icons.checkboxOutline,
|
2025-06-20 00:09:12 +07:00
|
|
|
text: $t('shopping-list.check-all-items'),
|
2025-01-26 08:04:40 -06:00
|
|
|
event: 'check',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
icon: $globals.icons.dotsVertical,
|
|
|
|
text: '',
|
|
|
|
event: 'three-dot',
|
|
|
|
children: [
|
|
|
|
{
|
|
|
|
icon: $globals.icons.tags,
|
2025-06-20 00:09:12 +07:00
|
|
|
text: $t('shopping-list.toggle-label-sort'),
|
2025-01-26 08:04:40 -06:00
|
|
|
event: 'sort-by-labels',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
icon: $globals.icons.tags,
|
2025-06-20 00:09:12 +07:00
|
|
|
text: $t('shopping-list.reorder-labels'),
|
2025-01-26 08:04:40 -06:00
|
|
|
event: 'reorder-labels',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
icon: $globals.icons.tags,
|
2025-06-20 00:09:12 +07:00
|
|
|
text: $t('shopping-list.manage-labels'),
|
2025-01-26 08:04:40 -06:00
|
|
|
event: 'manage-labels',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
]"
|
|
|
|
@edit="edit = true"
|
|
|
|
@three-dot="threeDot = true"
|
|
|
|
@check="openCheckAll"
|
|
|
|
@sort-by-labels="sortByLabels"
|
|
|
|
@copy-plain="copyListItems('plain')"
|
|
|
|
@copy-markdown="copyListItems('markdown')"
|
|
|
|
@reorder-labels="toggleReorderLabelsDialog()"
|
|
|
|
@manage-labels="$router.push(`/group/data/labels`)"
|
|
|
|
/>
|
|
|
|
</v-col>
|
2024-06-21 22:17:39 +10:00
|
|
|
</v-row>
|
|
|
|
</v-container>
|
2022-01-08 22:24:34 -09:00
|
|
|
</template>
|
2025-06-20 00:09:12 +07:00
|
|
|
<template #title>
|
|
|
|
{{ shoppingList.name }}
|
|
|
|
</template>
|
2022-01-08 22:24:34 -09:00
|
|
|
</BasePageTitle>
|
2024-06-29 04:58:58 -05:00
|
|
|
<BannerWarning
|
2025-06-20 00:09:12 +07:00
|
|
|
v-if="isOffline"
|
|
|
|
:title="$t('shopping-list.you-are-offline')"
|
|
|
|
:description="$t('shopping-list.you-are-offline-description')"
|
2024-06-29 04:58:58 -05:00
|
|
|
/>
|
2022-01-16 15:24:24 -09:00
|
|
|
|
2022-01-08 22:24:34 -09:00
|
|
|
<!-- Viewer -->
|
2025-06-20 00:09:12 +07:00
|
|
|
<section
|
|
|
|
v-if="!edit"
|
|
|
|
class="py-2"
|
|
|
|
>
|
2023-02-21 21:58:41 -06:00
|
|
|
<div v-if="!preferences.viewByLabel">
|
2025-06-20 00:09:12 +07:00
|
|
|
<VueDraggable
|
|
|
|
v-model="listItems.unchecked"
|
|
|
|
handle=".handle"
|
|
|
|
:delay="250"
|
|
|
|
:delay-on-touch-only="true"
|
|
|
|
@start="loadingCounter += 1"
|
|
|
|
@end="loadingCounter -= 1"
|
|
|
|
@update:model-value="updateIndexUnchecked"
|
|
|
|
>
|
|
|
|
<v-lazy
|
|
|
|
v-for="(item, index) in listItems.unchecked"
|
|
|
|
:key="item.id"
|
|
|
|
class="my-2"
|
|
|
|
>
|
2022-01-16 15:24:24 -09:00
|
|
|
<ShoppingListItem
|
|
|
|
v-model="listItems.unchecked[index]"
|
|
|
|
class="my-2 my-sm-0"
|
2025-06-20 00:09:12 +07:00
|
|
|
:show-label="true"
|
2022-10-21 20:35:45 -08:00
|
|
|
:labels="allLabels || []"
|
2022-01-16 15:24:24 -09:00
|
|
|
:units="allUnits || []"
|
|
|
|
:foods="allFoods || []"
|
2023-08-21 12:18:37 -05:00
|
|
|
:recipes="recipeMap"
|
2023-01-08 11:23:24 -06:00
|
|
|
@checked="saveListItem"
|
|
|
|
@save="saveListItem"
|
2022-01-16 15:24:24 -09:00
|
|
|
@delete="deleteListItem(item)"
|
|
|
|
/>
|
|
|
|
</v-lazy>
|
2025-06-20 00:09:12 +07:00
|
|
|
</VueDraggable>
|
2022-01-08 22:24:34 -09:00
|
|
|
</div>
|
2022-01-16 15:24:24 -09:00
|
|
|
|
|
|
|
<!-- View By Label -->
|
2022-01-08 22:24:34 -09:00
|
|
|
<div v-else>
|
2025-06-20 00:09:12 +07:00
|
|
|
<div
|
|
|
|
v-for="(value, key) in itemsByLabel"
|
|
|
|
:key="key"
|
|
|
|
class="pb-4"
|
|
|
|
>
|
2024-10-25 02:07:44 -05:00
|
|
|
<v-btn
|
|
|
|
:color="getLabelColor(value[0]) ? getLabelColor(value[0]) : '#959595'"
|
|
|
|
:style="{
|
2025-06-20 00:09:12 +07:00
|
|
|
'color': getTextColor(getLabelColor(value[0])),
|
|
|
|
'letter-spacing': 'normal',
|
|
|
|
}"
|
|
|
|
@click="toggleShowLabel(key.toString())"
|
2024-10-25 02:07:44 -05:00
|
|
|
>
|
|
|
|
<v-icon>
|
|
|
|
{{ labelOpenState[key] ? $globals.icons.chevronDown : $globals.icons.chevronRight }}
|
|
|
|
</v-icon>
|
|
|
|
{{ key }}
|
2024-10-01 10:47:51 -05:00
|
|
|
</v-btn>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-divider />
|
|
|
|
<v-expand-transition>
|
|
|
|
<div v-if="labelOpenState[key]">
|
|
|
|
<VueDraggable
|
|
|
|
:model-value="value"
|
|
|
|
handle=".handle"
|
|
|
|
:delay="250"
|
|
|
|
:delay-on-touch-only="true"
|
|
|
|
@start="loadingCounter += 1"
|
|
|
|
@end="loadingCounter -= 1"
|
|
|
|
@update:model-value="updateIndexUncheckedByLabel(key.toString(), $event)"
|
|
|
|
>
|
|
|
|
<v-lazy
|
|
|
|
v-for="(item, index) in value"
|
|
|
|
:key="item.id"
|
|
|
|
class="ml-2 my-2"
|
|
|
|
>
|
|
|
|
<ShoppingListItem
|
|
|
|
v-model="value[index]"
|
|
|
|
:show-label="false"
|
|
|
|
:labels="allLabels || []"
|
|
|
|
:units="allUnits || []"
|
|
|
|
:foods="allFoods || []"
|
|
|
|
:recipes="recipeMap"
|
|
|
|
@checked="saveListItem"
|
|
|
|
@save="saveListItem"
|
|
|
|
@delete="deleteListItem(item)"
|
|
|
|
/>
|
|
|
|
</v-lazy>
|
|
|
|
</VueDraggable>
|
|
|
|
</div>
|
|
|
|
</v-expand-transition>
|
2022-01-08 22:24:34 -09:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
2025-02-24 02:41:04 -06:00
|
|
|
<!-- Create Item -->
|
|
|
|
<div v-if="createEditorOpen">
|
|
|
|
<ShoppingListItemEditor
|
|
|
|
v-model="createListItemData"
|
|
|
|
class="my-4"
|
|
|
|
:labels="allLabels || []"
|
|
|
|
:units="allUnits || []"
|
|
|
|
:foods="allFoods || []"
|
|
|
|
:allow-delete="false"
|
|
|
|
@delete="createEditorOpen = false"
|
|
|
|
@cancel="createEditorOpen = false"
|
|
|
|
@save="createListItem"
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<div v-else class="d-flex justify-end">
|
2025-06-20 00:09:12 +07:00
|
|
|
<BaseButton
|
|
|
|
create
|
|
|
|
@click="createEditorOpen = true"
|
|
|
|
>
|
|
|
|
{{ $t('general.add') }}
|
|
|
|
</BaseButton>
|
2025-02-24 02:41:04 -06:00
|
|
|
</div>
|
|
|
|
|
2023-02-21 21:58:41 -06:00
|
|
|
<!-- Reorder Labels -->
|
2024-05-04 22:27:04 +02:00
|
|
|
<BaseDialog
|
|
|
|
v-model="reorderLabelsDialog"
|
|
|
|
:icon="$globals.icons.tagArrowUp"
|
|
|
|
:title="$t('shopping-list.reorder-labels')"
|
|
|
|
:submit-icon="$globals.icons.save"
|
2025-06-20 00:09:12 +07:00
|
|
|
:submit-text="$t('general.save')"
|
|
|
|
can-submit
|
2024-05-04 22:27:04 +02:00
|
|
|
@submit="saveLabelOrder"
|
2025-06-20 00:09:12 +07:00
|
|
|
@close="cancelLabelOrder"
|
|
|
|
>
|
|
|
|
<v-card
|
|
|
|
height="fit-content"
|
|
|
|
max-height="70vh"
|
|
|
|
style="overflow-y: auto;"
|
|
|
|
>
|
|
|
|
<VueDraggable
|
2024-10-24 11:24:42 -05:00
|
|
|
v-if="localLabels"
|
2025-06-20 00:09:12 +07:00
|
|
|
v-model="localLabels"
|
2024-10-24 11:24:42 -05:00
|
|
|
handle=".handle"
|
2025-06-20 00:09:12 +07:00
|
|
|
:delay="250"
|
2024-10-25 14:49:07 -05:00
|
|
|
:delay-on-touch-only="true"
|
2024-10-24 11:24:42 -05:00
|
|
|
class="my-2"
|
2025-06-20 00:09:12 +07:00
|
|
|
@update:model-value="updateLabelOrder"
|
2024-10-24 11:24:42 -05:00
|
|
|
>
|
2025-06-20 00:09:12 +07:00
|
|
|
<div
|
|
|
|
v-for="(labelSetting, index) in localLabels"
|
|
|
|
:key="labelSetting.id"
|
|
|
|
>
|
|
|
|
<MultiPurposeLabelSection
|
|
|
|
v-model="localLabels[index]"
|
|
|
|
use-color
|
|
|
|
/>
|
2023-02-21 21:58:41 -06:00
|
|
|
</div>
|
2025-06-20 00:09:12 +07:00
|
|
|
</VueDraggable>
|
2023-02-21 21:58:41 -06:00
|
|
|
</v-card>
|
|
|
|
</BaseDialog>
|
|
|
|
|
2022-01-16 15:24:24 -09:00
|
|
|
<!-- Checked Items -->
|
2025-06-20 00:09:12 +07:00
|
|
|
<div
|
|
|
|
v-if="listItems.checked && listItems.checked.length > 0"
|
|
|
|
class="mt-6"
|
|
|
|
>
|
2025-01-26 08:04:40 -06:00
|
|
|
<div class="d-flex">
|
|
|
|
<div class="flex-grow-1">
|
|
|
|
<button @click="toggleShowChecked()">
|
|
|
|
<span>
|
|
|
|
<v-icon>
|
|
|
|
{{ showChecked ? $globals.icons.chevronDown : $globals.icons.chevronRight }}
|
|
|
|
</v-icon>
|
|
|
|
</span>
|
2025-06-20 00:09:12 +07:00
|
|
|
{{ $t('shopping-list.items-checked-count', listItems.checked ? listItems.checked.length : 0) }}
|
2025-01-26 08:04:40 -06:00
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
<div class="justify-end mt-n2">
|
|
|
|
<BaseButtonGroup
|
2025-06-20 00:09:12 +07:00
|
|
|
:buttons="[
|
|
|
|
{
|
|
|
|
icon: $globals.icons.checkboxBlankOutline,
|
|
|
|
text: $t('shopping-list.uncheck-all-items'),
|
|
|
|
event: 'uncheck',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
icon: $globals.icons.delete,
|
|
|
|
text: $t('shopping-list.delete-checked'),
|
|
|
|
event: 'delete',
|
|
|
|
},
|
|
|
|
]"
|
|
|
|
@uncheck="openUncheckAll"
|
|
|
|
@delete="openDeleteChecked"
|
|
|
|
/>
|
|
|
|
</div>
|
2025-01-26 08:04:40 -06:00
|
|
|
</div>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-divider class="my-4" />
|
2022-01-08 22:24:34 -09:00
|
|
|
<v-expand-transition>
|
2025-06-20 00:09:12 +07:00
|
|
|
<div v-if="showChecked">
|
|
|
|
<div
|
|
|
|
v-for="(item, idx) in listItems.checked"
|
|
|
|
:key="item.id"
|
|
|
|
>
|
2022-01-16 15:24:24 -09:00
|
|
|
<ShoppingListItem
|
|
|
|
v-model="listItems.checked[idx]"
|
|
|
|
class="strike-through-note"
|
2023-01-28 18:45:02 -06:00
|
|
|
:labels="allLabels || []"
|
2022-01-16 15:24:24 -09:00
|
|
|
:units="allUnits || []"
|
|
|
|
:foods="allFoods || []"
|
2023-01-08 11:23:24 -06:00
|
|
|
@checked="saveListItem"
|
|
|
|
@save="saveListItem"
|
2022-01-16 15:24:24 -09:00
|
|
|
@delete="deleteListItem(item)"
|
|
|
|
/>
|
2022-01-08 22:24:34 -09:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</v-expand-transition>
|
|
|
|
</div>
|
|
|
|
</section>
|
|
|
|
|
2022-01-16 15:24:24 -09:00
|
|
|
<!-- Recipe References -->
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-lazy
|
|
|
|
v-if="shoppingList.recipeReferences && shoppingList.recipeReferences.length > 0"
|
|
|
|
>
|
2022-01-16 15:24:24 -09:00
|
|
|
<section>
|
|
|
|
<div>
|
|
|
|
<span>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-icon start class="mb-1">
|
2022-01-16 15:24:24 -09:00
|
|
|
{{ $globals.icons.primary }}
|
|
|
|
</v-icon>
|
|
|
|
</span>
|
2025-06-20 00:09:12 +07:00
|
|
|
{{ $t('shopping-list.linked-recipes-count', shoppingList.recipeReferences
|
|
|
|
? shoppingList.recipeReferences.length
|
|
|
|
: 0) }}
|
2022-01-08 22:24:34 -09:00
|
|
|
</div>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-divider class="my-4" />
|
|
|
|
<RecipeList
|
|
|
|
:recipes="Array.from(recipeMap.values())"
|
|
|
|
show-description
|
|
|
|
:disabled="isOffline"
|
|
|
|
>
|
|
|
|
<template
|
|
|
|
v-for="(recipe, index) in recipeMap.values()"
|
|
|
|
#[`actions-${recipe.id}`]
|
|
|
|
:key="'item-actions-decrease' + recipe.id"
|
|
|
|
>
|
|
|
|
<v-list-item-action>
|
|
|
|
<v-btn
|
|
|
|
v-if="recipe"
|
|
|
|
icon
|
|
|
|
:disabled="isOffline"
|
|
|
|
@click.prevent="removeRecipeReferenceToList(recipe.id!)"
|
|
|
|
>
|
|
|
|
<v-icon color="grey-lighten-1">
|
|
|
|
{{ $globals.icons.minus }}
|
|
|
|
</v-icon>
|
2022-01-16 15:24:24 -09:00
|
|
|
</v-btn>
|
|
|
|
</v-list-item-action>
|
2025-06-20 00:09:12 +07:00
|
|
|
<div class="pl-3">
|
2022-01-16 15:24:24 -09:00
|
|
|
{{ shoppingList.recipeReferences[index].recipeQuantity }}
|
|
|
|
</div>
|
2025-06-20 00:09:12 +07:00
|
|
|
<v-list-item-action>
|
|
|
|
<v-btn
|
|
|
|
icon
|
|
|
|
:disabled="isOffline"
|
|
|
|
@click.prevent="addRecipeReferenceToList(recipe.id!)"
|
|
|
|
>
|
|
|
|
<v-icon color="grey-lighten-1">
|
|
|
|
{{ $globals.icons.createAlt }}
|
|
|
|
</v-icon>
|
2022-01-16 15:24:24 -09:00
|
|
|
</v-btn>
|
|
|
|
</v-list-item-action>
|
|
|
|
</template>
|
|
|
|
</RecipeList>
|
|
|
|
</section>
|
|
|
|
</v-lazy>
|
2025-06-20 00:09:12 +07:00
|
|
|
<WakelockSwitch />
|
2022-01-08 22:24:34 -09:00
|
|
|
</v-container>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script lang="ts">
|
2025-06-20 00:09:12 +07:00
|
|
|
import { VueDraggable } from "vue-draggable-plus";
|
|
|
|
import { useIdle, useOnline, useToggle } from "@vueuse/core";
|
2022-01-16 15:24:24 -09:00
|
|
|
import { useCopyList } from "~/composables/use-copy";
|
2022-01-08 22:24:34 -09:00
|
|
|
import { useUserApi } from "~/composables/api";
|
2025-06-20 00:09:12 +07:00
|
|
|
import MultiPurposeLabelSection from "~/components/Domain/ShoppingList/MultiPurposeLabelSection.vue";
|
2022-01-08 22:24:34 -09:00
|
|
|
import ShoppingListItem from "~/components/Domain/ShoppingList/ShoppingListItem.vue";
|
2025-06-20 00:09:12 +07:00
|
|
|
import type { ShoppingListItemOut, ShoppingListMultiPurposeLabelOut, ShoppingListOut } from "~/lib/api/types/household";
|
2022-01-16 15:24:24 -09:00
|
|
|
import RecipeList from "~/components/Domain/Recipe/RecipeList.vue";
|
|
|
|
import ShoppingListItemEditor from "~/components/Domain/ShoppingList/ShoppingListItemEditor.vue";
|
2022-06-25 14:39:38 -05:00
|
|
|
import { useFoodStore, useLabelStore, useUnitStore } from "~/composables/store";
|
2024-06-29 04:58:58 -05:00
|
|
|
import { useShoppingListItemActions } from "~/composables/use-shopping-list-item-actions";
|
2023-02-21 21:58:41 -06:00
|
|
|
import { useShoppingListPreferences } from "~/composables/use-users/preferences";
|
2024-10-01 10:47:51 -05:00
|
|
|
import { getTextColor } from "~/composables/use-text-color";
|
2024-06-29 04:58:58 -05:00
|
|
|
import { uuid4 } from "~/composables/use-utils";
|
2022-01-16 15:24:24 -09:00
|
|
|
|
2022-01-08 22:24:34 -09:00
|
|
|
type CopyTypes = "plain" | "markdown";
|
|
|
|
|
|
|
|
interface PresentLabel {
|
|
|
|
id: string;
|
|
|
|
name: string;
|
|
|
|
}
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
export default defineNuxtComponent({
|
2022-01-08 22:24:34 -09:00
|
|
|
components: {
|
2025-06-20 00:09:12 +07:00
|
|
|
VueDraggable,
|
2023-02-21 21:58:41 -06:00
|
|
|
MultiPurposeLabelSection,
|
2022-01-08 22:24:34 -09:00
|
|
|
ShoppingListItem,
|
2022-01-16 15:24:24 -09:00
|
|
|
RecipeList,
|
|
|
|
ShoppingListItemEditor,
|
2022-01-08 22:24:34 -09:00
|
|
|
},
|
2025-06-20 00:09:12 +07:00
|
|
|
// middleware: "sidebase-auth",
|
2022-01-08 22:24:34 -09:00
|
|
|
setup() {
|
2025-06-20 11:59:13 +02:00
|
|
|
const { mdAndUp } = useDisplay();
|
2025-06-20 00:09:12 +07:00
|
|
|
const i18n = useI18n();
|
|
|
|
const $auth = useMealieAuth();
|
2023-02-21 21:58:41 -06:00
|
|
|
const preferences = useShoppingListPreferences();
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
const isOffline = computed(() => useOnline().value === false);
|
|
|
|
|
|
|
|
useSeoMeta({
|
|
|
|
title: i18n.t("shopping-list.shopping-list"),
|
|
|
|
});
|
|
|
|
|
|
|
|
const { idle } = useIdle(5 * 60 * 1000); // 5 minutes
|
2023-01-08 11:23:24 -06:00
|
|
|
const loadingCounter = ref(1);
|
|
|
|
const recipeReferenceLoading = ref(false);
|
2022-01-08 22:24:34 -09:00
|
|
|
const userApi = useUserApi();
|
|
|
|
|
|
|
|
const edit = ref(false);
|
2025-01-26 08:04:40 -06:00
|
|
|
const threeDot = ref(false);
|
2023-02-21 21:58:41 -06:00
|
|
|
const reorderLabelsDialog = ref(false);
|
2024-04-19 06:00:40 -05:00
|
|
|
const preserveItemOrder = ref(false);
|
2022-01-08 22:24:34 -09:00
|
|
|
|
|
|
|
const route = useRoute();
|
2025-06-20 00:09:12 +07:00
|
|
|
const groupSlug = computed(() => route.params.groupSlug as string || $auth.user.value?.groupSlug || "");
|
|
|
|
const id = route.params.id as string;
|
2024-06-29 04:58:58 -05:00
|
|
|
const shoppingListItemActions = useShoppingListItemActions(id);
|
2022-01-08 22:24:34 -09:00
|
|
|
|
2024-06-28 10:37:21 +01:00
|
|
|
const state = reactive({
|
|
|
|
checkAllDialog: false,
|
|
|
|
uncheckAllDialog: false,
|
|
|
|
deleteCheckedDialog: false,
|
|
|
|
});
|
|
|
|
|
2022-01-16 15:24:24 -09:00
|
|
|
// ===============================================================
|
|
|
|
// Shopping List Actions
|
|
|
|
|
2023-04-25 12:46:58 -05:00
|
|
|
const shoppingList = ref<ShoppingListOut | null>(null);
|
2022-01-08 22:24:34 -09:00
|
|
|
async function fetchShoppingList() {
|
2024-06-29 04:58:58 -05:00
|
|
|
const data = await shoppingListItemActions.getList();
|
2022-01-08 22:24:34 -09:00
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
async function refresh() {
|
2023-01-08 11:23:24 -06:00
|
|
|
loadingCounter.value += 1;
|
2024-06-29 04:58:58 -05:00
|
|
|
try {
|
|
|
|
await shoppingListItemActions.process();
|
2025-06-20 00:09:12 +07:00
|
|
|
}
|
|
|
|
catch (error) {
|
2024-06-29 04:58:58 -05:00
|
|
|
console.error(error);
|
|
|
|
}
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
let newListValue: typeof shoppingList.value = null;
|
2024-06-29 04:58:58 -05:00
|
|
|
try {
|
|
|
|
newListValue = await fetchShoppingList();
|
2025-06-20 00:09:12 +07:00
|
|
|
}
|
|
|
|
catch (error) {
|
2024-06-29 04:58:58 -05:00
|
|
|
console.error(error);
|
|
|
|
}
|
|
|
|
|
2023-01-08 11:23:24 -06:00
|
|
|
loadingCounter.value -= 1;
|
|
|
|
|
|
|
|
// only update the list with the new value if we're not loading, to prevent UI jitter
|
2024-04-19 06:00:40 -05:00
|
|
|
if (loadingCounter.value) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
// Prevent overwriting local changes with stale backend data when offline
|
|
|
|
if (isOffline.value) {
|
|
|
|
// Do not update shoppingList.value from backend when offline
|
|
|
|
updateListItemOrder();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-06-29 04:58:58 -05:00
|
|
|
// if we're not connected to the network, this will be null, so we don't want to clear the list
|
|
|
|
if (newListValue) {
|
|
|
|
shoppingList.value = newListValue;
|
|
|
|
}
|
|
|
|
|
2024-04-19 06:00:40 -05:00
|
|
|
updateListItemOrder();
|
|
|
|
}
|
|
|
|
|
|
|
|
function updateListItemOrder() {
|
|
|
|
if (!preserveItemOrder.value) {
|
|
|
|
groupAndSortListItemsByFood();
|
2025-06-20 00:09:12 +07:00
|
|
|
}
|
|
|
|
else {
|
2023-10-07 16:06:00 -05:00
|
|
|
sortListItems();
|
2023-01-08 11:23:24 -06:00
|
|
|
}
|
2024-06-29 04:58:58 -05:00
|
|
|
updateItemsByLabel();
|
2022-01-08 22:24:34 -09:00
|
|
|
}
|
|
|
|
|
2022-11-10 18:17:49 -06:00
|
|
|
// constantly polls for changes
|
|
|
|
async function pollForChanges() {
|
2022-11-30 23:29:27 -06:00
|
|
|
// pause polling if the user isn't active or we're busy
|
2023-01-08 11:23:24 -06:00
|
|
|
if (idle.value || loadingCounter.value) {
|
2022-11-10 18:17:49 -06:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
await refresh();
|
|
|
|
|
|
|
|
if (shoppingList.value) {
|
|
|
|
attempts = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if the refresh was unsuccessful, the shopping list will be null, so we increment the attempt counter
|
2025-06-20 00:09:12 +07:00
|
|
|
attempts++;
|
2022-11-10 18:17:49 -06:00
|
|
|
}
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
catch {
|
|
|
|
attempts++;
|
2022-11-10 18:17:49 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// if we hit too many errors, stop polling
|
|
|
|
if (attempts >= maxAttempts) {
|
|
|
|
clearInterval(pollTimer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// start polling
|
2023-01-08 11:23:24 -06:00
|
|
|
loadingCounter.value -= 1;
|
2025-06-20 00:09:12 +07:00
|
|
|
pollForChanges(); // populate initial list
|
2022-11-10 18:17:49 -06:00
|
|
|
|
2024-06-29 04:58:58 -05:00
|
|
|
// max poll time = pollFrequency * maxAttempts = 24 hours
|
|
|
|
// we use a long max poll time since polling stops when the user is idle anyway
|
|
|
|
const pollFrequency = 5000;
|
|
|
|
const maxAttempts = 17280;
|
2022-11-10 18:17:49 -06:00
|
|
|
let attempts = 0;
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
const pollTimer: ReturnType<typeof setInterval> = setInterval(() => {
|
|
|
|
pollForChanges();
|
|
|
|
}, pollFrequency);
|
2022-11-10 18:17:49 -06:00
|
|
|
onUnmounted(() => {
|
|
|
|
clearInterval(pollTimer);
|
|
|
|
});
|
|
|
|
|
2022-01-08 22:24:34 -09:00
|
|
|
// =====================================
|
2022-01-16 15:24:24 -09:00
|
|
|
// List Item CRUD
|
2022-01-08 22:24:34 -09:00
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
// Hydrate listItems from shoppingList.value?.listItems
|
|
|
|
const listItems = reactive({
|
|
|
|
unchecked: [] as ShoppingListItemOut[],
|
|
|
|
checked: [] as ShoppingListItemOut[],
|
2022-01-08 22:24:34 -09:00
|
|
|
});
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
function sortCheckedItems(a: ShoppingListItemOut, b: ShoppingListItemOut) {
|
|
|
|
if (a.updatedAt! === b.updatedAt!) {
|
|
|
|
return ((a.position || 0) > (b.position || 0)) ? -1 : 1;
|
|
|
|
}
|
|
|
|
return a.updatedAt! < b.updatedAt! ? 1 : -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
watch(
|
|
|
|
() => shoppingList.value?.listItems,
|
|
|
|
(items) => {
|
|
|
|
listItems.unchecked = (items?.filter(item => !item.checked) ?? []);
|
|
|
|
listItems.checked = (items?.filter(item => item.checked)
|
|
|
|
.sort(sortCheckedItems) ?? []);
|
|
|
|
},
|
|
|
|
{ immediate: true },
|
|
|
|
);
|
|
|
|
|
2024-10-25 02:07:44 -05:00
|
|
|
// =====================================
|
2024-11-05 17:12:52 -06:00
|
|
|
// Collapsable Labels
|
2024-10-25 02:07:44 -05:00
|
|
|
const labelOpenState = ref<{ [key: string]: boolean }>({});
|
|
|
|
|
|
|
|
const initializeLabelOpenStates = () => {
|
|
|
|
if (!shoppingList.value?.listItems) return;
|
|
|
|
|
|
|
|
const existingLabels = new Set(Object.keys(labelOpenState.value));
|
|
|
|
let hasChanges = false;
|
|
|
|
|
|
|
|
for (const item of shoppingList.value.listItems) {
|
2025-06-20 00:09:12 +07:00
|
|
|
const labelName = item.label?.name || i18n.t("shopping-list.no-label");
|
2024-11-05 17:12:52 -06:00
|
|
|
if (!existingLabels.has(labelName) && !(labelName in labelOpenState.value)) {
|
2024-10-25 02:07:44 -05:00
|
|
|
labelOpenState.value[labelName] = true;
|
|
|
|
hasChanges = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hasChanges) {
|
|
|
|
labelOpenState.value = { ...labelOpenState.value };
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-11-05 17:12:52 -06:00
|
|
|
const labelNames = computed(() => {
|
|
|
|
return new Set(
|
|
|
|
shoppingList.value?.listItems
|
2025-06-20 00:09:12 +07:00
|
|
|
?.map(item => item.label?.name || i18n.t("shopping-list.no-label"))
|
|
|
|
.filter(Boolean) ?? [],
|
2024-11-05 17:12:52 -06:00
|
|
|
);
|
|
|
|
});
|
2024-10-25 02:07:44 -05:00
|
|
|
|
|
|
|
watch(labelNames, initializeLabelOpenStates, { immediate: true });
|
|
|
|
|
|
|
|
function toggleShowLabel(key: string) {
|
|
|
|
labelOpenState.value[key] = !labelOpenState.value[key];
|
|
|
|
}
|
|
|
|
|
2022-01-08 22:24:34 -09:00
|
|
|
const [showChecked, toggleShowChecked] = useToggle(false);
|
|
|
|
|
|
|
|
// =====================================
|
|
|
|
// Copy List Items
|
|
|
|
|
2022-01-16 15:24:24 -09:00
|
|
|
const copy = useCopyList();
|
2022-01-08 22:24:34 -09:00
|
|
|
|
2022-01-16 15:24:24 -09:00
|
|
|
function copyListItems(copyType: CopyTypes) {
|
2024-02-28 22:06:04 +00:00
|
|
|
const text: string[] = [];
|
|
|
|
|
|
|
|
if (preferences.value.viewByLabel) {
|
|
|
|
// if we're sorting by label, we want the copied text in subsections
|
|
|
|
Object.entries(itemsByLabel.value).forEach(([label, items], idx) => {
|
|
|
|
// for every group except the first, add a blank line
|
|
|
|
if (idx) {
|
2025-06-20 00:09:12 +07:00
|
|
|
text.push("");
|
2024-02-28 22:06:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// add an appropriate heading for the label depending on the copy format
|
2025-06-20 00:09:12 +07:00
|
|
|
text.push(formatCopiedLabelHeading(copyType, label));
|
2024-02-28 22:06:04 +00:00
|
|
|
|
|
|
|
// now add the appropriately formatted list items with the given label
|
2025-06-20 00:09:12 +07:00
|
|
|
items.forEach(item => text.push(formatCopiedListItem(copyType, item)));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
else {
|
2024-02-28 22:06:04 +00:00
|
|
|
// labels are toggled off, so just copy in the order they come in
|
2025-06-20 00:09:12 +07:00
|
|
|
const items = shoppingList.value?.listItems?.filter(item => !item.checked);
|
2024-02-28 22:06:04 +00:00
|
|
|
|
|
|
|
items?.forEach((item) => {
|
2025-06-20 00:09:12 +07:00
|
|
|
text.push(formatCopiedListItem(copyType, item));
|
2024-02-28 22:06:04 +00:00
|
|
|
});
|
2022-01-08 22:24:34 -09:00
|
|
|
}
|
|
|
|
|
2024-02-28 22:06:04 +00:00
|
|
|
copy.copyPlain(text);
|
|
|
|
}
|
|
|
|
|
|
|
|
function formatCopiedListItem(copyType: CopyTypes, item: ShoppingListItemOut): string {
|
2025-06-20 00:09:12 +07:00
|
|
|
const display = item.display || "";
|
2024-02-28 22:06:04 +00:00
|
|
|
switch (copyType) {
|
|
|
|
case "markdown":
|
2025-06-20 00:09:12 +07:00
|
|
|
return `- [ ] ${display}`;
|
2024-02-28 22:06:04 +00:00
|
|
|
default:
|
2025-06-20 00:09:12 +07:00
|
|
|
return display;
|
2024-02-28 22:06:04 +00:00
|
|
|
}
|
|
|
|
}
|
2022-01-08 22:24:34 -09:00
|
|
|
|
2024-02-28 22:06:04 +00:00
|
|
|
function formatCopiedLabelHeading(copyType: CopyTypes, label: string): string {
|
2022-01-08 22:24:34 -09:00
|
|
|
switch (copyType) {
|
|
|
|
case "markdown":
|
2025-06-20 00:09:12 +07:00
|
|
|
return `# ${label}`;
|
2022-01-08 22:24:34 -09:00
|
|
|
default:
|
2025-06-20 00:09:12 +07:00
|
|
|
return `[${label}]`;
|
2022-01-08 22:24:34 -09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// =====================================
|
|
|
|
// Check / Uncheck All
|
2024-06-28 10:37:21 +01:00
|
|
|
function openCheckAll() {
|
2025-06-20 00:09:12 +07:00
|
|
|
if (shoppingList.value?.listItems?.some(item => !item.checked)) {
|
2024-06-28 10:37:21 +01:00
|
|
|
state.checkAllDialog = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function checkAll() {
|
|
|
|
state.checkAllDialog = false;
|
|
|
|
let hasChanged = false;
|
|
|
|
shoppingList.value?.listItems?.forEach((item) => {
|
|
|
|
if (!item.checked) {
|
|
|
|
hasChanged = true;
|
|
|
|
item.checked = true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if (hasChanged) {
|
2025-07-10 11:41:34 -05:00
|
|
|
updateUncheckedListItems();
|
2024-06-28 10:37:21 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function openUncheckAll() {
|
2025-06-20 00:09:12 +07:00
|
|
|
if (shoppingList.value?.listItems?.some(item => item.checked)) {
|
2024-06-28 10:37:21 +01:00
|
|
|
state.uncheckAllDialog = true;
|
|
|
|
}
|
|
|
|
}
|
2022-01-08 22:24:34 -09:00
|
|
|
|
|
|
|
function uncheckAll() {
|
2024-06-28 10:37:21 +01:00
|
|
|
state.uncheckAllDialog = false;
|
2022-01-08 22:24:34 -09:00
|
|
|
let hasChanged = false;
|
2022-01-16 15:24:24 -09:00
|
|
|
shoppingList.value?.listItems?.forEach((item) => {
|
2022-01-08 22:24:34 -09:00
|
|
|
if (item.checked) {
|
|
|
|
hasChanged = true;
|
|
|
|
item.checked = false;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if (hasChanged) {
|
2025-07-10 11:41:34 -05:00
|
|
|
listItems.unchecked = [...listItems.unchecked, ...listItems.checked];
|
|
|
|
listItems.checked = [];
|
|
|
|
updateUncheckedListItems();
|
2022-01-08 22:24:34 -09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-28 10:37:21 +01:00
|
|
|
function openDeleteChecked() {
|
2025-06-20 00:09:12 +07:00
|
|
|
if (shoppingList.value?.listItems?.some(item => item.checked)) {
|
2024-06-28 10:37:21 +01:00
|
|
|
state.deleteCheckedDialog = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-08 22:24:34 -09:00
|
|
|
function deleteChecked() {
|
2025-06-20 00:09:12 +07:00
|
|
|
const checked = shoppingList.value?.listItems?.filter(item => item.checked);
|
2022-01-08 22:24:34 -09:00
|
|
|
|
2022-01-16 15:24:24 -09:00
|
|
|
if (!checked || checked?.length === 0) {
|
2022-01-08 22:24:34 -09:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-01-08 11:23:24 -06:00
|
|
|
loadingCounter.value += 1;
|
2022-01-16 15:24:24 -09:00
|
|
|
deleteListItems(checked);
|
2022-01-08 22:24:34 -09:00
|
|
|
|
2023-01-08 11:23:24 -06:00
|
|
|
loadingCounter.value -= 1;
|
2022-01-16 15:24:24 -09:00
|
|
|
refresh();
|
2022-01-08 22:24:34 -09:00
|
|
|
}
|
|
|
|
|
|
|
|
// =====================================
|
|
|
|
// List Item Context Menu
|
|
|
|
|
|
|
|
const contextActions = {
|
|
|
|
delete: "delete",
|
|
|
|
setIngredient: "setIngredient",
|
|
|
|
};
|
|
|
|
|
|
|
|
const contextMenu = [
|
2025-06-20 00:09:12 +07:00
|
|
|
{ title: i18n.t("general.delete"), action: contextActions.delete },
|
|
|
|
{ title: i18n.t("recipe.ingredient"), action: contextActions.setIngredient },
|
2022-01-08 22:24:34 -09:00
|
|
|
];
|
|
|
|
|
2022-01-16 15:24:24 -09:00
|
|
|
function contextMenuAction(action: string, item: ShoppingListItemOut, idx: number) {
|
2022-01-08 22:24:34 -09:00
|
|
|
if (!shoppingList.value?.listItems) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (action) {
|
|
|
|
case contextActions.delete:
|
2025-06-20 00:09:12 +07:00
|
|
|
shoppingList.value.listItems = shoppingList.value?.listItems.filter(itm => itm.id !== item.id);
|
2022-01-08 22:24:34 -09:00
|
|
|
break;
|
|
|
|
case contextActions.setIngredient:
|
|
|
|
shoppingList.value.listItems[idx].isFood = !shoppingList.value.listItems[idx].isFood;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// =====================================
|
2022-01-16 15:24:24 -09:00
|
|
|
// Labels, Units, Foods
|
|
|
|
// TODO: Extract to Composable
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
const localLabels = ref<ShoppingListMultiPurposeLabelOut[]>();
|
2024-05-04 22:27:04 +02:00
|
|
|
|
2024-09-22 09:59:20 -05:00
|
|
|
const { store: allLabels } = useLabelStore();
|
|
|
|
const { store: allUnits } = useUnitStore();
|
|
|
|
const { store: allFoods } = useFoodStore();
|
2022-01-08 22:24:34 -09:00
|
|
|
|
2023-04-25 12:46:58 -05:00
|
|
|
function getLabelColor(item: ShoppingListItemOut | null) {
|
|
|
|
return item?.label?.color;
|
|
|
|
}
|
|
|
|
|
2022-01-08 22:24:34 -09:00
|
|
|
function sortByLabels() {
|
2023-02-21 21:58:41 -06:00
|
|
|
preferences.value.viewByLabel = !preferences.value.viewByLabel;
|
|
|
|
}
|
|
|
|
|
|
|
|
function toggleReorderLabelsDialog() {
|
2024-05-04 22:27:04 +02:00
|
|
|
// stop polling and populate localLabels
|
2025-06-20 00:09:12 +07:00
|
|
|
loadingCounter.value += 1;
|
|
|
|
reorderLabelsDialog.value = !reorderLabelsDialog.value;
|
|
|
|
localLabels.value = shoppingList.value?.labelSettings;
|
2023-02-21 21:58:41 -06:00
|
|
|
}
|
|
|
|
|
2024-05-04 22:27:04 +02:00
|
|
|
function updateLabelOrder(labelSettings: ShoppingListMultiPurposeLabelOut[]) {
|
2023-02-21 21:58:41 -06:00
|
|
|
if (!shoppingList.value) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
labelSettings.forEach((labelSetting, index) => {
|
|
|
|
labelSetting.position = index;
|
|
|
|
return labelSetting;
|
|
|
|
});
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
localLabels.value = labelSettings;
|
2024-05-04 22:27:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function cancelLabelOrder() {
|
2025-06-20 00:09:12 +07:00
|
|
|
loadingCounter.value -= 1;
|
2024-05-04 22:27:04 +02:00
|
|
|
if (!shoppingList.value) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// restore original state
|
2025-06-20 00:09:12 +07:00
|
|
|
localLabels.value = shoppingList.value.labelSettings;
|
2024-05-04 22:27:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async function saveLabelOrder() {
|
|
|
|
if (!shoppingList.value || !localLabels.value || (localLabels.value === shoppingList.value.labelSettings)) {
|
|
|
|
return;
|
|
|
|
}
|
2023-02-21 21:58:41 -06:00
|
|
|
|
|
|
|
loadingCounter.value += 1;
|
2024-05-04 22:27:04 +02:00
|
|
|
const { data } = await userApi.shopping.lists.updateLabelSettings(shoppingList.value.id, localLabels.value);
|
2023-02-21 21:58:41 -06:00
|
|
|
loadingCounter.value -= 1;
|
|
|
|
|
|
|
|
if (data) {
|
2024-05-04 22:27:04 +02:00
|
|
|
// update shoppingList labels using the API response
|
|
|
|
shoppingList.value.labelSettings = (data as ShoppingListOut).labelSettings;
|
|
|
|
updateItemsByLabel();
|
2023-02-21 21:58:41 -06:00
|
|
|
}
|
2022-01-08 22:24:34 -09:00
|
|
|
}
|
|
|
|
|
|
|
|
const presentLabels = computed(() => {
|
|
|
|
const labels: PresentLabel[] = [];
|
|
|
|
|
2022-01-16 15:24:24 -09:00
|
|
|
shoppingList.value?.listItems?.forEach((item) => {
|
|
|
|
if (item.labelId && item.label) {
|
2022-01-08 22:24:34 -09:00
|
|
|
labels.push({
|
|
|
|
name: item.label.name,
|
|
|
|
id: item.labelId,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return labels;
|
|
|
|
});
|
|
|
|
|
2022-06-02 09:12:05 -08:00
|
|
|
const itemsByLabel = ref<{ [key: string]: ShoppingListItemOut[] }>({});
|
|
|
|
|
2024-04-19 06:00:40 -05:00
|
|
|
interface ListItemGroup {
|
|
|
|
position: number;
|
|
|
|
createdAt: string;
|
|
|
|
items: ShoppingListItemOut[];
|
|
|
|
}
|
|
|
|
|
2024-06-29 04:58:58 -05:00
|
|
|
function sortItems(a: ShoppingListItemOut | ListItemGroup, b: ShoppingListItemOut | ListItemGroup) {
|
2025-06-20 00:09:12 +07:00
|
|
|
// Sort by position ASC, then by createdAt ASC
|
|
|
|
const posA = a.position ?? 0;
|
|
|
|
const posB = b.position ?? 0;
|
|
|
|
if (posA !== posB) {
|
|
|
|
return posA - posB;
|
|
|
|
}
|
|
|
|
const createdA = a.createdAt ?? "";
|
|
|
|
const createdB = b.createdAt ?? "";
|
|
|
|
if (createdA !== createdB) {
|
|
|
|
return createdA < createdB ? -1 : 1;
|
|
|
|
}
|
|
|
|
return 0;
|
2024-06-29 04:58:58 -05:00
|
|
|
}
|
|
|
|
|
2024-04-19 06:00:40 -05:00
|
|
|
function groupAndSortListItemsByFood() {
|
|
|
|
if (!shoppingList.value?.listItems?.length) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
const checkedItemKey = "__checkedItem";
|
2024-04-19 06:00:40 -05:00
|
|
|
const listItemGroupsMap = new Map<string, ListItemGroup>();
|
2025-06-20 00:09:12 +07:00
|
|
|
listItemGroupsMap.set(checkedItemKey, { position: Number.MAX_SAFE_INTEGER, createdAt: "", items: [] });
|
2024-04-19 06:00:40 -05:00
|
|
|
|
|
|
|
// group items by checked status, food, or note
|
|
|
|
shoppingList.value.listItems.forEach((item) => {
|
2025-06-20 00:09:12 +07:00
|
|
|
const key = item.checked
|
|
|
|
? checkedItemKey
|
|
|
|
: item.isFood && item.food?.name
|
|
|
|
? item.food.name
|
|
|
|
: item.note || "";
|
2024-04-19 06:00:40 -05:00
|
|
|
|
|
|
|
const group = listItemGroupsMap.get(key);
|
|
|
|
if (!group) {
|
2025-06-20 00:09:12 +07:00
|
|
|
listItemGroupsMap.set(key, { position: item.position || 0, createdAt: item.createdAt || "", items: [item] });
|
|
|
|
}
|
|
|
|
else {
|
2024-04-19 06:00:40 -05:00
|
|
|
group.items.push(item);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
const listItemGroups = Array.from(listItemGroupsMap.values());
|
2024-06-29 04:58:58 -05:00
|
|
|
listItemGroups.sort(sortItems);
|
2024-04-19 06:00:40 -05:00
|
|
|
|
2024-06-29 04:58:58 -05:00
|
|
|
// sort group items, then aggregate them
|
2024-04-19 06:00:40 -05:00
|
|
|
const sortedItems: ShoppingListItemOut[] = [];
|
|
|
|
let nextPosition = 0;
|
|
|
|
listItemGroups.forEach((listItemGroup) => {
|
2024-06-29 04:58:58 -05:00
|
|
|
listItemGroup.items.sort(sortItems);
|
2024-04-19 06:00:40 -05:00
|
|
|
listItemGroup.items.forEach((item) => {
|
|
|
|
item.position = nextPosition;
|
|
|
|
nextPosition += 1;
|
|
|
|
sortedItems.push(item);
|
2025-06-20 00:09:12 +07:00
|
|
|
});
|
2024-04-19 06:00:40 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
shoppingList.value.listItems = sortedItems;
|
|
|
|
}
|
|
|
|
|
2023-10-07 16:06:00 -05:00
|
|
|
function sortListItems() {
|
|
|
|
if (!shoppingList.value?.listItems?.length) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
shoppingList.value.listItems.sort(sortItems);
|
2023-10-07 16:06:00 -05:00
|
|
|
}
|
|
|
|
|
2022-06-02 09:12:05 -08:00
|
|
|
function updateItemsByLabel() {
|
|
|
|
const items: { [prop: string]: ShoppingListItemOut[] } = {};
|
2025-06-20 00:09:12 +07:00
|
|
|
const noLabelText = i18n.t("shopping-list.no-label");
|
2023-01-29 02:39:51 +01:00
|
|
|
const noLabel = [] as ShoppingListItemOut[];
|
2022-01-08 22:24:34 -09:00
|
|
|
|
2022-01-16 15:24:24 -09:00
|
|
|
shoppingList.value?.listItems?.forEach((item) => {
|
|
|
|
if (item.checked) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-01-08 22:24:34 -09:00
|
|
|
if (item.labelId) {
|
|
|
|
if (item.label && item.label.name in items) {
|
|
|
|
items[item.label.name].push(item);
|
2025-06-20 00:09:12 +07:00
|
|
|
}
|
|
|
|
else if (item.label) {
|
2022-01-08 22:24:34 -09:00
|
|
|
items[item.label.name] = [item];
|
|
|
|
}
|
2025-06-20 00:09:12 +07:00
|
|
|
}
|
|
|
|
else {
|
2023-01-29 02:39:51 +01:00
|
|
|
noLabel.push(item);
|
2022-01-08 22:24:34 -09:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2023-01-29 02:39:51 +01:00
|
|
|
if (noLabel.length > 0) {
|
|
|
|
items[noLabelText] = noLabel;
|
2022-01-08 22:24:34 -09:00
|
|
|
}
|
|
|
|
|
2023-02-21 21:58:41 -06:00
|
|
|
// sort the map by label order
|
2025-06-20 00:09:12 +07:00
|
|
|
const orderedLabelNames = shoppingList.value?.labelSettings?.map(labelSetting => labelSetting.label.name);
|
2023-02-21 21:58:41 -06:00
|
|
|
if (!orderedLabelNames) {
|
|
|
|
itemsByLabel.value = items;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const itemsSorted: { [prop: string]: ShoppingListItemOut[] } = {};
|
|
|
|
if (noLabelText in items) {
|
|
|
|
itemsSorted[noLabelText] = items[noLabelText];
|
|
|
|
}
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
orderedLabelNames.forEach((labelName) => {
|
2023-02-21 21:58:41 -06:00
|
|
|
if (labelName in items) {
|
|
|
|
itemsSorted[labelName] = items[labelName];
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
itemsByLabel.value = itemsSorted;
|
2022-06-02 09:12:05 -08:00
|
|
|
}
|
|
|
|
|
2022-01-16 15:24:24 -09:00
|
|
|
// =====================================
|
|
|
|
// Add/Remove Recipe References
|
|
|
|
|
2023-08-21 12:18:37 -05:00
|
|
|
const recipeMap = computed(() => new Map(
|
2025-06-20 00:09:12 +07:00
|
|
|
(shoppingList.value?.recipeReferences?.map(ref => ref.recipe) ?? [])
|
|
|
|
.map(recipe => [recipe.id || "", recipe])),
|
2023-08-21 12:18:37 -05:00
|
|
|
);
|
2022-01-16 15:24:24 -09:00
|
|
|
|
2022-02-13 12:23:42 -09:00
|
|
|
async function addRecipeReferenceToList(recipeId: string) {
|
2023-01-08 11:23:24 -06:00
|
|
|
if (!shoppingList.value || recipeReferenceLoading.value) {
|
2022-01-16 15:24:24 -09:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-01-08 11:23:24 -06:00
|
|
|
loadingCounter.value += 1;
|
|
|
|
recipeReferenceLoading.value = true;
|
2025-02-27 07:58:40 -06:00
|
|
|
const { data } = await userApi.shopping.lists.addRecipes(shoppingList.value.id, [{ recipeId }]);
|
2023-01-08 11:23:24 -06:00
|
|
|
recipeReferenceLoading.value = false;
|
|
|
|
loadingCounter.value -= 1;
|
2022-01-16 15:24:24 -09:00
|
|
|
|
|
|
|
if (data) {
|
|
|
|
refresh();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-13 12:23:42 -09:00
|
|
|
async function removeRecipeReferenceToList(recipeId: string) {
|
2023-01-08 11:23:24 -06:00
|
|
|
if (!shoppingList.value || recipeReferenceLoading.value) {
|
2022-01-16 15:24:24 -09:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-01-08 11:23:24 -06:00
|
|
|
loadingCounter.value += 1;
|
|
|
|
recipeReferenceLoading.value = true;
|
2022-01-16 15:24:24 -09:00
|
|
|
const { data } = await userApi.shopping.lists.removeRecipe(shoppingList.value.id, recipeId);
|
2023-01-08 11:23:24 -06:00
|
|
|
recipeReferenceLoading.value = false;
|
|
|
|
loadingCounter.value -= 1;
|
2022-01-16 15:24:24 -09:00
|
|
|
|
|
|
|
if (data) {
|
|
|
|
refresh();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// =====================================
|
|
|
|
// List Item CRUD
|
|
|
|
|
2022-10-21 20:35:45 -08:00
|
|
|
/*
|
|
|
|
* saveListItem updates and update on the backend server. Additionally, if the item is
|
|
|
|
* checked it will also append that item to the end of the list so that the unchecked items
|
|
|
|
* are at the top of the list.
|
|
|
|
*/
|
2024-06-29 04:58:58 -05:00
|
|
|
function saveListItem(item: ShoppingListItemOut) {
|
2022-01-16 15:24:24 -09:00
|
|
|
if (!shoppingList.value) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2025-06-17 19:41:35 +01:00
|
|
|
// set a temporary updatedAt timestamp prior to refresh so it appears at the top of the checked items
|
|
|
|
item.updatedAt = new Date().toISOString();
|
2022-10-21 20:35:45 -08:00
|
|
|
|
2023-04-25 12:46:58 -05:00
|
|
|
// make updates reflect immediately
|
|
|
|
if (shoppingList.value.listItems) {
|
|
|
|
shoppingList.value.listItems.forEach((oldListItem: ShoppingListItemOut, idx: number) => {
|
|
|
|
if (oldListItem.id === item.id && shoppingList.value?.listItems) {
|
|
|
|
shoppingList.value.listItems[idx] = item;
|
|
|
|
}
|
|
|
|
});
|
2025-06-20 00:09:12 +07:00
|
|
|
// Immediately update checked/unchecked arrays for UI
|
|
|
|
listItems.unchecked = shoppingList.value.listItems.filter(i => !i.checked);
|
|
|
|
listItems.checked = shoppingList.value.listItems.filter(i => i.checked)
|
|
|
|
.sort(sortCheckedItems);
|
2023-04-25 12:46:58 -05:00
|
|
|
}
|
|
|
|
|
2025-07-10 11:41:34 -05:00
|
|
|
// Update the item if it's checked, otherwise updateUncheckedListItems will handle it
|
|
|
|
if (item.checked) {
|
|
|
|
shoppingListItemActions.updateItem(item);
|
|
|
|
}
|
|
|
|
|
2024-04-19 06:00:40 -05:00
|
|
|
updateListItemOrder();
|
2025-07-10 11:41:34 -05:00
|
|
|
updateUncheckedListItems();
|
2022-01-16 15:24:24 -09:00
|
|
|
}
|
|
|
|
|
2024-06-29 04:58:58 -05:00
|
|
|
function deleteListItem(item: ShoppingListItemOut) {
|
2022-01-16 15:24:24 -09:00
|
|
|
if (!shoppingList.value) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-06-29 04:58:58 -05:00
|
|
|
shoppingListItemActions.deleteItem(item);
|
2022-01-16 15:24:24 -09:00
|
|
|
|
2024-06-29 04:58:58 -05:00
|
|
|
// remove the item from the list immediately so the user sees the change
|
|
|
|
if (shoppingList.value.listItems) {
|
2025-06-20 00:09:12 +07:00
|
|
|
shoppingList.value.listItems = shoppingList.value.listItems.filter(itm => itm.id !== item.id);
|
2022-01-16 15:24:24 -09:00
|
|
|
}
|
2024-06-29 04:58:58 -05:00
|
|
|
|
|
|
|
refresh();
|
2022-01-16 15:24:24 -09:00
|
|
|
}
|
|
|
|
|
|
|
|
// =====================================
|
|
|
|
// Create New Item
|
|
|
|
|
|
|
|
const createEditorOpen = ref(false);
|
2024-06-29 04:58:58 -05:00
|
|
|
const createListItemData = ref<ShoppingListItemOut>(listItemFactory());
|
2022-01-16 15:24:24 -09:00
|
|
|
|
2024-06-29 04:58:58 -05:00
|
|
|
function listItemFactory(isFood = false): ShoppingListItemOut {
|
2022-01-16 15:24:24 -09:00
|
|
|
return {
|
2024-06-29 04:58:58 -05:00
|
|
|
id: uuid4(),
|
2022-01-16 15:24:24 -09:00
|
|
|
shoppingListId: id,
|
|
|
|
checked: false,
|
|
|
|
position: shoppingList.value?.listItems?.length || 1,
|
2024-02-18 13:59:03 -06:00
|
|
|
isFood,
|
|
|
|
quantity: 0,
|
2022-01-16 15:24:24 -09:00
|
|
|
note: "",
|
|
|
|
labelId: undefined,
|
2023-01-28 18:45:02 -06:00
|
|
|
unitId: undefined,
|
|
|
|
foodId: undefined,
|
2025-06-20 00:09:12 +07:00
|
|
|
} as ShoppingListItemOut;
|
2022-01-16 15:24:24 -09:00
|
|
|
}
|
|
|
|
|
2025-06-20 00:09:12 +07:00
|
|
|
/* const newMeal = reactive({
|
|
|
|
date: "",
|
|
|
|
title: "",
|
|
|
|
text: "",
|
|
|
|
recipeId: undefined as string | undefined,
|
|
|
|
entryType: "dinner" as PlanEntryType,
|
|
|
|
existing: false,
|
|
|
|
id: 0,
|
|
|
|
groupId: "",
|
|
|
|
userId: $auth.user.value?.id || "",
|
|
|
|
}); */
|
|
|
|
|
2024-06-29 04:58:58 -05:00
|
|
|
function createListItem() {
|
2022-01-16 15:24:24 -09:00
|
|
|
if (!shoppingList.value) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-02-19 17:29:45 +01:00
|
|
|
if (!createListItemData.value.foodId && !createListItemData.value.note) {
|
|
|
|
// don't create an empty item
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-01-08 11:23:24 -06:00
|
|
|
loadingCounter.value += 1;
|
|
|
|
|
|
|
|
// make sure it's inserted into the end of the list, which may have been updated
|
2023-10-07 16:06:00 -05:00
|
|
|
createListItemData.value.position = shoppingList.value?.listItems?.length
|
|
|
|
? (shoppingList.value.listItems.reduce((a, b) => (a.position || 0) > (b.position || 0) ? a : b).position || 0) + 1
|
|
|
|
: 0;
|
2024-06-29 04:58:58 -05:00
|
|
|
|
|
|
|
createListItemData.value.createdAt = new Date().toISOString();
|
2024-08-22 10:14:32 -05:00
|
|
|
createListItemData.value.updatedAt = createListItemData.value.createdAt;
|
2024-06-29 04:58:58 -05:00
|
|
|
|
|
|
|
updateListItemOrder();
|
|
|
|
|
|
|
|
shoppingListItemActions.createItem(createListItemData.value);
|
2023-01-08 11:23:24 -06:00
|
|
|
loadingCounter.value -= 1;
|
2022-01-16 15:24:24 -09:00
|
|
|
|
2024-06-29 04:58:58 -05:00
|
|
|
if (shoppingList.value.listItems) {
|
2025-06-20 00:09:12 +07:00
|
|
|
// add the item to the list immediately so the user sees the change
|
|
|
|
shoppingList.value.listItems.push(createListItemData.value);
|
|
|
|
updateListItemOrder();
|
|
|
|
}
|
2024-06-29 04:58:58 -05:00
|
|
|
createListItemData.value = listItemFactory(createListItemData.value.isFood || false);
|
|
|
|
refresh();
|
2022-01-16 15:24:24 -09:00
|
|
|
}
|
|
|
|
|
2023-01-28 18:45:02 -06:00
|
|
|
function updateIndexUnchecked(uncheckedItems: ShoppingListItemOut[]) {
|
2025-06-20 00:09:12 +07:00
|
|
|
listItems.unchecked = uncheckedItems;
|
|
|
|
listItems.checked = shoppingList.value?.listItems?.filter(item => item.checked) || [];
|
2022-01-16 15:24:24 -09:00
|
|
|
|
2024-04-19 06:00:40 -05:00
|
|
|
// since the user has manually reordered the list, we should preserve this order
|
|
|
|
preserveItemOrder.value = true;
|
|
|
|
|
2025-07-10 11:41:34 -05:00
|
|
|
updateUncheckedListItems();
|
2022-01-16 15:24:24 -09:00
|
|
|
}
|
|
|
|
|
2023-02-21 21:58:41 -06:00
|
|
|
function updateIndexUncheckedByLabel(labelName: string, labeledUncheckedItems: ShoppingListItemOut[]) {
|
|
|
|
if (!itemsByLabel.value[labelName]) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// update this label's item order
|
|
|
|
itemsByLabel.value[labelName] = labeledUncheckedItems;
|
|
|
|
|
|
|
|
// reset list order of all items
|
|
|
|
const allUncheckedItems: ShoppingListItemOut[] = [];
|
|
|
|
for (labelName in itemsByLabel.value) {
|
|
|
|
allUncheckedItems.push(...itemsByLabel.value[labelName]);
|
|
|
|
}
|
|
|
|
|
2024-04-19 06:00:40 -05:00
|
|
|
// since the user has manually reordered the list, we should preserve this order
|
|
|
|
preserveItemOrder.value = true;
|
|
|
|
|
2023-02-21 21:58:41 -06:00
|
|
|
// save changes
|
2025-06-20 00:09:12 +07:00
|
|
|
listItems.unchecked = allUncheckedItems;
|
|
|
|
listItems.checked = shoppingList.value?.listItems?.filter(item => item.checked) || [];
|
2025-07-10 11:41:34 -05:00
|
|
|
updateUncheckedListItems();
|
2023-02-21 21:58:41 -06:00
|
|
|
}
|
|
|
|
|
2024-06-29 04:58:58 -05:00
|
|
|
function deleteListItems(items: ShoppingListItemOut[]) {
|
2022-01-16 15:24:24 -09:00
|
|
|
if (!shoppingList.value) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-06-29 04:58:58 -05:00
|
|
|
items.forEach((item) => {
|
|
|
|
shoppingListItemActions.deleteItem(item);
|
|
|
|
});
|
|
|
|
// remove the items from the list immediately so the user sees the change
|
|
|
|
if (shoppingList.value?.listItems) {
|
|
|
|
const deletedItems = new Set(items.map(item => item.id));
|
2025-06-20 00:09:12 +07:00
|
|
|
shoppingList.value.listItems = shoppingList.value.listItems.filter(itm => !deletedItems.has(itm.id));
|
2022-01-16 15:24:24 -09:00
|
|
|
}
|
2024-06-29 04:58:58 -05:00
|
|
|
|
|
|
|
refresh();
|
2022-01-16 15:24:24 -09:00
|
|
|
}
|
|
|
|
|
2025-07-10 11:41:34 -05:00
|
|
|
function updateUncheckedListItems() {
|
2022-01-16 15:24:24 -09:00
|
|
|
if (!shoppingList.value?.listItems) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2025-07-10 11:41:34 -05:00
|
|
|
// Set position for unchecked items
|
|
|
|
listItems.unchecked.forEach((item: ShoppingListItemOut, idx: number) => {
|
|
|
|
item.position = idx;
|
2024-06-29 04:58:58 -05:00
|
|
|
shoppingListItemActions.updateItem(item);
|
|
|
|
});
|
2025-07-10 11:41:34 -05:00
|
|
|
|
2024-06-29 04:58:58 -05:00
|
|
|
refresh();
|
2022-01-16 15:24:24 -09:00
|
|
|
}
|
|
|
|
|
2022-01-08 22:24:34 -09:00
|
|
|
return {
|
2024-06-28 10:37:21 +01:00
|
|
|
...toRefs(state),
|
2022-01-16 15:24:24 -09:00
|
|
|
addRecipeReferenceToList,
|
2022-01-08 22:24:34 -09:00
|
|
|
allLabels,
|
|
|
|
contextMenu,
|
2022-01-16 15:24:24 -09:00
|
|
|
contextMenuAction,
|
|
|
|
copyListItems,
|
|
|
|
createEditorOpen,
|
|
|
|
createListItem,
|
|
|
|
createListItemData,
|
2022-01-08 22:24:34 -09:00
|
|
|
deleteChecked,
|
2024-06-28 10:37:21 +01:00
|
|
|
openDeleteChecked,
|
2022-01-16 15:24:24 -09:00
|
|
|
deleteListItem,
|
2022-01-08 22:24:34 -09:00
|
|
|
edit,
|
2025-01-26 08:04:40 -06:00
|
|
|
threeDot,
|
2023-04-25 12:46:58 -05:00
|
|
|
getLabelColor,
|
2023-11-05 19:07:02 -06:00
|
|
|
groupSlug,
|
2022-01-16 15:24:24 -09:00
|
|
|
itemsByLabel,
|
|
|
|
listItems,
|
2023-01-08 11:23:24 -06:00
|
|
|
loadingCounter,
|
2023-02-21 21:58:41 -06:00
|
|
|
preferences,
|
2022-01-16 15:24:24 -09:00
|
|
|
presentLabels,
|
2023-08-21 12:18:37 -05:00
|
|
|
recipeMap,
|
2022-01-16 15:24:24 -09:00
|
|
|
removeRecipeReferenceToList,
|
2023-02-21 21:58:41 -06:00
|
|
|
reorderLabelsDialog,
|
|
|
|
toggleReorderLabelsDialog,
|
2024-05-04 22:27:04 +02:00
|
|
|
localLabels,
|
2023-02-21 21:58:41 -06:00
|
|
|
updateLabelOrder,
|
2024-05-04 22:27:04 +02:00
|
|
|
cancelLabelOrder,
|
|
|
|
saveLabelOrder,
|
2022-01-16 15:24:24 -09:00
|
|
|
saveListItem,
|
2022-01-08 22:24:34 -09:00
|
|
|
shoppingList,
|
2022-01-16 15:24:24 -09:00
|
|
|
showChecked,
|
|
|
|
sortByLabels,
|
2024-10-25 02:07:44 -05:00
|
|
|
labelOpenState,
|
|
|
|
toggleShowLabel,
|
2022-01-16 15:24:24 -09:00
|
|
|
toggleShowChecked,
|
|
|
|
uncheckAll,
|
2024-06-28 10:37:21 +01:00
|
|
|
openUncheckAll,
|
|
|
|
checkAll,
|
|
|
|
openCheckAll,
|
2023-01-28 18:45:02 -06:00
|
|
|
updateIndexUnchecked,
|
2023-02-21 21:58:41 -06:00
|
|
|
updateIndexUncheckedByLabel,
|
2022-01-16 15:24:24 -09:00
|
|
|
allUnits,
|
|
|
|
allFoods,
|
2024-10-01 10:47:51 -05:00
|
|
|
getTextColor,
|
2025-06-20 00:09:12 +07:00
|
|
|
isOffline,
|
2025-06-20 11:59:13 +02:00
|
|
|
mdAndUp,
|
2022-01-08 22:24:34 -09:00
|
|
|
};
|
|
|
|
},
|
|
|
|
});
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
.number-input-container {
|
|
|
|
max-width: 50px;
|
|
|
|
}
|
|
|
|
</style>
|