Compare commits
	
		
			18 Commits
		
	
	
		
			eb99ad9d7f
			...
			bayu/dev
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 4f2380fcdf | |||
| c23075304a | |||
| 17c7559158 | |||
| 711e5f3f52 | |||
| a99365fb0a | |||
| 41f0bb7a68 | |||
| c97130239d | |||
| 1b867227c7 | |||
| 4b07219928 | |||
| 9cdda42b8b | |||
| 6f5d3ccca8 | |||
| ff515e2621 | |||
| f057a346c2 | |||
| f84fe1017d | |||
| 27ba55314b | |||
| 59e046bd92 | |||
| 4461b78565 | |||
| f2f688f9f3 | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/icon-card.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.3 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/icon-chat.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.0 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/icon-education.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.8 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/icon-fund.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.3 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/icon-gadget-outline.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.7 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/icon-home.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.7 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/icon-logout.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.7 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/icon-profile.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.1 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/icon-setting.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.4 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/icons/icon-shop.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 3.7 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/images/bg-profile.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 4.0 MiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/images/img-cat-outlined.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 14 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/images/img-empty-transaction.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 70 KiB | 
							
								
								
									
										
											BIN
										
									
								
								assets/images/img-expand-purchase.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 32 KiB | 
| @@ -49,12 +49,28 @@ class PathAssets { | ||||
|   static const String iconTicket = 'assets/icons/icon-ticket.png'; | ||||
|   static const String iconGadget = 'assets/icons/icon-gadget.png'; | ||||
|   static const String iconCar = 'assets/icons/icon-car.png'; | ||||
|   static const String iconNavigationHome = 'assets/icons/icon-navigation-home.png'; | ||||
|   static const String iconNavigationPlan = 'assets/icons/icon-navigation-plan.png'; | ||||
|   static const String iconNavigationTransaction = 'assets/icons/icon-navigation-transaction.png'; | ||||
|   static const String iconNavigationPortfolio = 'assets/icons/icon-navigation-portfolio.png'; | ||||
|   static const String iconNavigationProfile = 'assets/icons/icon-navigation-profile.png'; | ||||
|   static const String iconNavigationHome = | ||||
|       'assets/icons/icon-navigation-home.png'; | ||||
|   static const String iconNavigationPlan = | ||||
|       'assets/icons/icon-navigation-plan.png'; | ||||
|   static const String iconNavigationTransaction = | ||||
|       'assets/icons/icon-navigation-transaction.png'; | ||||
|   static const String iconNavigationPortfolio = | ||||
|       'assets/icons/icon-navigation-portfolio.png'; | ||||
|   static const String iconNavigationProfile = | ||||
|       'assets/icons/icon-navigation-profile.png'; | ||||
|   static const String iconRemove = 'assets/icons/icon-remove.png'; | ||||
|   static const String iconEducation = 'assets/icons/icon-education.png'; | ||||
|   static const String iconFund = 'assets/icons/icon-fund.png'; | ||||
|   static const String iconHome = 'assets/icons/icon-home.png'; | ||||
|   static const String iconShop = 'assets/icons/icon-shop.png'; | ||||
|   static const String iconCard = 'assets/icons/icon-card.png'; | ||||
|   static const String iconChat = 'assets/icons/icon-chat.png'; | ||||
|   static const String iconLogout = 'assets/icons/icon-logout.png'; | ||||
|   static const String iconProfile = 'assets/icons/icon-profile.png'; | ||||
|   static const String iconSetting = 'assets/icons/icon-setting.png'; | ||||
|   static const String iconGadgetOutline = | ||||
|       'assets/icons/icon-gadget-outline.png'; | ||||
|  | ||||
|   /// IMAGE | ||||
|   static const String imgSplashLogo = 'assets/images/splash-logo.png'; | ||||
| @@ -83,15 +99,22 @@ class PathAssets { | ||||
|   static const String imgMoneyIncome = 'assets/images/img-money-income.png'; | ||||
|   static const String imgGrowing = 'assets/images/img-growing.png'; | ||||
|   static const String imgCat = 'assets/images/img-cat.png'; | ||||
|   static const String imgCatOutlined = 'assets/images/img-cat-outlined.png'; | ||||
|   static const String imgDeer = 'assets/images/img-deer.png'; | ||||
|   static const String imgLion = 'assets/images/img-lion.png'; | ||||
|   static const String imgGuideBank = 'assets/images/img-guide-bank.png'; | ||||
|   static const String imgGuide1 = 'assets/images/img-guide1.png'; | ||||
|   static const String imgGuide2 = 'assets/images/img-guide2.png'; | ||||
|   static const String imgOpenShopping = 'assets/images/img-open-shopping.png'; | ||||
|   static const String imgPaymentSuccess = 'assets/images/img-payment-success.png'; | ||||
|   static const String imgPaymentSuccess = | ||||
|       'assets/images/img-payment-success.png'; | ||||
|   static const String frameSignature = 'assets/images/frame-signature.png'; | ||||
|   static const String imgFinish = 'assets/images/img-finish.png'; | ||||
|   static const String imgExpandPurchase = | ||||
|       'assets/images/img-expand-purchase.png'; | ||||
|   static const String imgEmptyTransaction = | ||||
|       'assets/images/img-empty-transaction.png'; | ||||
|   static const String bgProfile = 'assets/images/bg-profile.png'; | ||||
|  | ||||
|   static const Map<String, String> goalInvestIcon = { | ||||
|     'Education': iconToga, | ||||
|   | ||||
| @@ -0,0 +1,100 @@ | ||||
| import 'package:cims_apps/application/component/image/image_view.dart'; | ||||
| import 'package:cims_apps/application/theme/color_palette.dart'; | ||||
| import 'package:cims_apps/core/utils/size_config.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
|  | ||||
| class CardTransactionView extends StatelessWidget { | ||||
|   final VoidCallback onTap; | ||||
|   final String iconPath, type, amount, subs, step; | ||||
|   final String? timeTransaction; | ||||
|   const CardTransactionView({ | ||||
|     Key? key, | ||||
|     required this.step, | ||||
|     required this.type, | ||||
|     required this.amount, | ||||
|     required this.iconPath, | ||||
|     required this.subs, | ||||
|     required this.onTap, | ||||
|     this.timeTransaction, | ||||
|   }) : super(key: key); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     TextTheme textTheme = Theme.of(context).textTheme; | ||||
|     return GestureDetector( | ||||
|       onTap: onTap, | ||||
|       child: Container( | ||||
|         margin: const EdgeInsets.symmetric(vertical: 16.0), | ||||
|         padding: const EdgeInsets.symmetric(vertical: 16.0, horizontal: 16.0), | ||||
|         decoration: BoxDecoration( | ||||
|           color: Colors.white, | ||||
|           border: Border.all(width: 1, color: ColorPalette.slate200), | ||||
|           borderRadius: const BorderRadius.all(Radius.circular(12)), | ||||
|         ), | ||||
|         child: Row( | ||||
|           mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|           crossAxisAlignment: CrossAxisAlignment.center, | ||||
|           children: [ | ||||
|             SizedBox( | ||||
|                 width: SizeConfig.width * .4, | ||||
|                 child: Column( | ||||
|                   crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                   children: [ | ||||
|                     Row( | ||||
|                       children: [ | ||||
|                         ImageView( | ||||
|                             image: iconPath, width: SizeConfig.width * .12), | ||||
|                         Padding( | ||||
|                           padding: const EdgeInsets.only(left: 8.0), | ||||
|                           child: Text( | ||||
|                             type, | ||||
|                             style: textTheme.headlineSmall, | ||||
|                           ), | ||||
|                         ), | ||||
|                       ], | ||||
|                     ), | ||||
|                     Padding( | ||||
|                       padding: const EdgeInsets.only(top: 16.0), | ||||
|                       child: Text( | ||||
|                         amount, | ||||
|                         style: textTheme.headlineSmall, | ||||
|                       ), | ||||
|                     ), | ||||
|                   ], | ||||
|                 )), | ||||
|             SizedBox( | ||||
|                 width: SizeConfig.width * .4, | ||||
|                 child: Column( | ||||
|                   crossAxisAlignment: CrossAxisAlignment.end, | ||||
|                   mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                   children: [ | ||||
|                     SizedBox( | ||||
|                         height: SizeConfig.height * .08, | ||||
|                         child: Text( | ||||
|                           subs, | ||||
|                           style: const TextStyle(color: ColorPalette.primary), | ||||
|                         )), | ||||
|                     Row( | ||||
|                       mainAxisAlignment: MainAxisAlignment.end, | ||||
|                       children: [ | ||||
|                         step == 'waiting' | ||||
|                             ? const Icon(Icons.access_time_sharp, | ||||
|                                 color: ColorPalette.slate400) | ||||
|                             : const SizedBox(), | ||||
|                         step == 'waiting' | ||||
|                             ? Text(timeTransaction.toString()) | ||||
|                             : const SizedBox(), | ||||
|                         const Padding( | ||||
|                           padding: EdgeInsets.only(left: 16.0), | ||||
|                           child: Icon(Icons.arrow_forward_ios), | ||||
|                         ), | ||||
|                       ], | ||||
|                     ), | ||||
|                   ], | ||||
|                 )), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| @@ -0,0 +1,44 @@ | ||||
| import 'package:cims_apps/application/assets/path_assets.dart'; | ||||
| import 'package:cims_apps/application/component/button/button_view.dart'; | ||||
| import 'package:cims_apps/application/component/image/image_view.dart'; | ||||
| import 'package:cims_apps/core/utils/size_config.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
|  | ||||
| class EmptyCardTransaction extends StatelessWidget { | ||||
|   final VoidCallback onPressedButton; | ||||
|   const EmptyCardTransaction({Key? key, required this.onPressedButton}) | ||||
|       : super(key: key); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     TextTheme textTheme = Theme.of(context).textTheme; | ||||
|     return Container( | ||||
|       padding: const EdgeInsets.symmetric(vertical: 8.0), | ||||
|       child: Center( | ||||
|         child: Column( | ||||
|           mainAxisAlignment: MainAxisAlignment.spaceAround, | ||||
|           children: [ | ||||
|             ImageView( | ||||
|               image: PathAssets.imgEmptyTransaction, | ||||
|               width: SizeConfig.width * .4, | ||||
|             ), | ||||
|             Text( | ||||
|               'No Transaction Yet', | ||||
|               style: textTheme.headlineSmall, | ||||
|             ), | ||||
|             Text( | ||||
|               "Let's keep building your investment for even greater financial growth!", | ||||
|               style: textTheme.bodyMedium, | ||||
|               textAlign: TextAlign.center, | ||||
|             ), | ||||
|             ButtonView( | ||||
|               name: 'Investing Now', | ||||
|               width: SizeConfig.width * .5, | ||||
|               onPressed: onPressedButton, | ||||
|             ), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| @@ -8,22 +8,26 @@ class ListTileView extends StatelessWidget { | ||||
|   final String title; | ||||
|   final VoidCallback? onPressed; | ||||
|   final Widget? prefixIcon, suffixIcon; | ||||
|   final Color? colorTitle; | ||||
|   final EdgeInsetsGeometry? padding, margin; | ||||
|   final TextStyle? textStyle; | ||||
|   const ListTileView( | ||||
|       {Key? key, | ||||
|       required this.title, | ||||
|       this.onPressed, | ||||
|       this.prefixIcon, | ||||
|       this.suffixIcon, | ||||
|       this.colorTitle}) | ||||
|       this.padding, | ||||
|       this.textStyle, | ||||
|       this.margin}) | ||||
|       : super(key: key); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Container( | ||||
|       width: SizeConfig.width, | ||||
|       padding: const EdgeInsets.symmetric(vertical: 16.0, horizontal: 8.0), | ||||
|       margin: const EdgeInsets.symmetric(vertical: 16.0), | ||||
|       padding: padding ?? | ||||
|           const EdgeInsets.symmetric(vertical: 16.0, horizontal: 8.0), | ||||
|       margin: margin ?? const EdgeInsets.symmetric(vertical: 16.0), | ||||
|       decoration: BoxDecoration( | ||||
|         color: ColorPalette.blue50, | ||||
|         borderRadius: BorderRadius.circular(10), | ||||
| @@ -47,22 +51,23 @@ class ListTileView extends StatelessWidget { | ||||
|           Expanded( | ||||
|             child: Text( | ||||
|               title, | ||||
|               style: TextStyle( | ||||
|               style: textStyle ?? | ||||
|                   const TextStyle( | ||||
|                     fontWeight: FontWeight.w600, | ||||
|                 color: colorTitle ?? ColorPalette.slate500, | ||||
|                     color: ColorPalette.slate500, | ||||
|                   ), | ||||
|             ), | ||||
|           ), | ||||
|           suffixIcon != null | ||||
|               ? IconButton( | ||||
|           suffixIcon ?? | ||||
|               IconButton( | ||||
|                 onPressed: onPressed, | ||||
|                 icon: const Icon( | ||||
|                   Icons.arrow_forward_ios, | ||||
|                   color: ColorPalette.primary, | ||||
|                   size: 20, | ||||
|                 ), | ||||
|                 ) | ||||
|               : const SizedBox(), | ||||
|               ), | ||||
|           // : const SizedBox(), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   | ||||
| @@ -2,6 +2,7 @@ import 'dart:io'; | ||||
|  | ||||
| import 'package:cims_apps/application/assets/path_assets.dart'; | ||||
| import 'package:cims_apps/application/component/button/button_view.dart'; | ||||
| import 'package:cims_apps/application/component/custom_app_bar/custom_app_bar.dart'; | ||||
| import 'package:cims_apps/application/component/image/image_view.dart'; | ||||
| import 'package:cims_apps/application/component/take_picture_screen/take_picture_screen.dart'; | ||||
| import 'package:cims_apps/application/theme/color_palette.dart'; | ||||
| @@ -94,10 +95,8 @@ class _DisplayPictureScreenState extends State<DisplayPictureScreen> { | ||||
|           return Consumer<SubmissionDataViewModel>( | ||||
|               builder: (context, provider, child) { | ||||
|             return Scaffold( | ||||
|               appBar: AppBar( | ||||
|                 title: const Text('Preview'), | ||||
|                 automaticallyImplyLeading: false, | ||||
|               ), | ||||
|               appBar: CustomAppBar( | ||||
|                   height: SizeConfig.height * .08, title: 'Preview'), | ||||
|               body: Container( | ||||
|                   padding: const EdgeInsets.symmetric(horizontal: 24.0), | ||||
|                   child: Column( | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import 'package:camera/camera.dart'; | ||||
| import 'package:cims_apps/application/assets/path_assets.dart'; | ||||
| import 'package:cims_apps/application/component/custom_app_bar/custom_app_bar.dart'; | ||||
| import 'package:cims_apps/application/component/image/image_view.dart'; | ||||
| import 'package:cims_apps/application/component/take_picture_screen/display_picture_screen.dart'; | ||||
| import 'package:cims_apps/core/route/route.dart'; | ||||
| @@ -27,7 +28,6 @@ class TakePictureScreenState extends State<TakePictureScreen> { | ||||
|   late String _takeContent; | ||||
|  | ||||
|   Future<void> changeFlash() async { | ||||
|     await _controller.setFlashMode(FlashMode.auto); | ||||
|     setState(() { | ||||
|       isFlash = !isFlash; | ||||
|     }); | ||||
| @@ -44,6 +44,7 @@ class TakePictureScreenState extends State<TakePictureScreen> { | ||||
|     // Next, initialize the controller. This returns a Future. | ||||
|     _initializeControllerFuture = _controller.initialize(); | ||||
|     _takeContent = widget.takeContent; | ||||
|     _controller.setFlashMode(isFlash ? FlashMode.torch : FlashMode.off); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
| @@ -57,9 +58,10 @@ class TakePictureScreenState extends State<TakePictureScreen> { | ||||
|   Widget build(BuildContext context) { | ||||
|     // Fill this out in the next steps. | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: const Text('Registration'), | ||||
|         actions: [ | ||||
|       appBar: CustomAppBar( | ||||
|         title: 'Registration', | ||||
|         height: SizeConfig.height * .08, | ||||
|         trailing: [ | ||||
|           IconButton( | ||||
|               onPressed: () { | ||||
|                 changeFlash(); | ||||
|   | ||||
| @@ -1,3 +1,8 @@ | ||||
| import 'package:cims_apps/application/theme/color_palette.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter/services.dart'; | ||||
| import 'package:intl/intl.dart'; | ||||
|  | ||||
| class StringUtils { | ||||
|   static bool emailValidation(String email) { | ||||
|     return RegExp( | ||||
| @@ -13,4 +18,24 @@ class StringUtils { | ||||
|     return RegExp(r'^(?=.*[a-z])(?=.*[A-Z])(?=.*[0-9])(?=.*?[\W_])(?=.{8,})') | ||||
|         .hasMatch(password); | ||||
|   } | ||||
|  | ||||
|   static String formatTime(DateTime? dateTime) { | ||||
|     if (dateTime != null) { | ||||
|       DateFormat formatter = DateFormat('HH:mm:ss'); | ||||
|       return formatter.format(dateTime); | ||||
|     } | ||||
|     return '--:--:--'; | ||||
|   } | ||||
|  | ||||
|   static void iCopyToClipboard(BuildContext context, | ||||
|       {String? desc, required String text}) { | ||||
|     Clipboard.setData(ClipboardData(text: text)); | ||||
|     ScaffoldMessenger.of(context) | ||||
|         .showSnackBar(SnackBar( | ||||
|           backgroundColor: ColorPalette.primary, | ||||
|           content: Text(desc ?? "Text copied to clipboard"), | ||||
|         )) | ||||
|         .closed | ||||
|         .then((value) => ScaffoldMessenger.of(context).removeCurrentSnackBar()); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -6,19 +6,26 @@ import 'package:cims_apps/application/theme/color_palette.dart'; | ||||
| import 'package:cims_apps/core/route/route.dart'; | ||||
| import 'package:cims_apps/core/utils/size_config.dart'; | ||||
| import 'package:cims_apps/features/auth/registration/view/submission_data/submission_parent.dart'; | ||||
| import 'package:cims_apps/features/auth/registration/viewmodel/submission_data_viewmodel.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
|  | ||||
| class ModelDataBank { | ||||
|   final String? title, subtitle; | ||||
|   ModelDataBank(this.title, this.subtitle); | ||||
| } | ||||
|  | ||||
| class ConfirmBankAccount extends StatelessWidget { | ||||
|   const ConfirmBankAccount({Key? key}) : super(key: key); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     List listData = [ | ||||
|       {'title': 'Bank Name', 'subtitle': 'Bank Mandiri'}, | ||||
|       {'title': 'Account Number', 'subtitle': '123002212084'}, | ||||
|       {'title': 'Account Owner Name', 'subtitle': 'Muhamad Rosyidin'}, | ||||
|       {'title': 'Name on ID card', 'subtitle': 'Muhamad Rosyidin'}, | ||||
|     ]; | ||||
|     final listDataBank = | ||||
|         ModalRoute.of(context)!.settings.arguments as List<ModelDataBank>; | ||||
|  | ||||
|     return ChangeNotifierProvider( | ||||
|         create: (context) => SubmissionDataViewModel(), | ||||
|         builder: (context, child) { | ||||
|           return Scaffold( | ||||
|             appBar: AppBar( | ||||
|               toolbarHeight: 70, | ||||
| @@ -42,7 +49,9 @@ class ConfirmBankAccount extends StatelessWidget { | ||||
|               padding: const EdgeInsets.all(16.0), | ||||
|               child: SizedBox( | ||||
|                 height: SizeConfig.height * .85, | ||||
|           child: Column( | ||||
|                 child: Consumer<SubmissionDataViewModel>( | ||||
|                     builder: (context, provider, child) { | ||||
|                   return Column( | ||||
|                     mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                     crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                     children: [ | ||||
| @@ -53,7 +62,7 @@ class ConfirmBankAccount extends StatelessWidget { | ||||
|                           mainAxisAlignment: MainAxisAlignment.start, | ||||
|                           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                           children: [ | ||||
|                     ...listData.map((e) { | ||||
|                             ...listDataBank.map((e) { | ||||
|                               return Padding( | ||||
|                                 padding: const EdgeInsets.only(bottom: 16.0), | ||||
|                                 child: Column( | ||||
| @@ -61,12 +70,13 @@ class ConfirmBankAccount extends StatelessWidget { | ||||
|                                   crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                                   children: [ | ||||
|                                     Text( | ||||
|                               e['title'], | ||||
|                                       e.title!, | ||||
|                                       style: const TextStyle( | ||||
|                                   color: ColorPalette.slate400, fontSize: 16), | ||||
|                                           color: ColorPalette.slate400, | ||||
|                                           fontSize: 16), | ||||
|                                     ), | ||||
|                                     Text( | ||||
|                               e['subtitle'], | ||||
|                                       e.subtitle!, | ||||
|                                       style: const TextStyle( | ||||
|                                           fontSize: 16, | ||||
|                                           color: ColorPalette.slate800, | ||||
| @@ -98,15 +108,18 @@ class ConfirmBankAccount extends StatelessWidget { | ||||
|                             name: 'Confirm', | ||||
|                             width: SizeConfig.width * .42, | ||||
|                             onPressed: () { | ||||
|                       routePush(context, page: const SubmissionParent()); | ||||
|                               routePush(context, | ||||
|                                   page: const SubmissionParent()); | ||||
|                             }, | ||||
|                           ), | ||||
|                         ], | ||||
|                       ) | ||||
|                     ], | ||||
|           ), | ||||
|                   ); | ||||
|                 }), | ||||
|               ), | ||||
|             ), | ||||
|           ); | ||||
|         }); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -63,9 +63,13 @@ class SubmitBankAccount extends StatelessWidget { | ||||
|           return SizedBox( | ||||
|             child: Consumer<SubmissionDataViewModel>( | ||||
|                 builder: (context, provider, child) { | ||||
|               return SingleChildScrollView( | ||||
|               return SizedBox( | ||||
|                 height: SizeConfig.height * .8, | ||||
|                 child: Form( | ||||
|                   key: provider.formKeySubmitDataBank, | ||||
|                   child: Column( | ||||
|                     crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                     mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                     children: [ | ||||
|                       const TextCaption(title: 'Input your bank account data'), | ||||
|                       TextFormView( | ||||
| @@ -79,11 +83,24 @@ class SubmitBankAccount extends StatelessWidget { | ||||
|                         suffixIcon: const Icon( | ||||
|                           Icons.keyboard_arrow_down_outlined, | ||||
|                         ), | ||||
|                         validator: (value) { | ||||
|                           if (value!.isEmpty) { | ||||
|                             return 'Field must be filled'; | ||||
|                           } | ||||
|                           return null; | ||||
|                         }, | ||||
|                       ), | ||||
|                       TextFormView( | ||||
|                         name: 'Account Number', | ||||
|                         hintText: 'Input Account Number', | ||||
|                         ctrl: provider.ctrlNoAccountBank, | ||||
|                         keyboardType: TextInputType.number, | ||||
|                         validator: (value) { | ||||
|                           if (value!.isEmpty) { | ||||
|                             return 'Field must be filled'; | ||||
|                           } | ||||
|                           return null; | ||||
|                         }, | ||||
|                         trailingTitleWidget: SizedBox( | ||||
|                           width: 24, | ||||
|                           child: GestureDetector( | ||||
| @@ -98,6 +115,13 @@ class SubmitBankAccount extends StatelessWidget { | ||||
|                       TextFormView( | ||||
|                         name: 'Account Owner Name', | ||||
|                         hintText: 'Input Account Name', | ||||
|                         ctrl: provider.ctrlNameAccountBank, | ||||
|                         validator: (value) { | ||||
|                           if (value!.isEmpty) { | ||||
|                             return 'Field must be filled'; | ||||
|                           } | ||||
|                           return null; | ||||
|                         }, | ||||
|                       ), | ||||
|                       const Text( | ||||
|                         "Make sure the account you use is in your name, not someone else's", | ||||
| @@ -109,16 +133,29 @@ class SubmitBankAccount extends StatelessWidget { | ||||
|                       ButtonView( | ||||
|                         name: 'Next', | ||||
|                         onPressed: () { | ||||
|                           if (provider.formKeySubmitDataBank.currentState! | ||||
|                               .validate()) { | ||||
|                             provider | ||||
|                                 .submitDataBank( | ||||
|                               bankName: provider.ctrlBankName.text, | ||||
|                               accountNumber: provider.ctrlNoAccountBank.text, | ||||
|                               accountName: provider.ctrlNameAccountBank.text, | ||||
|                             ) | ||||
|                                 .then((values) { | ||||
|                               provider.next(context).then((value) { | ||||
|                                 if (value) { | ||||
|                                   routePush(context, | ||||
|                                 page: const ConfirmBankAccount()); | ||||
|                                       page: const ConfirmBankAccount(), | ||||
|                                       arguments: values); | ||||
|                                 } | ||||
|                               }); | ||||
|                             }); | ||||
|                           } | ||||
|                         }, | ||||
|                       ) | ||||
|                     ], | ||||
|                   ), | ||||
|                 ), | ||||
|               ); | ||||
|             }), | ||||
|           ); | ||||
|   | ||||
| @@ -14,12 +14,12 @@ class RiskProfileView extends StatelessWidget { | ||||
|   Widget build(BuildContext context) { | ||||
|     return Container( | ||||
|       width: SizeConfig.width, | ||||
|       height: SizeConfig.height, | ||||
|       padding: EdgeInsets.all(24), | ||||
|       height: SizeConfig.height * .8, | ||||
|       padding: const EdgeInsets.all(24), | ||||
|       child: Column( | ||||
|         mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|         children: [ | ||||
|           Column( | ||||
|           const Column( | ||||
|             children: [ | ||||
|               ImageView(image: PathAssets.imgDataReport), | ||||
|               SizedBox( | ||||
| @@ -48,7 +48,7 @@ class RiskProfileView extends StatelessWidget { | ||||
|           ), | ||||
|           Column( | ||||
|             children: [ | ||||
|               Row( | ||||
|               const Row( | ||||
|                 mainAxisAlignment: MainAxisAlignment.center, | ||||
|                 children: [ | ||||
|                   ImageView( | ||||
| @@ -68,7 +68,7 @@ class RiskProfileView extends StatelessWidget { | ||||
|                   ) | ||||
|                 ], | ||||
|               ), | ||||
|               SizedBox( | ||||
|               const SizedBox( | ||||
|                 height: 24, | ||||
|               ), | ||||
|               ButtonView( | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import 'package:cims_apps/application/component/button/back_button_view.dart'; | ||||
| import 'package:cims_apps/application/component/custom_app_bar/custom_app_bar.dart'; | ||||
| import 'package:cims_apps/application/theme/color_palette.dart'; | ||||
| import 'package:cims_apps/core/route/route.dart'; | ||||
| @@ -66,21 +67,33 @@ class _SubmissionParentState extends State<SubmissionParent> { | ||||
|     return ChangeNotifierProvider( | ||||
|         create: (context) => SubmissionDataViewModel(), | ||||
|         builder: (context, child) { | ||||
|           return WillPopScope( | ||||
|             onWillPop: () async { | ||||
|           return PopScope( | ||||
|             canPop: false, | ||||
|             onPopInvoked: (didPop) async { | ||||
|               if (didPop) { | ||||
|                 return; | ||||
|               } | ||||
|               await routePush(context, | ||||
|                   page: const BottomNavigationView(), | ||||
|                   routeType: RouteType.pushReplace); | ||||
|               return false; | ||||
|             }, | ||||
|             child: Consumer<SubmissionDataViewModel>( | ||||
|                 builder: (context, provider, child) { | ||||
|               return Scaffold( | ||||
|                 appBar: CustomAppBar( | ||||
|                     height: SizeConfig.height * .1, title: 'Registration'), | ||||
|                 body: Stack( | ||||
|                   children: [ | ||||
|                     Column( | ||||
|                   height: SizeConfig.height * .1, | ||||
|                   title: 'Registration', | ||||
|                   leading: BackButtonView( | ||||
|                     onPress: () { | ||||
|                       routePush( | ||||
|                         context, | ||||
|                         page: const BottomNavigationView(), | ||||
|                         routeType: RouteType.pushReplace, | ||||
|                       ); | ||||
|                     }, | ||||
|                   ), | ||||
|                 ), | ||||
|                 body: Column( | ||||
|                   crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                   mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                   children: [ | ||||
| @@ -89,8 +102,7 @@ class _SubmissionParentState extends State<SubmissionParent> { | ||||
|                           horizontal: 16.0, vertical: 16.0), | ||||
|                       child: Row( | ||||
|                         mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                             children: | ||||
|                                 List.generate(provider.stepAmount, (index) { | ||||
|                         children: List.generate(provider.stepAmount, (index) { | ||||
|                           return _stepItem( | ||||
|                             isCurrentStep: | ||||
|                                 provider.getCurrentStep == index + 1 || | ||||
| @@ -100,16 +112,13 @@ class _SubmissionParentState extends State<SubmissionParent> { | ||||
|                       ), | ||||
|                     ), | ||||
|                     Expanded( | ||||
|                           child: Container( | ||||
|                             padding: | ||||
|                                 const EdgeInsets.symmetric(horizontal: 16.0), | ||||
|                       child: SingleChildScrollView( | ||||
|                         padding: const EdgeInsets.symmetric(horizontal: 16.0), | ||||
|                         child: _content(provider.getCurrentStep), | ||||
|                       ), | ||||
|                     ), | ||||
|                   ], | ||||
|                 ), | ||||
|                   ], | ||||
|                 ), | ||||
|               ); | ||||
|             }), | ||||
|           ); | ||||
|   | ||||
| @@ -176,7 +176,9 @@ class SubmitDataIdCard extends StatelessWidget { | ||||
|           return SingleChildScrollView( | ||||
|             child: Consumer<SubmissionDataViewModel>( | ||||
|                 builder: (context, provider, child) { | ||||
|               return Column( | ||||
|               return Form( | ||||
|                 key: provider.formKeySubmitIdCard, | ||||
|                 child: Column( | ||||
|                   crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                   children: [ | ||||
|                     const TextCaption( | ||||
| @@ -184,14 +186,35 @@ class SubmitDataIdCard extends StatelessWidget { | ||||
|                     TextFormView( | ||||
|                       name: 'NIK', | ||||
|                       keyboardType: TextInputType.number, | ||||
|                       validator: (value) { | ||||
|                         if (value!.isEmpty) { | ||||
|                           return 'Field must be filled'; | ||||
|                         } | ||||
|                         return null; | ||||
|                       }, | ||||
|                     ), | ||||
|                     TextFormView( | ||||
|                       name: 'Full Name', | ||||
|                       validator: (value) { | ||||
|                         if (value!.isEmpty) { | ||||
|                           return 'Field must be filled'; | ||||
|                         } | ||||
|                         return null; | ||||
|                       }, | ||||
|                     ), | ||||
|                   TextFormView(name: 'Full Name'), | ||||
|                     DatePickerView( | ||||
|                       name: 'Birth Date', | ||||
|                       ctrl: provider.ctrlBirthDate, | ||||
|                       maxDate: DateTime.now(), | ||||
|                       isMultipleSelection: false, | ||||
|                       enabled: true), | ||||
|                       enabled: true, | ||||
|                       validatorDate: (value) { | ||||
|                         if (value!.isEmpty) { | ||||
|                           return 'Field must be filled'; | ||||
|                         } | ||||
|                         return null; | ||||
|                       }, | ||||
|                     ), | ||||
|                     photoDocument(provider), | ||||
|                     Container( | ||||
|                       width: SizeConfig.width, | ||||
| @@ -241,14 +264,19 @@ class SubmitDataIdCard extends StatelessWidget { | ||||
|                     ButtonView( | ||||
|                       name: 'Next', | ||||
|                       onPressed: () async { | ||||
|                         if (provider.formKeySubmitIdCard.currentState! | ||||
|                             .validate()) { | ||||
|                           await provider.next(context).then((value) { | ||||
|                             if (value) { | ||||
|                           routePush(context, page: const SubmissionParent()); | ||||
|                               routePush(context, | ||||
|                                   page: const SubmissionParent()); | ||||
|                             } | ||||
|                           }); | ||||
|                         } | ||||
|                       }, | ||||
|                     ) | ||||
|                   ], | ||||
|                 ), | ||||
|               ); | ||||
|             }), | ||||
|           ); | ||||
|   | ||||
| @@ -15,41 +15,70 @@ import 'package:provider/provider.dart'; | ||||
| class SubmitEmail extends StatelessWidget { | ||||
|   const SubmitEmail({Key? key}) : super(key: key); | ||||
|  | ||||
|   Widget _emailVerify() { | ||||
|     return Column( | ||||
|   Widget _emailVerify(BuildContext context, SubmissionDataViewModel provider) { | ||||
|     var textTheme = Theme.of(context).textTheme; | ||||
|     return Padding( | ||||
|       padding: const EdgeInsets.all(24.0), | ||||
|       child: Column( | ||||
|         crossAxisAlignment: CrossAxisAlignment.start, | ||||
|         children: [ | ||||
|           const TextCaption(title: 'Check your e-mail'), | ||||
|           const ImageView(image: PathAssets.imgEmail), | ||||
|           Align( | ||||
|             alignment: Alignment.center, | ||||
|             child: RichText( | ||||
|               textAlign: TextAlign.center, | ||||
|               text: TextSpan(children: [ | ||||
|               const TextSpan( | ||||
|                 TextSpan( | ||||
|                   text: | ||||
|                       'We have sent a verification link to your e-mail. \nPlease check your email for ', | ||||
|                 style: TextStyle( | ||||
|                   color: Colors.black, | ||||
|                   decoration: TextDecoration.none, | ||||
|                 ), | ||||
|                   style: textTheme.displayMedium, | ||||
|                 ), | ||||
|                 TextSpan( | ||||
|                 recognizer: TapGestureRecognizer()..onTap = () {}, | ||||
|                   recognizer: TapGestureRecognizer() | ||||
|                     ..onTap = () async { | ||||
|                       await provider.next(context).then((value) { | ||||
|                         if (value) { | ||||
|                           routePush(context, page: const SubmissionParent()); | ||||
|                         } | ||||
|                       }); | ||||
|                     }, | ||||
|                   text: 'verification', | ||||
|                   style: const TextStyle( | ||||
|                     color: Colors.blue, | ||||
|                   ), | ||||
|                 ), | ||||
|               const TextSpan( | ||||
|                 TextSpan( | ||||
|                   text: ' to \ncontinue registration.', | ||||
|                 style: TextStyle( | ||||
|                   color: Colors.black, | ||||
|                   decoration: TextDecoration.none, | ||||
|                 ), | ||||
|                   style: textTheme.displayMedium, | ||||
|                 ), | ||||
|               ]), | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   showEmailVerify(BuildContext context, SubmissionDataViewModel provider) { | ||||
|     showModalBottomSheet( | ||||
|       context: context, | ||||
|       isScrollControlled: true, | ||||
|       enableDrag: false, | ||||
|       shape: const RoundedRectangleBorder( | ||||
|         borderRadius: BorderRadius.vertical( | ||||
|           top: Radius.zero, | ||||
|         ), | ||||
|       ), | ||||
|       builder: (context) { | ||||
|         var flutterView = View.of(context); | ||||
|         return Padding( | ||||
|           padding: EdgeInsets.only( | ||||
|             top: MediaQueryData.fromView(flutterView).padding.top, | ||||
|           ), | ||||
|           child: _emailVerify(context, provider), | ||||
|         ); | ||||
|       }, | ||||
|     ); | ||||
|   } | ||||
|  | ||||
| @@ -60,16 +89,16 @@ class SubmitEmail extends StatelessWidget { | ||||
|         builder: (context, child) { | ||||
|           return Consumer<SubmissionDataViewModel>( | ||||
|               builder: (context, provider, child) { | ||||
|             return SingleChildScrollView( | ||||
|             return SizedBox( | ||||
|               height: SizeConfig.height * .78, | ||||
|               child: Form( | ||||
|                 key: provider.formKeySubmitEmail, | ||||
|                 child: Column( | ||||
|                   crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                 // mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                   mainAxisAlignment: MainAxisAlignment.spaceAround, | ||||
|                   children: [ | ||||
|                   !provider.isEmailVerify | ||||
|                       ? const TextCaption(title: 'Enter your e-mail') | ||||
|                       : const TextCaption(title: 'Check your e-mail '), | ||||
|                   !provider.isEmailVerify | ||||
|                       ? TextFormView( | ||||
|                     const TextCaption(title: 'Enter your e-mail'), | ||||
|                     TextFormView( | ||||
|                       name: 'E-mail Address', | ||||
|                       hintText: 'Input e-mail address', | ||||
|                       keyboardType: TextInputType.emailAddress, | ||||
| @@ -82,24 +111,20 @@ class SubmitEmail extends StatelessWidget { | ||||
|                           return null; | ||||
|                         } | ||||
|                       }, | ||||
|                           // onTap: () { | ||||
|                           //   provider.submitEmail(); | ||||
|                           // }, | ||||
|                         ) | ||||
|                       : _emailVerify(), | ||||
|                   SizedBox(height: SizeConfig.height * .42), | ||||
|                     ), | ||||
|                     SizedBox(height: SizeConfig.height * .43), | ||||
|                     ButtonView( | ||||
|                       name: 'Next', | ||||
|                       onPressed: () async { | ||||
|                       await provider.next(context).then((value) { | ||||
|                         if (value) { | ||||
|                           routePush(context, page: const SubmissionParent()); | ||||
|                         if (provider.formKeySubmitEmail.currentState! | ||||
|                             .validate()) { | ||||
|                           showEmailVerify(context, provider); | ||||
|                         } | ||||
|                       }); | ||||
|                       }, | ||||
|                     ) | ||||
|                   ], | ||||
|                 ), | ||||
|               ), | ||||
|             ); | ||||
|           }); | ||||
|         }); | ||||
|   | ||||
| @@ -2,6 +2,7 @@ import 'package:cims_apps/application/component/button/button_view.dart'; | ||||
| import 'package:cims_apps/application/component/select_form/select_form_view.dart'; | ||||
| import 'package:cims_apps/application/component/text_caption/text_caption.dart'; | ||||
| import 'package:cims_apps/core/route/route.dart'; | ||||
| import 'package:cims_apps/core/utils/size_config.dart'; | ||||
| import 'package:cims_apps/features/auth/registration/view/submission_data/submission_parent.dart'; | ||||
| import 'package:cims_apps/features/auth/registration/viewmodel/submission_data_viewmodel.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| @@ -17,12 +18,13 @@ class SubmitPersonalData extends StatelessWidget { | ||||
|         builder: (context, child) { | ||||
|           return Consumer<SubmissionDataViewModel>( | ||||
|               builder: (context, provider, child) { | ||||
|             return SingleChildScrollView( | ||||
|             return SizedBox( | ||||
|               height: SizeConfig.height * .78, | ||||
|               child: Form( | ||||
|                 key: provider.formKeyPersonalData, | ||||
|                 child: Column( | ||||
|                   crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                   mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                   mainAxisAlignment: MainAxisAlignment.spaceAround, | ||||
|                   children: [ | ||||
|                     const TextCaption(title: 'Your personal details'), | ||||
|                     SelectFormView( | ||||
| @@ -31,6 +33,12 @@ class SubmitPersonalData extends StatelessWidget { | ||||
|                       ctrl: provider.ctrlOccupation, | ||||
|                       listItem: provider.listOccupation, | ||||
|                       onSelect: (value) {}, | ||||
|                       validator: (value) { | ||||
|                         if (value!.isEmpty) { | ||||
|                           return 'Field must be filled'; | ||||
|                         } | ||||
|                         return null; | ||||
|                       }, | ||||
|                     ), | ||||
|                     SelectFormView( | ||||
|                       name: 'Income Level (IDR)', | ||||
| @@ -38,6 +46,12 @@ class SubmitPersonalData extends StatelessWidget { | ||||
|                       ctrl: provider.ctrlIncome, | ||||
|                       listItem: provider.listIncome, | ||||
|                       onSelect: (value) {}, | ||||
|                       validator: (value) { | ||||
|                         if (value!.isEmpty) { | ||||
|                           return 'Field must be filled'; | ||||
|                         } | ||||
|                         return null; | ||||
|                       }, | ||||
|                     ), | ||||
|                     SelectFormView( | ||||
|                       name: 'Marital Status', | ||||
| @@ -45,6 +59,12 @@ class SubmitPersonalData extends StatelessWidget { | ||||
|                       ctrl: provider.ctrlMarital, | ||||
|                       listItem: provider.listMarital, | ||||
|                       onSelect: (value) {}, | ||||
|                       validator: (value) { | ||||
|                         if (value!.isEmpty) { | ||||
|                           return 'Field must be filled'; | ||||
|                         } | ||||
|                         return null; | ||||
|                       }, | ||||
|                     ), | ||||
|                     SelectFormView( | ||||
|                       name: 'Source of Fund', | ||||
| @@ -52,20 +72,28 @@ class SubmitPersonalData extends StatelessWidget { | ||||
|                       ctrl: provider.ctrlSourceFund, | ||||
|                       listItem: provider.listSourceFund, | ||||
|                       onSelect: (value) {}, | ||||
|                       validator: (value) { | ||||
|                         if (value!.isEmpty) { | ||||
|                           return 'Field must be filled'; | ||||
|                         } | ||||
|                         return null; | ||||
|                       }, | ||||
|                     ), | ||||
|                     Align( | ||||
|                       alignment: Alignment.bottomCenter, | ||||
|                       child: ButtonView( | ||||
|                     // SizedBox(height: SizeConfig.height * .18), | ||||
|                     ButtonView( | ||||
|                       marginVertical: 0, | ||||
|                       name: 'Next', | ||||
|                       onPressed: () async { | ||||
|                         if (provider.formKeyPersonalData.currentState! | ||||
|                             .validate()) { | ||||
|                           await provider.next(context).then((value) { | ||||
|                             if (value) { | ||||
|                               routePush(context, | ||||
|                                   page: const SubmissionParent()); | ||||
|                             } | ||||
|                           }); | ||||
|                         } | ||||
|                       }, | ||||
|                       ), | ||||
|                     ) | ||||
|                   ], | ||||
|                 ), | ||||
|   | ||||
| @@ -24,15 +24,18 @@ class InitialSignature extends StatelessWidget { | ||||
|         builder: (context, child) { | ||||
|           return Consumer<SubmissionDataViewModel>( | ||||
|               builder: (context, provider, child) { | ||||
|             return Column( | ||||
|             return SizedBox( | ||||
|               height: SizeConfig.height * .8, | ||||
|               child: Column( | ||||
|                 crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                 mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|                 children: [ | ||||
|                   const TextCaption(title: 'Draw your digital sign'), | ||||
|                   const ImageView(image: PathAssets.frameSignature), | ||||
|                   const ListTileView( | ||||
|                       title: | ||||
|                           'Make sure the sign you draw is match with your ID Card'), | ||||
|                 SizedBox(height: SizeConfig.height * .07), | ||||
|                   SizedBox(height: SizeConfig.height * .1), | ||||
|                   ButtonView( | ||||
|                     name: 'Next', | ||||
|                     onPressed: () { | ||||
| @@ -44,6 +47,7 @@ class InitialSignature extends StatelessWidget { | ||||
|                     }, | ||||
|                   ) | ||||
|                 ], | ||||
|               ), | ||||
|             ); | ||||
|           }); | ||||
|         }); | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| import 'package:camera/camera.dart'; | ||||
| import 'package:cims_apps/application/component/select_form/select_form_view.dart'; | ||||
| import 'package:cims_apps/features/auth/registration/view/submission_data/data_bank/confirm_bank_account.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:shared_preferences/shared_preferences.dart'; | ||||
|  | ||||
| @@ -14,11 +15,16 @@ class SubmissionDataViewModel extends ChangeNotifier { | ||||
|   bool _isEmailVerify = false; | ||||
|   bool get isEmailVerify => _isEmailVerify; | ||||
|   var formKeyPersonalData = GlobalKey<FormState>(); | ||||
|   var formKeySubmitEmail = GlobalKey<FormState>(); | ||||
|   var formKeySubmitIdCard = GlobalKey<FormState>(); | ||||
|   var formKeySubmitDataBank = GlobalKey<FormState>(); | ||||
|   TextEditingController ctrlOccupation = TextEditingController(); | ||||
|   TextEditingController ctrlIncome = TextEditingController(); | ||||
|   TextEditingController ctrlMarital = TextEditingController(); | ||||
|   TextEditingController ctrlSourceFund = TextEditingController(); | ||||
|   TextEditingController ctrlBankName = TextEditingController(); | ||||
|   TextEditingController ctrlNameAccountBank = TextEditingController(); | ||||
|   TextEditingController ctrlNoAccountBank = TextEditingController(); | ||||
|   TextEditingController ctrlBankNameSearch = TextEditingController(); | ||||
|   TextEditingController ctrlBirthDate = TextEditingController(); | ||||
|   int step = 1; | ||||
| @@ -120,4 +126,21 @@ class SubmissionDataViewModel extends ChangeNotifier { | ||||
|     valueBank = value; | ||||
|     notifyListeners(); | ||||
|   } | ||||
|  | ||||
|   List<ModelDataBank> listDataBank = []; | ||||
|  | ||||
|   Future<List<ModelDataBank>?> submitDataBank( | ||||
|       {required String bankName, accountNumber, accountName}) async { | ||||
|     List<ModelDataBank> listResponse = [ | ||||
|       ModelDataBank('Bank Name', bankName), | ||||
|       ModelDataBank('Account Number', accountNumber), | ||||
|       ModelDataBank('Account Owner Name', accountName), | ||||
|       ModelDataBank('Name on ID card', accountName), | ||||
|     ]; | ||||
|     if (listResponse.first.subtitle != null) { | ||||
|       // listDataBank = listResponse; | ||||
|       return listResponse; | ||||
|     } | ||||
|     return null; | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										8
									
								
								lib/features/profile/model/list_menu_model.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,8 @@ | ||||
| import 'package:flutter/material.dart'; | ||||
|  | ||||
| class ListMenuModel { | ||||
|   final String title, pathAsset; | ||||
|   final Widget page; | ||||
|   ListMenuModel( | ||||
|       {required this.title, required this.pathAsset, required this.page}); | ||||
| } | ||||
| @@ -1,5 +1,8 @@ | ||||
| import 'package:cims_apps/application/component/custom_app_bar/custom_app_bar.dart'; | ||||
| import 'package:cims_apps/application/assets/path_assets.dart'; | ||||
| import 'package:cims_apps/application/component/image/image_view.dart'; | ||||
| import 'package:cims_apps/application/theme/color_palette.dart'; | ||||
| import 'package:cims_apps/core/utils/size_config.dart'; | ||||
| import 'package:cims_apps/features/profile/model/list_menu_model.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
|  | ||||
| class ProfileView extends StatelessWidget { | ||||
| @@ -7,8 +10,110 @@ class ProfileView extends StatelessWidget { | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     List<ListMenuModel> listMenuGeneral = [ | ||||
|       ListMenuModel( | ||||
|           title: 'Personal Data', | ||||
|           pathAsset: PathAssets.iconProfile, | ||||
|           page: Container()), | ||||
|       ListMenuModel( | ||||
|           title: 'Change Password', | ||||
|           pathAsset: PathAssets.iconLock, | ||||
|           page: Container()), | ||||
|       ListMenuModel( | ||||
|           title: 'Add Card', pathAsset: PathAssets.iconCard, page: Container()), | ||||
|       ListMenuModel( | ||||
|           title: 'Settings', | ||||
|           pathAsset: PathAssets.iconSetting, | ||||
|           page: Container()), | ||||
|     ]; | ||||
|  | ||||
|     List<ListMenuModel> listMenuPreferences = [ | ||||
|       ListMenuModel( | ||||
|           title: 'FAQs', pathAsset: PathAssets.iconChat, page: Container()), | ||||
|       ListMenuModel( | ||||
|           title: 'Log Out', | ||||
|           pathAsset: PathAssets.iconLogout, | ||||
|           page: Container()), | ||||
|     ]; | ||||
|     TextStyle textStyle = const TextStyle( | ||||
|       fontSize: 20, | ||||
|       fontWeight: FontWeight.bold, | ||||
|       color: Colors.white, | ||||
|     ); | ||||
|  | ||||
|     Widget cardContent( | ||||
|         {required String title, required List<ListMenuModel> listMenu}) { | ||||
|       return Container( | ||||
|         width: SizeConfig.width, | ||||
|         padding: const EdgeInsets.all(16.0), | ||||
|         decoration: const BoxDecoration( | ||||
|             color: Colors.white, | ||||
|             borderRadius: BorderRadius.all(Radius.circular(12))), | ||||
|         child: Column( | ||||
|           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|           children: [ | ||||
|             Text(title), | ||||
|             ...listMenu.map((e) { | ||||
|               return Padding( | ||||
|                 padding: const EdgeInsets.only(top: 8.0), | ||||
|                 child: ListTile( | ||||
|                   title: Text(e.title), | ||||
|                   contentPadding: EdgeInsets.zero, | ||||
|                   leading: ImageView( | ||||
|                       image: e.pathAsset, width: SizeConfig.width * .08), | ||||
|                   trailing: const Icon(Icons.arrow_forward_ios, | ||||
|                       color: ColorPalette.slate400), | ||||
|                   onTap: () { | ||||
|                     // routePush(context, page: e.page); | ||||
|                   }, | ||||
|                 ), | ||||
|               ); | ||||
|             }), | ||||
|           ], | ||||
|         ), | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     return Scaffold( | ||||
|       appBar: CustomAppBar(height: SizeConfig.height * .08, title: 'Profile'), | ||||
|       appBar: null, | ||||
|       body: Stack( | ||||
|         children: [ | ||||
|           ImageView(image: PathAssets.bgProfile, width: SizeConfig.width), | ||||
|           Container( | ||||
|             width: SizeConfig.width, | ||||
|             padding: const EdgeInsets.only(top: 40.0, left: 16.0, right: 16.0), | ||||
|             child: SingleChildScrollView( | ||||
|               padding: const EdgeInsets.only(bottom: 60.0), | ||||
|               child: Column( | ||||
|                 crossAxisAlignment: CrossAxisAlignment.center, | ||||
|                 children: [ | ||||
|                   Text('Profile', style: textStyle), | ||||
|                   Padding( | ||||
|                     padding: const EdgeInsets.only(top: 40.0, bottom: 8.0), | ||||
|                     child: ImageView( | ||||
|                       image: PathAssets.imgCatOutlined, | ||||
|                       width: SizeConfig.width * .28, | ||||
|                     ), | ||||
|                   ), | ||||
|                   Text('Muhamad Rosyidin', | ||||
|                       style: textStyle.copyWith(fontSize: 24)), | ||||
|                   Text('Investor Conservative', | ||||
|                       style: textStyle.copyWith( | ||||
|                         fontSize: 16, | ||||
|                         fontWeight: FontWeight.normal, | ||||
|                         color: ColorPalette.green400, | ||||
|                       )), | ||||
|                   SizedBox(height: SizeConfig.height * .02), | ||||
|                   cardContent(title: 'General', listMenu: listMenuGeneral), | ||||
|                   SizedBox(height: SizeConfig.height * .03), | ||||
|                   cardContent( | ||||
|                       title: 'Preference', listMenu: listMenuPreferences), | ||||
|                 ], | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										51
									
								
								lib/features/transaction/view/cancel_view.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,51 @@ | ||||
| import 'package:cims_apps/application/assets/path_assets.dart'; | ||||
| import 'package:cims_apps/application/component/card_transaction/card_transaction_view.dart'; | ||||
| import 'package:cims_apps/application/component/card_transaction/empty_card_transaction.dart'; | ||||
| import 'package:cims_apps/core/route/route.dart'; | ||||
| import 'package:cims_apps/core/utils/number_formatter.dart'; | ||||
| import 'package:cims_apps/core/utils/string_utils.dart'; | ||||
| import 'package:cims_apps/features/transaction/view/subscribe_detail_view.dart'; | ||||
| import 'package:cims_apps/features/transaction/viewmodel/transaction_viewmodel.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
|  | ||||
| class CancelView extends StatelessWidget { | ||||
|   const CancelView({Key? key}) : super(key: key); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return SingleChildScrollView( | ||||
|       child: Consumer<TransactionViewModel>( | ||||
|         builder: (context, provider, child) => Column( | ||||
|           children: [ | ||||
|             if (provider.listCancelTransaction.isEmpty) | ||||
|               EmptyCardTransaction( | ||||
|                 onPressedButton: () {}, | ||||
|               ), | ||||
|             ...provider.listCancelTransaction.map((e) { | ||||
|               return CardTransactionView( | ||||
|                 onTap: () { | ||||
|                   routePush(context, | ||||
|                       page: const SubscribeDetailView( | ||||
|                         type: 'virtual', | ||||
|                       )); | ||||
|                 }, | ||||
|                 iconPath: PathAssets.iconEducation, | ||||
|                 type: 'Education', | ||||
|                 amount: NumberFormatter.numberCurrency( | ||||
|                   6000000, | ||||
|                   'Rp ', | ||||
|                   'id_ID', | ||||
|                   decimalDigits: 0, | ||||
|                 ), | ||||
|                 timeTransaction: StringUtils.formatTime(DateTime.now()), | ||||
|                 subs: '3 Subscription', | ||||
|                 step: 'cancel', | ||||
|               ); | ||||
|             }), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										53
									
								
								lib/features/transaction/view/done_view.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,53 @@ | ||||
| import 'package:cims_apps/application/assets/path_assets.dart'; | ||||
| import 'package:cims_apps/application/component/card_transaction/card_transaction_view.dart'; | ||||
| import 'package:cims_apps/application/component/card_transaction/empty_card_transaction.dart'; | ||||
| import 'package:cims_apps/core/route/route.dart'; | ||||
| import 'package:cims_apps/core/utils/number_formatter.dart'; | ||||
| import 'package:cims_apps/core/utils/string_utils.dart'; | ||||
| import 'package:cims_apps/features/transaction/view/subscribe_detail_view.dart'; | ||||
| import 'package:cims_apps/features/transaction/viewmodel/transaction_viewmodel.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
|  | ||||
| class DoneView extends StatelessWidget { | ||||
|   const DoneView({Key? key}) : super(key: key); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return SingleChildScrollView( | ||||
|       child: Consumer<TransactionViewModel>( | ||||
|         builder: (context, provider, child) { | ||||
|           return Column( | ||||
|             children: [ | ||||
|               if (provider.listDoneTransaction.isEmpty) | ||||
|                 EmptyCardTransaction( | ||||
|                   onPressedButton: () {}, | ||||
|                 ), | ||||
|               ...provider.listDoneTransaction.map((e) { | ||||
|                 return CardTransactionView( | ||||
|                   onTap: () { | ||||
|                     routePush(context, | ||||
|                         page: const SubscribeDetailView( | ||||
|                           type: 'virtual', | ||||
|                         )); | ||||
|                   }, | ||||
|                   iconPath: PathAssets.iconEducation, | ||||
|                   type: 'Education', | ||||
|                   amount: NumberFormatter.numberCurrency( | ||||
|                     6000000, | ||||
|                     'Rp ', | ||||
|                     'id_ID', | ||||
|                     decimalDigits: 0, | ||||
|                   ), | ||||
|                   timeTransaction: StringUtils.formatTime(DateTime.now()), | ||||
|                   subs: '3 Subscription', | ||||
|                   step: 'done', | ||||
|                 ); | ||||
|               }), | ||||
|             ], | ||||
|           ); | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										50
									
								
								lib/features/transaction/view/onprocess_view.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,50 @@ | ||||
| import 'package:cims_apps/application/assets/path_assets.dart'; | ||||
| import 'package:cims_apps/application/component/card_transaction/card_transaction_view.dart'; | ||||
| import 'package:cims_apps/application/component/card_transaction/empty_card_transaction.dart'; | ||||
| import 'package:cims_apps/core/route/route.dart'; | ||||
| import 'package:cims_apps/core/utils/number_formatter.dart'; | ||||
| import 'package:cims_apps/core/utils/string_utils.dart'; | ||||
| import 'package:cims_apps/features/transaction/view/subscribe_detail_view.dart'; | ||||
| import 'package:cims_apps/features/transaction/viewmodel/transaction_viewmodel.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
|  | ||||
| class OnProcessView extends StatelessWidget { | ||||
|   const OnProcessView({Key? key}) : super(key: key); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return SingleChildScrollView( | ||||
|       child: | ||||
|           Consumer<TransactionViewModel>(builder: (context, provider, child) { | ||||
|         return Column(children: [ | ||||
|           if (provider.listOnProcessTransaction.isEmpty) | ||||
|             EmptyCardTransaction( | ||||
|               onPressedButton: () {}, | ||||
|             ), | ||||
|           ...provider.listOnProcessTransaction.map((e) { | ||||
|             return CardTransactionView( | ||||
|               onTap: () { | ||||
|                 routePush(context, | ||||
|                     page: const SubscribeDetailView( | ||||
|                       type: 'virtual', | ||||
|                     )); | ||||
|               }, | ||||
|               iconPath: PathAssets.iconHome, | ||||
|               type: 'Home', | ||||
|               amount: NumberFormatter.numberCurrency( | ||||
|                 6000000, | ||||
|                 'Rp ', | ||||
|                 'id_ID', | ||||
|                 decimalDigits: 0, | ||||
|               ), | ||||
|               timeTransaction: StringUtils.formatTime(DateTime.now()), | ||||
|               subs: '3 Subscription', | ||||
|               step: 'on process', | ||||
|             ); | ||||
|           }), | ||||
|         ]); | ||||
|       }), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										516
									
								
								lib/features/transaction/view/subscribe_detail_view.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,516 @@ | ||||
| import 'package:cims_apps/application/assets/path_assets.dart'; | ||||
| import 'package:cims_apps/application/component/button/back_button_view.dart'; | ||||
| import 'package:cims_apps/application/component/button/button_view.dart'; | ||||
| import 'package:cims_apps/application/component/image/image_view.dart'; | ||||
| import 'package:cims_apps/application/component/list_tile/list_tile_view.dart'; | ||||
| import 'package:cims_apps/application/theme/color_palette.dart'; | ||||
| import 'package:cims_apps/core/utils/size_config.dart'; | ||||
| import 'package:cims_apps/core/utils/string_utils.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
|  | ||||
| class SubscribeDetailView extends StatelessWidget { | ||||
|   final String type; | ||||
|   const SubscribeDetailView({Key? key, required this.type}) : super(key: key); | ||||
|  | ||||
|   Widget _stepper( | ||||
|     TextTheme textTheme, { | ||||
|     required String description, | ||||
|     String? date, | ||||
|     time, | ||||
|     bool isActive = false, | ||||
|     bool isDone = false, | ||||
|     bool isLast = false, | ||||
|   }) { | ||||
|     return Container( | ||||
|       padding: const EdgeInsets.only(top: 8.0, left: 16.0), | ||||
|       child: Row( | ||||
|         crossAxisAlignment: CrossAxisAlignment.start, | ||||
|         children: [ | ||||
|           Column( | ||||
|             children: [ | ||||
|               Padding( | ||||
|                 padding: const EdgeInsets.only(left: 2.0, top: 8.0), | ||||
|                 child: Icon( | ||||
|                   Icons.circle_rounded, | ||||
|                   size: 10, | ||||
|                   color: isDone | ||||
|                       ? ColorPalette.primary | ||||
|                       : isActive | ||||
|                           ? ColorPalette.primary | ||||
|                           : ColorPalette.slate400, | ||||
|                 ), | ||||
|               ), | ||||
|               if (!isLast) | ||||
|                 ConstrainedBox( | ||||
|                   constraints: BoxConstraints.expand( | ||||
|                       height: SizeConfig.width * .2, width: 0.0), | ||||
|                   child: Padding( | ||||
|                     padding: const EdgeInsets.symmetric(vertical: 4.0), | ||||
|                     child: VerticalDivider( | ||||
|                       color: | ||||
|                           isDone ? ColorPalette.primary : ColorPalette.slate400, | ||||
|                       thickness: 2.0, | ||||
|                     ), | ||||
|                   ), | ||||
|                 ), | ||||
|             ], | ||||
|           ), | ||||
|           Padding( | ||||
|             padding: const EdgeInsets.only(left: 16.0), | ||||
|             child: Column( | ||||
|               crossAxisAlignment: CrossAxisAlignment.start, | ||||
|               children: [ | ||||
|                 Padding( | ||||
|                   padding: const EdgeInsets.only(bottom: 8.0), | ||||
|                   child: Text( | ||||
|                     description, | ||||
|                     style: TextStyle( | ||||
|                       fontSize: 16, | ||||
|                       fontWeight: FontWeight.w600, | ||||
|                       color: isDone | ||||
|                           ? Colors.black87 | ||||
|                           : isActive | ||||
|                               ? ColorPalette.primary | ||||
|                               : ColorPalette.slate400, | ||||
|                     ), | ||||
|                   ), | ||||
|                 ), | ||||
|                 isDone | ||||
|                     ? Row( | ||||
|                         children: [ | ||||
|                           Text(date ?? ''), | ||||
|                           const Padding( | ||||
|                             padding: EdgeInsets.symmetric(horizontal: 4.0), | ||||
|                             child: Icon( | ||||
|                               Icons.circle, | ||||
|                               size: 6, | ||||
|                               color: ColorPalette.slate400, | ||||
|                             ), | ||||
|                           ), | ||||
|                           Text(time ?? ''), | ||||
|                         ], | ||||
|                       ) | ||||
|                     : const SizedBox(), | ||||
|               ], | ||||
|             ), | ||||
|           ) | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   Widget _listProduct(BuildContext context) { | ||||
|     TextTheme textTheme = Theme.of(context).textTheme; | ||||
|     return Padding( | ||||
|       padding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0), | ||||
|       child: Column( | ||||
|         children: [ | ||||
|           Row( | ||||
|             children: [ | ||||
|               ImageView( | ||||
|                 image: PathAssets.imgProduct, | ||||
|                 width: SizeConfig.width * .1, | ||||
|               ), | ||||
|               Padding( | ||||
|                 padding: const EdgeInsets.only(left: 8.0), | ||||
|                 child: Column( | ||||
|                   crossAxisAlignment: CrossAxisAlignment.start, | ||||
|                   children: [ | ||||
|                     Text( | ||||
|                       'Gemilang Dana Kas Maxima', | ||||
|                       style: textTheme.headlineSmall, | ||||
|                     ), | ||||
|                     Container( | ||||
|                         padding: const EdgeInsets.symmetric( | ||||
|                             horizontal: 8.0, vertical: 2.0), | ||||
|                         decoration: BoxDecoration( | ||||
|                           color: ColorPalette.purple100, | ||||
|                           border: Border.all( | ||||
|                             width: 1, | ||||
|                             color: ColorPalette.purple, | ||||
|                           ), | ||||
|                           borderRadius: | ||||
|                               const BorderRadius.all(Radius.circular(24)), | ||||
|                         ), | ||||
|                         child: const Text( | ||||
|                           'Money Market', | ||||
|                           style: TextStyle( | ||||
|                             color: ColorPalette.purple500, | ||||
|                           ), | ||||
|                         )), | ||||
|                   ], | ||||
|                 ), | ||||
|               ), | ||||
|             ], | ||||
|           ), | ||||
|           Padding( | ||||
|             padding: const EdgeInsets.symmetric(vertical: 8.0), | ||||
|             child: Row( | ||||
|               mainAxisAlignment: MainAxisAlignment.spaceBetween, | ||||
|               children: [ | ||||
|                 const Text('Investment Amount'), | ||||
|                 Text( | ||||
|                   'Rp 2.000.000', | ||||
|                   style: textTheme.headlineSmall, | ||||
|                 ), | ||||
|               ], | ||||
|             ), | ||||
|           ), | ||||
|           const Divider( | ||||
|             color: ColorPalette.slate200, | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   Widget _cardVA(BuildContext context) { | ||||
|     return Container( | ||||
|       margin: const EdgeInsets.only(top: 16.0, bottom: 8.0), | ||||
|       padding: const EdgeInsets.only(bottom: 16.0), | ||||
|       decoration: BoxDecoration( | ||||
|         color: Colors.white, | ||||
|         border: Border.all(width: 1, color: ColorPalette.slate200), | ||||
|         borderRadius: const BorderRadius.all(Radius.circular(12)), | ||||
|       ), | ||||
|       child: Padding( | ||||
|         padding: const EdgeInsets.symmetric(horizontal: 16.0), | ||||
|         child: Column( | ||||
|           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|           children: [ | ||||
|             SizedBox(height: SizeConfig.height * .02), | ||||
|             RichText( | ||||
|                 text: const TextSpan(children: [ | ||||
|               TextSpan( | ||||
|                 text: 'Transfer to  ', | ||||
|                 style: TextStyle( | ||||
|                   color: ColorPalette.slate400, | ||||
|                   fontWeight: FontWeight.normal, | ||||
|                 ), | ||||
|               ), | ||||
|               TextSpan( | ||||
|                 text: 'ABC Virtual Account', | ||||
|                 style: TextStyle( | ||||
|                   color: ColorPalette.slate800, | ||||
|                   fontWeight: FontWeight.bold, | ||||
|                 ), | ||||
|               ), | ||||
|             ])), | ||||
|             SizedBox(height: SizeConfig.height * .01), | ||||
|             const Text( | ||||
|               'Ferdy Maulana', | ||||
|               style: TextStyle( | ||||
|                 color: Colors.black87, | ||||
|                 fontSize: 16, | ||||
|               ), | ||||
|             ), | ||||
|             ListTileView( | ||||
|               title: '8785 0000 3165 5512', | ||||
|               prefixIcon: const SizedBox(), | ||||
|               padding: const EdgeInsets.only(right: 16.0), | ||||
|               margin: const EdgeInsets.only(top: 8.0, bottom: 8.0), | ||||
|               textStyle: const TextStyle( | ||||
|                   fontWeight: FontWeight.bold, color: ColorPalette.primary), | ||||
|               suffixIcon: ButtonView( | ||||
|                 name: 'Copy', | ||||
|                 width: SizeConfig.width * .3, | ||||
|                 height: SizeConfig.height * .052, | ||||
|                 sizeBorderRadius: 8.0, | ||||
|                 widthPrefix: 8.0, | ||||
|                 marginVertical: 10.0, | ||||
|                 prefixIcon: const Icon( | ||||
|                   Icons.file_copy_outlined, | ||||
|                   color: Colors.white, | ||||
|                   size: 20, | ||||
|                 ), | ||||
|                 onPressed: () { | ||||
|                   StringUtils.iCopyToClipboard( | ||||
|                     context, | ||||
|                     text: '8785 0000 3165 5512', | ||||
|                   ); | ||||
|                 }, | ||||
|               ), | ||||
|             ), | ||||
|             const Divider(color: ColorPalette.slate200), | ||||
|             const Text( | ||||
|               'Amount to be transferred', | ||||
|               style: TextStyle( | ||||
|                 color: Colors.black87, | ||||
|                 fontSize: 16, | ||||
|               ), | ||||
|             ), | ||||
|             ListTileView( | ||||
|               title: 'Rp 10.000.000', | ||||
|               prefixIcon: const SizedBox(), | ||||
|               padding: const EdgeInsets.only(right: 16.0), | ||||
|               margin: const EdgeInsets.only(top: 8.0), | ||||
|               textStyle: const TextStyle( | ||||
|                   fontWeight: FontWeight.bold, color: ColorPalette.primary), | ||||
|               suffixIcon: ButtonView( | ||||
|                 name: 'Copy', | ||||
|                 width: SizeConfig.width * .3, | ||||
|                 height: SizeConfig.height * .052, | ||||
|                 sizeBorderRadius: 8.0, | ||||
|                 widthPrefix: 8.0, | ||||
|                 marginVertical: 10.0, | ||||
|                 prefixIcon: const Icon( | ||||
|                   Icons.file_copy_outlined, | ||||
|                   color: Colors.white, | ||||
|                   size: 20, | ||||
|                 ), | ||||
|                 onPressed: () { | ||||
|                   StringUtils.iCopyToClipboard( | ||||
|                     context, | ||||
|                     text: '10.000.000', | ||||
|                   ); | ||||
|                 }, | ||||
|               ), | ||||
|             ), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   Widget _cardContent(BuildContext context) { | ||||
|     TextTheme textTheme = Theme.of(context).textTheme; | ||||
|     List mySteps = const [ | ||||
|       { | ||||
|         'desc': 'Payment Being Verified', | ||||
|         'date': '07 Feb 2024 ', | ||||
|         'time': '21:01', | ||||
|         'isActive': false, | ||||
|         'isDone': true, | ||||
|         'isLast': false, | ||||
|       }, | ||||
|       { | ||||
|         'desc': 'Successful Payment', | ||||
|         'date': '07 Feb 2024 ', | ||||
|         'time': '21:01', | ||||
|         'isActive': true, | ||||
|         'isDone': false, | ||||
|         'isLast': false, | ||||
|       }, | ||||
|       { | ||||
|         'desc': 'Investment Manager Verification', | ||||
|         'date': '07 Feb 2024 ', | ||||
|         'time': '21:01', | ||||
|         'isActive': false, | ||||
|         'isDone': false, | ||||
|         'isLast': false, | ||||
|       }, | ||||
|       { | ||||
|         'desc': 'Successful Purchase', | ||||
|         'date': '07 Feb 2024 ', | ||||
|         'time': '21:01', | ||||
|         'isActive': false, | ||||
|         'isDone': false, | ||||
|         'isLast': true, | ||||
|       }, | ||||
|     ]; | ||||
|  | ||||
|     return SingleChildScrollView( | ||||
|       padding: const EdgeInsets.symmetric(horizontal: 16.0), | ||||
|       child: Column( | ||||
|         crossAxisAlignment: CrossAxisAlignment.start, | ||||
|         children: [ | ||||
|           Row( | ||||
|             mainAxisAlignment: MainAxisAlignment.center, | ||||
|             children: [ | ||||
|               const Icon(Icons.access_time, color: ColorPalette.slate400), | ||||
|               const Padding( | ||||
|                 padding: EdgeInsets.symmetric(horizontal: 8.0), | ||||
|                 child: Text('Waiting for Payment'), | ||||
|               ), | ||||
|               Text('23:56:42', style: textTheme.headlineSmall), | ||||
|             ], | ||||
|           ), | ||||
|           type == 'virtual' ? _cardVA(context) : const SizedBox(), | ||||
|           Padding( | ||||
|             padding: const EdgeInsets.symmetric(vertical: 16.0), | ||||
|             child: Text('Purchase Details', style: textTheme.headlineSmall), | ||||
|           ), | ||||
|           Container( | ||||
|             margin: const EdgeInsets.symmetric(vertical: 8.0), | ||||
|             padding: const EdgeInsets.only(bottom: 16.0), | ||||
|             decoration: BoxDecoration( | ||||
|               color: Colors.white, | ||||
|               border: Border.all(width: 1, color: ColorPalette.slate200), | ||||
|               borderRadius: const BorderRadius.all(Radius.circular(12)), | ||||
|             ), | ||||
|             child: ExpansionTile( | ||||
|               shape: ShapeBorder.lerp( | ||||
|                   const StadiumBorder(side: BorderSide.none), | ||||
|                   InputBorder.none, | ||||
|                   0), | ||||
|               title: Row( | ||||
|                 children: [ | ||||
|                   ImageView( | ||||
|                     image: PathAssets.iconEducation, | ||||
|                     width: SizeConfig.width * .08, | ||||
|                   ), | ||||
|                   const Padding( | ||||
|                     padding: EdgeInsets.only(left: 8.0), | ||||
|                     child: Text('Education'), | ||||
|                   ), | ||||
|                 ], | ||||
|               ), | ||||
|               subtitle: Padding( | ||||
|                 padding: const EdgeInsets.only(left: 40.0), | ||||
|                 child: Text('3 Subscriptions', style: textTheme.bodyMedium), | ||||
|               ), | ||||
|               children: [ | ||||
|                 _listProduct(context), | ||||
|                 _listProduct(context), | ||||
|               ], | ||||
|             ), | ||||
|           ), | ||||
|           ...mySteps.map((e) { | ||||
|             return _stepper( | ||||
|               textTheme, | ||||
|               description: '${e['desc']}', | ||||
|               date: e['date'], | ||||
|               time: e['time'], | ||||
|               isActive: e['isActive'], | ||||
|               isDone: e['isDone'], | ||||
|               isLast: e['isLast'], | ||||
|             ); | ||||
|           }), | ||||
|           Container( | ||||
|               margin: const EdgeInsets.symmetric(vertical: 8.0), | ||||
|               // padding: const EdgeInsets.only(bottom: 16.0), | ||||
|               decoration: BoxDecoration( | ||||
|                 color: Colors.white, | ||||
|                 border: Border.all(width: 1, color: ColorPalette.slate200), | ||||
|                 borderRadius: const BorderRadius.all(Radius.circular(12)), | ||||
|               ), | ||||
|               child: ExpansionTile( | ||||
|                   shape: ShapeBorder.lerp( | ||||
|                       const StadiumBorder(side: BorderSide.none), | ||||
|                       InputBorder.none, | ||||
|                       0), | ||||
|                   title: Row( | ||||
|                     children: [ | ||||
|                       const Expanded( | ||||
|                           child: Text('When Is My Purchase Complete?')), | ||||
|                       ImageView( | ||||
|                         image: PathAssets.imgExpandPurchase, | ||||
|                         width: SizeConfig.width * .15, | ||||
|                       ), | ||||
|                     ], | ||||
|                   ))), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Scaffold( | ||||
|       body: SizedBox( | ||||
|         child: Stack( | ||||
|           children: [ | ||||
|             const ImageView(image: PathAssets.imgDashboardAccount), | ||||
|             Column( | ||||
|               children: [ | ||||
|                 SizedBox(height: SizeConfig.height * .05), | ||||
|                 Padding( | ||||
|                   padding: const EdgeInsets.all(16.0), | ||||
|                   child: Row( | ||||
|                     mainAxisAlignment: MainAxisAlignment.start, | ||||
|                     children: [ | ||||
|                       BackButtonView( | ||||
|                         onPress: () => Navigator.pop(context), | ||||
|                       ), | ||||
|                       const Padding( | ||||
|                         padding: EdgeInsets.symmetric(horizontal: 80.0), | ||||
|                         child: Text( | ||||
|                           'Subscribe Detail', | ||||
|                           style: TextStyle( | ||||
|                             fontSize: 18, | ||||
|                             color: Colors.white, | ||||
|                             fontWeight: FontWeight.bold, | ||||
|                           ), | ||||
|                         ), | ||||
|                       ), | ||||
|                     ], | ||||
|                   ), | ||||
|                 ), | ||||
|                 type == 'virtual' | ||||
|                     ? const Padding( | ||||
|                         padding: EdgeInsets.only(bottom: 8.0), | ||||
|                         child: Text( | ||||
|                           'Virtual Account', | ||||
|                           style: TextStyle( | ||||
|                             color: Colors.white, | ||||
|                             fontSize: 16, | ||||
|                           ), | ||||
|                         ), | ||||
|                       ) | ||||
|                     : const SizedBox(), | ||||
|                 Row( | ||||
|                   mainAxisAlignment: MainAxisAlignment.center, | ||||
|                   children: [ | ||||
|                     ImageView( | ||||
|                       image: PathAssets.imgProduct, | ||||
|                       width: SizeConfig.width * .08, | ||||
|                     ), | ||||
|                     Padding( | ||||
|                       padding: const EdgeInsets.only(left: 8.0), | ||||
|                       child: Text( | ||||
|                         type == 'virtual' ? 'ABC' : 'Shopping Pay', | ||||
|                         style: const TextStyle( | ||||
|                           fontWeight: FontWeight.bold, | ||||
|                           color: Colors.white, | ||||
|                         ), | ||||
|                       ), | ||||
|                     ), | ||||
|                   ], | ||||
|                 ), | ||||
|                 const Padding( | ||||
|                   padding: EdgeInsets.only(top: 8.0), | ||||
|                   child: Row( | ||||
|                     mainAxisAlignment: MainAxisAlignment.center, | ||||
|                     children: [ | ||||
|                       Text( | ||||
|                         'No.Subscribe : ', | ||||
|                         style: TextStyle( | ||||
|                           fontWeight: FontWeight.normal, | ||||
|                           color: Colors.white, | ||||
|                         ), | ||||
|                       ), | ||||
|                       Text( | ||||
|                         'PI9393084SDMI1', | ||||
|                         style: TextStyle( | ||||
|                           fontWeight: FontWeight.bold, | ||||
|                           color: Colors.white, | ||||
|                         ), | ||||
|                       ), | ||||
|                     ], | ||||
|                   ), | ||||
|                 ), | ||||
|                 Expanded( | ||||
|                   child: Container( | ||||
|                     color: Colors.transparent, | ||||
|                     width: SizeConfig.width, | ||||
|                     padding: const EdgeInsets.only(top: 16.0), | ||||
|                     child: Container( | ||||
|                         margin: const EdgeInsets.only(top: 32.0), | ||||
|                         padding: const EdgeInsets.only(top: 32.0), | ||||
|                         decoration: const BoxDecoration( | ||||
|                           color: Colors.white, | ||||
|                           borderRadius: BorderRadius.only( | ||||
|                               topLeft: Radius.circular(24), | ||||
|                               topRight: Radius.circular(24)), | ||||
|                         ), | ||||
|                         child: _cardContent(context)), | ||||
|                   ), | ||||
|                 ), | ||||
|               ], | ||||
|             ) | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| @@ -1,14 +1,124 @@ | ||||
| import 'package:cims_apps/application/component/custom_app_bar/custom_app_bar.dart'; | ||||
| import 'package:cims_apps/application/assets/path_assets.dart'; | ||||
| import 'package:cims_apps/application/component/image/image_view.dart'; | ||||
| import 'package:cims_apps/application/theme/color_palette.dart'; | ||||
| import 'package:cims_apps/core/utils/size_config.dart'; | ||||
| import 'package:cims_apps/features/transaction/view/cancel_view.dart'; | ||||
| import 'package:cims_apps/features/transaction/view/done_view.dart'; | ||||
| import 'package:cims_apps/features/transaction/view/onprocess_view.dart'; | ||||
| import 'package:cims_apps/features/transaction/view/waiting_view.dart'; | ||||
| import 'package:cims_apps/features/transaction/viewmodel/transaction_viewmodel.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter_toggle_tab/flutter_toggle_tab.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
|  | ||||
| class TransactionView extends StatelessWidget { | ||||
|   const TransactionView({Key? key}) : super(key: key); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     List<Tab> textTabs = const [ | ||||
|       Tab(text: 'Waiting'), | ||||
|       Tab(text: 'On process'), | ||||
|       Tab(text: 'Done'), | ||||
|       Tab(text: 'Cancel'), | ||||
|     ]; | ||||
|     List<Widget> listTabBarView = const [ | ||||
|       WaitingView(), | ||||
|       OnProcessView(), | ||||
|       DoneView(), | ||||
|       CancelView(), | ||||
|     ]; | ||||
|  | ||||
|     return ChangeNotifierProvider( | ||||
|         create: (context) => TransactionViewModel(), | ||||
|         builder: (context, child) { | ||||
|           return Scaffold( | ||||
|       appBar: CustomAppBar(height: SizeConfig.height * 0.08, title: 'Transaction'), | ||||
|             backgroundColor: ColorPalette.primary, | ||||
|             body: SizedBox( | ||||
|               child: Stack( | ||||
|                 children: [ | ||||
|                   const ImageView(image: PathAssets.imgDashboardAccount), | ||||
|                   Column( | ||||
|                     children: [ | ||||
|                       SizedBox( | ||||
|                         height: SizeConfig.height * .05, | ||||
|                       ), | ||||
|                       const Center( | ||||
|                         child: Text( | ||||
|                           'Transaction', | ||||
|                           style: TextStyle( | ||||
|                               fontSize: 20, | ||||
|                               fontWeight: FontWeight.w700, | ||||
|                               color: Colors.white), | ||||
|                         ), | ||||
|                       ), | ||||
|                       SizedBox( | ||||
|                         height: SizeConfig.height * .04, | ||||
|                       ), | ||||
|                       Container( | ||||
|                         margin: const EdgeInsets.symmetric(horizontal: 24), | ||||
|                         child: FlutterToggleTab( | ||||
|                           height: SizeConfig.height * .065, | ||||
|                           width: SizeConfig.width * .2, | ||||
|                           marginSelected: const EdgeInsets.all(8.0), | ||||
|                           isScroll: false, | ||||
|                           selectedTextStyle: const TextStyle( | ||||
|                             color: ColorPalette.primary, | ||||
|                             fontWeight: FontWeight.w700, | ||||
|                           ), | ||||
|                           unSelectedTextStyle: const TextStyle( | ||||
|                             color: ColorPalette.blackFont, | ||||
|                             fontWeight: FontWeight.w700, | ||||
|                           ), | ||||
|                           unSelectedBackgroundColors: const [ | ||||
|                             ColorPalette.blue50 | ||||
|                           ], | ||||
|                           selectedBackgroundColors: const [ColorPalette.white], | ||||
|                           labels: const ['Subscribe', 'Reedem'], | ||||
|                           selectedLabelIndex: (p0) {}, | ||||
|                           selectedIndex: 0, | ||||
|                         ), | ||||
|                       ), | ||||
|                       Expanded( | ||||
|                           child: DefaultTabController( | ||||
|                         length: textTabs.length, | ||||
|                         child: Container( | ||||
|                           color: Colors.transparent, | ||||
|                           padding: const EdgeInsets.only(top: 32.0), | ||||
|                           child: Container( | ||||
|                             margin: const EdgeInsets.only(top: 24), | ||||
|                             padding: const EdgeInsets.only(top: 16.0), | ||||
|                             decoration: const BoxDecoration( | ||||
|                               color: Colors.white, | ||||
|                               borderRadius: BorderRadius.only( | ||||
|                                   topLeft: Radius.circular(24), | ||||
|                                   topRight: Radius.circular(24)), | ||||
|                             ), | ||||
|                             child: Column( | ||||
|                               children: [ | ||||
|                                 TabBar( | ||||
|                                   tabs: textTabs, | ||||
|                                   indicatorColor: Colors.blueAccent, | ||||
|                                 ), | ||||
|                                 Expanded( | ||||
|                                     child: Padding( | ||||
|                                   padding: const EdgeInsets.symmetric( | ||||
|                                     horizontal: 16.0, | ||||
|                                     vertical: 4.0, | ||||
|                                   ), | ||||
|                                   child: TabBarView(children: listTabBarView), | ||||
|                                 )) | ||||
|                               ], | ||||
|                             ), | ||||
|                           ), | ||||
|                         ), | ||||
|                       )), | ||||
|                     ], | ||||
|                   ) | ||||
|                 ], | ||||
|               ), | ||||
|             ), | ||||
|           ); | ||||
|         }); | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										49
									
								
								lib/features/transaction/view/waiting_view.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,49 @@ | ||||
| import 'package:cims_apps/application/assets/path_assets.dart'; | ||||
| import 'package:cims_apps/application/component/card_transaction/card_transaction_view.dart'; | ||||
| import 'package:cims_apps/application/component/card_transaction/empty_card_transaction.dart'; | ||||
| import 'package:cims_apps/core/route/route.dart'; | ||||
| import 'package:cims_apps/core/utils/number_formatter.dart'; | ||||
| import 'package:cims_apps/core/utils/string_utils.dart'; | ||||
| import 'package:cims_apps/features/transaction/view/subscribe_detail_view.dart'; | ||||
| import 'package:cims_apps/features/transaction/viewmodel/transaction_viewmodel.dart'; | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:provider/provider.dart'; | ||||
|  | ||||
| class WaitingView extends StatelessWidget { | ||||
|   const WaitingView({Key? key}) : super(key: key); | ||||
|  | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return SingleChildScrollView( | ||||
|       child: | ||||
|           Consumer<TransactionViewModel>(builder: (context, provider, child) { | ||||
|         return Column( | ||||
|           children: [ | ||||
|             provider.listWaitingTransaction.isNotEmpty | ||||
|                 ? CardTransactionView( | ||||
|                     onTap: () { | ||||
|                       routePush(context, | ||||
|                           page: const SubscribeDetailView( | ||||
|                             type: 'normal', | ||||
|                           )); | ||||
|                     }, | ||||
|                     iconPath: PathAssets.iconEducation, | ||||
|                     type: 'Education', | ||||
|                     amount: NumberFormatter.numberCurrency( | ||||
|                       6000000, | ||||
|                       'Rp ', | ||||
|                       'id_ID', | ||||
|                       decimalDigits: 0, | ||||
|                     ), | ||||
|                     timeTransaction: StringUtils.formatTime(DateTime.now()), | ||||
|                     subs: '3 Subscription', | ||||
|                     step: 'waiting') | ||||
|                 : EmptyCardTransaction( | ||||
|                     onPressedButton: () {}, | ||||
|                   ), | ||||
|           ], | ||||
|         ); | ||||
|       }), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| @@ -0,0 +1,8 @@ | ||||
| import 'package:flutter/material.dart'; | ||||
|  | ||||
| class TransactionViewModel extends ChangeNotifier { | ||||
|   List listWaitingTransaction = [1]; | ||||
|   List listOnProcessTransaction = [1]; | ||||
|   List listDoneTransaction = []; | ||||
|   List listCancelTransaction = []; | ||||
| } | ||||
| @@ -76,10 +76,8 @@ class MyApp extends StatelessWidget { | ||||
|             secondary: const Color(0xFFFECDA6), | ||||
|             onBackground: const Color(0xFFA9A9A9), | ||||
|           ), | ||||
|         bottomSheetTheme: BottomSheetThemeData( | ||||
|           backgroundColor: Colors.white, | ||||
|           surfaceTintColor: Colors.white | ||||
|         ) | ||||
|           bottomSheetTheme: const BottomSheetThemeData( | ||||
|               backgroundColor: Colors.white, surfaceTintColor: Colors.white) | ||||
|           // useMaterial3: true, | ||||
|           ), | ||||
|       initialRoute: initialRoute, | ||||
|   | ||||
| @@ -259,6 +259,14 @@ packages: | ||||
|     description: flutter | ||||
|     source: sdk | ||||
|     version: "0.0.0" | ||||
|   flutter_toggle_tab: | ||||
|     dependency: "direct main" | ||||
|     description: | ||||
|       name: flutter_toggle_tab | ||||
|       sha256: "90ad0d050f656df677998825f985637d010117a1793828cd7e6dadada4ecd2c5" | ||||
|       url: "https://pub.dev" | ||||
|     source: hosted | ||||
|     version: "1.4.1" | ||||
|   flutter_web_plugins: | ||||
|     dependency: transitive | ||||
|     description: flutter | ||||
|   | ||||
| @@ -53,6 +53,7 @@ dependencies: | ||||
|   shared_preferences: ^2.2.2 | ||||
|   calendar_date_picker2: ^0.5.3 | ||||
|   google_sign_in: ^6.2.1 | ||||
|   flutter_toggle_tab: ^1.4.1 | ||||
|  | ||||
|  | ||||
|  | ||||
|   | ||||