Compare commits
10 Commits
v1.0.0-alp
...
develop
Author | SHA1 | Date |
---|---|---|
|
11d1991279 | |
|
e38e8602ff | |
|
774db1873a | |
|
33d91c4b9a | |
|
76b0caa62e | |
|
c5080a2642 | |
|
a5e343ca37 | |
|
bcb661cc3f | |
|
b0c3eeb6fc | |
|
3d1f07ef68 |
|
@ -44,7 +44,7 @@ module.exports = {
|
|||
|
||||
// https://github.com/prettier/eslint-config-prettier#installation
|
||||
// usage with Prettier, provided by 'eslint-config-prettier'.
|
||||
'prettier', //'plugin:prettier/recommended'
|
||||
'plugin:prettier/recommended',
|
||||
],
|
||||
|
||||
plugins: [
|
||||
|
@ -54,10 +54,6 @@ module.exports = {
|
|||
// https://eslint.vuejs.org/user-guide/#why-doesn-t-it-work-on-vue-file
|
||||
// required to lint *.vue files
|
||||
'vue',
|
||||
|
||||
// https://github.com/typescript-eslint/typescript-eslint/issues/389#issuecomment-509292674
|
||||
// Prettier has not been included as plugin to avoid performance impact
|
||||
// add it as an extension for your IDE
|
||||
],
|
||||
|
||||
// add your custom rules here
|
||||
|
@ -65,9 +61,10 @@ module.exports = {
|
|||
'prefer-promise-reject-errors': 'off',
|
||||
|
||||
// TypeScript
|
||||
quotes: ['warn', 'single', { avoidEscape: true }],
|
||||
quotes: ['error', 'single', { avoidEscape: true }],
|
||||
'@typescript-eslint/explicit-function-return-type': 'off',
|
||||
'@typescript-eslint/explicit-module-boundary-types': 'off',
|
||||
'vue/multi-word-component-names': 'off',
|
||||
|
||||
// allow debugger during development only
|
||||
'no-debugger': process.env.NODE_ENV === 'production' ? 'error' : 'off',
|
||||
|
|
|
@ -2,4 +2,5 @@ node_modules
|
|||
yarn-error.log
|
||||
# No need for sharing this
|
||||
yarn.lock
|
||||
.idea
|
||||
|
||||
|
|
|
@ -0,0 +1 @@
|
|||
.woodpecker/
|
|
@ -0,0 +1,14 @@
|
|||
pipeline:
|
||||
deploy:
|
||||
when:
|
||||
event: tag
|
||||
tag: v*
|
||||
image: node:lts-alpine
|
||||
commands:
|
||||
- echo "//registry.npmjs.org/:_authToken=$NODE_AUTH_TOKEN" > .npmrc
|
||||
- yarn publish --non-interactive
|
||||
secrets: [ node_auth_token ]
|
||||
|
||||
depends_on:
|
||||
- lint
|
||||
|
|
@ -0,0 +1,9 @@
|
|||
pipeline:
|
||||
lint:
|
||||
when:
|
||||
branch: [main, develop]
|
||||
image: node:lts-alpine
|
||||
commands:
|
||||
- yarn install
|
||||
- yarn lint
|
||||
|
|
@ -1,4 +1,5 @@
|
|||
# Flaschengeist `pricelist` fontend-plugin
|
||||

|
||||
|
||||
This package provides the [Flaschengeist](https://flaschengeist.dev/Flaschengeist/flaschengeist) frontend for the pricelist plugin (build and manage drinks, show pricelist and calculate prices).
|
||||
|
||||
|
|
30
package.json
30
package.json
|
@ -1,6 +1,6 @@
|
|||
{
|
||||
"license": "MIT",
|
||||
"version": "1.0.0-alpha.2",
|
||||
"version": "1.0.0-alpha.5",
|
||||
"name": "@flaschengeist/pricelist",
|
||||
"author": "Tim Gröger <flaschengeist@wu5.de>",
|
||||
"homepage": "https://flaschengeist.dev/Flaschengeist",
|
||||
|
@ -19,25 +19,25 @@
|
|||
"lint": "eslint --ext .js,.ts,.vue ./src"
|
||||
},
|
||||
"dependencies": {
|
||||
"vuedraggable": "^4.0.1"
|
||||
"vuedraggable": "^4.1.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@flaschengeist/types": "^1.0.0-alpha.1",
|
||||
"@quasar/app": "^3.0.0",
|
||||
"@typescript-eslint/eslint-plugin": "^4.24.0",
|
||||
"@typescript-eslint/parser": "^4.24.0",
|
||||
"axios": "^0.21.1",
|
||||
"eslint": "^7.26.0",
|
||||
"@flaschengeist/types": "^1.0.0-alpha.10",
|
||||
"@quasar/app": "^3.2.4",
|
||||
"@typescript-eslint/eslint-plugin": "^5.5.0",
|
||||
"@typescript-eslint/parser": "^5.5.0",
|
||||
"eslint": "^8.2.0",
|
||||
"eslint-config-prettier": "^8.3.0",
|
||||
"eslint-plugin-vue": "^7.9.0",
|
||||
"pinia": "^2.0.0-beta.3",
|
||||
"prettier": "^2.3.0",
|
||||
"quasar": "^2.0.0",
|
||||
"typescript": "^4.2.4"
|
||||
"eslint-plugin-prettier": "^4.0.0",
|
||||
"eslint-plugin-vue": "^8.1.1",
|
||||
"pinia": "^2.0.6",
|
||||
"prettier": "^2.5.1",
|
||||
"quasar": "^2.3.3",
|
||||
"typescript": "^4.5.2"
|
||||
},
|
||||
"peerDependencies": {
|
||||
"@flaschengeist/api": "^1.0.0-alpha.1",
|
||||
"@flaschengeist/users": "^1.0.0-alpha.1"
|
||||
"@flaschengeist/api": "^1.0.0-alpha.7",
|
||||
"@flaschengeist/users": "^1.0.0-alpha.3"
|
||||
},
|
||||
"prettier": {
|
||||
"singleQuote": true,
|
||||
|
|
|
@ -7,7 +7,7 @@ declare namespace FG {
|
|||
volume?: number;
|
||||
cost_per_volume?: number;
|
||||
cost_per_package?: number;
|
||||
uuid: string;
|
||||
has_image: boolean;
|
||||
receipt?: Array<string>;
|
||||
tags?: Array<Tag>;
|
||||
type?: DrinkType;
|
||||
|
@ -23,13 +23,13 @@ declare namespace FG {
|
|||
interface DrinkPrice {
|
||||
id: number;
|
||||
price: number;
|
||||
volume: Array<DrinkPriceVolume>;
|
||||
volume?: DrinkPriceVolume;
|
||||
public: boolean;
|
||||
description?: string;
|
||||
}
|
||||
interface DrinkPriceVolume {
|
||||
id: number;
|
||||
drink: Array<Drink>;
|
||||
drink?: Drink;
|
||||
volume: number;
|
||||
min_prices: Array<MinPrices>;
|
||||
prices: Array<DrinkPrice>;
|
||||
|
|
|
@ -38,7 +38,12 @@
|
|||
<template #item="props">
|
||||
<div class="q-pa-xs col-xs-12 col-sm-6 col-md-4">
|
||||
<q-card>
|
||||
<q-img v-if="showPic" style="max-height: 256px" :src="image(props.row.uuid)">
|
||||
<q-img
|
||||
v-if="showPic && props.row.has_image"
|
||||
style="max-height: 256px"
|
||||
:src="image(props.row.id)"
|
||||
fit="contain"
|
||||
>
|
||||
<div
|
||||
v-if="!public && !nodetails && editable"
|
||||
class="absolute-top-right justify-end"
|
||||
|
@ -83,6 +88,33 @@
|
|||
</div>
|
||||
</template>
|
||||
</q-img>
|
||||
<q-img
|
||||
v-if="showPic && !props.row.has_image"
|
||||
class="bg-white"
|
||||
style="max-height: 256px"
|
||||
src="no-image.svg"
|
||||
>
|
||||
<div
|
||||
v-if="!public && !nodetails && editable"
|
||||
class="absolute-top-right justify-end"
|
||||
style="background-color: transparent"
|
||||
>
|
||||
<q-btn
|
||||
round
|
||||
icon="mdi-pencil"
|
||||
style="background-color: rgba(0, 0, 0, 0.5)"
|
||||
@click="editDrink = props.row"
|
||||
/>
|
||||
</div>
|
||||
<div class="absolute-bottom-right justify-end">
|
||||
<div class="text-subtitle1 text-right">
|
||||
{{ props.row.name }}
|
||||
</div>
|
||||
<div class="text-caption text-right">
|
||||
{{ props.row.type.name }}
|
||||
</div>
|
||||
</div>
|
||||
</q-img>
|
||||
<q-card-section v-if="!showPic">
|
||||
<div class="row">
|
||||
<div class="col">
|
||||
|
@ -191,8 +223,8 @@ import { filter, Search } from '../utils/filter';
|
|||
import SearchInput from './SearchInput.vue';
|
||||
import DrinkModify from './DrinkModify.vue';
|
||||
import { DeleteObjects } from '../utils/utils';
|
||||
import { setNewImage, getNewImage } from '../utils/image';
|
||||
import { PERMISSIONS } from '../permissions';
|
||||
import { sort } from '../utils/sort';
|
||||
import { Notify } from 'quasar';
|
||||
|
||||
export default defineComponent({
|
||||
|
@ -222,7 +254,7 @@ export default defineComponent({
|
|||
|
||||
onBeforeMount(() => {
|
||||
//void store.getDrinks();
|
||||
onRequest({
|
||||
void onRequest({
|
||||
pagination: pagination.value,
|
||||
filter: undefined,
|
||||
});
|
||||
|
@ -237,8 +269,6 @@ export default defineComponent({
|
|||
name: 'name',
|
||||
label: 'Name',
|
||||
field: 'name',
|
||||
sortable: true,
|
||||
sort,
|
||||
filterable: true,
|
||||
public: true,
|
||||
},
|
||||
|
@ -248,8 +278,6 @@ export default defineComponent({
|
|||
label: 'Kategorie',
|
||||
field: 'type',
|
||||
format: (val: FG.DrinkType) => `${val.name}`,
|
||||
sortable: true,
|
||||
sort: (a: FG.DrinkType, b: FG.DrinkType) => sort(a.name, b.name),
|
||||
filterable: true,
|
||||
public: true,
|
||||
},
|
||||
|
@ -275,7 +303,6 @@ export default defineComponent({
|
|||
label: 'Artikelnummer',
|
||||
field: 'article_id',
|
||||
sortable: true,
|
||||
sort,
|
||||
filterable: true,
|
||||
public: false,
|
||||
},
|
||||
|
@ -284,7 +311,6 @@ export default defineComponent({
|
|||
label: 'Inhalt in l des Gebinde',
|
||||
field: 'volume',
|
||||
sortable: true,
|
||||
sort,
|
||||
public: false,
|
||||
},
|
||||
{
|
||||
|
@ -292,7 +318,6 @@ export default defineComponent({
|
|||
label: 'Gebindegröße',
|
||||
field: 'package_size',
|
||||
sortable: true,
|
||||
sort,
|
||||
public: false,
|
||||
},
|
||||
{
|
||||
|
@ -301,7 +326,6 @@ export default defineComponent({
|
|||
field: 'cost_per_package',
|
||||
format: (val: number | null) => (val ? `${val.toFixed(3)}€` : ''),
|
||||
sortable: true,
|
||||
sort,
|
||||
public: false,
|
||||
},
|
||||
{
|
||||
|
@ -310,7 +334,6 @@ export default defineComponent({
|
|||
field: 'cost_per_volume',
|
||||
format: (val: number | null) => (val ? `${val.toFixed(3)}€` : ''),
|
||||
sortable: true,
|
||||
sort: (a: ComputedRef, b: ComputedRef) => sort(a.value, b.value),
|
||||
},
|
||||
{
|
||||
name: 'volumes',
|
||||
|
@ -439,12 +462,12 @@ export default defineComponent({
|
|||
void store.updateDrink(drink);
|
||||
}
|
||||
|
||||
async function deleteDrink() {
|
||||
function deleteDrink() {
|
||||
if (editDrink.value) {
|
||||
await store.deleteDrink(editDrink.value);
|
||||
store.deleteDrink(editDrink.value);
|
||||
}
|
||||
editDrink.value = undefined;
|
||||
onRequest({ pagination: pagination.value, filter: search.value });
|
||||
void onRequest({ pagination: pagination.value, filter: search.value });
|
||||
}
|
||||
|
||||
const showNewDrink = ref(false);
|
||||
|
@ -496,7 +519,7 @@ export default defineComponent({
|
|||
tags: [],
|
||||
type: undefined,
|
||||
volumes: [],
|
||||
uuid: '',
|
||||
has_image: false,
|
||||
};
|
||||
|
||||
function newDrink() {
|
||||
|
@ -532,12 +555,15 @@ export default defineComponent({
|
|||
editDrink.value.id = _drink.id;
|
||||
}
|
||||
}
|
||||
if (drinkPic instanceof File) {
|
||||
if (drinkPic instanceof File && drinkPic.name) {
|
||||
await savePicture(drinkPic);
|
||||
if (drink.id > 0) {
|
||||
setNewImage(drink.id);
|
||||
}
|
||||
}
|
||||
editDrink.value = undefined;
|
||||
notLoading.value = true;
|
||||
onRequest({ pagination: pagination.value, filter: search.value });
|
||||
void onRequest({ pagination: pagination.value, filter: search.value });
|
||||
}
|
||||
|
||||
function get_volumes(drink_id: number) {
|
||||
|
@ -546,18 +572,15 @@ export default defineComponent({
|
|||
|
||||
const notLoading = ref(true);
|
||||
|
||||
const imageloading = ref<Array<{ id: number; loading: boolean }>>([]);
|
||||
function getImageLoading(id: number) {
|
||||
const loading = imageloading.value.find((a) => a.id === id);
|
||||
if (loading) {
|
||||
return loading.loading;
|
||||
function image(id: number | undefined) {
|
||||
if (id) {
|
||||
const _newImage = getNewImage(id);
|
||||
if (_newImage) {
|
||||
return `${
|
||||
api.defaults.baseURL || ''
|
||||
}/pricelist/drinks/${id}/picture?thumbnail?t=${_newImage.lastModified.toString()}`;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
function image(uuid: string | undefined) {
|
||||
if (uuid) {
|
||||
return `${api.defaults.baseURL || ''}/pricelist/picture/${uuid}?size=256`;
|
||||
return `${api.defaults.baseURL || ''}/pricelist/drinks/${id}/picture?thumbnail`;
|
||||
}
|
||||
return 'no-image.svg';
|
||||
}
|
||||
|
@ -585,7 +608,6 @@ export default defineComponent({
|
|||
editing_drink,
|
||||
get_volumes,
|
||||
notLoading,
|
||||
getImageLoading,
|
||||
newDrink,
|
||||
hasPermission,
|
||||
PERMISSIONS,
|
||||
|
|
|
@ -203,7 +203,7 @@ export default defineComponent({
|
|||
id: -1,
|
||||
ingredient_id: newIngredient.value.id,
|
||||
volume: newIngredientVolume.value,
|
||||
cost_per_volume: newIngredient.value.cost_per_volume,
|
||||
cost_per_volume: <number>(<FG.Drink>newIngredient.value).cost_per_volume,
|
||||
name: newIngredient.value.name,
|
||||
},
|
||||
extra_ingredient: undefined,
|
||||
|
@ -243,17 +243,11 @@ export default defineComponent({
|
|||
emit('update:modelValue', unref(edit_ingredients));
|
||||
update();
|
||||
}
|
||||
//const drinks = computed(() =>
|
||||
// store.drinks.filter((drink) => {
|
||||
// console.log('computed drinks', drink.name, drink.cost_per_volume);
|
||||
// return drink.cost_per_volume;
|
||||
// })
|
||||
//);
|
||||
const drinks = ref<Array<Drink>>([]);
|
||||
const _extra_ingredients = computed(() => store.extraIngredients);
|
||||
const extra_ingredients = ref(_extra_ingredients.value);
|
||||
|
||||
async function filter_drinks(val, update) {
|
||||
async function filter_drinks(val: string, update: (a: () => void) => void) {
|
||||
let result = <Array<Drink>>[];
|
||||
if (val === '') {
|
||||
result = await store.getDrinks_no_store({ limit: 5, ingredient: true });
|
||||
|
@ -269,7 +263,7 @@ export default defineComponent({
|
|||
drinks.value = result;
|
||||
});
|
||||
}
|
||||
function filter_extra_ingredients(val, update) {
|
||||
function filter_extra_ingredients(val: string, update: (a: () => void) => void) {
|
||||
if (val === '') {
|
||||
update(() => {
|
||||
extra_ingredients.value = _extra_ingredients.value;
|
||||
|
@ -283,7 +277,7 @@ export default defineComponent({
|
|||
}
|
||||
}
|
||||
|
||||
async function filter(val, update) {
|
||||
async function filter(val: string, update: () => void) {
|
||||
await filter_drinks(val, update);
|
||||
filter_extra_ingredients(val, update);
|
||||
}
|
||||
|
|
|
@ -24,28 +24,30 @@
|
|||
</div>
|
||||
</q-card-section>
|
||||
<q-card-section>
|
||||
<q-img :src="image" style="max-height: 256px" fit="contain" />
|
||||
<div class="full-width row">
|
||||
<div class="col-10 q-pa-sm">
|
||||
<q-file
|
||||
v-model="drinkPic"
|
||||
filled
|
||||
clearable
|
||||
:clearable="drinkPic.name != ''"
|
||||
dense
|
||||
accept=".jpg, .jpeg, .png, image/*"
|
||||
@update:model-value="imagePreview"
|
||||
@clear="imgsrc = undefined"
|
||||
@clear="clear"
|
||||
>
|
||||
<template #prepend>
|
||||
<q-icon name="mdi-image" />
|
||||
<template #file>
|
||||
<q-img :src="image" style="max-height: 256px" fit="contain" />
|
||||
</template>
|
||||
</q-file>
|
||||
</div>
|
||||
<div class="col-2 q-pa-sm text-right">
|
||||
<div class="column justify-center">
|
||||
<div class="col-2 q-pa-sm">
|
||||
<q-btn round icon="mdi-delete" color="negative" size="sm" @click="delete_pic">
|
||||
<q-tooltip> Bild entfernen </q-tooltip>
|
||||
</q-btn>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</q-card-section>
|
||||
<q-card-section>
|
||||
<q-select
|
||||
|
@ -138,13 +140,15 @@
|
|||
</q-card-section>
|
||||
<q-card-actions class="justify-around">
|
||||
<q-btn label="Abbrechen" @click="cancel" />
|
||||
<q-btn v-if="can_delete" label="Löschen" color="negative" @click="delete_drink" />
|
||||
<!--<q-btn v-if="can_delete" label="Löschen" color="negative" @click="delete_drink" />-->
|
||||
<q-btn label="Löschen" color="negative" @click="delete_drink" />
|
||||
<q-btn label="Speichern" color="primary" @click="save" />
|
||||
</q-card-actions>
|
||||
</q-card>
|
||||
</template>
|
||||
<script lang="ts">
|
||||
import { clone, calc_min_prices, DeleteObjects, calc_cost_per_volume } from '../utils/utils';
|
||||
import { getNewImage } from '../utils/image';
|
||||
import { defineComponent, PropType, ref, onBeforeMount, computed } from 'vue';
|
||||
import DrinkPriceVolumes from './CalculationTable/DrinkPriceVolumes.vue';
|
||||
import BuildManual from '../components/CalculationTable/BuildManual.vue';
|
||||
|
@ -174,7 +178,6 @@ export default defineComponent({
|
|||
setup(props, { emit }) {
|
||||
onBeforeMount(() => {
|
||||
//edit_drink.value = <Drink>JSON.parse(JSON.stringify(props.drink));
|
||||
edit_drink.value = clone(props.drink);
|
||||
edit_volumes.value = clone(props.drink.volumes);
|
||||
});
|
||||
|
||||
|
@ -188,7 +191,7 @@ export default defineComponent({
|
|||
ingredients: [],
|
||||
});
|
||||
|
||||
const edit_drink = ref<Drink>();
|
||||
const edit_drink = ref<Drink>(clone(props.drink));
|
||||
const edit_volumes = ref<Array<DrinkPriceVolume>>([]);
|
||||
function save() {
|
||||
(<Drink>edit_drink.value).volumes = edit_volumes.value;
|
||||
|
@ -240,7 +243,7 @@ export default defineComponent({
|
|||
edit_drink.value?.receipt?.splice(event, 1);
|
||||
}
|
||||
|
||||
const drinkPic = ref();
|
||||
const drinkPic = ref<File>(new File([], '', {}));
|
||||
const imgsrc = ref();
|
||||
|
||||
const deletePic = ref(false);
|
||||
|
@ -248,9 +251,9 @@ export default defineComponent({
|
|||
function delete_pic() {
|
||||
deletePic.value = true;
|
||||
imgsrc.value = undefined;
|
||||
drinkPic.value = undefined;
|
||||
drinkPic.value = new File([], '', {});
|
||||
if (edit_drink.value) {
|
||||
edit_drink.value.uuid = '';
|
||||
edit_drink.value.has_image = false;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -267,14 +270,23 @@ export default defineComponent({
|
|||
}
|
||||
|
||||
const image = computed(() => {
|
||||
if (deletePic.value) {
|
||||
return 'no-image.svg';
|
||||
}
|
||||
console.log(imgsrc.value, deletePic.value, edit_drink.value);
|
||||
if (imgsrc.value) {
|
||||
return <string>imgsrc.value;
|
||||
}
|
||||
if (edit_drink.value?.uuid) {
|
||||
return `${api.defaults.baseURL || ''}/pricelist/picture/${edit_drink.value.uuid}?size=256`;
|
||||
if (deletePic.value && !imgsrc.value) {
|
||||
return 'no-image.svg';
|
||||
}
|
||||
if (edit_drink.value?.has_image) {
|
||||
const _image = getNewImage(edit_drink.value?.id);
|
||||
if (_image) {
|
||||
return `${api.defaults.baseURL || ''}/pricelist/drinks/${
|
||||
edit_drink.value.id
|
||||
}/picture?thumbnail?t=${_image.lastModified.toString()}`;
|
||||
}
|
||||
return `${api.defaults.baseURL || ''}/pricelist/drinks/${
|
||||
edit_drink.value.id
|
||||
}/picture?thumbnail`;
|
||||
}
|
||||
return 'no-image.svg';
|
||||
});
|
||||
|
@ -323,6 +335,11 @@ export default defineComponent({
|
|||
edit_volumes.value?.some((a) => a.ingredients.length > 0)
|
||||
);
|
||||
|
||||
function clear(val: File) {
|
||||
drinkPic.value = new File([], '', {});
|
||||
imgsrc.value = undefined;
|
||||
}
|
||||
|
||||
return {
|
||||
edit_drink,
|
||||
save,
|
||||
|
@ -349,6 +366,7 @@ export default defineComponent({
|
|||
hasIngredients,
|
||||
hasPermission,
|
||||
PERMISSIONS,
|
||||
clear,
|
||||
};
|
||||
},
|
||||
});
|
||||
|
|
|
@ -82,7 +82,7 @@ import SearchInput from '../components/SearchInput.vue';
|
|||
import { usePricelistStore, Order } from '../store';
|
||||
import { useMainStore } from '@flaschengeist/api';
|
||||
import { Search, filter } from '../utils/filter';
|
||||
import drag from 'vuedraggable';
|
||||
import draggableComponent from 'vuedraggable';
|
||||
|
||||
interface Row {
|
||||
name: string;
|
||||
|
@ -96,7 +96,7 @@ interface Row {
|
|||
|
||||
export default defineComponent({
|
||||
name: 'Pricelist',
|
||||
components: { SearchInput, drag: <ComponentPublicInstance>drag },
|
||||
components: { SearchInput, drag: draggableComponent },
|
||||
props: {
|
||||
public: {
|
||||
type: Boolean,
|
||||
|
@ -115,7 +115,7 @@ export default defineComponent({
|
|||
} else {
|
||||
user.value = '';
|
||||
}
|
||||
onRequest({ pagination: pagination.value, filter: { limit: 10 } });
|
||||
void onRequest({ pagination: pagination.value });
|
||||
});
|
||||
|
||||
const _order = ref<Array<Order>>([
|
||||
|
@ -174,7 +174,7 @@ export default defineComponent({
|
|||
label: 'Name',
|
||||
//field: 'name',
|
||||
field: 'volume',
|
||||
format: (val: FG.DrinkPriceVolume) => val.drink.name,
|
||||
format: (val: FG.DrinkPriceVolume) => <string>val.drink?.name,
|
||||
sortable: true,
|
||||
filterable: true,
|
||||
align: 'left',
|
||||
|
@ -186,7 +186,7 @@ export default defineComponent({
|
|||
field: 'volume',
|
||||
sortable: true,
|
||||
filterable: true,
|
||||
format: (val: FG.DrinkPriceVolume) => val.drink.type.name,
|
||||
format: (val: FG.DrinkPriceVolume) => <string>val.drink?.type?.name,
|
||||
},
|
||||
{
|
||||
name: 'tags',
|
||||
|
@ -197,8 +197,8 @@ export default defineComponent({
|
|||
|
||||
format: (val: FG.DrinkPriceVolume) => {
|
||||
let retVal = '';
|
||||
val.drink.tags.forEach((tag, index) => {
|
||||
if (index >= val.length - 1 && index > 0) {
|
||||
val.drink?.tags?.forEach((tag, index) => {
|
||||
if (index >= (<Array<FG.Tag>>val.drink?.tags).length - 1 && index > 0) {
|
||||
retVal += ', ';
|
||||
}
|
||||
retVal += tag.name;
|
||||
|
|
|
@ -1,13 +1,12 @@
|
|||
import { innerRoutes, outerRoutes } from './routes';
|
||||
import { FG_Plugin } from '@flaschengeist/typings';
|
||||
import { FG_Plugin } from '@flaschengeist/types';
|
||||
|
||||
const plugin: FG_Plugin.Plugin = {
|
||||
id: 'pricelist',
|
||||
name: 'Pricelist',
|
||||
innerRoutes,
|
||||
outerRoutes,
|
||||
requiredModules: [],
|
||||
requiredBackendModules: ['pricelist'],
|
||||
requiredModules: [['pricelist']],
|
||||
version: '0.0.1',
|
||||
widgets: [],
|
||||
};
|
||||
|
|
|
@ -39,7 +39,7 @@ export default defineComponent({
|
|||
void store.get_min_prices().finally(() => {
|
||||
volume.value.min_prices = calc_min_prices(volume.value, undefined, store.min_prices);
|
||||
});
|
||||
void store.getDrinks();
|
||||
void store.getDrinks({ limit: 10 });
|
||||
void store.getDrinkTypes();
|
||||
void store.getExtraIngredients();
|
||||
});
|
||||
|
|
|
@ -25,7 +25,7 @@ export default defineComponent({
|
|||
const mainStore = useMainStore();
|
||||
|
||||
onBeforeMount(() => {
|
||||
void store.getDrinks();
|
||||
void store.getDrinks({ limit: 10 });
|
||||
void store.getPriceListView(mainStore.currentUser.userid);
|
||||
});
|
||||
|
||||
|
|
|
@ -21,7 +21,7 @@
|
|||
<q-img
|
||||
style="max-height: 256px"
|
||||
loading="lazy"
|
||||
:src="image(props.row.uuid)"
|
||||
:src="image(props.row.id)"
|
||||
placeholder-src="no-image.svg"
|
||||
>
|
||||
<div class="absolute-bottom-right justify-end">
|
||||
|
@ -71,10 +71,8 @@ export default defineComponent({
|
|||
setup() {
|
||||
const store = usePricelistStore();
|
||||
onBeforeMount(() => {
|
||||
//void store.getDrinks();
|
||||
onRequest({
|
||||
void onRequest({
|
||||
pagination: pagination.value,
|
||||
filter: { limit: 10, receipt: true },
|
||||
});
|
||||
});
|
||||
const drinks = computed(
|
||||
|
@ -166,9 +164,9 @@ export default defineComponent({
|
|||
|
||||
const search = ref<Search>({ value: '', key: '', label: '' });
|
||||
const search_keys = computed(() => columns_drinks.filter((column) => column.filterable));
|
||||
function image(uuid: string | undefined) {
|
||||
if (uuid) {
|
||||
return `${api.defaults.baseURL || ''}/pricelist/picture/${uuid}?size=256`;
|
||||
function image(id: number) {
|
||||
if (id) {
|
||||
return `${api.defaults.baseURL || ''}/pricelist/drinks/${id}/picture?thumbnail`;
|
||||
}
|
||||
return 'no-image.svg';
|
||||
}
|
||||
|
@ -192,8 +190,8 @@ export default defineComponent({
|
|||
offset: startRow,
|
||||
limit: fetchCount,
|
||||
descending,
|
||||
search_name: props.filter.value,
|
||||
search_key: props.filter.key,
|
||||
search_name: props.filter?.value,
|
||||
search_key: props.filter?.key,
|
||||
receipt: true,
|
||||
});
|
||||
pagination.value.page = page;
|
||||
|
|
|
@ -77,7 +77,7 @@ export default defineComponent({
|
|||
.catch((err) => console.log(err));
|
||||
void store.getTags();
|
||||
void store.getDrinkTypes();
|
||||
void store.getDrinks();
|
||||
void store.getDrinks({ limit: 10 });
|
||||
void store.get_min_prices();
|
||||
});
|
||||
|
||||
|
|
65
src/store.ts
65
src/store.ts
|
@ -14,16 +14,6 @@ interface Drink extends Omit<Omit<FG.Drink, 'cost_per_volume'>, 'volumes'> {
|
|||
_cost_per_volume?: number;
|
||||
}
|
||||
|
||||
interface Pricelist {
|
||||
name: string;
|
||||
type: FG.DrinkType;
|
||||
tags: Array<FG.Tag>;
|
||||
volume: number;
|
||||
price: number;
|
||||
public: boolean;
|
||||
description: string;
|
||||
}
|
||||
|
||||
class DrinkPriceVolume implements DrinkPriceVolume {
|
||||
constructor({ id, volume, prices, ingredients }: FG.DrinkPriceVolume) {
|
||||
this.id = id;
|
||||
|
@ -46,7 +36,7 @@ class Drink {
|
|||
cost_per_package,
|
||||
tags,
|
||||
type,
|
||||
uuid,
|
||||
has_image,
|
||||
receipt,
|
||||
}: FG.Drink) {
|
||||
this.id = id;
|
||||
|
@ -60,7 +50,7 @@ class Drink {
|
|||
this.tags = tags;
|
||||
this.type = type;
|
||||
this.volumes = [];
|
||||
this.uuid = uuid;
|
||||
this.has_image = has_image;
|
||||
this.receipt = receipt || [];
|
||||
}
|
||||
}
|
||||
|
@ -149,9 +139,12 @@ export const usePricelistStore = defineStore({
|
|||
}) {
|
||||
if (!filter) filter = { limit: 10 };
|
||||
console.log('filter_api', filter);
|
||||
const { data } = await api.get<Array<FG.Drink>>('pricelist/drinks', {
|
||||
const { data } = await api.get<{ drinks: Array<FG.Drink>; count: number }>(
|
||||
'pricelist/drinks',
|
||||
{
|
||||
params: filter,
|
||||
});
|
||||
}
|
||||
);
|
||||
this.drinks = [];
|
||||
data.drinks.forEach((drink) => {
|
||||
const _drink = new Drink(drink);
|
||||
|
@ -187,7 +180,7 @@ export const usePricelistStore = defineStore({
|
|||
params: filter,
|
||||
}
|
||||
);
|
||||
const drinks = [];
|
||||
const drinks: Array<Drink> = [];
|
||||
data.drinks.forEach((drink) => {
|
||||
const _drink = new Drink(drink);
|
||||
drink.volumes.forEach((volume) => {
|
||||
|
@ -266,7 +259,9 @@ export const usePricelistStore = defineStore({
|
|||
_ingredient.drink_ingredient &&
|
||||
_ingredient.drink_ingredient.ingredient_id === drink.id
|
||||
) {
|
||||
if (drink.cost_per_volume != null) {
|
||||
_ingredient.drink_ingredient.cost_per_volume = drink.cost_per_volume;
|
||||
}
|
||||
_ingredient.drink_ingredient.name = drink.name;
|
||||
}
|
||||
});
|
||||
|
@ -304,12 +299,12 @@ export const usePricelistStore = defineStore({
|
|||
});
|
||||
const _drink = this.drinks.find((a) => a.id === drink.id);
|
||||
if (_drink) {
|
||||
_drink.uuid = data.uuid;
|
||||
_drink.has_image = data.has_image;
|
||||
}
|
||||
},
|
||||
async delete_drink_picture(drink: Drink) {
|
||||
await api.delete(`pricelist/drinks/${drink.id}/picture`);
|
||||
drink.uuid = '';
|
||||
drink.has_image = false;
|
||||
},
|
||||
async getTags() {
|
||||
const { data } = await api.get<Array<FG.Tag>>('/pricelist/tags');
|
||||
|
@ -360,41 +355,7 @@ export const usePricelistStore = defineStore({
|
|||
this.pricelist_columns_order = data;
|
||||
},
|
||||
},
|
||||
getters: {
|
||||
/*pricelist() {
|
||||
const retVal: Array<Pricelist> = [];
|
||||
this.drinks.forEach((drink) => {
|
||||
drink.volumes.forEach((volume) => {
|
||||
volume.prices.forEach((price) => {
|
||||
retVal.push({
|
||||
name: drink.name,
|
||||
type: <FG.DrinkType>drink.type,
|
||||
tags: <Array<FG.Tag>>drink.tags,
|
||||
volume: <number>volume.volume,
|
||||
price: price.price,
|
||||
public: price.public,
|
||||
description: <string>price.description,
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
||||
return retVal;
|
||||
},*/
|
||||
computed_pricelist() {
|
||||
const retVal: Array<Pricelist> = [];
|
||||
this.pricelist.forEach((price) => {
|
||||
retVal.push({
|
||||
name: price.volume.drink.name,
|
||||
type: <FG.DrinkType>price.volume.drink.type,
|
||||
tags: <Array<FG.Tag>>price.volume.drink.tags,
|
||||
volume: <number>price.volume.volume,
|
||||
public: price.public,
|
||||
price: price.price,
|
||||
description: <string>price.description,
|
||||
});
|
||||
});
|
||||
},
|
||||
},
|
||||
getters: {},
|
||||
});
|
||||
|
||||
export { DrinkPriceVolume, Drink, Order };
|
||||
|
|
|
@ -0,0 +1,20 @@
|
|||
import { ref } from 'vue';
|
||||
|
||||
const newImage = ref<Array<{ id: number; lastModified: Date }>>([]);
|
||||
|
||||
export function getNewImage(id: number) {
|
||||
const image = newImage.value.find((a) => a.id === id);
|
||||
if (image) {
|
||||
return image;
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
export function setNewImage(id: number) {
|
||||
const image = newImage.value.find((a) => a.id === id);
|
||||
if (image) {
|
||||
image.lastModified = new Date();
|
||||
} else {
|
||||
newImage.value.push({ id, lastModified: new Date() });
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue