- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我已关注 FireShip.io关于如何实现 in_app_purchase 的教程插件,因为我试图在我的应用程序中实现对消费品的购买。我已经取得了一些成功,并且已经完成了 95%,但是有一个错误我似乎无法修复。
预期:
1. 使用测试卡购买耗材,始终批准。
2. 消耗品添加到用户应用程序
3.使用测试卡购买耗材,一律拒绝。
4. 应用程序返回空或弹窗说交易被拒绝。
实际
1. 使用测试卡购买耗材,始终批准。
2. 消耗品添加到用户应用程序
3.使用测试卡购买耗材,一律拒绝。
4. 消耗品添加到用户应用程序(即代码返回最后一次购买)
本教程和我的实现之间的主要区别是他只有 1 个消耗品,而我的应用程序有 4 个。我已经包含了下面的所有代码,但我相信问题出在这种方法中:
// Returns purchase of specific product ID
PurchaseDetails _hasPurchased(List<String> productID) {
return _purchases.lastWhere(
(purchase) => productID.any((String prod) => prod == purchase.productID),
orElse: () => null);
/// Returns purchase of specific product ID
PurchaseDetails _hasPurchased(String productID) {
return _purchases.firstWhere( (purchase) => purchase.productID == productID, orElse: () => null);
}
import 'dart:io';
import 'package:find_the_treasure/models/user_model.dart';
import 'package:find_the_treasure/services/database.dart';
import 'package:find_the_treasure/widgets_common/buy_diamond_key_button.dart';
import 'package:find_the_treasure/widgets_common/platform_alert_dialog.dart';
import 'package:find_the_treasure/widgets_common/quests/diamondAndKeyContainer.dart';
import 'package:flutter/material.dart';
import 'package:in_app_purchase/in_app_purchase.dart';
import 'package:provider/provider.dart';
// App Store and Google Play consumable IDS
const String _diamond50 = 'diamond_50';
const String _diamond150 = 'diamond_150';
const String _diamond300 = 'diamond_300';
const String _diamond500 = 'diamond_500';
class ShopScreen extends StatefulWidget {
@override
_ShopScreenState createState() => _ShopScreenState();
}
class _ShopScreenState extends State<ShopScreen> {
// IAP Plugin Interface
final InAppPurchaseConnection _iap = InAppPurchaseConnection.instance;
// Updates to purchases
StreamSubscription<List<PurchaseDetails>> _subscription;
// Products for sale
List<ProductDetails> _products = [];
// Past purchases
List<PurchaseDetails> _purchases = [];
// Is the API available on the device
bool _isAvailable = false;
// Diasble button if purchase pending
bool _isPurchasePending = false;
// Consumable credits the user can buy
int _diamonds = 0;
int _keys = 0;
@override
void initState() {
_initialise();
super.initState();
}
@override
void dispose() {
_subscription.cancel();
super.dispose();
}
void _initialise() async {
// Check availilbility of In App Purchases
_isAvailable = await _iap.isAvailable();
if (_isAvailable) {
List<Future> futures = [_getProducts(), _getPastPurchases()];
await Future.wait(futures);
_verifyPurchase();
// Listen to new purchases
_subscription = _iap.purchaseUpdatedStream.listen((data) => setState(() {
_purchases.addAll(data);
_verifyPurchase();
}));
}
}
// Get all products available for sale
Future<void> _getProducts() async {
Set<String> ids =
Set.from([_diamond50, _diamond150, _diamond300, _diamond500]);
ProductDetailsResponse response = await _iap.queryProductDetails(ids);
setState(() {
_products = response.productDetails;
_products.sort((a, b) => a.title.length.compareTo(b.title.length));
});
}
// Gets past purchases
Future<void> _getPastPurchases() async {
QueryPurchaseDetailsResponse response = await _iap.queryPastPurchases();
for (PurchaseDetails purchase in response.pastPurchases) {
if (Platform.isIOS) {
InAppPurchaseConnection.instance.completePurchase(purchase);
}
}
setState(() {
_purchases = response.pastPurchases;
});
}
// Returns purchase of specific product ID
PurchaseDetails _hasPurchased(List<String> productID) {
return _purchases.lastWhere(
(purchase) => productID.any((String prod) => prod == purchase.productID),
orElse: () => null);
}
void _verifyPurchase() async {
DatabaseService _databaseService =
Provider.of<DatabaseService>(context, listen: false);
UserData _userData = Provider.of<UserData>(context, listen: false);
PurchaseDetails purchase =
_hasPurchased([_diamond50, _diamond150, _diamond300, _diamond500]);
if (purchase != null && purchase.status == PurchaseStatus.purchased) {
_isPurchasePending = true;
_getCorrect(purchase);
final _updateUserData = UserData(
displayName: _userData.displayName,
email: _userData.email,
photoURL: _userData.photoURL,
uid: _userData.uid,
userDiamondCount: _userData.userDiamondCount + _diamonds,
userKeyCount: _userData.userKeyCount + _keys);
final _didSelectOK = await PlatformAlertDialog(
title: 'Jackpot!',
content:
'I\'ll add the loot to ye treasure chest. Happy adventures.',
image: Image.asset('images/ic_thnx.png'),
defaultActionText: 'Add Loot')
.show(context);
if (_didSelectOK) {
_databaseService.updateUserDiamondAndKey(userData: _updateUserData);
_isPurchasePending = false;
}
} else if (purchase != null && purchase.status == PurchaseStatus.pending) {
_isPurchasePending = true;
PlatformAlertDialog(
title: 'Purchase Pending',
content:
'Your order is being processed, you\'ll recieve an order update very soon.',
image: Image.asset('images/ic_credit_card.png'),
defaultActionText: 'OK')
.show(context);
} else if (purchase != null && purchase.status == PurchaseStatus.error) {
_isPurchasePending = true;
PlatformAlertDialog(
title: 'Shiver Me Timbers!',
content:
'There has been an error whilst processing your payment. Please try again.',
image: Image.asset('images/ic_owl_wrong.png'),
defaultActionText: 'OK')
.show(context);
}
}
void _getCorrect(PurchaseDetails purchase) {
switch (purchase.productID) {
case _diamond50:
_diamonds = 50;
_keys = 1;
break;
case _diamond150:
_diamonds = 150;
_keys = 2;
break;
case _diamond300:
_diamonds = 300;
_keys = 3;
break;
case _diamond500:
_diamonds = 500;
_keys = 5;
break;
}
}
void _buyProduct(ProductDetails productDetails) {
final PurchaseParam purchaseParam =
PurchaseParam(productDetails: productDetails);
_iap.buyConsumable(
purchaseParam: purchaseParam,
);
}
@override
Widget build(BuildContext context) {
return SafeArea(
child: Scaffold(
appBar: AppBar(
backgroundColor: Colors.brown,
iconTheme: IconThemeData(color: Colors.black87),
actions: <Widget>[
Consumer<UserData>(
builder: (_, _userData, __) => DiamondAndKeyContainer(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
numberOfDiamonds: _userData.userDiamondCount,
numberOfKeys: _userData.userKeyCount,
color: Colors.white,
),
),
SizedBox(
width: 20,
)
],
),
body: Container(
height: MediaQuery.of(context).size.height,
width: MediaQuery.of(context).size.width,
decoration: BoxDecoration(
image: DecorationImage(
image: AssetImage(
"images/background_shop.png",
),
fit: BoxFit.fill,
),
),
child: _isAvailable
? Column(mainAxisAlignment: MainAxisAlignment.start, children: [
// Image.asset(),
SizedBox(
height: 10,
),
IconButton(
icon: Icon(
Icons.help_outline,
color: Colors.brown.shade300,
size: 40,
),
tooltip: 'Store questions',
onPressed: () async {
final _didSelectOK = await PlatformAlertDialog(
title: 'Welcome to The Shop',
content:
'Stock ye treasure chest with diamonds and keys. Use \'em to unlock quests and ye can also trade \'em for hints!',
image: Image.asset('images/ic_thnx.png'),
defaultActionText: 'OK')
.show(context);
if (_didSelectOK) {}
},
),
SizedBox(
height: 10,
),
for (var prod in _products)
BuyDiamondOrKeyButton(
numberOfDiamonds: numberOfDiamonds(prod.price),
diamondCost: prod.price,
bonusKey: numberOfKeys(prod.price),
onPressed: () => _buyProduct(prod),
isPending: _isPurchasePending,
),
// BuyTreasureChest()
])
: Center(
child: Container(
color: Colors.white.withOpacity(0.5),
height: MediaQuery.of(context).size.height / 3,
width: double.infinity,
margin: EdgeInsets.symmetric(horizontal: 15),
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
Text('Loading Store...'),
SizedBox(
height: 20,
),
Image.asset(
'images/compass.gif',
height: 200,
)
],
),
))),
),
);
}
String numberOfDiamonds(String productPrice) {
String _numberOfDiamonds;
switch (productPrice) {
case '\$4.99':
_numberOfDiamonds = '50';
break;
case '\$9.99':
_numberOfDiamonds = '150';
break;
case '\$19.99':
_numberOfDiamonds = '300';
break;
case '\$39.99':
_numberOfDiamonds = '500';
break;
default:
_numberOfDiamonds = '50';
}
return _numberOfDiamonds;
}
String numberOfKeys(String productPrice) {
String _numberOfKeys;
switch (productPrice) {
case '\$4.99':
_numberOfKeys = '1';
break;
case '\$9.99':
_numberOfKeys = '2';
break;
case '\$19.99':
_numberOfKeys = '3';
break;
case '\$39.99':
_numberOfKeys = '5';
break;
default:
_numberOfKeys = 'err';
}
return _numberOfKeys;
}
}
class BuyTreasureChest extends StatelessWidget {
const BuyTreasureChest({
Key key,
}) : super(key: key);
@override
Widget build(BuildContext context) {
return Container(
padding: EdgeInsets.symmetric(vertical: 10),
width: MediaQuery.of(context).size.width * 0.9,
child: Row(mainAxisAlignment: MainAxisAlignment.spaceEvenly, children: [
Column(
children: <Widget>[
Image.asset(
'images/chest.png',
height: 40,
),
],
),
Row(
children: <Widget>[
Text('? x', style: TextStyle(color: Colors.white, fontSize: 25)),
SizedBox(width: 8),
Image.asset(
'images/2.0x/ic_diamond.png',
height: 20,
),
],
),
Row(
children: <Widget>[
Text('? x', style: TextStyle(color: Colors.white, fontSize: 25)),
Image.asset(
'images/skull_key_outline.png',
height: 30,
),
],
),
Container(
padding: EdgeInsets.symmetric(vertical: 5, horizontal: 24),
decoration: BoxDecoration(
border: Border.all(width: 1, color: Colors.grey),
borderRadius: BorderRadius.circular(5),
color: Colors.grey.shade800),
child: Text(
'\$9.99',
textAlign: TextAlign.center,
style: TextStyle(color: Colors.white, fontSize: 15),
),
),
]),
decoration: BoxDecoration(
color: Colors.grey.shade800,
border: Border.all(color: Colors.amberAccent, width: 2),
borderRadius: BorderRadius.circular(35)),
);
}
}
最佳答案
你需要添加这个:
InAppPurchaseConnection.completePurchase
关于flutter - Flutter in_app_purchase 插件的正确实现多耗材,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61166125/
我正在尝试从flutter应用程序构建apk,但出现此错误: Note: /mnt/Software/Linux/Flutter/flutter/.pub-cache/hosted/pub.dartl
我有一个名为 X 的较大应用程序,还有另一个名为 Y 的较小应用程序。他们现在彼此分开,并且工作正常。我想将应用程序 Y 集成到 X 中。我想将 Y 的代码放入 X 项目中,但它们应该有不同的 Mai
在android Studio中选择Create New Flutter Project,出现如下4个选项。 Flutter 应用程序 Flutter 插件 Flutter 包 flutter 模块
我看到我的 flutter 项目生成了一个文件 ios/Flutter/Flutter.podspec ,这个文件有什么用? 如果它与生成的 Flutter.framework 有关? 我应该将它包含
我尝试过的 在包含flutter SDK的位置添加/编辑.bash_profile.rtf文件。 导出PATH = / Users / temur / Documents / Projects / F
Flutter 日志会打印数千个详细/垃圾邮件日志。 我正在尝试调试一个复杂的应用程序,但是打印太冗长了,以至于我很难找到我自己打印的东西。 有没有办法禁用详细? 就像是: Logger.level.
在flutter 1.22更新之后,我在Lineargradient colors属性中遇到错误,这给我一个错误,即未定义名称colors参数。.在Android中更新flutter和flutter插
在下面的代码 widget.hintText 中给出错误,我试图将日期选择器作为单独的组件,并在从其他文件调用它时动态传递提示文本值。 import 'package:date_field/date_
在下面的代码 widget.hintText 中给出错误,我试图将日期选择器作为单独的组件,并在从其他文件调用它时动态传递提示文本值。 import 'package:date_field/date_
Flutter 1.0 发布后,我正在按照步骤搭建 Flutter 开发环境。 在步骤中(如所附屏幕截图所示),它说要更新 $PATH 两次,一次使用 flutter 工具的路径 export PAT
我有一个用 flutter 编写的移动应用程序,我想将其转换为 flutter_web 应用程序(集成 flutter_web 尚不可用)。我目前遇到包裹问题。 我已按照本网站中列出的说明进行操作 h
如何向我的 Flutter 路由添加自定义转换?这是我目前的路线结构。 class MyApp extends StatelessWidget { // This widget is the
我正在尝试通过 URL 在 webview 中显示网页。我试过 flutter_webview_plugin 插件,但是当我在浏览器上运行项目时它不起作用。 在 flutter web applica
我正在使用 animatedContainer 在按下按钮时显示 listView.builder()。这是一个要显示的简单子(monad)列表,但问题是我不知道 ListView 构建器的高度会传递
我目前正在我的应用程序中制作渐变背景动画......我正在使用 lottie 动画的帮助下这样做!我试图将它封装在一个容器中并成功地做到了。但是有一个问题,尽管我将高度更改为大于 2000 的东西,但
美好的一天! 我无法弄清楚如何使用 google 标签管理器设置 flutter。我找到了 this package包括标签管理器 api。但是我不知道如何正确配置它。 (在网络上我只需要复制粘贴一个
我的购物车模型如下 class Cart { String description; double unitCost; double amount; int quantity; S
在 Flutter 应用程序中,我想为在线托管的资源(图像、视频等)实现缓存。 我希望它能在原生平台 (Android/iOS)(例如使用文件系统)和网络(例如使用 IndexedDB)上运行。 Fl
我写了一个页面,在顶部一切都很好,应该是这样。在底部我有一个事件的历史。我的容器的宽度是自动确定的(取决于屏幕的宽度),而高度 - 不,在不同的设备上有不同的高度(底部的缩进是不同的)。是否可以自动确
我正在处理一个页面,其中有一些字段,例如 textfield 和 slider。在页面的末尾必须有一个用于进行下一步的按钮,该按钮被包裹在 Align 中以在页面之间具有固定位置。 另一方面,resi
我是一名优秀的程序员,十分优秀!