基础库
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

turn_chain_util.dart 17 KiB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  1. import 'dart:io';
  2. import 'package:flutter/cupertino.dart';
  3. import 'package:flutter_alibc/alibc_const_key.dart';
  4. import 'package:flutter_alibc/alibc_model.dart';
  5. import 'package:flutter_alibc/flutter_alibc.dart';
  6. import 'package:fluttertoast/fluttertoast.dart';
  7. import 'package:jdsdk/jdsdk.dart';
  8. import 'package:url_launcher/url_launcher.dart';
  9. import 'package:zhiying_comm/models/user/user_info_model.dart';
  10. import 'package:zhiying_comm/util/dialog/loading/loading.dart';
  11. import 'package:zhiying_comm/util/empty_util.dart';
  12. import 'package:zhiying_comm/util/global_config.dart';
  13. import 'package:zhiying_comm/util/log/let_log.dart';
  14. import 'package:zhiying_comm/util/net_util.dart';
  15. import 'package:zhiying_comm/util/pdd_auth/pdd_auth.dart';
  16. import 'package:zhiying_comm/util/taobao/taobao_auth.dart';
  17. import 'package:zhiying_comm/util/turn_chain/turn_chain_dialog_repository.dart';
  18. import 'package:zhiying_comm/util/turn_chain/turn_chain_style_model.dart';
  19. import 'package:zhiying_comm/zhiying_comm.dart';
  20. import '../router_util.dart';
  21. import 'turn_chain_dialog.dart';
  22. import 'package:zhiying_comm/util/shared_prefe_util.dart';
  23. class TurnChainUtil {
  24. ///
  25. /// 跳转app或者打开url进行领券
  26. /// userInfoModel: 用户登陆的带有token 与 淘宝是否授权的 model类
  27. /// provider: 商品的渠道
  28. /// data: 转链需要的请求参数
  29. ///
  30. /// 一些常用的APP的 URL scheme:
  31. /// 来源:https://blog.csdn.net/jdazy/article/details/79208754
  32. /// QQ: mqq://
  33. /// 微信: weixin://
  34. /// 京东: openapp.jdmobile://
  35. /// 淘宝: taobao://
  36. /// 美团: imeituan://
  37. /// 点评: dianping://
  38. /// 1号店: wccbyihaodian://
  39. /// 支付宝: alipay://
  40. /// 微博: sinaweibo://
  41. /// 腾讯微博: TencentWeibo://
  42. /// weico微博: weico://
  43. /// 知乎: zhihu://
  44. /// 豆瓣fm: doubanradio://
  45. /// 网易公开课: ntesopen://
  46. /// Chrome: googlechrome://
  47. /// QQ浏览器: mqqbrowser://
  48. /// uc浏览器: ucbrowser://
  49. /// 搜狗浏览器: SogouMSE://
  50. /// 百度地图: baidumap:// bdmap://
  51. /// 优酷: youku://
  52. /// 人人: renren://
  53. /// 我查查: wcc://
  54. /// 有道词典: yddictproapp://
  55. /// 微盘: sinavdisk://
  56. /// 名片全能王: camcard://
  57. ///
  58. static Future<void> openReceiveCoupon(BuildContext context, UserInfoModel userInfoModel, String goodsId, String provider, Map<String, dynamic> data,
  59. {bool isFree = false}) async {
  60. /// iOS 审核状态
  61. // String is_ios_review = await SharedPreferencesUtil.getStringValue(GlobalConfig.IS_IOS_REVIEW, defaultVal: '0');
  62. /// 1、先判断是否登陆
  63. if (EmptyUtil.isEmpty(userInfoModel) || EmptyUtil.isEmpty(userInfoModel?.token)) {
  64. RouterUtil.goLogin(context);
  65. return;
  66. }
  67. /// 2、如果是淘宝或者天猫,判断是否授权
  68. ///
  69. /// ⚠️ 因业务修改,淘宝或者天猫每次都必须请求网络请求状态
  70. // if ((provider == GlobalConfig.PROVIDER_TB || provider == GlobalConfig.PROVIDER_TM ) && !userInfoModel.isTBAuth) {
  71. // TaobaoAuth.auth(context);
  72. // return;
  73. // }
  74. if (provider == GlobalConfig.PROVIDER_TB || provider == GlobalConfig.PROVIDER_TM) {
  75. bool isAuth = await TaobaoAuth.auth(context);
  76. if (!isAuth) return;
  77. }
  78. /// 3、获取转链,进行跳转
  79. Map<String, dynamic> result = await getTurnChainResult(context, goodsId, provider, data, isShare: false, isFree: isFree);
  80. if (!EmptyUtil.isEmpty(result)) {
  81. String openAppUrl = result['open_app_url'];
  82. String appUrl = result['app_url'];
  83. String webUrl = result['no_open_app_url'];
  84. bool jumpResult = await jumpNative(context, provider: provider, openAppUrl: openAppUrl, appUrl: appUrl, webUrl: webUrl);
  85. if (!jumpResult) {
  86. Fluttertoast.cancel();
  87. Fluttertoast.showToast(msg: '购买链接不存在');
  88. }
  89. // /// 4、根据渠道进行不同的跳转
  90. // switch (provider) {
  91. // case GlobalConfig.PROVIDER_TB:
  92. // case GlobalConfig.PROVIDER_TM:
  93. // if (!EmptyUtil.isEmpty(openAppUrl)) {
  94. // TradeResult tradeResult;
  95. // if (Platform.isAndroid) {
  96. // // print("跳转链接"+openAppUrl);
  97. // tradeResult = await FlutterAlibc.openByUrl(url: openAppUrl, backUrl: "alisdk://");
  98. // } else if (Platform.isIOS) {
  99. // if (is_ios_review == '1') {
  100. // print('iOS审核:' + is_ios_review);
  101. // RouterUtil.openWebview(webUrl, context);
  102. // } else {
  103. // tradeResult = await FlutterAlibc.openByUrl(url: openAppUrl);
  104. // }
  105. // }
  106. // Logger.debug('taobao result = ${tradeResult.errorCode} ${tradeResult.errorMessage} ');
  107. // } else if (!EmptyUtil.isEmpty(webUrl)) {
  108. // RouterUtil.openWebview(webUrl, context);
  109. // } else {
  110. // Fluttertoast.cancel();
  111. // Fluttertoast.showToast(msg: '购买链接不存在');
  112. // }
  113. //
  114. // break;
  115. // case GlobalConfig.PROVIDER_JD:
  116. // String tempURLScheme1 = 'openapp.jdmobile://virtual?params=%';
  117. // String tempURLScheme2 = 'openapp.jdmobile://';
  118. // if (!EmptyUtil.isEmpty(openAppUrl) && await canLaunch(tempURLScheme1) || await canLaunch(tempURLScheme2)) {
  119. // Jdsdk.openUrl(url: openAppUrl);
  120. // } else if (!EmptyUtil.isEmpty(webUrl)) {
  121. // RouterUtil.openWebview(webUrl, context);
  122. // } else {
  123. // Fluttertoast.cancel();
  124. // Fluttertoast.showToast(msg: '购买链接不存在');
  125. // }
  126. // break;
  127. // case GlobalConfig.PROVIDER_KL:
  128. // case GlobalConfig.PROVIDER_PDD:
  129. // case GlobalConfig.PROVIDER_SN:
  130. // bool launchable = await canLaunch(appUrl);
  131. // if (Platform.isIOS) {
  132. // launchable = await launch(appUrl);
  133. // }
  134. // if (launchable) {
  135. // if (!Platform.isIOS) {
  136. // if (appUrl.startsWith("suning")) {
  137. // RouterUtil.openWebview(webUrl, context);
  138. // } else {
  139. // RouterUtil.openWebview(webUrl, context);
  140. // }
  141. // }
  142. // } else if (!EmptyUtil.isEmpty(webUrl)) {
  143. // Logger.log('打开${provider} webUrl, url = ${webUrl}');
  144. // RouterUtil.openWebview(webUrl, context);
  145. // } else {
  146. // Fluttertoast.cancel();
  147. // Fluttertoast.showToast(msg: '购买链接不存在');
  148. // }
  149. // break;
  150. // case GlobalConfig.PROVIDER_VIP:
  151. // bool launchable = await canLaunch(appUrl);
  152. // if (Platform.isIOS) {
  153. // launchable = await launch(appUrl);
  154. // }
  155. // if (launchable) {
  156. // if (!Platform.isIOS) {
  157. // await launch(appUrl);
  158. // }
  159. // } else if (!EmptyUtil.isEmpty(webUrl)) {
  160. // Logger.log('打开${provider} webUrl, url = ${webUrl}');
  161. // RouterUtil.openWebview(webUrl, context);
  162. // } else {
  163. // Fluttertoast.cancel();
  164. // Fluttertoast.showToast(msg: '购买链接不存在');
  165. // }
  166. // break;
  167. // default:
  168. // if (!EmptyUtil.isEmpty(openAppUrl)) {
  169. // bool launchable = await canLaunch(appUrl);
  170. // if (launchable) {
  171. // launchable = await launch(appUrl);
  172. // }
  173. // if (launchable) {
  174. // if (!Platform.isIOS) {
  175. // RouterUtil.openWebview(webUrl, context);
  176. // }
  177. // } else if (!EmptyUtil.isEmpty(webUrl)) {
  178. // Logger.log('打开${provider} webUrl, url = ${webUrl}');
  179. // RouterUtil.openWebview(webUrl, context);
  180. // } else {
  181. // Fluttertoast.cancel();
  182. // Fluttertoast.showToast(msg: '购买链接不存在');
  183. // }
  184. // } else {
  185. // Fluttertoast.cancel();
  186. // Fluttertoast.showToast(msg: '购买链接不存在');
  187. // }
  188. // break;
  189. // }
  190. } else {
  191. Fluttertoast.cancel();
  192. Fluttertoast.showToast(msg: '购买链接不存在');
  193. }
  194. }
  195. ///
  196. /// 跳转原生或者webView打开购买链接
  197. ///
  198. /// provider: 商品渠道
  199. /// openAppUrl: 打开商品链接
  200. /// appUrl: 打开商品链接,不是Http开头,类似 taobao://, opennapp.jdxxx://
  201. /// webUrl: 打开商品WebView链接
  202. ///
  203. static Future<bool> jumpNative(BuildContext context, {String provider = GlobalConfig.PROVIDER_TB, String openAppUrl, String appUrl, String webUrl}) async {
  204. bool rlt = false;
  205. ///iOS 审核状态
  206. String isIosReview = await SharedPreferencesUtil.getStringValue(GlobalConfig.IS_IOS_REVIEW, defaultVal: '0');
  207. print("渠道" + provider);
  208. /// 4、根据渠道进行不同的跳转
  209. switch (provider) {
  210. case GlobalConfig.PROVIDER_TB:
  211. case GlobalConfig.PROVIDER_TM:
  212. if (!EmptyUtil.isEmpty(openAppUrl)) {
  213. TradeResult tradeResult;
  214. if (Platform.isAndroid) {
  215. tradeResult = await FlutterAlibc.openByUrl(url: openAppUrl, backUrl: "alisdk://");
  216. rlt = true;
  217. } else if (Platform.isIOS) {
  218. if (isIosReview == '1') {
  219. print('iOS审核:' + isIosReview);
  220. RouterUtil.openWebview(webUrl, context);
  221. rlt = true;
  222. } else {
  223. // IOS强制用淘宝打开
  224. tradeResult = await FlutterAlibc.openByUrl(url: openAppUrl, schemeType: AlibcSchemeType.AlibcSchemeTaoBao);
  225. rlt = true;
  226. }
  227. }
  228. Logger.debug('taobao result = ${tradeResult.errorCode} ${tradeResult.errorMessage} ');
  229. } else if (!EmptyUtil.isEmpty(webUrl)) {
  230. RouterUtil.openWebview(webUrl, context);
  231. rlt = true;
  232. } else {
  233. rlt = false;
  234. }
  235. break;
  236. case GlobalConfig.PROVIDER_JD:
  237. String tempURLScheme1 = 'openapp.jdmobile://virtual?params=%';
  238. String tempURLScheme2 = 'openapp.jdmobile://';
  239. if (!EmptyUtil.isEmpty(openAppUrl) && await canLaunch(tempURLScheme1) || await canLaunch(tempURLScheme2)) {
  240. Jdsdk.openUrl(url: openAppUrl);
  241. rlt = true;
  242. } else if (!EmptyUtil.isEmpty(webUrl)) {
  243. RouterUtil.openWebview(webUrl, context);
  244. rlt = true;
  245. } else {
  246. rlt = false;
  247. }
  248. break;
  249. case GlobalConfig.PROVIDER_KL:
  250. case GlobalConfig.PROVIDER_PDD:
  251. case GlobalConfig.PROVIDER_SN:
  252. bool launchable = await canLaunch(appUrl);
  253. if (Platform.isIOS) {
  254. launchable = await launch(appUrl);
  255. }
  256. if (launchable) {
  257. if (!Platform.isIOS) {
  258. if (provider == GlobalConfig.PROVIDER_SN) {
  259. RouterUtil.openWebview(webUrl, context);
  260. } else {
  261. if (appUrl.startsWith("http")) {
  262. RouterUtil.openWebview(webUrl, context);
  263. } else {
  264. await launch(appUrl);
  265. }
  266. }
  267. rlt = true;
  268. }
  269. } else if (!EmptyUtil.isEmpty(webUrl)) {
  270. Logger.log('打开${provider} webUrl, url = ${webUrl}');
  271. RouterUtil.openWebview(webUrl, context);
  272. rlt = true;
  273. } else {
  274. rlt = false;
  275. }
  276. break;
  277. case GlobalConfig.PROVIDER_VIP:
  278. bool launchable = await canLaunch(appUrl);
  279. if (Platform.isIOS) {
  280. launchable = await launch(appUrl);
  281. }
  282. if (launchable) {
  283. if (!Platform.isIOS) {
  284. await launch(appUrl);
  285. rlt = true;
  286. }
  287. } else if (!EmptyUtil.isEmpty(webUrl)) {
  288. Logger.log('打开${provider} webUrl, url = ${webUrl}');
  289. RouterUtil.openWebview(webUrl, context);
  290. rlt = true;
  291. } else {
  292. rlt = false;
  293. }
  294. break;
  295. default:
  296. if (!EmptyUtil.isEmpty(openAppUrl)) {
  297. bool launchable = await canLaunch(appUrl);
  298. if (launchable) {
  299. launchable = await launch(appUrl);
  300. }
  301. if (launchable) {
  302. if (!Platform.isIOS) {
  303. RouterUtil.openWebview(webUrl, context);
  304. rlt = true;
  305. }
  306. } else if (!EmptyUtil.isEmpty(webUrl)) {
  307. Logger.log('打开${provider} webUrl, url = ${webUrl}');
  308. RouterUtil.openWebview(webUrl, context);
  309. rlt = true;
  310. } else {
  311. rlt = false;
  312. }
  313. } else {
  314. rlt = false;
  315. }
  316. break;
  317. }
  318. return Future.value(rlt);
  319. }
  320. ///
  321. /// 获取分享的转链
  322. /// userInfoModel: 用户登陆的带有token 与 淘宝是否授权的 model类
  323. /// provider: 商品的渠道
  324. /// data: 转链需要的请求参数
  325. ///
  326. /// 返回参数: 只需要获取返回结果的 open_app_url 值即可。
  327. /// 例如: Map<String, dynamic> result = await getShareTurnChain(context, _user, provider, data);
  328. /// String buyUrl = result['open_app_url']
  329. ///
  330. static Future<Map<String, dynamic>> getShareTurnChain(BuildContext context, UserInfoModel userInfoModel, String goodsId, String provider, Map<String, dynamic> data) async {
  331. /// 1、先判断是否登陆
  332. if (EmptyUtil.isEmpty(userInfoModel) || EmptyUtil.isEmpty(userInfoModel?.token)) {
  333. RouterUtil.goLogin(context);
  334. return null;
  335. }
  336. /// 2、如果是淘宝,判断是否授权
  337. // if ((provider == GlobalConfig.PROVIDER_TB || provider == GlobalConfig.PROVIDER_TM) && !userInfoModel.isTBAuth) {
  338. // TaobaoAuth.auth(context);
  339. // return null;
  340. // }
  341. /// ⚠️ 因业务修改,淘宝或者天猫每次都必须请求网络请求状态
  342. if (provider == GlobalConfig.PROVIDER_TB || provider == GlobalConfig.PROVIDER_TM) {
  343. bool isAuth = await TaobaoAuth.auth(context);
  344. if (!isAuth) return null;
  345. }
  346. /// 2.5 如果是拼多多,判断是否需要授权
  347. if (provider == GlobalConfig.PROVIDER_PDD) {
  348. // 如果为true,说明需要进行授权,停止执行。
  349. bool result = await PddAuth.auth(context);
  350. if (result) return null;
  351. }
  352. /// 3、获取转链的结果
  353. Map<String, dynamic> result = await getTurnChainResult(context, goodsId, provider, data, isShare: true);
  354. if (!EmptyUtil.isEmpty(result) && !EmptyUtil.isEmpty(result['open_app_url'])) {
  355. return result;
  356. }
  357. Fluttertoast.cancel();
  358. Fluttertoast.showToast(msg: '购买链接不存在');
  359. return null;
  360. }
  361. ///
  362. /// 获取跳转的dialog样式
  363. ///
  364. static Future<dynamic> getCacheTurnChainDialogStyle(String goodsId, String provider, String commission, String coupon) async {
  365. TurnChainDialogRepository repository = TurnChainDialogRepository();
  366. var result = await repository.fetchCacheData(goodsId, provider, commission, coupon);
  367. if (!EmptyUtil.isEmpty(result)) {
  368. return result;
  369. }
  370. return null;
  371. }
  372. ///
  373. /// 缓存跳转的dialog样式
  374. ///
  375. static Future<bool> cacheTurnChainDialogStyle(String goodsId, String provider, String commission, String coupon) async {
  376. TurnChainDialogRepository repository = TurnChainDialogRepository();
  377. bool result = await repository.cacheData(goodsId, provider, commission, coupon);
  378. return result;
  379. }
  380. ///
  381. /// 接口文档:https://www.showdoc.com.cn/1003739271891029?page_id=5760575662067820
  382. /// 根据商品id等信息,获取领券或者分享的转链接
  383. ///
  384. ///
  385. static Future<Map<String, dynamic>> getTurnChainResult(BuildContext context, String goodsId, String provider, Map<String, dynamic> data,
  386. {bool isShare = false, bool isFree = false}) async {
  387. try {
  388. TurnChainDialogRepository repository = TurnChainDialogRepository();
  389. if (!EmptyUtil.isEmpty(context) && !EmptyUtil.isEmpty(provider) && !EmptyUtil.isEmpty(data) && !EmptyUtil.isEmpty('gid')) {
  390. TurnChainStyleModel model = await repository.fetchCacheData(goodsId, provider, data['commission'], data['coupon_price']);
  391. // 设置是否分享还是转链
  392. data['is_share'] = isShare ? '1' : '0';
  393. ///1为免单,0为普通
  394. data['is_free'] = (isFree ?? false) ? "1" : "0";
  395. (isFree ?? false) ? print("免单") : print("普通");
  396. if (EmptyUtil.isEmpty(model)) {
  397. // 开启loading
  398. Loading.show(context);
  399. } else {
  400. TurnChainLoading.show(context, model);
  401. }
  402. var result = await NetUtil.post('/api/v1/convert/$provider', params: data, method: NetMethod.POST);
  403. if (EmptyUtil.isEmpty(model)) {
  404. // 关闭loading
  405. Loading.dismiss();
  406. } else {
  407. TurnChainLoading.dismiss();
  408. }
  409. if (NetUtil.isSuccess(result) && !EmptyUtil.isEmpty(result[GlobalConfig.HTTP_RESPONSE_KEY_DATA])) {
  410. return result[GlobalConfig.HTTP_RESPONSE_KEY_DATA];
  411. }
  412. }
  413. } catch (e, s) {
  414. Logger.error(e, s);
  415. }
  416. return null;
  417. }
  418. }