INFRA: Set Up Project.
This commit is contained in:
339
lib/payment/orangePayScreen.dart
Normal file
339
lib/payment/orangePayScreen.dart
Normal file
@@ -0,0 +1,339 @@
|
||||
// ignore_for_file: must_be_immutable
|
||||
|
||||
import 'dart:async';
|
||||
import 'dart:convert';
|
||||
import 'package:customer/models/payment_model/orange_money.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:get/get.dart';
|
||||
import 'package:http/http.dart' as http;
|
||||
import 'package:webview_flutter/webview_flutter.dart';
|
||||
|
||||
class OrangeMoneyScreen extends StatefulWidget {
|
||||
String initialURl;
|
||||
OrangeMoney orangePay;
|
||||
String accessToken = '';
|
||||
String payToken = '';
|
||||
String orderId = '';
|
||||
String amount = '';
|
||||
|
||||
OrangeMoneyScreen({super.key, required this.initialURl, required this.orangePay, required this.accessToken, required this.payToken, required this.orderId, required this.amount});
|
||||
|
||||
@override
|
||||
State<OrangeMoneyScreen> createState() => _OrangeMoneyScreenState();
|
||||
}
|
||||
|
||||
class _OrangeMoneyScreenState extends State<OrangeMoneyScreen> {
|
||||
WebViewController controller = WebViewController();
|
||||
bool isLoading = true;
|
||||
Timer? timer;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
controller.clearCache();
|
||||
initController();
|
||||
startTransactionPolling();
|
||||
super.initState();
|
||||
}
|
||||
|
||||
// 🔹 Poll Orange API every 3 seconds to check status
|
||||
void startTransactionPolling() {
|
||||
timer = Timer.periodic(const Duration(seconds: 3), (Timer t) async {
|
||||
if (!mounted) return;
|
||||
|
||||
String status = await transactionStatus(accessToken: widget.accessToken, amount: widget.amount, orderId: widget.orderId, payToken: widget.payToken);
|
||||
|
||||
if (status == 'SUCCESS') {
|
||||
timer?.cancel();
|
||||
debugPrint('✅ Payment successful for Order ID: ${widget.orderId}');
|
||||
Get.back(result: true);
|
||||
} else if (status == 'FAILED' || status == 'CANCELLED') {
|
||||
timer?.cancel();
|
||||
debugPrint('❌ Payment failed or cancelled.');
|
||||
Get.back(result: false);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
void initController() {
|
||||
controller =
|
||||
WebViewController()
|
||||
..setJavaScriptMode(JavaScriptMode.unrestricted)
|
||||
..setBackgroundColor(const Color(0x00000000))
|
||||
..setNavigationDelegate(
|
||||
NavigationDelegate(
|
||||
onPageFinished: (url) {
|
||||
setState(() {
|
||||
isLoading = false;
|
||||
});
|
||||
},
|
||||
),
|
||||
)
|
||||
..loadRequest(Uri.parse(widget.initialURl));
|
||||
}
|
||||
|
||||
Future<String> transactionStatus({required String orderId, required String amount, required String payToken, required String accessToken}) async {
|
||||
String apiUrl = widget.orangePay.isSandbox == true ? 'https://api.orange.com/orange-money-webpay/dev/v1/transactionstatus' : 'https://api.orange.com/orange-money-webpay/cm/v1/transactionstatus';
|
||||
|
||||
Map<String, String> requestBody = {"order_id": orderId, "amount": amount, "pay_token": payToken};
|
||||
|
||||
try {
|
||||
var response = await http.post(
|
||||
Uri.parse(apiUrl),
|
||||
headers: {'Authorization': 'Bearer $accessToken', 'Content-Type': 'application/json', 'Accept': 'application/json'},
|
||||
body: json.encode(requestBody),
|
||||
);
|
||||
|
||||
if (response.statusCode == 201) {
|
||||
Map<String, dynamic> responseData = jsonDecode(response.body);
|
||||
debugPrint('🔍 Transaction Status: ${responseData['status']}');
|
||||
return responseData['status'];
|
||||
} else {
|
||||
return '';
|
||||
}
|
||||
} catch (e) {
|
||||
debugPrint('⚠️ Transaction check error: $e');
|
||||
return '';
|
||||
}
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return WillPopScope(
|
||||
onWillPop: () async {
|
||||
_showCancelDialog();
|
||||
return false;
|
||||
},
|
||||
child: Scaffold(
|
||||
appBar: AppBar(
|
||||
backgroundColor: Colors.black,
|
||||
leading: IconButton(icon: const Icon(Icons.arrow_back, color: Colors.white), onPressed: _showCancelDialog),
|
||||
title: Text('Orange Money Payment'.tr),
|
||||
),
|
||||
body: isLoading ? const Center(child: CircularProgressIndicator()) : WebViewWidget(controller: controller),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
Future<void> _showCancelDialog() async {
|
||||
return showDialog<void>(
|
||||
context: context,
|
||||
barrierDismissible: true,
|
||||
builder: (BuildContext context) {
|
||||
return AlertDialog(
|
||||
title: Text('Cancel Payment'.tr),
|
||||
content: Text('Are you sure you want to cancel this payment?'.tr),
|
||||
actions: [
|
||||
TextButton(child: Text('No', style: TextStyle(color: Colors.green)), onPressed: () => Get.back()),
|
||||
TextButton(
|
||||
child: Text('Yes'.tr, style: TextStyle(color: Colors.red)),
|
||||
onPressed: () {
|
||||
timer?.cancel();
|
||||
Get.back(); // close dialog
|
||||
Get.back(result: false); // close WebView and mark as failed
|
||||
},
|
||||
),
|
||||
],
|
||||
);
|
||||
},
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
timer?.cancel();
|
||||
super.dispose();
|
||||
}
|
||||
}
|
||||
|
||||
// // ignore_for_file: must_be_immutable
|
||||
//
|
||||
// import 'dart:async';
|
||||
// import 'dart:convert';
|
||||
// import 'package:customer/models/payment_model/orange_money.dart';
|
||||
// import 'package:flutter/material.dart';
|
||||
// import 'package:get/get.dart';
|
||||
// import 'package:http/http.dart' as http;
|
||||
// import 'package:webview_flutter/webview_flutter.dart';
|
||||
//
|
||||
// class OrangeMoneyScreen extends StatefulWidget {
|
||||
// String initialURl;
|
||||
// OrangeMoney orangePay;
|
||||
// String accessToken = '';
|
||||
// String payToken = '';
|
||||
// String orderId = '';
|
||||
// String amount = '';
|
||||
//
|
||||
// OrangeMoneyScreen({
|
||||
// super.key,
|
||||
// required this.initialURl,
|
||||
// required this.orangePay,
|
||||
// required this.accessToken,
|
||||
// required this.payToken,
|
||||
// required this.orderId,
|
||||
// required this.amount,
|
||||
// });
|
||||
//
|
||||
// @override
|
||||
// State<OrangeMoneyScreen> createState() => _OrangeMoneyScreenState();
|
||||
// }
|
||||
//
|
||||
// class _OrangeMoneyScreenState extends State<OrangeMoneyScreen> {
|
||||
// WebViewController controller = WebViewController();
|
||||
// bool isLoading = true;
|
||||
//
|
||||
// @override
|
||||
// void initState() {
|
||||
// controller.clearCache();
|
||||
// initController();
|
||||
// callTransaction();
|
||||
// super.initState();
|
||||
// }
|
||||
//
|
||||
// Timer? timer;
|
||||
//
|
||||
// void callTransaction() {
|
||||
// timer = Timer.periodic(const Duration(seconds: 3), (Timer t) {
|
||||
// if (mounted) {
|
||||
// transactionstatus(accessToken: widget.accessToken, amount: widget.amount, orderId: widget.orderId, payToken: widget.payToken).then((value) {
|
||||
// if (value == 'SUCCESS') {
|
||||
// if (timer != null) {
|
||||
// timer!.cancel();
|
||||
// }
|
||||
// Get.back(result: true);
|
||||
// } else if (value == 'FAILED') {
|
||||
// if (timer != null) {
|
||||
// timer!.cancel();
|
||||
// }
|
||||
// Get.back(result: false);
|
||||
// }
|
||||
// });
|
||||
// }
|
||||
// });
|
||||
// }
|
||||
//
|
||||
// void initController() {
|
||||
// controller = WebViewController()
|
||||
// ..setJavaScriptMode(JavaScriptMode.unrestricted)
|
||||
// ..setBackgroundColor(const Color(0x00000000))
|
||||
// ..setNavigationDelegate(
|
||||
// NavigationDelegate(
|
||||
// onPageFinished: ((url) {
|
||||
// setState(() {
|
||||
// isLoading = false;
|
||||
// });
|
||||
// }),
|
||||
// onNavigationRequest: (NavigationRequest navigation) async {
|
||||
// return NavigationDecision.navigate;
|
||||
// },
|
||||
// ),
|
||||
// )
|
||||
// ..loadRequest(Uri.parse(widget.initialURl));
|
||||
// }
|
||||
//
|
||||
// Future transactionstatus({
|
||||
// required String orderId,
|
||||
// required String amount,
|
||||
// required String payToken,
|
||||
// required String accessToken,
|
||||
// }) async {
|
||||
// String apiUrl = widget.orangePay.isSandbox == true
|
||||
// ? 'https://api.orange.com/orange-money-webpay/dev/v1/transactionstatus'
|
||||
// : 'https://api.orange.com/orange-money-webpay/cm/v1/transactionstatus';
|
||||
// Map<String, String> requestBody = {
|
||||
// "order_id": orderId,
|
||||
// "amount": amount, // "OUV",
|
||||
// "pay_token": payToken
|
||||
// };
|
||||
//
|
||||
// var response = await http.post(Uri.parse(apiUrl),
|
||||
// headers: <String, String>{
|
||||
// 'Authorization': 'Bearer $accessToken',
|
||||
// 'Content-Type': 'application/json',
|
||||
// 'Accept': 'application/json',
|
||||
// },
|
||||
// body: json.encode(requestBody));
|
||||
//
|
||||
// // Handle the response
|
||||
// if (response.statusCode == 201) {
|
||||
// Map<String, dynamic> responseData = jsonDecode(response.body);
|
||||
// return responseData['status'];
|
||||
// } else {
|
||||
// return '';
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// @override
|
||||
// Widget build(BuildContext context) {
|
||||
// // ignore: deprecated_member_use
|
||||
// return WillPopScope(
|
||||
// onWillPop: () async {
|
||||
// _showMyDialog();
|
||||
// return false;
|
||||
// },
|
||||
// child: Scaffold(
|
||||
// appBar: AppBar(
|
||||
// backgroundColor: Colors.black,
|
||||
// centerTitle: false,
|
||||
// leading: GestureDetector(
|
||||
// onTap: () {
|
||||
// _showMyDialog();
|
||||
// },
|
||||
// child: const Icon(
|
||||
// Icons.arrow_back,
|
||||
// color: Colors.white,
|
||||
// ),
|
||||
// )),
|
||||
// body: isLoading
|
||||
// ? const Center(
|
||||
// child: CircularProgressIndicator(),
|
||||
// )
|
||||
// : WebViewWidget(controller: controller),
|
||||
// ),
|
||||
// );
|
||||
// }
|
||||
//
|
||||
// Future<void> _showMyDialog() async {
|
||||
// return showDialog<void>(
|
||||
// context: context,
|
||||
// barrierDismissible: true, // user must tap button!
|
||||
// builder: (BuildContext context) {
|
||||
// return AlertDialog(
|
||||
// title: Text('Cancel Payment'.tr),
|
||||
// content: SingleChildScrollView(
|
||||
// child: Text("cancelPayment?".tr),
|
||||
// ),
|
||||
// actions: <Widget>[
|
||||
// TextButton(
|
||||
// child: Text(
|
||||
// 'Cancel'.tr,
|
||||
// style: const TextStyle(color: Colors.red),
|
||||
// ),
|
||||
// onPressed: () {
|
||||
// Get.back(result: false);
|
||||
// Get.back(result: false);
|
||||
// },
|
||||
// ),
|
||||
// TextButton(
|
||||
// child: Text(
|
||||
// 'Continue'.tr,
|
||||
// style: const TextStyle(color: Colors.green),
|
||||
// ),
|
||||
// onPressed: () {
|
||||
// Get.back(result: false);
|
||||
// },
|
||||
// ),
|
||||
// ],
|
||||
// );
|
||||
// },
|
||||
// );
|
||||
// }
|
||||
//
|
||||
// @override
|
||||
// void dispose() {
|
||||
// if (timer != null) {
|
||||
// timer!.cancel();
|
||||
// }
|
||||
// // TODO: implement dispose
|
||||
// super.dispose();
|
||||
// }
|
||||
// }
|
||||
Reference in New Issue
Block a user