<template> <div class="mk-notes" v-size="[{ max: 500 }]"> <div class="empty" v-if="empty"> <img src="https://xn--931a.moe/assets/info.jpg" alt=""/> <div>{{ $t('noNotes') }}</div> </div> <mk-error v-if="error" @retry="init()"/> <x-list ref="notes" class="notes" :items="notes" v-slot="{ item: note }"> <x-note :note="note" :detail="detail" :key="note.id"/> </x-list> <footer v-if="more"> <button @click="fetchMore()" :disabled="moreFetching" :style="{ cursor: moreFetching ? 'wait' : 'pointer' }" class="_buttonPrimary"> <template v-if="!moreFetching">{{ $t('loadMore') }}</template> <template v-if="moreFetching"><fa :icon="faSpinner" pulse fixed-width/></template> </button> </footer> </div> </template> <script lang="ts"> import Vue from 'vue'; import { faSpinner } from '@fortawesome/free-solid-svg-icons'; import i18n from '../i18n'; import paging from '../scripts/paging'; import XNote from './note.vue'; import XList from './date-separated-list.vue'; export default Vue.extend({ i18n, components: { XNote, XList }, mixins: [ paging({ before: (self) => { self.$emit('before'); }, after: (self, e) => { self.$emit('after', e); } }), ], props: { pagination: { required: true }, detail: { type: Boolean, required: false, default: false }, extract: { required: false } }, data() { return { faSpinner }; }, computed: { notes(): any[] { return this.extract ? this.extract(this.items) : this.items; }, }, methods: { focus() { this.$refs.notes.focus(); } } }); </script> <style lang="scss" scoped> .mk-notes { > .empty { padding: 32px; text-align: center; > img { vertical-align: bottom; height: 128px; margin-bottom: 16px; border-radius: 16px; } } > .notes { > ::v-deep * { margin-bottom: var(--marginFull); } } &.max-width_500px { > .notes { > ::v-deep * { margin-bottom: var(--marginHalf); } } } > footer { text-align: center; &:empty { display: none; } > button { margin: 0; padding: 16px; width: 100%; border-radius: var(--radius); &:disabled { opacity: 0.7; } } } } </style>