diff --git a/assets/image/2x/register_account.webp b/assets/image/2x/register_account.webp new file mode 100644 index 00000000..c2305773 Binary files /dev/null and b/assets/image/2x/register_account.webp differ diff --git a/assets/image/2x/retrieve_password.webp b/assets/image/2x/retrieve_password.webp new file mode 100644 index 00000000..e826dc06 Binary files /dev/null and b/assets/image/2x/retrieve_password.webp differ diff --git a/assets/image/3x/register_account.webp b/assets/image/3x/register_account.webp new file mode 100644 index 00000000..c3bb79f3 Binary files /dev/null and b/assets/image/3x/register_account.webp differ diff --git a/assets/image/3x/retrieve_password.webp b/assets/image/3x/retrieve_password.webp new file mode 100644 index 00000000..5cae0ff4 Binary files /dev/null and b/assets/image/3x/retrieve_password.webp differ diff --git a/assets/image/register_account.webp b/assets/image/register_account.webp new file mode 100644 index 00000000..12ec3878 Binary files /dev/null and b/assets/image/register_account.webp differ diff --git a/assets/image/retrieve_password.webp b/assets/image/retrieve_password.webp new file mode 100644 index 00000000..31e49ad5 Binary files /dev/null and b/assets/image/retrieve_password.webp differ diff --git a/lib/business_system/goods/add_goods/add_goods_page.dart b/lib/business_system/goods/add_goods/add_goods_page.dart new file mode 100644 index 00000000..27932ed1 --- /dev/null +++ b/lib/business_system/goods/add_goods/add_goods_page.dart @@ -0,0 +1,1211 @@ +import 'package:dio/dio.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_easyloading/flutter_easyloading.dart'; +import 'package:flutter_smart_dialog/flutter_smart_dialog.dart'; +import 'package:huixiang/view_widget/my_appbar.dart'; +import 'package:flutter_screenutil/flutter_screenutil.dart'; +import '../../../generated/l10n.dart'; +import '../../../retrofit/business_api.dart'; +import '../../../retrofit/data/base_data.dart'; +import '../../../retrofit/data/goods_category_list.dart'; +import '../../../retrofit/data/product_group_list.dart'; +import '../../../utils/business_instance.dart'; +import '../../../utils/flutter_utils.dart'; +import '../../../utils/font_weight.dart'; +import '../../../view_widget/settlement_tips_dialog.dart'; + +class AddGoodsPage extends StatefulWidget { + final Map arguments; + + AddGoodsPage({this.arguments}); + + @override + State createState() { + return _AddGoodsPage(); + } +} + +class _AddGoodsPage extends State { + TextEditingController goodsNameController = TextEditingController(); + TextEditingController profileController = TextEditingController(); + TextEditingController skuController = TextEditingController(); + TextEditingController heftController = TextEditingController(); + TextEditingController priceController = TextEditingController(); + TextEditingController originalPriceController = TextEditingController(); + TextEditingController packingChargeController = TextEditingController(); + TextEditingController stockController = TextEditingController(); + BusinessApiService businessService; + ProductGroupList productGroupList; + List goodsCategoryList = []; + String networkError = ""; + int networkStatus = 0; + String categoryName = ""; + int categoryIndex = 0; + String categoryId = ""; + String groupName = ""; + int groupIndex = 0; + String groupId = ""; + bool isLogistics = false; + bool isGround = true; + bool isOversold = true; + bool isSetMeal = false; + bool isAttrStyle = false; + bool isKeyBoardShow = false; + FocusNode _focusNode = FocusNode(); + + @override + void initState() { + super.initState(); + WidgetsBinding.instance.addPostFrameCallback((_) { + setState(() { + print("object: ${MediaQuery.of(context).viewInsets.bottom}"); + if (MediaQuery.of(context).viewInsets.bottom == 0) { + if (isKeyBoardShow) { + isKeyBoardShow = false; + //关闭键盘 软键盘关闭了, 清除输入控件的焦点, 否则重新进入页面会导致软键盘再弹出问题 + FocusScope.of(context).requestFocus(FocusNode()); + } + } else { + isKeyBoardShow = true; + } + }); + }); + _onRefresh(); + } + + ///离开页面记着销毁和清除 + @override + void dispose() { + _focusNode.unfocus(); + super.dispose(); + } + + _onRefresh() async { + EasyLoading.show( + status: S.current.zhengzaijiazai, + maskType: EasyLoadingMaskType.black); + await queryProductGroupList(); + await queryCategorize(); + EasyLoading.dismiss(); + if (mounted) + setState(() {}); + } + + ///分组列表 + queryProductGroupList() async { + if (businessService == null) { + businessService = BusinessApiService(Dio(), + context: context, + token: BusinessInstance.instance.businessToken, + tenant: BusinessInstance.instance.businessTenant, + storeId: widget.arguments["storeId"]); + } + BaseData baseData = await businessService.productGroup({ + "current": 1, + "map": {}, + "model": {"groupImg": "", "groupName": "", "isDelete": 0}, + "order": "ascending", + "size": 100, + "sort": "sort" + }).catchError((error) { + networkError = AppUtils.dioErrorTypeToString(error.type); + networkStatus = -1; + setState(() {}); + }); + if (baseData != null && baseData.isSuccess) { + productGroupList = baseData.data; + networkStatus = 1; + } + } + + ///分类列表 + queryCategorize() async { + if (businessService == null) { + businessService = BusinessApiService(Dio(), + context: context, + token: BusinessInstance.instance.businessToken, + tenant: BusinessInstance.instance.businessTenant, + storeId: widget.arguments["storeId"]); + } + BaseData> baseData = await businessService.findCategoryListByDepth().catchError((error) {}); + if (baseData != null && baseData.isSuccess) { + goodsCategoryList = baseData.data; + } + } + + ///新建商品 + addGoods() async { + try { + if (businessService == null) { + businessService = BusinessApiService(Dio(), + context: context, + token: BusinessInstance.instance.businessToken, + tenant: BusinessInstance.instance.businessTenant, + storeId: widget.arguments["storeId"]); + } + BaseData baseData = await businessService.productSave({ + "storeId": widget.arguments["storeId"], + "attrStyle": isAttrStyle == true ? 0:1,//款式类型:1-多款式 0-单款式 + "categoryId": "1343391656220557312",//分类id + "oversold": isOversold == true ? 1:0,//允许超卖:1-允许 0-不允许 + "setMeal": isSetMeal == true ? 0:1,//套餐专属商品 0是 1不是 + "groupId": "1693436138259218432",//分组id + "imgs": [ + { + "imgPath": "https://pos.upload.lotus-wallet.com/1195/2023/10/1ba87000-d940-49e5-a6ca-f15623d5f841.jpg" + } + ],//商品图片 + "needLogistics": isLogistics == true ? 1:0,//需要物流:1-需要 0-不需要 + "productName": "新建商品",//商品名字 + "productType": 0,//商品类型(0:普通类型;1:拼盘类型)app新增商品用不到,默认传0 + "shortName": "商品简介",//商品简介 + "skuAttrList": [ + { + "attrCode": "",//规格的数据编码,app新增商品用不到,但必须传"" + "attrName": "",//属性名称,app新增商品用不到,但必须传"" + "needImg": 0,//是否需要图片,app新增商品用不到,但必须传0 + "attrValueList": [ + {"attrValue": "商品规格",//商品规格 + "attrValueCode": "", //规格值的数据编码,app新增商品用不到,但必须传"" + }]//规格对应的规格值 + } + ],//规格列表 + "skuList": [ + { + "applyPrice": "234",//商品原价 + "skuAttrCodeDTOList": [{"attrCode": "", "attrValueCode": ""}],//sku对应的规格编码,app新增商品用不到,但必须传该段 + "skuPrice": "123",//商品售价 + "packagingFee": "789",//打包费 + "skuStock": "666",//库存 + "weight": "222"//重量 + }],//商品sku列表 + "status":isGround == true ? 1:0 //状态:1-上架 0-下架 + }).catchError((error) { + networkError = AppUtils.dioErrorTypeToString(error.type); + networkStatus = -1; + setState(() {}); + }); + if (!mounted) return; + if (baseData != null && baseData.isSuccess) { + SmartDialog.show( + widget: SettlementTips( + () {}, + text: "新建商品成功", + color: Color(0xFF30415B), + )); + networkStatus = 1; + setState(() {}); + } else { + SmartDialog.show( + widget: SettlementTips( + () {}, + text: baseData.msg, + color: Color(0xFF30415B), + )); + } + } finally {} + } + + @override + Widget build(BuildContext context) { + return GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + FocusScope.of(context).unfocus(); + }, + child: Scaffold( + resizeToAvoidBottomInset: false, + backgroundColor: Color(0xFFF8F8FA), + appBar: MyAppBar( + title: "新建商品", + titleColor: Colors.black, + background: Colors.white, + leadingColor: Colors.black, + brightness: Brightness.dark, + ), + body: SingleChildScrollView( + physics: BouncingScrollPhysics(), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Padding( + padding: EdgeInsets.only(top: 15.h, bottom: 15.h, left: 16.w), + child: Text( + "基本信息", + style: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + ), + ), + basicInformation(), + Container( + color: Colors.white, + margin: EdgeInsets.symmetric(vertical: 12.h), + padding: EdgeInsets.only(top: 16.h), + child: Column( + children: [ + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: (){ + officialCategory(); + }, + child:textSelectItem("官方分类",categoryName == "" ? "请选择":categoryName), + ), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: (){ + goodsGroup(); + }, + child:textSelectItem("商品分组",groupName == "" ? "请选择":groupName)), + ], + ), + ), + buttonSelect(), + Container( + color: Colors.white, + margin: EdgeInsets.only(top: 12.h), + child: Column( + children: [ + Padding(padding: EdgeInsets.only(top: 18.h,left: 16.w,right: 16.w), + child: Row( + children: [ + Padding( + padding: EdgeInsets.only(right: 20.h), + child: Text( + "多规格", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: FontWeight.bold, + ), + )), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap:(){ + setState((){ + isAttrStyle = !isAttrStyle; + }); + }, + child: Image.asset( + isAttrStyle == true ? "assets/image/reservation_switch.webp":"assets/image/reservation_unswitch.webp", + width: 44.w, + height: 24.h, + ), + ), + ], + )), + Container( + width: double.infinity, + height: 1.w, + color: Color(0x14000000), + margin: EdgeInsets.only(top: 16.h,left: 16.w,right: 16.w), + ), + isAttrStyle == true ? GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: (){ + Navigator.of(context).pushNamed( + '/router/set_goods_specs'); + }, + child:Container( + padding: EdgeInsets.only(left: 16.w,right: 16.w,top: 16.h), + child:Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.start, + children: [ + Expanded(child:Text( + "商品规格", + style: TextStyle( + color: Color(0xFF0D0D0D), + fontSize: 14.sp, + fontWeight: FontWeight.bold, + ), + )), + Text( + "请设置商品规格", + style: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: FontWeight.bold, + ), + ), + Image.asset( + "assets/image/bs_right.webp", + width: 16.h, + height: 16.h, + ), + ], + ), + Container( + margin: EdgeInsets.only(top: 16.h), + color: Color(0x14000000), + height: 1.h, + width: double.infinity, + ), + ], + ), + ), + ): textItem("商品规格", skuController, "请输入商品规格"), + if(isAttrStyle == false) + textItem("商品重量", heftController, "请输入商品重量"), + if(isAttrStyle == false) + textItem("商品售价", priceController, "请输入商品售卖价格"), + if(isAttrStyle == false) + textItem("商品原价", originalPriceController, "请输入商品划线价格"), + if(isAttrStyle == false) + textItem("打包费", packingChargeController, "请输入商品打包费用"), + if(isAttrStyle == false) + textItem("商品库存", stockController, "请输入商品库存"), + if(isAttrStyle == false) + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap:(){ + Navigator.of(context).pushNamed( + '/router/set_meal',arguments:{"storeId":widget.arguments["storeId"]}); + }, + child: Padding(padding: EdgeInsets.symmetric(horizontal: 16.w,vertical:16.h), + child: Column( + children: [ + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child:Text( + "设置套餐", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: FontWeight.bold, + ), + ), + ), + Text( + "套餐选择", + style: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: FontWeight.bold, + ), + ), + Image.asset( + "assets/image/bs_right.webp", + width: 16.h, + height: 16.h, + ), + ], + ), + Container( + margin:EdgeInsets.only(top: 16.h), + color: Color(0x14000000), + height: 1.h, + width: double.infinity, + ), + ], + ),), + ), + Container( + width: double.infinity, + padding: EdgeInsets.symmetric(vertical: 16.h), + margin: EdgeInsets.only(top: 111.h, bottom: 34.h,left:16.w,right:16.w), + decoration: BoxDecoration( + color: Color(0xFF30415B), + borderRadius: BorderRadius.circular(27), + ), + alignment: Alignment.center, + child: Text( + S + .of(context) + .baocun, + style: TextStyle( + color: Colors.white, + fontSize: 16.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + ], + ), + ) + ], + ), + ), + ), + ); + } + + ///基础信息1 + Widget basicInformation() { + return Container( + padding: EdgeInsets.symmetric(horizontal: 16.w), + color: Colors.white, + child: Column( + children: [ + mustTextItem("商品名称", goodsNameController, "请输入商品名称"), + Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text.rich( + TextSpan( + children: [ + TextSpan( + text: "*", + style: TextStyle( + color: Color(0xFFE02020), + fontSize: 14.sp, + fontWeight: FontWeight.w500, + ), + ), + TextSpan( + text: "商品图片", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ], + ), + ), + ), + Container( + padding: EdgeInsets.all(15), + margin: EdgeInsets.only(right: 4.w), + decoration: BoxDecoration( + color: Color(0xFFF0F0F0), + borderRadius: BorderRadius.circular(4), + ), + child: Icon( + Icons.add, + size: 24, + color: Color(0xFFD8D8D8), + ), + ), + Image.asset( + "assets/image/vip_code.webp", + fit: BoxFit.cover, + width: 54.h, + height: 54.h, + ), + ], + ), + Container( + margin: EdgeInsets.symmetric(vertical: 16.h), + color: Color(0x14000000), + height: 1.h, + width: double.infinity, + ), + ], + ), + Row( + crossAxisAlignment: CrossAxisAlignment.start, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Text.rich( + TextSpan( + children: [ + TextSpan( + text: "*", + style: TextStyle( + color: Color(0xFFE02020), + fontSize: 14.sp, + fontWeight: FontWeight.w500, + ), + ), + TextSpan( + text: "商品简介", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ], + ), + ), + Expanded( + child: Container( + padding: EdgeInsets.only(left: 8.w), + margin: EdgeInsets.only(left: 22.w, bottom: 16.h), + decoration: BoxDecoration( + color: Color(0xFFFCFCFC), + border: Border.all( + color: Color(0xFFEBEBEB), + width: 1, + ), + borderRadius: BorderRadius.circular(2), + ), + alignment: Alignment.topLeft, + child: TextField( + maxLines: 5, + controller: profileController, + onChanged: (value) { + setState(() { + // textLength = value.length; + }); + }, + decoration: InputDecoration( + border: InputBorder.none, + hintText: "请输入商品商品简介", + hintStyle: TextStyle( + fontSize: 12.sp, + fontWeight: MyFontWeight.bold, + color: Color(0xFF7A797F), + ), + ), + ), + )) + ], + ), + ], + ), + ); + } + + ///按钮选择板块 + Widget buttonSelect() { + return Container( + padding: EdgeInsets.only(left: 17.w, top: 16.h), + color: Colors.white, + child: Column( + children: [ + Row( + children: [ + Expanded( + child:Row( + children: [ + Padding( + padding: EdgeInsets.only(right: 20.h), + child: Text( + "需要物流", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: FontWeight.bold, + ), + )), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap:(){ + setState((){ + isLogistics = !isLogistics; + }); + }, + child: Image.asset( + isLogistics == true ? "assets/image/reservation_switch.webp":"assets/image/reservation_unswitch.webp", + width: 44.w, + height: 24.h, + ), + ), + ], + ), + ), + Expanded( + child:Row( + children: [ + Padding( + padding: EdgeInsets.only(right: 20.h), + child: Text( + "上架状态", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: FontWeight.bold, + ), + )), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap:(){ + setState((){ + isGround = !isGround; + }); + }, + child: Image.asset( + isGround == true ? "assets/image/reservation_switch.webp":"assets/image/reservation_unswitch.webp", + width: 44.w, + height: 24.h, + ), + ), + ], + ), + ), + ], + ), + Container( + width: double.infinity, + height: 1.w, + color: Color(0x14000000), + margin: EdgeInsets.symmetric(vertical: 16.h), + ), + Row( + children: [ + Expanded( + child:Row( + children: [ + Padding( + padding: EdgeInsets.only(right: 20.h), + child: Text( + "允许超卖", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: FontWeight.bold, + ), + )), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap:(){ + setState((){ + isOversold = !isOversold; + }); + }, + child: Image.asset( + isOversold == true ? "assets/image/reservation_switch.webp":"assets/image/reservation_unswitch.webp", + width: 44.w, + height: 24.h, + ), + ), + ], + ), + ), Expanded( + child: Row( + children: [ + Padding( + padding: EdgeInsets.only(right: 20.h), + child: Text( + "套餐商品", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: FontWeight.bold, + ), + )), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap:(){ + setState((){ + isSetMeal = !isSetMeal; + }); + }, + child: Image.asset( + isSetMeal == true ? "assets/image/reservation_switch.webp":"assets/image/reservation_unswitch.webp", + width: 44.w, + height: 24.h, + ), + ), + ], + ), + ) + ], + ), + Container( + width: double.infinity, + height: 1.w, + color: Color(0x14000000), + margin: EdgeInsets.symmetric(vertical: 16.h), + ), + ], + ), + ); + } + + Widget textSelectItem(left,selectName) { + return Container( + padding: EdgeInsets.symmetric(horizontal: 16.w), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text.rich( + TextSpan( + children: [ + TextSpan( + text: "*", + style: TextStyle( + color: Color(0xFFE02020), + fontSize: 14.sp, + fontWeight: FontWeight.w500, + ), + ), + TextSpan( + text: left, + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ], + ), + ), + ), + Text( + selectName, + style: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: FontWeight.bold, + ), + ), + Image.asset( + "assets/image/bs_right.webp", + width: 16.h, + height: 16.h, + ), + ], + ), + Container( + margin: EdgeInsets.symmetric(vertical: 16.h), + color: Color(0x14000000), + height: 1.h, + width: double.infinity, + ), + ], + )); + } + + Widget mustTextItem(left, rightController, right) { + return Container( + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text.rich( + TextSpan( + children: [ + TextSpan( + text: "*", + style: TextStyle( + color: Color(0xFFE02020), + fontSize: 14.sp, + fontWeight: FontWeight.w500, + ), + ), + TextSpan( + text: left, + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ], + ), + ), + ), + Expanded( + child: TextField( + controller: rightController, + decoration: InputDecoration( + hintText: right ?? "", + hintTextDirection: TextDirection.rtl, + hintStyle: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + border: InputBorder.none, + ), + textAlign: TextAlign.right, + style: TextStyle( + color: Color(0xFF000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold), + ), + flex: 2, + ), + ], + ), + Container( + margin: EdgeInsets.only(bottom: 16.h), + color: Color(0x14000000), + height: 1.h, + width: double.infinity, + ), + ], + )); + } + + Widget textItem(left, rightController, right) { + return Container( + padding: EdgeInsets.symmetric(horizontal: 16.w), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text( + left, + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + Expanded( + child: TextField( + controller: rightController, + decoration: InputDecoration( + hintText: right ?? "", + hintTextDirection: TextDirection.rtl, + hintStyle: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + border: InputBorder.none, + ), + textAlign: TextAlign.right, + style: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold), + ), + flex: 2, + ), + ], + ), + Container( + color: Color(0x14000000), + height: 1.h, + width: double.infinity, + ), + ], + )); + } + + ///官方分类弹窗选择 + officialCategory() { + showModalBottomSheet( + context: context, + backgroundColor: Colors.transparent, + builder: (context) { + return StatefulBuilder(builder: ( + context, + state, + ) { + return Container( + width: double.infinity, + height: 365.h, + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.only( + topLeft: Radius.circular(8), + topRight: Radius.circular(8), + ), + ), + child: Column( + mainAxisAlignment: MainAxisAlignment.start, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Row( + mainAxisAlignment: MainAxisAlignment.start, + crossAxisAlignment: CrossAxisAlignment.center, + children: [ + Expanded( + child: Container( + alignment: Alignment.center, + margin: EdgeInsets.only( + top: 12.h, bottom: 12.h, left: 41.w), + child: Text( + "选择官方分类", + style: TextStyle( + fontWeight: MyFontWeight.bold, + fontSize: 16.sp, + color: Color(0xFF1A1A1A), + ), + ), + )), + GestureDetector( + onTap: () { + state(() { + Navigator.of(context).pop(); + }); + }, + child: Padding( + padding: EdgeInsets.only(right: 16.w), + child: Image.asset( + "assets/image/cancel.webp", + width: 25.h, + height: 25.h, + ), + ), + ), + ], + ), + Expanded(child:ListView.builder( + padding: EdgeInsets.zero, + itemCount: goodsCategoryList?.length ??0, + scrollDirection: Axis.vertical, + shrinkWrap: true, + physics: BouncingScrollPhysics(), + itemBuilder: (context, position) { + return GestureDetector( + onTap: () { + state(() { + categoryIndex = position; + }); + }, + child: categoryItem(goodsCategoryList[position],position), + ); + }, + )), + GestureDetector( + onTap: () { + setState(() { + categoryName = goodsCategoryList[categoryIndex].categoryName; + Navigator.of(context).pop(); + }); + }, + child: Container( + width: double.infinity, + alignment: Alignment.center, + margin: EdgeInsets.symmetric(vertical:25.h,horizontal:16.w), + padding: EdgeInsets.symmetric(vertical: 16.h), + decoration: BoxDecoration( + borderRadius: BorderRadius.circular(27), + color: Color(0xFF30415B)), + child: Text( + S.of(context).queding, + style: TextStyle( + fontWeight: MyFontWeight.semi_bold, + fontSize: 16.sp, + color: Colors.white, + ), + ), + ), + ) + ], + ), + ); + }); + }); + } + + Widget categoryItem(GoodsCategoryList goodsCategoryList,index){ + return Container( + height: 52.h, + margin: EdgeInsets.only(bottom: 12,left:16.w,right: 16.w), + child: Stack( + alignment: Alignment.bottomRight, + children: [ + Container( + height: 52.h, + width: double.infinity, + decoration: BoxDecoration( + color: categoryIndex == index + ? Color(0xFFEFF5FF) + : Color(0xFFF7F7F7), + borderRadius: BorderRadius.circular(4.w), + border: Border.all( + color: categoryIndex == index + ? Color(0xFF30415B) + : Colors.white, + width: categoryIndex == index ? 1.w : 0, + ), + ), + padding: EdgeInsets.only( + top: 16.h, + bottom: 16.h, + left: 16.w, + right: 17.w), + child: Text( + goodsCategoryList?.categoryName ?? "", + style: TextStyle( + fontSize: 14.sp, + fontWeight: MyFontWeight.medium, + color: categoryIndex == index + ? Color(0xFF30415B) + : Color(0xFF0D0D0D)), + ), + ), + if (categoryIndex == index) + Image.asset( + "assets/image/bs_shop.webp", + width: 20, + height: 20, + fit: BoxFit.fill, + ), + ], + ), + ); + } + + ///商品分组弹窗选择 + goodsGroup() { + showModalBottomSheet( + context: context, + backgroundColor: Colors.transparent, + builder: (context) { + return StatefulBuilder(builder: ( + context, + state, + ) { + return Container( + width: double.infinity, + height: 365.h, + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.only( + topLeft: Radius.circular(8), + topRight: Radius.circular(8), + ), + ), + child: Column( + mainAxisAlignment: MainAxisAlignment.start, + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Row( + mainAxisAlignment: MainAxisAlignment.start, + crossAxisAlignment: CrossAxisAlignment.center, + children: [ + Expanded( + child: Container( + alignment: Alignment.center, + margin: EdgeInsets.only( + top: 12.h, bottom: 12.h, left: 41.w), + child: Text( + "选择官方分类", + style: TextStyle( + fontWeight: MyFontWeight.bold, + fontSize: 16.sp, + color: Color(0xFF1A1A1A), + ), + ), + )), + GestureDetector( + onTap: () { + state(() { + Navigator.of(context).pop(); + }); + }, + child: Padding( + padding: EdgeInsets.only(right: 16.w), + child: Image.asset( + "assets/image/cancel.webp", + width: 25.h, + height: 25.h, + ), + ), + ), + ], + ), + Expanded(child:ListView.builder( + padding: EdgeInsets.zero, + itemCount: productGroupList?.records?.length ?? 0, + scrollDirection: Axis.vertical, + shrinkWrap: true, + physics: BouncingScrollPhysics(), + itemBuilder: (context, position) { + return GestureDetector( + onTap: () { + state(() { + groupIndex = position; + }); + }, + child: groupItem(productGroupList.records[position],position), + ); + }, + )), + GestureDetector( + onTap: () { + setState(() { + groupName = productGroupList?.records[groupIndex]?.groupName ?? ""; + Navigator.of(context).pop(); + }); + }, + child: Container( + width: double.infinity, + alignment: Alignment.center, + margin: EdgeInsets.symmetric(vertical:25.h,horizontal:16.w), + padding: EdgeInsets.symmetric(vertical: 16.h), + decoration: BoxDecoration( + borderRadius: BorderRadius.circular(27), + color: Color(0xFF30415B)), + child: Text( + S.of(context).queding, + style: TextStyle( + fontWeight: MyFontWeight.semi_bold, + fontSize: 16.sp, + color: Colors.white, + ), + ), + ), + ) + ], + ), + ); + }); + }); + } + + Widget groupItem(Records records,index){ + return Container( + height: 52.h, + margin: EdgeInsets.only(bottom: 12,left:16.w,right: 16.w), + child: Stack( + alignment: Alignment.bottomRight, + children: [ + Container( + height: 52.h, + width: double.infinity, + decoration: BoxDecoration( + color: groupIndex == index + ? Color(0xFFEFF5FF) + : Color(0xFFF7F7F7), + borderRadius: BorderRadius.circular(4.w), + border: Border.all( + color: groupIndex == index + ? Color(0xFF30415B) + : Colors.white, + width: groupIndex == index ? 1.w : 0, + ), + ), + padding: EdgeInsets.only( + top: 16.h, + bottom: 16.h, + left: 16.w, + right: 17.w), + child: Text( + records?.groupName ?? "", + style: TextStyle( + fontSize: 14.sp, + fontWeight: MyFontWeight.medium, + color: categoryIndex == index + ? Color(0xFF30415B) + : Color(0xFF0D0D0D)), + ), + ), + if (groupIndex == index) + Image.asset( + "assets/image/bs_shop.webp", + width: 20, + height: 20, + fit: BoxFit.fill, + ), + ], + ), + ); + } +} diff --git a/lib/business_system/goods/add_goods/edit_specs_detail.dart b/lib/business_system/goods/add_goods/edit_specs_detail.dart new file mode 100644 index 00000000..8941c6ad --- /dev/null +++ b/lib/business_system/goods/add_goods/edit_specs_detail.dart @@ -0,0 +1,440 @@ +import 'package:flutter/material.dart'; +import 'package:flutter_smart_dialog/flutter_smart_dialog.dart'; +import 'package:huixiang/view_widget/my_appbar.dart'; +import 'package:flutter_screenutil/flutter_screenutil.dart'; +import '../../../generated/l10n.dart'; +import '../../../retrofit/business_api.dart'; +import '../../../retrofit/data/edit_specs_detail_list.dart'; +import '../../../utils/font_weight.dart'; +import '../../../view_widget/settlement_tips_dialog.dart'; + +class EditSpecsDetail extends StatefulWidget { + final Map arguments; + + EditSpecsDetail({this.arguments}); + + @override + State createState() { + return _EditSpecsDetail(); + } +} + +class _EditSpecsDetail extends State { + List editSpecsDetailList = []; + bool isKeyBoardShow = false; + FocusNode _focusNode = FocusNode(); + BusinessApiService businessService; + + @override + void initState() { + super.initState(); + WidgetsBinding.instance.addPostFrameCallback((_) { + setState(() { + print("object: ${MediaQuery.of(context).viewInsets.bottom}"); + if (MediaQuery.of(context).viewInsets.bottom == 0) { + if (isKeyBoardShow) { + isKeyBoardShow = false; + //关闭键盘 软键盘关闭了, 清除输入控件的焦点, 否则重新进入页面会导致软键盘再弹出问题 + FocusScope.of(context).requestFocus(FocusNode()); + } + } else { + isKeyBoardShow = true; + } + }); + }); + List skus = widget?.arguments["skus"] ?? []; + List> specsDetails = + widget.arguments["specsDetails"] ?? []; + skus.forEach((element) { + var edit = EditSpecsDetailList(element); + for(var e1 in specsDetails){ + if(e1["skuName"] == element){ + edit.goodPriceController.text = e1["goodPrice"]; + edit.originalPriceController.text = e1["originalPrice"]; + edit.packagingFeeController.text = e1["packagingFee"]; + edit.specsWeightController.text = e1["specsWeight"]; + edit.skuStockController.text = e1["skuStock"]; + break; + } + } + editSpecsDetailList.add(edit); + }); + } + + ///离开页面记着销毁和清除 + @override + void dispose() { + _focusNode.unfocus(); + super.dispose(); + } + + @override + Widget build(BuildContext context) { + return Scaffold( + resizeToAvoidBottomInset: false, + appBar: MyAppBar( + title: "编辑规格明细", + titleColor: Colors.black, + background: Colors.white, + leadingColor: Colors.black, + brightness: Brightness.dark, + ), + body: Stack( + children: [ + Container( + margin: EdgeInsets.only(bottom: 120.h), + child: ListView.builder( + padding: EdgeInsets.zero, + itemCount: editSpecsDetailList.length, + scrollDirection: Axis.vertical, + shrinkWrap: true, + physics: BouncingScrollPhysics(), + itemBuilder: (context, position) { + return GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + setState(() { + FocusScope.of(context).unfocus(); + }); + }, + child: editSpecsItem(position)); + }, + ), + ), + + ///确认 + Align( + alignment: Alignment.bottomCenter, + child: GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + bool flag = false; + String editType = ""; + List> specsDetail = []; + editSpecsDetailList.forEach((element) { + if (element.goodPriceController.text.trim() == "") { + flag = true; + editType = "请输入商品售价"; + return; + } else if (element.originalPriceController.text.trim() == + "") { + flag = true; + editType = "请输入商品原价"; + return; + } + specsDetail.add({ + "skuName": element.specsDetailName, + "goodPrice": element.goodPriceController.text, + "originalPrice": element.originalPriceController.text, + "packagingFee": element.packagingFeeController.text, + "specsWeight": element.specsWeightController.text, + "skuStock": element.skuStockController.text + }); + }); + if (flag) { + SmartDialog.show( + clickBgDismissTemp: false, + widget: SettlementTips( + () {}, + text: editType, + color: Color(0xFF30415B), + )); + } else { + Navigator.of(context).pop(specsDetail); + } + }, + child: Container( + color: Colors.white, + width: double.infinity, + padding: EdgeInsets.only( + top: 10.h, left: 16.w, right: 16.w, bottom: 34.h), + child: Container( + alignment: Alignment.center, + height: 54.h, + decoration: BoxDecoration( + color: Color(0xFF30415B), + borderRadius: BorderRadius.circular(27), + ), + child: Text( + S.of(context).baocun, + style: TextStyle( + color: Colors.white, + fontSize: 16.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + ), + )) + ], + ), + ); + } + + ///编辑规格list + Widget editSpecsItem(index) { + return Container( + color: Colors.white, + margin: EdgeInsets.symmetric(vertical: 12.h), + padding: EdgeInsets.symmetric(horizontal: 16.w, vertical: 20.h), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text( + editSpecsDetailList[index].specsDetailName ?? "", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + Container( + width: double.infinity, + height: 1.h, + margin: EdgeInsets.only(top: 20.h), + color: Color(0x14000000), + ), + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text.rich( + TextSpan( + children: [ + TextSpan( + text: "*", + style: TextStyle( + color: Color(0xFFE02020), + fontSize: 14.sp, + fontWeight: FontWeight.w500, + ), + ), + TextSpan( + text: "商品售价", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + ), + ], + ), + ), + ), + Expanded( + child: TextField( + controller: editSpecsDetailList[index].goodPriceController, + decoration: InputDecoration( + hintText: "请输入商品售卖价格", + hintTextDirection: TextDirection.rtl, + hintStyle: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + border: InputBorder.none, + ), + textAlign: TextAlign.right, + style: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + ), + flex: 2, + ), + ], + ), + Container( + width: double.infinity, + height: 1.h, + color: Color(0x14000000), + ), + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text.rich( + TextSpan( + children: [ + TextSpan( + text: "*", + style: TextStyle( + color: Color(0xFFE02020), + fontSize: 14.sp, + fontWeight: FontWeight.w500, + ), + ), + TextSpan( + text: "商品原价", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + ), + ], + ), + ), + ), + Expanded( + child: TextField( + controller: + editSpecsDetailList[index].originalPriceController, + decoration: InputDecoration( + hintText: "请输入商品划线价格", + hintTextDirection: TextDirection.rtl, + hintStyle: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + border: InputBorder.none, + ), + textAlign: TextAlign.right, + style: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + ), + flex: 2, + ), + ], + ), + Container( + width: double.infinity, + height: 1.h, + color: Color(0x14000000), + ), + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text( + "商品打包费", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + ), + ), + Expanded( + child: TextField( + controller: editSpecsDetailList[index].packagingFeeController, + decoration: InputDecoration( + hintText: "请输入商品打包费", + hintTextDirection: TextDirection.rtl, + hintStyle: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + border: InputBorder.none, + ), + textAlign: TextAlign.right, + style: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + ), + flex: 2, + ), + ], + ), + Container( + width: double.infinity, + height: 1.h, + color: Color(0x14000000), + ), + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text( + "商品重量", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + ), + ), + Expanded( + child: TextField( + controller: editSpecsDetailList[index].specsWeightController, + decoration: InputDecoration( + hintText: "请输入商品重量(单位kg)", + hintTextDirection: TextDirection.rtl, + hintStyle: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + border: InputBorder.none, + ), + textAlign: TextAlign.right, + style: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + ), + flex: 2, + ), + ], + ), + Container( + width: double.infinity, + height: 1.h, + color: Color(0x14000000), + ), + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text( + "商品库存", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + ), + ), + Expanded( + child: TextField( + controller: editSpecsDetailList[index].skuStockController, + decoration: InputDecoration( + hintText: "请输入商品库存", + hintTextDirection: TextDirection.rtl, + hintStyle: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + border: InputBorder.none, + ), + textAlign: TextAlign.right, + style: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + ), + flex: 2, + ), + ], + ), + Container( + width: double.infinity, + height: 1.h, + color: Color(0x14000000), + ), + ], + ), + ); + } +} diff --git a/lib/business_system/goods/add_goods/set_goods_specs.dart b/lib/business_system/goods/add_goods/set_goods_specs.dart new file mode 100644 index 00000000..7c7611d4 --- /dev/null +++ b/lib/business_system/goods/add_goods/set_goods_specs.dart @@ -0,0 +1,444 @@ +import 'package:flutter/material.dart'; +import 'package:flutter_smart_dialog/flutter_smart_dialog.dart'; +import 'package:huixiang/view_widget/my_appbar.dart'; +import 'package:flutter_screenutil/flutter_screenutil.dart'; +import '../../../retrofit/business_api.dart'; +import '../../../retrofit/data/edit_specs_detail_list.dart'; +import '../../../retrofit/data/set_specs_list.dart'; +import '../../../utils/font_weight.dart'; +import '../../../view_widget/settlement_tips_dialog.dart'; + +class SetGoodsSpecs extends StatefulWidget { + final Map arguments; + + SetGoodsSpecs({this.arguments}); + + @override + State createState() { + return _SetGoodsSpecs(); + } +} + +class _SetGoodsSpecs extends State { + List specs = []; + List> specsDetails = []; + bool isKeyBoardShow = false; + FocusNode _focusNode = FocusNode(); + BusinessApiService businessService; + + @override + void initState() { + super.initState(); + WidgetsBinding.instance.addPostFrameCallback((_) { + setState(() { + print("object: ${MediaQuery.of(context).viewInsets.bottom}"); + if (MediaQuery.of(context).viewInsets.bottom == 0) { + if (isKeyBoardShow) { + isKeyBoardShow = false; + //关闭键盘 软键盘关闭了, 清除输入控件的焦点, 否则重新进入页面会导致软键盘再弹出问题 + FocusScope.of(context).requestFocus(FocusNode()); + } + } else { + isKeyBoardShow = true; + } + }); + }); + } + + ///离开页面记着销毁和清除 + @override + void dispose() { + _focusNode.unfocus(); + super.dispose(); + } + + ///sku计算 + List mergeArr(List> arr) { + var result = arr.removeAt(0); + while (arr.isNotEmpty) { + var curArr = arr.removeAt(0); + var lastArr = result; + result = []; + for (var lastVal in lastArr) { + for (var curVal in curArr) { + result.add("$lastVal $curVal"); + } + } + } + return result; + } + + @override + Widget build(BuildContext context) { + return GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + FocusScope.of(context).unfocus(); + }, + child: Scaffold( + resizeToAvoidBottomInset: false, + appBar: MyAppBar( + title: "设置商品规格", + titleColor: Colors.black, + background: Colors.white, + leadingColor: Colors.black, + brightness: Brightness.dark, + ), + body: Stack( + children: [ + SingleChildScrollView( + physics: BouncingScrollPhysics(), + child: Container( + margin: EdgeInsets.only(bottom: 150.h), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + ListView.builder( + padding: EdgeInsets.zero, + itemCount: specs.length, + scrollDirection: Axis.vertical, + shrinkWrap: true, + physics: BouncingScrollPhysics(), + itemBuilder: (context, position) { + return GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + FocusScope.of(context).unfocus(); + }, + child: specsText(position)); + }, + ), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + setState(() { + specs.add(SetSpecsList()); + }); + }, + child: Container( + color: Colors.white, + padding: EdgeInsets.symmetric(vertical: 20.h), + child: Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Icon( + Icons.add_circle, + color: Color(0xFF30415B), + size: 20, + ), + Padding( + padding: EdgeInsets.only(left: 7.w), + child: Text( + "添加规格", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + ], + ), + ), + ), + if (specs.length > 0) + Padding( + padding: EdgeInsets.all(16), + child: Text( + "设置库存/价格等规格明细", + style: TextStyle( + color: Color(0xFF7A797F), + fontSize: 12.sp, + fontWeight: MyFontWeight.bold, + ), + )), + if (specs.length > 0) + Container( + color: Colors.white, + padding: EdgeInsets.only( + top: 20.h, + bottom: 20.h, + left: 16.w, + right: 20.h), + child: Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text( + "规格明细", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + )), + Expanded( + child: GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + bool flag = false; + String tipText = ""; + specs.forEach((element) { + if (element.specsNameController.text + .trim() == + "") { + flag = true; + tipText = "规格名称未输入,请先输入规格名称"; + return; + } else if (element + .specsValues.length == + 0) { + flag = true; + tipText = "未添加规格值,请先添加规格值"; + return; + } + }); + if (flag) { + SmartDialog.show( + clickBgDismissTemp: false, + widget: SettlementTips( + () {}, + text: tipText, + color: Color(0xFF30415B), + )); + } else { + List skus = mergeArr(specs + .map((e) => e.specsValues) + .toList()); + Navigator.of(context).pushNamed( + '/router/edit_specs_detail', + arguments: { + "skus": skus, + "specsDetails": specsDetails + }).then((value) { + if (value != null){ + specsDetails.clear(); + specsDetails.addAll(value); + setState((){}); + } + }); + } + }, + child: Row( + mainAxisAlignment: + MainAxisAlignment.end, + children: [ + Padding( + padding: + EdgeInsets.only(right: 5.w), + child: Text( + "去设置", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + Image.asset( + "assets/image/bs_right.webp", + width: 16.h, + height: 16.h, + ), + ], + ))) + ], + ), + ), + ], + ), + )), + + ///确认 + Align( + alignment: Alignment.bottomCenter, + child: GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () {}, + child: Container( + color: Colors.white, + width: double.infinity, + padding: EdgeInsets.only( + top: 10.h, left: 16.w, right: 16.w, bottom: 34.h), + child: Container( + alignment: Alignment.center, + height: 54.h, + decoration: BoxDecoration( + color: Color(0xFF30415B), + borderRadius: BorderRadius.circular(27), + ), + child: Text( + "确认", + style: TextStyle( + color: Colors.white, + fontSize: 16.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + ), + )) + ], + ), + )); + } + + ///添加规格样式 + Widget specsText(index) { + return Container( + color: Colors.white, + margin: EdgeInsets.symmetric(vertical: 12.h), + padding: EdgeInsets.symmetric(horizontal: 16.w), + child: Column( + children: [ + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text( + "添加规格", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + Expanded( + child: TextField( + controller: specs[index].specsNameController, + decoration: InputDecoration( + hintText: "请输入规格名称", + hintTextDirection: TextDirection.rtl, + hintStyle: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + border: InputBorder.none, + ), + textAlign: TextAlign.right, + style: TextStyle( + color: Color(0xFF0D0D0D), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold), + ), + flex: 2, + ), + ], + ), + Container( + width: double.infinity, + height: 1.h, + color: Color(0x14000000), + ), + Padding( + padding: EdgeInsets.symmetric(vertical: 16.h), + child: Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + flex: 1, + child: Text( + "规格植", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + Expanded( + child: GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + Navigator.of(context).pushNamed( + '/router/set_goods_specs_value', + arguments: { + "storeId": "", + "specsValues": specs[index].specsValues + }).then((value) { + if (value != null) + setState(() { + specs[index].specsValues = value; + }); + }); + }, + child: Row( + mainAxisAlignment: MainAxisAlignment.end, + crossAxisAlignment: CrossAxisAlignment.center, + children: [ + Expanded( + child: Text( + specs[index].specsValues.length != 0 + ? specs[index] + .specsValues + .map((e) => e) + .toString() + .replaceAll("(", "") + .replaceAll(")", "") + : "请添加规格值", + overflow: TextOverflow.ellipsis, + maxLines: 1, + textAlign: TextAlign.end, + style: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + )), + Image.asset( + "assets/image/bs_right.webp", + width: 16.h, + height: 16.h, + ), + ], + ), + ), + flex: 2, + ), + ], + )), + Container( + width: double.infinity, + height: 1.h, + color: Color(0x14000000), + margin: EdgeInsets.only(bottom: 18.h), + ), + GestureDetector( + onTap: () { + setState(() { + specs.removeAt(index); + }); + }, + child: Padding( + padding: EdgeInsets.only(bottom: 20.h), + child: Row( + mainAxisAlignment: MainAxisAlignment.end, + children: [ + Icon( + Icons.remove_circle, + color: Color(0xFFFA5151), + size: 24, + ), + Padding( + padding: EdgeInsets.only(left: 7.w), + child: Text( + "删除", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + ], + )), + ) + ], + ), + ); + } +} diff --git a/lib/business_system/goods/add_goods/set_goods_specs_value.dart b/lib/business_system/goods/add_goods/set_goods_specs_value.dart new file mode 100644 index 00000000..de56fb95 --- /dev/null +++ b/lib/business_system/goods/add_goods/set_goods_specs_value.dart @@ -0,0 +1,252 @@ +import 'package:flutter/material.dart'; +import 'package:flutter_smart_dialog/flutter_smart_dialog.dart'; +import 'package:huixiang/view_widget/my_appbar.dart'; +import 'package:flutter_screenutil/flutter_screenutil.dart'; +import '../../../retrofit/business_api.dart'; +import '../../../retrofit/data/set_specs_list.dart'; +import '../../../retrofit/data/set_specs_value_list.dart'; +import '../../../utils/font_weight.dart'; +import '../../../view_widget/settlement_tips_dialog.dart'; + +class SetGoodsSpecsValue extends StatefulWidget { + final Map arguments; + + SetGoodsSpecsValue({this.arguments}); + + @override + State createState() { + return _SetGoodsSpecsValue(); + } +} + +class _SetGoodsSpecsValue extends State { + List specsValue = []; + bool isKeyBoardShow = false; + FocusNode _focusNode = FocusNode(); + BusinessApiService businessService; + + @override + void initState() { + super.initState(); + WidgetsBinding.instance.addPostFrameCallback((_) { + setState(() { + print("object: ${MediaQuery.of(context).viewInsets.bottom}"); + if (MediaQuery.of(context).viewInsets.bottom == 0) { + if (isKeyBoardShow) { + isKeyBoardShow = false; + //关闭键盘 软键盘关闭了, 清除输入控件的焦点, 否则重新进入页面会导致软键盘再弹出问题 + FocusScope.of(context).requestFocus(FocusNode()); + } + } else { + isKeyBoardShow = true; + } + }); + }); + List specsValues = widget.arguments["specsValues"] ?? []; + specsValues.forEach((element) { + var temp = SetSpecsValueList(); + temp.specsValueNameController.text = element; + specsValue.add(temp); + }); + } + + ///离开页面记着销毁和清除 + @override + void dispose() { + _focusNode.unfocus(); + super.dispose(); + } + + @override + Widget build(BuildContext context) { + return GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + FocusScope.of(context).unfocus(); + }, + child: Scaffold( + resizeToAvoidBottomInset: false, + appBar: MyAppBar( + title: "设置商品规格值", + titleColor: Colors.black, + background: Colors.white, + leadingColor: Colors.black, + brightness: Brightness.dark, + ), + body: Stack( + children: [ + SingleChildScrollView( + physics: BouncingScrollPhysics(), + child: Container( + margin: EdgeInsets.only(bottom: 150.h), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Container( + color: Colors.white, + child: ListView.builder( + padding: EdgeInsets.zero, + itemCount: specsValue.length, + scrollDirection: Axis.vertical, + shrinkWrap: true, + physics: BouncingScrollPhysics(), + itemBuilder: (context, position) { + return GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + setState(() {}); + }, + child: specsValueText(position)); + }, + )), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + setState(() { + specsValue.add(SetSpecsValueList()); + FocusScope.of(context).unfocus(); + }); + }, + child: Container( + color: Colors.white, + padding: EdgeInsets.symmetric(vertical: 20.h), + margin: EdgeInsets.only(top: 12.h), + child: Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Icon( + Icons.add_circle, + color: Color(0xFF30415B), + size: 20, + ), + Padding( + padding: EdgeInsets.only(left: 7.w), + child: Text( + "添加规格值", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + ], + ), + ), + ), + ], + ), + )), + + ///确认 + Align( + alignment: Alignment.bottomCenter, + child: GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + bool flag = false; + List values = []; + specsValue.forEach((element) { + if (element.specsValueNameController.text == "") { + flag = true; + return; + } + values.add(element.specsValueNameController.text); + }); + if (flag) { + SmartDialog.show( + clickBgDismissTemp: false, + widget: SettlementTips( + () {}, + text: "还有规格未输入值,请输入完整", + color: Color(0xFF30415B), + )); + } else { + Navigator.of(context).pop(values); + } + }, + child: Container( + color: Colors.white, + width: double.infinity, + padding: EdgeInsets.only( + top: 10.h, left: 16.w, right: 16.w, bottom: 34.h), + child: Container( + alignment: Alignment.center, + height: 54.h, + decoration: BoxDecoration( + color: Color(0xFF30415B), + borderRadius: BorderRadius.circular(27), + ), + child: Text( + "确认", + style: TextStyle( + color: Colors.white, + fontSize: 16.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + ), + )) + ], + ), + )); + } + + ///添加规格样式 + Widget specsValueText(index) { + return Container( + padding: EdgeInsets.symmetric(horizontal: 16.w), + child: Column( + children: [ + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + GestureDetector( + onTap: () { + setState(() { + specsValue.removeAt(index); + }); + }, + child: Container( + padding: EdgeInsets.only(right: 12.w), + child: Icon( + Icons.remove_circle, + color: Color(0xFFFA5151), + size: 24, + ), + ), + ), + Expanded( + child: TextField( + controller: specsValue[index].specsValueNameController, + decoration: InputDecoration( + hintText: "请输入规格值名称", + hintStyle: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + border: InputBorder.none, + ), + style: TextStyle( + color: Color(0xFF0D0D0D), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold), + ), + flex: 2, + ), + ], + ), + if (specsValue.length - 1 > index) + Container( + width: double.infinity, + height: 1.h, + color: Color(0x14000000), + ), + ], + ), + ); + } +} diff --git a/lib/business_system/goods/add_goods/set_meal.dart b/lib/business_system/goods/add_goods/set_meal.dart new file mode 100644 index 00000000..4aa5b01d --- /dev/null +++ b/lib/business_system/goods/add_goods/set_meal.dart @@ -0,0 +1,491 @@ +import 'package:flutter/material.dart'; +import 'package:huixiang/view_widget/my_appbar.dart'; +import 'package:flutter_screenutil/flutter_screenutil.dart'; +import '../../../retrofit/business_api.dart'; +import '../../../retrofit/data/set_specs_meal_list.dart'; +import '../../../utils/font_weight.dart'; + +class SetMeal extends StatefulWidget { + final Map arguments; + + SetMeal({this.arguments}); + + @override + State createState() { + return _SetMeal(); + } +} + +class _SetMeal extends State { + List specsMeal = []; + bool isKeyBoardShow = false; + FocusNode _focusNode = FocusNode(); + BusinessApiService businessService; + + @override + void initState() { + super.initState(); + WidgetsBinding.instance.addPostFrameCallback((_) { + setState(() { + print("object: ${MediaQuery.of(context).viewInsets.bottom}"); + if (MediaQuery.of(context).viewInsets.bottom == 0) { + if (isKeyBoardShow) { + isKeyBoardShow = false; + //关闭键盘 软键盘关闭了, 清除输入控件的焦点, 否则重新进入页面会导致软键盘再弹出问题 + FocusScope.of(context).requestFocus(FocusNode()); + } + } else { + isKeyBoardShow = true; + } + }); + }); + } + + ///离开页面记着销毁和清除 + @override + void dispose() { + _focusNode.unfocus(); + super.dispose(); + } + + @override + Widget build(BuildContext context) { + return GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + FocusScope.of(context).unfocus(); + }, + child: Scaffold( + resizeToAvoidBottomInset: false, + appBar: MyAppBar( + title: "套餐选择", + titleColor: Colors.black, + background: Colors.white, + leadingColor: Colors.black, + brightness: Brightness.dark, + action: GestureDetector( + onTap: () { + }, + child: Text( + "确定", + style: TextStyle( + color: Color(0xFF30415B), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + ), + body: Stack( + children: [ + SingleChildScrollView( + physics: BouncingScrollPhysics(), + child: Container( + margin: EdgeInsets.only(bottom: 150.h), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + ListView.builder( + padding: EdgeInsets.zero, + itemCount: specsMeal.length, + scrollDirection: Axis.vertical, + shrinkWrap: true, + physics: BouncingScrollPhysics(), + itemBuilder: (context, position) { + return GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + setState(() {}); + }, + child: specsValueText(position)); + }, + ), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + setState(() { + specsMeal.add(SetSpecsMealList()); + }); + }, + child: Container( + color: Colors.white, + padding: EdgeInsets.symmetric(vertical: 20.h), + child: Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Icon( + Icons.add_circle, + color: Color(0xFF30415B), + size: 20, + ), + Padding( + padding: EdgeInsets.only(left: 7.w), + child: Text( + "添加分组", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + ], + ), + ), + ), + ], + ), + )), + ], + ), + )); + } + + ///添加规格样式 + Widget specsValueText(index) { + return Column( + children: [ + Container( + color: Colors.white, + margin: EdgeInsets.symmetric(vertical: 12.h), + padding: EdgeInsets.only(left: 16.w, right: 16.w, bottom: 12.h), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text( + "分组1", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + Expanded( + child: TextField( + controller: specsMeal[index].groupsNameController, + decoration: InputDecoration( + hintText: "请输入分组名称", + hintTextDirection: TextDirection.rtl, + hintStyle: TextStyle( + color: Color(0xFF7A797F), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + border: InputBorder.none, + ), + textAlign: TextAlign.right, + style: TextStyle( + color: Color(0xFF0D0D0D), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold), + ), + flex: 2, + ), + ], + ), + Container( + width: double.infinity, + height: 1.h, + color: Color(0x14000000), + ), + Padding( + padding: EdgeInsets.symmetric(vertical: 14.h), + child: Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text( + "分组总数", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + setState(() { + if (specsMeal[index].groupsTotal <= 1) return; + specsMeal[index].groupsTotal = + specsMeal[index].groupsTotal - 1; + }); + }, + child: Container( + padding: EdgeInsets.symmetric(horizontal: 20.w), + child: Icon( + Icons.remove_circle_outline, + color: Color(0xFF30415B), + size: 24, + ))), + Text( + specsMeal[index].groupsTotal.toString(), + style: TextStyle( + color: Color(0xD9000000), + fontSize: 20.sp, + fontWeight: MyFontWeight.regular, + ), + ), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + setState(() { + specsMeal[index].groupsTotal = + specsMeal[index].groupsTotal + 1; + }); + }, + child: Container( + padding: EdgeInsets.only(left: 20.w), + child: Icon( + Icons.add_circle, + color: Color(0xFF30415B), + size: 24, + )), + ), + ], + )), + Container( + width: double.infinity, + height: 1.h, + color: Color(0x14000000), + margin: EdgeInsets.only(bottom: 18.h), + ), + Padding( + padding: EdgeInsets.only(bottom: 14.h), + child: Row( + crossAxisAlignment: CrossAxisAlignment.center, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text( + "可选数量", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + setState(() { + if (specsMeal[index].optionalNum <= 1) return; + specsMeal[index].optionalNum = + specsMeal[index].optionalNum - 1; + }); + }, + child: Container( + padding: EdgeInsets.symmetric(horizontal: 20.w), + child: Icon( + Icons.remove_circle_outline, + color: Color(0xFF30415B), + size: 24, + ))), + Text( + specsMeal[index].optionalNum.toString(), + style: TextStyle( + color: Color(0xD9000000), + fontSize: 20.sp, + fontWeight: MyFontWeight.regular, + ), + ), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + setState(() { + specsMeal[index].optionalNum = + specsMeal[index].optionalNum + 1; + }); + }, + child: Container( + padding: EdgeInsets.only(left: 20.w), + child: Icon( + Icons.add_circle, + color: Color(0xFF30415B), + size: 24, + )), + ), + ], + )), + Container( + width: double.infinity, + height: 1.h, + color: Color(0x14000000), + margin: EdgeInsets.only(bottom: 18.h), + ), + Padding( + padding: EdgeInsets.only(bottom: 12.h), + child: Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + Navigator.of(context).pushNamed('/router/batch_shelf', + arguments: { + "storeId": widget.arguments["storeId"], + "titleName": "选择商品" + }).then((value) { + specsMeal[index].goodsMeal.clear(); + specsMeal[index].goodsMeal.addAll(value); + setState(() {}); + }); + }, + child: Container( + alignment: Alignment.center, + padding: EdgeInsets.only( + left: 8.w, right: 8.w, bottom: 3.h), + decoration: BoxDecoration( + color: Colors.white, + border: Border.all( + color: Color(0xFF2E3552), + width: 1, + ), + borderRadius: BorderRadius.circular(100), + ), + child: Text( + "+增加商品", + style: TextStyle( + color: Color(0xFF30415B), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + ), + )), + Spacer(), + GestureDetector( + onTap: () { + setState(() { + specsMeal.removeAt(index); + }); + }, + child: Row( + mainAxisAlignment: MainAxisAlignment.end, + children: [ + Icon( + Icons.remove_circle, + color: Color(0xFFFA5151), + size: 24, + ), + Padding( + padding: EdgeInsets.only(left: 7.w), + child: Text( + "删除分组", + style: TextStyle( + color: Color(0xD9000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.bold, + ), + ), + ), + ], + ), + ) + ], + ), + ), + Text( + "*请添加或减少商品,直到达到分组总数", + style: TextStyle( + color: Color(0xFFFA5151), + fontSize: 12.sp, + fontWeight: MyFontWeight.regular, + ), + ), + ], + ), + ), + ListView.builder( + itemCount: specsMeal[index]?.goodsMeal?.length ?? 0, + physics: BouncingScrollPhysics(), + shrinkWrap: true, + itemBuilder: (context, position) { + return GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () {}, + child: specsInfoItem(index, position), + ); + }, + ) + ], + ); + } + + ///规格信息样式 + Widget specsInfoItem(specsIndex, index) { + return Container( + color: Colors.white, + margin: EdgeInsets.symmetric(vertical: 12.h), + padding: EdgeInsets.only( + left: 16.w, + top: 20.h, + bottom: 20.h, + ), + child: Row( + crossAxisAlignment: CrossAxisAlignment.start, + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Expanded( + child: Text( + specsMeal[specsIndex]?.goodsMeal[index]["goodsName"] ?? "", + maxLines: 1, + overflow: TextOverflow.ellipsis, + style: TextStyle( + color: Color(0xFF000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + ), + flex: 2, + ), + Expanded( + child: Text( + "所有规格", + textAlign: TextAlign.center, + style: TextStyle( + color: Color(0xFF000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + ), + ), + Expanded( + child: Text( + specsMeal[specsIndex]?.goodsMeal[index]["goodsNum"] ?? "", + textAlign: TextAlign.center, + style: TextStyle( + color: Color(0xFF000000), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular, + ), + )), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + setState(() { + specsMeal[specsIndex].goodsMeal.removeAt(index); + }); + }, + child: Container( + padding: EdgeInsets.only(right: 31.h), + child: Icon( + Icons.remove_circle, + color: Color(0xFFFA5151), + size: 20, + ), + ), + ) + ], + ), + ); + } +} diff --git a/lib/business_system/login/captcha/bus_block_puzzle_captcha.dart b/lib/business_system/login/captcha/bus_block_puzzle_captcha.dart new file mode 100644 index 00000000..0f7c5e16 --- /dev/null +++ b/lib/business_system/login/captcha/bus_block_puzzle_captcha.dart @@ -0,0 +1,551 @@ +import 'dart:convert'; +import 'dart:math'; + +import 'package:dio/dio.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_screenutil/flutter_screenutil.dart'; +import 'package:huixiang/login/captcha/click_word_captcha.dart'; + +import '../../../retrofit/business_api.dart'; +import '../../../utils/captcha_util.dart'; +import '../../../utils/widget_util.dart'; + +typedef VoidSuccessCallback = dynamic Function(String v); + +class BusBlockPuzzleCaptchaPage extends StatefulWidget { + final VoidSuccessCallback onSuccess; //拖放完成后验证成功回调 + final VoidCallback onFail; //拖放完成后验证失败回调 + + BusBlockPuzzleCaptchaPage({this.onSuccess, this.onFail}); + + @override + _BusBlockPuzzleCaptchaPageState createState() => _BusBlockPuzzleCaptchaPageState(); +} + +class _BusBlockPuzzleCaptchaPageState extends State + with TickerProviderStateMixin { +// String baseImageBase64 = +// "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"; + String baseImageBase64 = ""; + String slideImageBase64 = ""; + String captchaToken = ""; + String secretKey = ""; //加密key + + Size baseSize = Size.zero; //底部基类图片 + Size slideSize = Size.zero; //滑块图片 + + var sliderColor = Colors.white; //滑块的背景色 + var sliderIcon = Icons.arrow_forward; //滑块的图标 + var movedXBorderColor = Colors.white; //滑块拖动时,左边已滑的区域边框颜色 + double sliderStartX = 0; //滑块未拖前的X坐标 + double sliderXMoved = 0; + bool sliderMoveFinish = false; //滑块拖动结束 + bool checkResultAfterDrag = false; //拖动后的校验结果 + + //-------------动画------------ + int _checkMilliseconds = 0; //滑动时间 + bool _showTimeLine = false; //是否显示动画部件 + bool _checkSuccess = false; //校验是否成功 + AnimationController controller; + + //高度动画 + Animation offsetAnimation; + + //底部部件key + GlobalKey _containerKey = new GlobalKey(); + + //背景图key + GlobalKey _baseImageKey = new GlobalKey(); + + //滑块 + GlobalKey _slideImageKey = new GlobalKey(); + double _bottomSliderSize = 60.h; + + //------------动画------------ + + //校验通过 + void checkSuccess(String content) { + setState(() { + checkResultAfterDrag = true; + _checkSuccess = true; + _showTimeLine = true; + }); + _forwardAnimation(); + updateSliderColorIcon(); + + //刷新验证码 + Future.delayed(Duration(milliseconds: 1000)).then((v) { + _reverseAnimation().then((v) { + setState(() { + _showTimeLine = false; + }); + //回调 + if (widget.onSuccess != null) { + widget.onSuccess(content); + } + //关闭验证码 + print(content); + Navigator.pop(context); + }); + }); + } + + //校验失败 + void checkFail() { + setState(() { + _showTimeLine = true; + _checkSuccess = false; + checkResultAfterDrag = false; + }); + _forwardAnimation(); + updateSliderColorIcon(); + + //刷新验证码 + Future.delayed(Duration(milliseconds: 1000)).then((v) { + _reverseAnimation().then((v) { + setState(() { + _showTimeLine = false; + }); + loadCaptcha(); + //回调 + if (widget.onFail != null) { + widget.onFail(); + } + }); + }); + } + + //重设滑动颜色与图标 + void updateSliderColorIcon() { + var _sliderColor; //滑块的背景色 + var _sliderIcon; //滑块的图标 + var _movedXBorderColor; //滑块拖动时,左边已滑的区域边框颜色 + + //滑块的背景色 + if (sliderMoveFinish) { + //拖动结束 + _sliderColor = checkResultAfterDrag ? Colors.green : Colors.red; + _sliderIcon = checkResultAfterDrag ? Icons.check : Icons.close; + _movedXBorderColor = checkResultAfterDrag ? Colors.green : Colors.red; + } else { + //拖动未开始或正在拖动中 + _sliderColor = sliderXMoved > 0 ? Color(0xff447ab2) : Colors.white; + _sliderIcon = Icons.arrow_forward; + _movedXBorderColor = Color(0xff447ab2); + } + + sliderColor = _sliderColor; + sliderIcon = _sliderIcon; + movedXBorderColor = _movedXBorderColor; + setState(() {}); + } + + //加载验证码 + void loadCaptcha() async { + setState(() { + _showTimeLine = false; + sliderMoveFinish = false; + checkResultAfterDrag = false; + sliderColor = Colors.white; //滑块的背景色 + sliderIcon = Icons.arrow_forward; //滑块的图标 + movedXBorderColor = Colors.white; //滑块拖动时,左边已滑的区域边框颜色 + }); + BusinessApiService businessService = BusinessApiService(Dio(), context: context); + ClickWordCaptchaModel baseData = await businessService + .captchaGet({"captchaType": "blockPuzzle"}).catchError((onError) {}); + if (baseData == null) { + setState(() { + secretKey = ""; + }); + return; + } + + sliderXMoved = 0; + sliderStartX = 0; + captchaToken = ''; + checkResultAfterDrag = false; + + baseImageBase64 = baseData.imgStr; + secretKey = baseData.secretKey; + baseImageBase64 = baseImageBase64.replaceAll('\n', ''); + slideImageBase64 = baseData.jigsawImageBase64; + slideImageBase64 = slideImageBase64.replaceAll('\n', ''); + captchaToken = baseData.token; + + var baseR = await WidgetUtil.getImageWH( + image: Image.memory(Base64Decoder().convert(baseImageBase64))); + baseSize = baseR.size; + + var silderR = await WidgetUtil.getImageWH( + image: Image.memory(Base64Decoder().convert(slideImageBase64))); + slideSize = silderR.size; + + setState(() {}); + } + + //校验验证码 + void checkCaptcha(sliderXMoved, captchaToken, {BuildContext myContext}) { + setState(() { + sliderMoveFinish = true; + }); + //滑动结束,改变滑块的图标及颜色 +// updateSliderColorIcon(); + + //pointJson参数需要aes加密 + +// MediaQueryData mediaQuery = MediaQuery.of(myContext); + var pointMap = {"x": sliderXMoved, "y": 5}; + var pointStr = json.encode(pointMap); + var cryptedStr = pointStr; + + // secretKey 不为空 进行as加密 + if (!CaptchaUtil.isEmpty(secretKey)) { + cryptedStr = CaptchaUtil.aesEncode(key: secretKey, content: pointStr); + // var dcrypt = CaptchaUtil.aesDecode(key: secretKey, content: cryptedStr); + // json.decode(dcrypt); + } + + BusinessApiService businessService = BusinessApiService(Dio(), context: context); + businessService + .captchaCheck({ + "pointJson": cryptedStr, + "captchaType": "blockPuzzle", + "token": captchaToken + }) + .catchError((onError) {}) + .then((res) { + if (res) { + checkFail(); + return; + } + //如果不加密 将 token 和 坐标序列化 通过 --- 链接成字符串 + var captchaVerification = "$captchaToken---$pointStr"; + if (!CaptchaUtil.isEmpty(secretKey)) { + //如果加密 将 token 和 坐标序列化 通过 --- 链接成字符串 进行加密 加密密钥为 _clickWordCaptchaModel.secretKey + captchaVerification = CaptchaUtil.aesEncode( + key: secretKey, content: captchaVerification); + } + checkSuccess(captchaVerification); + }) + .catchError((error) { + loadCaptcha(); + print(error); + }); + } + + @override + void initState() { + super.initState(); + initAnimation(); + loadCaptcha(); + } + + @override + void dispose() { + controller.dispose(); + super.dispose(); + } + + // 初始化动画 + void initAnimation() { + controller = + AnimationController(duration: Duration(milliseconds: 500), vsync: this); + + offsetAnimation = Tween(begin: 0.5, end: 0) + .animate(CurvedAnimation(parent: controller, curve: Curves.ease)) + ..addListener(() { + this.setState(() {}); + }); + } + + // 反向执行动画 + _reverseAnimation() async { + await controller.reverse(); + } + + // 正向执行动画 + _forwardAnimation() async { + await controller.forward(); + } + + @override + void didUpdateWidget(BusBlockPuzzleCaptchaPage oldWidget) { + // TODO: implement didUpdateWidget + super.didUpdateWidget(oldWidget); + } + + @override + Widget build(BuildContext context) { + return MaxScaleTextWidget( + child: buildContent(context), + ); + } + + Widget buildContent(BuildContext context) { + var mediaQuery = MediaQuery.of(context); + var dialogWidth = 0.9 * mediaQuery.size.width; + if (dialogWidth < 330) { + dialogWidth = mediaQuery.size.width; + } + + return Scaffold( + backgroundColor: Colors.transparent, + body: Center( + child: Container( + key: _containerKey, + width: dialogWidth, + height: MediaQuery.of(context).size.height >= 750 ? 310.h:325.h, + color: Colors.white, + child: Column( + mainAxisAlignment: MainAxisAlignment.start, + crossAxisAlignment: CrossAxisAlignment.center, + children: [ + _topContainer(), + _middleContainer(), + _bottomContainer(), + ], + ), + ), + ), + ); + } + + ///顶部,提示+关闭 + _topContainer() { + return Container( + height: 50.h, + padding: EdgeInsets.fromLTRB(10.w, 0, 10.w, 0), + decoration: BoxDecoration( + border: Border(bottom: BorderSide(width: 1.w, color: Color(0xffe5e5e5))), + ), + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Text( + '请完成安全验证', + style: TextStyle(fontSize: 18), + ), + IconButton( + icon: Icon(Icons.highlight_off), + iconSize: 30, + color: Colors.black38, + onPressed: () { + //退出 + Navigator.pop(context); + }), + ], + ), + ); + } + + _middleContainer() { + ////显示验证码 + return Container( + margin: EdgeInsets.symmetric(vertical: 10.h), + child: Stack( + children: [ + ///底图 310*155 + baseImageBase64.length > 0 + ? Image.memory( + Base64Decoder().convert(baseImageBase64), + fit: BoxFit.fitWidth, + key: _baseImageKey, + gaplessPlayback: true, + ) + : Container( + width: 310.w, + height: 155.h, + alignment: Alignment.center, + child: CircularProgressIndicator(), + ), + + ///滑块图 + (baseImageBase64.length > 0 && slideImageBase64.length > 0) + ? Container( + margin: EdgeInsets.fromLTRB(sliderXMoved, 0, 0, 0), + child: Image.memory( + Base64Decoder().convert(slideImageBase64), + fit: BoxFit.fitHeight, + key: _slideImageKey, + gaplessPlayback: true, + ), + ) + : Container(), + + //刷新按钮 + Positioned( + top: 0, + right: 0, + child: IconButton( + icon: Icon(Icons.refresh), + iconSize: 30, + color: Colors.black54, + onPressed: () { + //刷新 + loadCaptcha(); + }), + ), + Positioned( + bottom: 0, + left: -10.w, + right: -10.w, + child: Offstage( + offstage: !_showTimeLine, + child: FractionalTranslation( + translation: Offset(0, offsetAnimation.value), + child: Container( + margin: EdgeInsets.only(left: 10.w, right: 10.w), + height: 40.h, + color: _checkSuccess + ? Color(0x7F66BB6A) + : Color.fromRGBO(200, 100, 100, 0.4), + alignment: Alignment.centerLeft, + child: Text( + _checkSuccess + ? "${(_checkMilliseconds / (60.0 * 12)).toStringAsFixed(2)}s验证成功" + : "验证失败", + style: TextStyle(color: Colors.white), + ), + ), + ), + )), + Positioned( + bottom: -20.h, + left: 0, + right: 0, + child: Offstage( + offstage: !_showTimeLine, + child: Container( + margin: EdgeInsets.only(left: 10.w, right: 10.w), + height: 20.h, + color: Colors.white, + ), + )) + ], + ), + ); + } + + ///底部,滑动区域 + _bottomContainer() { + return baseSize.width > 0 + ? Container( + height: 70.h, + width: baseSize.width, +// color: Colors.cyanAccent, + child: Stack( + alignment: AlignmentDirectional.centerStart, + children: [ + Container( + height: _bottomSliderSize, + decoration: BoxDecoration( + border: Border.all( + width: 1.w, + color: Color(0xffe5e5e5), + ), + color: Color(0xfff8f9fb), + ), + ), + Container( + alignment: Alignment.center, + child: Text( + '向右拖动滑块填充拼图', + style: TextStyle(fontSize: 14.sp), + ), + ), + Container( + width: sliderXMoved, + height: _bottomSliderSize - 2.h, + decoration: BoxDecoration( + border: Border.all( + width: sliderXMoved > 0 ? 1 : 0, + color: movedXBorderColor, + ), + color: Color(0xfff3fef1), + ), + ), + GestureDetector( + onPanStart: (startDetails) { + ///开始 + _checkMilliseconds = + new DateTime.now().millisecondsSinceEpoch; + // print(startDetails.localPosition); + sliderStartX = startDetails.localPosition.dx; + }, + onPanUpdate: (updateDetails) { + ///更新 + // print(updateDetails.localPosition); + double _w1 = _baseImageKey.currentContext.size.width - + _slideImageKey.currentContext.size.width; + double offset = + updateDetails.localPosition.dx - sliderStartX; + if (offset < 0) { + offset = 0; + } + if (offset > _w1) { + offset = _w1; + } + // print("offset ------ $offset"); + setState(() { + sliderXMoved = offset; + }); + //滑动过程,改变滑块左边框颜色 + updateSliderColorIcon(); + }, + onPanEnd: (endDetails) { + //结束 + // print("endDetails"); + checkCaptcha(sliderXMoved, captchaToken); + int _nowTime = new DateTime.now().millisecondsSinceEpoch; + _checkMilliseconds = _nowTime - _checkMilliseconds; + }, + child: Container( + width: _bottomSliderSize, + height: _bottomSliderSize, + margin: EdgeInsets.only( + left: sliderXMoved > 0 ? sliderXMoved : 1), + decoration: BoxDecoration( + border: Border( + top: BorderSide( + width: 1.w, + color: Color(0xffe5e5e5), + ), + right: BorderSide( + width: 1.w, + color: Color(0xffe5e5e5), + ), + bottom: BorderSide( + width: 1.w, + color: Color(0xffe5e5e5), + ), + ), + color: sliderColor, + ), + child: IconButton( + icon: Icon(sliderIcon), + iconSize: 30, + color: Colors.black54, + onPressed: () {}, + ), + ), + ) + ], + )) + : Container(); + } +} + +class MaxScaleTextWidget extends StatelessWidget { + final double max; + final Widget child; + + MaxScaleTextWidget({Key key, this.max = 1.0, this.child}) : super(key: key); + + @override + Widget build(BuildContext context) { + var data = MediaQuery.of(context); + var textScaleFactor = min(max, data.textScaleFactor); + return MediaQuery( + data: data.copyWith(textScaleFactor: textScaleFactor), child: child); + } +} diff --git a/lib/business_system/login/captcha/bus_click_word_captcha.dart b/lib/business_system/login/captcha/bus_click_word_captcha.dart new file mode 100644 index 00000000..4ec1cda9 --- /dev/null +++ b/lib/business_system/login/captcha/bus_click_word_captcha.dart @@ -0,0 +1,367 @@ +import 'dart:convert'; +import 'package:dio/dio.dart'; +import 'package:flutter/cupertino.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_screenutil/flutter_screenutil.dart'; + +import '../../../retrofit/business_api.dart'; +import '../../../utils/captcha_util.dart'; +import '../../../utils/widget_util.dart'; + +typedef VoidSuccessCallback = dynamic Function(String v); + +class BusClickWordCaptcha extends StatefulWidget { + final VoidSuccessCallback onSuccess; //文字点击后验证成功回调 + final VoidCallback onFail; //文字点击完成后验证失败回调 + + const BusClickWordCaptcha({Key key, this.onSuccess, this.onFail}) + : super(key: key); + + @override + _BusClickWordCaptchaState createState() => _BusClickWordCaptchaState(); +} + +class _BusClickWordCaptchaState extends State { + BusClickWordCaptchaState _busClickWordCaptchaState = BusClickWordCaptchaState.none; + List _tapOffsetList = []; + BusClickWordCaptchaModel _busClickWordCaptchaModel = BusClickWordCaptchaModel(); + + Color titleColor = Colors.black; + Color borderColor = Color(0xffdddddd); + String bottomTitle = ""; + Size baseSize = Size(310.0, 155.0); + + //改变底部样式及字段 + _changeResultState() { + switch (_busClickWordCaptchaState) { + case BusClickWordCaptchaState.normal: + titleColor = Colors.black; + borderColor = Color(0xffdddddd); + break; + case BusClickWordCaptchaState.success: + _tapOffsetList = []; + titleColor = Colors.green; + borderColor = Colors.green; + bottomTitle = "验证成功"; + break; + case BusClickWordCaptchaState.fail: + _tapOffsetList = []; + titleColor = Colors.red; + borderColor = Colors.red; + bottomTitle = "验证失败"; + break; + default: + titleColor = Colors.black; + borderColor = Color(0xffdddddd); + bottomTitle = "数据加载中……"; + break; + } + setState(() {}); + } + + @override + void initState() { + super.initState(); + _loadCaptcha(); + } + + //加载验证码 + _loadCaptcha() async { + _tapOffsetList = []; + _busClickWordCaptchaState = BusClickWordCaptchaState.none; + _changeResultState(); + BusinessApiService businessService = BusinessApiService(Dio(), context: context); + BusClickWordCaptchaModel baseData = (await businessService.captchaGet({"captchaType": "clickWord"}).catchError((onError) {})) as BusClickWordCaptchaModel; + if (baseData == null) { + _busClickWordCaptchaModel.secretKey = ""; + bottomTitle = "加载失败,请刷新"; + _busClickWordCaptchaState = BusClickWordCaptchaState.normal; + _changeResultState(); + return; + } + else { + _busClickWordCaptchaModel = baseData; + var baseR = await WidgetUtil.getImageWH( + image: Image.memory( + Base64Decoder().convert(_busClickWordCaptchaModel.imgStr))); + baseSize = baseR.size; + + bottomTitle = "请依次点击【${_busClickWordCaptchaModel.wordStr}】"; + } + + _busClickWordCaptchaState = BusClickWordCaptchaState.normal; + _changeResultState(); + } + + //校验验证码 + _checkCaptcha() async { + List> mousePos = []; + _tapOffsetList.map((size) { + mousePos + .add({"x": size.dx.roundToDouble(), "y": size.dy.roundToDouble()}); + }).toList(); + var pointStr = json.encode(mousePos); + + var cryptedStr = pointStr; + + // secretKey 不为空 进行as加密 + if (!CaptchaUtil.isEmpty(_busClickWordCaptchaModel.secretKey)) { + cryptedStr = CaptchaUtil.aesEncode( + key: _busClickWordCaptchaModel.secretKey, content: pointStr); + // var dcrypt = CaptchaUtil.aesDecode( + // key: _busClickWordCaptchaModel.secretKey, content: cryptedStr); + } + +// Map _map = json.decode(dcrypt); + BusinessApiService businessService = BusinessApiService(Dio(), context: context); + bool baseData = await businessService.captchaCheck({ + "pointJson": cryptedStr, + "captchaType": "clickWord", + "token": _busClickWordCaptchaModel.token + }).catchError((onError) {}); + if (baseData) { + _checkFail(); + return; + } + //如果不加密 将 token 和 坐标序列化 通过 --- 链接成字符串 + var captchaVerification = "${_busClickWordCaptchaModel.token}---$pointStr"; + if (!CaptchaUtil.isEmpty(_busClickWordCaptchaModel.secretKey)) { + //如果加密 将 token 和 坐标序列化 通过 --- 链接成字符串 进行加密 加密密钥为 _busClickWordCaptchaModel.secretKey + captchaVerification = CaptchaUtil.aesEncode( + key: _busClickWordCaptchaModel.secretKey, + content: captchaVerification); + } + _checkSuccess(captchaVerification); + } + + //校验失败 + _checkFail() async { + _busClickWordCaptchaState = BusClickWordCaptchaState.fail; + _changeResultState(); + + await Future.delayed(Duration(milliseconds: 1000)); + _loadCaptcha(); + //回调 + if (widget.onFail != null) { + widget.onFail(); + } + } + + //校验成功 + _checkSuccess(String pointJson) async { + _busClickWordCaptchaState = BusClickWordCaptchaState.success; + _changeResultState(); + + await Future.delayed(Duration(milliseconds: 1000)); + + var cryptedStr = CaptchaUtil.aesEncode(key: 'BGxdEUOZkXka4HSj', content: pointJson); + + print(cryptedStr); + //回调 pointJson 是经过es加密之后的信息 + if (widget.onSuccess != null) { + widget.onSuccess(cryptedStr); + } + //关闭 + Navigator.pop(context); + } + + @override + Widget build(BuildContext context) { + var data = MediaQuery.of(context); + var dialogWidth = 0.9 * data.size.width; + var isRatioCross = false; + if (dialogWidth < 320.0) { + dialogWidth = data.size.width; + isRatioCross = true; + } + return Scaffold( + backgroundColor: Colors.transparent, + body: Center( + child: Container( + width: dialogWidth, + height: 320.h, + color: Colors.white, + child: Column( + children: [ + _topConttainer(), + _captchaContainer(), + _bottomContainer() + ], + ), + ), + ), + ); + } + + //图片验证码 + _captchaContainer() { + List _widgetList = []; + if (!CaptchaUtil.isEmpty(_busClickWordCaptchaModel.imgStr)) { + _widgetList.add(Image( + width: baseSize.width, + height: baseSize.height, + gaplessPlayback: true, + image: MemoryImage( + Base64Decoder().convert(_busClickWordCaptchaModel.imgStr)))); + } + + double _widgetW = 20; + for (int i = 0; i < _tapOffsetList.length; i++) { + Offset offset = _tapOffsetList[i]; + _widgetList.add(Positioned( + left: offset.dx - _widgetW * 0.5, + top: offset.dy - _widgetW * 0.5, + child: Container( + alignment: Alignment.center, + width: _widgetW, + height: _widgetW, + decoration: BoxDecoration( + color: Color(0xCC43A047), + borderRadius: BorderRadius.all(Radius.circular(_widgetW))), + child: Text( + "${i + 1}", + style: TextStyle(color: Colors.white, fontSize: 15), + ), + ))); + } + _widgetList.add(//刷新按钮 + Positioned( + top: 0, + right: 0, + child: IconButton( + icon: Icon(Icons.refresh), + iconSize: 30, + color: Colors.deepOrangeAccent, + onPressed: () { + //刷新 + _loadCaptcha(); + }), + )); + + return GestureDetector( + onTapDown: (TapDownDetails details) { + debugPrint( + "onTapDown globalPosition全局坐标系位置: ${details.globalPosition} localPosition组件坐标系位置: ${details.localPosition} "); + if (!CaptchaUtil.isListEmpty(_busClickWordCaptchaModel.wordList) && + _tapOffsetList.length < _busClickWordCaptchaModel.wordList.length) { + _tapOffsetList.add( + Offset(details.localPosition.dx, details.localPosition.dy)); + } + setState(() {}); + if (!CaptchaUtil.isListEmpty(_busClickWordCaptchaModel.wordList) && + _tapOffsetList.length == _busClickWordCaptchaModel.wordList.length) { + _checkCaptcha(); + } + }, + child: Container( + width: baseSize.width, + height: baseSize.height, + child: Stack( + children: _widgetList, + ), + )); + } + + //底部提示部件 + _bottomContainer() { + return Container( + height: 50.h, + margin: EdgeInsets.only(top: 10), + alignment: Alignment.center, + width: baseSize.width, + decoration: BoxDecoration( + borderRadius: BorderRadius.all(Radius.circular(4)), + border: Border.all(color: borderColor)), + child: + Text(bottomTitle, style: TextStyle(fontSize: 18, color: titleColor)), + ); + } + + //顶部,提示+关闭 + _topConttainer() { + return Container( + padding: EdgeInsets.fromLTRB(10, 0, 10, 0), + margin: EdgeInsets.only(bottom: 20, top: 5), + decoration: BoxDecoration( + border: Border(bottom: BorderSide(width: 1, color: Color(0xffe5e5e5))), + ), + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + Text( + '请完成安全验证', + style: TextStyle(fontSize: 18), + ), + IconButton( + icon: Icon(Icons.highlight_off), + iconSize: 35, + color: Colors.black54, + onPressed: () { + //退出 + Navigator.pop(context); + }), + ], + ), + ); + } +} + +//校验状态 +enum BusClickWordCaptchaState { + normal, //默认 可自定义描述 + success, //成功 + fail, //失败 + none, //无状态 用于加载使用 +} + +//请求数据模型 +class BusClickWordCaptchaModel { + String imgStr; //图表url 目前用base64 data + String jigsawImageBase64; //图表url 目前用base64 data + String token; // 获取的token 用于校验 + List wordList; //显示需要点选的字 + String wordStr; //显示需要点选的字转换为字符串 + String secretKey; //加密key + + BusClickWordCaptchaModel( + {this.imgStr = "", + this.jigsawImageBase64 = "", + this.token = "", + this.secretKey = "", + this.wordList = const [], + this.wordStr = ""}); + + //解析数据转换模型 + static BusClickWordCaptchaModel fromMap(Map map) { + BusClickWordCaptchaModel captchaModel = BusClickWordCaptchaModel(); + captchaModel.imgStr = map["originalImageBase64"] ?? ""; + captchaModel.jigsawImageBase64 = map["jigsawImageBase64"] ?? ""; + captchaModel.token = map["token"] ?? ""; + captchaModel.secretKey = map["secretKey"] ?? ""; + captchaModel.wordList = map["wordList"] ?? []; + + if (!CaptchaUtil.isListEmpty(captchaModel.wordList)) { + captchaModel.wordStr = captchaModel.wordList.join(","); + } + + return captchaModel; + } + + //将模型转换 + Map toJson() { + var map = new Map(); + map['imgStr'] = imgStr; + map['jigsawImageBase64'] = jigsawImageBase64; + map['token'] = token; + map['secretKey'] = token; + map['wordList'] = wordList; + map['wordStr'] = wordStr; + return map; + } + + @override + String toString() { + // TODO: implement toString + return JsonEncoder.withIndent(' ').convert(toJson()); + } +} diff --git a/lib/business_system/login/register_retrieve_password.dart b/lib/business_system/login/register_retrieve_password.dart new file mode 100644 index 00000000..3454beef --- /dev/null +++ b/lib/business_system/login/register_retrieve_password.dart @@ -0,0 +1,760 @@ +import 'dart:async'; + +import 'package:dio/dio.dart'; +import 'package:flutter/cupertino.dart'; +import 'package:flutter/gestures.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter/services.dart'; +import 'package:flutter_easyloading/flutter_easyloading.dart'; +import 'package:flutter_screenutil/flutter_screenutil.dart'; +import 'package:flutter_smart_dialog/flutter_smart_dialog.dart'; +import 'package:shared_preferences/shared_preferences.dart'; +import 'package:sharesdk_plugin/sharesdk_interface.dart'; + +import '../../generated/l10n.dart'; +import '../../retrofit/business_api.dart'; +import '../../retrofit/data/base_data.dart'; +import '../../utils/flutter_utils.dart'; +import '../../utils/font_weight.dart'; +import '../../view_widget/border_text.dart'; +import '../../view_widget/my_appbar.dart'; +import '../../view_widget/round_button.dart'; +import '../../view_widget/settlement_tips_dialog.dart'; +import 'captcha/bus_block_puzzle_captcha.dart'; + +class RegisterRetrievePassword extends StatefulWidget { + final Map arguments; + + RegisterRetrievePassword({this.arguments}); + + @override + State createState() { + return _RegisterRetrievePassword(); + } +} + +class _RegisterRetrievePassword extends State { + TextEditingController phoneController = TextEditingController(); + TextEditingController codeController = TextEditingController(); + TextEditingController passwordController = TextEditingController(); + TextEditingController changePasswordController = TextEditingController(); + TextEditingController businessNameController = TextEditingController(); + final TapGestureRecognizer tapGestureRecognizer = TapGestureRecognizer(); + var checkStatus = false; + String titleName; + BusinessApiService businessService; + Color statusCodeTextColor = Color(0xFF353535); + var verifyStatus = 0; + var mobileStatus = 0; + var btnText = "获取验证码"; + var sendCodeStatus = 0; + Timer _timer; + int phoneState = 1; + String mobile; + String newMobile; + + @override + void initState() { + super.initState(); + titleName = widget?.arguments["titleName"] ?? ""; + } + + @override + void dispose() { + super.dispose(); + if (_timer != null && _timer.isActive) _timer.cancel(); + } + + ///发送验证码 + sendVerifyCode(v) async { + BusinessApiService businessService = + BusinessApiService(Dio(), context: context); + BaseData baseData = await businessService.busSendVerify({ + "areaCode": "+86", + "mobile": phoneController.text, + "verification": v + }).catchError((onError) { + SmartDialog.showToast(AppUtils.dioErrorTypeToString(onError.type), + alignment: Alignment.center); + }); + if (baseData != null && baseData.isSuccess) { + sendCodeStatus = 1; + countdown(); + SmartDialog.showToast(baseData.data, alignment: Alignment.center); + } else { + btnText = S.of(context).send_code; + sendCodeStatus = 0; + SmartDialog.showToast("${baseData.msg}", alignment: Alignment.center); + refresh(); + } + } + + ///刷新页面 + void refresh() { + if (!mounted) return; + setState(() {}); + } + + ///发送验证码成功倒计时 + countdown() { + if (_timer != null && _timer.isActive) return; + int countdown = 60; + _timer = Timer.periodic(Duration(seconds: 1), (timer) { + countdown--; + if (countdown == 0) { + btnText = "重新发送"; + sendCodeStatus = 0; + _timer.cancel(); + } else { + btnText = "${countdown}s"; + } + if (!mounted) return; + setState(() {}); + }); + } + + ///注册接口 + register() async { + try { + if (codeController.text == "") { + SmartDialog.showToast("请输入验证码", alignment: Alignment.center); + return; + } + if (phoneController.text == "") { + SmartDialog.showToast(S.of(context).qingshurushoujihao, + alignment: Alignment.center); + return; + } + if (passwordController.text == "") { + SmartDialog.showToast("请输入密码", alignment: Alignment.center); + return; + } + if (businessNameController.text == "") { + SmartDialog.showToast("请输入商户名称", alignment: Alignment.center); + return; + } + if (!checkStatus) { + SmartDialog.showToast(S.of(context).gouxuanxieyi, + alignment: Alignment.center); + return; + } + + var param = { + "areaCode": "+86", + "mobile": phoneController.text, + "capcha": codeController.text, + "tenantName": businessNameController.text, + "password": passwordController.text, + }; + EasyLoading.show( + status: S.of(context).zhengzaijiazai, + maskType: EasyLoadingMaskType.black); + businessService = BusinessApiService(Dio(), context: context); + BaseData baseData = + await businessService.tenantAppLogin(param).catchError((error) { + print(error.message); + }); + if (baseData != null && baseData.isSuccess) { + SmartDialog.show( + clickBgDismissTemp: false, + widget: SettlementTips( + () {}, + text: baseData.data, + color: Color(0xFF30415B), + )); + Navigator.of(context).pop(); + } else { + if (baseData.msg != null) + SmartDialog.show( + clickBgDismissTemp: false, + widget: SettlementTips( + () {}, + text: baseData.msg, + color: Color(0xFF30415B), + )); + } + } finally { + EasyLoading.dismiss(); + } + } + + ///找回密码 + userChangePsd() async { + try { + if (phoneController.text == "") { + SmartDialog.showToast(S.of(context).qingshurushoujihao, + alignment: Alignment.center); + return; + } + if (codeController.text == "") { + SmartDialog.showToast("请输入验证码", alignment: Alignment.center); + return; + } + if (passwordController.text == "") { + SmartDialog.showToast("请输入密码", alignment: Alignment.center); + return; + } + if (changePasswordController.text == "") { + SmartDialog.showToast("请输入确认密码", alignment: Alignment.center); + return; + } + if (passwordController.text != changePasswordController.text) { + SmartDialog.showToast("两次密码不一致,请重新输入", alignment: Alignment.center); + return; + } + + var param = { + "areaCode": "+86", + "mobile": phoneController.text, + "capcha": codeController.text, + "newPassword": changePasswordController.text, + }; + EasyLoading.show( + status: S.of(context).zhengzaijiazai, + maskType: EasyLoadingMaskType.black); + businessService = BusinessApiService(Dio(), context: context); + BaseData baseData = + await businessService.tenantUserChangePsd(param).catchError((error) { + print(error.message); + }); + if (baseData != null && baseData.isSuccess) { + SmartDialog.show( + clickBgDismissTemp: false, + widget: SettlementTips( + () {}, + text: baseData.data, + color: Color(0xFF30415B), + )); + Navigator.of(context).pop(); + } else { + if (baseData.msg != null) + SmartDialog.show( + clickBgDismissTemp: false, + widget: SettlementTips( + () {}, + text: baseData.msg, + color: Color(0xFF30415B), + )); + } + } finally { + EasyLoading.dismiss(); + } + } + + @override + Widget build(BuildContext context) { + return Scaffold( + backgroundColor: Colors.white, + resizeToAvoidBottomInset: false, + appBar: MyAppBar( + title: titleName, + titleColor: Colors.black, + background: Colors.white, + leadingColor: Colors.black, + brightness: Brightness.dark, + ), + body: GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + FocusScope.of(context).requestFocus(FocusNode()); + }, + child: Column( + children: [ + registerAccount(), + GestureDetector( + onTap: () { + FocusScope.of(context).requestFocus(FocusNode()); + if (titleName == "注册账号") { + register(); + } else { + userChangePsd(); + } + }, + child: Container( + padding: EdgeInsets.symmetric(vertical: 15.h), + margin: + EdgeInsets.symmetric(horizontal: 16.w, vertical: 30.h), + alignment: Alignment.center, + width: double.infinity, + decoration: BoxDecoration( + color: Color(0xFF30415B), + borderRadius: BorderRadius.circular(4), + ), + child: Text( + titleName == "注册账号" ? "注册" : "确定", + style: TextStyle( + fontSize: 16.sp, + color: Colors.white, + fontWeight: MyFontWeight.bold), + ), + ), + ), + if (titleName == "注册账号") + Row( + mainAxisAlignment: MainAxisAlignment.start, + crossAxisAlignment: CrossAxisAlignment.center, + children: [ + Checkbox( + value: checkStatus, + onChanged: (a) { + setState(() { + checkStatus = !checkStatus; + }); + }, + checkColor: Color(0xFFFFFFFF), + fillColor: MaterialStateProperty.all(Color(0xFF30415B)), + ), + Expanded( + child: Text.rich( + TextSpan(children: [ + TextSpan( + text: S.of(context).privacy_policy1, + style: TextStyle( + fontSize: 11.sp, + color: Color(0xFF010101), + ), + ), + TextSpan( + text: "《一心回乡用户协议》", + recognizer: TapGestureRecognizer() + ..onTap = () { + Navigator.of(context) + .pushNamed('/router/user_service_page'); + }, + style: TextStyle( + fontSize: 11.sp, + color: Color(0xFF010101), + ), + ), + TextSpan( + text: "、", + style: TextStyle( + fontSize: 11.sp, + color: Colors.black, + ), + ), + TextSpan( + text: S.of(context).privacy_policy3, + recognizer: tapGestureRecognizer, + style: TextStyle( + fontSize: 11.sp, + color: Color(0xFF010101), + ), + ), + TextSpan( + text: S.of(context).privacy_policy4, + style: TextStyle( + fontSize: 11.sp, + height: 1.2, + color: Color(0xFF010101), + ), + ), + ]), + )), + SizedBox( + width: 30, + ) + ], + ), + ], + )), + ); + } + + Widget registerAccount() { + return Container( + padding: + EdgeInsets.only(top: 21.h, bottom: 16.h, left: 15.w, right: 18.w), + margin: EdgeInsets.only(left: 16.w, right: 16.w, top: 16.h), + decoration: BoxDecoration( + color: Colors.white, + boxShadow: [ + BoxShadow( + color: Colors.black.withAlpha(12), + offset: Offset(0, 3), + blurRadius: 14, + spreadRadius: 0, + ) + ], + borderRadius: BorderRadius.circular(8), + ), + child: Column( + children: [ + ///手机号 + Row( + children: [ + Text( + "手机号码", + style: TextStyle( + fontSize: 14.sp, + color: Color(0xFF1A1A1A), + fontWeight: MyFontWeight.regular), + ), + Expanded( + child: TextField( + controller: phoneController, + decoration: InputDecoration( + hintText: "请输入手机号码", + hintStyle: TextStyle( + color: Color(0xFF808080), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + border: InputBorder.none, + contentPadding: EdgeInsets.only(left: 23.w), + ), + inputFormatters: [LengthLimitingTextInputFormatter(11)], + keyboardType: TextInputType.phone, + style: TextStyle( + color: Color(0xFF808080), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + ), + ), + ], + ), + Container( + height: 1.h, + width: double.infinity, + color: Color(0xFFEBECEF), + margin: EdgeInsets.only(bottom: 16.h), + ), + + ///验证码 + Row( + children: [ + Text( + "验证码", + style: TextStyle( + fontSize: 14.sp, + color: Color(0xFF1A1A1A), + fontWeight: MyFontWeight.regular), + ), + Expanded( + flex: 2, + child: TextField( + controller: codeController, + decoration: InputDecoration( + hintText: "请输入验证码", + hintStyle: TextStyle( + color: Color(0xFF808080), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + border: InputBorder.none, + contentPadding: EdgeInsets.only(left: 23.w), + ), + keyboardType: TextInputType.phone, + style: TextStyle( + color: Color(0xFF808080), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + ), + ), + Expanded( + child: GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + mobile = phoneController.text; + if (mobile == "") { + SmartDialog.showToast(S.of(context).qingshurushoujihao, + alignment: Alignment.center); + return; + } + loadingBlockPuzzle(context); + }, + child: Text( + btnText, + textAlign: TextAlign.right, + style: TextStyle( + fontSize: 14.sp, + color: Color(0xFF30415B), + fontWeight: MyFontWeight.regular), + ), + )), + ], + ), + Container( + height: 1.h, + width: double.infinity, + color: Color(0xFFEBECEF), + margin: EdgeInsets.only(bottom: 16.h), + ), + + ///密码 + Row( + children: [ + Text( + "密码", + style: TextStyle( + fontSize: 14.sp, + color: Color(0xFF1A1A1A), + fontWeight: MyFontWeight.regular), + ), + Expanded( + child: TextField( + controller: passwordController, + decoration: InputDecoration( + hintText: "请输入密码", + hintStyle: TextStyle( + color: Color(0xFF808080), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + border: InputBorder.none, + contentPadding: EdgeInsets.only(left: 23.w), + ), + keyboardType: TextInputType.phone, + style: TextStyle( + color: Color(0xFF808080), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + ), + ), + ], + ), + Container( + height: 1.h, + width: double.infinity, + color: Color(0xFFEBECEF), + margin: EdgeInsets.only(bottom: 16.h), + ), + + ///商户名称/确认密码 + titleName == "注册账号" + ? Row( + children: [ + Text( + "商户名称", + style: TextStyle( + fontSize: 14.sp, + color: Color(0xFF1A1A1A), + fontWeight: MyFontWeight.regular), + ), + Expanded( + child: TextField( + controller: businessNameController, + decoration: InputDecoration( + hintText: "请输入商户名称", + hintStyle: TextStyle( + color: Color(0xFF808080), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + border: InputBorder.none, + contentPadding: EdgeInsets.only(left: 23.w), + ), + keyboardType: TextInputType.phone, + style: TextStyle( + color: Color(0xFF808080), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + ), + ), + ], + ) + : Row( + children: [ + Text( + "确认密码", + style: TextStyle( + fontSize: 14.sp, + color: Color(0xFF1A1A1A), + fontWeight: MyFontWeight.regular), + ), + Expanded( + child: TextField( + controller: changePasswordController, + decoration: InputDecoration( + hintText: "请再次输入密码", + hintStyle: TextStyle( + color: Color(0xFF808080), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + border: InputBorder.none, + contentPadding: EdgeInsets.only(left: 23.w), + ), + keyboardType: TextInputType.phone, + style: TextStyle( + color: Color(0xFF808080), + fontSize: 14.sp, + fontWeight: MyFontWeight.regular), + ), + ), + ], + ) + ], + )); + } + + showAlertDialog() { + //显示对话框 + showDialog( + context: context, + builder: (BuildContext context) { + return WillPopScope( + onWillPop: () async => false, + child: SimpleDialog( + titlePadding: EdgeInsets.all(10), + backgroundColor: Colors.transparent, + elevation: 0, + shape: RoundedRectangleBorder( + borderRadius: BorderRadius.circular(6), + ), + children: [ + Stack( + alignment: Alignment.bottomCenter, + children: [ + Container( + alignment: Alignment.center, + width: double.infinity, + height: 325.h, + padding: EdgeInsets.only(left: 16.w, right: 16.w), + decoration: new BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(8), + ), + child: Column( + children: [ + Padding( + padding: EdgeInsets.only(top: 24.h, bottom: 10.h), + child: Text( + S.of(context).xieyitanchuang, + style: TextStyle( + color: Color(0xff4D4D4D), + fontSize: 18.sp, + fontWeight: FontWeight.bold, + ), + ), + ), + Text.rich( + TextSpan(children: [ + TextSpan( + text: S.of(context).yinsizhengce1, + style: TextStyle( + fontWeight: MyFontWeight.medium, + fontSize: 14.sp, + height: 1.3.h, + color: Color(0xff727272), + ), + ), + TextSpan( + text: S.of(context).yinsixieyi, + style: TextStyle( + fontWeight: MyFontWeight.medium, + fontSize: 14.sp, + color: Color(0xff32A060)), + recognizer: TapGestureRecognizer() + ..onTap = () { + Navigator.of(context) + .popAndPushNamed('/router/treaty_page'); + }, + ), + ]), + ), + SizedBox( + height: 10.h, + ), + Text( + S.of(context).yinsizhengce2, + style: TextStyle( + color: Color(0xff727272), + fontSize: 14.sp, + height: 1.3.h, + fontWeight: MyFontWeight.medium, + ), + ), + SizedBox( + height: 16.h, + ), + ], + ), + ), + Row( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + GestureDetector( + onTap: () { + Navigator.of(context).pop(); + // exit(0); + }, + child: Container( + height: 40.h, + alignment: Alignment.bottomCenter, + margin: EdgeInsets.only(bottom: 20.h), + child: BorderText( + padding: EdgeInsets.only( + top: 10.h, + bottom: 10.h, + left: 36.w, + right: 36.w, + ), + text: S.of(context).jujue, + fontSize: 12.sp, + textColor: Color(0xFF32A060), + borderColor: Color(0xFF32A060), + borderWidth: 1.w, + radius: 23, + ), + ), + ), + SizedBox( + width: 21.w, + ), + Container( + height: 40.h, + margin: EdgeInsets.only(bottom: 20.h), + alignment: Alignment.bottomCenter, + child: RoundButton( + text: S.of(context).tongyibingjixu, + textColor: Colors.white, + fontSize: 12.sp, + callback: () { + SharedPreferences.getInstance().then((value) { + value.setBool("isShowPrivacyPolicy", true); + }); + SharesdkPlugin.uploadPrivacyPermissionStatus( + 1, + (success) => { + Navigator.of(context).pop(), + }, + ); + }, + padding: EdgeInsets.only( + top: 10.h, + bottom: 10.h, + left: 21.5.w, + right: 21.5.w, + ), + backgroup: Color(0xff32A060), + radius: 23, + ), + ), + SizedBox( + height: 20.h, + ), + ], + ), + ], + ) + ], + ), + ); + }, + ); + } + + //滑动拼图 + loadingBlockPuzzle(BuildContext context, {barrierDismissible = true}) { + showDialog( + context: context, + barrierDismissible: barrierDismissible, + builder: (BuildContext context) { + return BusBlockPuzzleCaptchaPage( + onSuccess: (v) { + sendVerifyCode(v); + }, + onFail: () { + print("onFail"); + }, + ); + }, + ); + } +} diff --git a/lib/business_system/mine/service_subscription/function_version_detail.dart b/lib/business_system/mine/service_subscription/function_version_detail.dart new file mode 100644 index 00000000..1a0dfb69 --- /dev/null +++ b/lib/business_system/mine/service_subscription/function_version_detail.dart @@ -0,0 +1,44 @@ +import 'package:flutter/material.dart'; +import 'package:huixiang/view_widget/my_appbar.dart'; + +class FunctionVersionDetail extends StatefulWidget { + final Map arguments; + + FunctionVersionDetail({this.arguments}); + + @override + State createState() { + return _FunctionVersionDetail(); + } +} + +class _FunctionVersionDetail extends State { + + @override + void initState() { + super.initState(); + } + + @override + Widget build(BuildContext context) { + return Scaffold( + appBar: MyAppBar( + title: "功能版本介绍", + titleColor: Colors.black, + background: Colors.white, + leadingColor: Colors.black, + brightness: Brightness.dark, + ), + body:Container( + width: double.infinity, + height: double.infinity, + child:Image.asset( + "assets/image/activity_q.webp", + width: double.infinity, + height: double.infinity, + fit: BoxFit.fill, + ), + ), + ); + } +} diff --git a/lib/business_system/mine/service_subscription/service_purchase_record.dart b/lib/business_system/mine/service_subscription/service_purchase_record.dart new file mode 100644 index 00000000..3cf30a2b --- /dev/null +++ b/lib/business_system/mine/service_subscription/service_purchase_record.dart @@ -0,0 +1,459 @@ +import 'package:dio/dio.dart'; +import 'package:flutter/cupertino.dart'; +import 'package:flutter/material.dart'; +import 'package:flutter_easyloading/flutter_easyloading.dart'; +import 'package:huixiang/view_widget/classic_header.dart'; +import 'package:huixiang/view_widget/my_footer.dart'; +import 'package:pull_to_refresh/pull_to_refresh.dart'; +import 'package:flutter_screenutil/flutter_screenutil.dart'; +import 'package:shimmer/shimmer.dart'; + +import '../../../generated/l10n.dart'; +import '../../../retrofit/business_api.dart'; +import '../../../retrofit/data/base_data.dart'; +import '../../../retrofit/data/service_bug_list.dart'; +import '../../../utils/business_instance.dart'; +import '../../../utils/flutter_utils.dart'; +import '../../../utils/font_weight.dart'; +import '../../../view_widget/my_appbar.dart'; +import '../../../view_widget/no_data_view.dart'; + +class ServicePurchaseRecord extends StatefulWidget { + final Map arguments; + + ServicePurchaseRecord({this.arguments}); + + @override + State createState() { + return _ServicePurchaseRecord(); + } +} + +class _ServicePurchaseRecord extends State { + final RefreshController refreshController = RefreshController(); + BusinessApiService businessService; + int _current = 1; + String networkError = ""; + int networkStatus = 0; + List records = []; + + ///离开页面记着销毁和清除 + @override + void dispose() { + super.dispose(); + refreshController.dispose(); + } + + @override + void initState() { + super.initState(); + _onRefresh(); + } + + _onRefresh({bool isShowLoad = true}) async { + if (isShowLoad) + EasyLoading.show( + status: S.current.zhengzaijiazai, + maskType: EasyLoadingMaskType.black); + await queryPackageBuyList(); + EasyLoading.dismiss(); + if (!mounted) return; + if (refreshController.isRefresh) refreshController.refreshCompleted(); + setState(() {}); + } + + ///服务购买记录 + queryPackageBuyList() async { + try { + if (businessService == null) { + businessService = BusinessApiService(Dio(), + context: context, + token: BusinessInstance.instance.businessToken, + tenant: BusinessInstance.instance.businessTenant, + storeId: widget.arguments["storeId"]); + } + BaseData baseData = await businessService.packageBuyList({ + "current": _current, + "map": {}, + "model": {"tenantCode": BusinessInstance.instance.businessTenant}, + "order": "descending", + "size": 10, + "sort": "createTime" + }).catchError((error) { + networkError = AppUtils.dioErrorTypeToString(error.type); + networkStatus = -1; + setState(() {}); + refreshController.refreshFailed(); + refreshController.loadFailed(); + }); + if (!mounted) return; + if (baseData != null && baseData.isSuccess) { + records.addAll(baseData?.data?.records ?? []); + if ((baseData?.data?.records ?? []).isEmpty || + records.length.toString() == baseData.data.total) + refreshController.loadNoData(); + else + refreshController.loadComplete(); + networkStatus = 1; + } + } finally { + } + } + + @override + Widget build(BuildContext context) { + return Scaffold( + appBar: MyAppBar( + title: "购买记录", + titleColor: Colors.black, + background: Colors.white, + leadingColor: Colors.black, + ), + body: SmartRefresher( + controller: refreshController, + enablePullDown: true, + enablePullUp: records.length == 0 ? false : true, + header: MyHeader(color: Color(0xFF30415B)), + physics: BouncingScrollPhysics(), + footer: CustomFooter( + builder: (context, mode) { + return MyFooter(mode); + }, + ), + onRefresh: () { + _current = 1; + records.clear(); + _onRefresh(isShowLoad: false); + }, + onLoading: () { + _current++; + _onRefresh(isShowLoad: false); + }, + child: networkStatus == -1 + ? noNetwork() + : ((networkStatus == 0) + ? Container( + margin: EdgeInsets.only(top: 12.h), + child: ListView.builder( + padding: EdgeInsets.zero, + itemCount: 10, + scrollDirection: Axis.vertical, + shrinkWrap: true, + physics: BouncingScrollPhysics(), + itemBuilder: (context, position) { + return purchaseRecordItemSm(); + }, + )) + : ((records == null || records.length == 0) + ? NoDataView( + src: "assets/image/bs_no data_logo.webp", + isShowBtn: false, + text: "暂无购买记录", + fontSize: 16.sp, + iconHeight: 120.h, + margin: EdgeInsets.all(50.h), + ) + : Container( + margin: EdgeInsets.only(top: 12.h), + child: ListView.builder( + itemCount: records?.length ?? 0, + physics: BouncingScrollPhysics(), + shrinkWrap: true, + itemBuilder: (context, position) { + return GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () {}, + child: purchaseRecordItem(records[position]), + ); + }, + ), + )))), + ); + } + + Widget purchaseRecordItem(Records records) { + return Container( + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(8), + ), + margin: EdgeInsets.only(left: 16.w, right: 16.w, bottom: 12.h), + padding: EdgeInsets.only( + left: 12.h, + top: 12.h, + bottom: 12.h, + ), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text( + records?.packageName ?? "", + style: TextStyle( + fontSize: 14.sp, + color: Color(0xFF0D0D0D), + fontWeight: MyFontWeight.bold), + ), + Container( + width: double.infinity, + height: 1.h, + color: Color(0xFFF4F6F7), + margin: EdgeInsets.symmetric(vertical: 13.h), + ), + + ///服务内容暂未定义,暂时隐藏 + // Padding( + // padding: EdgeInsets.only(bottom: 12.h), + // child: Text( + // "套餐内容: 会员管理", + // style: TextStyle( + // fontSize: 12.sp, + // color: Color(0xFF1A1A1A), + // fontWeight: MyFontWeight.regular), + // ), + // ), + Padding( + padding: EdgeInsets.only(bottom: 12.h), + child: Text( + "套餐年限: ${records?.packageNum ?? "0"}年", + style: TextStyle( + fontSize: 12.sp, + color: Color(0xFF1A1A1A), + fontWeight: MyFontWeight.regular), + )), + Padding( + padding: EdgeInsets.only(bottom: 12.h), + child: Text( + "套餐金额: ${records?.packagePrice ?? "0"}元", + style: TextStyle( + fontSize: 12.sp, + color: Color(0xFF1A1A1A), + fontWeight: MyFontWeight.regular), + )), + Padding( + padding: EdgeInsets.only(bottom: 12.h), + child: Text( + "购买时间: ${records?.createTime ?? ""}", + style: TextStyle( + fontSize: 12.sp, + color: Color(0xFF1A1A1A), + fontWeight: MyFontWeight.regular), + )), + Text( + "支付方式: ${records?.payChannel == 1 ? "微信支付" :"支付宝支付"}", + style: TextStyle( + fontSize: 12.sp, + color: Color(0xFF1A1A1A), + fontWeight: MyFontWeight.regular), + ) + ], + ), + ); + } + + Widget purchaseRecordItemSm() { + return Container( + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(8), + ), + margin: EdgeInsets.only(left: 16.w, right: 16.w, bottom: 12.h), + padding: EdgeInsets.only( + left: 12.h, + top: 16.h, + bottom: 16.h, + ), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Shimmer.fromColors( + baseColor: Color(0XFFD8D8D8), + highlightColor: Color(0XFFD8D8D8), + child: Container( + decoration: BoxDecoration( + color: Color(0XFFD8D8D8), + borderRadius: BorderRadius.circular(2), + ), + width: 70.w, + height: 20.h, + ), + ), + Container( + width: double.infinity, + height: 1.h, + color: Color(0xFFF4F6F7), + margin: EdgeInsets.symmetric(vertical: 13.h), + ), + Padding( + padding: EdgeInsets.only(bottom: 12.h), + child: Row( + children: [ + Shimmer.fromColors( + baseColor: Color(0XFFD8D8D8), + highlightColor: Color(0XFFD8D8D8), + child: Container( + decoration: BoxDecoration( + color: Color(0XFFD8D8D8), + borderRadius: BorderRadius.circular(2), + ), + width: 52.w, + height: 18.h, + ), + ), + Shimmer.fromColors( + baseColor: Color(0XFFD8D8D8), + highlightColor: Color(0XFFD8D8D8), + child: Container( + margin: EdgeInsets.only(left: 15.w), + decoration: BoxDecoration( + color: Color(0XFFD8D8D8), + borderRadius: BorderRadius.circular(2), + ), + width: 48.w, + height: 18.h, + ), + ), + ], + )), + Padding( + padding: EdgeInsets.only(bottom: 12.h), + child: Row( + children: [ + Shimmer.fromColors( + baseColor: Color(0XFFD8D8D8), + highlightColor: Color(0XFFD8D8D8), + child: Container( + decoration: BoxDecoration( + color: Color(0XFFD8D8D8), + borderRadius: BorderRadius.circular(2), + ), + width: 52.w, + height: 18.h, + ), + ), + Shimmer.fromColors( + baseColor: Color(0XFFD8D8D8), + highlightColor: Color(0XFFD8D8D8), + child: Container( + margin: EdgeInsets.only(left: 15.w), + decoration: BoxDecoration( + color: Color(0XFFD8D8D8), + borderRadius: BorderRadius.circular(2), + ), + width: 30.w, + height: 18.h, + ), + ), + ], + )), + Padding( + padding: EdgeInsets.only(bottom: 12.h), + child: Row( + children: [ + Shimmer.fromColors( + baseColor: Color(0XFFD8D8D8), + highlightColor: Color(0XFFD8D8D8), + child: Container( + decoration: BoxDecoration( + color: Color(0XFFD8D8D8), + borderRadius: BorderRadius.circular(2), + ), + width: 52.w, + height: 18.h, + ), + ), + Shimmer.fromColors( + baseColor: Color(0XFFD8D8D8), + highlightColor: Color(0XFFD8D8D8), + child: Container( + margin: EdgeInsets.only(left: 15.w), + decoration: BoxDecoration( + color: Color(0XFFD8D8D8), + borderRadius: BorderRadius.circular(2), + ), + width: 123.w, + height: 18.h, + ), + ), + ], + )), + Row( + children: [ + Shimmer.fromColors( + baseColor: Color(0XFFD8D8D8), + highlightColor: Color(0XFFD8D8D8), + child: Container( + decoration: BoxDecoration( + color: Color(0XFFD8D8D8), + borderRadius: BorderRadius.circular(2), + ), + width: 52.w, + height: 18.h, + ), + ), + Shimmer.fromColors( + baseColor: Color(0XFFD8D8D8), + highlightColor: Color(0XFFD8D8D8), + child: Container( + margin: EdgeInsets.only(left: 15.w), + decoration: BoxDecoration( + color: Color(0XFFD8D8D8), + borderRadius: BorderRadius.circular(2), + ), + width: 48.w, + height: 18.h, + ), + ), + ], + ) + ], + )); + } + + Widget noNetwork() { + return Container( + margin: EdgeInsets.only(top: 120.h), + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Text( + // "无法连接到网络", + networkError.substring(0, networkError.indexOf(",")), + style: TextStyle( + fontSize: 14.sp, + color: Color(0xFF0D0D0D), + fontWeight: MyFontWeight.bold), + ), + Padding( + padding: EdgeInsets.symmetric(vertical: 10.h), + child: Text( + "请检查网络设置或稍后重试", + style: TextStyle( + fontSize: 12.sp, + color: Color(0xFF7A797F), + fontWeight: MyFontWeight.regular), + ), + ), + GestureDetector( + behavior: HitTestBehavior.opaque, + onTap: () { + _onRefresh(); + }, + child: Container( + decoration: BoxDecoration( + color: Color(0xFF30415B), + borderRadius: BorderRadius.circular(15), + ), + padding: EdgeInsets.symmetric(horizontal: 12.w, vertical: 3.h), + child: Text( + "重试", + style: TextStyle( + fontSize: 14.sp, + color: Colors.white, + fontWeight: MyFontWeight.regular), + )), + ) + ], + ), + ); + } +} diff --git a/lib/main.dart b/lib/main.dart index 2073baa6..4e634d07 100644 --- a/lib/main.dart +++ b/lib/main.dart @@ -93,6 +93,7 @@ import 'business_system/date_select/week_report_page.dart'; import 'business_system/goods/add_goods/add_goods_page.dart'; import 'business_system/goods/add_goods/edit_specs_detail.dart'; import 'business_system/goods/add_goods/set_goods_specs.dart'; +import 'business_system/goods/add_goods/set_goods_specs_value.dart'; import 'business_system/goods/add_goods/set_meal.dart'; import 'business_system/goods/goods_search_page.dart'; import 'business_system/goods/on_sale/add_assort.dart'; @@ -113,11 +114,14 @@ import 'business_system/home/vip/member_details_page.dart'; import 'business_system/home/vip/pay_success_page.dart'; import 'business_system/home/vip/vip_recharge_page.dart'; import 'business_system/login/business_login_page.dart'; +import 'business_system/login/register_retrieve_password.dart'; import 'business_system/mine/account_information.dart'; import 'business_system/mine/clerk_manage/add_new_clerk_page.dart'; import 'business_system/mine/clerk_manage/clerk_manage_page.dart'; import 'business_system/mine/merchant_info.dart'; import 'business_system/mine/security_setting.dart'; +import 'business_system/mine/service_subscription/function_version_detail.dart'; +import 'business_system/mine/service_subscription/service_purchase_record.dart'; import 'business_system/mine/service_subscription/service_subscription_page.dart'; import 'business_system/mine/shop_image_info.dart'; import 'business_system/mine/shop_reservation_code.dart'; @@ -566,4 +570,12 @@ Map routers = { EditSpecsDetail(arguments:arguments), '/router/set_meal': (context, {arguments}) => SetMeal(arguments:arguments), + '/router/function_version_detail': (context, {arguments}) => + FunctionVersionDetail(arguments:arguments), + '/router/register_retrieve_password': (context, {arguments}) => + RegisterRetrievePassword(arguments:arguments), + '/router/service_purchase_record': (context, {arguments}) => + ServicePurchaseRecord(arguments:arguments), + '/router/set_goods_specs_value': (context, {arguments}) => + SetGoodsSpecsValue(arguments:arguments), }; diff --git a/lib/retrofit/data/edit_specs_detail_list.dart b/lib/retrofit/data/edit_specs_detail_list.dart new file mode 100644 index 00000000..b061b1cb --- /dev/null +++ b/lib/retrofit/data/edit_specs_detail_list.dart @@ -0,0 +1,32 @@ +import 'package:flutter/cupertino.dart'; + +class EditSpecsDetailList { + String _specsDetailName; + TextEditingController _goodPriceController; + TextEditingController _originalPriceController; + TextEditingController _packagingFeeController; + TextEditingController _specsWeightController; + TextEditingController _skuStockController; + + TextEditingController get goodPriceController => _goodPriceController; + TextEditingController get originalPriceController => _originalPriceController; + TextEditingController get packagingFeeController => _packagingFeeController; + TextEditingController get specsWeightController => _specsWeightController; + TextEditingController get skuStockController => _skuStockController; + + + String get specsDetailName => _specsDetailName; + + set specsDetailName(String value) { + _specsDetailName = value; + } + + EditSpecsDetailList(String specsDetailName){ + _specsDetailName = specsDetailName; + _goodPriceController = TextEditingController(); + _originalPriceController = TextEditingController(); + _packagingFeeController = TextEditingController(); + _specsWeightController = TextEditingController(); + _skuStockController = TextEditingController(); + } +} \ No newline at end of file diff --git a/lib/retrofit/data/service_bug_list.dart b/lib/retrofit/data/service_bug_list.dart new file mode 100644 index 00000000..5b59a119 --- /dev/null +++ b/lib/retrofit/data/service_bug_list.dart @@ -0,0 +1,313 @@ +/// records : [{"id":"1745006589485514752","createTime":"2024-01-10 16:56:06","createUser":"1739884285034233856","updateTime":"2024-01-11 07:51:48","updateUser":"1739884285034233856","tenantId":"1739884282001752064","tenantCode":"1197","packageId":"1743090505270427648","oldPackageId":"1742842291845857280","packagePrice":"0.01","packageNum":1,"packageDiscount":100,"expirationTime":"2025-01-10 16:56:06","upgradeTime":null,"status":0,"payChannel":1,"isDelete":0,"packageName":"品牌商户"},{"id":"1745002175626477568","createTime":"2024-01-10 16:38:34","createUser":"1739884285034233856","updateTime":"2024-01-11 07:51:47","updateUser":"1739884285034233856","tenantId":"1739884282001752064","tenantCode":"1197","packageId":"1742842291845857280","oldPackageId":null,"packagePrice":"200.00","packageNum":10,"packageDiscount":100,"expirationTime":"2049-09-11 15:39:12","upgradeTime":null,"status":0,"payChannel":1,"isDelete":0,"packageName":"普通商户"},{"id":"1744984806560628736","createTime":"2024-01-10 15:29:33","createUser":"1739884285034233856","updateTime":"2024-01-11 07:51:46","updateUser":"1739884285034233856","tenantId":"1739884282001752064","tenantCode":"1197","packageId":"1743090505270427648","oldPackageId":null,"packagePrice":"0.01","packageNum":2,"packageDiscount":100,"expirationTime":"2030-01-10 14:23:37","upgradeTime":null,"status":1,"payChannel":1,"isDelete":0,"packageName":"品牌商户"},{"id":"1744983525037178880","createTime":"2024-01-10 15:24:27","createUser":"1739884285034233856","updateTime":"2024-01-11 07:51:46","updateUser":"1739884285034233856","tenantId":"1739884282001752064","tenantCode":"1197","packageId":"1743090505270427648","oldPackageId":null,"packagePrice":"0.01","packageNum":3,"packageDiscount":100,"expirationTime":"2028-01-10 14:23:37","upgradeTime":null,"status":1,"payChannel":1,"isDelete":0,"packageName":"品牌商户"},{"id":"1744968216842600448","createTime":"2024-01-10 14:23:37","createUser":"1739884285034233856","updateTime":"2024-01-11 07:51:45","updateUser":"1739884285034233856","tenantId":"1739884282001752064","tenantCode":"1197","packageId":"1743090505270427648","oldPackageId":"1742842291845857280","packagePrice":"0.01","packageNum":1,"packageDiscount":100,"expirationTime":"2025-01-10 14:23:37","upgradeTime":null,"status":1,"payChannel":1,"isDelete":0,"packageName":"品牌商户"},{"id":"1744965441966571520","createTime":"2024-01-10 14:12:36","createUser":"1739884285034233856","updateTime":"2024-01-11 07:51:44","updateUser":"1739884285034233856","tenantId":"1739884282001752064","tenantCode":"1197","packageId":"1742842291845857280","oldPackageId":null,"packagePrice":"0.01","packageNum":10,"packageDiscount":100,"expirationTime":"2030-01-10 18:00:00","upgradeTime":"2024-01-10 16:56:05","status":3,"payChannel":1,"isDelete":0,"packageName":"普通商户"},{"id":"1744965184633438208","createTime":"2024-01-10 14:11:35","createUser":"1739884285034233856","updateTime":"2024-01-11 07:51:44","updateUser":"1739884285034233856","tenantId":"1739884282001752064","tenantCode":"1197","packageId":"1742842291845857280","oldPackageId":null,"packagePrice":"0.01","packageNum":5,"packageDiscount":100,"expirationTime":"2027-01-10 18:00:00","upgradeTime":"2024-01-10 16:38:57","status":3,"payChannel":1,"isDelete":0,"packageName":"普通商户"},{"id":"1744965083273887744","createTime":"2024-01-10 14:11:10","createUser":"1739884285034233856","updateTime":"2024-01-11 07:51:43","updateUser":"1739884285034233856","tenantId":"1739884282001752064","tenantCode":"1197","packageId":"1742842291845857280","oldPackageId":null,"packagePrice":"0.01","packageNum":1,"packageDiscount":100,"expirationTime":"2025-01-10 18:00:00","upgradeTime":"2024-05-10 18:00:00","status":3,"payChannel":1,"isDelete":0,"packageName":"普通商户"}] +/// total : "8" +/// size : "10" +/// current : "1" +/// orders : [{"column":"create_time","asc":false}] +/// hitCount : false +/// searchCount : true +/// pages : "1" + +class ServiceBugList { + ServiceBugList({ + List records, + String total, + String size, + String current, + List orders, + bool hitCount, + bool searchCount, + String pages,}){ + _records = records; + _total = total; + _size = size; + _current = current; + _orders = orders; + _hitCount = hitCount; + _searchCount = searchCount; + _pages = pages; +} + + ServiceBugList.fromJson(dynamic json) { + if (json['records'] != null) { + _records = []; + json['records'].forEach((v) { + _records.add(Records.fromJson(v)); + }); + } + _total = json['total']; + _size = json['size']; + _current = json['current']; + if (json['orders'] != null) { + _orders = []; + json['orders'].forEach((v) { + _orders.add(Orders.fromJson(v)); + }); + } + _hitCount = json['hitCount']; + _searchCount = json['searchCount']; + _pages = json['pages']; + } + List _records; + String _total; + String _size; + String _current; + List _orders; + bool _hitCount; + bool _searchCount; + String _pages; +ServiceBugList copyWith({ List records, + String total, + String size, + String current, + List orders, + bool hitCount, + bool searchCount, + String pages, +}) => ServiceBugList( records: records ?? _records, + total: total ?? _total, + size: size ?? _size, + current: current ?? _current, + orders: orders ?? _orders, + hitCount: hitCount ?? _hitCount, + searchCount: searchCount ?? _searchCount, + pages: pages ?? _pages, +); + List get records => _records; + String get total => _total; + String get size => _size; + String get current => _current; + List get orders => _orders; + bool get hitCount => _hitCount; + bool get searchCount => _searchCount; + String get pages => _pages; + + Map toJson() { + final map = {}; + if (_records != null) { + map['records'] = _records.map((v) => v.toJson()).toList(); + } + map['total'] = _total; + map['size'] = _size; + map['current'] = _current; + if (_orders != null) { + map['orders'] = _orders.map((v) => v.toJson()).toList(); + } + map['hitCount'] = _hitCount; + map['searchCount'] = _searchCount; + map['pages'] = _pages; + return map; + } + +} + +/// column : "create_time" +/// asc : false + +class Orders { + Orders({ + String column, + bool asc,}){ + _column = column; + _asc = asc; +} + + Orders.fromJson(dynamic json) { + _column = json['column']; + _asc = json['asc']; + } + String _column; + bool _asc; +Orders copyWith({ String column, + bool asc, +}) => Orders( column: column ?? _column, + asc: asc ?? _asc, +); + String get column => _column; + bool get asc => _asc; + + Map toJson() { + final map = {}; + map['column'] = _column; + map['asc'] = _asc; + return map; + } + +} + +/// id : "1745006589485514752" +/// createTime : "2024-01-10 16:56:06" +/// createUser : "1739884285034233856" +/// updateTime : "2024-01-11 07:51:48" +/// updateUser : "1739884285034233856" +/// tenantId : "1739884282001752064" +/// tenantCode : "1197" +/// packageId : "1743090505270427648" +/// oldPackageId : "1742842291845857280" +/// packagePrice : "0.01" +/// packageNum : 1 +/// packageDiscount : 100 +/// expirationTime : "2025-01-10 16:56:06" +/// upgradeTime : null +/// status : 0 +/// payChannel : 1 +/// isDelete : 0 +/// packageName : "品牌商户" + +class Records { + Records({ + String id, + String createTime, + String createUser, + String updateTime, + String updateUser, + String tenantId, + String tenantCode, + String packageId, + String oldPackageId, + String packagePrice, + num packageNum, + num packageDiscount, + String expirationTime, + dynamic upgradeTime, + num status, + num payChannel, + num isDelete, + String packageName,}){ + _id = id; + _createTime = createTime; + _createUser = createUser; + _updateTime = updateTime; + _updateUser = updateUser; + _tenantId = tenantId; + _tenantCode = tenantCode; + _packageId = packageId; + _oldPackageId = oldPackageId; + _packagePrice = packagePrice; + _packageNum = packageNum; + _packageDiscount = packageDiscount; + _expirationTime = expirationTime; + _upgradeTime = upgradeTime; + _status = status; + _payChannel = payChannel; + _isDelete = isDelete; + _packageName = packageName; +} + + Records.fromJson(dynamic json) { + _id = json['id']; + _createTime = json['createTime']; + _createUser = json['createUser']; + _updateTime = json['updateTime']; + _updateUser = json['updateUser']; + _tenantId = json['tenantId']; + _tenantCode = json['tenantCode']; + _packageId = json['packageId']; + _oldPackageId = json['oldPackageId']; + _packagePrice = json['packagePrice']; + _packageNum = json['packageNum']; + _packageDiscount = json['packageDiscount']; + _expirationTime = json['expirationTime']; + _upgradeTime = json['upgradeTime']; + _status = json['status']; + _payChannel = json['payChannel']; + _isDelete = json['isDelete']; + _packageName = json['packageName']; + } + String _id; + String _createTime; + String _createUser; + String _updateTime; + String _updateUser; + String _tenantId; + String _tenantCode; + String _packageId; + String _oldPackageId; + String _packagePrice; + num _packageNum; + num _packageDiscount; + String _expirationTime; + dynamic _upgradeTime; + num _status; + num _payChannel; + num _isDelete; + String _packageName; +Records copyWith({ String id, + String createTime, + String createUser, + String updateTime, + String updateUser, + String tenantId, + String tenantCode, + String packageId, + String oldPackageId, + String packagePrice, + num packageNum, + num packageDiscount, + String expirationTime, + dynamic upgradeTime, + num status, + num payChannel, + num isDelete, + String packageName, +}) => Records( id: id ?? _id, + createTime: createTime ?? _createTime, + createUser: createUser ?? _createUser, + updateTime: updateTime ?? _updateTime, + updateUser: updateUser ?? _updateUser, + tenantId: tenantId ?? _tenantId, + tenantCode: tenantCode ?? _tenantCode, + packageId: packageId ?? _packageId, + oldPackageId: oldPackageId ?? _oldPackageId, + packagePrice: packagePrice ?? _packagePrice, + packageNum: packageNum ?? _packageNum, + packageDiscount: packageDiscount ?? _packageDiscount, + expirationTime: expirationTime ?? _expirationTime, + upgradeTime: upgradeTime ?? _upgradeTime, + status: status ?? _status, + payChannel: payChannel ?? _payChannel, + isDelete: isDelete ?? _isDelete, + packageName: packageName ?? _packageName, +); + String get id => _id; + String get createTime => _createTime; + String get createUser => _createUser; + String get updateTime => _updateTime; + String get updateUser => _updateUser; + String get tenantId => _tenantId; + String get tenantCode => _tenantCode; + String get packageId => _packageId; + String get oldPackageId => _oldPackageId; + String get packagePrice => _packagePrice; + num get packageNum => _packageNum; + num get packageDiscount => _packageDiscount; + String get expirationTime => _expirationTime; + dynamic get upgradeTime => _upgradeTime; + num get status => _status; + num get payChannel => _payChannel; + num get isDelete => _isDelete; + String get packageName => _packageName; + + Map toJson() { + final map = {}; + map['id'] = _id; + map['createTime'] = _createTime; + map['createUser'] = _createUser; + map['updateTime'] = _updateTime; + map['updateUser'] = _updateUser; + map['tenantId'] = _tenantId; + map['tenantCode'] = _tenantCode; + map['packageId'] = _packageId; + map['oldPackageId'] = _oldPackageId; + map['packagePrice'] = _packagePrice; + map['packageNum'] = _packageNum; + map['packageDiscount'] = _packageDiscount; + map['expirationTime'] = _expirationTime; + map['upgradeTime'] = _upgradeTime; + map['status'] = _status; + map['payChannel'] = _payChannel; + map['isDelete'] = _isDelete; + map['packageName'] = _packageName; + return map; + } + +} \ No newline at end of file diff --git a/lib/retrofit/data/set_specs_list.dart b/lib/retrofit/data/set_specs_list.dart new file mode 100644 index 00000000..5f28f98a --- /dev/null +++ b/lib/retrofit/data/set_specs_list.dart @@ -0,0 +1,19 @@ +import 'package:flutter/cupertino.dart'; + +class SetSpecsList { + TextEditingController _specsNameController; + List _specsValues = []; + + TextEditingController get specsNameController => _specsNameController; + + + List get specsValues => _specsValues; + + set specsValues(List value) { + _specsValues = value; + } + + SetSpecsList(){ + _specsNameController = TextEditingController(); + } +} \ No newline at end of file diff --git a/lib/retrofit/data/set_specs_meal_list.dart b/lib/retrofit/data/set_specs_meal_list.dart new file mode 100644 index 00000000..d025bbe3 --- /dev/null +++ b/lib/retrofit/data/set_specs_meal_list.dart @@ -0,0 +1,34 @@ +import 'package:flutter/cupertino.dart'; + +class SetSpecsMealList { + TextEditingController _groupsNameController; + List> _goodsMeal = []; + num _groupsTotal = 1; + num _optionalNum = 1; + + TextEditingController get groupsNameController => _groupsNameController; + + + List> get goodsMeal => _goodsMeal; + + set goodsMeal(List> value) { + _goodsMeal = value; + } + + + num get groupsTotal => _groupsTotal; + + set groupsTotal(num value) { + _groupsTotal = value; + } + + SetSpecsMealList(){ + _groupsNameController = TextEditingController(); + } + + num get optionalNum => _optionalNum; + + set optionalNum(num value) { + _optionalNum = value; + } +} \ No newline at end of file diff --git a/lib/retrofit/data/set_specs_value_list.dart b/lib/retrofit/data/set_specs_value_list.dart new file mode 100644 index 00000000..362ccefe --- /dev/null +++ b/lib/retrofit/data/set_specs_value_list.dart @@ -0,0 +1,10 @@ +import 'package:flutter/cupertino.dart'; + +class SetSpecsValueList { + TextEditingController _specsValueNameController; + + TextEditingController get specsValueNameController => _specsValueNameController; + SetSpecsValueList(){ + _specsValueNameController = TextEditingController(); + } +} \ No newline at end of file