You've already forked isop-mirror
feat: add filter for internships
This commit is contained in:
@@ -11,33 +11,55 @@ use Mpdf\Mpdf;
|
|||||||
|
|
||||||
class InternshipController extends Controller
|
class InternshipController extends Controller
|
||||||
{
|
{
|
||||||
public function all()
|
public function all(Request $request)
|
||||||
{
|
{
|
||||||
$user = auth()->user();
|
$user = $request->user();
|
||||||
|
|
||||||
if ($user->role !== 'ADMIN') {
|
$request->validate([
|
||||||
abort(403, 'Unauthorized');
|
'year' => 'nullable|integer',
|
||||||
|
'company' => 'nullable|string|min:3|max:32',
|
||||||
|
'study_programe' => 'nullable|string|min:3|max:32',
|
||||||
|
'student' => 'nullable|string|min:3|max:32',
|
||||||
|
'page' => 'nullable|integer|min:1',
|
||||||
|
'per_page' => 'nullable|integer|min:-1|max:100',
|
||||||
|
]);
|
||||||
|
|
||||||
|
$perPage = $request->input('per_page', 15);
|
||||||
|
|
||||||
|
// Handle "All" items (-1)
|
||||||
|
if ($perPage == -1) {
|
||||||
|
$perPage = Internship::count();
|
||||||
}
|
}
|
||||||
|
|
||||||
$internships = Internship::all();
|
$internships = Internship::query()
|
||||||
return response()->json($internships);
|
->with(['student.studentData'])
|
||||||
}
|
->when($request->year, function ($query, $year) {
|
||||||
|
$query->whereYear('start', $year);
|
||||||
public function all_my()
|
})
|
||||||
{
|
->when($request->company, function ($query, $company) {
|
||||||
$user = auth()->user();
|
$query->whereHas('company', function ($q) use ($company) {
|
||||||
|
$q->where('name', 'like', "%$company%");
|
||||||
if ($user->role === 'STUDENT') {
|
});
|
||||||
$internships = Internship::whereUserId($user->id)->get();
|
})
|
||||||
} elseif ($user->role === 'EMPLOYER') {
|
->when($request->study_programe, function ($query, $studyPrograme) {
|
||||||
$company = Company::whereContact($user->id)->first();
|
$query->whereHas('student.studentData', function ($q) use ($studyPrograme) {
|
||||||
if (!$company) {
|
$q->where('study_field', 'like', "%$studyPrograme%");
|
||||||
return response()->json(['message' => 'No company associated with this user.'], 404);
|
});
|
||||||
}
|
})
|
||||||
$internships = Internship::whereCompanyId($company->id)->get();
|
->when($request->student, function ($query, $student) {
|
||||||
} else {
|
$query->whereHas('student', function ($q) use ($student) {
|
||||||
abort(403, 'Unauthorized');
|
$q->where('name', 'like', "%$student%");
|
||||||
}
|
});
|
||||||
|
})
|
||||||
|
->when($user->role === 'STUDENT', function ($query) use ($user) {
|
||||||
|
$query->where('user_id', '=', $user->id);
|
||||||
|
})
|
||||||
|
->when($user->role === 'EMPLOYER', function ($query) use ($user) {
|
||||||
|
$query->whereHas('company', function ($q) use ($user) {
|
||||||
|
$q->where('contact', 'like', $user->id);
|
||||||
|
});
|
||||||
|
})
|
||||||
|
->paginate($perPage);
|
||||||
|
|
||||||
return response()->json($internships);
|
return response()->json($internships);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,8 +15,14 @@ class AdministratorOnly
|
|||||||
*/
|
*/
|
||||||
public function handle(Request $request, Closure $next): Response
|
public function handle(Request $request, Closure $next): Response
|
||||||
{
|
{
|
||||||
if ($request->user()->role !== 'ADMIN') {
|
$user = $request->user();
|
||||||
return response(status: 403);
|
|
||||||
|
if ($user === null) {
|
||||||
|
abort(403, 'Unauthorized');
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($user->role !== 'ADMIN') {
|
||||||
|
abort(403, 'Unauthorized');
|
||||||
}
|
}
|
||||||
|
|
||||||
return $next($request);
|
return $next($request);
|
||||||
|
|||||||
@@ -28,7 +28,7 @@ class DatabaseSeeder extends Seeder
|
|||||||
]);
|
]);
|
||||||
|
|
||||||
// create employers and companies
|
// create employers and companies
|
||||||
User::factory(10)
|
User::factory(20)
|
||||||
->create([
|
->create([
|
||||||
'role' => 'EMPLOYER'
|
'role' => 'EMPLOYER'
|
||||||
])
|
])
|
||||||
@@ -39,7 +39,7 @@ class DatabaseSeeder extends Seeder
|
|||||||
});
|
});
|
||||||
|
|
||||||
// create students
|
// create students
|
||||||
User::factory(10)
|
User::factory(20)
|
||||||
->create([
|
->create([
|
||||||
'role' => 'STUDENT'
|
'role' => 'STUDENT'
|
||||||
])
|
])
|
||||||
|
|||||||
@@ -42,8 +42,7 @@ Route::post('/password-reset', [RegisteredUserController::class, 'reset_password
|
|||||||
->name('password.reset');
|
->name('password.reset');
|
||||||
|
|
||||||
Route::prefix('/internships')->group(function () {
|
Route::prefix('/internships')->group(function () {
|
||||||
Route::get("/", [InternshipController::class, 'all'])->name("api.internships");
|
Route::get("/", [InternshipController::class, 'all'])->middleware(['auth:sanctum'])->name("api.internships");
|
||||||
Route::get("/my", [InternshipController::class, 'all_my'])->name("api.internships.my");
|
|
||||||
|
|
||||||
Route::prefix('/{id}')->middleware("auth:sanctum")->group(function () {
|
Route::prefix('/{id}')->middleware("auth:sanctum")->group(function () {
|
||||||
Route::get("/", [InternshipController::class, 'get'])->name("api.internships.get");
|
Route::get("/", [InternshipController::class, 'get'])->name("api.internships.get");
|
||||||
|
|||||||
112
frontend/app/components/InternshipListView.vue
Normal file
112
frontend/app/components/InternshipListView.vue
Normal file
@@ -0,0 +1,112 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
import type { Internship } from '~/types/internships';
|
||||||
|
import type { Paginated } from '~/types/pagination';
|
||||||
|
import { prettyInternshipStatus } from '~/types/internship_status';
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
mode: 'admin' | 'company' | 'student';
|
||||||
|
}>();
|
||||||
|
|
||||||
|
const filters = ref({
|
||||||
|
year: null,
|
||||||
|
company: null,
|
||||||
|
study_programe: null,
|
||||||
|
student: null,
|
||||||
|
});
|
||||||
|
|
||||||
|
const page = ref(1);
|
||||||
|
const itemsPerPage = ref(15);
|
||||||
|
const totalItems = ref(0);
|
||||||
|
|
||||||
|
const allHeaders = [
|
||||||
|
{ title: "Študent", key: "student.name", sortable: false },
|
||||||
|
{ title: "Firma", key: "company.name", sortable: false },
|
||||||
|
{ title: "Od", key: "start", sortable: false },
|
||||||
|
{ title: "Do", key: "end", sortable: false },
|
||||||
|
{ title: "Semester", key: "semester", sortable: false },
|
||||||
|
{ title: "Ročník", key: "year_of_study", sortable: false },
|
||||||
|
{ title: "Študijný odbor", key: "student.student_data.study_field", sortable: false },
|
||||||
|
{ title: "Stav", key: "status", sortable: false },
|
||||||
|
{ title: "Operácie", key: "operations", sortable: false }
|
||||||
|
];
|
||||||
|
|
||||||
|
const headers = props.mode === 'company'
|
||||||
|
? allHeaders.filter(header => header.key !== "company.name")
|
||||||
|
: allHeaders;
|
||||||
|
|
||||||
|
const { data, error, pending } = await useLazySanctumFetch<Paginated<Internship>>('/api/internships', () => ({
|
||||||
|
params: {
|
||||||
|
...filters.value,
|
||||||
|
page: page.value,
|
||||||
|
per_page: itemsPerPage.value,
|
||||||
|
}
|
||||||
|
}), {
|
||||||
|
watch: [filters.value, page, itemsPerPage]
|
||||||
|
});
|
||||||
|
|
||||||
|
watch(data, (newData) => {
|
||||||
|
totalItems.value = newData?.total ?? 0;
|
||||||
|
});
|
||||||
|
|
||||||
|
watch(filters, async () => {
|
||||||
|
page.value = 1;
|
||||||
|
}, { deep: true });
|
||||||
|
|
||||||
|
async function delteInternship(internship: Internship) {
|
||||||
|
// TODO: unimplemented
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<ErrorAlert v-if="error" :error="error.statusMessage ?? error.message" />
|
||||||
|
|
||||||
|
<v-row>
|
||||||
|
<v-col cols="12" md="3">
|
||||||
|
<v-text-field v-model="filters.year" label="Rok" type="number" clearable density="compact" />
|
||||||
|
</v-col>
|
||||||
|
<v-col cols="12" md="3" v-if="mode !== 'company'">
|
||||||
|
<v-text-field v-model="filters.company" label="Názov firmy" clearable density="compact" />
|
||||||
|
</v-col>
|
||||||
|
<v-col cols="12" md="3" v-if="mode !== 'student'">
|
||||||
|
<v-text-field v-model="filters.study_programe" label="Študijný program" clearable density="compact" />
|
||||||
|
</v-col>
|
||||||
|
<v-col cols="12" md="3" v-if="mode !== 'student'">
|
||||||
|
<v-text-field v-model="filters.student" label="Študent" clearable density="compact" />
|
||||||
|
</v-col>
|
||||||
|
</v-row>
|
||||||
|
|
||||||
|
<v-data-table-server v-model:items-per-page="itemsPerPage" v-model:page="page" :headers="headers"
|
||||||
|
:items="data?.data" :items-length="totalItems" :loading="pending">
|
||||||
|
|
||||||
|
<template #item.status="{ item }">
|
||||||
|
{{ prettyInternshipStatus(item.status.status) }}
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template #item.semester="{ item }">
|
||||||
|
{{ item.semester === "WINTER" ? "Zimný" : "Letný" }}
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template #item.operations="{ item }">
|
||||||
|
<v-tooltip text="Editovať">
|
||||||
|
<template #activator="{ props }">
|
||||||
|
<v-btn icon="mdi-pencil" size="small" variant="text"
|
||||||
|
:to="`/dashboard/${mode}/internships/edit/${item.id}`" />
|
||||||
|
</template>
|
||||||
|
</v-tooltip>
|
||||||
|
<v-tooltip text="Vymazať">
|
||||||
|
<template #activator="{ props }">
|
||||||
|
<v-btn icon="mdi-delete" size="small" variant="text" color="error"
|
||||||
|
@click="async () => delteInternship(item)" />
|
||||||
|
</template>
|
||||||
|
</v-tooltip>
|
||||||
|
</template>
|
||||||
|
</v-data-table-server>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
:deep(.v-data-table-header__content) {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -1,7 +1,4 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { prettyInternshipStatus } from '~/types/internship_status';
|
|
||||||
import type { Internship } from '~/types/internships';
|
|
||||||
|
|
||||||
definePageMeta({
|
definePageMeta({
|
||||||
middleware: ['sanctum:auth', 'admin-only'],
|
middleware: ['sanctum:auth', 'admin-only'],
|
||||||
});
|
});
|
||||||
@@ -23,8 +20,6 @@ const headers = [
|
|||||||
{ title: 'Stav', key: 'status', align: 'middle' },
|
{ title: 'Stav', key: 'status', align: 'middle' },
|
||||||
{ title: 'Operácie', key: 'ops', align: 'middle' },
|
{ title: 'Operácie', key: 'ops', align: 'middle' },
|
||||||
];
|
];
|
||||||
|
|
||||||
const { data, error, pending } = await useLazySanctumFetch<Internship[]>('/api/internships');
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
@@ -35,42 +30,7 @@ const { data, error, pending } = await useLazySanctumFetch<Internship[]>('/api/i
|
|||||||
<!-- spacer -->
|
<!-- spacer -->
|
||||||
<div style="height: 40px;"></div>
|
<div style="height: 40px;"></div>
|
||||||
|
|
||||||
<!-- Čakajúca hláška -->
|
<InternshipListView mode="admin" />
|
||||||
<LoadingAlert v-if="pending" />
|
|
||||||
|
|
||||||
<!-- Chybová hláška -->
|
|
||||||
<ErrorAlert v-else-if="error" :error="error?.message" />
|
|
||||||
|
|
||||||
<v-table v-else>
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th v-for="header in headers" :class="'text-' + header.align">
|
|
||||||
<strong>{{ header.title }}</strong>
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
<tr v-for="item in data">
|
|
||||||
<td>{{ item.company.name }}</td>
|
|
||||||
<td>{{ item.student.name }}</td>
|
|
||||||
<td>{{ item.start }}</td>
|
|
||||||
<td>{{ item.end }}</td>
|
|
||||||
<td>{{ item.year_of_study }}</td>
|
|
||||||
<td>{{ item.semester === "WINTER" ? "Zimný" : "Letný" }}</td>
|
|
||||||
<td>
|
|
||||||
<v-btn class="m-1" density="compact" base-color="grey">
|
|
||||||
{{ prettyInternshipStatus(item.status.status) }}
|
|
||||||
</v-btn>
|
|
||||||
</td>
|
|
||||||
<td class="text-left">
|
|
||||||
<v-btn class="m-1 op-btn" density="compact" append-icon="mdi-pencil" base-color="orange"
|
|
||||||
:to="'/dashboard/admin/internships/edit/' + item.id">Editovať</v-btn>
|
|
||||||
<v-btn class="m-1 op-btn" density="compact" append-icon="mdi-trash-can-outline"
|
|
||||||
base-color="red" @click="async () => { }">Vymazať</v-btn>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
</v-table>
|
|
||||||
</v-card>
|
</v-card>
|
||||||
</v-container>
|
</v-container>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -1,7 +1,4 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import type { Internship } from '~/types/internships';
|
|
||||||
import { prettyInternshipStatus } from '~/types/internship_status';
|
|
||||||
|
|
||||||
definePageMeta({
|
definePageMeta({
|
||||||
middleware: ['sanctum:auth', 'company-only'],
|
middleware: ['sanctum:auth', 'company-only'],
|
||||||
});
|
});
|
||||||
@@ -22,8 +19,6 @@ const headers = [
|
|||||||
{ title: 'Stav', key: 'status', align: 'middle' },
|
{ title: 'Stav', key: 'status', align: 'middle' },
|
||||||
{ title: 'Operácie', key: 'ops', align: 'middle' },
|
{ title: 'Operácie', key: 'ops', align: 'middle' },
|
||||||
];
|
];
|
||||||
|
|
||||||
const { data, error, pending } = await useLazySanctumFetch<Internship[]>('/api/internships/my');
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
@@ -34,41 +29,7 @@ const { data, error, pending } = await useLazySanctumFetch<Internship[]>('/api/i
|
|||||||
<!-- spacer -->
|
<!-- spacer -->
|
||||||
<div style="height: 40px;"></div>
|
<div style="height: 40px;"></div>
|
||||||
|
|
||||||
<!-- Čakajúca hláška -->
|
<InternshipListView mode="company" />
|
||||||
<LoadingAlert v-if="pending" />
|
|
||||||
|
|
||||||
<!-- Chybová hláška -->
|
|
||||||
<ErrorAlert v-if="error" :error="error?.message" />
|
|
||||||
|
|
||||||
<v-table v-else>
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th v-for="header in headers" :class="'text-' + header.align">
|
|
||||||
<strong>{{ header.title }}</strong>
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
<tr v-for="item in data">
|
|
||||||
<td>{{ item.student.name }}</td>
|
|
||||||
<td>{{ item.start }}</td>
|
|
||||||
<td>{{ item.end }}</td>
|
|
||||||
<td>{{ item.year_of_study }}</td>
|
|
||||||
<td>{{ item.semester === "WINTER" ? "Zimný" : "Letný" }}</td>
|
|
||||||
<td>
|
|
||||||
<v-btn class="m-1" density="compact" base-color="grey">
|
|
||||||
{{ prettyInternshipStatus(item.status.status) }}
|
|
||||||
</v-btn>
|
|
||||||
</td>
|
|
||||||
<td class="text-left">
|
|
||||||
<v-btn class="m-1 op-btn" density="compact" append-icon="mdi-pencil" base-color="orange"
|
|
||||||
:to="'/dashboard/company/internships/edit/' + item.id">Editovať</v-btn>
|
|
||||||
<v-btn class="m-1 op-btn" density="compact" append-icon="mdi-trash-can-outline"
|
|
||||||
base-color="red" @click="async () => { }">Zmazať</v-btn>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
</v-table>
|
|
||||||
</v-card>
|
</v-card>
|
||||||
</v-container>
|
</v-container>
|
||||||
</template>
|
</template>
|
||||||
@@ -78,12 +39,4 @@ const { data, error, pending } = await useLazySanctumFetch<Internship[]>('/api/i
|
|||||||
padding-left: 10px;
|
padding-left: 10px;
|
||||||
padding-right: 10px;
|
padding-right: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.alert {
|
|
||||||
margin-bottom: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.op-btn {
|
|
||||||
margin: 10px;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
@@ -1,6 +1,4 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { prettyInternshipStatus } from '~/types/internship_status';
|
|
||||||
import type { Internship } from '~/types/internships';
|
|
||||||
import type { User } from '~/types/user';
|
import type { User } from '~/types/user';
|
||||||
|
|
||||||
definePageMeta({
|
definePageMeta({
|
||||||
@@ -14,18 +12,7 @@ useSeoMeta({
|
|||||||
ogDescription: "Portál študenta",
|
ogDescription: "Portál študenta",
|
||||||
});
|
});
|
||||||
|
|
||||||
const headers = [
|
|
||||||
{ title: 'Firma', key: 'company', align: 'left' },
|
|
||||||
{ title: 'Od', key: 'start', align: 'left' },
|
|
||||||
{ title: 'Do', key: 'end', align: 'left' },
|
|
||||||
{ title: 'Ročník', key: 'year_of_study', align: 'middle' },
|
|
||||||
{ title: 'Semester', key: 'semester', align: 'middle' },
|
|
||||||
{ title: 'Stav', key: 'status', align: 'middle' },
|
|
||||||
{ title: 'Operácie', key: 'ops', align: 'middle' },
|
|
||||||
];
|
|
||||||
|
|
||||||
const user = useSanctumUser<User>();
|
const user = useSanctumUser<User>();
|
||||||
const { data, error, pending } = await useLazySanctumFetch<Internship[]>('/api/internships/my');
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
@@ -38,7 +25,7 @@ const { data, error, pending } = await useLazySanctumFetch<Internship[]>('/api/i
|
|||||||
<!-- spacer -->
|
<!-- spacer -->
|
||||||
<div style="height: 40px;"></div>
|
<div style="height: 40px;"></div>
|
||||||
|
|
||||||
<v-btn prepend-icon="mdi-plus" color="blue" class="mr-2" to="/dashboard/student/internship/create">
|
<v-btn prepend-icon="mdi-plus" color="blue" class="mr-2" to="/dashboard/student/internships/create">
|
||||||
Pridať
|
Pridať
|
||||||
</v-btn>
|
</v-btn>
|
||||||
<v-btn prepend-icon="mdi-domain" color="blue" class="mr-2" to="/dashboard/student/companies">
|
<v-btn prepend-icon="mdi-domain" color="blue" class="mr-2" to="/dashboard/student/companies">
|
||||||
@@ -53,41 +40,7 @@ const { data, error, pending } = await useLazySanctumFetch<Internship[]>('/api/i
|
|||||||
|
|
||||||
<h3>Moje praxe</h3>
|
<h3>Moje praxe</h3>
|
||||||
|
|
||||||
<!-- Čakajúca hláška -->
|
<InternshipListView mode="student" />
|
||||||
<LoadingAlert v-if="pending" />
|
|
||||||
|
|
||||||
<!-- Chybová hláška -->
|
|
||||||
<ErrorAlert v-else-if="error" :error="error?.message" />
|
|
||||||
|
|
||||||
<v-table v-else>
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th v-for="header in headers" :class="'text-' + header.align">
|
|
||||||
<strong>{{ header.title }}</strong>
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
<tr v-for="item in data">
|
|
||||||
<td>{{ item.company.name }}</td>
|
|
||||||
<td>{{ item.start }}</td>
|
|
||||||
<td>{{ item.end }}</td>
|
|
||||||
<td>{{ item.year_of_study }}</td>
|
|
||||||
<td>{{ item.semester === "WINTER" ? "Zimný" : "Letný" }}</td>
|
|
||||||
<td>
|
|
||||||
<v-btn class="m-1" density="compact" base-color="grey">
|
|
||||||
{{ prettyInternshipStatus(item.status.status) }}
|
|
||||||
</v-btn>
|
|
||||||
</td>
|
|
||||||
<td class="text-left">
|
|
||||||
<v-btn class="m-1 op-btn" density="compact" append-icon="mdi-pencil" base-color="orange"
|
|
||||||
:to="'/dashboard/student/internship/edit/' + item.id">Editovať</v-btn>
|
|
||||||
<v-btn class="m-1 op-btn" density="compact" append-icon="mdi-trash-can-outline"
|
|
||||||
base-color="red" @click="async () => { }">Zmazať</v-btn>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
</v-table>
|
|
||||||
</v-card>
|
</v-card>
|
||||||
</v-container>
|
</v-container>
|
||||||
</template>
|
</template>
|
||||||
@@ -97,8 +50,4 @@ const { data, error, pending } = await useLazySanctumFetch<Internship[]>('/api/i
|
|||||||
padding-left: 10px;
|
padding-left: 10px;
|
||||||
padding-right: 10px;
|
padding-right: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.op-btn {
|
|
||||||
margin: 10px;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
4
frontend/app/types/pagination.ts
Normal file
4
frontend/app/types/pagination.ts
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
export type Paginated<T> = {
|
||||||
|
data: T[],
|
||||||
|
total: number;
|
||||||
|
};
|
||||||
Reference in New Issue
Block a user