added auto meal planner back

This commit is contained in:
vabene1111
2025-09-04 22:23:50 +02:00
parent c56dd9563c
commit 0fbb95438a
5 changed files with 137 additions and 6 deletions

View File

@@ -1245,8 +1245,8 @@ class MealPlanSerializer(SpacedModelSerializer, WritableNestedModelSerializer):
class AutoMealPlanSerializer(serializers.Serializer):
start_date = serializers.DateField()
end_date = serializers.DateField()
start_date = serializers.DateTimeField()
end_date = serializers.DateTimeField()
meal_type_id = serializers.IntegerField()
keyword_ids = serializers.ListField()
servings = CustomDecimalField()

View File

@@ -0,0 +1,114 @@
<template>
<v-dialog max-width="600px" :activator="props.activator" v-model="dialog">
<v-card :loading="loading">
<v-closable-card-title v-model="dialog" :title="$t('Auto_Planner')" icon="fa-solid fa-calendar-plus"></v-closable-card-title>
<v-card-text>
<v-form>
<model-select model="MealType" v-model="autoMealPlan.mealTypeId" :object="false"></model-select>
<model-select model="Keyword" v-model="autoMealPlan.keywordIds" mode="tags" :object="false"></model-select>
<v-number-input :label="$t('Servings')" v-model="autoMealPlan.servings"></v-number-input>
<v-date-input :label="$t('Date')"
multiple="range"
v-model="dateRangeValue"
:first-day-of-week="useUserPreferenceStore().deviceSettings.mealplan_startingDayOfWeek"
:show-week="useUserPreferenceStore().deviceSettings.mealplan_displayWeekNumbers"
prepend-icon=""
prepend-inner-icon="$calendar"
></v-date-input>
<model-select model="User" v-model="autoMealPlan.shared" mode="tags"></model-select>
<v-checkbox v-model="autoMealPlan.addshopping" :label="$t('AddToShopping')" hide-details></v-checkbox>
</v-form>
</v-card-text>
<v-card-actions>
<v-btn @click="dialog = false">{{ $t('Cancel') }}</v-btn>
<v-btn color="create" prepend-icon="fa-solid fa-person-running" @click="doAutoPlan()" :loading="loading">{{ $t('Create') }}</v-btn>
</v-card-actions>
</v-card>
</v-dialog>
</template>
<script setup lang="ts">
import {useI18n} from "vue-i18n";
import ModelSelect from "@/components/inputs/ModelSelect.vue";
import {ApiApi, AutoMealPlan} from "@/openapi";
import {onMounted, ref} from "vue";
import VClosableCardTitle from "@/components/dialogs/VClosableCardTitle.vue";
import {VDateInput} from 'vuetify/labs/VDateInput'
import {DateTime} from "luxon";
import {useUserPreferenceStore} from "@/stores/UserPreferenceStore.ts";
import {ErrorMessageType, PreparedMessage, useMessageStore} from "@/stores/MessageStore.ts";
import {useMealPlanStore} from "@/stores/MealPlanStore.ts";
const emit = defineEmits(['change'])
const props = defineProps({
activator: {type: String, default: 'parent'},
})
const {t} = useI18n()
const dialog = defineModel<boolean>({default: false})
const loading = ref(false)
const dateRangeValue = ref([] as Date[])
const autoMealPlan = ref({} as AutoMealPlan)
onMounted(() => {
initializeRequest()
})
/**
* load default values for auto plan creation
*/
function initializeRequest() {
autoMealPlan.value = {
servings: 1,
startDate: DateTime.now().toJSDate(),
endDate: DateTime.now().plus({day: 7}).toJSDate(),
shared: useUserPreferenceStore().userSettings.planShare,
addshopping: useUserPreferenceStore().userSettings.mealplanAutoaddShopping,
} as AutoMealPlan
dateRangeValue.value = []
let currentDate = DateTime.fromJSDate(autoMealPlan.value.startDate).plus({day: 1}).toJSDate()
while (currentDate <= autoMealPlan.value.endDate) {
dateRangeValue.value.push(currentDate)
currentDate = DateTime.fromJSDate(currentDate).plus({day: 1}).toJSDate()
}
}
/**
* perform auto plan creation
*/
function doAutoPlan() {
let api = new ApiApi()
loading.value = true
autoMealPlan.value.startDate = dateRangeValue.value[0]
autoMealPlan.value.endDate = dateRangeValue.value[dateRangeValue.value.length - 1]
console.log('requesting auto plan from ', autoMealPlan.value.startDate, ' to ', autoMealPlan.value.endDate)
api.apiAutoPlanCreate({autoMealPlan: autoMealPlan.value}).then(r => {
dialog.value = false
useMealPlanStore().refreshLastUpdatedPeriod()
initializeRequest()
useMessageStore().addPreparedMessage(PreparedMessage.CREATE_SUCCESS)
}).catch(err => {
useMessageStore().addError(ErrorMessageType.CREATE_ERROR, err)
}).finally(() => {
loading.value = false
})
}
</script>
<style scoped>
</style>

View File

@@ -13,12 +13,17 @@
<template v-if="route.name == 'MealPlanPage'">
<v-divider></v-divider>
<v-list-item prepend-icon="fa-solid fa-calendar-plus" link>
{{$t('Auto_Planner')}}
<auto-plan-dialog></auto-plan-dialog>
</v-list-item>
<v-list-subheader>{{$t('Settings')}}</v-list-subheader>
<v-list-item>
<meal-plan-device-settings></meal-plan-device-settings>
</v-list-item>
</template>
</template>
<script setup lang="ts">
@@ -27,6 +32,7 @@ import {useRoute} from "vue-router";
import {getListModels} from "@/types/Models";
import {useUserPreferenceStore} from "@/stores/UserPreferenceStore";
import MealPlanDeviceSettings from "@/components/settings/MealPlanDeviceSettings.vue";
import AutoPlanDialog from "@/components/dialogs/AutoPlanDialog.vue";
const route = useRoute()

View File

@@ -109,8 +109,8 @@ export function AutoMealPlanToJSON(value?: AutoMealPlan | null): any {
}
return {
'start_date': ((value['startDate']).toISOString().substring(0,10)),
'end_date': ((value['endDate']).toISOString().substring(0,10)),
'start_date': ((value['startDate']).toISOString()),
'end_date': ((value['endDate']).toISOString()),
'meal_type_id': value['mealTypeId'],
'keyword_ids': value['keywordIds'],
'servings': value['servings'],

View File

@@ -16,6 +16,9 @@ export const useMealPlanStore = defineStore(_STORE_ID, () => {
const loading = ref(false)
let settings = ref({})
const lastStartDate = ref(new Date())
const lastEndDate = ref(new Date())
const planList = computed(() => {
let plan_list = [] as MealPlan[]
@@ -49,10 +52,18 @@ export const useMealPlanStore = defineStore(_STORE_ID, () => {
// return this.settings
// })
/**
* based on the last API refresh period, refresh the meal plan list
*/
function refreshLastUpdatedPeriod() {
refreshFromAPI(lastStartDate.value, lastEndDate.value)
}
function refreshFromAPI(from_date: Date, to_date: Date) {
console.log('refresh from api', from_date, to_date)
if (currently_updating.value[0] !== from_date || currently_updating.value[1] !== to_date) {
lastStartDate.value = from_date
lastEndDate.value = to_date
currently_updating.value = [from_date, to_date] // certainly no perfect check but better than nothing
loading.value = true
const api = new ApiApi()
@@ -148,7 +159,7 @@ export const useMealPlanStore = defineStore(_STORE_ID, () => {
// return JSON.parse(s)
// }
// }
return {plans, currently_updating, planList, loading, refreshFromAPI, createObject, updateObject, deleteObject, createOrUpdate}
return {plans, currently_updating, planList, loading, refreshFromAPI, createObject, updateObject, deleteObject, refreshLastUpdatedPeriod, createOrUpdate}
})
// enable hot reload for store