add pagination

This commit is contained in:
AntoXa PRO 2023-10-05 14:19:28 +03:00
parent ad27e021fd
commit 03e73d5789
7 changed files with 146 additions and 5 deletions

View File

@ -1,7 +1,7 @@
{ {
"name": "axp-ui", "name": "axp-ui",
"descriiption": "My helper ui lib", "descriiption": "My helper ui lib",
"version": "1.7.3", "version": "1.8.0",
"homepage": "https://antoxahub.ru/antoxa/axp-ui", "homepage": "https://antoxahub.ru/antoxa/axp-ui",
"repository": { "repository": {
"type": "git", "type": "git",

View File

@ -1,6 +1,6 @@
<script setup lang="ts"> <script setup lang="ts">
import { ref } from 'vue' import { ref } from 'vue'
import { cFieldsSchema } from 'axp-ts' import { cFieldsSchema, Pagination } from 'axp-ts'
import { colors, icons } from '.' import { colors, icons } from '.'
import { import {
UiBtn, UiBtn,
@ -20,7 +20,8 @@ import {
UiFieldFile, UiFieldFile,
UiTable, UiTable,
UiCard, UiCard,
UiPickerDays UiPickerDays,
UiPagination
} from './components' } from './components'
// Test fields. // Test fields.
@ -51,6 +52,9 @@ const days = ref([1, 3, 5])
// File. // File.
const testFileValue = ref<FileList | undefined>() const testFileValue = ref<FileList | undefined>()
// Pagination.
const pagination = ref(new Pagination({ page: 4, total: 93 }).toObject())
</script> </script>
<template> <template>
@ -278,8 +282,15 @@ const testFileValue = ref<FileList | undefined>()
</div> </div>
</div> </div>
<div class="item pagination">
<div class="item-title">Pagination</div>
<div class="item-content">
<UiPagination v-model="pagination" />
</div>
</div>
<div class="item complex"> <div class="item complex">
<div class="item-title">Сложные элементы</div> <div class="item-title">Complex elements</div>
<div class="item-content"> <div class="item-content">
<div class="grid grid-cols-2 gap-4"> <div class="grid grid-cols-2 gap-4">
<ui-card title="Карточка"> <ui-card title="Карточка">

View File

@ -0,0 +1,108 @@
<script lang="ts">
import type { TPagination } from 'axp-ts'
export type TUiPaginationPage = {
value: number
active?: boolean
}
export type TUiPaginationProps = {
modelValue?: TPagination
length?: number
}
export type TUiPaginationEmits = {
(e: 'update:model-value', v: TPagination): void
}
</script>
<script setup lang="ts">
import UiBtn from './Btn.vue'
import { computed } from 'vue'
import { Pagination } from 'axp-ts'
// Props.
const props = withDefaults(defineProps<TUiPaginationProps>(), {
// @ts-ignore
modelValue: new Pagination().toObject(),
length: 3
})
// Emits.
const emit = defineEmits<TUiPaginationEmits>()
// Data.
const page = computed(() => props.modelValue.page || 1)
const pages = computed(() => {
let pages: TUiPaginationPage[] = []
const length = props.length
// Pre pages.
for (let i = 1; i < length; i++) {
const value = page.value - i
if (value < 1) break
pages.push({ value })
}
pages = pages.reverse()
// Curren page.
pages.push({ value: page.value, active: true })
// Next pages.
for (let i = 1; i <= length; i++) {
const value = page.value + i
if (value > (props.modelValue.pages || 0)) break
pages.push({ value })
}
// for(page in props.modelValue)
return pages
})
// Handlers.
const updateHandler = (page: number) => {
const pagination = new Pagination({ ...props.modelValue, page })
emit('update:model-value', pagination.toObject())
}
// Etc.
const isShowFirst = computed(() => page.value - props.length > 0)
const isShowLast = computed(
() =>
props.modelValue.pages &&
page.value + props.length < props.modelValue.pages
)
</script>
<template>
<div v-if="props.modelValue.pages" class="ui-pagination">
<div class="pages">
<div v-if="isShowFirst" class="pages-first">
<UiBtn
label="1"
class="page-item"
@click="updateHandler(1)"
/>
<span class="page-separator">...</span>
</div>
<div class="pages-numbers">
<UiBtn
v-for="item in pages"
:label="item.value.toString()"
:class="{ 'page-item': true, active: item.active }"
:disabled="item.active"
@click="updateHandler(item.value)"
/>
</div>
<div v-if="isShowLast" class="pages-last">
<span class="page-separator">...</span>
<UiBtn
:label="props.modelValue.pages.toString()"
@click="updateHandler(props.modelValue.pages)"
class="page-item"
/>
</div>
</div>
</div>
</template>

View File

@ -63,3 +63,6 @@ export * from './PickerDays.vue'
export { default as UiForm } from './Form.vue' export { default as UiForm } from './Form.vue'
export * from './Form.vue' export * from './Form.vue'
export { default as UiPagination } from './Pagination.vue'
export * from './Pagination.vue'

View File

@ -149,3 +149,17 @@
&-toggle-theme &-toggle-theme
.ui-icon .ui-icon
@apply cursor-pointer @apply cursor-pointer
&-pagination
.pages
@apply flex
&-first,
&-numbers,
&-last
@apply flex items-end
.page-item
@apply w-[38px] h-[38px]
.page-separator
@apply block mx-2
&-numbers
.page-item:not(:last-child)
@apply mr-2

View File

@ -57,3 +57,8 @@
&-field .input > *, &-field .input > *,
&-card &-card
@apply rounded @apply rounded
&-pagination
.pages
.active
@apply bg-primary text-white

View File

@ -6,7 +6,7 @@ const config: Config = {
theme: { theme: {
extend: { extend: {
colors: { colors: {
primary: '#3c87a4', primary: '#6d6bff',
accent: '#ab6dfd', accent: '#ab6dfd',
dark: '#363636', dark: '#363636',