@goplusvn/core 0.1.0 → 0.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/package.json +2 -1
- package/src/assets/erp_wallpaper.png +0 -0
- package/src/assets/goeat_logo.png +0 -0
- package/src/audit/audit-manager.ts +139 -0
- package/src/audit/index.ts +11 -0
- package/src/audit/memory-audit-logger.ts +86 -0
- package/src/audit/types.ts +50 -0
- package/src/auth/auth-service.ts +97 -0
- package/src/auth/index.ts +266 -0
- package/src/code-generation/index.ts +69 -0
- package/src/configs/auth-routes.ts +17 -0
- package/src/configs/crud.ts +136 -0
- package/src/configs/data/navigations.ts +781 -0
- package/src/configs/data/oauth-links.ts +10 -0
- package/src/configs/entities/material-categories.config.ts +125 -0
- package/src/configs/i18n.ts +12 -0
- package/src/configs/index.ts +26 -0
- package/src/configs/status.ts +25 -0
- package/src/configs/themes.ts +100 -0
- package/src/crud/components/crud-bulk-actions.tsx +91 -0
- package/src/crud/components/crud-card-view.tsx +241 -0
- package/src/crud/components/crud-context.tsx +122 -0
- package/src/crud/components/crud-delete-dialog.tsx +145 -0
- package/src/crud/components/crud-dialog.tsx +406 -0
- package/src/crud/components/crud-empty-state.tsx +104 -0
- package/src/crud/components/crud-export-button.tsx +170 -0
- package/src/crud/components/crud-field-renderer.tsx +653 -0
- package/src/crud/components/crud-filter-chips.tsx +102 -0
- package/src/crud/components/crud-filters/checkbox-filter.tsx +97 -0
- package/src/crud/components/crud-filters/datetime-filter.tsx +83 -0
- package/src/crud/components/crud-filters/filter-builder.tsx +66 -0
- package/src/crud/components/crud-filters/index.tsx +76 -0
- package/src/crud/components/crud-filters/radio-filter.tsx +86 -0
- package/src/crud/components/crud-filters/select-filter.tsx +141 -0
- package/src/crud/components/crud-filters/text-filter.tsx +86 -0
- package/src/crud/components/crud-form.tsx +642 -0
- package/src/crud/components/crud-import-dialog.tsx +440 -0
- package/src/crud/components/crud-infinite-scroll.tsx +116 -0
- package/src/crud/components/crud-page.tsx +1017 -0
- package/src/crud/components/crud-provider.tsx +277 -0
- package/src/crud/components/crud-row-actions.tsx +189 -0
- package/src/crud/components/crud-search.tsx +82 -0
- package/src/crud/components/crud-sheet.tsx +336 -0
- package/src/crud/components/crud-table-skeleton.tsx +26 -0
- package/src/crud/components/crud-table-toolbar.tsx +91 -0
- package/src/crud/components/crud-table.tsx +352 -0
- package/src/crud/components/crud-virtual-table.tsx +55 -0
- package/src/crud/components/index.tsx +20 -0
- package/src/crud/crud-filters/checkbox-filter.tsx +87 -0
- package/src/crud/crud-filters/datetime-filter.tsx +82 -0
- package/src/crud/crud-filters/filter-builder.tsx +64 -0
- package/src/crud/crud-filters/index.tsx +78 -0
- package/src/crud/crud-filters/radio-filter.tsx +79 -0
- package/src/crud/crud-filters/select-filter.tsx +148 -0
- package/src/crud/crud-filters/text-filter.tsx +81 -0
- package/src/crud/index.ts +43 -0
- package/src/crud/lib/crud-service.test.ts +334 -0
- package/src/crud/lib/crud-service.ts +358 -0
- package/src/crud/lib/crud-utils.test.ts +354 -0
- package/src/crud/lib/crud-utils.ts +299 -0
- package/src/crud/lib/crud-validator.ts +247 -0
- package/src/crud/lib/data-loader.ts +234 -0
- package/src/crud/lib/field-calculator.ts +241 -0
- package/src/crud/lib/field-formatter.ts +240 -0
- package/src/crud/lib/import-export-service.test.ts +290 -0
- package/src/crud/lib/import-export-service.ts +352 -0
- package/src/crud/lib/import-server-utils.ts +109 -0
- package/src/crud/lib/lazy-loader.ts +241 -0
- package/src/crud/lib/parse-filters.ts +85 -0
- package/src/crud/lib/permissions.ts +52 -0
- package/src/crud/lib/serialize-config.ts +60 -0
- package/src/crud/lib/stream-loader.ts +145 -0
- package/src/crud/lib/translate-config.ts +335 -0
- package/src/crud/lib/types.ts +11 -0
- package/src/crud/pages/entity-crud-page.tsx +144 -0
- package/src/crud/server.ts +8 -0
- package/src/home/constants.tsx +142 -0
- package/src/home/feature-showcase.tsx +171 -0
- package/src/home/home-page.tsx +191 -0
- package/src/home/hooks/index.ts +1 -0
- package/src/home/hooks/useWidgetPreferences.ts +167 -0
- package/src/home/index.ts +33 -0
- package/src/home/quick-access-dialog.tsx +271 -0
- package/src/home/quick-access-menu.tsx +267 -0
- package/src/home/types.ts +140 -0
- package/src/home/welcome-card.tsx +92 -0
- package/src/home/widget-container.tsx +258 -0
- package/src/home/widgets/base-widget.tsx +200 -0
- package/src/home/widgets/customers-widget.tsx +74 -0
- package/src/home/widgets/index.ts +6 -0
- package/src/home/widgets/orders-widget.tsx +87 -0
- package/src/home/widgets/revenue-widget.tsx +71 -0
- package/src/home/widgets/stock-widget.tsx +109 -0
- package/src/hooks/index.tsx +598 -0
- package/src/hooks/use-tenant.test.tsx +30 -0
- package/src/hooks/use-tenant.ts +5 -0
- package/src/index.ts +17 -0
- package/src/infrastructure/__tests__/architecture-verification.spec.ts +103 -0
- package/src/infrastructure/api-service.ts +317 -0
- package/src/infrastructure/cache/cache-manager.ts +107 -0
- package/src/infrastructure/cache/cache.ts +120 -0
- package/src/infrastructure/cache/index.ts +8 -0
- package/src/infrastructure/cache/types.ts +48 -0
- package/src/infrastructure/cron/cron-manager.ts +239 -0
- package/src/infrastructure/cron/index.ts +6 -0
- package/src/infrastructure/cron/types.ts +41 -0
- package/src/infrastructure/event-bus/event-bus.ts +145 -0
- package/src/infrastructure/event-bus/index.ts +2 -0
- package/src/infrastructure/event-bus/types.ts +22 -0
- package/src/infrastructure/index.ts +32 -0
- package/src/infrastructure/lock/decorators.ts +67 -0
- package/src/infrastructure/lock/index.ts +2 -0
- package/src/infrastructure/lock/lock-manager.ts +33 -0
- package/src/infrastructure/logger/index.ts +2 -0
- package/src/infrastructure/logger/logger.ts +96 -0
- package/src/infrastructure/logger/types.ts +25 -0
- package/src/layout/index.tsx +185 -0
- package/src/navigation/index.ts +91 -0
- package/src/notification/index.ts +14 -0
- package/src/notification/notification-service.ts +120 -0
- package/src/notification/storage/in-memory.ts +56 -0
- package/src/notification/storage/index.ts +1 -0
- package/src/notification/types.ts +51 -0
- package/src/organization/branch-service.ts +299 -0
- package/src/organization/branches.config.ts +154 -0
- package/src/organization/index.ts +5 -0
- package/src/plugin/apps-registry.ts +97 -0
- package/src/plugin/index.ts +5 -0
- package/src/plugin/types.ts +41 -0
- package/src/providers/index.tsx +109 -0
- package/src/providers/tenant-provider.tsx +45 -0
- package/src/rbac/components/roles/role-card.tsx +158 -0
- package/src/rbac/components/roles/role-stats-cards.tsx +29 -0
- package/src/rbac/components/roles/role-toolbar.tsx +123 -0
- package/src/rbac/hooks/use-role-operations.ts +159 -0
- package/src/rbac/hooks/use-roles-data.ts +59 -0
- package/src/rbac/index.ts +297 -0
- package/src/rbac/lib/permission-helpers.ts +63 -0
- package/src/rbac/pages/action-list-page.tsx +25 -0
- package/src/rbac/pages/resource-list-page.tsx +25 -0
- package/src/rbac/pages/role-list-page.tsx +378 -0
- package/src/rbac/permission-service.ts +140 -0
- package/src/rbac/permissions.ts +135 -0
- package/src/rbac/resource-service.ts +115 -0
- package/src/rbac/resource-validator.ts +119 -0
- package/src/rbac/role-service.ts +165 -0
- package/src/rbac/server.ts +16 -0
- package/src/rbac/types.ts +38 -0
- package/src/schemas/action.schema.ts +66 -0
- package/src/schemas/branch.schema.ts +52 -0
- package/src/schemas/coming-soon-schema.ts +9 -0
- package/src/schemas/company.schema.ts +44 -0
- package/src/schemas/forgot-passward-schema.ts +9 -0
- package/src/schemas/index.ts +30 -0
- package/src/schemas/material-category.schema.ts +43 -0
- package/src/schemas/material-pricing.schema.ts +74 -0
- package/src/schemas/material.schema.ts +76 -0
- package/src/schemas/materials.ts +52 -0
- package/src/schemas/new-passward-schema.ts +15 -0
- package/src/schemas/partner-company.schema.ts +149 -0
- package/src/schemas/register-schema.ts +36 -0
- package/src/schemas/resource.schema.ts +133 -0
- package/src/schemas/role.schema.ts +11 -0
- package/src/schemas/sign-in-schema.ts +24 -0
- package/src/schemas/supplier-pricing.schema.ts +15 -0
- package/src/schemas/supplier.schema.ts +120 -0
- package/src/schemas/system-category-group.schema.ts +67 -0
- package/src/schemas/system-category.schema.ts +77 -0
- package/src/schemas/system-config.schema.ts +118 -0
- package/src/schemas/uom.schema.ts +75 -0
- package/src/schemas/user-supplier.schema.ts +179 -0
- package/src/schemas/user.schema.ts +18 -0
- package/src/schemas/verify-email-schema.ts +9 -0
- package/src/schemas/warehouse.schema.ts +49 -0
- package/src/system/components/categories/category-list.tsx +529 -0
- package/src/system/components/categories/category-manager.tsx +89 -0
- package/src/system/components/categories/group-sidebar.tsx +308 -0
- package/src/system/components/settings/setting-dialogs.tsx +197 -0
- package/src/system/components/settings/setting-field.tsx +291 -0
- package/src/system/components/settings/setting-form-dialog.tsx +308 -0
- package/src/system/components/settings/settings-groups.ts +80 -0
- package/src/system/components/settings/settings-search.tsx +71 -0
- package/src/system/components/settings/settings-section.tsx +74 -0
- package/src/system/components/settings/settings-sidebar.tsx +81 -0
- package/src/system/constants.ts +3 -0
- package/src/system/index.ts +150 -0
- package/src/system/job-manager.ts +176 -0
- package/src/system/pages/components/categories/category-list.tsx +537 -0
- package/src/system/pages/components/categories/category-manager.tsx +90 -0
- package/src/system/pages/components/categories/group-sidebar.tsx +311 -0
- package/src/system/pages/components/settings/sales-rules-settings.tsx +222 -0
- package/src/system/pages/components/settings/setting-dialogs.tsx +197 -0
- package/src/system/pages/components/settings/setting-field.tsx +292 -0
- package/src/system/pages/components/settings/setting-form-dialog.tsx +308 -0
- package/src/system/pages/components/settings/settings-groups.ts +87 -0
- package/src/system/pages/components/settings/settings-page.tsx +372 -0
- package/src/system/pages/components/settings/settings-search.tsx +71 -0
- package/src/system/pages/components/settings/settings-section.tsx +74 -0
- package/src/system/pages/components/settings/settings-sidebar.tsx +81 -0
- package/src/system/pages/components/settings/system-settings.tsx +244 -0
- package/src/system/pages/system-category-page.tsx +15 -0
- package/src/system/pages/system-settings-page.tsx +380 -0
- package/src/system/schemas/system-category-group.schema.ts +46 -0
- package/src/system/schemas/system-category.schema.ts +56 -0
- package/src/system/services/settings-service.ts +127 -0
- package/src/system/services/system-category-service.ts +63 -0
- package/src/system/types.ts +45 -0
- package/src/types/index.ts +703 -0
- package/src/ui/auth/auth-layout.tsx +135 -0
- package/src/ui/auth/forgot-password-form.tsx +98 -0
- package/src/ui/auth/index.tsx +7 -0
- package/src/ui/auth/new-password-form.tsx +107 -0
- package/src/ui/auth/oauth-links.tsx +30 -0
- package/src/ui/auth/register-form.tsx +202 -0
- package/src/ui/auth/sign-in-form.tsx +238 -0
- package/src/ui/auth/verify-email-form.tsx +104 -0
- package/src/ui/crud/index.tsx +10 -0
- package/src/ui/data-display/accordion.tsx +65 -0
- package/src/ui/data-display/aspect-ratio.tsx +11 -0
- package/src/ui/data-display/avatar.tsx +163 -0
- package/src/ui/data-display/bento-grid.tsx +77 -0
- package/src/ui/data-display/carousel.tsx +249 -0
- package/src/ui/data-display/chart.tsx +363 -0
- package/src/ui/data-display/code-block-highlight.tsx +54 -0
- package/src/ui/data-display/collapsible.tsx +42 -0
- package/src/ui/data-display/compact-stat-bar.tsx +149 -0
- package/src/ui/data-display/data-table/data-table-context.tsx +255 -0
- package/src/ui/data-display/data-table/data-table-empty-state.tsx +133 -0
- package/src/ui/data-display/data-table/data-table-skeleton.tsx +145 -0
- package/src/ui/data-display/data-table/data-table-toolbar.tsx +353 -0
- package/src/ui/data-display/data-table/data-table.tsx +597 -0
- package/src/ui/data-display/data-table/index.ts +44 -0
- package/src/ui/data-display/data-table-column-header.tsx +75 -0
- package/src/ui/data-display/data-table-pagination.tsx +130 -0
- package/src/ui/data-display/data-table-view-options.tsx +59 -0
- package/src/ui/data-display/formatted-number-input.tsx +210 -0
- package/src/ui/data-display/highlight.tsx +20 -0
- package/src/ui/data-display/hover-card.tsx +48 -0
- package/src/ui/data-display/index.tsx +50 -0
- package/src/ui/data-display/iphone-15-pro.tsx +114 -0
- package/src/ui/data-display/kanban/index.ts +4 -0
- package/src/ui/data-display/kanban/kanban-board.tsx +192 -0
- package/src/ui/data-display/kanban/kanban-column.tsx +74 -0
- package/src/ui/data-display/kanban/kanban-item.tsx +50 -0
- package/src/ui/data-display/kanban/kanban-types.ts +21 -0
- package/src/ui/data-display/kpi-card.tsx +68 -0
- package/src/ui/data-display/media-grid.tsx +110 -0
- package/src/ui/data-display/safari.tsx +175 -0
- package/src/ui/data-display/show-more-text.tsx +55 -0
- package/src/ui/data-display/tabs.tsx +68 -0
- package/src/ui/data-display/timeline.tsx +256 -0
- package/src/ui/feedback/alert.tsx +60 -0
- package/src/ui/feedback/context-menu.tsx +245 -0
- package/src/ui/feedback/drawer.tsx +132 -0
- package/src/ui/feedback/error-dialog.tsx +273 -0
- package/src/ui/feedback/index.tsx +183 -0
- package/src/ui/feedback/progress.tsx +32 -0
- package/src/ui/feedback/sheet.tsx +148 -0
- package/src/ui/feedback/sonner.tsx +36 -0
- package/src/ui/forms/command.tsx +157 -0
- package/src/ui/forms/date-picker.tsx +73 -0
- package/src/ui/forms/date-range-picker.tsx +76 -0
- package/src/ui/forms/date-time-picker.tsx +109 -0
- package/src/ui/forms/editor/editor-menu-bar.tsx +394 -0
- package/src/ui/forms/editor/index.tsx +130 -0
- package/src/ui/forms/editor/multi-select-example.tsx +1234 -0
- package/src/ui/forms/emoji-picker.tsx +109 -0
- package/src/ui/forms/file-dropzone.tsx +169 -0
- package/src/ui/forms/file-thumbnail.tsx +29 -0
- package/src/ui/forms/index.tsx +201 -0
- package/src/ui/forms/input-file.tsx +99 -0
- package/src/ui/forms/input-group.tsx +46 -0
- package/src/ui/forms/input-otp.tsx +81 -0
- package/src/ui/forms/input-phone.tsx +172 -0
- package/src/ui/forms/input-spin.tsx +116 -0
- package/src/ui/forms/input-tags.tsx +219 -0
- package/src/ui/forms/input-time.tsx +42 -0
- package/src/ui/forms/multi-select.tsx +629 -0
- package/src/ui/forms/multiple-date-picker.tsx +74 -0
- package/src/ui/forms/radio-group.tsx +42 -0
- package/src/ui/forms/rating.tsx +158 -0
- package/src/ui/forms/time-picker.tsx +57 -0
- package/src/ui/index.tsx +17 -0
- package/src/ui/layout/animated-list.tsx +77 -0
- package/src/ui/layout/animated-sidebar.tsx +294 -0
- package/src/ui/layout/command-menu.tsx +355 -0
- package/src/ui/layout/customizer.tsx +324 -0
- package/src/ui/layout/footer.tsx +43 -0
- package/src/ui/layout/full-screen-toggle.tsx +52 -0
- package/src/ui/layout/header-breadcrumb.tsx +77 -0
- package/src/ui/layout/horizontal-layout-header.tsx +83 -0
- package/src/ui/layout/horizontal-layout.tsx +50 -0
- package/src/ui/layout/index.tsx +25 -0
- package/src/ui/layout/language-dropdown.tsx +103 -0
- package/src/ui/layout/logo.tsx +63 -0
- package/src/ui/layout/main-layout.tsx +57 -0
- package/src/ui/layout/mode-dropdown.tsx +58 -0
- package/src/ui/layout/notification-dropdown.tsx +127 -0
- package/src/ui/layout/page-tabs.tsx +306 -0
- package/src/ui/layout/route-cache.tsx +214 -0
- package/src/ui/layout/sidebar-group-icon-menu.tsx +195 -0
- package/src/ui/layout/sidebar.tsx +279 -0
- package/src/ui/layout/tab-content-cache.tsx +201 -0
- package/src/ui/layout/tab-navigation-provider.tsx +536 -0
- package/src/ui/layout/toggle-mobile-sidebar.tsx +33 -0
- package/src/ui/layout/top-bar-header-menubar.tsx +412 -0
- package/src/ui/layout/user-dropdown.tsx +188 -0
- package/src/ui/layout/vertical-layout-header.tsx +65 -0
- package/src/ui/layout/vertical-layout.tsx +47 -0
- package/src/ui/management/audit-log-page.tsx +209 -0
- package/src/ui/management/cache-management.tsx +349 -0
- package/src/ui/management/index.ts +3 -0
- package/src/ui/management/job-management.tsx +308 -0
- package/src/ui/pages/not-found.tsx +30 -0
- package/src/ui/primitives/badge.tsx +66 -0
- package/src/ui/primitives/breadcrumb.tsx +103 -0
- package/src/ui/primitives/button.tsx +129 -0
- package/src/ui/primitives/calendar.tsx +74 -0
- package/src/ui/primitives/card.tsx +86 -0
- package/src/ui/primitives/checkbox.tsx +31 -0
- package/src/ui/primitives/client.ts +30 -0
- package/src/ui/primitives/combobox.tsx +290 -0
- package/src/ui/primitives/dialog.tsx +121 -0
- package/src/ui/primitives/dropdown-menu.tsx +239 -0
- package/src/ui/primitives/dynamic-icon.tsx +24 -0
- package/src/ui/primitives/index.tsx +134 -0
- package/src/ui/primitives/input-number.tsx +131 -0
- package/src/ui/primitives/input.tsx +22 -0
- package/src/ui/primitives/keyboard.tsx +23 -0
- package/src/ui/primitives/label.tsx +24 -0
- package/src/ui/primitives/menubar.tsx +262 -0
- package/src/ui/primitives/navigation-menu.tsx +157 -0
- package/src/ui/primitives/pagination.tsx +118 -0
- package/src/ui/primitives/popover.tsx +56 -0
- package/src/ui/primitives/prefetch-link.tsx +60 -0
- package/src/ui/primitives/resizable.tsx +59 -0
- package/src/ui/primitives/scroll-area.tsx +63 -0
- package/src/ui/primitives/select.tsx +172 -0
- package/src/ui/primitives/separator.tsx +51 -0
- package/src/ui/primitives/sidebar.tsx +844 -0
- package/src/ui/primitives/slider.tsx +27 -0
- package/src/ui/primitives/status-badge.tsx +47 -0
- package/src/ui/primitives/sticky-layout.tsx +50 -0
- package/src/ui/primitives/switch.tsx +29 -0
- package/src/ui/primitives/table.tsx +116 -0
- package/src/ui/primitives/tabs.tsx +55 -0
- package/src/ui/primitives/toggle-group.tsx +70 -0
- package/src/ui/primitives/toggle.tsx +47 -0
- package/src/ui/primitives/tooltip.tsx +59 -0
- package/src/user/components/dangerous-zone.tsx +34 -0
- package/src/user/components/delete-account-form.tsx +40 -0
- package/src/user/components/index.ts +4 -0
- package/src/user/components/profile-info-form.tsx +390 -0
- package/src/user/components/profile-info.tsx +32 -0
- package/src/user/components/unified-profile-dialog.tsx +1019 -0
- package/src/user/components/user-stats.tsx +27 -0
- package/src/user/components/user-toolbar.tsx +137 -0
- package/src/user/components/users-card-view.tsx +253 -0
- package/src/user/index.ts +11 -0
- package/src/user/pages/user-list-page.tsx +234 -0
- package/src/user/pages/users-client-page.tsx +385 -0
- package/src/user/profile-page.tsx +19 -0
- package/src/user/schemas.ts +68 -0
- package/src/user/types.ts +34 -0
- package/src/user/user-service.ts +538 -0
- package/src/utils/index.ts +906 -0
- package/src/workflow/activity-timeline.tsx +412 -0
- package/src/workflow/approval-workflow.tsx +31 -0
- package/src/workflow/index.ts +2 -0
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
import { format } from "date-fns";
|
|
4
|
+
import { vi } from "date-fns/locale";
|
|
5
|
+
import { TrendingUp, Zap } from "lucide-react";
|
|
6
|
+
import { cn } from "../utils";
|
|
7
|
+
|
|
8
|
+
interface WelcomeCardProps {
|
|
9
|
+
userName?: string;
|
|
10
|
+
className?: string;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
function getGreeting(hour: number): { text: string; emoji: string } {
|
|
14
|
+
if (hour >= 5 && hour < 12) {
|
|
15
|
+
return { text: "Chào buổi sáng", emoji: "🌅" };
|
|
16
|
+
} else if (hour >= 12 && hour < 18) {
|
|
17
|
+
return { text: "Chào buổi chiều", emoji: "☀️" };
|
|
18
|
+
} else {
|
|
19
|
+
return { text: "Chào buổi tối", emoji: "🌙" };
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
export function WelcomeCard({ userName = "Bạn", className }: WelcomeCardProps) {
|
|
24
|
+
const now = new Date();
|
|
25
|
+
const hour = now.getHours();
|
|
26
|
+
const greeting = getGreeting(hour);
|
|
27
|
+
|
|
28
|
+
const formattedDate = format(now, "EEEE, dd/MM/yyyy", { locale: vi });
|
|
29
|
+
const formattedTime = format(now, "HH:mm");
|
|
30
|
+
|
|
31
|
+
return (
|
|
32
|
+
<div
|
|
33
|
+
className={cn(
|
|
34
|
+
"w-full bg-gradient-to-r from-blue-900 to-primary rounded-xl overflow-hidden shadow-lg group relative animate-in fade-in slide-in-from-bottom-4 duration-500",
|
|
35
|
+
className,
|
|
36
|
+
)}
|
|
37
|
+
>
|
|
38
|
+
{/* Background decoration */}
|
|
39
|
+
<div
|
|
40
|
+
className="absolute inset-0 opacity-20 pointer-events-none"
|
|
41
|
+
style={{
|
|
42
|
+
backgroundImage:
|
|
43
|
+
"radial-gradient(circle at 80% 20%, white 0%, transparent 40%), radial-gradient(circle at 10% 80%, white 0%, transparent 30%)",
|
|
44
|
+
}}
|
|
45
|
+
/>
|
|
46
|
+
|
|
47
|
+
<div className="flex flex-col md:flex-row items-center relative overflow-hidden">
|
|
48
|
+
{/* Content Left */}
|
|
49
|
+
<div className="p-8 flex-1 z-10 w-full">
|
|
50
|
+
<div className="flex items-center gap-2 mb-4">
|
|
51
|
+
<span className="bg-white/20 text-white text-xs font-bold px-2 py-1 rounded uppercase tracking-wider backdrop-blur-sm">
|
|
52
|
+
{formattedDate}
|
|
53
|
+
</span>
|
|
54
|
+
</div>
|
|
55
|
+
<h2 className="text-white text-2xl md:text-3xl font-bold mb-3">
|
|
56
|
+
{greeting.text}, {userName}
|
|
57
|
+
</h2>
|
|
58
|
+
<div className="text-blue-100 mb-6 max-w-lg">
|
|
59
|
+
Chúc bạn một ngày làm việc hiệu quả. Hệ thống đang hoạt động ổn
|
|
60
|
+
định.
|
|
61
|
+
</div>
|
|
62
|
+
<div className="flex gap-3">
|
|
63
|
+
<div className="bg-white/10 text-white px-5 py-2.5 rounded-lg text-sm font-bold shadow-sm backdrop-blur-sm border border-white/20 flex items-center gap-2">
|
|
64
|
+
<span className="w-2 h-2 rounded-full bg-green-400 animate-pulse" />
|
|
65
|
+
<span>Online • {formattedTime}</span>
|
|
66
|
+
</div>
|
|
67
|
+
</div>
|
|
68
|
+
</div>
|
|
69
|
+
|
|
70
|
+
{/* Visuals Right (Hidden on mobile) */}
|
|
71
|
+
<div className="hidden md:flex flex-1 justify-end items-end h-full p-8 z-10 self-stretch">
|
|
72
|
+
<div className="grid grid-cols-2 gap-3 opacity-90 transform translate-y-4">
|
|
73
|
+
<div className="bg-white/10 backdrop-blur-md border border-white/20 p-3 rounded-lg flex items-center gap-3">
|
|
74
|
+
<TrendingUp className="text-white w-6 h-6" />
|
|
75
|
+
<div>
|
|
76
|
+
<div className="h-1.5 w-12 bg-white/40 rounded mb-1" />
|
|
77
|
+
<div className="h-1.5 w-8 bg-white/20 rounded" />
|
|
78
|
+
</div>
|
|
79
|
+
</div>
|
|
80
|
+
<div className="bg-white/10 backdrop-blur-md border border-white/20 p-3 rounded-lg flex items-center gap-3">
|
|
81
|
+
<Zap className="text-white w-6 h-6" />
|
|
82
|
+
<div>
|
|
83
|
+
<div className="h-1.5 w-12 bg-white/40 rounded mb-1" />
|
|
84
|
+
<div className="h-1.5 w-8 bg-white/20 rounded" />
|
|
85
|
+
</div>
|
|
86
|
+
</div>
|
|
87
|
+
</div>
|
|
88
|
+
</div>
|
|
89
|
+
</div>
|
|
90
|
+
</div>
|
|
91
|
+
);
|
|
92
|
+
}
|
|
@@ -0,0 +1,258 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
import { useState, useCallback } from "react";
|
|
4
|
+
import {
|
|
5
|
+
DndContext,
|
|
6
|
+
closestCenter,
|
|
7
|
+
KeyboardSensor,
|
|
8
|
+
PointerSensor,
|
|
9
|
+
useSensor,
|
|
10
|
+
useSensors,
|
|
11
|
+
} from "@dnd-kit/core";
|
|
12
|
+
import type { DragEndEvent } from "@dnd-kit/core";
|
|
13
|
+
import {
|
|
14
|
+
arrayMove,
|
|
15
|
+
SortableContext,
|
|
16
|
+
sortableKeyboardCoordinates,
|
|
17
|
+
rectSortingStrategy,
|
|
18
|
+
} from "@dnd-kit/sortable";
|
|
19
|
+
import { motion, AnimatePresence } from "framer-motion";
|
|
20
|
+
import { Plus, Eye, EyeOff, RotateCcw } from "lucide-react";
|
|
21
|
+
import { cn } from "../utils";
|
|
22
|
+
import type { WidgetConfig } from "./types";
|
|
23
|
+
import { DEFAULT_WIDGETS, WIDGET_LABELS } from "./types";
|
|
24
|
+
import { RevenueWidget } from "./widgets/revenue-widget";
|
|
25
|
+
import { OrdersWidget } from "./widgets/orders-widget";
|
|
26
|
+
import { CustomersWidget } from "./widgets/customers-widget";
|
|
27
|
+
import { StockWidget } from "./widgets/stock-widget";
|
|
28
|
+
|
|
29
|
+
interface WidgetContainerProps {
|
|
30
|
+
widgets: WidgetConfig[];
|
|
31
|
+
onWidgetsChange: (widgets: WidgetConfig[]) => void;
|
|
32
|
+
widgetData?: {
|
|
33
|
+
revenue?: { total: number; change?: number };
|
|
34
|
+
orders?: {
|
|
35
|
+
total: number;
|
|
36
|
+
pending?: number;
|
|
37
|
+
completed?: number;
|
|
38
|
+
change?: number;
|
|
39
|
+
};
|
|
40
|
+
customers?: { total: number; newToday?: number; change?: number };
|
|
41
|
+
stock?: {
|
|
42
|
+
totalItems: number;
|
|
43
|
+
lowStockCount: number;
|
|
44
|
+
lowStockItems?: Array<{
|
|
45
|
+
id: string;
|
|
46
|
+
name: string;
|
|
47
|
+
quantity: number;
|
|
48
|
+
minStock: number;
|
|
49
|
+
}>;
|
|
50
|
+
};
|
|
51
|
+
};
|
|
52
|
+
loading?: boolean;
|
|
53
|
+
className?: string;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
export function WidgetContainer({
|
|
57
|
+
widgets,
|
|
58
|
+
onWidgetsChange,
|
|
59
|
+
widgetData,
|
|
60
|
+
loading = false,
|
|
61
|
+
className,
|
|
62
|
+
}: WidgetContainerProps) {
|
|
63
|
+
const [showHiddenWidgets, setShowHiddenWidgets] = useState(false);
|
|
64
|
+
|
|
65
|
+
const sensors = useSensors(
|
|
66
|
+
useSensor(PointerSensor, {
|
|
67
|
+
activationConstraint: {
|
|
68
|
+
distance: 8,
|
|
69
|
+
},
|
|
70
|
+
}),
|
|
71
|
+
useSensor(KeyboardSensor, {
|
|
72
|
+
coordinateGetter: sortableKeyboardCoordinates,
|
|
73
|
+
}),
|
|
74
|
+
);
|
|
75
|
+
|
|
76
|
+
const handleDragEnd = useCallback(
|
|
77
|
+
(event: DragEndEvent) => {
|
|
78
|
+
const { active, over } = event;
|
|
79
|
+
|
|
80
|
+
if (over && active.id !== over.id) {
|
|
81
|
+
const oldIndex = widgets.findIndex((w) => w.id === active.id);
|
|
82
|
+
const newIndex = widgets.findIndex((w) => w.id === over.id);
|
|
83
|
+
|
|
84
|
+
const newWidgets = arrayMove(widgets, oldIndex, newIndex).map(
|
|
85
|
+
(widget, index) => ({
|
|
86
|
+
...widget,
|
|
87
|
+
position: index,
|
|
88
|
+
}),
|
|
89
|
+
);
|
|
90
|
+
|
|
91
|
+
onWidgetsChange(newWidgets);
|
|
92
|
+
}
|
|
93
|
+
},
|
|
94
|
+
[widgets, onWidgetsChange],
|
|
95
|
+
);
|
|
96
|
+
|
|
97
|
+
const handleToggleVisibility = useCallback(
|
|
98
|
+
(widgetId: string) => {
|
|
99
|
+
const newWidgets = widgets.map((widget) =>
|
|
100
|
+
widget.id === widgetId
|
|
101
|
+
? { ...widget, visible: !widget.visible }
|
|
102
|
+
: widget,
|
|
103
|
+
);
|
|
104
|
+
onWidgetsChange(newWidgets);
|
|
105
|
+
},
|
|
106
|
+
[widgets, onWidgetsChange],
|
|
107
|
+
);
|
|
108
|
+
|
|
109
|
+
const handleRemoveWidget = useCallback(
|
|
110
|
+
(widgetId: string) => {
|
|
111
|
+
const newWidgets = widgets.map((widget) =>
|
|
112
|
+
widget.id === widgetId ? { ...widget, visible: false } : widget,
|
|
113
|
+
);
|
|
114
|
+
onWidgetsChange(newWidgets);
|
|
115
|
+
},
|
|
116
|
+
[widgets, onWidgetsChange],
|
|
117
|
+
);
|
|
118
|
+
|
|
119
|
+
const handleResetWidgets = useCallback(() => {
|
|
120
|
+
onWidgetsChange(DEFAULT_WIDGETS);
|
|
121
|
+
}, [onWidgetsChange]);
|
|
122
|
+
|
|
123
|
+
const visibleWidgets = widgets
|
|
124
|
+
.filter((w) => w.visible)
|
|
125
|
+
.sort((a, b) => a.position - b.position);
|
|
126
|
+
|
|
127
|
+
const hiddenWidgets = widgets.filter((w) => !w.visible);
|
|
128
|
+
|
|
129
|
+
const renderWidget = (config: WidgetConfig) => {
|
|
130
|
+
const commonProps = {
|
|
131
|
+
config,
|
|
132
|
+
loading,
|
|
133
|
+
onRemove: () => handleRemoveWidget(config.id),
|
|
134
|
+
onToggleVisibility: () => handleToggleVisibility(config.id),
|
|
135
|
+
};
|
|
136
|
+
|
|
137
|
+
switch (config.type) {
|
|
138
|
+
case "revenue":
|
|
139
|
+
return <RevenueWidget {...commonProps} data={widgetData?.revenue} />;
|
|
140
|
+
case "orders":
|
|
141
|
+
return <OrdersWidget {...commonProps} data={widgetData?.orders} />;
|
|
142
|
+
case "customers":
|
|
143
|
+
return (
|
|
144
|
+
<CustomersWidget {...commonProps} data={widgetData?.customers} />
|
|
145
|
+
);
|
|
146
|
+
case "stock":
|
|
147
|
+
return <StockWidget {...commonProps} data={widgetData?.stock} />;
|
|
148
|
+
default:
|
|
149
|
+
return null;
|
|
150
|
+
}
|
|
151
|
+
};
|
|
152
|
+
|
|
153
|
+
return (
|
|
154
|
+
<div className={cn("space-y-4", className)}>
|
|
155
|
+
{/* Header */}
|
|
156
|
+
<div className="flex items-center justify-between">
|
|
157
|
+
<h2 className="text-xl font-semibold text-foreground">
|
|
158
|
+
Widget Dashboard
|
|
159
|
+
</h2>
|
|
160
|
+
<div className="flex items-center gap-2">
|
|
161
|
+
{hiddenWidgets.length > 0 && (
|
|
162
|
+
<button
|
|
163
|
+
onClick={() => setShowHiddenWidgets(!showHiddenWidgets)}
|
|
164
|
+
className="inline-flex items-center gap-1.5 rounded-lg px-3 py-1.5 text-sm font-medium text-muted-foreground hover:bg-muted hover:text-foreground transition-colors"
|
|
165
|
+
>
|
|
166
|
+
{showHiddenWidgets ? (
|
|
167
|
+
<EyeOff className="h-4 w-4" />
|
|
168
|
+
) : (
|
|
169
|
+
<Eye className="h-4 w-4" />
|
|
170
|
+
)}
|
|
171
|
+
<span>{hiddenWidgets.length} ẩn</span>
|
|
172
|
+
</button>
|
|
173
|
+
)}
|
|
174
|
+
<button
|
|
175
|
+
onClick={handleResetWidgets}
|
|
176
|
+
className="inline-flex items-center gap-1.5 rounded-lg px-3 py-1.5 text-sm font-medium text-muted-foreground hover:bg-muted hover:text-foreground transition-colors"
|
|
177
|
+
title="Khôi phục mặc định"
|
|
178
|
+
>
|
|
179
|
+
<RotateCcw className="h-4 w-4" />
|
|
180
|
+
</button>
|
|
181
|
+
</div>
|
|
182
|
+
</div>
|
|
183
|
+
|
|
184
|
+
{/* Hidden widgets panel */}
|
|
185
|
+
<AnimatePresence>
|
|
186
|
+
{showHiddenWidgets && hiddenWidgets.length > 0 && (
|
|
187
|
+
<motion.div
|
|
188
|
+
initial={{ opacity: 0, height: 0 }}
|
|
189
|
+
animate={{ opacity: 1, height: "auto" }}
|
|
190
|
+
exit={{ opacity: 0, height: 0 }}
|
|
191
|
+
// @ts-ignore className is valid for motion.div
|
|
192
|
+
className="overflow-hidden"
|
|
193
|
+
>
|
|
194
|
+
<div className="rounded-lg border border-dashed border-border bg-muted/30 p-4">
|
|
195
|
+
<p className="mb-3 text-sm font-medium text-muted-foreground">
|
|
196
|
+
Widget đã ẩn - Click để hiển thị lại
|
|
197
|
+
</p>
|
|
198
|
+
<div className="flex flex-wrap gap-2">
|
|
199
|
+
{hiddenWidgets.map((widget) => (
|
|
200
|
+
<button
|
|
201
|
+
key={widget.id}
|
|
202
|
+
onClick={() => handleToggleVisibility(widget.id)}
|
|
203
|
+
className="inline-flex items-center gap-1.5 rounded-lg bg-background px-3 py-2 text-sm font-medium shadow-sm border border-border hover:bg-muted transition-colors"
|
|
204
|
+
>
|
|
205
|
+
<Plus className="h-4 w-4" />
|
|
206
|
+
{WIDGET_LABELS[widget.type]}
|
|
207
|
+
</button>
|
|
208
|
+
))}
|
|
209
|
+
</div>
|
|
210
|
+
</div>
|
|
211
|
+
</motion.div>
|
|
212
|
+
)}
|
|
213
|
+
</AnimatePresence>
|
|
214
|
+
|
|
215
|
+
{/* Sortable widget grid */}
|
|
216
|
+
<DndContext
|
|
217
|
+
sensors={sensors}
|
|
218
|
+
collisionDetection={closestCenter}
|
|
219
|
+
onDragEnd={handleDragEnd}
|
|
220
|
+
>
|
|
221
|
+
<SortableContext
|
|
222
|
+
items={visibleWidgets.map((w) => w.id)}
|
|
223
|
+
strategy={rectSortingStrategy}
|
|
224
|
+
>
|
|
225
|
+
<div className="grid gap-4 sm:grid-cols-2 lg:grid-cols-4">
|
|
226
|
+
<AnimatePresence mode="popLayout">
|
|
227
|
+
{visibleWidgets.map((widget) => (
|
|
228
|
+
<div key={widget.id}>{renderWidget(widget)}</div>
|
|
229
|
+
))}
|
|
230
|
+
</AnimatePresence>
|
|
231
|
+
</div>
|
|
232
|
+
</SortableContext>
|
|
233
|
+
</DndContext>
|
|
234
|
+
|
|
235
|
+
{/* Empty state */}
|
|
236
|
+
{visibleWidgets.length === 0 && (
|
|
237
|
+
<div className="flex flex-col items-center justify-center rounded-lg border border-dashed border-border py-12 text-center">
|
|
238
|
+
<div className="rounded-full bg-muted p-3 mb-4">
|
|
239
|
+
<Plus className="h-6 w-6 text-muted-foreground" />
|
|
240
|
+
</div>
|
|
241
|
+
<h3 className="font-medium text-foreground mb-1">
|
|
242
|
+
Chưa có widget nào
|
|
243
|
+
</h3>
|
|
244
|
+
<p className="text-sm text-muted-foreground mb-4">
|
|
245
|
+
Thêm widget để theo dõi các chỉ số quan trọng
|
|
246
|
+
</p>
|
|
247
|
+
<button
|
|
248
|
+
onClick={handleResetWidgets}
|
|
249
|
+
className="inline-flex items-center gap-2 rounded-lg bg-primary px-4 py-2 text-sm font-medium text-primary-foreground hover:bg-primary/90 transition-colors"
|
|
250
|
+
>
|
|
251
|
+
<RotateCcw className="h-4 w-4" />
|
|
252
|
+
Khôi phục mặc định
|
|
253
|
+
</button>
|
|
254
|
+
</div>
|
|
255
|
+
)}
|
|
256
|
+
</div>
|
|
257
|
+
);
|
|
258
|
+
}
|
|
@@ -0,0 +1,200 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
import { motion } from "framer-motion";
|
|
4
|
+
import { useSortable } from "@dnd-kit/sortable";
|
|
5
|
+
import { CSS } from "@dnd-kit/utilities";
|
|
6
|
+
import { GripVertical, Settings, X, Eye, EyeOff } from "lucide-react";
|
|
7
|
+
import { cn } from "../../utils";
|
|
8
|
+
import type { WidgetConfig, WidgetSize } from "../types";
|
|
9
|
+
|
|
10
|
+
interface BaseWidgetProps {
|
|
11
|
+
config: WidgetConfig;
|
|
12
|
+
title: string;
|
|
13
|
+
children: React.ReactNode;
|
|
14
|
+
loading?: boolean;
|
|
15
|
+
error?: string;
|
|
16
|
+
onRemove?: () => void;
|
|
17
|
+
onToggleVisibility?: () => void;
|
|
18
|
+
onSettings?: () => void;
|
|
19
|
+
className?: string;
|
|
20
|
+
isDragging?: boolean;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
const sizeClasses: Record<WidgetSize, string> = {
|
|
24
|
+
small: "col-span-1",
|
|
25
|
+
medium: "col-span-1 md:col-span-1",
|
|
26
|
+
large: "col-span-1 md:col-span-2",
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
export function BaseWidget({
|
|
30
|
+
config,
|
|
31
|
+
title,
|
|
32
|
+
children,
|
|
33
|
+
loading = false,
|
|
34
|
+
error,
|
|
35
|
+
onRemove,
|
|
36
|
+
onToggleVisibility,
|
|
37
|
+
onSettings,
|
|
38
|
+
className,
|
|
39
|
+
isDragging = false,
|
|
40
|
+
}: BaseWidgetProps) {
|
|
41
|
+
const {
|
|
42
|
+
attributes,
|
|
43
|
+
listeners,
|
|
44
|
+
setNodeRef,
|
|
45
|
+
transform,
|
|
46
|
+
transition,
|
|
47
|
+
isDragging: isSortableDragging,
|
|
48
|
+
} = useSortable({ id: config.id });
|
|
49
|
+
|
|
50
|
+
const style = {
|
|
51
|
+
transform: CSS.Transform.toString(transform),
|
|
52
|
+
transition,
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
const isCurrentlyDragging = isDragging || isSortableDragging;
|
|
56
|
+
|
|
57
|
+
if (!config.visible) {
|
|
58
|
+
return null;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
return (
|
|
62
|
+
<motion.div
|
|
63
|
+
ref={setNodeRef}
|
|
64
|
+
style={style}
|
|
65
|
+
initial={{ opacity: 0, scale: 0.95 }}
|
|
66
|
+
animate={{ opacity: 1, scale: 1 }}
|
|
67
|
+
exit={{ opacity: 0, scale: 0.95 }}
|
|
68
|
+
// @ts-ignore className is valid for motion.div
|
|
69
|
+
className={cn(
|
|
70
|
+
"group relative flex flex-col justify-between gap-3 rounded-xl border border-slate-200 bg-white p-5 shadow-sm transition-all hover:border-primary/30 hover:shadow-md dark:border-slate-700 dark:bg-slate-800",
|
|
71
|
+
sizeClasses[config.size],
|
|
72
|
+
isCurrentlyDragging && "z-50 shadow-lg ring-2 ring-primary/20",
|
|
73
|
+
className,
|
|
74
|
+
)}
|
|
75
|
+
>
|
|
76
|
+
{/* Header */}
|
|
77
|
+
<div className="flex items-center justify-between mb-1">
|
|
78
|
+
<div className="flex items-center gap-2">
|
|
79
|
+
{/* Drag handle */}
|
|
80
|
+
<button
|
|
81
|
+
{...attributes}
|
|
82
|
+
{...listeners}
|
|
83
|
+
className="cursor-grab touch-none rounded text-slate-400 hover:text-slate-600 dark:hover:text-slate-300 active:cursor-grabbing opacity-0 group-hover:opacity-100 transition-opacity"
|
|
84
|
+
aria-label="Drag to reorder"
|
|
85
|
+
>
|
|
86
|
+
<GripVertical className="h-4 w-4" />
|
|
87
|
+
</button>
|
|
88
|
+
<h4 className="flex items-center gap-2 text-sm font-semibold text-slate-900 dark:text-white">
|
|
89
|
+
{title}
|
|
90
|
+
</h4>
|
|
91
|
+
</div>
|
|
92
|
+
|
|
93
|
+
{/* Actions */}
|
|
94
|
+
<div className="flex items-center gap-1 opacity-0 transition-opacity group-hover:opacity-100">
|
|
95
|
+
{onToggleVisibility && (
|
|
96
|
+
<button
|
|
97
|
+
onClick={onToggleVisibility}
|
|
98
|
+
className="rounded p-1 text-slate-400 hover:bg-slate-100 hover:text-slate-700 dark:hover:bg-slate-700 dark:hover:text-slate-300"
|
|
99
|
+
aria-label={config.visible ? "Hide widget" : "Show widget"}
|
|
100
|
+
>
|
|
101
|
+
{config.visible ? (
|
|
102
|
+
<Eye className="h-4 w-4" />
|
|
103
|
+
) : (
|
|
104
|
+
<EyeOff className="h-4 w-4" />
|
|
105
|
+
)}
|
|
106
|
+
</button>
|
|
107
|
+
)}
|
|
108
|
+
{onSettings && (
|
|
109
|
+
<button
|
|
110
|
+
onClick={onSettings}
|
|
111
|
+
className="rounded p-1 text-slate-400 hover:bg-slate-100 hover:text-slate-700 dark:hover:bg-slate-700 dark:hover:text-slate-300"
|
|
112
|
+
aria-label="Widget settings"
|
|
113
|
+
>
|
|
114
|
+
<Settings className="h-4 w-4" />
|
|
115
|
+
</button>
|
|
116
|
+
)}
|
|
117
|
+
{onRemove && (
|
|
118
|
+
<button
|
|
119
|
+
onClick={onRemove}
|
|
120
|
+
className="rounded p-1 text-slate-400 hover:bg-red-50 hover:text-red-500 dark:hover:bg-red-900/20"
|
|
121
|
+
aria-label="Remove widget"
|
|
122
|
+
>
|
|
123
|
+
<X className="h-4 w-4" />
|
|
124
|
+
</button>
|
|
125
|
+
)}
|
|
126
|
+
</div>
|
|
127
|
+
</div>
|
|
128
|
+
|
|
129
|
+
{/* Content */}
|
|
130
|
+
<div className="flex-1">
|
|
131
|
+
{loading ? (
|
|
132
|
+
<div className="flex items-center justify-center py-4">
|
|
133
|
+
<div className="h-6 w-6 animate-spin rounded-full border-2 border-primary border-t-transparent" />
|
|
134
|
+
</div>
|
|
135
|
+
) : error ? (
|
|
136
|
+
<div className="flex items-center justify-center py-4 text-xs text-red-500">
|
|
137
|
+
{error}
|
|
138
|
+
</div>
|
|
139
|
+
) : (
|
|
140
|
+
children
|
|
141
|
+
)}
|
|
142
|
+
</div>
|
|
143
|
+
</motion.div>
|
|
144
|
+
);
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
// Stat display component for widgets
|
|
148
|
+
interface WidgetStatProps {
|
|
149
|
+
value: string | number;
|
|
150
|
+
label?: string;
|
|
151
|
+
change?: {
|
|
152
|
+
value: number;
|
|
153
|
+
type: "increase" | "decrease";
|
|
154
|
+
period?: string;
|
|
155
|
+
};
|
|
156
|
+
valueClassName?: string;
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
export function WidgetStat({
|
|
160
|
+
value,
|
|
161
|
+
label,
|
|
162
|
+
change,
|
|
163
|
+
valueClassName,
|
|
164
|
+
}: WidgetStatProps) {
|
|
165
|
+
return (
|
|
166
|
+
<div className="space-y-2">
|
|
167
|
+
<div className={cn("text-3xl font-bold text-foreground", valueClassName)}>
|
|
168
|
+
{value}
|
|
169
|
+
</div>
|
|
170
|
+
{label && <p className="text-sm text-muted-foreground">{label}</p>}
|
|
171
|
+
{change && (
|
|
172
|
+
<div
|
|
173
|
+
className={cn(
|
|
174
|
+
"inline-flex items-center gap-1 rounded-full px-2 py-0.5 text-xs font-medium",
|
|
175
|
+
change.type === "increase"
|
|
176
|
+
? "bg-green-100 text-green-700 dark:bg-green-900/30 dark:text-green-400"
|
|
177
|
+
: "bg-red-100 text-red-700 dark:bg-red-900/30 dark:text-red-400",
|
|
178
|
+
)}
|
|
179
|
+
>
|
|
180
|
+
{change.type === "increase" ? (
|
|
181
|
+
<svg className="h-3 w-3" viewBox="0 0 12 12" fill="currentColor">
|
|
182
|
+
<path d="M6 2L10 7H2L6 2Z" />
|
|
183
|
+
</svg>
|
|
184
|
+
) : (
|
|
185
|
+
<svg className="h-3 w-3" viewBox="0 0 12 12" fill="currentColor">
|
|
186
|
+
<path d="M6 10L2 5H10L6 10Z" />
|
|
187
|
+
</svg>
|
|
188
|
+
)}
|
|
189
|
+
<span>
|
|
190
|
+
{change.value > 0 ? "+" : ""}
|
|
191
|
+
{change.value}%
|
|
192
|
+
</span>
|
|
193
|
+
{change.period && (
|
|
194
|
+
<span className="text-muted-foreground">vs {change.period}</span>
|
|
195
|
+
)}
|
|
196
|
+
</div>
|
|
197
|
+
)}
|
|
198
|
+
</div>
|
|
199
|
+
);
|
|
200
|
+
}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
import { Users } from "lucide-react";
|
|
4
|
+
import { BaseWidget, WidgetStat } from "./base-widget";
|
|
5
|
+
import type { WidgetConfig } from "../types";
|
|
6
|
+
|
|
7
|
+
interface CustomersWidgetProps {
|
|
8
|
+
config: WidgetConfig;
|
|
9
|
+
data?: {
|
|
10
|
+
total: number;
|
|
11
|
+
newToday?: number;
|
|
12
|
+
change?: number;
|
|
13
|
+
changePeriod?: string;
|
|
14
|
+
};
|
|
15
|
+
loading?: boolean;
|
|
16
|
+
error?: string;
|
|
17
|
+
onRemove?: () => void;
|
|
18
|
+
onToggleVisibility?: () => void;
|
|
19
|
+
onSettings?: () => void;
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export function CustomersWidget({
|
|
23
|
+
config,
|
|
24
|
+
data,
|
|
25
|
+
loading,
|
|
26
|
+
error,
|
|
27
|
+
onRemove,
|
|
28
|
+
onToggleVisibility,
|
|
29
|
+
onSettings,
|
|
30
|
+
}: CustomersWidgetProps) {
|
|
31
|
+
return (
|
|
32
|
+
<BaseWidget
|
|
33
|
+
config={config}
|
|
34
|
+
title="Khách hàng"
|
|
35
|
+
loading={loading}
|
|
36
|
+
error={error}
|
|
37
|
+
onRemove={onRemove}
|
|
38
|
+
onToggleVisibility={onToggleVisibility}
|
|
39
|
+
onSettings={onSettings}
|
|
40
|
+
>
|
|
41
|
+
<div className="flex items-start justify-between">
|
|
42
|
+
<div className="space-y-3">
|
|
43
|
+
<WidgetStat
|
|
44
|
+
value={data?.total ?? 0}
|
|
45
|
+
label="Tổng khách hàng"
|
|
46
|
+
change={
|
|
47
|
+
data?.change !== undefined
|
|
48
|
+
? {
|
|
49
|
+
value: data.change,
|
|
50
|
+
type: data.change >= 0 ? "increase" : "decrease",
|
|
51
|
+
period: data.changePeriod || "tháng trước",
|
|
52
|
+
}
|
|
53
|
+
: undefined
|
|
54
|
+
}
|
|
55
|
+
valueClassName="text-purple-600 dark:text-purple-400"
|
|
56
|
+
/>
|
|
57
|
+
|
|
58
|
+
{data?.newToday !== undefined && data.newToday > 0 && (
|
|
59
|
+
<div className="text-sm">
|
|
60
|
+
<span className="text-muted-foreground">Khách mới hôm nay: </span>
|
|
61
|
+
<span className="font-medium text-green-600 dark:text-green-400">
|
|
62
|
+
+{data.newToday}
|
|
63
|
+
</span>
|
|
64
|
+
</div>
|
|
65
|
+
)}
|
|
66
|
+
</div>
|
|
67
|
+
|
|
68
|
+
<div className="rounded-full bg-purple-100 p-3 dark:bg-purple-900/30">
|
|
69
|
+
<Users className="h-6 w-6 text-purple-600 dark:text-purple-400" />
|
|
70
|
+
</div>
|
|
71
|
+
</div>
|
|
72
|
+
</BaseWidget>
|
|
73
|
+
);
|
|
74
|
+
}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
// Widget exports
|
|
2
|
+
export { BaseWidget, WidgetStat } from "./base-widget";
|
|
3
|
+
export { RevenueWidget } from "./revenue-widget";
|
|
4
|
+
export { OrdersWidget } from "./orders-widget";
|
|
5
|
+
export { CustomersWidget } from "./customers-widget";
|
|
6
|
+
export { StockWidget } from "./stock-widget";
|