Compare commits

..

No commits in common. "main" and "v1.0.0-alpha.1" have entirely different histories.

25 changed files with 313 additions and 794 deletions

View File

@ -17,11 +17,11 @@ module.exports = {
project: resolve(__dirname, './tsconfig.json'), project: resolve(__dirname, './tsconfig.json'),
tsconfigRootDir: __dirname, tsconfigRootDir: __dirname,
ecmaVersion: 2019, // Allows for the parsing of modern ECMAScript features ecmaVersion: 2019, // Allows for the parsing of modern ECMAScript features
sourceType: 'module', // Allows for the use of imports sourceType: 'module' // Allows for the use of imports
}, },
env: { env: {
browser: true, browser: true
}, },
// Rules order is important, please avoid shuffling them // Rules order is important, please avoid shuffling them
@ -44,7 +44,7 @@ module.exports = {
// https://github.com/prettier/eslint-config-prettier#installation // https://github.com/prettier/eslint-config-prettier#installation
// usage with Prettier, provided by 'eslint-config-prettier'. // usage with Prettier, provided by 'eslint-config-prettier'.
'plugin:prettier/recommended', 'prettier', //'plugin:prettier/recommended'
], ],
plugins: [ plugins: [
@ -54,6 +54,10 @@ module.exports = {
// https://eslint.vuejs.org/user-guide/#why-doesn-t-it-work-on-vue-file // https://eslint.vuejs.org/user-guide/#why-doesn-t-it-work-on-vue-file
// required to lint *.vue files // required to lint *.vue files
'vue', '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 // add your custom rules here
@ -61,11 +65,11 @@ module.exports = {
'prefer-promise-reject-errors': 'off', 'prefer-promise-reject-errors': 'off',
// TypeScript // TypeScript
quotes: ['error', 'single', { avoidEscape: true }], quotes: ['warn', 'single', { avoidEscape: true }],
'@typescript-eslint/explicit-function-return-type': 'off', '@typescript-eslint/explicit-function-return-type': 'off',
'@typescript-eslint/explicit-module-boundary-types': 'off', '@typescript-eslint/explicit-module-boundary-types': 'off',
// allow debugger during development only // allow debugger during development only
'no-debugger': process.env.NODE_ENV === 'production' ? 'error' : 'off', 'no-debugger': process.env.NODE_ENV === 'production' ? 'error' : 'off'
}, }
}; }

3
.gitignore vendored
View File

@ -1,6 +1,3 @@
node_modules node_modules
yarn-error.log yarn-error.log
# No need for pinned dependencies, there are only peer dependencies
yarn.lock yarn.lock
.idea

View File

@ -1,3 +0,0 @@
yarn-error.log
.woodpecker/

View File

@ -1,14 +0,0 @@
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

View File

@ -1,9 +0,0 @@
pipeline:
lint:
when:
branch: [main, develop]
image: node:lts-alpine
commands:
- yarn install
- yarn lint

View File

@ -1,5 +1,4 @@
# Flaschengeist `balance` fontend-plugin # Flaschengeist `balance` fontend-plugin
![status-badge](https://ci.os-sc.org/api/badges/Flaschengeist/flaschengeist-balance/status.svg)
This package provides the [Flaschengeist](https://flaschengeist.dev/Flaschengeist/flaschengeist) frontend for the `balance` plugin. This package provides the [Flaschengeist](https://flaschengeist.dev/Flaschengeist/flaschengeist) frontend for the `balance` plugin.

View File

@ -1,6 +1,6 @@
{ {
"license": "MIT", "license": "MIT",
"version": "1.0.0", "version": "1.0.0-alpha.1",
"name": "@flaschengeist/balance", "name": "@flaschengeist/balance",
"author": "Ferdinand Thiessen <rpm@fthiessen.de>", "author": "Ferdinand Thiessen <rpm@fthiessen.de>",
"homepage": "https://flaschengeist.dev/Flaschengeist", "homepage": "https://flaschengeist.dev/Flaschengeist",
@ -15,30 +15,26 @@
"main": "src/index.ts", "main": "src/index.ts",
"types": "src/api.d.ts", "types": "src/api.d.ts",
"scripts": { "scripts": {
"format": "prettier --config ./package.json --write '{,!(node_modules)/**/}*.{js,ts,vue}'", "pretty": "prettier --config ./package.json --write '{,!(node_modules)/**/}*.ts'",
"lint": "eslint --ext .js,.ts,.vue ./src" "lint": "eslint --ext .js,.ts,.vue ./src"
}, },
"devDependencies": { "devDependencies": {
"@flaschengeist/api": "^1.0.0", "@flaschengeist/types": "^1.0.0-alpha.1",
"@flaschengeist/types": "^1.0.0", "@quasar/app": "^3.0.0-beta.25",
"@quasar/app-webpack": "^3.7.2", "@typescript-eslint/eslint-plugin": "^4.24.0",
"@typescript-eslint/eslint-plugin": "^5.8.0", "@typescript-eslint/parser": "^4.24.0",
"@typescript-eslint/parser": "^5.8.0", "axios": "^0.21.1",
"axios": "^0.24.0", "eslint": "^7.26.0",
"eslint": "^8.5.0",
"eslint-config-prettier": "^8.3.0", "eslint-config-prettier": "^8.3.0",
"eslint-plugin-prettier": "^4.0.0", "eslint-plugin-vue": "^7.9.0",
"eslint-plugin-vue": "^8.2.0", "pinia": "^2.0.0-alpha.19",
"eslint-webpack-plugin": "^3.1.1", "prettier": "^2.3.0",
"modify-source-webpack-plugin": "^3.0.0", "quasar": "^2.0.0-beta.18",
"pinia": "^2.0.8", "typescript": "^4.2.4"
"prettier": "^2.5.1",
"quasar": "^2.11.10",
"typescript": "^4.5.4"
}, },
"peerDependencies": { "peerDependencies": {
"@flaschengeist/api": "1.0.0", "@flaschengeist/api": "1.0.0-alpha.1",
"@flaschengeist/users": "1.0.0" "@flaschengeist/users": "1.0.0-alpha.1"
}, },
"prettier": { "prettier": {
"singleQuote": true, "singleQuote": true,
@ -46,4 +42,5 @@
"printWidth": 120, "printWidth": 120,
"arrowParens": "always" "arrowParens": "always"
} }
} }

20
src/balance.d.ts vendored
View File

@ -1,20 +0,0 @@
import { FG_Plugin } from '@flaschengeist/types';
export interface SendFromNotification {
receiver_id: string;
}
export interface SendToNotification {
sender_id: string;
}
export interface TransactionNotification {
author_id: string;
}
export interface BalanceNotification extends FG_Plugin.Notification {
data: {
type: number;
amount: number;
} & (SendFromNotification | SendToNotification | TransactionNotification);
}

View File

@ -13,15 +13,45 @@
:label="shortcut.toFixed(2).toString() + ' €'" :label="shortcut.toFixed(2).toString() + ' €'"
@click="changeBalance(shortcut)" @click="changeBalance(shortcut)"
> >
<q-menu anchor="bottom middle" self="top middle" context-menu> <q-popup-proxy context-menu>
<q-btn label="Entfernen" @click="removeShortcut(shortcut)" /> <q-btn label="Entfernen" @click="removeShortcut(shortcut)" />
</q-menu> </q-popup-proxy>
<q-tooltip> Rechtsklick um Verknüpfung zu entfernen </q-tooltip> <q-tooltip>Rechtsklick um Verknüpfung zu entfernen</q-tooltip>
</q-btn>
</div></q-card-section
>
<q-card-section class="row q-col-gutter-md items-center">
<div class="col-sm-4 col-xs-12">
<q-input
v-model.number="amount"
type="number"
filled
label="Eigener Betrag"
step="0.1"
min="0"
/>
</div>
<div class="col-sm-4 col-xs-6">
<q-btn
style="width: 100%"
color="primary"
label="Anschreiben"
@click="changeBalance(amount * -1)"
><q-tooltip>Rechtsklick um Betrag als Verknüpfung hinzuzufügen</q-tooltip>
<q-popup-proxy v-model="showAddShortcut" context-menu>
<q-btn label="neue Verknüpfung" @click="addShortcut"></q-btn>
</q-popup-proxy>
</q-btn> </q-btn>
</div> </div>
</q-card-section> <div class="col-sm-4 col-xs-6">
<q-card-section class="row q-col-gutter-md items-center"> <q-btn
<balance-add-body :user="user" can-add-shortcut /> v-if="canAddCredit"
style="width: 100%"
color="secondary"
label="Gutschreiben"
@click="changeBalance(amount)"
/>
</div>
</q-card-section> </q-card-section>
</q-card> </q-card>
</template> </template>
@ -30,13 +60,12 @@
import { computed, ref, defineComponent, onBeforeMount } from 'vue'; import { computed, ref, defineComponent, onBeforeMount } from 'vue';
import { hasPermission, useMainStore } from '@flaschengeist/api'; import { hasPermission, useMainStore } from '@flaschengeist/api';
import BalanceHeader from '../components/BalanceHeader.vue'; import BalanceHeader from '../components/BalanceHeader.vue';
import BalanceAddBody from '../components/BalanceAddBody.vue';
import PERMISSIONS from '../permissions'; import PERMISSIONS from '../permissions';
import { useBalanceStore } from '../store'; import { useBalanceStore } from '../store';
export default defineComponent({ export default defineComponent({
name: 'BalanceAdd', name: 'BalanceAdd',
components: { BalanceHeader, BalanceAddBody }, components: { BalanceHeader },
emits: { 'open-history': () => true }, emits: { 'open-history': () => true },
setup(_, { emit }) { setup(_, { emit }) {
const store = useBalanceStore(); const store = useBalanceStore();
@ -46,29 +75,39 @@ export default defineComponent({
void store.getShortcuts(); void store.getShortcuts();
}); });
const amount = ref<number>(0);
const showAddShortcut = ref(false);
const user = ref(mainStore.currentUser); const user = ref(mainStore.currentUser);
const shortCuts = computed(() => store.shortcuts); const shortCuts = computed(() => store.shortcuts);
const canAddCredit = hasPermission(PERMISSIONS.CREDIT);
const showSelector = hasPermission(PERMISSIONS.DEBIT) || hasPermission(PERMISSIONS.CREDIT); const showSelector = hasPermission(PERMISSIONS.DEBIT) || hasPermission(PERMISSIONS.CREDIT);
function addShortcut() {
if (amount.value != 0) void store.createShortcut(amount.value * -1);
}
function removeShortcut(shortcut: number) { function removeShortcut(shortcut: number) {
void store.removeShortcut(shortcut); void store.removeShortcut(shortcut);
} }
function openHistory() {
emit('open-history');
}
async function changeBalance(amount: number) { async function changeBalance(amount: number) {
await store.changeBalance(amount, user.value); await store.changeBalance(amount, user.value);
} }
function openHistory() {
emit('open-history');
}
return { return {
user, user,
addShortcut,
canAddCredit,
removeShortcut, removeShortcut,
showAddShortcut,
changeBalance,
amount,
showSelector, showSelector,
shortCuts, shortCuts,
openHistory, openHistory,
changeBalance,
}; };
}, },
}); });

View File

@ -1,83 +0,0 @@
<template>
<div class="col-sm-4 col-xs-12">
<q-input
v-model.number="amount"
ref="refAmount"
type="number"
filled
label="Eigener Betrag"
step="0.1"
min="0"
suffix="€"
:rules="[check_cent_step]"
/>
</div>
<div class="col-sm-4 col-xs-6">
<q-btn style="width: 100%" color="primary" label="Anschreiben" @click="changeBalance(amount * -1)">
<q-tooltip v-if="canAddShortcut"> Rechtsklick um Betrag als Verknüpfung hinzuzufügen </q-tooltip>
<q-menu v-if="canAddShortcut" anchor="bottom middle" self="top middle" context-menu>
<q-btn label="neue Verknüpfung" @click="addShortcut"></q-btn>
</q-menu>
</q-btn>
</div>
<div class="col-sm-4 col-xs-6">
<q-btn
v-if="canAddCredit"
style="width: 100%"
color="secondary"
label="Gutschreiben"
@click="changeBalance(amount)"
/>
</div>
</template>
<script lang="ts">
import { defineComponent, ref, PropType, computed, watch } from 'vue';
import { QInput } from 'quasar';
import { useBalanceStore } from '../store';
import { hasPermission, useUserStore } from '@flaschengeist/api';
import PERMISSIONS from '../permissions';
import { check_cent_step } from '../utils';
export default defineComponent({
name: 'BalanceAddBody',
props: {
user: {
type: [Object, String] as PropType<FG.User | string>,
required: true,
},
canAddShortcut: {
type: Boolean,
default: false,
},
},
emits: {
'change-balance': (user: FG.User) => user,
},
setup(props, { emit }) {
const store = useBalanceStore();
const userStore = useUserStore();
const amount = ref<number>(0);
const refAmount = ref<QInput>();
async function changeBalance(amount: number) {
await refAmount.value?.validate();
if (amount != 0 && !refAmount.value?.hasError) await store.changeBalance(amount, user.value);
emit('change-balance', user.value);
}
async function addShortcut() {
await refAmount.value?.validate();
if (amount.value != 0 && !refAmount.value?.hasError) void store.createShortcut(amount.value * -1);
}
const canAddCredit = hasPermission(PERMISSIONS.CREDIT);
const user = computed(() =>
(<FG.User>props.user).userid
? <FG.User>props.user
: <FG.User>userStore.users.find((a) => a.userid === <string>props.user)
);
watch(amount, (a) => {
amount.value = Math.abs(a);
});
return { changeBalance, addShortcut, canAddCredit, amount, check_cent_step, refAmount };
},
});
</script>
<style scoped></style>

View File

@ -2,7 +2,7 @@
<q-card-section class="fit row justify-left content-center items-center q-col-gutter-sm"> <q-card-section class="fit row justify-left content-center items-center q-col-gutter-sm">
<div class="col-5"> <div class="col-5">
<div v-if="balance" class="text-h6"> <div v-if="balance" class="text-h6">
Aktueller Stand: {{ balance.balance ? balance.balance.toFixed(2) : parseInt('0').toFixed(2) }} Aktueller Stand: {{ balance.balance.toFixed(2) }}
<q-badge v-if="isLocked" color="negative" align="top"> gesperrt </q-badge> <q-badge v-if="isLocked" color="negative" align="top"> gesperrt </q-badge>
</div> </div>
<q-spinner v-else color="primary" size="3em" /> <q-spinner v-else color="primary" size="3em" />
@ -10,11 +10,9 @@
<div v-if="showSelector" class="col-6"> <div v-if="showSelector" class="col-6">
<UserSelector v-model="user" /> <UserSelector v-model="user" />
</div> </div>
<div class="col"> <div class="col-1 justify-end">
<div class="row fit justify-end content-end items-end">
<q-btn round flat icon="mdi-format-list-checks" @click="openHistory" /> <q-btn round flat icon="mdi-format-list-checks" @click="openHistory" />
</div> </div>
</div>
</q-card-section> </q-card-section>
</template> </template>
@ -24,6 +22,7 @@ import UserSelector from '@flaschengeist/users/src/components/UserSelector.vue';
import { useMainStore } from '@flaschengeist/api'; import { useMainStore } from '@flaschengeist/api';
import { useBalanceStore } from '../store'; import { useBalanceStore } from '../store';
export default defineComponent({ export default defineComponent({
name: 'BalanceHeader', name: 'BalanceHeader',
components: { UserSelector }, components: { UserSelector },
@ -41,7 +40,9 @@ export default defineComponent({
onBeforeMount(() => void store.getBalance(mainStore.currentUser)); onBeforeMount(() => void store.getBalance(mainStore.currentUser));
const balance = computed(() => store.balances.find((x) => x.userid === props.modelValue.userid)); const balance = computed(() =>
store.balances.find((x) => x.userid === props.modelValue.userid)
);
const isLocked = computed( const isLocked = computed(
() => () =>

View File

@ -3,7 +3,21 @@
<BalanceHeader v-model="sender" :show-selector="showSelector" @open-history="openHistory" /> <BalanceHeader v-model="sender" :show-selector="showSelector" @open-history="openHistory" />
<q-separator /> <q-separator />
<q-card-section class="row q-col-gutter-md items-center"> <q-card-section class="row q-col-gutter-md items-center">
<balance-transfer-body :user="sender" /> <div class="col-sm-4 col-xs-12">
<q-input v-model.number="amount" type="number" filled label="Betrag" step="0.1" min="0" />
</div>
<div class="col-sm-4 col-xs-6">
<UserSelector v-model="receiver" label="Empfänger" />
</div>
<div class="col-sm-4 col-xs-6">
<q-btn
style="width: 100%"
color="primary"
:disable="sendDisabled"
label="Senden"
@click="sendAmount"
/>
</div>
</q-card-section> </q-card-section>
</q-card> </q-card>
</template> </template>
@ -11,20 +25,36 @@
<script lang="ts"> <script lang="ts">
import { computed, ref, defineComponent } from 'vue'; import { computed, ref, defineComponent } from 'vue';
import { hasPermission, useMainStore } from '@flaschengeist/api'; import { hasPermission, useMainStore } from '@flaschengeist/api';
import UserSelector from '@flaschengeist/users/src/components/UserSelector.vue';
import BalanceHeader from '../components/BalanceHeader.vue'; import BalanceHeader from '../components/BalanceHeader.vue';
import BalanceTransferBody from '../components/BalanceTransferBody.vue';
import PERMISSIONS from '../permissions'; import PERMISSIONS from '../permissions';
import { useBalanceStore } from '../store';
export default defineComponent({ export default defineComponent({
name: 'BalanceTransfer', name: 'BalanceTransfer',
components: { BalanceHeader, BalanceTransferBody }, components: { BalanceHeader, UserSelector },
emits: { 'open-history': () => true }, emits: { 'open-history': () => true },
setup(_, { emit }) { setup(_, { emit }) {
const store = useBalanceStore();
const mainStore = useMainStore(); const mainStore = useMainStore();
const showSelector = computed(() => hasPermission(PERMISSIONS.SEND_OTHER)); const showSelector = computed(() => hasPermission(PERMISSIONS.SEND_OTHER));
const sender = ref<FG.User | undefined>(mainStore.currentUser); const sender = ref<FG.User | undefined>(mainStore.currentUser);
const receiver = ref<FG.User | undefined>(undefined);
const amount = ref<number>(0);
const sendDisabled = computed(() => {
return !(
receiver.value &&
sender.value &&
sender.value.userid != receiver.value.userid &&
amount.value > 0
);
});
async function sendAmount() {
if (receiver.value) await store.changeBalance(amount.value, receiver.value, sender.value);
}
function openHistory() { function openHistory() {
emit('open-history'); emit('open-history');
@ -32,7 +62,11 @@ export default defineComponent({
return { return {
sender, sender,
receiver,
amount,
sendAmount,
showSelector, showSelector,
sendDisabled,
openHistory, openHistory,
}; };
}, },

View File

@ -1,83 +0,0 @@
<template>
<div class="col-sm-4 col-xs-12">
<q-input
v-model.number="amount"
ref="refAmount"
type="number"
filled
label="Betrag"
step="0.1"
min="0"
suffix="€"
:rules="[check_cent_step]"
/>
</div>
<div class="col-sm-4 col-xs-6">
<UserSelector v-model="receiver" label="Empfänger" />
</div>
<div class="col-sm-4 col-xs-6">
<q-btn style="width: 100%" color="primary" :disable="sendDisabled" label="Senden" @click="sendAmount" />
</div>
</template>
<script lang="ts">
import { computed, defineComponent, PropType, ref } from 'vue';
import { QInput } from 'quasar';
import UserSelector from '@flaschengeist/users/src/components/UserSelector.vue';
import { useBalanceStore } from '../store';
import { check_cent_step } from '../utils';
import { useUserStore } from '@flaschengeist/api';
export default defineComponent({
name: 'BalanceTransferBody',
components: { UserSelector },
props: {
user: {
type: [Object, String] as PropType<FG.User | string>,
required: true,
},
},
emits: {
changeBalance: ({ sender, receiver }: { sender: FG.User; receiver: FG.User }) => sender && receiver,
},
setup(props, { emit }) {
const store = useBalanceStore();
const userStore = useUserStore();
const receiver = ref<FG.User | undefined>(undefined);
const amount = ref<number>(0);
const refAmount = ref<QInput>();
const sender = computed(() =>
(<FG.User>props.user).userid
? <FG.User>props.user
: <FG.User>userStore.users.find((a) => a.userid === <string>props.user)
);
const sendDisabled = computed(() => {
return !(
receiver.value &&
sender.value &&
sender.value.userid != receiver.value.userid &&
amount.value > 0 &&
!refAmount.value?.hasError
);
});
async function sendAmount() {
await refAmount.value?.validate();
if (receiver.value && !refAmount.value?.hasError) {
await store.changeBalance(amount.value, receiver.value, sender.value);
emit('changeBalance', { sender: sender.value, receiver: receiver.value });
}
}
return {
receiver,
sender,
amount,
sendAmount,
sendDisabled,
refAmount,
check_cent_step,
};
},
});
</script>
<style scoped></style>

View File

@ -35,7 +35,7 @@ import { ref, computed, defineComponent, onUnmounted, onMounted, PropType } from
import { useBalanceStore } from '../store'; import { useBalanceStore } from '../store';
export default defineComponent({ export default defineComponent({
name: 'BalanceTransaction', name: 'Transaction',
props: { props: {
transaction: { transaction: {
required: true, required: true,

View File

@ -21,7 +21,7 @@ export default defineComponent({
void store.getBalance(mainStore.currentUser); void store.getBalance(mainStore.currentUser);
}); });
const balance = computed(() => store.balance?.balance || 0); const balance = computed(() => store.balance?.balance || NaN);
return { balance }; return { balance };
}, },

View File

@ -1,46 +1,13 @@
import { FG_Plugin } from '@flaschengeist/types'; import { FG_Plugin } from '@flaschengeist/types';
import { defineAsyncComponent } from 'vue'; import { defineAsyncComponent } from 'vue';
import routes from './routes'; import routes from './routes';
import { BalanceNotification, SendFromNotification, SendToNotification, TransactionNotification } from 'app/balance';
import { useUserStore } from '@flaschengeist/api';
const BalanceTypes = {
send_to: 0x01,
send_from: 0x02,
add_from: 0x03,
sub_from: 0x04,
};
function transpile (msg: FG_Plugin.Notification) {
console.log('notification:', msg);
const message = msg as BalanceNotification;
message.icon = 'mdi-cash';
const store = useUserStore();
void store.getUsers();
if (message.data.type === BalanceTypes.send_from) {
const receiver = <FG.User>store.findUser((<SendFromNotification>message.data).receiver_id);
const author = <FG.User>store.findUser((<SendFromNotification>message.data).author_id);
message.text = `${author.display_name} hat ${message.data.amount.toFixed(2)}€ von dir zu ${receiver.display_name
} überwiesen.`;
} else if (message.data.type === BalanceTypes.send_to) {
const sender = <FG.User>store.findUser((<SendToNotification>message.data).sender_id);
console.log(sender);
message.text = `${sender.display_name} hat dir ${message.data.amount.toFixed(2)}€ überwiesen.`;
} else {
const author = <FG.User>store.findUser((<TransactionNotification>message.data).author_id);
const abgebucht = message.data.type === BalanceTypes.add_from ? 'aufgeladen' : 'abgebucht';
message.text = `${author.display_name} hat ${message.data.amount.toFixed(2)}€ dir ${abgebucht}.`;
}
return message;
}
const plugin: FG_Plugin.Plugin = { const plugin: FG_Plugin.Plugin = {
id: 'balance', id: 'balance',
name: 'Balance', name: 'Balance',
innerRoutes: routes, innerRoutes: routes,
requiredModules: [['balance']], requiredModules: [['balance']],
version: '1.0.0', version: '0.0.2',
notification: transpile,
widgets: [ widgets: [
{ {
priority: 0, priority: 0,

59
src/pages/Admin.vue Normal file
View File

@ -0,0 +1,59 @@
<template>
<div>
<q-page padding>
<q-card>
<q-card-section>
<q-table :rows="rows" row-key="userid" :columns="columns" />
</q-card-section>
</q-card>
</q-page>
</div>
</template>
<script lang="ts">
// TODO: Fill usefull data
import { ref, defineComponent, onMounted } from 'vue';
import { useBalanceStore } from '../store';
export default defineComponent({
// name: 'PageName'
setup() {
const store = useBalanceStore();
onMounted(() => void store.getBalances().then((balances) => rows.value.push(...balances)));
const rows = ref(store.balances);
const columns = [
{
name: 'userid',
label: 'Benutzer ID',
field: 'userid',
required: true,
align: 'left',
sortable: true
},
{
name: 'credit',
label: 'Haben',
field: 'credit',
format: (val: number) => val.toFixed(2)
},
{
name: 'debit',
label: 'Soll',
field: 'debit',
format: (val: number) => val.toFixed(2)
},
{
name: 'balance',
label: 'Kontostand',
format: (_: undefined, row: { debit: number; credit: number }) =>
(row.credit - row.debit).toFixed(2)
}
];
return { rows, columns };
}
});
</script>

View File

@ -1,314 +0,0 @@
<template>
<q-page padding>
<q-table
v-model:pagination="pagination"
:rows="rows"
row-key="userid"
:columns="columns"
:filter="filter"
@request="onRequest"
>
<template #top-right>
<div class="full-width row q-gutter-sm">
<q-input v-model="filter" label="Filter" filled dense debounce="300">
<template #append>
<q-icon name="mdi-magnify" />
</template>
</q-input>
<q-input
ref="refLimit"
v-model.number="limit"
label="Limit"
type="number"
step="0.01"
suffix="€"
filled
dense
:rules="[check_cent_step]"
>
<template #prepend>
<q-btn
:icon="sign === '+' ? 'mdi-plus' : 'mdi-minus'"
round
flat
:text-color="sign === '+' ? 'secondary' : 'negative'"
@click="changeSign"
/>
</template>
</q-input>
<q-btn label="Limits Setzen" color="primary" dense @click="setLimits(limit)" />
</div>
</template>
<template #body="props">
<q-tr :props="props">
<q-td key="firstname" :props="props">
{{ getFirstname(props.row.userid) }}
</q-td>
<q-td key="lastname" :props="props">
{{ getLastname(props.row.userid) }}
</q-td>
<q-td key="limit" :props="props">
{{ getLimit(props.row.userid) || 'Kein Limit' }}
<q-popup-edit
v-slot="scope"
v-model="limit"
buttons
label-cancel="Abbrechen"
label-set="Speichern"
@save="setLimit($event, props.row.userid)"
@cancel="limit = undefined"
>
<q-input v-model.number="scope.value" label="Limit" type="number" step="0.01" suffix="€" filled dense>
<template #prepend>
<q-btn :icon="sign === '+' ? 'mdi-plus' : 'mdi-minus'" round flat @click="changeSign" />
</template>
</q-input>
</q-popup-edit>
</q-td>
<q-td key="balance" :props="props">
{{ getBalance(props.row.debit, props.row.credit) }}
<q-menu
v-model="showMenu[props.row.userid]"
anchor="bottom middle"
self="top middle"
:persistent="$q.platform.is.mobile"
>
<q-card>
<q-card-section>
<q-tab-panels v-model="tab" animated>
<q-tab-panel name="add" class="fit column q-gutter-sm">
<balance-add-body
:user="props.row.userid"
:can-add-shortcut="false"
@change-balance="updateBalance($event, props.row.userid)"
/>
</q-tab-panel>
<q-tab-panel name="transfer" class="fit column q-gutter-sm">
<balance-transfer-body
:user="props.row.userid"
@change-balance="updateBalances($event, $event, props.row.userid)"
/>
</q-tab-panel>
</q-tab-panels>
</q-card-section>
<div v-if="$q.platform.is.mobile" class="full-width row justify-center">
<q-btn v-close-popup label="Abbrechen" flat color="primary" />
</div>
<q-tabs
v-model="tab"
dense
class="text-grey"
active-color="primary"
indicator-color="primary"
align="justify"
narrow-indicator
>
<q-tab name="add" label="Anschreiben" />
<q-tab name="transfer" label="Übertragen" />
</q-tabs>
</q-card>
</q-menu>
</q-td>
</q-tr>
</template>
</q-table>
</q-page>
</template>
<script lang="ts">
import { ref, defineComponent, computed, onBeforeMount } from 'vue';
import { QInput } from 'quasar';
import { check_cent_step } from '../utils';
import { useBalanceStore } from '../store';
import { useUserStore } from '@flaschengeist/api';
import BalanceAddBody from '../components/BalanceAddBody.vue';
import BalanceTransferBody from '../components/BalanceTransferBody.vue';
export default defineComponent({
name: 'BalanceAdminPage',
components: { BalanceTransferBody, BalanceAddBody },
setup() {
const store = useBalanceStore();
const userStore = useUserStore();
const showMenu = ref<{ [userid: string]: boolean }>({});
const refLimit = ref<QInput>();
onBeforeMount(() => {
void userStore.getUsers();
void store.getLimits();
void onRequest({ pagination: pagination.value, filter: filter.value });
});
const rows = computed(() => store.balances);
const limit = ref<number>();
const sign = ref<'+' | '-'>('-');
const filter = ref<string>();
const columns = [
{
name: 'firstname',
label: 'Vorname',
field: 'userid',
align: 'left',
sortable: true,
},
{
name: 'lastname',
label: 'Vorname',
field: 'userid',
align: 'left',
sortable: true,
},
{
name: 'limit',
label: 'Limit',
align: 'right',
field: 'userid',
format: (_: undefined, row: { userid: string }) => getLimit(row.userid),
sortable: true,
},
{
name: 'balance',
label: 'Kontostand',
align: 'right',
field: 'userid',
format: (_: undefined, row: { debit: number; credit: number }) => getBalance(row.debit, row.credit),
sortable: true,
},
];
interface PaginationInterface {
sortBy: string;
descending: boolean;
page: number;
rowsPerPage: number;
rowsNumber: number;
}
const pagination = ref<PaginationInterface>({
sortBy: 'lastname',
rowsPerPage: 5,
rowsNumber: 10,
descending: false,
page: 1,
});
const loading = ref(false);
async function onRequest(props: { pagination: PaginationInterface; filter?: string }) {
const { page, rowsPerPage, sortBy, descending } = props.pagination;
const filter = props.filter;
loading.value = true;
const fetchCount = rowsPerPage === 0 ? pagination.value.rowsNumber : rowsPerPage;
const startRow = (page - 1) * rowsPerPage;
try {
const result = await store.getBalances({
limit: fetchCount,
offset: startRow,
descending: descending ? true : undefined,
sortBy,
filter,
});
pagination.value.page = page;
pagination.value.rowsPerPage = rowsPerPage;
pagination.value.sortBy = sortBy;
pagination.value.descending = descending;
if (result.count) pagination.value.rowsNumber = result.count;
} catch (error) {
console.warn(error);
}
//console.log(pagination.value);
loading.value = false;
}
function getName(val: string) {
return userStore.users.find((a) => a.userid === val)?.display_name;
}
function getLimit(val: string) {
const limit = store.user_limits.find((a) => a.userid === val)?.limit?.toFixed(2);
return limit ? `${limit}` : undefined;
}
function getBalance(debit: number, credit: number) {
return (credit - debit).toFixed(2);
}
async function updateBalance(user: FG.User, ref_showMenu?: string) {
await store.getBalance(user);
if (ref_showMenu) {
showMenu.value[ref_showMenu] = false;
}
}
async function updateBalances({ sender, receiver }: { sender: FG.User; receiver: FG.User }, ref_showMenu?: string) {
await updateBalance(sender);
await updateBalance(receiver);
if (ref_showMenu) {
showMenu.value[ref_showMenu] = false;
}
}
async function setLimit(l: number, userid: string) {
if (sign.value === '-') {
l = -l;
}
await refLimit.value?.validate();
if (refLimit.value?.hasError) {
return;
}
await store.setLimit(l, userid);
limit.value = undefined;
}
function getFirstname(userid: string) {
return userStore.users.find((a) => a.userid === userid)?.firstname;
}
function getLastname(userid: string) {
return userStore.users.find((a) => a.userid === userid)?.lastname;
}
function changeSign() {
sign.value = sign.value === '+' ? '-' : '+';
}
async function setLimits(l: number) {
if (sign.value === '-') {
l = -l;
}
await refLimit.value?.validate();
if (refLimit.value?.hasError) {
return;
}
void store.setLimits(l);
}
const tab = ref('add');
return {
rows,
columns,
limit,
setLimits,
getName,
getLimit,
setLimit,
getBalance,
updateBalance,
updateBalances,
tab,
pagination,
onRequest,
getFirstname,
getLastname,
filter,
showMenu,
sign,
changeSign,
check_cent_step,
refLimit,
};
},
});
</script>

View File

@ -1,7 +1,12 @@
<template> <template>
<q-page padding class="fit row justify-center content-start items-start q-gutter-sm"> <q-page padding class="fit row justify-center content-start items-start q-gutter-sm">
<q-tabs v-if="$q.screen.gt.sm" v-model="tab"> <q-tabs v-if="$q.screen.gt.sm" v-model="tab">
<q-tab v-for="(tabindex, index) in tabs" :key="'tab' + index" :name="tabindex.name" :label="tabindex.label" /> <q-tab
v-for="(tabindex, index) in tabs"
:key="'tab' + index"
:name="tabindex.name"
:label="tabindex.label"
/>
</q-tabs> </q-tabs>
<div v-else class="fit row justify-end"> <div v-else class="fit row justify-end">
<q-btn <q-btn
@ -28,13 +33,18 @@
</q-list> </q-list>
<q-list v-if="show"> <q-list v-if="show">
<div v-for="(transaction, index) in transactions" :key="index" class="col-sm-12"> <div v-for="(transaction, index) in transactions" :key="index" class="col-sm-12">
<balance-transaction v-model:transaction="transactions[index]" @update:transaction="updateBalance" /> <Transaction v-model:transaction="transactions[index]" />
</div> </div>
</q-list> </q-list>
</q-drawer> </q-drawer>
<q-tab-panels v-model="tab" style="background-color: transparent" class="q-pa-none col-12" animated> <q-tab-panels
v-model="tab"
style="background-color: transparent"
class="q-pa-none col-12"
animated
>
<q-tab-panel name="add" class="q-px-xs"> <q-tab-panel name="add" class="q-px-xs">
<balance-add <BalanceAdd
@open-history=" @open-history="
showDrawer = !showDrawer; showDrawer = !showDrawer;
show = true; show = true;
@ -42,7 +52,7 @@
/> />
</q-tab-panel> </q-tab-panel>
<q-tab-panel name="transfer" class="q-px-xs"> <q-tab-panel name="transfer" class="q-px-xs">
<balance-transfer <BalanceTransfer
@open-history=" @open-history="
showDrawer = !showDrawer; showDrawer = !showDrawer;
show = true; show = true;
@ -54,31 +64,23 @@
</template> </template>
<script lang="ts"> <script lang="ts">
import { computed, defineComponent, ref, onMounted, watch } from 'vue'; import { computed, defineComponent, ref, onMounted } from 'vue';
import { useRoute, useRouter } from 'vue-router';
import { hasSomePermissions, useMainStore } from '@flaschengeist/api'; import { hasSomePermissions, useMainStore } from '@flaschengeist/api';
import BalanceTransfer from '../components/BalanceTransfer.vue';
import Transaction from '../components/Transaction.vue';
import BalanceAdd from '../components/BalanceAdd.vue';
import { useBalanceStore } from '../store'; import { useBalanceStore } from '../store';
import PERMISSIONS from '../permissions'; import PERMISSIONS from '../permissions';
import BalanceTransaction from '../components/BalanceTransaction.vue';
import BalanceTransfer from '../components/BalanceTransfer.vue';
import BalanceAdd from '../components/BalanceAdd.vue';
export default defineComponent({ export default defineComponent({
name: 'BalanceManage', name: 'BalanceManage',
components: { BalanceAdd, BalanceTransfer, BalanceTransaction }, components: { BalanceAdd, BalanceTransfer, Transaction },
setup() { setup() {
const balanceStore = useBalanceStore(); const balanceStore = useBalanceStore();
const mainStore = useMainStore(); const mainStore = useMainStore();
const router = useRouter();
const route = useRoute();
const now = new Date(); const now = new Date();
onMounted(async () => { onMounted(() => {
if (tabs.some((value) => value.name == route.query.q_tab)) {
tab.value = route.query.q_tab as string;
}
await router.replace({ query: { q_tab: tab.value } });
void balanceStore.getTransactions(mainStore.currentUser, { void balanceStore.getTransactions(mainStore.currentUser, {
from: new Date(now.getFullYear(), now.getMonth(), now.getDate()), from: new Date(now.getFullYear(), now.getMonth(), now.getDate()),
}); });
@ -91,7 +93,8 @@ export default defineComponent({
.sort((a, b) => (a.time >= b.time ? -1 : 1)); .sort((a, b) => (a.time >= b.time ? -1 : 1));
}); });
const canAdd = () => hasSomePermissions([PERMISSIONS.DEBIT, PERMISSIONS.CREDIT, PERMISSIONS.DEBIT_OWN]); const canAdd = () =>
hasSomePermissions([PERMISSIONS.DEBIT, PERMISSIONS.CREDIT, PERMISSIONS.DEBIT_OWN]);
interface Tab { interface Tab {
name: string; name: string;
@ -118,9 +121,6 @@ export default defineComponent({
*/ */
const showDrawer = ref<boolean>(false); const showDrawer = ref<boolean>(false);
const tab = ref<string>(canAdd() ? 'add' : 'transfer'); const tab = ref<string>(canAdd() ? 'add' : 'transfer');
watch(tab, (val) => {
void router.replace({ query: { q_tab: val } });
});
const show = ref<boolean>(false); const show = ref<boolean>(false);
return { return {
showDrawer, showDrawer,
@ -128,7 +128,6 @@ export default defineComponent({
tabs, tabs,
transactions, transactions,
show, show,
updateBalance: () => balanceStore.getBalance(mainStore.currentUser),
}; };
}, },
}); });

View File

@ -10,7 +10,6 @@
<q-table <q-table
v-model:pagination="pagination" v-model:pagination="pagination"
title="Buchungen" title="Buchungen"
flat
:rows="data" :rows="data"
:columns="columns" :columns="columns"
row-key="id" row-key="id"
@ -18,12 +17,8 @@
binary-state-sort binary-state-sort
@request="onRequest" @request="onRequest"
> >
<template #top="props"> <template #top>
<q-toggle <q-toggle v-model="showCancelled" label="Stornierte einblenden" />
v-model="showCancelled"
label="Stornierte einblenden"
@update:model-value="onRequest({ pagination: props.pagination })"
/>
</template> </template>
<template #body-cell="props"> <template #body-cell="props">
<q-td :props="props" :class="{ 'bg-grey': props.row.reversal_id != null }"> <q-td :props="props" :class="{ 'bg-grey': props.row.reversal_id != null }">
@ -38,27 +33,22 @@
<script lang="ts"> <script lang="ts">
import { formatDateTime, useMainStore, useUserStore } from '@flaschengeist/api'; import { formatDateTime, useMainStore, useUserStore } from '@flaschengeist/api';
import { computed, defineComponent, onMounted, ref, watch } from 'vue'; import { computed, defineComponent, onMounted, ref } from 'vue';
import { useRoute, useRouter } from 'vue-router';
import { useBalanceStore } from '../store'; import { useBalanceStore } from '../store';
export default defineComponent({ export default defineComponent({
name: 'BalanceOverviewPage', // name: 'PageName'
setup() { setup() {
const store = useBalanceStore(); const store = useBalanceStore();
const mainStore = useMainStore(); const mainStore = useMainStore();
const userStore = useUserStore(); const userStore = useUserStore();
const router = useRouter();
const route = useRoute();
onMounted(async () => { onMounted(() => {
if (route.query?.showCancelled == 'true') showCancelled.value = true;
await router.replace({ query: { showCancelled: showCancelled.value } });
void store.getBalance(mainStore.currentUser); void store.getBalance(mainStore.currentUser);
void userStore.getUsers().then(() => void userStore.getUsers().then(() =>
onRequest({ onRequest({
pagination: pagination.value, pagination: pagination.value,
filter: undefined, filter: undefined
}) })
); );
}); });
@ -68,10 +58,10 @@ export default defineComponent({
const loading = ref(false); const loading = ref(false);
const pagination = ref({ const pagination = ref({
sortBy: 'time', sortBy: 'time',
descending: true, descending: false,
page: 1, page: 1,
rowsPerPage: 10, rowsPerPage: 3,
rowsNumber: 10, rowsNumber: 10
}); });
interface PaginationInterface { interface PaginationInterface {
@ -95,8 +85,7 @@ export default defineComponent({
offset: startRow, offset: startRow,
limit: fetchCount, limit: fetchCount,
showCancelled: showCancelled.value, showCancelled: showCancelled.value,
showReversals: false, showReversals: false
descending,
}); });
// clear out existing data and add new // clear out existing data and add new
data.value.splice(0, data.value.length, ...result.transactions); data.value.splice(0, data.value.length, ...result.transactions);
@ -112,7 +101,7 @@ export default defineComponent({
loading.value = false; loading.value = false;
} }
const balance = computed(() => store.balance?.balance || 0); const balance = computed(() => store.balance?.balance || NaN);
const columns = [ const columns = [
{ {
@ -121,7 +110,7 @@ export default defineComponent({
field: 'time', field: 'time',
required: true, required: true,
sortable: true, sortable: true,
format: (val: Date) => formatDateTime(new Date(val), true, true, true), format: (val: Date) => formatDateTime(new Date(val), true, true, true)
}, },
{ {
name: 'type', name: 'type',
@ -131,18 +120,31 @@ export default defineComponent({
else { else {
if (row.receiver_id == null) return 'Angeschrieben'; if (row.receiver_id == null) return 'Angeschrieben';
else { else {
if (row.receiver_id === mainStore.currentUser.userid) if (row.receiver_id === mainStore.currentUser.userid) return 'Bekommen von X';
return `Bekommen von ${<string>getName(row.sender_id)}`; else return 'Gesendet an X';
else return `Gesendet an ${<string>getName(row.receiver_id)}`; }
} }
} }
}, },
{
name: 'text',
label: 'Text',
format: (_: undefined, row: FG.Transaction) => {
if (row.sender_id == null) return 'Gutschrift';
else {
if (row.receiver_id == null) return 'Angeschrieben';
else {
if (row.receiver_id === mainStore.currentUser.userid) return 'Bekommen von X';
else return 'Gesendet an X';
}
}
}
}, },
{ {
name: 'amount', name: 'amount',
label: 'Betrag', label: 'Betrag',
field: 'amount', field: 'amount',
format: (val: number) => `${val.toFixed(2)}`, format: (val: number) => `${val.toFixed(2)}`
}, },
{ {
name: 'author_id', name: 'author_id',
@ -152,23 +154,11 @@ export default defineComponent({
const user = userStore.users.filter((x) => x.userid == val); const user = userStore.users.filter((x) => x.userid == val);
if (user.length > 0) return user[0].display_name; if (user.length > 0) return user[0].display_name;
else return val; else return val;
}, }
}, }
]; ];
function getName(userid: string) {
return userStore.users.find((a) => a.userid === userid)?.display_name;
}
watch(showCancelled, async () => {
await router.replace({ query: { showCancelled: showCancelled.value } });
await onRequest({
pagination: pagination.value,
filter: undefined,
});
});
return { data, pagination, onRequest, loading, balance, columns, showCancelled }; return { data, pagination, onRequest, loading, balance, columns, showCancelled };
}, }
}); });
</script> </script>

View File

@ -19,7 +19,7 @@ const mainRoutes: FG_Plugin.MenuRoute[] = [
route: { route: {
path: 'overview', path: 'overview',
name: 'balance-view', name: 'balance-view',
component: () => import('../pages/BalanceOverviewPage.vue'), component: () => import('../pages/Overview.vue'),
}, },
}, },
{ {
@ -40,7 +40,7 @@ const mainRoutes: FG_Plugin.MenuRoute[] = [
route: { route: {
path: 'admin', path: 'admin',
name: 'balance-admin', name: 'balance-admin',
component: () => import('../pages/BalanceAdminPage.vue'), component: () => import('../pages/Admin.vue'),
}, },
}, },
], ],

View File

@ -1,4 +1,4 @@
import { api, useMainStore, useUserStore } from '@flaschengeist/api'; import { api, useMainStore } from '@flaschengeist/api';
import { defineStore } from 'pinia'; import { defineStore } from 'pinia';
import { AxiosResponse } from 'axios'; import { AxiosResponse } from 'axios';
import { Notify } from 'quasar'; import { Notify } from 'quasar';
@ -19,11 +19,6 @@ export interface TransactionsResponse {
count?: number; count?: number;
} }
export interface UserLimit {
userid: string;
limit?: number;
}
function fixTransaction(t: FG.Transaction) { function fixTransaction(t: FG.Transaction) {
t.time = new Date(t.time); t.time = new Date(t.time);
} }
@ -36,7 +31,6 @@ export const useBalanceStore = defineStore({
shortcuts: [] as number[], shortcuts: [] as number[],
transactions: [] as FG.Transaction[], transactions: [] as FG.Transaction[],
_balances_dirty: 0, _balances_dirty: 0,
user_limits: [] as Array<UserLimit>,
}), }),
getters: { getters: {
@ -64,7 +58,9 @@ export const useBalanceStore = defineStore({
async getShortcuts(force = false) { async getShortcuts(force = false) {
if (force || this.shortcuts.length == 0) { if (force || this.shortcuts.length == 0) {
const mainStore = useMainStore(); const mainStore = useMainStore();
const { data } = await api.get<number[]>(`/users/${mainStore.currentUser.userid}/balance/shortcuts`); const { data } = await api.get<number[]>(
`/users/${mainStore.currentUser.userid}/balance/shortcuts`
);
this.shortcuts = data; this.shortcuts = data;
} }
}, },
@ -77,18 +73,16 @@ export const useBalanceStore = defineStore({
return data; return data;
}, },
async getBalances(filter?: { async getBalances(force = false) {
limit?: number; if (
offset?: number; force ||
sortBy?: string; this.balances.length == 0 ||
descending?: boolean; new Date().getTime() - this._balances_dirty > 60000
filter?: string; ) {
}) { const { data } = await api.get<BalancesResponse[]>('/balance');
const { data } = await api.get<{ balances: BalancesResponse[]; count: number }>('/balance', { this.balances = data;
params: filter, }
}); return this.balances;
this.balances = data.balances;
return { balances: this.balances, count: data.count };
}, },
async changeBalance(amount: number, user: FG.User, sender: FG.User | undefined = undefined) { async changeBalance(amount: number, user: FG.User, sender: FG.User | undefined = undefined) {
@ -100,7 +94,10 @@ export const useBalanceStore = defineStore({
sender: sender?.userid, sender: sender?.userid,
}); });
fixTransaction(data); fixTransaction(data);
if (user.userid === mainStore.currentUser.userid || sender?.userid === mainStore.currentUser.userid) if (
user.userid === mainStore.currentUser.userid ||
sender?.userid === mainStore.currentUser.userid
)
this.transactions.push(data); this.transactions.push(data);
const f = this.balances.find((x) => x.userid === user.userid); const f = this.balances.find((x) => x.userid === user.userid);
if (f) f.balance += amount; if (f) f.balance += amount;
@ -137,22 +134,16 @@ export const useBalanceStore = defineStore({
to?: Date; to?: Date;
showReversals?: boolean; showReversals?: boolean;
showCancelled?: boolean; showCancelled?: boolean;
descending?: boolean;
} }
| undefined = undefined | undefined = undefined
) { ) {
if (!filter) filter = { limit: 10 }; if (!filter) filter = { limit: 10 };
const { data } = await api.get<TransactionsResponse>(`/users/${user.userid}/balance/transactions`, { const { data } = await api.get<TransactionsResponse>(
params: filter, `/users/${user.userid}/balance/transactions`,
}); { params: filter }
);
data.transactions.forEach((t) => fixTransaction(t)); data.transactions.forEach((t) => fixTransaction(t));
if (data.transactions) { if (data.transactions) this.transactions.push(...data.transactions);
data.transactions.forEach((t) => {
const idx = this.transactions.findIndex((x) => x.id === t.id);
if (idx == -1) this.transactions.push(t);
else this.transactions[idx] = t;
});
}
return data; return data;
}, },
@ -169,34 +160,5 @@ export const useBalanceStore = defineStore({
} }
this._balances_dirty = 0; this._balances_dirty = 0;
}, },
async getLimits(filter?: { userids: undefined | string }) {
const { data } = await api.get<Array<UserLimit>>('users/balance/limit', {
params: filter,
});
this.user_limits = data;
},
async setLimits(limit: number) {
await api.put('users/balance/limit', { limit });
useUserStore().users.forEach((user) => {
const user_limit = this.user_limits.find((a) => a.userid === user.userid);
if (user_limit) {
user_limit.limit = limit;
} else {
this.user_limits.push({ userid: user.userid, limit });
}
});
},
async setLimit(limit: number, userid: string) {
await api.put(`users/${userid}/balance/limit`, { limit });
const user_limit = this.user_limits.find((a) => a.userid === userid);
if (user_limit) {
user_limit.limit = limit;
} else {
this.user_limits.push({ userid, limit });
}
},
}, },
}); });

View File

@ -1,3 +0,0 @@
export function check_cent_step(value: number): boolean | string {
return (value * 100) % 1 === 0 || 'Betrag muss in 1-Cent-Schritten angegeben werden';
}

View File

@ -1,5 +1,5 @@
{ {
"extends": "@quasar/app-webpack/tsconfig-preset", "extends": "@quasar/app/tsconfig-preset",
"target": "esnext", "target": "esnext",
"compilerOptions": { "compilerOptions": {
"baseUrl": "src/", "baseUrl": "src/",