Merge pull request 'dev' (#1) from dev into main
Reviewed-on: https://gits.altru.id/apksel-dev/apskel-pos-flutter/pulls/1
@ -23,7 +23,7 @@ if (flutterVersionName == null) {
|
||||
}
|
||||
|
||||
android {
|
||||
namespace "com.example.enaklo_pos"
|
||||
namespace "com.appscale.pos"
|
||||
compileSdkVersion 35
|
||||
ndkVersion flutter.ndkVersion
|
||||
|
||||
@ -42,7 +42,7 @@ android {
|
||||
|
||||
defaultConfig {
|
||||
// TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html).
|
||||
applicationId "com.example.enaklo_pos"
|
||||
applicationId "com.appscale.pos"
|
||||
// You can update the following values to match your application needs.
|
||||
// For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-gradle-build-configuration.
|
||||
minSdkVersion 21
|
||||
|
||||
@ -11,7 +11,7 @@
|
||||
<!-- Izin khusus untuk akses foto (media images) di Android 33 ke atas -->
|
||||
<uses-permission android:name="android.permission.READ_MEDIA_IMAGES" />
|
||||
<application
|
||||
android:label="EnakloPOS"
|
||||
android:label="ApskelPOS"
|
||||
android:name="${applicationName}"
|
||||
android:icon="@mipmap/launcher_icon">
|
||||
<activity
|
||||
|
||||
@ -1,4 +1,4 @@
|
||||
package com.example.enaklo_pos
|
||||
package com.appscale.pos
|
||||
|
||||
import io.flutter.embedding.android.FlutterActivity
|
||||
|
||||
|
||||
|
Before Width: | Height: | Size: 6.8 KiB After Width: | Height: | Size: 15 KiB |
|
Before Width: | Height: | Size: 4.9 KiB After Width: | Height: | Size: 9.4 KiB |
|
Before Width: | Height: | Size: 8.6 KiB After Width: | Height: | Size: 22 KiB |
|
Before Width: | Height: | Size: 11 KiB After Width: | Height: | Size: 38 KiB |
|
Before Width: | Height: | Size: 13 KiB After Width: | Height: | Size: 57 KiB |
@ -1,5 +1,9 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<background android:drawable="@color/ic_launcher_background"/>
|
||||
<foreground android:drawable="@drawable/ic_launcher_foreground"/>
|
||||
<foreground>
|
||||
<inset
|
||||
android:drawable="@drawable/ic_launcher_foreground"
|
||||
android:inset="16%" />
|
||||
</foreground>
|
||||
</adaptive-icon>
|
||||
|
||||
|
Before Width: | Height: | Size: 3.4 KiB After Width: | Height: | Size: 5.6 KiB |
|
Before Width: | Height: | Size: 2.3 KiB After Width: | Height: | Size: 3.1 KiB |
|
Before Width: | Height: | Size: 4.5 KiB After Width: | Height: | Size: 8.3 KiB |
|
Before Width: | Height: | Size: 6.4 KiB After Width: | Height: | Size: 13 KiB |
|
Before Width: | Height: | Size: 7.7 KiB After Width: | Height: | Size: 19 KiB |
@ -1,4 +1,4 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<color name="ic_launcher_background">#FFFFFF</color>
|
||||
<color name="ic_launcher_background">#ffffff</color>
|
||||
</resources>
|
||||
BIN
assets/fonts/quicksand/Quicksand-Bold.ttf
Normal file
BIN
assets/fonts/quicksand/Quicksand-Light.ttf
Normal file
BIN
assets/fonts/quicksand/Quicksand-Medium.ttf
Normal file
BIN
assets/fonts/quicksand/Quicksand-Regular.ttf
Normal file
BIN
assets/fonts/quicksand/Quicksand-SemiBold.ttf
Normal file
10
assets/icons/people.svg
Normal file
@ -0,0 +1,10 @@
|
||||
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<g clip-path="url(#clip0_17_4760)">
|
||||
<path d="M9 13.75C6.66 13.75 2 14.92 2 17.25V19H16V17.25C16 14.92 11.34 13.75 9 13.75ZM4.34 17C5.18 16.42 7.21 15.75 9 15.75C10.79 15.75 12.82 16.42 13.66 17H4.34ZM9 12C10.93 12 12.5 10.43 12.5 8.5C12.5 6.57 10.93 5 9 5C7.07 5 5.5 6.57 5.5 8.5C5.5 10.43 7.07 12 9 12ZM9 7C9.83 7 10.5 7.67 10.5 8.5C10.5 9.33 9.83 10 9 10C8.17 10 7.5 9.33 7.5 8.5C7.5 7.67 8.17 7 9 7ZM16.04 13.81C17.2 14.65 18 15.77 18 17.25V19H22V17.25C22 15.23 18.5 14.08 16.04 13.81ZM15 12C16.93 12 18.5 10.43 18.5 8.5C18.5 6.57 16.93 5 15 5C14.46 5 13.96 5.13 13.5 5.35C14.13 6.24 14.5 7.33 14.5 8.5C14.5 9.67 14.13 10.76 13.5 11.65C13.96 11.87 14.46 12 15 12Z" fill="white"/>
|
||||
</g>
|
||||
<defs>
|
||||
<clipPath id="clip0_17_4760">
|
||||
<rect width="24" height="24" fill="white"/>
|
||||
</clipPath>
|
||||
</defs>
|
||||
</svg>
|
||||
|
After Width: | Height: | Size: 892 B |
BIN
assets/images/gojek.png
Normal file
|
After Width: | Height: | Size: 81 KiB |
BIN
assets/images/grab.png
Normal file
|
After Width: | Height: | Size: 135 KiB |
|
Before Width: | Height: | Size: 19 KiB After Width: | Height: | Size: 79 KiB |
@ -5,7 +5,7 @@ import 'lib/core/utils/app_icon_generator.dart';
|
||||
void main() async {
|
||||
WidgetsFlutterBinding.ensureInitialized();
|
||||
|
||||
print('Generating EnakloPOS app icon...');
|
||||
print('Generating ApskelPOS app icon...');
|
||||
|
||||
try {
|
||||
final iconData = await AppIconGenerator.generateAppIcon();
|
||||
@ -17,17 +17,16 @@ void main() async {
|
||||
}
|
||||
|
||||
// Write the generated icon to file
|
||||
final iconFile = File('assets/logo/logo_app_icon.png');
|
||||
final iconFile = File('assets/logo/ic_launcher.png');
|
||||
await iconFile.writeAsBytes(iconData);
|
||||
|
||||
print('✅ App icon generated successfully at: assets/logo/logo_app_icon.png');
|
||||
print('✅ App icon generated successfully at: assets/logo/ic_launcher.png');
|
||||
print('📱 The icon features:');
|
||||
print(' - White background for visibility');
|
||||
print(' - Blue circular background');
|
||||
print(' - Gift box with "e" inside');
|
||||
print(' - "ENAKLO" and "POS" text');
|
||||
print(' - 1024x1024 resolution for high quality');
|
||||
|
||||
} catch (e) {
|
||||
print('❌ Error generating app icon: $e');
|
||||
}
|
||||
|
||||
@ -1,122 +1 @@
|
||||
{
|
||||
"images" : [
|
||||
{
|
||||
"size" : "20x20",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-20x20@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "20x20",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-20x20@3x.png",
|
||||
"scale" : "3x"
|
||||
},
|
||||
{
|
||||
"size" : "29x29",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-29x29@1x.png",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"size" : "29x29",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-29x29@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "29x29",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-29x29@3x.png",
|
||||
"scale" : "3x"
|
||||
},
|
||||
{
|
||||
"size" : "40x40",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-40x40@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "40x40",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-40x40@3x.png",
|
||||
"scale" : "3x"
|
||||
},
|
||||
{
|
||||
"size" : "60x60",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-60x60@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "60x60",
|
||||
"idiom" : "iphone",
|
||||
"filename" : "Icon-App-60x60@3x.png",
|
||||
"scale" : "3x"
|
||||
},
|
||||
{
|
||||
"size" : "20x20",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-20x20@1x.png",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"size" : "20x20",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-20x20@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "29x29",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-29x29@1x.png",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"size" : "29x29",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-29x29@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "40x40",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-40x40@1x.png",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"size" : "40x40",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-40x40@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "76x76",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-76x76@1x.png",
|
||||
"scale" : "1x"
|
||||
},
|
||||
{
|
||||
"size" : "76x76",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-76x76@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "83.5x83.5",
|
||||
"idiom" : "ipad",
|
||||
"filename" : "Icon-App-83.5x83.5@2x.png",
|
||||
"scale" : "2x"
|
||||
},
|
||||
{
|
||||
"size" : "1024x1024",
|
||||
"idiom" : "ios-marketing",
|
||||
"filename" : "Icon-App-1024x1024@1x.png",
|
||||
"scale" : "1x"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"version" : 1,
|
||||
"author" : "xcode"
|
||||
}
|
||||
}
|
||||
{"images":[{"size":"20x20","idiom":"iphone","filename":"Icon-App-20x20@2x.png","scale":"2x"},{"size":"20x20","idiom":"iphone","filename":"Icon-App-20x20@3x.png","scale":"3x"},{"size":"29x29","idiom":"iphone","filename":"Icon-App-29x29@1x.png","scale":"1x"},{"size":"29x29","idiom":"iphone","filename":"Icon-App-29x29@2x.png","scale":"2x"},{"size":"29x29","idiom":"iphone","filename":"Icon-App-29x29@3x.png","scale":"3x"},{"size":"40x40","idiom":"iphone","filename":"Icon-App-40x40@2x.png","scale":"2x"},{"size":"40x40","idiom":"iphone","filename":"Icon-App-40x40@3x.png","scale":"3x"},{"size":"57x57","idiom":"iphone","filename":"Icon-App-57x57@1x.png","scale":"1x"},{"size":"57x57","idiom":"iphone","filename":"Icon-App-57x57@2x.png","scale":"2x"},{"size":"60x60","idiom":"iphone","filename":"Icon-App-60x60@2x.png","scale":"2x"},{"size":"60x60","idiom":"iphone","filename":"Icon-App-60x60@3x.png","scale":"3x"},{"size":"20x20","idiom":"ipad","filename":"Icon-App-20x20@1x.png","scale":"1x"},{"size":"20x20","idiom":"ipad","filename":"Icon-App-20x20@2x.png","scale":"2x"},{"size":"29x29","idiom":"ipad","filename":"Icon-App-29x29@1x.png","scale":"1x"},{"size":"29x29","idiom":"ipad","filename":"Icon-App-29x29@2x.png","scale":"2x"},{"size":"40x40","idiom":"ipad","filename":"Icon-App-40x40@1x.png","scale":"1x"},{"size":"40x40","idiom":"ipad","filename":"Icon-App-40x40@2x.png","scale":"2x"},{"size":"50x50","idiom":"ipad","filename":"Icon-App-50x50@1x.png","scale":"1x"},{"size":"50x50","idiom":"ipad","filename":"Icon-App-50x50@2x.png","scale":"2x"},{"size":"72x72","idiom":"ipad","filename":"Icon-App-72x72@1x.png","scale":"1x"},{"size":"72x72","idiom":"ipad","filename":"Icon-App-72x72@2x.png","scale":"2x"},{"size":"76x76","idiom":"ipad","filename":"Icon-App-76x76@1x.png","scale":"1x"},{"size":"76x76","idiom":"ipad","filename":"Icon-App-76x76@2x.png","scale":"2x"},{"size":"83.5x83.5","idiom":"ipad","filename":"Icon-App-83.5x83.5@2x.png","scale":"2x"},{"size":"1024x1024","idiom":"ios-marketing","filename":"Icon-App-1024x1024@1x.png","scale":"1x"}],"info":{"version":1,"author":"xcode"}}
|
||||
|
Before Width: | Height: | Size: 41 KiB After Width: | Height: | Size: 150 KiB |
|
Before Width: | Height: | Size: 664 B After Width: | Height: | Size: 666 B |
|
Before Width: | Height: | Size: 1.5 KiB After Width: | Height: | Size: 2.2 KiB |
|
Before Width: | Height: | Size: 2.2 KiB After Width: | Height: | Size: 3.8 KiB |
|
Before Width: | Height: | Size: 984 B After Width: | Height: | Size: 1.3 KiB |
|
Before Width: | Height: | Size: 2.1 KiB After Width: | Height: | Size: 3.7 KiB |
|
Before Width: | Height: | Size: 2.9 KiB After Width: | Height: | Size: 6.4 KiB |
|
Before Width: | Height: | Size: 1.5 KiB After Width: | Height: | Size: 2.2 KiB |
|
Before Width: | Height: | Size: 2.7 KiB After Width: | Height: | Size: 5.8 KiB |
|
Before Width: | Height: | Size: 3.6 KiB After Width: | Height: | Size: 9.8 KiB |
|
Before Width: | Height: | Size: 1.8 KiB After Width: | Height: | Size: 3.0 KiB |
|
Before Width: | Height: | Size: 3.1 KiB After Width: | Height: | Size: 7.8 KiB |
|
Before Width: | Height: | Size: 2.0 KiB After Width: | Height: | Size: 3.6 KiB |
|
Before Width: | Height: | Size: 3.4 KiB After Width: | Height: | Size: 9.1 KiB |
|
Before Width: | Height: | Size: 3.6 KiB After Width: | Height: | Size: 9.8 KiB |
|
Before Width: | Height: | Size: 3.9 KiB After Width: | Height: | Size: 16 KiB |
|
Before Width: | Height: | Size: 2.4 KiB After Width: | Height: | Size: 5.0 KiB |
|
Before Width: | Height: | Size: 3.9 KiB After Width: | Height: | Size: 12 KiB |
|
Before Width: | Height: | Size: 2.5 KiB After Width: | Height: | Size: 5.4 KiB |
|
Before Width: | Height: | Size: 4.0 KiB After Width: | Height: | Size: 13 KiB |
|
Before Width: | Height: | Size: 3.7 KiB After Width: | Height: | Size: 15 KiB |
@ -5,7 +5,7 @@
|
||||
<key>CFBundleDevelopmentRegion</key>
|
||||
<string>$(DEVELOPMENT_LANGUAGE)</string>
|
||||
<key>CFBundleDisplayName</key>
|
||||
<string>EnakloPOS</string>
|
||||
<string>ApskelPOS</string>
|
||||
<key>CFBundleExecutable</key>
|
||||
<string>$(EXECUTABLE_NAME)</string>
|
||||
<key>CFBundleIdentifier</key>
|
||||
|
||||
17
launcher_icon.yaml
Normal file
@ -0,0 +1,17 @@
|
||||
# Generate: dart run flutter_launcher_icons -f launcher_icon.yaml
|
||||
|
||||
flutter_launcher_icons:
|
||||
android: "launcher_icon"
|
||||
ios: true
|
||||
image_path: "assets/logo/logo.png"
|
||||
remove_alpha_ios: true
|
||||
min_sdk_android: 21 # android min sdk min:16, default 21
|
||||
adaptive_icon_background: "#ffffff"
|
||||
adaptive_icon_foreground: "assets/logo/logo.png"
|
||||
web:
|
||||
generate: true
|
||||
image_path: "assets/logo/logo.png"
|
||||
windows:
|
||||
generate: true
|
||||
image_path: "assets/logo/logo.png"
|
||||
icon_size: 48
|
||||
@ -101,6 +101,9 @@ class $AssetsIconsGen {
|
||||
/// File path: assets/icons/payments.svg
|
||||
SvgGenImage get payments => const SvgGenImage('assets/icons/payments.svg');
|
||||
|
||||
/// File path: assets/icons/people.svg
|
||||
SvgGenImage get people => const SvgGenImage('assets/icons/people.svg');
|
||||
|
||||
/// File path: assets/icons/print.svg
|
||||
SvgGenImage get print => const SvgGenImage('assets/icons/print.svg');
|
||||
|
||||
@ -152,6 +155,7 @@ class $AssetsIconsGen {
|
||||
orders,
|
||||
pajak,
|
||||
payments,
|
||||
people,
|
||||
print,
|
||||
qrCode,
|
||||
report,
|
||||
@ -186,6 +190,12 @@ class $AssetsImagesGen {
|
||||
/// File path: assets/images/drink7.png
|
||||
AssetGenImage get drink7 => const AssetGenImage('assets/images/drink7.png');
|
||||
|
||||
/// File path: assets/images/gojek.png
|
||||
AssetGenImage get gojek => const AssetGenImage('assets/images/gojek.png');
|
||||
|
||||
/// File path: assets/images/grab.png
|
||||
AssetGenImage get grab => const AssetGenImage('assets/images/grab.png');
|
||||
|
||||
/// File path: assets/images/logo.png
|
||||
AssetGenImage get logo => const AssetGenImage('assets/images/logo.png');
|
||||
|
||||
@ -265,6 +275,8 @@ class $AssetsImagesGen {
|
||||
drink5,
|
||||
drink6,
|
||||
drink7,
|
||||
gojek,
|
||||
grab,
|
||||
logo,
|
||||
managePrinter,
|
||||
manageProduct,
|
||||
|
||||
15
lib/core/assets/fonts.gen.dart
Normal file
@ -0,0 +1,15 @@
|
||||
/// GENERATED CODE - DO NOT MODIFY BY HAND
|
||||
/// *****************************************************
|
||||
/// FlutterGen
|
||||
/// *****************************************************
|
||||
|
||||
// coverage:ignore-file
|
||||
// ignore_for_file: type=lint
|
||||
// ignore_for_file: directives_ordering,unnecessary_import,implicit_dynamic_list_literal,deprecated_member_use
|
||||
|
||||
class FontFamily {
|
||||
FontFamily._();
|
||||
|
||||
/// Font family: Quicksand
|
||||
static const String quicksand = 'Quicksand';
|
||||
}
|
||||
@ -18,6 +18,10 @@ class Button extends StatelessWidget {
|
||||
this.icon,
|
||||
this.disabled = false,
|
||||
this.fontSize = 16.0,
|
||||
this.elevation,
|
||||
this.labelStyle,
|
||||
this.mainAxisAlignment = MainAxisAlignment.center,
|
||||
this.crossAxisAlignment = CrossAxisAlignment.center,
|
||||
});
|
||||
|
||||
const Button.outlined({
|
||||
@ -33,9 +37,13 @@ class Button extends StatelessWidget {
|
||||
this.icon,
|
||||
this.disabled = false,
|
||||
this.fontSize = 16.0,
|
||||
this.elevation,
|
||||
this.labelStyle,
|
||||
this.mainAxisAlignment = MainAxisAlignment.center,
|
||||
this.crossAxisAlignment = CrossAxisAlignment.center,
|
||||
});
|
||||
|
||||
final Function() onPressed;
|
||||
final Function()? onPressed;
|
||||
final String label;
|
||||
final ButtonStyle style;
|
||||
final Color color;
|
||||
@ -43,9 +51,13 @@ class Button extends StatelessWidget {
|
||||
final double? width;
|
||||
final double height;
|
||||
final double borderRadius;
|
||||
final double? elevation;
|
||||
final Widget? icon;
|
||||
final bool disabled;
|
||||
final double fontSize;
|
||||
final TextStyle? labelStyle;
|
||||
final MainAxisAlignment mainAxisAlignment;
|
||||
final CrossAxisAlignment crossAxisAlignment;
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
@ -60,11 +72,12 @@ class Button extends StatelessWidget {
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(borderRadius),
|
||||
),
|
||||
elevation: elevation,
|
||||
padding: const EdgeInsets.symmetric(horizontal: 16.0),
|
||||
),
|
||||
child: Row(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
mainAxisAlignment: mainAxisAlignment,
|
||||
crossAxisAlignment: crossAxisAlignment,
|
||||
children: [
|
||||
icon ?? const SizedBox.shrink(),
|
||||
if (icon != null) const SizedBox(width: 10.0),
|
||||
@ -73,11 +86,12 @@ class Button extends StatelessWidget {
|
||||
fit: BoxFit.scaleDown,
|
||||
child: Text(
|
||||
label,
|
||||
style: TextStyle(
|
||||
color: disabled ? Colors.grey : textColor,
|
||||
fontSize: fontSize,
|
||||
fontWeight: FontWeight.w600,
|
||||
),
|
||||
style: labelStyle ??
|
||||
TextStyle(
|
||||
color: disabled ? Colors.grey : textColor,
|
||||
fontSize: fontSize,
|
||||
fontWeight: FontWeight.bold,
|
||||
),
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
),
|
||||
@ -89,14 +103,15 @@ class Button extends StatelessWidget {
|
||||
onPressed: disabled ? null : onPressed,
|
||||
style: OutlinedButton.styleFrom(
|
||||
backgroundColor: color,
|
||||
side: const BorderSide(color: Colors.grey),
|
||||
side: const BorderSide(color: AppColors.primary),
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(borderRadius),
|
||||
),
|
||||
padding: const EdgeInsets.symmetric(horizontal: 16.0),
|
||||
),
|
||||
child: Row(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
mainAxisAlignment: mainAxisAlignment,
|
||||
crossAxisAlignment: crossAxisAlignment,
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
icon ?? const SizedBox.shrink(),
|
||||
@ -106,11 +121,12 @@ class Button extends StatelessWidget {
|
||||
fit: BoxFit.scaleDown,
|
||||
child: Text(
|
||||
label,
|
||||
style: TextStyle(
|
||||
color: disabled ? Colors.grey : textColor,
|
||||
fontSize: fontSize,
|
||||
fontWeight: FontWeight.w600,
|
||||
),
|
||||
style: labelStyle ??
|
||||
TextStyle(
|
||||
color: disabled ? Colors.grey : textColor,
|
||||
fontSize: fontSize,
|
||||
fontWeight: FontWeight.w600,
|
||||
),
|
||||
textAlign: TextAlign.center,
|
||||
),
|
||||
),
|
||||
|
||||
117
lib/core/components/custom_modal_dialog.dart
Normal file
@ -0,0 +1,117 @@
|
||||
import 'package:enaklo_pos/core/components/spaces.dart';
|
||||
import 'package:enaklo_pos/core/constants/colors.dart';
|
||||
import 'package:enaklo_pos/core/extensions/build_context_ext.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class CustomModalDialog extends StatelessWidget {
|
||||
final String title;
|
||||
final String? subtitle;
|
||||
final Widget child;
|
||||
final VoidCallback? onClose;
|
||||
final double? minWidth;
|
||||
final double? maxWidth;
|
||||
final double? minHeight;
|
||||
final double? maxHeight;
|
||||
final EdgeInsets? contentPadding;
|
||||
|
||||
const CustomModalDialog({
|
||||
super.key,
|
||||
required this.title,
|
||||
this.subtitle,
|
||||
required this.child,
|
||||
this.onClose,
|
||||
this.minWidth,
|
||||
this.maxWidth,
|
||||
this.minHeight,
|
||||
this.maxHeight,
|
||||
this.contentPadding,
|
||||
});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Dialog(
|
||||
backgroundColor: AppColors.white,
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(16),
|
||||
),
|
||||
child: ConstrainedBox(
|
||||
constraints: BoxConstraints(
|
||||
minWidth: minWidth ?? context.deviceWidth * 0.3,
|
||||
maxWidth: maxWidth ?? context.deviceWidth * 0.8,
|
||||
minHeight: minHeight ?? context.deviceHeight * 0.3,
|
||||
maxHeight: maxHeight ?? context.deviceHeight * 0.8,
|
||||
),
|
||||
child: IntrinsicWidth(
|
||||
child: Column(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
Container(
|
||||
padding: const EdgeInsets.all(16),
|
||||
width: double.infinity,
|
||||
decoration: BoxDecoration(
|
||||
gradient: LinearGradient(
|
||||
colors: [
|
||||
const Color.fromARGB(255, 81, 40, 134),
|
||||
AppColors.primary,
|
||||
],
|
||||
begin: Alignment.topLeft,
|
||||
end: Alignment.bottomRight,
|
||||
),
|
||||
borderRadius: BorderRadius.vertical(
|
||||
top: Radius.circular(16),
|
||||
),
|
||||
),
|
||||
child: Row(
|
||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||
children: [
|
||||
Expanded(
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text(
|
||||
title,
|
||||
style: TextStyle(
|
||||
color: AppColors.white,
|
||||
fontSize: 20,
|
||||
fontWeight: FontWeight.bold,
|
||||
),
|
||||
),
|
||||
if (subtitle != null)
|
||||
Text(
|
||||
subtitle ?? '',
|
||||
style: TextStyle(
|
||||
color: AppColors.grey,
|
||||
fontSize: 16,
|
||||
fontWeight: FontWeight.w500,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
SpaceWidth(12),
|
||||
IconButton(
|
||||
icon: Icon(Icons.close, color: AppColors.white),
|
||||
onPressed: () {
|
||||
if (onClose != null) {
|
||||
onClose!();
|
||||
} else {
|
||||
Navigator.of(context).pop();
|
||||
}
|
||||
},
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
Flexible(
|
||||
child: SingleChildScrollView(
|
||||
padding: contentPadding ?? EdgeInsets.zero,
|
||||
child: child,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
@ -14,6 +14,8 @@ class CustomTextField extends StatelessWidget {
|
||||
final Widget? prefixIcon;
|
||||
final Widget? suffixIcon;
|
||||
final bool readOnly;
|
||||
final int maxLines;
|
||||
final String? Function(String?)? validator;
|
||||
|
||||
const CustomTextField({
|
||||
super.key,
|
||||
@ -28,6 +30,8 @@ class CustomTextField extends StatelessWidget {
|
||||
this.prefixIcon,
|
||||
this.suffixIcon,
|
||||
this.readOnly = false,
|
||||
this.maxLines = 1,
|
||||
this.validator,
|
||||
});
|
||||
|
||||
@override
|
||||
@ -53,17 +57,11 @@ class CustomTextField extends StatelessWidget {
|
||||
textInputAction: textInputAction,
|
||||
textCapitalization: textCapitalization ?? TextCapitalization.none,
|
||||
readOnly: readOnly,
|
||||
maxLines: maxLines,
|
||||
validator: validator,
|
||||
decoration: InputDecoration(
|
||||
prefixIcon: prefixIcon,
|
||||
suffixIcon: suffixIcon,
|
||||
border: OutlineInputBorder(
|
||||
borderRadius: BorderRadius.circular(16.0),
|
||||
borderSide: const BorderSide(color: Colors.grey),
|
||||
),
|
||||
enabledBorder: OutlineInputBorder(
|
||||
borderRadius: BorderRadius.circular(16.0),
|
||||
borderSide: const BorderSide(color: Colors.grey),
|
||||
),
|
||||
hintText: label,
|
||||
),
|
||||
),
|
||||
|
||||
41
lib/core/components/dashed_divider.dart
Normal file
@ -0,0 +1,41 @@
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class DashedDivider extends StatelessWidget {
|
||||
final double height;
|
||||
final double dashWidth;
|
||||
final double dashSpacing;
|
||||
final Color color;
|
||||
|
||||
const DashedDivider({
|
||||
super.key,
|
||||
this.height = 1,
|
||||
this.dashWidth = 5,
|
||||
this.dashSpacing = 3,
|
||||
this.color = Colors.grey,
|
||||
});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return SizedBox(
|
||||
height: height,
|
||||
child: LayoutBuilder(
|
||||
builder: (context, constraints) {
|
||||
final boxWidth = constraints.constrainWidth();
|
||||
final dashCount = (boxWidth / (dashWidth + dashSpacing)).floor();
|
||||
return Row(
|
||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||
children: List.generate(dashCount, (_) {
|
||||
return SizedBox(
|
||||
width: dashWidth,
|
||||
height: height,
|
||||
child: DecoratedBox(
|
||||
decoration: BoxDecoration(color: color),
|
||||
),
|
||||
);
|
||||
}),
|
||||
);
|
||||
},
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
48
lib/core/components/flushbar.dart
Normal file
@ -0,0 +1,48 @@
|
||||
import 'package:another_flushbar/flushbar.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class AppFlushbar {
|
||||
static void showSuccess(BuildContext context, String message) {
|
||||
Flushbar(
|
||||
messageText: Text(
|
||||
message,
|
||||
style: const TextStyle(
|
||||
color: Colors.white,
|
||||
fontWeight: FontWeight.bold,
|
||||
fontSize: 16,
|
||||
),
|
||||
),
|
||||
icon: const Icon(
|
||||
Icons.check_circle,
|
||||
color: Colors.white,
|
||||
),
|
||||
duration: const Duration(seconds: 2),
|
||||
flushbarPosition: FlushbarPosition.BOTTOM,
|
||||
backgroundColor: Colors.green,
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
margin: const EdgeInsets.all(12),
|
||||
).show(context);
|
||||
}
|
||||
|
||||
static void showError(BuildContext context, String message) {
|
||||
Flushbar(
|
||||
messageText: Text(
|
||||
message,
|
||||
style: const TextStyle(
|
||||
color: Colors.white,
|
||||
fontWeight: FontWeight.bold,
|
||||
fontSize: 16,
|
||||
),
|
||||
),
|
||||
icon: const Icon(
|
||||
Icons.error,
|
||||
color: Colors.white,
|
||||
),
|
||||
duration: const Duration(seconds: 3),
|
||||
flushbarPosition: FlushbarPosition.BOTTOM,
|
||||
backgroundColor: Colors.red,
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
margin: const EdgeInsets.all(12),
|
||||
).show(context);
|
||||
}
|
||||
}
|
||||
@ -1,91 +1,189 @@
|
||||
import 'dart:io';
|
||||
|
||||
import 'package:enaklo_pos/presentation/setting/bloc/upload_file/upload_file_bloc.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||
import 'package:image_picker/image_picker.dart';
|
||||
import 'package:cached_network_image/cached_network_image.dart';
|
||||
|
||||
import '../assets/assets.gen.dart';
|
||||
import '../constants/colors.dart';
|
||||
import '../constants/variables.dart';
|
||||
import 'buttons.dart';
|
||||
import 'spaces.dart';
|
||||
|
||||
class ImagePickerWidget extends StatefulWidget {
|
||||
final String label;
|
||||
final void Function(XFile? file) onChanged;
|
||||
final void Function(String? uploadedUrl)? onUploaded;
|
||||
final bool showLabel;
|
||||
final String? initialImageUrl;
|
||||
final bool autoUpload;
|
||||
|
||||
const ImagePickerWidget({
|
||||
super.key,
|
||||
required this.label,
|
||||
required this.onChanged,
|
||||
this.onUploaded,
|
||||
this.showLabel = true,
|
||||
this.initialImageUrl,
|
||||
this.autoUpload = false,
|
||||
});
|
||||
|
||||
@override
|
||||
State<ImagePickerWidget> createState() => _ImagePickerWidgetState();
|
||||
}
|
||||
|
||||
class _ImagePickerWidgetState extends State<ImagePickerWidget> {
|
||||
class _ImagePickerWidgetState extends State<ImagePickerWidget>
|
||||
with TickerProviderStateMixin {
|
||||
String? imagePath;
|
||||
String? uploadedImageUrl;
|
||||
bool hasInitialImage = false;
|
||||
bool isHovering = false;
|
||||
bool isUploading = false;
|
||||
late AnimationController _scaleController;
|
||||
late AnimationController _fadeController;
|
||||
late AnimationController _uploadController;
|
||||
late Animation<double> _scaleAnimation;
|
||||
late Animation<double> _fadeAnimation;
|
||||
late Animation<double> _uploadAnimation;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
hasInitialImage = widget.initialImageUrl != null;
|
||||
|
||||
_scaleController = AnimationController(
|
||||
duration: const Duration(milliseconds: 200),
|
||||
vsync: this,
|
||||
);
|
||||
_fadeController = AnimationController(
|
||||
duration: const Duration(milliseconds: 300),
|
||||
vsync: this,
|
||||
);
|
||||
_uploadController = AnimationController(
|
||||
duration: const Duration(milliseconds: 1000),
|
||||
vsync: this,
|
||||
);
|
||||
|
||||
_scaleAnimation = Tween<double>(
|
||||
begin: 1.0,
|
||||
end: 0.95,
|
||||
).animate(CurvedAnimation(
|
||||
parent: _scaleController,
|
||||
curve: Curves.easeInOut,
|
||||
));
|
||||
|
||||
_fadeAnimation = Tween<double>(
|
||||
begin: 0.0,
|
||||
end: 1.0,
|
||||
).animate(CurvedAnimation(
|
||||
parent: _fadeController,
|
||||
curve: Curves.easeInOut,
|
||||
));
|
||||
|
||||
_uploadAnimation = Tween<double>(
|
||||
begin: 0.0,
|
||||
end: 1.0,
|
||||
).animate(CurvedAnimation(
|
||||
parent: _uploadController,
|
||||
curve: Curves.easeInOut,
|
||||
));
|
||||
|
||||
_fadeController.forward();
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
_scaleController.dispose();
|
||||
_fadeController.dispose();
|
||||
_uploadController.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
Future<void> _pickImage() async {
|
||||
_scaleController.forward().then((_) {
|
||||
_scaleController.reverse();
|
||||
});
|
||||
|
||||
final pickedFile = await ImagePicker().pickImage(
|
||||
source: ImageSource.gallery,
|
||||
);
|
||||
|
||||
setState(() {
|
||||
if (pickedFile != null) {
|
||||
if (pickedFile != null) {
|
||||
setState(() {
|
||||
imagePath = pickedFile.path;
|
||||
hasInitialImage = false; // Clear initial image when new image is picked
|
||||
widget.onChanged(pickedFile);
|
||||
} else {
|
||||
debugPrint('No image selected.');
|
||||
widget.onChanged(null);
|
||||
hasInitialImage = false;
|
||||
uploadedImageUrl = null;
|
||||
});
|
||||
|
||||
widget.onChanged(pickedFile);
|
||||
|
||||
// Auto upload if enabled
|
||||
if (widget.autoUpload) {
|
||||
_uploadImage(pickedFile.path);
|
||||
}
|
||||
});
|
||||
} else {
|
||||
debugPrint('No image selected.');
|
||||
widget.onChanged(null);
|
||||
}
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
if (widget.showLabel) ...[
|
||||
Text(
|
||||
widget.label,
|
||||
style: const TextStyle(
|
||||
fontSize: 14,
|
||||
fontWeight: FontWeight.w700,
|
||||
),
|
||||
void _uploadImage(String filePath) {
|
||||
setState(() {
|
||||
isUploading = true;
|
||||
});
|
||||
_uploadController.forward();
|
||||
|
||||
context.read<UploadFileBloc>().add(
|
||||
UploadFileEvent.upload(filePath),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildImageContainer() {
|
||||
return Container(
|
||||
width: 100.0,
|
||||
height: 100.0,
|
||||
decoration: BoxDecoration(
|
||||
borderRadius: BorderRadius.circular(20.0),
|
||||
boxShadow: [
|
||||
BoxShadow(
|
||||
color: Colors.black.withOpacity(0.1),
|
||||
blurRadius: 10,
|
||||
offset: const Offset(0, 4),
|
||||
),
|
||||
const SpaceHeight(12.0),
|
||||
],
|
||||
Container(
|
||||
padding: const EdgeInsets.all(6.0),
|
||||
decoration: BoxDecoration(
|
||||
borderRadius: BorderRadius.circular(16.0),
|
||||
border: Border.all(color: AppColors.primary),
|
||||
),
|
||||
child: Row(
|
||||
children: [
|
||||
SizedBox(
|
||||
width: 80.0,
|
||||
height: 80.0,
|
||||
child: ClipRRect(
|
||||
borderRadius: BorderRadius.circular(10.0),
|
||||
child: imagePath != null
|
||||
? Image.file(
|
||||
File(imagePath!),
|
||||
),
|
||||
child: ClipRRect(
|
||||
borderRadius: BorderRadius.circular(20.0),
|
||||
child: Stack(
|
||||
children: [
|
||||
Positioned.fill(
|
||||
child: imagePath != null
|
||||
? Image.file(
|
||||
File(imagePath!),
|
||||
fit: BoxFit.cover,
|
||||
)
|
||||
: uploadedImageUrl != null
|
||||
? CachedNetworkImage(
|
||||
imageUrl: uploadedImageUrl!.contains('http')
|
||||
? uploadedImageUrl!
|
||||
: '${Variables.baseUrl}/$uploadedImageUrl',
|
||||
placeholder: (context, url) => Container(
|
||||
decoration: BoxDecoration(
|
||||
gradient: LinearGradient(
|
||||
colors: [
|
||||
AppColors.primary.withOpacity(0.1),
|
||||
AppColors.primary.withOpacity(0.05),
|
||||
],
|
||||
begin: Alignment.topLeft,
|
||||
end: Alignment.bottomRight,
|
||||
),
|
||||
),
|
||||
child: const Center(
|
||||
child: CircularProgressIndicator(strokeWidth: 2),
|
||||
),
|
||||
),
|
||||
errorWidget: (context, url, error) =>
|
||||
_buildPlaceholder(),
|
||||
fit: BoxFit.cover,
|
||||
)
|
||||
: hasInitialImage && widget.initialImageUrl != null
|
||||
@ -93,38 +191,493 @@ class _ImagePickerWidgetState extends State<ImagePickerWidget> {
|
||||
imageUrl: widget.initialImageUrl!.contains('http')
|
||||
? widget.initialImageUrl!
|
||||
: '${Variables.baseUrl}/${widget.initialImageUrl}',
|
||||
placeholder: (context, url) =>
|
||||
const Center(child: CircularProgressIndicator()),
|
||||
errorWidget: (context, url, error) => Container(
|
||||
padding: const EdgeInsets.all(16.0),
|
||||
color: AppColors.black.withOpacity(0.05),
|
||||
child: Assets.icons.image.svg(),
|
||||
placeholder: (context, url) => Container(
|
||||
decoration: BoxDecoration(
|
||||
gradient: LinearGradient(
|
||||
colors: [
|
||||
AppColors.primary.withOpacity(0.1),
|
||||
AppColors.primary.withOpacity(0.05),
|
||||
],
|
||||
begin: Alignment.topLeft,
|
||||
end: Alignment.bottomRight,
|
||||
),
|
||||
),
|
||||
child: const Center(
|
||||
child:
|
||||
CircularProgressIndicator(strokeWidth: 2),
|
||||
),
|
||||
),
|
||||
errorWidget: (context, url, error) =>
|
||||
_buildPlaceholder(),
|
||||
fit: BoxFit.cover,
|
||||
)
|
||||
: Container(
|
||||
padding: const EdgeInsets.all(16.0),
|
||||
color: AppColors.black.withOpacity(0.05),
|
||||
child: Assets.icons.image.svg(),
|
||||
),
|
||||
: _buildPlaceholder(),
|
||||
),
|
||||
// Upload progress overlay
|
||||
if (isUploading)
|
||||
Positioned.fill(
|
||||
child: Container(
|
||||
decoration: BoxDecoration(
|
||||
color: Colors.black.withOpacity(0.6),
|
||||
),
|
||||
child: Center(
|
||||
child: Column(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
children: [
|
||||
SizedBox(
|
||||
width: 24,
|
||||
height: 24,
|
||||
child: CircularProgressIndicator(
|
||||
color: Colors.white,
|
||||
strokeWidth: 2,
|
||||
value: _uploadAnimation.value == 1.0
|
||||
? null
|
||||
: _uploadAnimation.value,
|
||||
),
|
||||
),
|
||||
const SizedBox(height: 8),
|
||||
const Text(
|
||||
'Uploading...',
|
||||
style: TextStyle(
|
||||
color: Colors.white,
|
||||
fontSize: 10,
|
||||
fontWeight: FontWeight.w500,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
const Spacer(),
|
||||
Padding(
|
||||
padding: const EdgeInsets.only(right: 10.0),
|
||||
child: Button.filled(
|
||||
height: 30.0,
|
||||
width: 140.0,
|
||||
onPressed: _pickImage,
|
||||
label: 'Choose Photo',
|
||||
fontSize: 12.0,
|
||||
borderRadius: 5.0,
|
||||
// Overlay gradient for better button visibility
|
||||
if ((imagePath != null ||
|
||||
uploadedImageUrl != null ||
|
||||
(hasInitialImage && widget.initialImageUrl != null)) &&
|
||||
!isUploading)
|
||||
Positioned.fill(
|
||||
child: Container(
|
||||
decoration: BoxDecoration(
|
||||
gradient: LinearGradient(
|
||||
colors: [
|
||||
Colors.transparent,
|
||||
Colors.black.withOpacity(0.3),
|
||||
],
|
||||
begin: Alignment.topCenter,
|
||||
end: Alignment.bottomCenter,
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildPlaceholder() {
|
||||
return Container(
|
||||
decoration: BoxDecoration(
|
||||
gradient: LinearGradient(
|
||||
colors: [
|
||||
AppColors.primary.withOpacity(0.1),
|
||||
AppColors.primary.withOpacity(0.05),
|
||||
],
|
||||
begin: Alignment.topLeft,
|
||||
end: Alignment.bottomRight,
|
||||
),
|
||||
),
|
||||
child: Center(
|
||||
child: Column(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
children: [
|
||||
Icon(
|
||||
Icons.add_photo_alternate_outlined,
|
||||
size: 32,
|
||||
color: AppColors.primary.withOpacity(0.6),
|
||||
),
|
||||
const SizedBox(height: 4),
|
||||
Text(
|
||||
'Photo',
|
||||
style: TextStyle(
|
||||
fontSize: 10,
|
||||
color: AppColors.primary.withOpacity(0.6),
|
||||
fontWeight: FontWeight.w500,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildActionButton() {
|
||||
bool hasImage = imagePath != null ||
|
||||
uploadedImageUrl != null ||
|
||||
(hasInitialImage && widget.initialImageUrl != null);
|
||||
|
||||
return Container(
|
||||
decoration: BoxDecoration(
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
gradient: LinearGradient(
|
||||
colors: [
|
||||
AppColors.primary,
|
||||
AppColors.primary.withOpacity(0.8),
|
||||
],
|
||||
begin: Alignment.topLeft,
|
||||
end: Alignment.bottomRight,
|
||||
),
|
||||
boxShadow: [
|
||||
BoxShadow(
|
||||
color: AppColors.primary.withOpacity(0.3),
|
||||
blurRadius: 8,
|
||||
offset: const Offset(0, 4),
|
||||
),
|
||||
],
|
||||
),
|
||||
child: Material(
|
||||
color: Colors.transparent,
|
||||
child: InkWell(
|
||||
onTap: isUploading ? null : _pickImage,
|
||||
onHover: (hover) {
|
||||
setState(() {
|
||||
isHovering = hover;
|
||||
});
|
||||
},
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
child: AnimatedContainer(
|
||||
duration: const Duration(milliseconds: 200),
|
||||
padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 12),
|
||||
decoration: BoxDecoration(
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
color: isHovering
|
||||
? Colors.white.withOpacity(0.1)
|
||||
: Colors.transparent,
|
||||
),
|
||||
child: Row(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
Icon(
|
||||
isUploading
|
||||
? Icons.cloud_upload_outlined
|
||||
: hasImage
|
||||
? Icons.edit_outlined
|
||||
: Icons.add_photo_alternate_outlined,
|
||||
color: Colors.white,
|
||||
size: 18,
|
||||
),
|
||||
const SizedBox(width: 8),
|
||||
Text(
|
||||
isUploading
|
||||
? 'Uploading...'
|
||||
: hasImage
|
||||
? 'Change Photo'
|
||||
: 'Choose Photo',
|
||||
style: const TextStyle(
|
||||
color: Colors.white,
|
||||
fontSize: 14,
|
||||
fontWeight: FontWeight.w600,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Widget _buildUploadButton() {
|
||||
if (!widget.autoUpload &&
|
||||
imagePath != null &&
|
||||
uploadedImageUrl == null &&
|
||||
!isUploading) {
|
||||
return Padding(
|
||||
padding: const EdgeInsets.only(top: 12),
|
||||
child: Container(
|
||||
decoration: BoxDecoration(
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
gradient: LinearGradient(
|
||||
colors: [
|
||||
Colors.green.shade600,
|
||||
Colors.green.shade500,
|
||||
],
|
||||
begin: Alignment.topLeft,
|
||||
end: Alignment.bottomRight,
|
||||
),
|
||||
boxShadow: [
|
||||
BoxShadow(
|
||||
color: Colors.green.withOpacity(0.3),
|
||||
blurRadius: 8,
|
||||
offset: const Offset(0, 4),
|
||||
),
|
||||
],
|
||||
),
|
||||
child: Material(
|
||||
color: Colors.transparent,
|
||||
child: InkWell(
|
||||
onTap: () => _uploadImage(imagePath!),
|
||||
borderRadius: BorderRadius.circular(12),
|
||||
child: Container(
|
||||
padding:
|
||||
const EdgeInsets.symmetric(horizontal: 20, vertical: 10),
|
||||
child: Row(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
Icon(
|
||||
Icons.cloud_upload_outlined,
|
||||
color: Colors.white,
|
||||
size: 16,
|
||||
),
|
||||
const SizedBox(width: 8),
|
||||
Text(
|
||||
'Upload to Server',
|
||||
style: const TextStyle(
|
||||
color: Colors.white,
|
||||
fontSize: 12,
|
||||
fontWeight: FontWeight.w600,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
}
|
||||
return const SizedBox.shrink();
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return BlocListener<UploadFileBloc, UploadFileState>(
|
||||
listener: (context, state) {
|
||||
state.when(
|
||||
initial: () {},
|
||||
loading: () {
|
||||
if (!isUploading) {
|
||||
setState(() {
|
||||
isUploading = true;
|
||||
});
|
||||
_uploadController.repeat();
|
||||
}
|
||||
},
|
||||
success: (fileData) {
|
||||
setState(() {
|
||||
isUploading = false;
|
||||
uploadedImageUrl = fileData.fileUrl;
|
||||
});
|
||||
_uploadController.reset();
|
||||
|
||||
if (widget.onUploaded != null) {
|
||||
widget.onUploaded!(fileData.fileUrl);
|
||||
}
|
||||
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(
|
||||
content: Row(
|
||||
children: [
|
||||
Icon(Icons.check_circle, color: Colors.white),
|
||||
SizedBox(width: 8),
|
||||
Text('Image uploaded successfully!'),
|
||||
],
|
||||
),
|
||||
backgroundColor: Colors.green,
|
||||
behavior: SnackBarBehavior.floating,
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(10),
|
||||
),
|
||||
duration: Duration(seconds: 2),
|
||||
),
|
||||
);
|
||||
},
|
||||
error: (message) {
|
||||
setState(() {
|
||||
isUploading = false;
|
||||
});
|
||||
_uploadController.reset();
|
||||
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(
|
||||
content: Row(
|
||||
children: [
|
||||
Icon(Icons.error_outline, color: Colors.white),
|
||||
SizedBox(width: 8),
|
||||
Expanded(child: Text('Upload failed: $message')),
|
||||
],
|
||||
),
|
||||
backgroundColor: Colors.red,
|
||||
behavior: SnackBarBehavior.floating,
|
||||
shape: RoundedRectangleBorder(
|
||||
borderRadius: BorderRadius.circular(10),
|
||||
),
|
||||
duration: Duration(seconds: 3),
|
||||
),
|
||||
);
|
||||
},
|
||||
);
|
||||
},
|
||||
child: FadeTransition(
|
||||
opacity: _fadeAnimation,
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
if (widget.showLabel) ...[
|
||||
Text(
|
||||
widget.label,
|
||||
style: const TextStyle(
|
||||
fontSize: 16,
|
||||
fontWeight: FontWeight.w700,
|
||||
color: Colors.black87,
|
||||
),
|
||||
),
|
||||
const SpaceHeight(16.0),
|
||||
],
|
||||
ScaleTransition(
|
||||
scale: _scaleAnimation,
|
||||
child: Container(
|
||||
padding: const EdgeInsets.all(20.0),
|
||||
decoration: BoxDecoration(
|
||||
borderRadius: BorderRadius.circular(24.0),
|
||||
color: Colors.white,
|
||||
border: Border.all(
|
||||
color: isUploading
|
||||
? Colors.orange.withOpacity(0.5)
|
||||
: uploadedImageUrl != null
|
||||
? Colors.green.withOpacity(0.5)
|
||||
: AppColors.primary.withOpacity(0.2),
|
||||
width: 1.5,
|
||||
),
|
||||
boxShadow: [
|
||||
BoxShadow(
|
||||
color: Colors.black.withOpacity(0.05),
|
||||
blurRadius: 20,
|
||||
offset: const Offset(0, 10),
|
||||
),
|
||||
],
|
||||
),
|
||||
child: Column(
|
||||
children: [
|
||||
_buildImageContainer(),
|
||||
const SizedBox(height: 20),
|
||||
_buildActionButton(),
|
||||
_buildUploadButton(),
|
||||
if ((imagePath != null ||
|
||||
uploadedImageUrl != null ||
|
||||
(hasInitialImage &&
|
||||
widget.initialImageUrl != null)) &&
|
||||
!isUploading) ...[
|
||||
const SizedBox(height: 12),
|
||||
TextButton.icon(
|
||||
onPressed: () {
|
||||
setState(() {
|
||||
imagePath = null;
|
||||
hasInitialImage = false;
|
||||
uploadedImageUrl = null;
|
||||
});
|
||||
widget.onChanged(null);
|
||||
if (widget.onUploaded != null) {
|
||||
widget.onUploaded!(null);
|
||||
}
|
||||
},
|
||||
icon: Icon(
|
||||
Icons.delete_outline,
|
||||
size: 16,
|
||||
color: Colors.red.shade400,
|
||||
),
|
||||
label: Text(
|
||||
'Remove Photo',
|
||||
style: TextStyle(
|
||||
color: Colors.red.shade400,
|
||||
fontSize: 12,
|
||||
fontWeight: FontWeight.w500,
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
// Upload status indicator
|
||||
if (uploadedImageUrl != null && !isUploading) ...[
|
||||
const SizedBox(height: 8),
|
||||
Container(
|
||||
padding: const EdgeInsets.symmetric(
|
||||
horizontal: 12, vertical: 6),
|
||||
decoration: BoxDecoration(
|
||||
color: Colors.green.withOpacity(0.1),
|
||||
borderRadius: BorderRadius.circular(20),
|
||||
border:
|
||||
Border.all(color: Colors.green.withOpacity(0.3)),
|
||||
),
|
||||
child: Row(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
Icon(
|
||||
Icons.cloud_done_outlined,
|
||||
size: 14,
|
||||
color: Colors.green.shade600,
|
||||
),
|
||||
const SizedBox(width: 6),
|
||||
Text(
|
||||
'Uploaded',
|
||||
style: TextStyle(
|
||||
fontSize: 10,
|
||||
fontWeight: FontWeight.w600,
|
||||
color: Colors.green.shade600,
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
],
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/// Cara menggunakan widget ini:
|
||||
///
|
||||
/// ```dart
|
||||
/// // 1. Basic usage tanpa auto upload
|
||||
/// ImagePickerWidget(
|
||||
/// label: 'Product Image',
|
||||
/// onChanged: (file) {
|
||||
/// // Handle selected file
|
||||
/// print('Selected file: ${file?.path}');
|
||||
/// },
|
||||
/// onUploaded: (url) {
|
||||
/// // Handle uploaded URL
|
||||
/// print('Uploaded URL: $url');
|
||||
/// },
|
||||
/// )
|
||||
///
|
||||
/// // 2. Auto upload setelah memilih gambar
|
||||
/// ImagePickerWidget(
|
||||
/// label: 'Profile Picture',
|
||||
/// autoUpload: true,
|
||||
/// onChanged: (file) => setState(() => selectedFile = file),
|
||||
/// onUploaded: (url) => setState(() => profileImageUrl = url),
|
||||
/// )
|
||||
///
|
||||
/// // 3. Dengan initial image
|
||||
/// ImagePickerWidget(
|
||||
/// label: 'Banner Image',
|
||||
/// initialImageUrl: existingImageUrl,
|
||||
/// onChanged: (file) => handleFileChange(file),
|
||||
/// onUploaded: (url) => handleUploadSuccess(url),
|
||||
/// )
|
||||
/// ```
|
||||
///
|
||||
/// Pastikan untuk wrap widget ini dengan BlocProvider:
|
||||
/// ```dart
|
||||
/// BlocProvider(
|
||||
/// create: (context) => UploadFileBloc(
|
||||
/// context.read<FileRemoteDataSource>(),
|
||||
/// ),
|
||||
/// child: ImagePickerWidget(...),
|
||||
/// )
|
||||
/// ```
|
||||
@ -2,8 +2,6 @@ import 'package:flutter/material.dart';
|
||||
|
||||
import '../constants/colors.dart';
|
||||
|
||||
|
||||
|
||||
class SearchInput extends StatelessWidget {
|
||||
final TextEditingController controller;
|
||||
final Function(String value)? onChanged;
|
||||
|
||||
@ -2,7 +2,7 @@ import 'package:flutter/material.dart';
|
||||
|
||||
class AppColors {
|
||||
/// primary = #3949AB
|
||||
static const Color primary = Color(0xff6466f1);
|
||||
static const Color primary = Color(0xff36175e);
|
||||
|
||||
/// grey = #B7B7B7
|
||||
static const Color grey = Color(0xffB7B7B7);
|
||||
@ -18,6 +18,7 @@ class AppColors {
|
||||
|
||||
/// white = #FFFFFF
|
||||
static const Color white = Color(0xffFFFFFF);
|
||||
static const Color whiteText = Color(0xfff1eaf9);
|
||||
|
||||
/// green = #50C474
|
||||
static const Color green = Color(0xff50C474);
|
||||
@ -36,4 +37,10 @@ class AppColors {
|
||||
|
||||
/// stroke = #EFF0F6
|
||||
static const Color stroke = Color(0xffEFF0F6);
|
||||
|
||||
static const Color background = Color.fromARGB(255, 241, 241, 241);
|
||||
|
||||
static const Color primaryLight = Color(0xFF5A3E8A);
|
||||
static const Color greyLight = Color(0xFFE0E0E0);
|
||||
static const Color greyDark = Color(0xFF707070);
|
||||
}
|
||||
|
||||
41
lib/core/constants/theme.dart
Normal file
@ -0,0 +1,41 @@
|
||||
import 'package:enaklo_pos/core/assets/fonts.gen.dart';
|
||||
import 'package:enaklo_pos/core/constants/colors.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
ThemeData getApplicationTheme = ThemeData(
|
||||
primaryColor: AppColors.primary,
|
||||
scaffoldBackgroundColor: AppColors.white,
|
||||
appBarTheme: AppBarTheme(
|
||||
color: AppColors.white,
|
||||
elevation: 0,
|
||||
titleTextStyle: TextStyle(
|
||||
color: AppColors.primary,
|
||||
fontSize: 16.0,
|
||||
fontWeight: FontWeight.w500,
|
||||
),
|
||||
iconTheme: const IconThemeData(
|
||||
color: AppColors.primary,
|
||||
),
|
||||
),
|
||||
fontFamily: FontFamily.quicksand,
|
||||
colorScheme: ColorScheme.fromSeed(seedColor: AppColors.primary),
|
||||
useMaterial3: true,
|
||||
inputDecorationTheme: InputDecorationTheme(
|
||||
contentPadding: const EdgeInsets.symmetric(horizontal: 16.0),
|
||||
hintStyle: const TextStyle(
|
||||
color: AppColors.grey,
|
||||
),
|
||||
border: OutlineInputBorder(
|
||||
borderRadius: BorderRadius.circular(8.0),
|
||||
borderSide: BorderSide(color: AppColors.primary),
|
||||
),
|
||||
focusedBorder: OutlineInputBorder(
|
||||
borderRadius: BorderRadius.circular(8.0),
|
||||
borderSide: BorderSide(color: AppColors.primary),
|
||||
),
|
||||
enabledBorder: OutlineInputBorder(
|
||||
borderRadius: BorderRadius.circular(8.0),
|
||||
borderSide: BorderSide(color: AppColors.primary),
|
||||
),
|
||||
),
|
||||
);
|
||||
@ -2,5 +2,6 @@ class Variables {
|
||||
static const String appName = 'POS Kasir Resto App';
|
||||
static const String apiVersion = 'v1';
|
||||
// static const String baseUrl = 'http://192.168.1.202:8000';
|
||||
static const String baseUrl = 'https://pos-app-tablet.enaklo.co.id';
|
||||
static const String baseUrl = 'https://enaklo-pos-be.altru.id';
|
||||
static const int defaultLimit = 10;
|
||||
}
|
||||
|
||||
@ -15,4 +15,12 @@ extension StringExt on String {
|
||||
decimalDigits: 0,
|
||||
).format(parsedValue);
|
||||
}
|
||||
|
||||
String toTitleCase() {
|
||||
if (isEmpty) return '';
|
||||
return split(' ').map((word) {
|
||||
if (word.isEmpty) return '';
|
||||
return word[0].toUpperCase() + word.substring(1).toLowerCase();
|
||||
}).join(' ');
|
||||
}
|
||||
}
|
||||
|
||||
183
lib/core/function/app_function.dart
Normal file
@ -0,0 +1,183 @@
|
||||
import 'dart:developer';
|
||||
import 'dart:typed_data';
|
||||
|
||||
import 'package:barcode_image/barcode_image.dart';
|
||||
import 'package:enaklo_pos/core/extensions/string_ext.dart';
|
||||
import 'package:enaklo_pos/core/utils/printer_service.dart';
|
||||
import 'package:enaklo_pos/data/dataoutputs/print_dataoutputs.dart';
|
||||
import 'package:enaklo_pos/data/datasources/auth_local_datasource.dart';
|
||||
import 'package:enaklo_pos/data/datasources/outlet_local_datasource.dart';
|
||||
import 'package:enaklo_pos/data/datasources/product_local_datasource.dart';
|
||||
import 'package:enaklo_pos/data/datasources/settings_local_datasource.dart';
|
||||
import 'package:enaklo_pos/data/models/response/order_response_model.dart';
|
||||
import 'package:enaklo_pos/data/type/bussines_type.dart';
|
||||
import 'package:enaklo_pos/presentation/home/models/product_quantity.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:barcode/barcode.dart';
|
||||
import 'package:image/image.dart' as img;
|
||||
|
||||
Future<void> onPrint(
|
||||
BuildContext context, {
|
||||
required List<ProductQuantity> productQuantity,
|
||||
required Order order,
|
||||
}) async {
|
||||
final outlet = await OutletLocalDatasource().get();
|
||||
if (outlet.businessType == BusinessType.restaurant) {
|
||||
final checkerPrinter =
|
||||
await ProductLocalDatasource.instance.getPrinterByCode('checker');
|
||||
final kitchenPrinter =
|
||||
await ProductLocalDatasource.instance.getPrinterByCode('kitchen');
|
||||
final barPrinter =
|
||||
await ProductLocalDatasource.instance.getPrinterByCode('bar');
|
||||
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
|
||||
// Checker printer
|
||||
if (checkerPrinter != null) {
|
||||
try {
|
||||
final printValue = await PrintDataoutputs.instance.printChecker(
|
||||
productQuantity,
|
||||
order.tableNumber ?? "",
|
||||
order.orderNumber ?? "",
|
||||
authData.user?.name ?? "",
|
||||
checkerPrinter.paper.toIntegerFromText,
|
||||
order.orderType ?? "",
|
||||
);
|
||||
|
||||
await PrinterService()
|
||||
// ignore: use_build_context_synchronously
|
||||
.printWithPrinter(checkerPrinter, printValue, context);
|
||||
} catch (e) {
|
||||
log("Error printing checker: $e");
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(content: Text('Error printing checker: $e')),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// Kitchen printer
|
||||
if (kitchenPrinter != null) {
|
||||
try {
|
||||
final printValue = await PrintDataoutputs.instance.printKitchen(
|
||||
productQuantity,
|
||||
order.tableNumber!,
|
||||
order.orderNumber ?? "",
|
||||
authData.user?.name ?? "",
|
||||
kitchenPrinter.paper.toIntegerFromText,
|
||||
order.orderType ?? "",
|
||||
);
|
||||
|
||||
await PrinterService()
|
||||
.printWithPrinter(kitchenPrinter, printValue, context);
|
||||
} catch (e) {
|
||||
log("Error printing kitchen order: $e");
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(content: Text('Error printing kitchen order: $e')),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// Bar printer
|
||||
if (barPrinter != null) {
|
||||
try {
|
||||
final printValue = await PrintDataoutputs.instance.printBar(
|
||||
productQuantity,
|
||||
order.tableNumber ?? "",
|
||||
order.orderNumber ?? "",
|
||||
authData.user?.name ?? "",
|
||||
barPrinter.paper.toIntegerFromText,
|
||||
order.orderType ?? "",
|
||||
);
|
||||
|
||||
await PrinterService()
|
||||
.printWithPrinter(barPrinter, printValue, context);
|
||||
} catch (e) {
|
||||
log("Error printing bar order: $e");
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(content: Text('Error printing bar order: $e')),
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (outlet.businessType == BusinessType.ticketing) {
|
||||
final ticketPrinter =
|
||||
await ProductLocalDatasource.instance.getPrinterByCode('ticket');
|
||||
|
||||
final barcode = await generateBarcodeAsUint8List(order.orderNumber ?? "");
|
||||
|
||||
if (ticketPrinter != null) {
|
||||
try {
|
||||
final printValue = await PrintDataoutputs.instance.printTicket(
|
||||
order.totalAmount ?? 0,
|
||||
barcode,
|
||||
ticketPrinter.paper.toIntegerFromText,
|
||||
);
|
||||
|
||||
await PrinterService()
|
||||
// ignore: use_build_context_synchronously
|
||||
.printWithPrinter(ticketPrinter, printValue, context);
|
||||
} catch (e) {
|
||||
log("Error printing ticket: $e");
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(content: Text('Error printing ticket: $e')),
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> onPrintRecipt(
|
||||
context, {
|
||||
required Order order,
|
||||
required String paymentMethod,
|
||||
required int nominalBayar,
|
||||
required int kembalian,
|
||||
}) async {
|
||||
final receiptPrinter =
|
||||
await ProductLocalDatasource.instance.getPrinterByCode('receipt');
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final settings = await SettingsLocalDatasource().getTax();
|
||||
final outlet = await OutletLocalDatasource().get();
|
||||
|
||||
if (receiptPrinter != null) {
|
||||
try {
|
||||
final printValue = await PrintDataoutputs.instance.printOrderV4(
|
||||
order,
|
||||
authData.user?.name ?? "",
|
||||
paymentMethod,
|
||||
nominalBayar,
|
||||
kembalian,
|
||||
settings.value,
|
||||
receiptPrinter.paper.toIntegerFromText,
|
||||
order.orderType ?? "",
|
||||
outlet,
|
||||
);
|
||||
await PrinterService()
|
||||
.printWithPrinter(receiptPrinter, printValue, context);
|
||||
} catch (e) {
|
||||
log("Error printing receipt order: $e");
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(content: Text('Error printing receipt order: $e')),
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Future<Uint8List> generateBarcodeAsUint8List(String data) async {
|
||||
// 1. Buat barcode instance (code128, qrCode, dll)
|
||||
final barcode = Barcode.code128();
|
||||
|
||||
// 2. Siapkan canvas image dari package `image`
|
||||
final image = img.Image(width: 600, height: 200);
|
||||
|
||||
// 3. Gambar barcode ke canvas menggunakan barcode_image
|
||||
drawBarcode(
|
||||
image,
|
||||
barcode,
|
||||
data,
|
||||
);
|
||||
|
||||
// 4. Encode image ke Uint8List PNG
|
||||
return Uint8List.fromList(img.encodePng(image));
|
||||
}
|
||||
181
lib/core/network/dio_client.dart
Normal file
@ -0,0 +1,181 @@
|
||||
import 'package:awesome_dio_interceptor/awesome_dio_interceptor.dart';
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:enaklo_pos/core/extensions/build_context_ext.dart';
|
||||
import 'package:enaklo_pos/presentation/auth/login_page.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
|
||||
class DioClient {
|
||||
static final Dio _dio = Dio(BaseOptions(
|
||||
connectTimeout: const Duration(seconds: 10),
|
||||
receiveTimeout: const Duration(seconds: 10),
|
||||
headers: {
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
))
|
||||
..interceptors.add(AuthInterceptor())
|
||||
..interceptors.add(
|
||||
AwesomeDioInterceptor(
|
||||
logRequestTimeout: true,
|
||||
logRequestHeaders: true,
|
||||
logResponseHeaders: true,
|
||||
),
|
||||
);
|
||||
|
||||
static Dio get instance => _dio;
|
||||
}
|
||||
|
||||
class AuthInterceptor extends Interceptor {
|
||||
static final GlobalKey<NavigatorState> navigatorKey =
|
||||
GlobalKey<NavigatorState>();
|
||||
|
||||
@override
|
||||
void onRequest(
|
||||
RequestOptions options, RequestInterceptorHandler handler) async {
|
||||
// Add token to request headers
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
final token = prefs.getString('auth_token');
|
||||
|
||||
if (token != null) {
|
||||
options.headers['Authorization'] = 'Bearer $token';
|
||||
}
|
||||
|
||||
handler.next(options);
|
||||
}
|
||||
|
||||
@override
|
||||
void onResponse(Response response, ResponseInterceptorHandler handler) {
|
||||
handler.next(response);
|
||||
}
|
||||
|
||||
@override
|
||||
void onError(DioException err, ErrorInterceptorHandler handler) async {
|
||||
// Check if error is 401 (Unauthorized) - token expired
|
||||
if (err.response?.statusCode == 401) {
|
||||
await _handleTokenExpired();
|
||||
}
|
||||
|
||||
handler.next(err);
|
||||
}
|
||||
|
||||
Future<void> _handleTokenExpired() async {
|
||||
// Clear stored token
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
await prefs.remove('auth_token');
|
||||
await prefs.remove('refresh_token');
|
||||
await prefs.clear(); // Optional: clear all user data
|
||||
|
||||
// Navigate to login page
|
||||
final context = navigatorKey.currentContext;
|
||||
if (context != null) {
|
||||
// Option 1: Navigate and remove all previous routes
|
||||
context.pushReplacement(LoginPage());
|
||||
|
||||
// Option 2: If using GoRouter, uncomment below:
|
||||
// GoRouter.of(context).go('/login');
|
||||
|
||||
// Option 3: If using custom routing, uncomment below:
|
||||
// Navigator.of(context).pushAndRemoveUntil(
|
||||
// MaterialPageRoute(builder: (context) => LoginPage()),
|
||||
// (route) => false,
|
||||
// );
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Alternative: Dengan Refresh Token Logic
|
||||
class AuthInterceptorWithRefresh extends Interceptor {
|
||||
static final GlobalKey<NavigatorState> navigatorKey =
|
||||
GlobalKey<NavigatorState>();
|
||||
|
||||
@override
|
||||
void onRequest(
|
||||
RequestOptions options, RequestInterceptorHandler handler) async {
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
final token = prefs.getString('auth_token');
|
||||
|
||||
if (token != null) {
|
||||
options.headers['Authorization'] = 'Bearer $token';
|
||||
}
|
||||
|
||||
handler.next(options);
|
||||
}
|
||||
|
||||
@override
|
||||
void onError(DioException err, ErrorInterceptorHandler handler) async {
|
||||
if (err.response?.statusCode == 401) {
|
||||
// Try refresh token first
|
||||
final success = await _tryRefreshToken();
|
||||
|
||||
if (success) {
|
||||
// Retry the original request
|
||||
final response = await _retryRequest(err.requestOptions);
|
||||
handler.resolve(response);
|
||||
return;
|
||||
} else {
|
||||
// Refresh failed, redirect to login
|
||||
await _handleTokenExpired();
|
||||
}
|
||||
}
|
||||
|
||||
handler.next(err);
|
||||
}
|
||||
|
||||
Future<bool> _tryRefreshToken() async {
|
||||
try {
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
final refreshToken = prefs.getString('refresh_token');
|
||||
|
||||
if (refreshToken == null) return false;
|
||||
|
||||
final response = await Dio().post(
|
||||
'YOUR_REFRESH_TOKEN_ENDPOINT',
|
||||
data: {'refresh_token': refreshToken},
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
final newToken = response.data['access_token'];
|
||||
final newRefreshToken = response.data['refresh_token'];
|
||||
|
||||
await prefs.setString('auth_token', newToken);
|
||||
await prefs.setString('refresh_token', newRefreshToken);
|
||||
|
||||
return true;
|
||||
}
|
||||
} catch (e) {
|
||||
print('Refresh token failed: $e');
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
Future<Response> _retryRequest(RequestOptions options) async {
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
final token = prefs.getString('auth_token');
|
||||
|
||||
options.headers['Authorization'] = 'Bearer $token';
|
||||
|
||||
return await DioClient.instance.request(
|
||||
options.path,
|
||||
options: Options(
|
||||
method: options.method,
|
||||
headers: options.headers,
|
||||
),
|
||||
data: options.data,
|
||||
queryParameters: options.queryParameters,
|
||||
);
|
||||
}
|
||||
|
||||
Future<void> _handleTokenExpired() async {
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
await prefs.clear();
|
||||
|
||||
final context = navigatorKey.currentContext;
|
||||
if (context != null) {
|
||||
Navigator.of(context).pushNamedAndRemoveUntil(
|
||||
'/login',
|
||||
(route) => false,
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -38,7 +38,7 @@ class ItemSalesInvoice {
|
||||
|
||||
return HelperPdfService.saveDocument(
|
||||
name:
|
||||
'Enaklo POS | Item Sales Report | ${DateTime.now().millisecondsSinceEpoch}.pdf',
|
||||
'Apskel POS | Item Sales Report | ${DateTime.now().millisecondsSinceEpoch}.pdf',
|
||||
pdf: pdf);
|
||||
}
|
||||
|
||||
@ -48,7 +48,7 @@ class ItemSalesInvoice {
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
SizedBox(height: 1 * PdfPageFormat.cm),
|
||||
Text('Enaklo POS | Item Sales Report',
|
||||
Text('Apskel POS | Item Sales Report',
|
||||
style: TextStyle(
|
||||
fontSize: 20,
|
||||
fontWeight: FontWeight.bold,
|
||||
|
||||
@ -28,7 +28,8 @@ class RevenueInvoice {
|
||||
|
||||
// Load logo image
|
||||
log("Loading logo image...");
|
||||
final ByteData dataImage = await rootBundle.load('assets/images/logo.png');
|
||||
final ByteData dataImage =
|
||||
await rootBundle.load('assets/images/logo.png');
|
||||
final Uint8List bytes = dataImage.buffer.asUint8List();
|
||||
final image = pw.MemoryImage(bytes);
|
||||
log("Logo image loaded successfully, size: ${bytes.length} bytes");
|
||||
@ -49,7 +50,7 @@ class RevenueInvoice {
|
||||
log("Saving PDF document...");
|
||||
return HelperPdfService.saveDocument(
|
||||
name:
|
||||
'Enaklo POS | Summary Sales Report | ${DateTime.now().millisecondsSinceEpoch}.pdf',
|
||||
'Apskel POS | Summary Sales Report | ${DateTime.now().millisecondsSinceEpoch}.pdf',
|
||||
pdf: pdf,
|
||||
);
|
||||
} catch (e) {
|
||||
@ -69,7 +70,7 @@ class RevenueInvoice {
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
SizedBox(height: 1 * PdfPageFormat.cm),
|
||||
Text('Enaklo POS | Summary Sales Report',
|
||||
Text('Apskel POS | Summary Sales Report',
|
||||
style: TextStyle(
|
||||
fontSize: 20,
|
||||
fontWeight: FontWeight.bold,
|
||||
@ -125,7 +126,8 @@ class RevenueInvoice {
|
||||
buildText(
|
||||
title: 'Discount',
|
||||
titleStyle: TextStyle(fontWeight: FontWeight.normal),
|
||||
value: "- ${safeParseInt(summaryModel.totalDiscount).currencyFormatRp}",
|
||||
value:
|
||||
"- ${safeParseInt(summaryModel.totalDiscount).currencyFormatRp}",
|
||||
unite: true,
|
||||
textStyle: TextStyle(
|
||||
color: PdfColor.fromHex('#FF0000'),
|
||||
@ -147,7 +149,8 @@ class RevenueInvoice {
|
||||
titleStyle: TextStyle(
|
||||
fontWeight: FontWeight.normal,
|
||||
),
|
||||
value: safeParseInt(summaryModel.totalServiceCharge).currencyFormatRp,
|
||||
value:
|
||||
safeParseInt(summaryModel.totalServiceCharge).currencyFormatRp,
|
||||
unite: true,
|
||||
),
|
||||
Divider(),
|
||||
|
||||
@ -5,7 +5,7 @@ import 'package:enaklo_pos/core/extensions/int_ext.dart';
|
||||
import 'package:flutter/services.dart';
|
||||
|
||||
import 'package:enaklo_pos/core/utils/helper_pdf_service.dart';
|
||||
import 'package:enaklo_pos/data/models/response/order_remote_datasource.dart';
|
||||
import 'package:enaklo_pos/data/models/response/order_response_model.dart';
|
||||
import 'package:pdf/widgets.dart';
|
||||
import 'package:pdf/pdf.dart';
|
||||
import 'package:pdf/widgets.dart' as pw;
|
||||
@ -13,7 +13,7 @@ import 'package:pdf/widgets.dart' as pw;
|
||||
class TransactionSalesInvoice {
|
||||
static late Font ttf;
|
||||
static Future<File> generate(
|
||||
List<ItemOrder> itemOrders, String searchDateFormatted) async {
|
||||
List<Order> itemOrders, String searchDateFormatted) async {
|
||||
final pdf = Document();
|
||||
// var data = await rootBundle.load("assets/fonts/noto-sans.ttf");
|
||||
// ttf = Font.ttf(data);
|
||||
@ -38,7 +38,7 @@ class TransactionSalesInvoice {
|
||||
|
||||
return HelperPdfService.saveDocument(
|
||||
name:
|
||||
'Enaklo POS | Transaction Sales Report | ${DateTime.now().millisecondsSinceEpoch}.pdf',
|
||||
'Apskel POS | Transaction Sales Report | ${DateTime.now().millisecondsSinceEpoch}.pdf',
|
||||
pdf: pdf);
|
||||
}
|
||||
|
||||
@ -48,7 +48,7 @@ class TransactionSalesInvoice {
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
SizedBox(height: 1 * PdfPageFormat.cm),
|
||||
Text('Enaklo POS | Transaction Sales Report',
|
||||
Text('Apskel POS | Transaction Sales Report',
|
||||
style: TextStyle(
|
||||
fontSize: 20,
|
||||
fontWeight: FontWeight.bold,
|
||||
@ -70,7 +70,7 @@ class TransactionSalesInvoice {
|
||||
),
|
||||
]);
|
||||
|
||||
static Widget buildInvoice(List<ItemOrder> itemOrders) {
|
||||
static Widget buildInvoice(List<Order> itemOrders) {
|
||||
final headers = [
|
||||
'Total',
|
||||
'Sub Total',
|
||||
@ -81,12 +81,13 @@ class TransactionSalesInvoice {
|
||||
];
|
||||
final data = itemOrders.map((item) {
|
||||
return [
|
||||
item.total!.currencyFormatRp,
|
||||
item.subTotal!.currencyFormatRp,
|
||||
item.tax!.currencyFormatRp,
|
||||
int.parse(item.discountAmount!.replaceAll('.00', '')).currencyFormatRp,
|
||||
item.serviceCharge!.currencyFormatRp,
|
||||
item.transactionTime!.toFormattedDate2(),
|
||||
item.totalAmount!.currencyFormatRp,
|
||||
item.subtotal!.currencyFormatRp,
|
||||
item.taxAmount!.currencyFormatRp,
|
||||
int.parse(item.discountAmount!.toString().replaceAll('.00', ''))
|
||||
.currencyFormatRp,
|
||||
0,
|
||||
item.createdAt!.toFormattedDate2(),
|
||||
];
|
||||
}).toList();
|
||||
|
||||
|
||||
@ -1,5 +1,7 @@
|
||||
import 'dart:math';
|
||||
|
||||
import 'package:enaklo_pos/data/models/response/order_response_model.dart';
|
||||
import 'package:enaklo_pos/presentation/home/models/outlet_model.dart';
|
||||
import 'package:esc_pos_utils_plus/esc_pos_utils_plus.dart';
|
||||
import 'package:flutter/services.dart';
|
||||
import 'package:enaklo_pos/core/extensions/int_ext.dart';
|
||||
@ -35,7 +37,7 @@ class PrintDataoutputs {
|
||||
final total = totalPrice + pajak;
|
||||
|
||||
bytes += generator.reset();
|
||||
bytes += generator.text('Enaklo POS',
|
||||
bytes += generator.text('Apskel POS',
|
||||
styles: const PosStyles(
|
||||
bold: true,
|
||||
align: PosAlign.center,
|
||||
@ -60,12 +62,12 @@ class PrintDataoutputs {
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text:
|
||||
'${product.product.price!.toIntegerFromText.currencyFormatRp} x ${product.quantity}',
|
||||
'${product.product.price!.currencyFormatRp} x ${product.quantity}',
|
||||
width: 8,
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: '${product.product.price!.toIntegerFromText * product.quantity}'
|
||||
text: '${product.product.price! * product.quantity}'
|
||||
.toIntegerFromText
|
||||
.currencyFormatRp,
|
||||
width: 4,
|
||||
@ -202,7 +204,7 @@ class PrintDataoutputs {
|
||||
// bytes += generator.feed(3);
|
||||
// }
|
||||
|
||||
bytes += generator.text('Enaklo POS',
|
||||
bytes += generator.text('Apskel POS',
|
||||
styles: const PosStyles(
|
||||
bold: true,
|
||||
align: PosAlign.center,
|
||||
@ -328,8 +330,7 @@ class PrintDataoutputs {
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: (product.product.price!.toIntegerFromText * product.quantity)
|
||||
.currencyFormatRp,
|
||||
text: (product.product.price! * product.quantity).currencyFormatRp,
|
||||
width: 4,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
),
|
||||
@ -398,8 +399,7 @@ class PrintDataoutputs {
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: (products[0].product.price!.toIntegerFromText *
|
||||
products[0].quantity)
|
||||
text: (products[0].product.price! * products[0].quantity)
|
||||
.currencyFormatRp,
|
||||
width: 4,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
@ -430,8 +430,7 @@ class PrintDataoutputs {
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: (products[0].product.price!.toIntegerFromText *
|
||||
products[0].quantity)
|
||||
text: (products[0].product.price! * products[0].quantity)
|
||||
.currencyFormatRp,
|
||||
width: 4,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
@ -474,21 +473,21 @@ class PrintDataoutputs {
|
||||
}
|
||||
|
||||
Future<List<int>> printOrderV3(
|
||||
List<ProductQuantity> products,
|
||||
int totalQuantity,
|
||||
int totalPrice,
|
||||
String paymentMethod,
|
||||
int nominalBayar,
|
||||
int kembalian,
|
||||
int subTotal,
|
||||
int discount,
|
||||
int pajak,
|
||||
int serviceCharge,
|
||||
String namaKasir,
|
||||
String customerName,
|
||||
int paper,
|
||||
{int taxPercentage = 11, int serviceChargePercentage = 5}
|
||||
) async {
|
||||
List<ProductQuantity> products,
|
||||
int totalQuantity,
|
||||
int totalPrice,
|
||||
String paymentMethod,
|
||||
int nominalBayar,
|
||||
int kembalian,
|
||||
int subTotal,
|
||||
int discount,
|
||||
int pajak,
|
||||
int serviceCharge,
|
||||
String namaKasir,
|
||||
String customerName,
|
||||
int paper,
|
||||
{int taxPercentage = 11,
|
||||
int serviceChargePercentage = 5}) async {
|
||||
List<int> bytes = [];
|
||||
|
||||
final profile = await CapabilityProfile.load();
|
||||
@ -610,8 +609,8 @@ class PrintDataoutputs {
|
||||
styles: const PosStyles(bold: true, align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: '${product.product.price!.toIntegerFromText * product.quantity}'
|
||||
.currencyFormatRpV2,
|
||||
text:
|
||||
'${product.product.price! * product.quantity}'.currencyFormatRpV2,
|
||||
width: 4,
|
||||
styles: const PosStyles(bold: true, align: PosAlign.right),
|
||||
),
|
||||
@ -626,8 +625,7 @@ class PrintDataoutputs {
|
||||
final subTotalPrice = products.fold<int>(
|
||||
0,
|
||||
(previousValue, element) =>
|
||||
previousValue +
|
||||
(element.product.price!.toIntegerFromText * element.quantity));
|
||||
previousValue + (element.product.price! * element.quantity));
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: 'Subtotal $totalQuantity Product',
|
||||
@ -743,6 +741,265 @@ class PrintDataoutputs {
|
||||
return bytes;
|
||||
}
|
||||
|
||||
Future<List<int>> printOrderV4(
|
||||
Order order,
|
||||
String chashierName,
|
||||
String paymentMethod,
|
||||
int nominalBayar,
|
||||
int kembalian,
|
||||
int taxPercentage,
|
||||
int paper,
|
||||
String orderType,
|
||||
Outlet outlet,
|
||||
) async {
|
||||
List<int> bytes = [];
|
||||
|
||||
final profile = await CapabilityProfile.load();
|
||||
final generator =
|
||||
Generator(paper == 58 ? PaperSize.mm58 : PaperSize.mm80, profile);
|
||||
|
||||
bytes += generator.reset();
|
||||
|
||||
bytes += generator.text(outlet.name ?? "",
|
||||
styles: const PosStyles(
|
||||
bold: true,
|
||||
align: PosAlign.center,
|
||||
height: PosTextSize.size1,
|
||||
width: PosTextSize.size1,
|
||||
));
|
||||
|
||||
bytes += generator.text(outlet.address ?? "",
|
||||
styles: const PosStyles(bold: false, align: PosAlign.center));
|
||||
bytes += generator.text(outlet.phoneNumber ?? "",
|
||||
styles: const PosStyles(bold: false, align: PosAlign.center));
|
||||
|
||||
bytes += generator.text(
|
||||
paper == 80
|
||||
? '------------------------------------------------'
|
||||
: '--------------------------------',
|
||||
styles: const PosStyles(bold: false, align: PosAlign.center));
|
||||
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: DateFormat('dd MMM yyyy').format(DateTime.now()),
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: DateFormat('HH:mm').format(DateTime.now()),
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
),
|
||||
]);
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: 'Receipt Number',
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: 'JF-${DateFormat('yyyyMMddhhmm').format(DateTime.now())}',
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
),
|
||||
]);
|
||||
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: 'Order ID',
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: Random().nextInt(100000).toString(),
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
),
|
||||
]);
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: 'Bill Name',
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: order.metadata?['customer_name'] ?? '',
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
),
|
||||
]);
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: 'Collected By',
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: chashierName,
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
),
|
||||
]);
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: 'Pembayaran',
|
||||
width: 8,
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: paymentMethod,
|
||||
width: 4,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
),
|
||||
]);
|
||||
|
||||
bytes += generator.text(
|
||||
paper == 80
|
||||
? '------------------------------------------------'
|
||||
: '--------------------------------',
|
||||
styles: const PosStyles(bold: false, align: PosAlign.center));
|
||||
bytes += generator.text(orderType,
|
||||
styles: const PosStyles(bold: true, align: PosAlign.center));
|
||||
bytes += generator.text(
|
||||
paper == 80
|
||||
? '------------------------------------------------'
|
||||
: '--------------------------------',
|
||||
styles: const PosStyles(bold: false, align: PosAlign.center));
|
||||
for (final product in (order.orderItems ?? <OrderItem>[])) {
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: '${product.quantity} x ${product.productName}',
|
||||
width: 8,
|
||||
styles: const PosStyles(bold: true, align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: (product.totalPrice ?? 0).currencyFormatRpV2,
|
||||
width: 4,
|
||||
styles: const PosStyles(bold: true, align: PosAlign.right),
|
||||
),
|
||||
]);
|
||||
}
|
||||
bytes += generator.text(
|
||||
paper == 80
|
||||
? '------------------------------------------------'
|
||||
: '--------------------------------',
|
||||
styles: const PosStyles(bold: false, align: PosAlign.center));
|
||||
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: 'Subtotal ${order.orderItems?.length ?? "0"} Product',
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: (order.subtotal ?? 0).currencyFormatRpV2,
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
),
|
||||
]);
|
||||
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: 'Discount',
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: (order.discountAmount ?? 0).currencyFormatRpV2,
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
),
|
||||
]);
|
||||
|
||||
// Only show tax if it's greater than 0
|
||||
if ((order.taxAmount ?? 0) > 0) {
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: 'Tax PB1 ($taxPercentage%)',
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: (order.taxAmount ?? 0).currencyFormatRpV2,
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
),
|
||||
]);
|
||||
}
|
||||
|
||||
// Only show service charge if it's greater than 0
|
||||
// if (serviceCharge > 0) {
|
||||
// bytes += generator.row([
|
||||
// PosColumn(
|
||||
// text: 'Service Charge($serviceChargePercentage%)',
|
||||
// width: 6,
|
||||
// styles: const PosStyles(align: PosAlign.left),
|
||||
// ),
|
||||
// PosColumn(
|
||||
// text: serviceCharge.currencyFormatRpV2,
|
||||
// width: 6,
|
||||
// styles: const PosStyles(align: PosAlign.right),
|
||||
// ),
|
||||
// ]);
|
||||
// }
|
||||
bytes += generator.text(
|
||||
paper == 80
|
||||
? '------------------------------------------------'
|
||||
: '--------------------------------',
|
||||
styles: const PosStyles(bold: false, align: PosAlign.center));
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: 'Total',
|
||||
width: 6,
|
||||
styles: const PosStyles(bold: true, align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: '${order.totalAmount ?? ""}'.currencyFormatRpV2,
|
||||
width: 6,
|
||||
styles: const PosStyles(bold: true, align: PosAlign.right),
|
||||
),
|
||||
]);
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: 'Dibayar',
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: nominalBayar.currencyFormatRpV2,
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
),
|
||||
]);
|
||||
bytes += generator.row([
|
||||
PosColumn(
|
||||
text: 'Kembali',
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.left),
|
||||
),
|
||||
PosColumn(
|
||||
text: kembalian.currencyFormatRpV2,
|
||||
width: 6,
|
||||
styles: const PosStyles(align: PosAlign.right),
|
||||
),
|
||||
]);
|
||||
bytes += generator.text(
|
||||
paper == 80
|
||||
? '------------------------------------------------'
|
||||
: '--------------------------------',
|
||||
styles: const PosStyles(bold: false, align: PosAlign.center));
|
||||
// bytes += generator.text('Notes',
|
||||
// styles: const PosStyles(bold: false, align: PosAlign.center));
|
||||
// bytes += generator.text('Pass Wifi: fic14jilid2',
|
||||
// styles: const PosStyles(bold: false, align: PosAlign.center));
|
||||
// //terima kasih
|
||||
// bytes += generator.text('Terima Kasih',
|
||||
// styles: const PosStyles(bold: true, align: PosAlign.center));
|
||||
paper == 80 ? bytes += generator.feed(3) : bytes += generator.feed(1);
|
||||
bytes += generator.cut();
|
||||
return bytes;
|
||||
}
|
||||
|
||||
Future<List<int>> printQRIS(
|
||||
int totalPrice, Uint8List imageQris, int paper) async {
|
||||
List<int> bytes = [];
|
||||
@ -778,8 +1035,13 @@ class PrintDataoutputs {
|
||||
return bytes;
|
||||
}
|
||||
|
||||
Future<List<int>> printChecker(List<ProductQuantity> products,
|
||||
String tableName, String draftName, String cashierName, int paper, String orderType) async {
|
||||
Future<List<int>> printChecker(
|
||||
List<ProductQuantity> products,
|
||||
String tableName,
|
||||
String draftName,
|
||||
String cashierName,
|
||||
int paper,
|
||||
String orderType) async {
|
||||
List<int> bytes = [];
|
||||
|
||||
final profile = await CapabilityProfile.load();
|
||||
@ -908,8 +1170,13 @@ class PrintDataoutputs {
|
||||
return bytes;
|
||||
}
|
||||
|
||||
Future<List<int>> printKitchen(List<ProductQuantity> products,
|
||||
String tableNumber, String draftName, String cashierName, int paper, String orderType) async {
|
||||
Future<List<int>> printKitchen(
|
||||
List<ProductQuantity> products,
|
||||
String tableNumber,
|
||||
String draftName,
|
||||
String cashierName,
|
||||
int paper,
|
||||
String orderType) async {
|
||||
List<int> bytes = [];
|
||||
|
||||
final profile = await CapabilityProfile.load();
|
||||
@ -1134,4 +1401,39 @@ class PrintDataoutputs {
|
||||
|
||||
return bytes;
|
||||
}
|
||||
|
||||
Future<List<int>> printTicket(
|
||||
int totalPrice, Uint8List imageQris, int paper) async {
|
||||
List<int> bytes = [];
|
||||
|
||||
final profile = await CapabilityProfile.load();
|
||||
final generator =
|
||||
Generator(paper == 58 ? PaperSize.mm58 : PaperSize.mm80, profile);
|
||||
|
||||
final img.Image? orginalImage = img.decodeImage(imageQris);
|
||||
bytes += generator.reset();
|
||||
|
||||
// final Uint8List bytesData = data.buffer.asUint8List();
|
||||
// final img.Image? orginalImage = img.decodeImage(bytesData);
|
||||
// bytes += generator.reset();
|
||||
|
||||
bytes += generator.text('Scan Ticket',
|
||||
styles: const PosStyles(bold: false, align: PosAlign.center));
|
||||
bytes += generator.feed(2);
|
||||
if (orginalImage != null) {
|
||||
final img.Image grayscalledImage = img.grayscale(orginalImage);
|
||||
final img.Image resizedImage =
|
||||
img.copyResize(grayscalledImage, width: 240);
|
||||
bytes += generator.imageRaster(resizedImage, align: PosAlign.center);
|
||||
bytes += generator.feed(1);
|
||||
}
|
||||
|
||||
bytes += generator.text('Price : ${totalPrice.currencyFormatRp}',
|
||||
styles: const PosStyles(bold: false, align: PosAlign.center));
|
||||
|
||||
bytes += generator.feed(4);
|
||||
bytes += generator.cut();
|
||||
|
||||
return bytes;
|
||||
}
|
||||
}
|
||||
|
||||
187
lib/data/datasources/analytic_remote_datasource.dart
Normal file
@ -0,0 +1,187 @@
|
||||
import 'dart:developer';
|
||||
|
||||
import 'package:dartz/dartz.dart';
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:enaklo_pos/core/constants/variables.dart';
|
||||
import 'package:enaklo_pos/core/network/dio_client.dart';
|
||||
import 'package:enaklo_pos/data/datasources/auth_local_datasource.dart';
|
||||
import 'package:enaklo_pos/data/models/response/dashboard_analytic_response_model.dart';
|
||||
import 'package:enaklo_pos/data/models/response/payment_method_analytic_response_model.dart';
|
||||
import 'package:enaklo_pos/data/models/response/product_analytic_response_model.dart';
|
||||
import 'package:enaklo_pos/data/models/response/profit_loss_response_model.dart';
|
||||
import 'package:enaklo_pos/data/models/response/sales_analytic_response_model.dart';
|
||||
import 'package:intl/intl.dart';
|
||||
|
||||
class AnalyticRemoteDatasource {
|
||||
final Dio dio = DioClient.instance;
|
||||
|
||||
Future<Either<String, PaymentMethodAnalyticResponseModel>> getPaymentMethod({
|
||||
required DateTime dateFrom,
|
||||
required DateTime dateTo,
|
||||
}) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final headers = {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
};
|
||||
|
||||
try {
|
||||
final response = await dio.get(
|
||||
'${Variables.baseUrl}/api/v1/analytics/payment-methods',
|
||||
queryParameters: {
|
||||
'date_from': DateFormat('dd-MM-yyyy').format(dateFrom),
|
||||
'date_to': DateFormat('dd-MM-yyyy').format(dateTo),
|
||||
},
|
||||
options: Options(headers: headers),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return right(PaymentMethodAnalyticResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return left('Terjadi Kesalahan, Coba lagi nanti.');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log('Dio error: ${e.message}');
|
||||
return left(e.response?.data.toString() ?? e.message ?? 'Unknown error');
|
||||
} catch (e) {
|
||||
log('Unexpected error: $e');
|
||||
return left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, SalesAnalyticResponseModel>> getSales({
|
||||
required DateTime dateFrom,
|
||||
required DateTime dateTo,
|
||||
}) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final headers = {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
};
|
||||
|
||||
try {
|
||||
final response = await dio.get(
|
||||
'${Variables.baseUrl}/api/v1/analytics/sales',
|
||||
queryParameters: {
|
||||
'date_from': DateFormat('dd-MM-yyyy').format(dateFrom),
|
||||
'date_to': DateFormat('dd-MM-yyyy').format(dateTo),
|
||||
},
|
||||
options: Options(headers: headers),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return right(SalesAnalyticResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return left('Terjadi Kesalahan, Coba lagi nanti.');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log('Dio error: ${e.message}');
|
||||
return left(e.response?.data.toString() ?? e.message ?? 'Unknown error');
|
||||
} catch (e) {
|
||||
log('Unexpected error: $e');
|
||||
return left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, ProductAnalyticResponseModel>> getProduct({
|
||||
required DateTime dateFrom,
|
||||
required DateTime dateTo,
|
||||
}) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final headers = {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
};
|
||||
|
||||
try {
|
||||
final response = await dio.get(
|
||||
'${Variables.baseUrl}/api/v1/analytics/products',
|
||||
queryParameters: {
|
||||
'date_from': DateFormat('dd-MM-yyyy').format(dateFrom),
|
||||
'date_to': DateFormat('dd-MM-yyyy').format(dateTo),
|
||||
},
|
||||
options: Options(headers: headers),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return right(ProductAnalyticResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return left('Terjadi Kesalahan, Coba lagi nanti.');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log('Dio error: ${e.message}');
|
||||
return left(e.response?.data.toString() ?? e.message ?? 'Unknown error');
|
||||
} catch (e) {
|
||||
log('Unexpected error: $e');
|
||||
return left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, DashboardAnalyticResponseModel>> getDashboard({
|
||||
required DateTime dateFrom,
|
||||
required DateTime dateTo,
|
||||
}) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final headers = {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
};
|
||||
|
||||
try {
|
||||
final response = await dio.get(
|
||||
'${Variables.baseUrl}/api/v1/analytics/dashboard',
|
||||
queryParameters: {
|
||||
'date_from': DateFormat('dd-MM-yyyy').format(dateFrom),
|
||||
'date_to': DateFormat('dd-MM-yyyy').format(dateTo),
|
||||
},
|
||||
options: Options(headers: headers),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return right(DashboardAnalyticResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return left('Terjadi Kesalahan, Coba lagi nanti.');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log('Dio error: ${e.message}');
|
||||
return left(e.response?.data.toString() ?? e.message ?? 'Unknown error');
|
||||
} catch (e) {
|
||||
log('Unexpected error: $e');
|
||||
return left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, ProfitLossResponseModel>> getProfitLoss({
|
||||
required DateTime dateFrom,
|
||||
required DateTime dateTo,
|
||||
}) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final headers = {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
};
|
||||
|
||||
try {
|
||||
final response = await dio.get(
|
||||
'${Variables.baseUrl}/api/v1/analytics/profit-loss',
|
||||
queryParameters: {
|
||||
'date_from': DateFormat('dd-MM-yyyy').format(dateFrom),
|
||||
'date_to': DateFormat('dd-MM-yyyy').format(dateTo),
|
||||
},
|
||||
options: Options(headers: headers),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return right(ProfitLossResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return left('Terjadi Kesalahan, Coba lagi nanti.');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log('Dio error: ${e.message}');
|
||||
return left(e.response?.data.toString() ?? e.message ?? 'Unknown error');
|
||||
} catch (e) {
|
||||
log('Unexpected error: $e');
|
||||
return left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -1,10 +1,16 @@
|
||||
import 'dart:developer';
|
||||
|
||||
import 'package:enaklo_pos/data/models/response/auth_response_model.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
|
||||
class AuthLocalDataSource {
|
||||
Future<void> saveAuthData(AuthResponseModel authResponseModel) async {
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
await prefs.setString('auth_data', authResponseModel.toJson());
|
||||
try {
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
await prefs.setString('auth_data', authResponseModel.toJson());
|
||||
} catch (e) {
|
||||
log('Error saving auth data: $e');
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> removeAuthData() async {
|
||||
@ -16,6 +22,8 @@ class AuthLocalDataSource {
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
final authData = prefs.getString('auth_data');
|
||||
|
||||
log('Auth data: $authData');
|
||||
|
||||
return AuthResponseModel.fromJson(authData!);
|
||||
}
|
||||
|
||||
|
||||
@ -1,44 +1,67 @@
|
||||
import 'dart:developer';
|
||||
|
||||
import 'package:dartz/dartz.dart';
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:enaklo_pos/core/constants/variables.dart';
|
||||
import 'package:enaklo_pos/core/network/dio_client.dart';
|
||||
import 'package:enaklo_pos/data/datasources/auth_local_datasource.dart';
|
||||
import 'package:enaklo_pos/data/models/response/auth_response_model.dart';
|
||||
import 'package:http/http.dart' as http;
|
||||
|
||||
class AuthRemoteDatasource {
|
||||
final Dio dio = DioClient.instance;
|
||||
Future<Either<String, AuthResponseModel>> login(
|
||||
String email, String password) async {
|
||||
final url = Uri.parse('${Variables.baseUrl}/api/login');
|
||||
final response = await http.post(
|
||||
url,
|
||||
body: {
|
||||
'email': email,
|
||||
'password': password,
|
||||
},
|
||||
);
|
||||
final url = '${Variables.baseUrl}/api/v1/auth/login';
|
||||
log(url);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(AuthResponseModel.fromJson(response.body));
|
||||
} else {
|
||||
return const Left('Failed to login');
|
||||
try {
|
||||
final response = await dio.post(
|
||||
url,
|
||||
data: {
|
||||
'email': email,
|
||||
'password': password,
|
||||
},
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(AuthResponseModel.fromMap(response.data['data']));
|
||||
} else {
|
||||
return const Left('Failed to login');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(e.response?.data['message'] ?? 'Login gagal');
|
||||
} catch (e) {
|
||||
log("Unexpected error: $e");
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
|
||||
//logout
|
||||
Future<Either<String, bool>> logout() async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = Uri.parse('${Variables.baseUrl}/api/logout');
|
||||
final response = await http.post(
|
||||
url,
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
);
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/auth/logout';
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return const Right(true);
|
||||
} else {
|
||||
return const Left('Failed to logout');
|
||||
final response = await dio.post(
|
||||
url,
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return const Right(true);
|
||||
} else {
|
||||
return const Left('Failed to logout');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
return Left(e.response?.data['message'] ?? 'Logout gagal');
|
||||
} catch (e) {
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@ -1,27 +1,48 @@
|
||||
import 'dart:developer';
|
||||
|
||||
import 'package:dartz/dartz.dart';
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:enaklo_pos/core/constants/variables.dart';
|
||||
import 'package:enaklo_pos/core/network/dio_client.dart';
|
||||
import 'package:enaklo_pos/data/datasources/auth_local_datasource.dart';
|
||||
import 'package:enaklo_pos/data/models/response/category_response_model.dart';
|
||||
import 'package:http/http.dart' as http;
|
||||
|
||||
class CategoryRemoteDatasource {
|
||||
Future<Either<String, CategroyResponseModel>> getCategories() async {
|
||||
final Dio dio = DioClient.instance;
|
||||
|
||||
Future<Either<String, CategoryResponseModel>> getCategories({
|
||||
int page = 1,
|
||||
int limit = 10,
|
||||
bool isActive = true,
|
||||
}) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final Map<String, String> headers = {
|
||||
final headers = {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
};
|
||||
final response = await http.get(
|
||||
Uri.parse('${Variables.baseUrl}/api/api-categories'),
|
||||
headers: headers);
|
||||
log(response.statusCode.toString());
|
||||
log(response.body);
|
||||
if (response.statusCode == 200) {
|
||||
return right(CategroyResponseModel.fromJson(response.body));
|
||||
} else {
|
||||
return left(response.body);
|
||||
|
||||
try {
|
||||
final response = await dio.get(
|
||||
'${Variables.baseUrl}/api/v1/categories',
|
||||
queryParameters: {
|
||||
'page': page,
|
||||
'limit': limit,
|
||||
'is_active': isActive,
|
||||
},
|
||||
options: Options(headers: headers),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return right(CategoryResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return left(response.data.toString());
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log('Dio error: ${e.message}');
|
||||
return left(e.response?.data.toString() ?? e.message ?? 'Unknown error');
|
||||
} catch (e) {
|
||||
log('Unexpected error: $e');
|
||||
return left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
101
lib/data/datasources/customer_remote_datasource.dart
Normal file
@ -0,0 +1,101 @@
|
||||
import 'dart:developer';
|
||||
import 'package:dartz/dartz.dart';
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:enaklo_pos/core/network/dio_client.dart';
|
||||
import 'package:enaklo_pos/data/models/response/customer_response_model.dart';
|
||||
import '../../core/constants/variables.dart';
|
||||
import 'auth_local_datasource.dart';
|
||||
|
||||
class CustomerRemoteDataSource {
|
||||
final Dio dio = DioClient.instance;
|
||||
|
||||
Future<Either<String, CustomerResponseModel>> getCustomers({
|
||||
int page = 1,
|
||||
int limit = Variables.defaultLimit,
|
||||
}) async {
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/customers';
|
||||
|
||||
final response = await dio.get(
|
||||
url,
|
||||
queryParameters: {
|
||||
'page': page,
|
||||
'limit': limit,
|
||||
'organization_id': authData.user?.organizationId,
|
||||
},
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(CustomerResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return const Left('Failed to get customers');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(e.response?.data['message'] ?? 'Gagal mengambil customer');
|
||||
} catch (e) {
|
||||
log("Unexpected error: $e");
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, bool>> createCustomer({
|
||||
required String name,
|
||||
required String phone,
|
||||
required String address,
|
||||
required String email,
|
||||
required bool isActive,
|
||||
}) async {
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/customers';
|
||||
|
||||
Map<String, dynamic> data = {
|
||||
'name': name,
|
||||
'is_active': isActive,
|
||||
};
|
||||
|
||||
if (phone.isNotEmpty) {
|
||||
data['phone'] = phone;
|
||||
}
|
||||
|
||||
if (address.isNotEmpty) {
|
||||
data['address'] = address;
|
||||
}
|
||||
|
||||
if (email.isNotEmpty) {
|
||||
data['email'] = email;
|
||||
}
|
||||
|
||||
final response = await dio.post(
|
||||
url,
|
||||
data: data,
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200 || response.statusCode == 201) {
|
||||
return const Right(true);
|
||||
} else {
|
||||
return const Left('Failed to create customer ');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(e.response?.data['message'] ?? 'Gagal membuat pelanggan');
|
||||
} catch (e) {
|
||||
log("Unexpected error: $e");
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
}
|
||||
15
lib/data/datasources/delivery_local_datasource.dart
Normal file
@ -0,0 +1,15 @@
|
||||
import 'package:enaklo_pos/core/assets/assets.gen.dart';
|
||||
import 'package:enaklo_pos/data/models/response/delivery_response_model.dart';
|
||||
|
||||
List<DeliveryModel> deliveries = [
|
||||
DeliveryModel(
|
||||
id: 'gojek',
|
||||
name: 'Gojek',
|
||||
imageUrl: Assets.images.gojek.path,
|
||||
),
|
||||
DeliveryModel(
|
||||
id: 'grab',
|
||||
name: 'Grab',
|
||||
imageUrl: Assets.images.grab.path,
|
||||
),
|
||||
];
|
||||
53
lib/data/datasources/file_remote_datasource.dart
Normal file
@ -0,0 +1,53 @@
|
||||
import 'package:dartz/dartz.dart';
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:enaklo_pos/core/constants/variables.dart';
|
||||
import 'package:enaklo_pos/core/network/dio_client.dart';
|
||||
import 'package:enaklo_pos/data/datasources/auth_local_datasource.dart';
|
||||
import 'package:enaklo_pos/data/models/response/file_response_model.dart';
|
||||
|
||||
class FileRemoteDataSource {
|
||||
final Dio dio = DioClient.instance;
|
||||
|
||||
Future<Either<String, FileResponseModel>> uploadFile({
|
||||
required String filePath,
|
||||
required String fileType,
|
||||
required String description,
|
||||
}) async {
|
||||
final url = '${Variables.baseUrl}/api/v1/files/upload';
|
||||
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
|
||||
// Membuat FormData
|
||||
final formData = FormData.fromMap({
|
||||
'file': await MultipartFile.fromFile(filePath,
|
||||
filename: filePath.split('/').last),
|
||||
'file_type': fileType,
|
||||
'description': description,
|
||||
});
|
||||
|
||||
final response = await dio.post(
|
||||
url,
|
||||
data: formData,
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
// Content-Type otomatis diatur oleh Dio untuk FormData
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 201 || response.statusCode == 200) {
|
||||
// Misal response.data['url'] adalah URL file yang diupload
|
||||
return Right(FileResponseModel.fromJson(response.data));
|
||||
} else {
|
||||
return Left('Upload gagal: ${response.statusMessage}');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
return Left(e.response?.data['message'] ?? 'Upload gagal');
|
||||
} catch (e) {
|
||||
return Left('Unexpected error: $e');
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -1,17 +1,25 @@
|
||||
import 'dart:convert';
|
||||
import 'dart:developer';
|
||||
|
||||
import 'dart:convert';
|
||||
|
||||
import 'package:dartz/dartz.dart';
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:enaklo_pos/core/constants/variables.dart';
|
||||
import 'package:enaklo_pos/core/network/dio_client.dart';
|
||||
import 'package:enaklo_pos/data/datasources/auth_local_datasource.dart';
|
||||
import 'package:enaklo_pos/data/models/response/order_remote_datasource.dart';
|
||||
import 'package:enaklo_pos/data/models/request/payment_request.dart';
|
||||
import 'package:enaklo_pos/data/models/response/order_response_model.dart';
|
||||
import 'package:enaklo_pos/data/models/response/payment_method_response_model.dart';
|
||||
import 'package:enaklo_pos/data/models/response/payment_methods_response_model.dart';
|
||||
import 'package:enaklo_pos/data/models/response/payment_response_model.dart';
|
||||
import 'package:enaklo_pos/data/models/response/summary_response_model.dart';
|
||||
import 'package:enaklo_pos/presentation/home/models/order_model.dart';
|
||||
import 'package:enaklo_pos/presentation/home/models/order_request.dart';
|
||||
import 'package:http/http.dart' as http;
|
||||
import 'package:intl/intl.dart';
|
||||
|
||||
class OrderRemoteDatasource {
|
||||
final Dio dio = DioClient.instance;
|
||||
|
||||
//save order to remote server
|
||||
Future<bool> saveOrder(OrderModel orderModel) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
@ -69,7 +77,8 @@ class OrderRemoteDatasource {
|
||||
print("✅ getOrderByRangeDate API call successful");
|
||||
return Right(OrderResponseModel.fromJson(response.body));
|
||||
} else {
|
||||
print("❌ getOrderByRangeDate API call failed - Status: ${response.statusCode}");
|
||||
print(
|
||||
"❌ getOrderByRangeDate API call failed - Status: ${response.statusCode}");
|
||||
print("❌ Error Response: ${response.body}");
|
||||
return const Left("Failed Load Data");
|
||||
}
|
||||
@ -102,7 +111,8 @@ class OrderRemoteDatasource {
|
||||
print("✅ getSummaryByRangeDate API call successful");
|
||||
return Right(SummaryResponseModel.fromJson(response.body));
|
||||
} else {
|
||||
print("❌ getSummaryByRangeDate API call failed - Status: ${response.statusCode}");
|
||||
print(
|
||||
"❌ getSummaryByRangeDate API call failed - Status: ${response.statusCode}");
|
||||
print("❌ Error Response: ${response.body}");
|
||||
return const Left("Failed Load Data");
|
||||
}
|
||||
@ -112,7 +122,8 @@ class OrderRemoteDatasource {
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, PaymentMethodResponseModel>> getPaymentMethodByRangeDate(
|
||||
Future<Either<String, PaymentMethodResponseModel>>
|
||||
getPaymentMethodByRangeDate(
|
||||
String startDate,
|
||||
String endDate,
|
||||
) async {
|
||||
@ -134,7 +145,8 @@ class OrderRemoteDatasource {
|
||||
print("✅ getPaymentMethodByRangeDate API call successful");
|
||||
return Right(PaymentMethodResponseModel.fromJson(response.body));
|
||||
} else {
|
||||
print("❌ getPaymentMethodByRangeDate API call failed - Status: ${response.statusCode}");
|
||||
print(
|
||||
"❌ getPaymentMethodByRangeDate API call failed - Status: ${response.statusCode}");
|
||||
print("❌ Error Response: ${response.body}");
|
||||
return const Left("Failed Load Payment Method Data");
|
||||
}
|
||||
@ -171,7 +183,8 @@ class OrderRemoteDatasource {
|
||||
print("✅ addOrderItems API call successful");
|
||||
return const Right(true);
|
||||
} else {
|
||||
print("❌ addOrderItems API call failed - Status: ${response.statusCode}");
|
||||
print(
|
||||
"❌ addOrderItems API call failed - Status: ${response.statusCode}");
|
||||
print("❌ Error Response: ${response.body}");
|
||||
return Left("Failed to add order items: ${response.body}");
|
||||
}
|
||||
@ -180,4 +193,432 @@ class OrderRemoteDatasource {
|
||||
return Left("Failed: $e");
|
||||
}
|
||||
}
|
||||
|
||||
// New Api
|
||||
Future<Either<String, OrderDetailResponseModel>> createOrder(
|
||||
OrderRequestModel orderModel,
|
||||
) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/orders';
|
||||
|
||||
try {
|
||||
final response = await dio.post(
|
||||
url,
|
||||
data: orderModel.toMap(),
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(OrderDetailResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return const Left('Gagal membuat pesanan');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
final errorMessage =
|
||||
e.response?.data['message'] ?? 'Terjadi kesalahan, coba lagi nanti.';
|
||||
log("💥 Dio error: ${e.message}");
|
||||
log("💥 Dio response: ${e.response?.data}");
|
||||
return Left(errorMessage);
|
||||
} catch (e) {
|
||||
log("💥 Unexpected error: $e");
|
||||
return const Left('Terjadi kesalahan tak terduga');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, PaymentSuccessResponseModel>> createPayment(
|
||||
PaymentRequestModel orderModel) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/payments';
|
||||
|
||||
try {
|
||||
final response = await dio.post(
|
||||
url,
|
||||
data: orderModel.toMap(),
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(PaymentSuccessResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return const Left('Gagal membuat pembayaran');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
final errorMessage =
|
||||
e.response?.data['message'] ?? 'Terjadi kesalahan, coba lagi nanti.';
|
||||
log("💥 Dio error: ${e.message}");
|
||||
log("💥 Dio response: ${e.response?.data}");
|
||||
return Left(errorMessage);
|
||||
} catch (e) {
|
||||
log("💥 Unexpected error: $e");
|
||||
return const Left('Terjadi kesalahan tak terduga');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, OrderResponseModel>> getOrder({
|
||||
int page = 1,
|
||||
int limit = Variables.defaultLimit,
|
||||
String status = 'completed',
|
||||
required DateTime dateFrom,
|
||||
required DateTime dateTo,
|
||||
String? search,
|
||||
}) async {
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
|
||||
Map<String, dynamic> params = {
|
||||
'page': page,
|
||||
'limit': limit,
|
||||
'status': status,
|
||||
'date_from': DateFormat('dd-MM-yyyy').format(dateFrom),
|
||||
'date_to': DateFormat('dd-MM-yyyy').format(dateTo),
|
||||
};
|
||||
|
||||
if (search != null && search.isNotEmpty) {
|
||||
params['search'] = search;
|
||||
}
|
||||
|
||||
final response = await dio.get(
|
||||
'${Variables.baseUrl}/api/v1/orders',
|
||||
queryParameters: params,
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(OrderResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
log("❌ getOrderByRangeDate API call failed - Status: ${response.statusCode}");
|
||||
return const Left("Failed Load Data");
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
final errorMessage = 'Something went wrong';
|
||||
log("💥 Dio error: ${e.message}");
|
||||
log("💥 Dio response: ${e.response?.data}");
|
||||
return Left(errorMessage);
|
||||
} catch (e) {
|
||||
log("💥 Unexpected error: $e");
|
||||
return Left("Unexpected Error: $e");
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, OrderDetailResponseModel>> getOrderById(
|
||||
{required String orderId}) async {
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final response = await dio.get(
|
||||
'${Variables.baseUrl}/api/v1/orders/$orderId',
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(OrderDetailResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
log("❌ OrderDetailResponseModel API call failed - Status: ${response.statusCode}");
|
||||
return const Left("Failed Load Data");
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
final errorMessage = 'Something went wrong';
|
||||
log("💥 Dio error: ${e.message}");
|
||||
log("💥 Dio response: ${e.response?.data}");
|
||||
return Left(errorMessage);
|
||||
} catch (e) {
|
||||
log("💥 Unexpected error: $e");
|
||||
return Left("Unexpected Error: $e");
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, OrderDetailResponseModel>> createOrderWithPayment(
|
||||
OrderRequestModel orderModel,
|
||||
PaymentMethod payment,
|
||||
) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/orders';
|
||||
|
||||
try {
|
||||
final response = await dio.post(
|
||||
url,
|
||||
data: orderModel.toMap(),
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
final data = OrderDetailResponseModel.fromMap(response.data);
|
||||
final orderData = data.data;
|
||||
final paymentRequest = PaymentRequestModel(
|
||||
orderId: orderData?.id,
|
||||
amount: orderData?.totalAmount,
|
||||
paymentMethodId: payment.id,
|
||||
splitDescription: '',
|
||||
splitNumber: 1,
|
||||
splitTotal: 1,
|
||||
paymentOrderItems: orderData?.orderItems
|
||||
?.map((item) => PaymentOrderItemModel(
|
||||
amount: item.totalPrice,
|
||||
orderItemId: item.id,
|
||||
))
|
||||
.toList(),
|
||||
);
|
||||
|
||||
createPayment(paymentRequest);
|
||||
return Right(data);
|
||||
} else {
|
||||
return const Left('Gagal membuat pesanan');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
final errorMessage =
|
||||
e.response?.data['message'] ?? 'Terjadi kesalahan, coba lagi nanti.';
|
||||
log("💥 Dio error: ${e.message}");
|
||||
log("💥 Dio response: ${e.response?.data}");
|
||||
return Left(errorMessage);
|
||||
} catch (e) {
|
||||
log("💥 Unexpected error: $e");
|
||||
return const Left('Terjadi kesalahan tak terduga');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, bool>> addToOrder({
|
||||
required String orderId,
|
||||
required List<OrderItemRequest> orderItems,
|
||||
}) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/orders/$orderId/add-items';
|
||||
|
||||
try {
|
||||
final response = await dio.post(
|
||||
url,
|
||||
data: {
|
||||
"order_items": orderItems.map((item) => item.toMap()).toList(),
|
||||
'notes': '',
|
||||
},
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(true);
|
||||
} else {
|
||||
return const Left('Gagal menambahkan pesanan pesanan');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
final errorMessage =
|
||||
e.response?.data['message'] ?? 'Terjadi kesalahan, coba lagi nanti.';
|
||||
log("💥 Dio error: ${e.message}");
|
||||
log("💥 Dio response: ${e.response?.data}");
|
||||
return Left(errorMessage);
|
||||
} catch (e) {
|
||||
log("💥 Unexpected error: $e");
|
||||
return const Left('Terjadi kesalahan, coba lagi nanti.');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, bool>> refund({
|
||||
required String orderId,
|
||||
required String reason,
|
||||
required List<OrderItem> orderItems,
|
||||
}) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/orders/$orderId/refund';
|
||||
|
||||
final int refundAmount = orderItems.fold(
|
||||
0,
|
||||
(sum, item) => sum + ((item.unitPrice ?? 0) * (item.quantity ?? 0)),
|
||||
);
|
||||
try {
|
||||
final response = await dio.post(
|
||||
url,
|
||||
data: {
|
||||
'refund_amount': refundAmount,
|
||||
"order_items": orderItems
|
||||
.map((item) => {
|
||||
'order_item_id': item.id,
|
||||
"refund_quantity": item.quantity,
|
||||
"refund_amount": item.totalPrice,
|
||||
"reason": ""
|
||||
})
|
||||
.toList(),
|
||||
'reason': reason,
|
||||
},
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(true);
|
||||
} else {
|
||||
return const Left('Gagal refund');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
final errorMessage =
|
||||
e.response?.data['message'] ?? 'Terjadi kesalahan, coba lagi nanti.';
|
||||
log("💥 Dio error: ${e.message}");
|
||||
log("💥 Dio response: ${e.response?.data}");
|
||||
return Left(errorMessage);
|
||||
} catch (e) {
|
||||
log("💥 Unexpected error: $e");
|
||||
return const Left('Terjadi kesalahan tak terduga');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, bool>> voidOrder({
|
||||
required String orderId,
|
||||
required String reason,
|
||||
String type = "ITEM", // TYPE: ALL, ITEM
|
||||
required List<OrderItem> orderItems,
|
||||
}) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/orders/void';
|
||||
|
||||
try {
|
||||
final response = await dio.post(
|
||||
url,
|
||||
data: {
|
||||
'order_id': orderId,
|
||||
'type': orderItems.isEmpty ? "ALL" : type,
|
||||
'reason': reason,
|
||||
"items": orderItems
|
||||
.map((item) => {
|
||||
'order_item_id': item.id,
|
||||
"quantity": item.quantity,
|
||||
})
|
||||
.toList(),
|
||||
},
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(true);
|
||||
} else {
|
||||
return const Left('Gagal refund');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
final errorMessage =
|
||||
e.response?.data['message'] ?? 'Terjadi kesalahan, coba lagi nanti.';
|
||||
log("💥 Dio error: ${e.message}");
|
||||
log("💥 Dio response: ${e.response?.data}");
|
||||
return Left(errorMessage);
|
||||
} catch (e) {
|
||||
log("💥 Unexpected error: $e");
|
||||
return const Left('Terjadi kesalahan tak terduga');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, bool>> refundPayment({
|
||||
required String orderId,
|
||||
required String reason,
|
||||
required int refundAmount,
|
||||
}) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/orders/$orderId/refund';
|
||||
|
||||
try {
|
||||
final response = await dio.post(
|
||||
url,
|
||||
data: {
|
||||
'refund_amount': refundAmount,
|
||||
'reason': reason,
|
||||
},
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(true);
|
||||
} else {
|
||||
return const Left('Gagal refund');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
final errorMessage = 'Terjadi kesalahan coba lagi nanti';
|
||||
log("💥 Dio error: ${e.message}");
|
||||
log("💥 Dio response: ${e.response?.data}");
|
||||
return Left(errorMessage);
|
||||
} catch (e) {
|
||||
log("💥 Unexpected error: $e");
|
||||
return const Left('Terjadi kesalahan tak terduga');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, PaymentSuccessResponseModel>> createPaymentSplitBill(
|
||||
PaymentSplitBillRequest request) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/orders/split-bill';
|
||||
|
||||
try {
|
||||
final response = await dio.post(
|
||||
url,
|
||||
data: request.toMap(),
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(PaymentSuccessResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return const Left('Gagal membuat pembayaran');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
final errorMessage =
|
||||
e.response?.data['message'] ?? 'Terjadi kesalahan, coba lagi nanti.';
|
||||
log("💥 Dio error: ${e.message}");
|
||||
log("💥 Dio response: ${e.response?.data}");
|
||||
return Left(errorMessage);
|
||||
} catch (e) {
|
||||
log("💥 Unexpected error: $e");
|
||||
return const Left('Terjadi kesalahan tak terduga');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
30
lib/data/datasources/outlet_local_datasource.dart
Normal file
@ -0,0 +1,30 @@
|
||||
import 'dart:developer';
|
||||
|
||||
import 'package:enaklo_pos/presentation/home/models/outlet_model.dart';
|
||||
import 'package:shared_preferences/shared_preferences.dart';
|
||||
|
||||
class OutletLocalDatasource {
|
||||
Future<void> save(Outlet outlet) async {
|
||||
try {
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
await prefs.setString('outlet', outlet.toJson());
|
||||
log('Outlet Local Data: ${outlet.toJson()}');
|
||||
} catch (e) {
|
||||
log('Error saving outlet: $e');
|
||||
}
|
||||
}
|
||||
|
||||
Future<void> remove() async {
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
await prefs.remove('outlet');
|
||||
}
|
||||
|
||||
Future<Outlet> get() async {
|
||||
final prefs = await SharedPreferences.getInstance();
|
||||
final outlet = prefs.getString('outlet');
|
||||
|
||||
log('Outlet Local Data: $outlet');
|
||||
|
||||
return Outlet.fromJson(outlet!);
|
||||
}
|
||||
}
|
||||
96
lib/data/datasources/outlet_remote_data_source.dart
Normal file
@ -0,0 +1,96 @@
|
||||
import 'dart:developer';
|
||||
import 'package:dartz/dartz.dart';
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:enaklo_pos/core/network/dio_client.dart';
|
||||
import 'package:enaklo_pos/data/datasources/outlet_local_datasource.dart';
|
||||
import 'package:enaklo_pos/data/datasources/settings_local_datasource.dart';
|
||||
import 'package:enaklo_pos/presentation/home/models/outlet_model.dart';
|
||||
import 'package:enaklo_pos/presentation/setting/models/tax_model.dart';
|
||||
import '../../core/constants/variables.dart';
|
||||
import 'auth_local_datasource.dart';
|
||||
|
||||
class OutletRemoteDataSource {
|
||||
final Dio dio = DioClient.instance;
|
||||
|
||||
Future<Either<String, OutletResponse>> getOutlets() async {
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
|
||||
final url = '${Variables.baseUrl}/api/v1/outlets/list';
|
||||
|
||||
final response = await dio.get(
|
||||
url,
|
||||
queryParameters: {
|
||||
'organization_id': authData.user?.organizationId,
|
||||
},
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
final data = OutletResponse.fromMap(response.data);
|
||||
return Right(data);
|
||||
} else {
|
||||
return const Left('Failed to get outlets');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(e.response?.data['message'] ?? 'Gagal mengambil outlet');
|
||||
} catch (e) {
|
||||
log("Unexpected error: $e");
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, OutletDetailResponse>> currentOutlet() async {
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
|
||||
if (authData.user?.outletId == null) {
|
||||
return const Left('Kamu belum memiliki bergabung dengan outlet');
|
||||
}
|
||||
|
||||
final url =
|
||||
'${Variables.baseUrl}/api/v1/outlets/detail/${authData.user?.outletId}';
|
||||
|
||||
final response = await dio.get(
|
||||
url,
|
||||
queryParameters: {
|
||||
'organization_id': authData.user?.organizationId,
|
||||
},
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
final data = OutletDetailResponse.fromMap(response.data);
|
||||
await SettingsLocalDatasource().saveTax(
|
||||
TaxModel(
|
||||
name: 'PB1',
|
||||
type: TaxType.pajak,
|
||||
value: data.data?.taxRate ?? 0,
|
||||
),
|
||||
);
|
||||
|
||||
await OutletLocalDatasource().save(data.data!);
|
||||
return Right(data);
|
||||
} else {
|
||||
return const Left('Failed to get outlets');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(e.response?.data['message'] ?? 'Gagal mengambil outlet');
|
||||
} catch (e) {
|
||||
log("Unexpected error: $e");
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -1,34 +1,40 @@
|
||||
import 'dart:developer';
|
||||
|
||||
import 'package:dartz/dartz.dart';
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:enaklo_pos/core/constants/variables.dart';
|
||||
import 'package:enaklo_pos/core/network/dio_client.dart';
|
||||
import 'package:enaklo_pos/data/datasources/auth_local_datasource.dart';
|
||||
import 'package:enaklo_pos/data/models/response/payment_methods_response_model.dart';
|
||||
import 'package:http/http.dart' as http;
|
||||
|
||||
class PaymentMethodsRemoteDatasource {
|
||||
Future<Either<String, PaymentMethodsResponseModel>> getPaymentMethods() async {
|
||||
final Dio dio = DioClient.instance;
|
||||
Future<Either<String, PaymentMethodsResponseModel>>
|
||||
getPaymentMethods() async {
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final response = await http.get(
|
||||
Uri.parse('${Variables.baseUrl}/api/payment-methods'),
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
final response = await dio.get(
|
||||
'${Variables.baseUrl}/api/v1/payment-methods',
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
log("Payment Methods Response Status: ${response.statusCode}");
|
||||
log("Payment Methods Response Body: ${response.body}");
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(PaymentMethodsResponseModel.fromJson(response.body));
|
||||
return Right(PaymentMethodsResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return const Left('Failed to get payment methods');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(
|
||||
e.response?.data['message'] ?? 'Failed to get payment methods');
|
||||
} catch (e) {
|
||||
log("Error getting payment methods: $e");
|
||||
return Left('Error: $e');
|
||||
return Left('Unexpected error');
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -7,7 +7,6 @@ import 'package:enaklo_pos/data/models/response/table_model.dart';
|
||||
import 'package:enaklo_pos/presentation/home/models/order_model.dart';
|
||||
import 'package:enaklo_pos/presentation/table/models/draft_order_item.dart';
|
||||
import 'package:enaklo_pos/presentation/table/models/draft_order_model.dart';
|
||||
import 'package:intl/intl.dart';
|
||||
import 'package:sqflite/sqflite.dart';
|
||||
|
||||
import '../../presentation/home/models/product_quantity.dart';
|
||||
@ -151,7 +150,7 @@ class ProductLocalDatasource {
|
||||
final dbExists = await databaseExists(path);
|
||||
if (dbExists) {
|
||||
log("Deleting existing database to ensure new schema with order_type column");
|
||||
await deleteDatabase(path);
|
||||
// await deleteDatabase(path);
|
||||
}
|
||||
} catch (e) {
|
||||
log("Error deleting database: $e");
|
||||
@ -169,7 +168,8 @@ class ProductLocalDatasource {
|
||||
if (oldVersion < 2) {
|
||||
// Add order_type column to orders table if it doesn't exist
|
||||
try {
|
||||
await db.execute('ALTER TABLE $tableOrder ADD COLUMN order_type TEXT DEFAULT "DINE IN"');
|
||||
await db.execute(
|
||||
'ALTER TABLE $tableOrder ADD COLUMN order_type TEXT DEFAULT "DINE IN"');
|
||||
log("Added order_type column to orders table");
|
||||
} catch (e) {
|
||||
log("order_type column might already exist: $e");
|
||||
@ -238,6 +238,31 @@ class ProductLocalDatasource {
|
||||
});
|
||||
}
|
||||
|
||||
Future<List<OrderModel>> getAllOrderByRange(
|
||||
DateTime start, DateTime end) async {
|
||||
final db = await instance.database;
|
||||
|
||||
// Format ke ISO 8601 untuk range, hasil: yyyy-MM-ddTHH:mm:ss
|
||||
final startIso = start.toIso8601String();
|
||||
final endIso = end.toIso8601String();
|
||||
|
||||
final startDateYYYYMMDD = startIso.substring(0, 10);
|
||||
final endDateYYYYMMDD = endIso.substring(0, 10);
|
||||
|
||||
final List<Map<String, dynamic>> maps = await db.query(
|
||||
tableOrder,
|
||||
where: 'substr(transaction_time, 1, 10) BETWEEN ? AND ?',
|
||||
whereArgs: [startDateYYYYMMDD, endDateYYYYMMDD],
|
||||
orderBy: 'transaction_time DESC',
|
||||
);
|
||||
log("Get All Order By Range: $startDateYYYYMMDD $endDateYYYYMMDD");
|
||||
|
||||
return List.generate(maps.length, (i) {
|
||||
log("Save save OrderModel: ${OrderModel.fromMap(maps[i])}");
|
||||
return OrderModel.fromMap(maps[i]);
|
||||
});
|
||||
}
|
||||
|
||||
//get order item by order id
|
||||
Future<List<ProductQuantity>> getOrderItemByOrderId(int orderId) async {
|
||||
final db = await instance.database;
|
||||
@ -283,7 +308,7 @@ class ProductLocalDatasource {
|
||||
tableProduct,
|
||||
product.toLocalMap(),
|
||||
where: 'product_id = ?',
|
||||
whereArgs: [product.productId],
|
||||
whereArgs: [product.id],
|
||||
);
|
||||
}
|
||||
|
||||
@ -294,7 +319,7 @@ class ProductLocalDatasource {
|
||||
for (var product in products) {
|
||||
await db.insert(tableProduct, product.toLocalMap(),
|
||||
conflictAlgorithm: ConflictAlgorithm.replace);
|
||||
log('inserted success id: ${product.productId} | name: ${product.name} | price: ${product.price} | Printer Type ${product.printerType}');
|
||||
log('inserted success id: ${product.id} | name: ${product.name} | price: ${product.price} ');
|
||||
}
|
||||
}
|
||||
|
||||
@ -333,19 +358,19 @@ class ProductLocalDatasource {
|
||||
|
||||
// generate table managent with count
|
||||
Future<void> createTableManagement(String tableName, Offset position) async {
|
||||
final db = await instance.database;
|
||||
TableModel newTable = TableModel(
|
||||
tableName: tableName,
|
||||
status: 'available',
|
||||
orderId: 0,
|
||||
paymentAmount: 0,
|
||||
startTime: DateTime.now().toIso8601String(),
|
||||
position: position,
|
||||
);
|
||||
await db.insert(
|
||||
tableManagement,
|
||||
newTable.toMap(),
|
||||
);
|
||||
// final db = await instance.database;
|
||||
// TableModel newTable = TableModel(
|
||||
// tableName: tableName,
|
||||
// status: 'available',
|
||||
// orderId: 0,
|
||||
// paymentAmount: 0,
|
||||
// startTime: DateTime.now().toIso8601String(),
|
||||
// position: position,
|
||||
// );
|
||||
// await db.insert(
|
||||
// tableManagement,
|
||||
// newTable.toMap(),
|
||||
// );
|
||||
}
|
||||
|
||||
// change position table
|
||||
@ -578,7 +603,8 @@ class ProductLocalDatasource {
|
||||
where: 'id_draft_order = ?', whereArgs: [draftOrder.id]);
|
||||
|
||||
for (var orderItem in draftOrder.orders) {
|
||||
await db.insert('draft_order_items', orderItem.toMapForLocal(draftOrder.id!));
|
||||
await db.insert(
|
||||
'draft_order_items', orderItem.toMapForLocal(draftOrder.id!));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@ -1,94 +1,126 @@
|
||||
import 'dart:developer';
|
||||
|
||||
import 'package:dartz/dartz.dart';
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:enaklo_pos/core/network/dio_client.dart';
|
||||
import 'package:enaklo_pos/data/models/request/product_request_model.dart';
|
||||
import 'package:enaklo_pos/data/models/response/add_product_response_model.dart';
|
||||
import 'package:enaklo_pos/data/models/response/product_response_model.dart';
|
||||
import 'package:http/http.dart' as http;
|
||||
|
||||
import '../../core/constants/variables.dart';
|
||||
import 'auth_local_datasource.dart';
|
||||
|
||||
class ProductRemoteDatasource {
|
||||
Future<Either<String, ProductResponseModel>> getProducts() async {
|
||||
final url = Uri.parse('${Variables.baseUrl}/api/products');
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final response = await http.get(url, headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
});
|
||||
log("Status Code: ${response.statusCode}");
|
||||
log("Body: ${response.body}");
|
||||
if (response.statusCode == 200) {
|
||||
return Right(ProductResponseModel.fromJson(response.body));
|
||||
} else {
|
||||
return const Left('Failed to get products');
|
||||
final Dio dio = DioClient.instance;
|
||||
|
||||
Future<Either<String, ProductResponseModel>> getProducts({
|
||||
int page = 1,
|
||||
int limit = Variables.defaultLimit,
|
||||
String? categoryId,
|
||||
String? search,
|
||||
}) async {
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/products';
|
||||
|
||||
Map<String, dynamic> queryParameters = {
|
||||
'page': page,
|
||||
'limit': limit,
|
||||
};
|
||||
|
||||
if (categoryId != null) {
|
||||
queryParameters['category_id'] = categoryId;
|
||||
}
|
||||
|
||||
if (search != null && search.isNotEmpty) {
|
||||
queryParameters['search'] = search;
|
||||
}
|
||||
|
||||
final response = await dio.get(
|
||||
url,
|
||||
queryParameters: queryParameters,
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(ProductResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return const Left('Failed to get products');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(e.response?.data['message'] ?? 'Gagal mengambil produk');
|
||||
} catch (e) {
|
||||
log("Unexpected error: $e");
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, AddProductResponseModel>> addProduct(
|
||||
ProductRequestModel productRequestModel) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final Map<String, String> headers = {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
};
|
||||
var request = http.MultipartRequest(
|
||||
'POST', Uri.parse('${Variables.baseUrl}/api/products'));
|
||||
request.fields.addAll(productRequestModel.toMap());
|
||||
request.files.add(await http.MultipartFile.fromPath(
|
||||
'image', productRequestModel.image!.path));
|
||||
request.headers.addAll(headers);
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/products';
|
||||
|
||||
http.StreamedResponse response = await request.send();
|
||||
final response = await dio.post(
|
||||
url,
|
||||
data: productRequestModel.toMap(),
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
final String body = await response.stream.bytesToString();
|
||||
log(response.stream.toString());
|
||||
log(response.statusCode.toString());
|
||||
if (response.statusCode == 201) {
|
||||
return right(AddProductResponseModel.fromJson(body));
|
||||
} else {
|
||||
return left(body);
|
||||
if (response.statusCode == 200) {
|
||||
return Right(AddProductResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return const Left('Failed to create products');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(e.response?.data['message'] ?? 'Gagal menambah produk');
|
||||
} catch (e) {
|
||||
log("Unexpected error: $e");
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, AddProductResponseModel>> updateProduct(
|
||||
ProductRequestModel productRequestModel) async {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final Map<String, String> headers = {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
};
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url =
|
||||
'${Variables.baseUrl}/api/v1/products/${productRequestModel.id}';
|
||||
|
||||
log("Update Product Request Data: ${productRequestModel.toMap()}");
|
||||
log("Update Product ID: ${productRequestModel.id}");
|
||||
log("Update Product Name: ${productRequestModel.name}");
|
||||
log("Update Product Price: ${productRequestModel.price}");
|
||||
log("Update Product Stock: ${productRequestModel.stock}");
|
||||
log("Update Product Category ID: ${productRequestModel.categoryId}");
|
||||
log("Update Product Is Best Seller: ${productRequestModel.isBestSeller}");
|
||||
log("Update Product Printer Type: ${productRequestModel.printerType}");
|
||||
log("Update Product Has Image: ${productRequestModel.image != null}");
|
||||
final response = await dio.put(
|
||||
url,
|
||||
data: productRequestModel.toMap(),
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
var request = http.MultipartRequest(
|
||||
'POST', Uri.parse('${Variables.baseUrl}/api/products/edit'));
|
||||
request.fields.addAll(productRequestModel.toMap());
|
||||
if (productRequestModel.image != null) {
|
||||
request.files.add(await http.MultipartFile.fromPath(
|
||||
'image', productRequestModel.image!.path));
|
||||
}
|
||||
request.headers.addAll(headers);
|
||||
|
||||
log("Update Product Request Fields: ${request.fields}");
|
||||
log("Update Product Request Files: ${request.files.length}");
|
||||
|
||||
http.StreamedResponse response = await request.send();
|
||||
|
||||
final String body = await response.stream.bytesToString();
|
||||
log("Update Product Status Code: ${response.statusCode}");
|
||||
log("Update Product Body: $body");
|
||||
if (response.statusCode == 200) {
|
||||
return right(AddProductResponseModel.fromJson(body));
|
||||
} else {
|
||||
return left(body);
|
||||
if (response.statusCode == 200) {
|
||||
return Right(AddProductResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return const Left('Failed to update products');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(e.response?.data['message'] ?? 'Gagal update produk');
|
||||
} catch (e) {
|
||||
log("Unexpected error: $e");
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
170
lib/data/datasources/table_remote_datasource.dart
Normal file
@ -0,0 +1,170 @@
|
||||
import 'dart:developer';
|
||||
import 'dart:ui';
|
||||
import 'package:dartz/dartz.dart';
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:enaklo_pos/core/network/dio_client.dart';
|
||||
import 'package:enaklo_pos/data/models/response/table_model.dart';
|
||||
import '../../core/constants/variables.dart';
|
||||
import 'auth_local_datasource.dart';
|
||||
|
||||
class TableRemoteDataSource {
|
||||
final Dio dio = DioClient.instance;
|
||||
|
||||
Future<Either<String, bool>> createTable({
|
||||
required String tableName,
|
||||
required int capacity,
|
||||
required String location,
|
||||
}) async {
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/tables';
|
||||
|
||||
final response = await dio.post(
|
||||
url,
|
||||
data: {
|
||||
"outlet_id": authData.user?.outletId,
|
||||
"table_name": tableName,
|
||||
"capacity": capacity,
|
||||
"location": location,
|
||||
"status": "available",
|
||||
"is_active": true,
|
||||
"position_x": 200,
|
||||
"position_y": 200,
|
||||
},
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200 || response.statusCode == 201) {
|
||||
return Right(true);
|
||||
} else {
|
||||
return const Left('Failed to create table');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(e.response?.data['message'] ?? 'Gagal membuat table');
|
||||
} catch (e) {
|
||||
log("Unexpected error: $e");
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, TableResponseModel>> getTable({
|
||||
int page = 1,
|
||||
int limit = 50,
|
||||
String? status,
|
||||
}) async {
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/tables';
|
||||
|
||||
Map<String, dynamic> queryParameters = {
|
||||
'page': page,
|
||||
'limit': limit,
|
||||
};
|
||||
|
||||
if (status != null) {
|
||||
queryParameters['status'] = status;
|
||||
}
|
||||
|
||||
final response = await dio.get(
|
||||
url,
|
||||
queryParameters: queryParameters,
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
return Right(TableResponseModel.fromMap(response.data));
|
||||
} else {
|
||||
return const Left('Failed to get tables');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(e.response?.data['message'] ?? 'Gagal mengambil data meja');
|
||||
} catch (e) {
|
||||
log("Unexpected error: $e");
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, bool>> updatePosition({
|
||||
required String tableId,
|
||||
required Offset position,
|
||||
}) async {
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/tables/$tableId';
|
||||
|
||||
final response = await dio.put(
|
||||
url,
|
||||
data: {
|
||||
"position_x": position.dx.round(),
|
||||
"position_y": position.dy.round(),
|
||||
},
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200 || response.statusCode == 201) {
|
||||
return Right(true);
|
||||
} else {
|
||||
return const Left('Failed to create table');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(e.response?.data['message'] ?? 'Gagal membuat table');
|
||||
} catch (e) {
|
||||
log("Unexpected error: $e");
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
|
||||
Future<Either<String, bool>> transferTable({
|
||||
required String fromTableId,
|
||||
required String toTableId,
|
||||
}) async {
|
||||
try {
|
||||
final authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/tables/transfer';
|
||||
|
||||
final response = await dio.put(
|
||||
url,
|
||||
data: {
|
||||
"from_table": fromTableId,
|
||||
"to_table": toTableId,
|
||||
},
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200 || response.statusCode == 201) {
|
||||
return Right(true);
|
||||
} else {
|
||||
return const Left('Failed to create table');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(e.response?.data['message'] ?? 'Gagal membuat table');
|
||||
} catch (e) {
|
||||
log("Unexpected error: $e");
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
}
|
||||
47
lib/data/datasources/user_remote_datasource.dart
Normal file
@ -0,0 +1,47 @@
|
||||
import 'dart:developer';
|
||||
|
||||
import 'package:dartz/dartz.dart';
|
||||
import 'package:dio/dio.dart';
|
||||
import 'package:enaklo_pos/core/constants/variables.dart';
|
||||
import 'package:enaklo_pos/core/network/dio_client.dart';
|
||||
import 'package:enaklo_pos/data/datasources/auth_local_datasource.dart';
|
||||
import 'package:enaklo_pos/data/models/response/auth_response_model.dart';
|
||||
|
||||
class UserRemoteDatasource {
|
||||
final Dio dio = DioClient.instance;
|
||||
|
||||
Future<Either<String, bool>> updateOutlet(String outletId) async {
|
||||
AuthResponseModel authData = await AuthLocalDataSource().getAuthData();
|
||||
final url = '${Variables.baseUrl}/api/v1/users/${authData.user?.id}';
|
||||
|
||||
try {
|
||||
final response = await dio.put(
|
||||
url,
|
||||
data: {
|
||||
'outlet_id': outletId,
|
||||
},
|
||||
options: Options(
|
||||
headers: {
|
||||
'Authorization': 'Bearer ${authData.token}',
|
||||
'Accept': 'application/json',
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
),
|
||||
);
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
authData.user?.outletId = response.data['outlet_id'];
|
||||
await AuthLocalDataSource().saveAuthData(authData);
|
||||
return Right(true);
|
||||
} else {
|
||||
return const Left('Failed to login');
|
||||
}
|
||||
} on DioException catch (e) {
|
||||
log("Dio error: ${e.message}");
|
||||
return Left(e.response?.data['message'] ?? 'Login gagal');
|
||||
} catch (e) {
|
||||
log("Unexpected error: $e");
|
||||
return const Left('Unexpected error occurred');
|
||||
}
|
||||
}
|
||||
}
|
||||
137
lib/data/models/request/payment_request.dart
Normal file
@ -0,0 +1,137 @@
|
||||
import 'dart:convert';
|
||||
|
||||
class PaymentRequestModel {
|
||||
final String? orderId;
|
||||
final String? paymentMethodId;
|
||||
final int? amount;
|
||||
final String? transactionId;
|
||||
final int? splitNumber;
|
||||
final int? splitTotal;
|
||||
final String? splitDescription;
|
||||
final List<PaymentOrderItemModel>? paymentOrderItems;
|
||||
|
||||
PaymentRequestModel({
|
||||
this.orderId,
|
||||
this.paymentMethodId,
|
||||
this.amount,
|
||||
this.transactionId,
|
||||
this.splitNumber,
|
||||
this.splitTotal,
|
||||
this.splitDescription,
|
||||
this.paymentOrderItems,
|
||||
});
|
||||
|
||||
factory PaymentRequestModel.fromJson(String str) =>
|
||||
PaymentRequestModel.fromMap(json.decode(str));
|
||||
|
||||
String toJson() => json.encode(toMap());
|
||||
|
||||
factory PaymentRequestModel.fromMap(Map<String, dynamic> json) =>
|
||||
PaymentRequestModel(
|
||||
orderId: json["order_id"],
|
||||
paymentMethodId: json["payment_method_id"],
|
||||
amount: json["amount"]?.toDouble(),
|
||||
transactionId: json["transaction_id"],
|
||||
splitNumber: json["split_number"],
|
||||
splitTotal: json["split_total"],
|
||||
splitDescription: json["split_description"],
|
||||
paymentOrderItems: json["payment_order_items"] == null
|
||||
? []
|
||||
: List<PaymentOrderItemModel>.from(json["payment_order_items"]
|
||||
.map((x) => PaymentOrderItemModel.fromMap(x))),
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"order_id": orderId,
|
||||
"payment_method_id": paymentMethodId,
|
||||
"amount": amount,
|
||||
"transaction_id": transactionId,
|
||||
"split_number": splitNumber,
|
||||
"split_total": splitTotal,
|
||||
"split_description": splitDescription,
|
||||
"payment_order_items": paymentOrderItems == null
|
||||
? []
|
||||
: List<dynamic>.from(paymentOrderItems!.map((x) => x.toMap())),
|
||||
};
|
||||
}
|
||||
|
||||
class PaymentOrderItemModel {
|
||||
final String? orderItemId;
|
||||
final int? amount;
|
||||
|
||||
PaymentOrderItemModel({
|
||||
this.orderItemId,
|
||||
this.amount,
|
||||
});
|
||||
|
||||
factory PaymentOrderItemModel.fromJson(String str) =>
|
||||
PaymentOrderItemModel.fromMap(json.decode(str));
|
||||
|
||||
factory PaymentOrderItemModel.fromMap(Map<String, dynamic> json) =>
|
||||
PaymentOrderItemModel(
|
||||
orderItemId: json["order_item_id"],
|
||||
amount: json["amount"]?.toDouble(),
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"order_item_id": orderItemId,
|
||||
"amount": amount,
|
||||
};
|
||||
}
|
||||
|
||||
class PaymentSplitBillRequest {
|
||||
final String orderId;
|
||||
final String paymentMethodId;
|
||||
final String customerId;
|
||||
final String type; // e.g., "AMOUNT" or "ITEM"
|
||||
final int amount;
|
||||
final List<SplitItem> items;
|
||||
|
||||
PaymentSplitBillRequest({
|
||||
required this.orderId,
|
||||
required this.paymentMethodId,
|
||||
required this.customerId,
|
||||
required this.type,
|
||||
required this.amount,
|
||||
required this.items,
|
||||
});
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
Map<String, dynamic> data = {
|
||||
'order_id': orderId,
|
||||
'payment_method_id': paymentMethodId,
|
||||
'type': type,
|
||||
};
|
||||
|
||||
if (customerId.isNotEmpty) {
|
||||
data['customer_id'] = customerId;
|
||||
}
|
||||
|
||||
if (type == "AMOUNT") {
|
||||
data['amount'] = amount;
|
||||
}
|
||||
|
||||
if (type == "ITEM") {
|
||||
data['items'] = items.map((e) => e.toJson()).toList();
|
||||
}
|
||||
|
||||
return data;
|
||||
}
|
||||
}
|
||||
|
||||
class SplitItem {
|
||||
final String orderItemId;
|
||||
final int quantity;
|
||||
|
||||
SplitItem({
|
||||
required this.orderItemId,
|
||||
required this.quantity,
|
||||
});
|
||||
|
||||
Map<String, dynamic> toJson() {
|
||||
return {
|
||||
'order_item_id': orderItemId,
|
||||
'quantity': quantity,
|
||||
};
|
||||
}
|
||||
}
|
||||
@ -1,40 +1,49 @@
|
||||
import 'dart:developer';
|
||||
|
||||
import 'package:image_picker/image_picker.dart';
|
||||
|
||||
class ProductRequestModel {
|
||||
final int? id;
|
||||
final String? id;
|
||||
final String name;
|
||||
final String? description;
|
||||
final String categoryId;
|
||||
final String? sku;
|
||||
final String? barcode;
|
||||
final int price;
|
||||
final int stock;
|
||||
final int categoryId;
|
||||
final int isBestSeller;
|
||||
final XFile? image;
|
||||
final int cost;
|
||||
final bool isActive;
|
||||
final bool hasVariants;
|
||||
final String imageUrl;
|
||||
final String? printerType;
|
||||
|
||||
ProductRequestModel({
|
||||
this.id,
|
||||
required this.name,
|
||||
required this.price,
|
||||
required this.stock,
|
||||
this.description,
|
||||
required this.categoryId,
|
||||
required this.isBestSeller,
|
||||
this.image,
|
||||
this.sku,
|
||||
this.barcode,
|
||||
required this.price,
|
||||
required this.cost,
|
||||
this.isActive = true,
|
||||
this.hasVariants = false,
|
||||
required this.imageUrl,
|
||||
this.printerType,
|
||||
});
|
||||
|
||||
Map<String, String> toMap() {
|
||||
log("toMap: $isBestSeller");
|
||||
final map = {
|
||||
Map<String, dynamic> toMap() {
|
||||
final map = <String, dynamic>{
|
||||
'name': name,
|
||||
'price': price.toString(),
|
||||
'stock': stock.toString(),
|
||||
'category_id': categoryId.toString(),
|
||||
'is_best_seller': isBestSeller.toString(),
|
||||
'description': description ?? '',
|
||||
'category_id': categoryId,
|
||||
'sku': sku ?? '',
|
||||
'barcode': barcode ?? '',
|
||||
'price': price,
|
||||
'cost': cost,
|
||||
'is_active': isActive,
|
||||
'has_variants': hasVariants,
|
||||
'image_url': imageUrl,
|
||||
'printer_type': printerType ?? '',
|
||||
};
|
||||
|
||||
if (id != null) {
|
||||
map['id'] = id.toString();
|
||||
map['id'] = id;
|
||||
}
|
||||
|
||||
return map;
|
||||
|
||||
@ -4,12 +4,10 @@ import 'package:enaklo_pos/data/models/response/product_response_model.dart';
|
||||
|
||||
class AddProductResponseModel {
|
||||
final bool success;
|
||||
final String message;
|
||||
final Product data;
|
||||
|
||||
AddProductResponseModel({
|
||||
required this.success,
|
||||
required this.message,
|
||||
required this.data,
|
||||
});
|
||||
|
||||
@ -21,13 +19,11 @@ class AddProductResponseModel {
|
||||
factory AddProductResponseModel.fromMap(Map<String, dynamic> json) =>
|
||||
AddProductResponseModel(
|
||||
success: json["success"],
|
||||
message: json["message"],
|
||||
data: Product.fromMap(json["data"]),
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"success": success,
|
||||
"message": message,
|
||||
"data": data.toMap(),
|
||||
};
|
||||
}
|
||||
|
||||
@ -1,85 +1,83 @@
|
||||
import 'dart:convert';
|
||||
|
||||
class AuthResponseModel {
|
||||
final String? status;
|
||||
final String? token;
|
||||
final User? user;
|
||||
final String? token;
|
||||
final User? user;
|
||||
|
||||
AuthResponseModel({
|
||||
this.status,
|
||||
this.token,
|
||||
this.user,
|
||||
});
|
||||
AuthResponseModel({
|
||||
this.token,
|
||||
this.user,
|
||||
});
|
||||
|
||||
factory AuthResponseModel.fromJson(String str) => AuthResponseModel.fromMap(json.decode(str));
|
||||
factory AuthResponseModel.fromJson(String str) =>
|
||||
AuthResponseModel.fromMap(json.decode(str));
|
||||
|
||||
String toJson() => json.encode(toMap());
|
||||
String toJson() => json.encode(toMap());
|
||||
|
||||
factory AuthResponseModel.fromMap(Map<String, dynamic> json) => AuthResponseModel(
|
||||
status: json["status"],
|
||||
factory AuthResponseModel.fromMap(Map<String, dynamic> json) =>
|
||||
AuthResponseModel(
|
||||
token: json["token"],
|
||||
user: json["user"] == null ? null : User.fromMap(json["user"]),
|
||||
);
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"status": status,
|
||||
Map<String, dynamic> toMap() => {
|
||||
"token": token,
|
||||
"user": user?.toMap(),
|
||||
};
|
||||
};
|
||||
}
|
||||
|
||||
class User {
|
||||
final int? id;
|
||||
final String? name;
|
||||
final String? email;
|
||||
final DateTime? emailVerifiedAt;
|
||||
final dynamic twoFactorSecret;
|
||||
final dynamic twoFactorRecoveryCodes;
|
||||
final dynamic twoFactorConfirmedAt;
|
||||
final DateTime? createdAt;
|
||||
final DateTime? updatedAt;
|
||||
final String? role;
|
||||
final String? id;
|
||||
final String? organizationId;
|
||||
String? outletId;
|
||||
final String? name;
|
||||
final String? email;
|
||||
final String? role;
|
||||
final bool? isActive;
|
||||
final DateTime? createdAt;
|
||||
final DateTime? updatedAt;
|
||||
|
||||
User({
|
||||
this.id,
|
||||
this.name,
|
||||
this.email,
|
||||
this.emailVerifiedAt,
|
||||
this.twoFactorSecret,
|
||||
this.twoFactorRecoveryCodes,
|
||||
this.twoFactorConfirmedAt,
|
||||
this.createdAt,
|
||||
this.updatedAt,
|
||||
this.role,
|
||||
});
|
||||
User({
|
||||
this.id,
|
||||
this.organizationId,
|
||||
this.outletId,
|
||||
this.name,
|
||||
this.role,
|
||||
this.isActive,
|
||||
this.email,
|
||||
this.createdAt,
|
||||
this.updatedAt,
|
||||
});
|
||||
|
||||
factory User.fromJson(String str) => User.fromMap(json.decode(str));
|
||||
factory User.fromJson(String str) => User.fromMap(json.decode(str));
|
||||
|
||||
String toJson() => json.encode(toMap());
|
||||
String toJson() => json.encode(toMap());
|
||||
|
||||
factory User.fromMap(Map<String, dynamic> json) => User(
|
||||
factory User.fromMap(Map<String, dynamic> json) => User(
|
||||
id: json["id"],
|
||||
organizationId: json["organization_id"],
|
||||
outletId: json["outlet_id"],
|
||||
name: json["name"],
|
||||
email: json["email"],
|
||||
emailVerifiedAt: json["email_verified_at"] == null ? null : DateTime.parse(json["email_verified_at"]),
|
||||
twoFactorSecret: json["two_factor_secret"],
|
||||
twoFactorRecoveryCodes: json["two_factor_recovery_codes"],
|
||||
twoFactorConfirmedAt: json["two_factor_confirmed_at"],
|
||||
createdAt: json["created_at"] == null ? null : DateTime.parse(json["created_at"]),
|
||||
updatedAt: json["updated_at"] == null ? null : DateTime.parse(json["updated_at"]),
|
||||
role: json["role"],
|
||||
);
|
||||
isActive: json["is_active"],
|
||||
createdAt: json["created_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["created_at"]),
|
||||
updatedAt: json["updated_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["updated_at"]),
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
Map<String, dynamic> toMap() => {
|
||||
"id": id,
|
||||
"organization_id": organizationId,
|
||||
"outlet_id": outletId,
|
||||
"name": name,
|
||||
"email": email,
|
||||
"email_verified_at": emailVerifiedAt?.toIso8601String(),
|
||||
"two_factor_secret": twoFactorSecret,
|
||||
"two_factor_recovery_codes": twoFactorRecoveryCodes,
|
||||
"two_factor_confirmed_at": twoFactorConfirmedAt,
|
||||
"role": role,
|
||||
"is_active": isActive,
|
||||
"created_at": createdAt?.toIso8601String(),
|
||||
"updated_at": updatedAt?.toIso8601String(),
|
||||
"role": role,
|
||||
};
|
||||
};
|
||||
}
|
||||
|
||||
@ -1,91 +1,120 @@
|
||||
// ignore_for_file: public_member_api_docs, sort_constructors_first
|
||||
import 'dart:convert';
|
||||
|
||||
class CategroyResponseModel {
|
||||
final String status;
|
||||
final List<CategoryModel> data;
|
||||
class CategoryResponseModel {
|
||||
final bool success;
|
||||
final CategoryData data;
|
||||
final dynamic errors;
|
||||
|
||||
CategroyResponseModel({
|
||||
required this.status,
|
||||
CategoryResponseModel({
|
||||
required this.success,
|
||||
required this.data,
|
||||
this.errors,
|
||||
});
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
return <String, dynamic>{
|
||||
'status': status,
|
||||
'data': data.map((x) => x.toMap()).toList(),
|
||||
};
|
||||
}
|
||||
|
||||
factory CategroyResponseModel.fromMap(Map<String, dynamic> map) {
|
||||
return CategroyResponseModel(
|
||||
status: map['status'] as String,
|
||||
data: List<CategoryModel>.from(
|
||||
(map['data']).map<CategoryModel>(
|
||||
(x) => CategoryModel.fromMap(x as Map<String, dynamic>),
|
||||
),
|
||||
),
|
||||
factory CategoryResponseModel.fromMap(Map<String, dynamic> map) {
|
||||
return CategoryResponseModel(
|
||||
success: map['success'] as bool,
|
||||
data: CategoryData.fromMap(map['data'] as Map<String, dynamic>),
|
||||
errors: map['errors'],
|
||||
);
|
||||
}
|
||||
|
||||
factory CategroyResponseModel.fromJson(String str) =>
|
||||
CategroyResponseModel.fromMap(json.decode(str));
|
||||
|
||||
String toJson() => json.encode(toMap());
|
||||
}
|
||||
|
||||
class CategoryModel {
|
||||
int? id;
|
||||
String? name;
|
||||
int? categoryId;
|
||||
int? isSync;
|
||||
String? image;
|
||||
// DateTime createdAt;
|
||||
// DateTime updatedAt;
|
||||
|
||||
CategoryModel({this.id, this.name, this.categoryId, this.isSync, this.image});
|
||||
factory CategoryResponseModel.fromJson(String str) =>
|
||||
CategoryResponseModel.fromMap(json.decode(str));
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
return <String, dynamic>{
|
||||
// 'id': id,
|
||||
'name': name,
|
||||
'is_sync': isSync ?? 1,
|
||||
'category_id': id,
|
||||
'image': image
|
||||
return {
|
||||
'success': success,
|
||||
'data': data.toMap(),
|
||||
'errors': errors,
|
||||
};
|
||||
}
|
||||
|
||||
factory CategoryModel.fromMap(Map<String, dynamic> map) {
|
||||
return CategoryModel(
|
||||
id: map['id'] as int?,
|
||||
name: map['name'] as String?,
|
||||
isSync: map['is_sync'] as int?,
|
||||
categoryId: map['id'],
|
||||
image: map['image']);
|
||||
}
|
||||
|
||||
factory CategoryModel.fromJson(String str) =>
|
||||
CategoryModel.fromMap(json.decode(str));
|
||||
|
||||
String toJson() => json.encode(toMap());
|
||||
}
|
||||
|
||||
@override
|
||||
bool operator ==(covariant CategoryModel other) {
|
||||
if (identical(this, other)) return true;
|
||||
class CategoryData {
|
||||
final List<CategoryModel> categories;
|
||||
final int totalCount;
|
||||
final int page;
|
||||
final int limit;
|
||||
final int totalPages;
|
||||
|
||||
return other.id == id &&
|
||||
other.name == name &&
|
||||
other.categoryId == categoryId &&
|
||||
other.isSync == isSync &&
|
||||
other.image == image;
|
||||
CategoryData({
|
||||
required this.categories,
|
||||
required this.totalCount,
|
||||
required this.page,
|
||||
required this.limit,
|
||||
required this.totalPages,
|
||||
});
|
||||
|
||||
factory CategoryData.fromMap(Map<String, dynamic> map) {
|
||||
return CategoryData(
|
||||
categories: List<CategoryModel>.from(
|
||||
(map['categories'] as List).map((x) => CategoryModel.fromMap(x)),
|
||||
),
|
||||
totalCount: map['total_count'] as int,
|
||||
page: map['page'] as int,
|
||||
limit: map['limit'] as int,
|
||||
totalPages: map['total_pages'] as int,
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
int get hashCode {
|
||||
return id.hashCode ^
|
||||
name.hashCode ^
|
||||
categoryId.hashCode ^
|
||||
isSync.hashCode ^
|
||||
image.hashCode;
|
||||
Map<String, dynamic> toMap() {
|
||||
return {
|
||||
'categories': categories.map((x) => x.toMap()).toList(),
|
||||
'total_count': totalCount,
|
||||
'page': page,
|
||||
'limit': limit,
|
||||
'total_pages': totalPages,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
class CategoryModel {
|
||||
String id;
|
||||
final String organizationId;
|
||||
final String name;
|
||||
final String? description;
|
||||
final String businessType;
|
||||
final Map<String, dynamic> metadata;
|
||||
final DateTime createdAt;
|
||||
final DateTime updatedAt;
|
||||
|
||||
CategoryModel({
|
||||
required this.id,
|
||||
required this.organizationId,
|
||||
required this.name,
|
||||
this.description,
|
||||
required this.businessType,
|
||||
required this.metadata,
|
||||
required this.createdAt,
|
||||
required this.updatedAt,
|
||||
});
|
||||
|
||||
factory CategoryModel.fromMap(Map<String, dynamic> map) {
|
||||
return CategoryModel(
|
||||
id: map['id'] as String,
|
||||
organizationId: map['organization_id'] as String,
|
||||
name: map['name'] as String,
|
||||
description: map['description'] as String?,
|
||||
businessType: map['business_type'] as String,
|
||||
metadata: Map<String, dynamic>.from(map['metadata'] ?? {}),
|
||||
createdAt: DateTime.parse(map['created_at'] as String),
|
||||
updatedAt: DateTime.parse(map['updated_at'] as String),
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
return {
|
||||
'id': id,
|
||||
'organization_id': organizationId,
|
||||
'name': name,
|
||||
'description': description,
|
||||
'business_type': businessType,
|
||||
'metadata': metadata,
|
||||
'created_at': createdAt.toIso8601String(),
|
||||
'updated_at': updatedAt.toIso8601String(),
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
127
lib/data/models/response/customer_response_model.dart
Normal file
@ -0,0 +1,127 @@
|
||||
import 'dart:convert';
|
||||
|
||||
class CustomerResponseModel {
|
||||
final bool? success;
|
||||
final CustomerData? data;
|
||||
final dynamic errors;
|
||||
|
||||
CustomerResponseModel({
|
||||
this.success,
|
||||
this.data,
|
||||
this.errors,
|
||||
});
|
||||
|
||||
factory CustomerResponseModel.fromJson(String str) =>
|
||||
CustomerResponseModel.fromMap(json.decode(str));
|
||||
|
||||
String toJson() => json.encode(toMap());
|
||||
|
||||
factory CustomerResponseModel.fromMap(Map<String, dynamic> json) =>
|
||||
CustomerResponseModel(
|
||||
success: json["success"],
|
||||
data: json["data"] == null ? null : CustomerData.fromMap(json["data"]),
|
||||
errors: json["errors"],
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"success": success,
|
||||
"data": data?.toMap(),
|
||||
"errors": errors,
|
||||
};
|
||||
}
|
||||
|
||||
class CustomerData {
|
||||
final List<Customer>? customers;
|
||||
final int? totalCount;
|
||||
final int? page;
|
||||
final int? limit;
|
||||
final int? totalPages;
|
||||
|
||||
CustomerData({
|
||||
this.customers,
|
||||
this.totalCount,
|
||||
this.page,
|
||||
this.limit,
|
||||
this.totalPages,
|
||||
});
|
||||
|
||||
factory CustomerData.fromMap(Map<String, dynamic> json) => CustomerData(
|
||||
customers: json["data"] == null
|
||||
? []
|
||||
: List<Customer>.from(json["data"].map((x) => Customer.fromMap(x))),
|
||||
totalCount: json["total_count"],
|
||||
page: json["page"],
|
||||
limit: json["limit"],
|
||||
totalPages: json["total_pages"],
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"data": customers == null
|
||||
? []
|
||||
: List<dynamic>.from(customers!.map((x) => x.toMap())),
|
||||
"total_count": totalCount,
|
||||
"page": page,
|
||||
"limit": limit,
|
||||
"total_pages": totalPages,
|
||||
};
|
||||
}
|
||||
|
||||
class Customer {
|
||||
final String? id;
|
||||
final String? organizationId;
|
||||
final String? name;
|
||||
final String? email;
|
||||
final String? phone;
|
||||
final String? address;
|
||||
final bool? isDefault;
|
||||
final bool? isActive;
|
||||
final Map<String, dynamic>? metadata;
|
||||
final DateTime? createdAt;
|
||||
final DateTime? updatedAt;
|
||||
|
||||
Customer({
|
||||
this.id,
|
||||
this.organizationId,
|
||||
this.name,
|
||||
this.email,
|
||||
this.phone,
|
||||
this.address,
|
||||
this.isDefault,
|
||||
this.isActive,
|
||||
this.metadata,
|
||||
this.createdAt,
|
||||
this.updatedAt,
|
||||
});
|
||||
|
||||
factory Customer.fromMap(Map<String, dynamic> json) => Customer(
|
||||
id: json["id"],
|
||||
organizationId: json["organization_id"],
|
||||
name: json["name"],
|
||||
email: json["email"],
|
||||
phone: json["phone"],
|
||||
address: json["address"],
|
||||
isDefault: json["is_default"],
|
||||
isActive: json["is_active"],
|
||||
metadata: json["metadata"] ?? {},
|
||||
createdAt: json["created_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["created_at"]),
|
||||
updatedAt: json["updated_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["updated_at"]),
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"id": id,
|
||||
"organization_id": organizationId,
|
||||
"name": name,
|
||||
"email": email,
|
||||
"phone": phone,
|
||||
"address": address,
|
||||
"is_default": isDefault,
|
||||
"is_active": isActive,
|
||||
"metadata": metadata,
|
||||
"created_at": createdAt?.toIso8601String(),
|
||||
"updated_at": updatedAt?.toIso8601String(),
|
||||
};
|
||||
}
|
||||
248
lib/data/models/response/dashboard_analytic_response_model.dart
Normal file
@ -0,0 +1,248 @@
|
||||
class DashboardAnalyticResponseModel {
|
||||
final bool success;
|
||||
final DashboardAnalyticData data;
|
||||
final dynamic errors;
|
||||
|
||||
DashboardAnalyticResponseModel({
|
||||
required this.success,
|
||||
required this.data,
|
||||
this.errors,
|
||||
});
|
||||
|
||||
/// Khusus untuk JSON string
|
||||
factory DashboardAnalyticResponseModel.fromJson(Map<String, dynamic> json) =>
|
||||
DashboardAnalyticResponseModel.fromMap(json);
|
||||
|
||||
/// Untuk menerima Map biasa (bukan dari JSON string)
|
||||
factory DashboardAnalyticResponseModel.fromMap(Map<String, dynamic> map) {
|
||||
return DashboardAnalyticResponseModel(
|
||||
success: map['success'] ?? false,
|
||||
data: DashboardAnalyticData.fromMap(map['data'] ?? {}),
|
||||
errors: map['errors'],
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() => toMap();
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
'success': success,
|
||||
'data': data.toMap(),
|
||||
'errors': errors,
|
||||
};
|
||||
}
|
||||
|
||||
class DashboardAnalyticData {
|
||||
final String organizationId;
|
||||
final String outletId;
|
||||
final String dateFrom;
|
||||
final String dateTo;
|
||||
final DashboardOverview overview;
|
||||
final List<TopProduct> topProducts;
|
||||
final List<PaymentMethodAnalytic> paymentMethods;
|
||||
final List<RecentSale> recentSales;
|
||||
|
||||
DashboardAnalyticData({
|
||||
required this.organizationId,
|
||||
required this.outletId,
|
||||
required this.dateFrom,
|
||||
required this.dateTo,
|
||||
required this.overview,
|
||||
required this.topProducts,
|
||||
required this.paymentMethods,
|
||||
required this.recentSales,
|
||||
});
|
||||
|
||||
factory DashboardAnalyticData.fromMap(Map<String, dynamic> map) =>
|
||||
DashboardAnalyticData(
|
||||
organizationId: map['organization_id'],
|
||||
outletId: map['outlet_id'],
|
||||
dateFrom: map['date_from'],
|
||||
dateTo: map['date_to'],
|
||||
overview: DashboardOverview.fromMap(map['overview']),
|
||||
topProducts: map['top_products'] == null
|
||||
? []
|
||||
: List<TopProduct>.from(
|
||||
map['top_products']?.map((x) => TopProduct.fromMap(x))),
|
||||
paymentMethods: map['payment_methods'] == null
|
||||
? []
|
||||
: List<PaymentMethodAnalytic>.from(map['payment_methods']
|
||||
?.map((x) => PaymentMethodAnalytic.fromMap(x))),
|
||||
recentSales: map['recent_sales'] == null
|
||||
? []
|
||||
: List<RecentSale>.from(
|
||||
map['recent_sales']?.map((x) => RecentSale.fromMap(x))),
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
'organization_id': organizationId,
|
||||
'outlet_id': outletId,
|
||||
'date_from': dateFrom,
|
||||
'date_to': dateTo,
|
||||
'overview': overview.toMap(),
|
||||
'top_products': topProducts.map((x) => x.toMap()).toList(),
|
||||
'payment_methods': paymentMethods.map((x) => x.toMap()).toList(),
|
||||
'recent_sales': recentSales.map((x) => x.toMap()).toList(),
|
||||
};
|
||||
}
|
||||
|
||||
class DashboardOverview {
|
||||
final int totalSales;
|
||||
final int totalOrders;
|
||||
final double averageOrderValue;
|
||||
final int totalCustomers;
|
||||
final int voidedOrders;
|
||||
final int refundedOrders;
|
||||
|
||||
DashboardOverview({
|
||||
required this.totalSales,
|
||||
required this.totalOrders,
|
||||
required this.averageOrderValue,
|
||||
required this.totalCustomers,
|
||||
required this.voidedOrders,
|
||||
required this.refundedOrders,
|
||||
});
|
||||
|
||||
factory DashboardOverview.fromMap(Map<String, dynamic> map) =>
|
||||
DashboardOverview(
|
||||
totalSales: map['total_sales'],
|
||||
totalOrders: map['total_orders'],
|
||||
averageOrderValue: map['average_order_value']?.toDouble() ?? 0.0,
|
||||
totalCustomers: map['total_customers'],
|
||||
voidedOrders: map['voided_orders'],
|
||||
refundedOrders: map['refunded_orders'],
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
'total_sales': totalSales,
|
||||
'total_orders': totalOrders,
|
||||
'average_order_value': averageOrderValue,
|
||||
'total_customers': totalCustomers,
|
||||
'voided_orders': voidedOrders,
|
||||
'refunded_orders': refundedOrders,
|
||||
};
|
||||
}
|
||||
|
||||
class TopProduct {
|
||||
final String productId;
|
||||
final String productName;
|
||||
final String categoryId;
|
||||
final String categoryName;
|
||||
final int quantitySold;
|
||||
final int revenue;
|
||||
final double averagePrice;
|
||||
final int orderCount;
|
||||
|
||||
TopProduct({
|
||||
required this.productId,
|
||||
required this.productName,
|
||||
required this.categoryId,
|
||||
required this.categoryName,
|
||||
required this.quantitySold,
|
||||
required this.revenue,
|
||||
required this.averagePrice,
|
||||
required this.orderCount,
|
||||
});
|
||||
|
||||
factory TopProduct.fromMap(Map<String, dynamic> map) => TopProduct(
|
||||
productId: map['product_id'],
|
||||
productName: map['product_name'],
|
||||
categoryId: map['category_id'],
|
||||
categoryName: map['category_name'],
|
||||
quantitySold: map['quantity_sold'],
|
||||
revenue: map['revenue'],
|
||||
averagePrice: map['average_price']?.toDouble() ?? 0.0,
|
||||
orderCount: map['order_count'],
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
'product_id': productId,
|
||||
'product_name': productName,
|
||||
'category_id': categoryId,
|
||||
'category_name': categoryName,
|
||||
'quantity_sold': quantitySold,
|
||||
'revenue': revenue,
|
||||
'average_price': averagePrice,
|
||||
'order_count': orderCount,
|
||||
};
|
||||
}
|
||||
|
||||
class PaymentMethodAnalytic {
|
||||
final String paymentMethodId;
|
||||
final String paymentMethodName;
|
||||
final String paymentMethodType;
|
||||
final int totalAmount;
|
||||
final int orderCount;
|
||||
final int paymentCount;
|
||||
final double percentage;
|
||||
|
||||
PaymentMethodAnalytic({
|
||||
required this.paymentMethodId,
|
||||
required this.paymentMethodName,
|
||||
required this.paymentMethodType,
|
||||
required this.totalAmount,
|
||||
required this.orderCount,
|
||||
required this.paymentCount,
|
||||
required this.percentage,
|
||||
});
|
||||
|
||||
factory PaymentMethodAnalytic.fromMap(Map<String, dynamic> map) =>
|
||||
PaymentMethodAnalytic(
|
||||
paymentMethodId: map['payment_method_id'],
|
||||
paymentMethodName: map['payment_method_name'],
|
||||
paymentMethodType: map['payment_method_type'],
|
||||
totalAmount: map['total_amount'],
|
||||
orderCount: map['order_count'],
|
||||
paymentCount: map['payment_count'],
|
||||
percentage: map['percentage']?.toDouble() ?? 0.0,
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
'payment_method_id': paymentMethodId,
|
||||
'payment_method_name': paymentMethodName,
|
||||
'payment_method_type': paymentMethodType,
|
||||
'total_amount': totalAmount,
|
||||
'order_count': orderCount,
|
||||
'payment_count': paymentCount,
|
||||
'percentage': percentage,
|
||||
};
|
||||
}
|
||||
|
||||
class RecentSale {
|
||||
final String date;
|
||||
final int sales;
|
||||
final int orders;
|
||||
final int items;
|
||||
final int tax;
|
||||
final int discount;
|
||||
final int netSales;
|
||||
|
||||
RecentSale({
|
||||
required this.date,
|
||||
required this.sales,
|
||||
required this.orders,
|
||||
required this.items,
|
||||
required this.tax,
|
||||
required this.discount,
|
||||
required this.netSales,
|
||||
});
|
||||
|
||||
factory RecentSale.fromMap(Map<String, dynamic> map) => RecentSale(
|
||||
date: map['date'],
|
||||
sales: map['sales'],
|
||||
orders: map['orders'],
|
||||
items: map['items'],
|
||||
tax: map['tax'],
|
||||
discount: map['discount'],
|
||||
netSales: map['net_sales'],
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
'date': date,
|
||||
'sales': sales,
|
||||
'orders': orders,
|
||||
'items': items,
|
||||
'tax': tax,
|
||||
'discount': discount,
|
||||
'net_sales': netSales,
|
||||
};
|
||||
}
|
||||
11
lib/data/models/response/delivery_response_model.dart
Normal file
@ -0,0 +1,11 @@
|
||||
class DeliveryModel {
|
||||
String id;
|
||||
String name;
|
||||
String imageUrl;
|
||||
|
||||
DeliveryModel({
|
||||
required this.id,
|
||||
required this.name,
|
||||
required this.imageUrl,
|
||||
});
|
||||
}
|
||||
154
lib/data/models/response/file_response_model.dart
Normal file
@ -0,0 +1,154 @@
|
||||
class FileResponseModel {
|
||||
final FileModel data;
|
||||
final String message;
|
||||
final bool success;
|
||||
|
||||
FileResponseModel({
|
||||
required this.data,
|
||||
required this.message,
|
||||
required this.success,
|
||||
});
|
||||
|
||||
factory FileResponseModel.fromJson(Map<String, dynamic> json) {
|
||||
return FileResponseModel(
|
||||
data: FileModel.fromJson(json['data']),
|
||||
message: json['message'] as String,
|
||||
success: json['success'] as bool,
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() => {
|
||||
'data': data.toJson(),
|
||||
'message': message,
|
||||
'success': success,
|
||||
};
|
||||
|
||||
factory FileResponseModel.fromMap(Map<String, dynamic> map) =>
|
||||
FileResponseModel.fromJson(map);
|
||||
|
||||
Map<String, dynamic> toMap() => toJson();
|
||||
|
||||
FileResponseModel copyWith({
|
||||
FileModel? data,
|
||||
String? message,
|
||||
bool? success,
|
||||
}) {
|
||||
return FileResponseModel(
|
||||
data: data ?? this.data,
|
||||
message: message ?? this.message,
|
||||
success: success ?? this.success,
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
String toString() =>
|
||||
'FileResponseModel(data: $data, message: $message, success: $success)';
|
||||
}
|
||||
|
||||
class FileModel {
|
||||
final String id;
|
||||
final String organizationId;
|
||||
final String userId;
|
||||
final String fileName;
|
||||
final String originalName;
|
||||
final String fileUrl;
|
||||
final int fileSize;
|
||||
final String mimeType;
|
||||
final String fileType;
|
||||
final String uploadPath;
|
||||
final bool isPublic;
|
||||
final DateTime createdAt;
|
||||
final DateTime updatedAt;
|
||||
|
||||
FileModel({
|
||||
required this.id,
|
||||
required this.organizationId,
|
||||
required this.userId,
|
||||
required this.fileName,
|
||||
required this.originalName,
|
||||
required this.fileUrl,
|
||||
required this.fileSize,
|
||||
required this.mimeType,
|
||||
required this.fileType,
|
||||
required this.uploadPath,
|
||||
required this.isPublic,
|
||||
required this.createdAt,
|
||||
required this.updatedAt,
|
||||
});
|
||||
|
||||
factory FileModel.fromJson(Map<String, dynamic> json) {
|
||||
return FileModel(
|
||||
id: json['id'] as String,
|
||||
organizationId: json['organization_id'] as String,
|
||||
userId: json['user_id'] as String,
|
||||
fileName: json['file_name'] as String,
|
||||
originalName: json['original_name'] as String,
|
||||
fileUrl: json['file_url'] as String,
|
||||
fileSize: json['file_size'] as int,
|
||||
mimeType: json['mime_type'] as String,
|
||||
fileType: json['file_type'] as String,
|
||||
uploadPath: json['upload_path'] as String,
|
||||
isPublic: json['is_public'] as bool,
|
||||
createdAt: DateTime.parse(json['created_at'] as String),
|
||||
updatedAt: DateTime.parse(json['updated_at'] as String),
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toJson() => {
|
||||
'id': id,
|
||||
'organization_id': organizationId,
|
||||
'user_id': userId,
|
||||
'file_name': fileName,
|
||||
'original_name': originalName,
|
||||
'file_url': fileUrl,
|
||||
'file_size': fileSize,
|
||||
'mime_type': mimeType,
|
||||
'file_type': fileType,
|
||||
'upload_path': uploadPath,
|
||||
'is_public': isPublic,
|
||||
'created_at': createdAt.toIso8601String(),
|
||||
'updated_at': updatedAt.toIso8601String(),
|
||||
};
|
||||
|
||||
factory FileModel.fromMap(Map<String, dynamic> map) =>
|
||||
FileModel.fromJson(map);
|
||||
|
||||
Map<String, dynamic> toMap() => toJson();
|
||||
|
||||
FileModel copyWith({
|
||||
String? id,
|
||||
String? organizationId,
|
||||
String? userId,
|
||||
String? fileName,
|
||||
String? originalName,
|
||||
String? fileUrl,
|
||||
int? fileSize,
|
||||
String? mimeType,
|
||||
String? fileType,
|
||||
String? uploadPath,
|
||||
bool? isPublic,
|
||||
DateTime? createdAt,
|
||||
DateTime? updatedAt,
|
||||
}) {
|
||||
return FileModel(
|
||||
id: id ?? this.id,
|
||||
organizationId: organizationId ?? this.organizationId,
|
||||
userId: userId ?? this.userId,
|
||||
fileName: fileName ?? this.fileName,
|
||||
originalName: originalName ?? this.originalName,
|
||||
fileUrl: fileUrl ?? this.fileUrl,
|
||||
fileSize: fileSize ?? this.fileSize,
|
||||
mimeType: mimeType ?? this.mimeType,
|
||||
fileType: fileType ?? this.fileType,
|
||||
uploadPath: uploadPath ?? this.uploadPath,
|
||||
isPublic: isPublic ?? this.isPublic,
|
||||
createdAt: createdAt ?? this.createdAt,
|
||||
updatedAt: updatedAt ?? this.updatedAt,
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
String toString() {
|
||||
return 'FileModel(id: $id, organizationId: $organizationId, userId: $userId, fileName: $fileName, originalName: $originalName, fileUrl: $fileUrl, fileSize: $fileSize, mimeType: $mimeType, fileType: $fileType, uploadPath: $uploadPath, isPublic: $isPublic, createdAt: $createdAt, updatedAt: $updatedAt)';
|
||||
}
|
||||
}
|
||||
@ -1,113 +0,0 @@
|
||||
import 'dart:convert';
|
||||
|
||||
class OrderResponseModel {
|
||||
String? status;
|
||||
List<ItemOrder>? data;
|
||||
|
||||
OrderResponseModel({
|
||||
this.status,
|
||||
this.data,
|
||||
});
|
||||
|
||||
factory OrderResponseModel.fromJson(String str) =>
|
||||
OrderResponseModel.fromMap(json.decode(str));
|
||||
|
||||
String toJson() => json.encode(toMap());
|
||||
|
||||
factory OrderResponseModel.fromMap(Map<String, dynamic> json) =>
|
||||
OrderResponseModel(
|
||||
status: json["status"],
|
||||
data: json["data"] == null
|
||||
? []
|
||||
: List<ItemOrder>.from(
|
||||
json["data"]!.map((x) => ItemOrder.fromMap(x))),
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"status": status,
|
||||
"data":
|
||||
data == null ? [] : List<dynamic>.from(data!.map((x) => x.toMap())),
|
||||
};
|
||||
}
|
||||
|
||||
class ItemOrder {
|
||||
int? id;
|
||||
int? paymentAmount;
|
||||
int? subTotal;
|
||||
int? tax;
|
||||
int? discount;
|
||||
String? discountAmount;
|
||||
int? serviceCharge;
|
||||
int? total;
|
||||
String? paymentMethod;
|
||||
int? totalItem;
|
||||
int? idKasir;
|
||||
String? namaKasir;
|
||||
DateTime? transactionTime;
|
||||
DateTime? createdAt;
|
||||
DateTime? updatedAt;
|
||||
|
||||
ItemOrder({
|
||||
this.id,
|
||||
this.paymentAmount,
|
||||
this.subTotal,
|
||||
this.tax,
|
||||
this.discount,
|
||||
this.discountAmount,
|
||||
this.serviceCharge,
|
||||
this.total,
|
||||
this.paymentMethod,
|
||||
this.totalItem,
|
||||
this.idKasir,
|
||||
this.namaKasir,
|
||||
this.transactionTime,
|
||||
this.createdAt,
|
||||
this.updatedAt,
|
||||
});
|
||||
|
||||
factory ItemOrder.fromJson(String str) => ItemOrder.fromMap(json.decode(str));
|
||||
|
||||
String toJson() => json.encode(toMap());
|
||||
|
||||
factory ItemOrder.fromMap(Map<String, dynamic> json) => ItemOrder(
|
||||
id: json["id"],
|
||||
paymentAmount: json["payment_amount"],
|
||||
subTotal: json["sub_total"],
|
||||
tax: json["tax"],
|
||||
discount: json["discount"],
|
||||
discountAmount: json["discount_amount"],
|
||||
serviceCharge: json["service_charge"],
|
||||
total: json["total"],
|
||||
paymentMethod: json["payment_method"]!,
|
||||
totalItem: json["total_item"],
|
||||
idKasir: json["id_kasir"],
|
||||
namaKasir: json["nama_kasir"],
|
||||
transactionTime: json["transaction_time"] == null
|
||||
? null
|
||||
: DateTime.parse(json["transaction_time"]),
|
||||
createdAt: json["created_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["created_at"]),
|
||||
updatedAt: json["updated_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["updated_at"]),
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"id": id,
|
||||
"payment_amount": paymentAmount,
|
||||
"sub_total": subTotal,
|
||||
"tax": tax,
|
||||
"discount": discount,
|
||||
"discount_amount": discountAmount,
|
||||
"service_charge": serviceCharge,
|
||||
"total": total,
|
||||
"payment_method": paymentMethod,
|
||||
"total_item": totalItem,
|
||||
"id_kasir": idKasir,
|
||||
"nama_kasir": namaKasir,
|
||||
"transaction_time": transactionTime?.toIso8601String(),
|
||||
"created_at": createdAt?.toIso8601String(),
|
||||
"updated_at": updatedAt?.toIso8601String(),
|
||||
};
|
||||
}
|
||||
382
lib/data/models/response/order_response_model.dart
Normal file
@ -0,0 +1,382 @@
|
||||
import 'dart:convert';
|
||||
|
||||
class OrderDetailResponseModel {
|
||||
final bool? success;
|
||||
final Order? data;
|
||||
final dynamic errors;
|
||||
|
||||
OrderDetailResponseModel({
|
||||
this.success,
|
||||
this.data,
|
||||
this.errors,
|
||||
});
|
||||
|
||||
factory OrderDetailResponseModel.fromJson(String str) =>
|
||||
OrderDetailResponseModel.fromMap(json.decode(str));
|
||||
|
||||
String toJson() => json.encode(toMap());
|
||||
|
||||
factory OrderDetailResponseModel.fromMap(Map<String, dynamic> json) =>
|
||||
OrderDetailResponseModel(
|
||||
success: json["success"],
|
||||
data: json["data"] == null ? null : Order.fromMap(json["data"]),
|
||||
errors: json["errors"],
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"success": success,
|
||||
"data": data?.toMap(),
|
||||
"errors": errors,
|
||||
};
|
||||
}
|
||||
|
||||
class OrderResponseModel {
|
||||
final bool? success;
|
||||
final OrderData? data;
|
||||
|
||||
OrderResponseModel({
|
||||
this.success,
|
||||
this.data,
|
||||
});
|
||||
|
||||
factory OrderResponseModel.fromJson(String str) =>
|
||||
OrderResponseModel.fromMap(json.decode(str));
|
||||
|
||||
String toJson() => json.encode(toMap());
|
||||
|
||||
factory OrderResponseModel.fromMap(Map<String, dynamic> map) {
|
||||
return OrderResponseModel(
|
||||
success: map['success'] ?? false,
|
||||
data: OrderData.fromMap(map['data']),
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
return {
|
||||
'success': success,
|
||||
'data': data?.toMap(),
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
class OrderData {
|
||||
final List<Order>? orders;
|
||||
final List<Payment>? payments;
|
||||
final int? totalCount;
|
||||
final int? page;
|
||||
final int? limit;
|
||||
final int? totalPages;
|
||||
|
||||
OrderData({
|
||||
required this.orders,
|
||||
required this.payments,
|
||||
required this.totalCount,
|
||||
required this.page,
|
||||
required this.limit,
|
||||
required this.totalPages,
|
||||
});
|
||||
|
||||
factory OrderData.fromMap(Map<String, dynamic> map) {
|
||||
return OrderData(
|
||||
orders: map["orders"] == null
|
||||
? []
|
||||
: List<Order>.from(map['orders']?.map((x) => Order.fromMap(x))),
|
||||
payments: map["payments"] == null
|
||||
? []
|
||||
: List<Payment>.from(map['payments']?.map((x) => Payment.fromMap(x))),
|
||||
totalCount: map['total_count'],
|
||||
page: map['page'],
|
||||
limit: map['limit'],
|
||||
totalPages: map['total_pages'],
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
return {
|
||||
'orders': orders?.map((x) => x.toMap()).toList(),
|
||||
'payments': payments?.map((x) => x.toMap()).toList(),
|
||||
'total_count': totalCount,
|
||||
'page': page,
|
||||
'limit': limit,
|
||||
'total_pages': totalPages,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
class Order {
|
||||
final String? id;
|
||||
final String? orderNumber;
|
||||
final String? outletId;
|
||||
final String? userId;
|
||||
final String? tableNumber;
|
||||
final String? orderType;
|
||||
final String? status;
|
||||
final int? subtotal;
|
||||
final int? taxAmount;
|
||||
final int? discountAmount;
|
||||
final int? totalAmount;
|
||||
final int? totalCost;
|
||||
final int? remainingAmount;
|
||||
final String? paymentStatus;
|
||||
final int? refundAmount;
|
||||
final bool? isVoid;
|
||||
final bool? isRefund;
|
||||
final String? notes;
|
||||
final Map<String, dynamic>? metadata;
|
||||
final DateTime? createdAt;
|
||||
final DateTime? updatedAt;
|
||||
final List<OrderItem>? orderItems;
|
||||
final List<Payment>? payments;
|
||||
final int? totalPaid;
|
||||
final int? paymentCount;
|
||||
final String? splitType;
|
||||
|
||||
Order({
|
||||
this.id,
|
||||
this.orderNumber,
|
||||
this.outletId,
|
||||
this.userId,
|
||||
this.tableNumber,
|
||||
this.orderType,
|
||||
this.status,
|
||||
this.subtotal,
|
||||
this.taxAmount,
|
||||
this.discountAmount,
|
||||
this.totalAmount,
|
||||
this.totalCost,
|
||||
this.remainingAmount,
|
||||
this.paymentStatus,
|
||||
this.refundAmount,
|
||||
this.isVoid,
|
||||
this.isRefund,
|
||||
this.notes,
|
||||
this.metadata,
|
||||
this.createdAt,
|
||||
this.updatedAt,
|
||||
this.orderItems,
|
||||
this.payments,
|
||||
this.totalPaid,
|
||||
this.paymentCount,
|
||||
this.splitType,
|
||||
});
|
||||
|
||||
factory Order.fromMap(Map<String, dynamic> map) {
|
||||
return Order(
|
||||
id: map['id'],
|
||||
orderNumber: map['order_number'],
|
||||
outletId: map['outlet_id'],
|
||||
userId: map['user_id'],
|
||||
tableNumber: map['table_number'],
|
||||
orderType: map['order_type'],
|
||||
status: map['status'],
|
||||
subtotal: map['subtotal'],
|
||||
taxAmount: map['tax_amount'],
|
||||
discountAmount: map['discount_amount'],
|
||||
totalAmount: map['total_amount'],
|
||||
totalCost: map['total_cost'],
|
||||
remainingAmount: map['remaining_amount'],
|
||||
paymentStatus: map['payment_status'],
|
||||
refundAmount: map['refund_amount'],
|
||||
isVoid: map['is_void'],
|
||||
isRefund: map['is_refund'],
|
||||
notes: map['notes'],
|
||||
metadata: map['metadata'] ?? {},
|
||||
createdAt: DateTime.parse(map['created_at']),
|
||||
updatedAt: DateTime.parse(map['updated_at']),
|
||||
orderItems: map["order_items"] == null
|
||||
? []
|
||||
: List<OrderItem>.from(
|
||||
map['order_items'].map((x) => OrderItem.fromMap(x))),
|
||||
payments: map["payments"] == null
|
||||
? []
|
||||
: List<Payment>.from(map['payments'].map((x) => Payment.fromMap(x))),
|
||||
totalPaid: map['total_paid'],
|
||||
paymentCount: map['payment_count'],
|
||||
splitType: map['split_type'] ?? "",
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
return {
|
||||
'id': id,
|
||||
'order_number': orderNumber,
|
||||
'outlet_id': outletId,
|
||||
'user_id': userId,
|
||||
'table_number': tableNumber,
|
||||
'order_type': orderType,
|
||||
'status': status,
|
||||
'subtotal': subtotal,
|
||||
'tax_amount': taxAmount,
|
||||
'discount_amount': discountAmount,
|
||||
'total_amount': totalAmount,
|
||||
'total_cost': totalCost,
|
||||
'remaining_amount': remainingAmount,
|
||||
'payment_status': paymentStatus,
|
||||
'refund_amount': refundAmount,
|
||||
'is_void': isVoid,
|
||||
'is_refund': isRefund,
|
||||
'notes': notes,
|
||||
'metadata': metadata,
|
||||
'created_at': createdAt?.toIso8601String(),
|
||||
'updated_at': updatedAt?.toIso8601String(),
|
||||
'order_items': orderItems?.map((x) => x.toMap()).toList(),
|
||||
'payments': payments?.map((x) => x.toMap()).toList(),
|
||||
'total_paid': totalPaid,
|
||||
'payment_count': paymentCount,
|
||||
'split_type': splitType,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
class OrderItem {
|
||||
String? id;
|
||||
String? orderId;
|
||||
String? productId;
|
||||
String? productName;
|
||||
String? productVariantId;
|
||||
String? productVariantName;
|
||||
int? quantity;
|
||||
int? unitPrice;
|
||||
int? totalPrice;
|
||||
List<dynamic>? modifiers;
|
||||
String? notes;
|
||||
String? status;
|
||||
DateTime? createdAt;
|
||||
DateTime? updatedAt;
|
||||
String? printerType;
|
||||
int? paidQuantity;
|
||||
|
||||
OrderItem({
|
||||
this.id,
|
||||
this.orderId,
|
||||
this.productId,
|
||||
this.productName,
|
||||
this.productVariantId,
|
||||
this.productVariantName,
|
||||
this.quantity,
|
||||
this.unitPrice,
|
||||
this.totalPrice,
|
||||
this.modifiers,
|
||||
this.notes,
|
||||
this.status,
|
||||
this.createdAt,
|
||||
this.updatedAt,
|
||||
this.printerType,
|
||||
this.paidQuantity,
|
||||
});
|
||||
|
||||
factory OrderItem.fromMap(Map<String, dynamic> map) {
|
||||
return OrderItem(
|
||||
id: map['id'],
|
||||
orderId: map['order_id'],
|
||||
productId: map['product_id'],
|
||||
productName: map['product_name'],
|
||||
productVariantId: map['product_variant_id'],
|
||||
productVariantName: map['product_variant_name'],
|
||||
quantity: map['quantity'],
|
||||
unitPrice: map['unit_price'],
|
||||
totalPrice: map['total_price'],
|
||||
modifiers:
|
||||
map['modifiers'] == null ? [] : List<dynamic>.from(map['modifiers']),
|
||||
notes: map['notes'],
|
||||
status: map['status'],
|
||||
createdAt: DateTime.parse(map['created_at']),
|
||||
updatedAt: DateTime.parse(map['updated_at']),
|
||||
printerType: map['printer_type'],
|
||||
paidQuantity: map['paid_quantity'],
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
return {
|
||||
'id': id,
|
||||
'order_id': orderId,
|
||||
'product_id': productId,
|
||||
'product_name': productName,
|
||||
'product_variant_id': productVariantId,
|
||||
'product_variant_name': productVariantName,
|
||||
'quantity': quantity,
|
||||
'unit_price': unitPrice,
|
||||
'total_price': totalPrice,
|
||||
'modifiers': modifiers,
|
||||
'notes': notes,
|
||||
'status': status,
|
||||
'created_at': createdAt?.toIso8601String(),
|
||||
'updated_at': updatedAt?.toIso8601String(),
|
||||
'printer_type': printerType,
|
||||
'paid_quantity': paidQuantity,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
class Payment {
|
||||
final String? id;
|
||||
final String? orderId;
|
||||
final String? paymentMethodId;
|
||||
final String? paymentMethodName;
|
||||
final String? paymentMethodType;
|
||||
final int? amount;
|
||||
final String? status;
|
||||
final int? splitNumber;
|
||||
final int? splitTotal;
|
||||
final String? splitDescription;
|
||||
final int? refundAmount;
|
||||
final Map<String, dynamic>? metadata;
|
||||
final DateTime? createdAt;
|
||||
final DateTime? updatedAt;
|
||||
|
||||
Payment({
|
||||
this.id,
|
||||
this.orderId,
|
||||
this.paymentMethodId,
|
||||
this.paymentMethodName,
|
||||
this.paymentMethodType,
|
||||
this.amount,
|
||||
this.status,
|
||||
this.splitNumber,
|
||||
this.splitTotal,
|
||||
this.splitDescription,
|
||||
this.refundAmount,
|
||||
this.metadata,
|
||||
this.createdAt,
|
||||
this.updatedAt,
|
||||
});
|
||||
|
||||
factory Payment.fromMap(Map<String, dynamic> map) {
|
||||
return Payment(
|
||||
id: map['id'],
|
||||
orderId: map['order_id'],
|
||||
paymentMethodId: map['payment_method_id'],
|
||||
paymentMethodName: map['payment_method_name'],
|
||||
paymentMethodType: map['payment_method_type'],
|
||||
amount: map['amount'],
|
||||
status: map['status'],
|
||||
splitNumber: map['split_number'],
|
||||
splitTotal: map['split_total'],
|
||||
splitDescription: map['split_description'],
|
||||
refundAmount: map['refund_amount'],
|
||||
metadata: map['metadata'] ?? {},
|
||||
createdAt: DateTime.parse(map['created_at']),
|
||||
updatedAt: DateTime.parse(map['updated_at']),
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
return {
|
||||
'id': id,
|
||||
'order_id': orderId,
|
||||
'payment_method_id': paymentMethodId,
|
||||
'payment_method_name': paymentMethodName,
|
||||
'payment_method_type': paymentMethodType,
|
||||
'amount': amount,
|
||||
'status': status,
|
||||
'split_number': splitNumber,
|
||||
'split_total': splitTotal,
|
||||
'split_description': splitDescription,
|
||||
'refund_amount': refundAmount,
|
||||
'metadata': metadata,
|
||||
'created_at': createdAt?.toIso8601String(),
|
||||
'updated_at': updatedAt?.toIso8601String(),
|
||||
};
|
||||
}
|
||||
}
|
||||
@ -0,0 +1,173 @@
|
||||
class PaymentMethodAnalyticResponseModel {
|
||||
final bool success;
|
||||
final PaymentMethodAnalyticData data;
|
||||
final dynamic errors;
|
||||
|
||||
PaymentMethodAnalyticResponseModel({
|
||||
required this.success,
|
||||
required this.data,
|
||||
this.errors,
|
||||
});
|
||||
|
||||
factory PaymentMethodAnalyticResponseModel.fromJson(
|
||||
Map<String, dynamic> json) =>
|
||||
PaymentMethodAnalyticResponseModel.fromMap(json);
|
||||
|
||||
Map<String, dynamic> toJson() => toMap();
|
||||
|
||||
factory PaymentMethodAnalyticResponseModel.fromMap(Map<String, dynamic> map) {
|
||||
return PaymentMethodAnalyticResponseModel(
|
||||
success: map['success'],
|
||||
data: PaymentMethodAnalyticData.fromMap(map['data']),
|
||||
errors: map['errors'],
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
return {
|
||||
'success': success,
|
||||
'data': data.toMap(),
|
||||
'errors': errors,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
class PaymentMethodAnalyticData {
|
||||
final String organizationId;
|
||||
final String outletId;
|
||||
final DateTime dateFrom;
|
||||
final DateTime dateTo;
|
||||
final String groupBy;
|
||||
final PaymentSummary summary;
|
||||
final List<PaymentMethodAnalyticItem> data;
|
||||
|
||||
PaymentMethodAnalyticData({
|
||||
required this.organizationId,
|
||||
required this.outletId,
|
||||
required this.dateFrom,
|
||||
required this.dateTo,
|
||||
required this.groupBy,
|
||||
required this.summary,
|
||||
required this.data,
|
||||
});
|
||||
|
||||
factory PaymentMethodAnalyticData.fromJson(Map<String, dynamic> json) =>
|
||||
PaymentMethodAnalyticData.fromMap(json);
|
||||
|
||||
Map<String, dynamic> toJson() => toMap();
|
||||
|
||||
factory PaymentMethodAnalyticData.fromMap(Map<String, dynamic> map) {
|
||||
return PaymentMethodAnalyticData(
|
||||
organizationId: map['organization_id'],
|
||||
outletId: map['outlet_id'],
|
||||
dateFrom: DateTime.parse(map['date_from']),
|
||||
dateTo: DateTime.parse(map['date_to']),
|
||||
groupBy: map['group_by'],
|
||||
summary: PaymentSummary.fromMap(map['summary']),
|
||||
data: map['data'] == null
|
||||
? []
|
||||
: List<PaymentMethodAnalyticItem>.from(
|
||||
map['data']?.map((x) => PaymentMethodAnalyticItem.fromMap(x)) ??
|
||||
[],
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
return {
|
||||
'organization_id': organizationId,
|
||||
'outlet_id': outletId,
|
||||
'date_from': dateFrom.toIso8601String(),
|
||||
'date_to': dateTo.toIso8601String(),
|
||||
'group_by': groupBy,
|
||||
'summary': summary.toMap(),
|
||||
'data': data.map((x) => x.toMap()).toList(),
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
class PaymentSummary {
|
||||
final int totalAmount;
|
||||
final int totalOrders;
|
||||
final int totalPayments;
|
||||
final double averageOrderValue;
|
||||
|
||||
PaymentSummary({
|
||||
required this.totalAmount,
|
||||
required this.totalOrders,
|
||||
required this.totalPayments,
|
||||
required this.averageOrderValue,
|
||||
});
|
||||
|
||||
factory PaymentSummary.fromJson(Map<String, dynamic> json) =>
|
||||
PaymentSummary.fromMap(json);
|
||||
|
||||
Map<String, dynamic> toJson() => toMap();
|
||||
|
||||
factory PaymentSummary.fromMap(Map<String, dynamic> map) {
|
||||
return PaymentSummary(
|
||||
totalAmount: map['total_amount'],
|
||||
totalOrders: map['total_orders'],
|
||||
totalPayments: map['total_payments'],
|
||||
averageOrderValue: (map['average_order_value'] as num).toDouble(),
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
return {
|
||||
'total_amount': totalAmount,
|
||||
'total_orders': totalOrders,
|
||||
'total_payments': totalPayments,
|
||||
'average_order_value': averageOrderValue,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
class PaymentMethodAnalyticItem {
|
||||
final String paymentMethodId;
|
||||
final String paymentMethodName;
|
||||
final String paymentMethodType;
|
||||
final int totalAmount;
|
||||
final int orderCount;
|
||||
final int paymentCount;
|
||||
final int percentage;
|
||||
|
||||
PaymentMethodAnalyticItem({
|
||||
required this.paymentMethodId,
|
||||
required this.paymentMethodName,
|
||||
required this.paymentMethodType,
|
||||
required this.totalAmount,
|
||||
required this.orderCount,
|
||||
required this.paymentCount,
|
||||
required this.percentage,
|
||||
});
|
||||
|
||||
factory PaymentMethodAnalyticItem.fromJson(Map<String, dynamic> json) =>
|
||||
PaymentMethodAnalyticItem.fromMap(json);
|
||||
|
||||
Map<String, dynamic> toJson() => toMap();
|
||||
|
||||
factory PaymentMethodAnalyticItem.fromMap(Map<String, dynamic> map) {
|
||||
return PaymentMethodAnalyticItem(
|
||||
paymentMethodId: map['payment_method_id'],
|
||||
paymentMethodName: map['payment_method_name'],
|
||||
paymentMethodType: map['payment_method_type'],
|
||||
totalAmount: map['total_amount'],
|
||||
orderCount: map['order_count'],
|
||||
paymentCount: map['payment_count'],
|
||||
percentage: map['percentage'],
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
return {
|
||||
'payment_method_id': paymentMethodId,
|
||||
'payment_method_name': paymentMethodName,
|
||||
'payment_method_type': paymentMethodType,
|
||||
'total_amount': totalAmount,
|
||||
'order_count': orderCount,
|
||||
'payment_count': paymentCount,
|
||||
'percentage': percentage,
|
||||
};
|
||||
}
|
||||
}
|
||||
@ -1,12 +1,14 @@
|
||||
import 'dart:convert';
|
||||
|
||||
class PaymentMethodsResponseModel {
|
||||
final String? status;
|
||||
final List<PaymentMethod>? data;
|
||||
final bool? success;
|
||||
final PaymentMethodsData? data;
|
||||
final dynamic errors;
|
||||
|
||||
PaymentMethodsResponseModel({
|
||||
this.status,
|
||||
this.success,
|
||||
this.data,
|
||||
this.errors,
|
||||
});
|
||||
|
||||
factory PaymentMethodsResponseModel.fromJson(String str) =>
|
||||
@ -16,51 +18,83 @@ class PaymentMethodsResponseModel {
|
||||
|
||||
factory PaymentMethodsResponseModel.fromMap(Map<String, dynamic> json) =>
|
||||
PaymentMethodsResponseModel(
|
||||
status: json["status"],
|
||||
success: json["success"],
|
||||
data: json["data"] == null
|
||||
? []
|
||||
: List<PaymentMethod>.from(
|
||||
json["data"]!.map((x) => PaymentMethod.fromMap(x))),
|
||||
? null
|
||||
: PaymentMethodsData.fromMap(json["data"]),
|
||||
errors: json["errors"],
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"status": status,
|
||||
"data": data == null
|
||||
"success": success,
|
||||
"data": data?.toMap(),
|
||||
"errors": errors,
|
||||
};
|
||||
}
|
||||
|
||||
class PaymentMethodsData {
|
||||
final List<PaymentMethod>? paymentMethods;
|
||||
final int? totalCount;
|
||||
final int? page;
|
||||
final int? limit;
|
||||
final int? totalPages;
|
||||
|
||||
PaymentMethodsData({
|
||||
this.paymentMethods,
|
||||
this.totalCount,
|
||||
this.page,
|
||||
this.limit,
|
||||
this.totalPages,
|
||||
});
|
||||
|
||||
factory PaymentMethodsData.fromMap(Map<String, dynamic> json) =>
|
||||
PaymentMethodsData(
|
||||
paymentMethods: json["payment_methods"] == null
|
||||
? []
|
||||
: List<dynamic>.from(data!.map((x) => x.toMap())),
|
||||
: List<PaymentMethod>.from(
|
||||
json["payment_methods"].map((x) => PaymentMethod.fromMap(x))),
|
||||
totalCount: json["total_count"],
|
||||
page: json["page"],
|
||||
limit: json["limit"],
|
||||
totalPages: json["total_pages"],
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"payment_methods": paymentMethods == null
|
||||
? []
|
||||
: List<dynamic>.from(paymentMethods!.map((x) => x.toMap())),
|
||||
"total_count": totalCount,
|
||||
"page": page,
|
||||
"limit": limit,
|
||||
"total_pages": totalPages,
|
||||
};
|
||||
}
|
||||
|
||||
class PaymentMethod {
|
||||
final int? id;
|
||||
final String? id;
|
||||
final String? organizationId;
|
||||
final String? name;
|
||||
final String? description;
|
||||
final String? type;
|
||||
final bool? isActive;
|
||||
final int? sortOrder;
|
||||
final DateTime? createdAt;
|
||||
final DateTime? updatedAt;
|
||||
|
||||
PaymentMethod({
|
||||
this.id,
|
||||
this.organizationId,
|
||||
this.name,
|
||||
this.description,
|
||||
this.type,
|
||||
this.isActive,
|
||||
this.sortOrder,
|
||||
this.createdAt,
|
||||
this.updatedAt,
|
||||
});
|
||||
|
||||
factory PaymentMethod.fromJson(String str) =>
|
||||
PaymentMethod.fromMap(json.decode(str));
|
||||
|
||||
String toJson() => json.encode(toMap());
|
||||
|
||||
factory PaymentMethod.fromMap(Map<String, dynamic> json) => PaymentMethod(
|
||||
id: json["id"],
|
||||
organizationId: json["organization_id"],
|
||||
name: json["name"],
|
||||
description: json["description"],
|
||||
type: json["type"],
|
||||
isActive: json["is_active"],
|
||||
sortOrder: json["sort_order"],
|
||||
createdAt: json["created_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["created_at"]),
|
||||
@ -71,10 +105,10 @@ class PaymentMethod {
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"id": id,
|
||||
"organization_id": organizationId,
|
||||
"name": name,
|
||||
"description": description,
|
||||
"type": type,
|
||||
"is_active": isActive,
|
||||
"sort_order": sortOrder,
|
||||
"created_at": createdAt?.toIso8601String(),
|
||||
"updated_at": updatedAt?.toIso8601String(),
|
||||
};
|
||||
|
||||
95
lib/data/models/response/payment_response_model.dart
Normal file
@ -0,0 +1,95 @@
|
||||
import 'dart:convert';
|
||||
|
||||
class PaymentSuccessResponseModel {
|
||||
final bool? success;
|
||||
final PaymentData? data;
|
||||
final dynamic errors;
|
||||
|
||||
PaymentSuccessResponseModel({
|
||||
this.success,
|
||||
this.data,
|
||||
this.errors,
|
||||
});
|
||||
|
||||
factory PaymentSuccessResponseModel.fromJson(String str) =>
|
||||
PaymentSuccessResponseModel.fromMap(json.decode(str));
|
||||
|
||||
String toJson() => json.encode(toMap());
|
||||
|
||||
factory PaymentSuccessResponseModel.fromMap(Map<String, dynamic> json) =>
|
||||
PaymentSuccessResponseModel(
|
||||
success: json["success"],
|
||||
data: json["data"] == null ? null : PaymentData.fromMap(json["data"]),
|
||||
errors: json["errors"],
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"success": success,
|
||||
"data": data?.toMap(),
|
||||
"errors": errors,
|
||||
};
|
||||
}
|
||||
|
||||
class PaymentData {
|
||||
final String? id;
|
||||
final String? orderId;
|
||||
final String? paymentMethodId;
|
||||
final int? amount;
|
||||
final String? status;
|
||||
final String? transactionId;
|
||||
final int? splitNumber;
|
||||
final int? splitTotal;
|
||||
final String? splitDescription;
|
||||
final int? refundAmount;
|
||||
final DateTime? createdAt;
|
||||
final DateTime? updatedAt;
|
||||
|
||||
PaymentData({
|
||||
this.id,
|
||||
this.orderId,
|
||||
this.paymentMethodId,
|
||||
this.amount,
|
||||
this.status,
|
||||
this.transactionId,
|
||||
this.splitNumber,
|
||||
this.splitTotal,
|
||||
this.splitDescription,
|
||||
this.refundAmount,
|
||||
this.createdAt,
|
||||
this.updatedAt,
|
||||
});
|
||||
|
||||
factory PaymentData.fromMap(Map<String, dynamic> json) => PaymentData(
|
||||
id: json["id"],
|
||||
orderId: json["order_id"],
|
||||
paymentMethodId: json["payment_method_id"],
|
||||
amount: json["amount"],
|
||||
status: json["status"],
|
||||
transactionId: json["transaction_id"],
|
||||
splitNumber: json["split_number"],
|
||||
splitTotal: json["split_total"],
|
||||
splitDescription: json["split_description"],
|
||||
refundAmount: json["refund_amount"],
|
||||
createdAt: json["created_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["created_at"]),
|
||||
updatedAt: json["updated_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["updated_at"]),
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"id": id,
|
||||
"order_id": orderId,
|
||||
"payment_method_id": paymentMethodId,
|
||||
"amount": amount,
|
||||
"status": status,
|
||||
"transaction_id": transactionId,
|
||||
"split_number": splitNumber,
|
||||
"split_total": splitTotal,
|
||||
"split_description": splitDescription,
|
||||
"refund_amount": refundAmount,
|
||||
"created_at": createdAt?.toIso8601String(),
|
||||
"updated_at": updatedAt?.toIso8601String(),
|
||||
};
|
||||
}
|
||||
145
lib/data/models/response/product_analytic_response_model.dart
Normal file
@ -0,0 +1,145 @@
|
||||
class ProductAnalyticResponseModel {
|
||||
final bool success;
|
||||
final ProductAnalyticData data;
|
||||
final dynamic errors;
|
||||
|
||||
ProductAnalyticResponseModel({
|
||||
required this.success,
|
||||
required this.data,
|
||||
this.errors,
|
||||
});
|
||||
|
||||
factory ProductAnalyticResponseModel.fromJson(Map<String, dynamic> json) =>
|
||||
ProductAnalyticResponseModel.fromMap(json);
|
||||
|
||||
Map<String, dynamic> toJson() => toMap();
|
||||
|
||||
factory ProductAnalyticResponseModel.fromMap(Map<String, dynamic> map) {
|
||||
return ProductAnalyticResponseModel(
|
||||
success: map['success'] ?? false,
|
||||
data: ProductAnalyticData.fromMap(map['data']),
|
||||
errors: map['errors'],
|
||||
);
|
||||
}
|
||||
|
||||
Map<String, dynamic> toMap() {
|
||||
return {
|
||||
'success': success,
|
||||
'data': data.toMap(),
|
||||
'errors': errors,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
class ProductAnalyticData {
|
||||
final String organizationId;
|
||||
final String outletId;
|
||||
final DateTime dateFrom;
|
||||
final DateTime dateTo;
|
||||
final List<ProductAnalyticItem> data;
|
||||
|
||||
ProductAnalyticData({
|
||||
required this.organizationId,
|
||||
required this.outletId,
|
||||
required this.dateFrom,
|
||||
required this.dateTo,
|
||||
required this.data,
|
||||
});
|
||||
|
||||
factory ProductAnalyticData.fromMap(Map<String, dynamic> map) =>
|
||||
ProductAnalyticData(
|
||||
organizationId: map['organization_id'],
|
||||
outletId: map['outlet_id'],
|
||||
dateFrom: DateTime.parse(map['date_from']),
|
||||
dateTo: DateTime.parse(map['date_to']),
|
||||
data: map['data'] == null
|
||||
? []
|
||||
: List<ProductAnalyticItem>.from(
|
||||
map['data'].map((x) => ProductAnalyticItem.fromMap(x)),
|
||||
),
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
'organization_id': organizationId,
|
||||
'outlet_id': outletId,
|
||||
'date_from': dateFrom.toIso8601String(),
|
||||
'date_to': dateTo.toIso8601String(),
|
||||
'data': data.map((x) => x.toMap()).toList(),
|
||||
};
|
||||
}
|
||||
|
||||
class ProductAnalyticItem {
|
||||
final String productId;
|
||||
final String productName;
|
||||
final String categoryId;
|
||||
final String categoryName;
|
||||
final int quantitySold;
|
||||
final int revenue;
|
||||
final double averagePrice;
|
||||
final int orderCount;
|
||||
|
||||
ProductAnalyticItem({
|
||||
required this.productId,
|
||||
required this.productName,
|
||||
required this.categoryId,
|
||||
required this.categoryName,
|
||||
required this.quantitySold,
|
||||
required this.revenue,
|
||||
required this.averagePrice,
|
||||
required this.orderCount,
|
||||
});
|
||||
|
||||
factory ProductAnalyticItem.fromMap(Map<String, dynamic> map) =>
|
||||
ProductAnalyticItem(
|
||||
productId: map['product_id'],
|
||||
productName: map['product_name'],
|
||||
categoryId: map['category_id'],
|
||||
categoryName: map['category_name'],
|
||||
quantitySold: map['quantity_sold'],
|
||||
revenue: map['revenue'],
|
||||
averagePrice: (map['average_price'] as num).toDouble(),
|
||||
orderCount: map['order_count'],
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
'product_id': productId,
|
||||
'product_name': productName,
|
||||
'category_id': categoryId,
|
||||
'category_name': categoryName,
|
||||
'quantity_sold': quantitySold,
|
||||
'revenue': revenue,
|
||||
'average_price': averagePrice,
|
||||
'order_count': orderCount,
|
||||
};
|
||||
}
|
||||
|
||||
class ProductInsights {
|
||||
final List<ProductAnalyticItem> topProducts;
|
||||
final ProductAnalyticItem? bestProduct;
|
||||
final List<CategorySummary> categorySummary;
|
||||
final int totalProducts;
|
||||
final int totalRevenue;
|
||||
final int totalQuantitySold;
|
||||
|
||||
ProductInsights({
|
||||
required this.topProducts,
|
||||
required this.bestProduct,
|
||||
required this.categorySummary,
|
||||
required this.totalProducts,
|
||||
required this.totalRevenue,
|
||||
required this.totalQuantitySold,
|
||||
});
|
||||
}
|
||||
|
||||
// Category summary class
|
||||
class CategorySummary {
|
||||
final String categoryName;
|
||||
int productCount;
|
||||
int totalRevenue;
|
||||
|
||||
CategorySummary({
|
||||
required this.categoryName,
|
||||
required this.productCount,
|
||||
required this.totalRevenue,
|
||||
});
|
||||
}
|
||||
@ -1,17 +1,15 @@
|
||||
// ignore_for_file: public_member_api_docs, sort_constructors_first
|
||||
import 'dart:convert';
|
||||
|
||||
import 'package:flutter/foundation.dart';
|
||||
|
||||
import 'package:enaklo_pos/presentation/home/pages/confirm_payment_page.dart';
|
||||
|
||||
class ProductResponseModel {
|
||||
final String? status;
|
||||
final List<Product>? data;
|
||||
final bool? success;
|
||||
final ProductData? data;
|
||||
final dynamic errors;
|
||||
|
||||
ProductResponseModel({
|
||||
this.status,
|
||||
this.success,
|
||||
this.data,
|
||||
this.errors,
|
||||
});
|
||||
|
||||
factory ProductResponseModel.fromJson(String str) =>
|
||||
@ -21,50 +19,90 @@ class ProductResponseModel {
|
||||
|
||||
factory ProductResponseModel.fromMap(Map<String, dynamic> json) =>
|
||||
ProductResponseModel(
|
||||
status: json["status"],
|
||||
data: json["data"] == null
|
||||
? []
|
||||
: List<Product>.from(json["data"]!.map((x) => Product.fromMap(x))),
|
||||
success: json["success"],
|
||||
data: json["data"] == null ? null : ProductData.fromMap(json["data"]),
|
||||
errors: json["errors"],
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"status": status,
|
||||
"data":
|
||||
data == null ? [] : List<dynamic>.from(data!.map((x) => x.toMap())),
|
||||
"success": success,
|
||||
"data": data?.toMap(),
|
||||
"errors": errors,
|
||||
};
|
||||
}
|
||||
|
||||
class ProductData {
|
||||
final List<Product>? products;
|
||||
final int? totalCount;
|
||||
final int? page;
|
||||
final int? limit;
|
||||
final int? totalPages;
|
||||
|
||||
ProductData({
|
||||
this.products,
|
||||
this.totalCount,
|
||||
this.page,
|
||||
this.limit,
|
||||
this.totalPages,
|
||||
});
|
||||
|
||||
factory ProductData.fromMap(Map<String, dynamic> json) => ProductData(
|
||||
products: json["products"] == null
|
||||
? []
|
||||
: List<Product>.from(
|
||||
json["products"].map((x) => Product.fromMap(x))),
|
||||
totalCount: json["total_count"],
|
||||
page: json["page"],
|
||||
limit: json["limit"],
|
||||
totalPages: json["total_pages"],
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"products": products == null
|
||||
? []
|
||||
: List<dynamic>.from(products!.map((x) => x.toMap())),
|
||||
"total_count": totalCount,
|
||||
"page": page,
|
||||
"limit": limit,
|
||||
"total_pages": totalPages,
|
||||
};
|
||||
}
|
||||
|
||||
class Product {
|
||||
final int? id;
|
||||
final int? productId;
|
||||
final int? categoryId;
|
||||
final String? id;
|
||||
final String? organizationId;
|
||||
final String? categoryId;
|
||||
final String? sku;
|
||||
final String? name;
|
||||
final String? description;
|
||||
final String? image;
|
||||
final String? price;
|
||||
final int? stock;
|
||||
final int? status;
|
||||
final int? isFavorite;
|
||||
final int? price;
|
||||
final int? cost;
|
||||
final String? businessType;
|
||||
final String? imageUrl;
|
||||
final String? printerType;
|
||||
final Map<String, dynamic>? metadata;
|
||||
final bool? isActive;
|
||||
final DateTime? createdAt;
|
||||
final DateTime? updatedAt;
|
||||
final Category? category;
|
||||
final String? printerType;
|
||||
final List<ProductVariant>? variants;
|
||||
|
||||
Product({
|
||||
this.id,
|
||||
this.productId,
|
||||
this.organizationId,
|
||||
this.categoryId,
|
||||
this.sku,
|
||||
this.name,
|
||||
this.description,
|
||||
this.image,
|
||||
this.price,
|
||||
this.stock,
|
||||
this.status,
|
||||
this.isFavorite,
|
||||
this.cost,
|
||||
this.businessType,
|
||||
this.imageUrl,
|
||||
this.printerType,
|
||||
this.metadata,
|
||||
this.isActive,
|
||||
this.createdAt,
|
||||
this.updatedAt,
|
||||
this.category,
|
||||
this.printerType,
|
||||
this.variants,
|
||||
});
|
||||
|
||||
factory Product.fromJson(String str) => Product.fromMap(json.decode(str));
|
||||
@ -72,91 +110,102 @@ class Product {
|
||||
String toJson() => json.encode(toMap());
|
||||
|
||||
factory Product.fromMap(Map<String, dynamic> json) => Product(
|
||||
id: json["id"] is String ? int.tryParse(json["id"]) : json["id"],
|
||||
productId: json["product_id"] is String ? int.tryParse(json["product_id"]) : json["product_id"],
|
||||
categoryId: json["category_id"] is String
|
||||
? int.tryParse(json["category_id"])
|
||||
: json["category_id"],
|
||||
id: json["id"],
|
||||
organizationId: json["organization_id"],
|
||||
categoryId: json["category_id"],
|
||||
sku: json["sku"],
|
||||
name: json["name"],
|
||||
description: json["description"],
|
||||
image: json["image"],
|
||||
// price: json["price"].substring(0, json["price"].length - 3),
|
||||
price: json["price"].toString().replaceAll('.00', ''),
|
||||
stock: json["stock"] is String ? int.tryParse(json["stock"]) : json["stock"],
|
||||
status: json["status"] is String ? int.tryParse(json["status"]) : json["status"],
|
||||
isFavorite: json["is_favorite"] is String ? int.tryParse(json["is_favorite"]) : json["is_favorite"],
|
||||
price: json["price"],
|
||||
cost: json["cost"],
|
||||
businessType: json["business_type"],
|
||||
imageUrl: json["image_url"],
|
||||
printerType: json["printer_type"],
|
||||
metadata: json["metadata"] ?? {},
|
||||
isActive: json["is_active"],
|
||||
createdAt: json["created_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["created_at"]),
|
||||
updatedAt: json["updated_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["updated_at"]),
|
||||
category: json["category"] == null
|
||||
? null
|
||||
: Category.fromMap(json["category"]),
|
||||
printerType: json["printer_type"] ?? 'bar',
|
||||
variants: json["variants"] == null
|
||||
? []
|
||||
: List<ProductVariant>.from(
|
||||
json["variants"].map((x) => ProductVariant.fromMap(x))),
|
||||
);
|
||||
|
||||
factory Product.fromOrderMap(Map<String, dynamic> json) => Product(
|
||||
id: json["id_product"],
|
||||
price: json["price"].toString(),
|
||||
price: json["price"],
|
||||
);
|
||||
|
||||
factory Product.fromLocalMap(Map<String, dynamic> json) => Product(
|
||||
id: json["id"],
|
||||
productId: json["product_id"],
|
||||
categoryId: json["categoryId"],
|
||||
category: Category(
|
||||
id: json["categoryId"],
|
||||
name: json["categoryName"],
|
||||
),
|
||||
organizationId: json["organization_id"],
|
||||
categoryId: json["category_id"],
|
||||
sku: json["sku"],
|
||||
name: json["name"],
|
||||
description: json["description"],
|
||||
image: json["image"],
|
||||
price: json["price"],
|
||||
stock: json["stock"],
|
||||
status: json["status"],
|
||||
isFavorite: json["isFavorite"],
|
||||
createdAt: json["createdAt"] == null
|
||||
cost: json["cost"],
|
||||
businessType: json["business_type"],
|
||||
imageUrl: json["image_url"],
|
||||
printerType: json["printer_type"],
|
||||
metadata: json["metadata"] ?? {},
|
||||
isActive: json["is_active"],
|
||||
createdAt: json["created_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["createdAt"]),
|
||||
updatedAt: json["updatedAt"] == null
|
||||
: DateTime.parse(json["created_at"]),
|
||||
updatedAt: json["updated_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["updatedAt"]),
|
||||
printerType: json["printer_type"] ?? 'bar',
|
||||
: DateTime.parse(json["updated_at"]),
|
||||
variants: json["variants"] == null
|
||||
? []
|
||||
: List<ProductVariant>.from(
|
||||
json["variants"].map((x) => ProductVariant.fromMap(x))),
|
||||
);
|
||||
|
||||
Map<String, dynamic> toLocalMap() => {
|
||||
"product_id": id,
|
||||
"categoryId": categoryId,
|
||||
"categoryName": category?.name,
|
||||
"id": id,
|
||||
"organization_id": organizationId,
|
||||
"category_id": categoryId,
|
||||
"sku": sku,
|
||||
"name": name,
|
||||
"description": description,
|
||||
"image": image,
|
||||
"price": price?.replaceAll(RegExp(r'\.0+$'), ''),
|
||||
"stock": stock,
|
||||
"status": status,
|
||||
"isFavorite": isFavorite,
|
||||
"createdAt": createdAt?.toIso8601String(),
|
||||
"updatedAt": updatedAt?.toIso8601String(),
|
||||
"price": price,
|
||||
"cost": cost,
|
||||
"business_type": businessType,
|
||||
"image_url": imageUrl,
|
||||
"printer_type": printerType,
|
||||
"metadata": metadata,
|
||||
"is_active": isActive,
|
||||
"created_at": createdAt?.toIso8601String(),
|
||||
"updated_at": updatedAt?.toIso8601String(),
|
||||
"variants": variants == null
|
||||
? []
|
||||
: List<dynamic>.from(variants!.map((x) => x.toMap())),
|
||||
};
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"id": id,
|
||||
"product_id": productId,
|
||||
"organization_id": organizationId,
|
||||
"category_id": categoryId,
|
||||
"sku": sku,
|
||||
"name": name,
|
||||
"description": description,
|
||||
"image": image,
|
||||
"price": price,
|
||||
"stock": stock,
|
||||
"status": status,
|
||||
"is_favorite": isFavorite,
|
||||
"cost": cost,
|
||||
"business_type": businessType,
|
||||
"image_url": imageUrl,
|
||||
"printer_type": printerType,
|
||||
"metadata": metadata,
|
||||
"is_active": isActive,
|
||||
"created_at": createdAt?.toIso8601String(),
|
||||
"updated_at": updatedAt?.toIso8601String(),
|
||||
"category": category?.toMap(),
|
||||
"printer_type": printerType,
|
||||
"variants": variants == null
|
||||
? []
|
||||
: List<dynamic>.from(variants!.map((x) => x.toMap())),
|
||||
};
|
||||
|
||||
@override
|
||||
@ -164,70 +213,88 @@ class Product {
|
||||
if (identical(this, other)) return true;
|
||||
|
||||
return other.id == id &&
|
||||
other.productId == productId &&
|
||||
other.organizationId == organizationId &&
|
||||
other.categoryId == categoryId &&
|
||||
other.sku == sku &&
|
||||
other.name == name &&
|
||||
other.description == description &&
|
||||
other.image == image &&
|
||||
other.price == price &&
|
||||
other.stock == stock &&
|
||||
other.status == status &&
|
||||
other.isFavorite == isFavorite &&
|
||||
other.cost == cost &&
|
||||
other.businessType == businessType &&
|
||||
other.imageUrl == imageUrl &&
|
||||
other.printerType == printerType &&
|
||||
other.metadata == metadata &&
|
||||
other.isActive == isActive &&
|
||||
other.createdAt == createdAt &&
|
||||
other.updatedAt == updatedAt &&
|
||||
other.category == category &&
|
||||
other.printerType == printerType;
|
||||
_listEquals(other.variants, variants);
|
||||
}
|
||||
|
||||
bool _listEquals(List<ProductVariant>? a, List<ProductVariant>? b) {
|
||||
if (a == null && b == null) return true;
|
||||
if (a == null || b == null) return false;
|
||||
if (a.length != b.length) return false;
|
||||
for (int i = 0; i < a.length; i++) {
|
||||
if (a[i] != b[i]) return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
@override
|
||||
int get hashCode {
|
||||
return id.hashCode ^
|
||||
productId.hashCode ^
|
||||
organizationId.hashCode ^
|
||||
categoryId.hashCode ^
|
||||
sku.hashCode ^
|
||||
name.hashCode ^
|
||||
description.hashCode ^
|
||||
image.hashCode ^
|
||||
price.hashCode ^
|
||||
stock.hashCode ^
|
||||
status.hashCode ^
|
||||
isFavorite.hashCode ^
|
||||
cost.hashCode ^
|
||||
businessType.hashCode ^
|
||||
imageUrl.hashCode ^
|
||||
printerType.hashCode ^
|
||||
metadata.hashCode ^
|
||||
isActive.hashCode ^
|
||||
createdAt.hashCode ^
|
||||
updatedAt.hashCode ^
|
||||
category.hashCode ^
|
||||
printerType.hashCode;
|
||||
variants.hashCode;
|
||||
}
|
||||
|
||||
Product copyWith({
|
||||
int? id,
|
||||
int? productId,
|
||||
int? categoryId,
|
||||
String? id,
|
||||
String? organizationId,
|
||||
String? categoryId,
|
||||
String? sku,
|
||||
String? name,
|
||||
String? description,
|
||||
String? image,
|
||||
String? price,
|
||||
int? stock,
|
||||
int? status,
|
||||
int? isFavorite,
|
||||
int? price,
|
||||
int? cost,
|
||||
String? businessType,
|
||||
String? imageUrl,
|
||||
String? printerType,
|
||||
Map<String, dynamic>? metadata,
|
||||
bool? isActive,
|
||||
DateTime? createdAt,
|
||||
DateTime? updatedAt,
|
||||
Category? category,
|
||||
String? printerType,
|
||||
List<ProductVariant>? variants,
|
||||
}) {
|
||||
return Product(
|
||||
id: id ?? this.id,
|
||||
productId: productId ?? this.productId,
|
||||
organizationId: organizationId ?? this.organizationId,
|
||||
categoryId: categoryId ?? this.categoryId,
|
||||
sku: sku ?? this.sku,
|
||||
name: name ?? this.name,
|
||||
description: description ?? this.description,
|
||||
image: image ?? this.image,
|
||||
price: price ?? this.price,
|
||||
stock: stock ?? this.stock,
|
||||
status: status ?? this.status,
|
||||
isFavorite: isFavorite ?? this.isFavorite,
|
||||
cost: cost ?? this.cost,
|
||||
businessType: businessType ?? this.businessType,
|
||||
imageUrl: imageUrl ?? this.imageUrl,
|
||||
printerType: printerType ?? this.printerType,
|
||||
metadata: metadata ?? this.metadata,
|
||||
isActive: isActive ?? this.isActive,
|
||||
createdAt: createdAt ?? this.createdAt,
|
||||
updatedAt: updatedAt ?? this.updatedAt,
|
||||
category: category ?? this.category,
|
||||
printerType: printerType ?? this.printerType,
|
||||
variants: variants ?? this.variants,
|
||||
);
|
||||
}
|
||||
}
|
||||
@ -297,3 +364,51 @@ class Category {
|
||||
updatedAt.hashCode;
|
||||
}
|
||||
}
|
||||
|
||||
class ProductVariant {
|
||||
final String? id;
|
||||
final String? productId;
|
||||
final String? name;
|
||||
final int? priceModifier;
|
||||
final int? cost;
|
||||
final Map<String, dynamic>? metadata;
|
||||
final DateTime? createdAt;
|
||||
final DateTime? updatedAt;
|
||||
|
||||
ProductVariant({
|
||||
this.id,
|
||||
this.productId,
|
||||
this.name,
|
||||
this.priceModifier,
|
||||
this.cost,
|
||||
this.metadata,
|
||||
this.createdAt,
|
||||
this.updatedAt,
|
||||
});
|
||||
|
||||
factory ProductVariant.fromMap(Map<String, dynamic> json) => ProductVariant(
|
||||
id: json["id"],
|
||||
productId: json["product_id"],
|
||||
name: json["name"],
|
||||
priceModifier: json["price_modifier"],
|
||||
cost: json["cost"],
|
||||
metadata: json["metadata"] ?? {},
|
||||
createdAt: json["created_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["created_at"]),
|
||||
updatedAt: json["updated_at"] == null
|
||||
? null
|
||||
: DateTime.parse(json["updated_at"]),
|
||||
);
|
||||
|
||||
Map<String, dynamic> toMap() => {
|
||||
"id": id,
|
||||
"product_id": productId,
|
||||
"name": name,
|
||||
"price_modifier": priceModifier,
|
||||
"cost": cost,
|
||||
"metadata": metadata,
|
||||
"created_at": createdAt?.toIso8601String(),
|
||||
"updated_at": updatedAt?.toIso8601String(),
|
||||
};
|
||||
}
|
||||
|
||||