SKPayment Ошибка, SKProduct продолжает возвращаться ноль
Я работаю в Spritekit и пытаюсь включить SKPayments в моей игре, чтобы дать пользователю 5 жизней после совершения покупки. Но каждый раз, когда я запускаю приложение и пытаюсь совершить покупку, я получаю ошибку, что мой SKProduct FiveLives равен нулю. Код и строка, по которой я продолжаю получать сообщение об ошибке, приведены ниже. Может кто-нибудь, пожалуйста, помогите мне?
self.product1ID = @"com.retrogames.5Lives";
[[SKPaymentQueue defaultQueue] addTransactionObserver:self];
[self getProduct1ID:self];
SKPayment *payment = [SKPayment paymentWithProduct:_FiveLives];
ERROR LINE [[SKPaymentQueue defaultQueue] addPayment:payment];
}
Остальная часть моего класса здесь
StoreViewController.h
#import <UIKit/UIKit.h>
#import <StoreKit/StoreKit.h>
@interface StoreViewController : UIViewController <SKPaymentTransactionObserver, SKProductsRequestDelegate>{
SKProductsRequest *productsRequest;
}
@property (strong, nonatomic) SKProduct *FiveLives;
@property (strong, nonatomic) NSString *product1ID;
- (IBAction)_5Lives:(id)sender;
-(void)getProduct1ID:(UIViewController *)viewcontroller;
@end
StoreViewController.M
#import "StoreViewController.h"
@interface StoreViewController ()
@end
@implementation StoreViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view.
}
- (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
-(void)getProduct1ID:(UIViewController *)viewcontroller{
if ([SKPaymentQueue canMakePayments]) {
SKProductsRequest *request = [[SKProductsRequest alloc]initWithProductIdentifiers:[NSSet setWithObject:self.product1ID]];
request.delegate = self;
[request start];
}
else{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Sorry purchase was unsuccessful" message:@"Please enable in app purchases in your settings" delegate:self cancelButtonTitle:nil otherButtonTitles:nil, nil];
[alert show];
}
}
#pragma mark _
#pragma mark SKProductsRequestDelegate
-(void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response {
NSArray *Products = response.products;
if (Products.count != 0) {
_FiveLives = Products[0];
}
Products = response.invalidProductIdentifiers;
for (SKProduct *FiveLives in Products) {
NSLog(@"Product Not Found : %@", FiveLives);
}
}
- (IBAction)_5Lives:(id)sender {
self.product1ID = @"com.retrogames.5Lives";
[[SKPaymentQueue defaultQueue] addTransactionObserver:self];
[self getProduct1ID:self];
SKPayment *payment = [SKPayment paymentWithProduct:_FiveLives];
[[SKPaymentQueue defaultQueue] addPayment:payment];
}
- (IBAction)RestorePurchases:(id)sender{
[[SKPaymentQueue defaultQueue] addTransactionObserver:self];
[[SKPaymentQueue defaultQueue]restoreCompletedTransactions];
}
-(void)add5Lives{
NSLog(@"BUY5LIVES");
GameLives = GameLives + 5;
[[NSUserDefaults standardUserDefaults] setInteger:GameLives forKey:@"Key"];
}
-(void)paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray *)transactions{
for (SKPaymentTransaction *transaction in transactions) {
switch (transaction.transactionState) {
case SKPaymentTransactionStatePurchased:[self add5Lives];{
NSLog(@"TRANSACTION SUCCESSFUL");
[[SKPaymentQueue defaultQueue] finishTransaction:transaction];
break;
}
case SKPaymentTransactionStateFailed:NSLog(@"TRANSACTION FAILED");{
[[SKPaymentQueue defaultQueue] finishTransaction:transaction];
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Payment was unsuccessful" message:@"Payment was unable to be processed." delegate:self cancelButtonTitle:nil otherButtonTitles:nil, nil];
[alert show];
}
break;
default:
break;
}
}
}
Новый код после того, как я использовал учебник
PURCHASELIVES
PurchaseLives.m
+ (PurchaseLives *)sharedInstance {
static dispatch_once_t once;
static PurchaseLives * sharedInstance;
dispatch_once(&once, ^{
NSSet * productIdentifiers = [NSSet setWithObjects:
@"com.retrogames.5Lives",
nil];
sharedInstance = [[self alloc] initWithProductIdentifiers:productIdentifiers];
});
return sharedInstance;
}
@end
IAPHelper
IAPHelper.m
NSString *const IAPHelperProductPurchasedNotification = @"IAPHelperProductPurchasedNotification";
@interface IAPHelper () <SKProductsRequestDelegate, SKPaymentTransactionObserver>
@end
@implementation IAPHelper {
// 3
SKProductsRequest * _productsRequest;
// 4
RequestProductsCompletionHandler _completionHandler;
NSSet * _productIdentifiers;
NSMutableSet * _purchasedProductIdentifiers;
}
- (id)initWithProductIdentifiers:(NSSet *)productIdentifiers {
if ((self = [super init])) {
// Store product identifiers
_productIdentifiers = productIdentifiers;
// Check for previously purchased products
_purchasedProductIdentifiers = [NSMutableSet set];
for (NSString * productIdentifier in _productIdentifiers) {
BOOL productPurchased = [[NSUserDefaults standardUserDefaults] boolForKey:productIdentifier];
if (productPurchased) {
[_purchasedProductIdentifiers addObject:productIdentifier];
NSLog(@"Previously purchased: %@", productIdentifier);
} else {
NSLog(@"Not purchased: %@", productIdentifier);
}
}
[[SKPaymentQueue defaultQueue] addTransactionObserver:self];
}
return self;
}
- (void)requestProductsWithCompletionHandler:(RequestProductsCompletionHandler)completionHandler {
// 1
_completionHandler = [completionHandler copy];
// 2
_productsRequest = [[SKProductsRequest alloc] initWithProductIdentifiers:_productIdentifiers];
_productsRequest.delegate = self;
[_productsRequest start];
}
#pragma mark - SKProductsRequestDelegate
- (void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response {
NSLog(@"Loaded list of products...");
_productsRequest = nil;
NSArray * skProducts = response.products;
for (SKProduct * skProduct in skProducts) {
NSLog(@"Found product: %@ %@ %0.2f",
skProduct.productIdentifier,
skProduct.localizedTitle,
skProduct.price.floatValue);
FiveLives = skProducts[0];
NSLog(@"Products = %@",skProducts);
}
_completionHandler(YES, skProducts);
_completionHandler = nil;
}
- (void)request:(SKRequest *)request didFailWithError:(NSError *)error {
NSLog(@"Failed to load list of products.");
_productsRequest = nil;
_completionHandler(NO, nil);
_completionHandler = nil;
}
- (BOOL)productPurchased:(NSString *)productIdentifier {
return [_purchasedProductIdentifiers containsObject:productIdentifier];
}
- (void)buyProduct:(SKProduct *)product {
NSLog(@"Buying %@...", FiveLives.productIdentifier);
SKPayment * payment = [SKPayment paymentWithProduct:FiveLives];
[[SKPaymentQueue defaultQueue] addPayment:payment];
}
- (void)paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray *)transactions
{
for (SKPaymentTransaction * transaction in transactions) {
switch (transaction.transactionState)
{
case SKPaymentTransactionStatePurchased:
[self completeTransaction:transaction];
break;
case SKPaymentTransactionStateFailed:
[self failedTransaction:transaction];
break;
case SKPaymentTransactionStateRestored:
[self restoreTransaction:transaction];
default:
break;
}
}
}
// Add new method
- (void)provideContentForProductIdentifier:(NSString *)productIdentifier {
[_purchasedProductIdentifiers addObject:productIdentifier];
[[NSUserDefaults standardUserDefaults] setBool:YES forKey:productIdentifier];
[[NSUserDefaults standardUserDefaults] synchronize];
[[NSNotificationCenter defaultCenter] postNotificationName:IAPHelperProductPurchasedNotification object:productIdentifier userInfo:nil];
}
- (void)completeTransaction:(SKPaymentTransaction *)transaction {
NSLog(@"completeTransaction...");
[self provideContentForProductIdentifier:transaction.payment.productIdentifier];
[[SKPaymentQueue defaultQueue] finishTransaction:transaction];
}
- (void)restoreTransaction:(SKPaymentTransaction *)transaction {
NSLog(@"restoreTransaction...");
[self provideContentForProductIdentifier:transaction.originalTransaction.payment.productIdentifier];
[[SKPaymentQueue defaultQueue] finishTransaction:transaction];
}
- (void)failedTransaction:(SKPaymentTransaction *)transaction {
NSLog(@"failedTransaction...");
if (transaction.error.code != SKErrorPaymentCancelled)
{
NSLog(@"Transaction error: %@", transaction.error.localizedDescription);
}
[[SKPaymentQueue defaultQueue] finishTransaction: transaction];
}
@end
BUYLIVES
#import "BuyLivesViewController.h"
#import "PurchaseLives.h"
#import "IAP Helper.h"
@interface BuyLivesViewController (){
NSArray *_products;
}
@end
@implementation BuyLivesViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view.
}
- (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
- (void)viewWillAppear:(BOOL)animated {
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(productPurchased:) name:IAPHelperProductPurchasedNotification object:nil];
}
- (void)viewWillDisappear:(BOOL)animated {
[[NSNotificationCenter defaultCenter] removeObserver:self];
}
- (IBAction)Buy5Lives:(id)sender {
[[PurchaseLives sharedInstance] buyProduct:FiveLives];
}
- (void)productPurchased:(NSNotification *)notification {
NSString * productIdentifier = notification.object;
[_products enumerateObjectsUsingBlock:^(SKProduct * product, NSUInteger idx, BOOL *stop) {
if ([product.productIdentifier isEqualToString:productIdentifier]) {
GameLives = GameLives + 5;
}
}];
}
@end
1 ответ
Проблема здесь:
[self getProduct1ID:self];
SKPayment *payment = [SKPayment paymentWithProduct:_FiveLives];
getProduct1ID
выдает запрос на получение всех товаров из магазина. Однако этот запрос не является блокирующим. это означает, что к моменту завершения запроса вы уже звоните paymentWithProduct
, И в этот раз _FiveLives
еще не было установлено с действующим продуктом, и это будет nil
,
То, что я рекомендую вам сделать, это позвонить getProduct1ID
в начале вашего приложения (я делаю это в методе делегата приложения) application:didFinishLaunchingWithOptions:
) и сохраните продукт как вы в переменной-члене, которую вы можете использовать позже.
Затем, когда пользователь нажимает кнопку покупки, просто используйте экземпляр продукта для совершения покупки.
Я также рекомендую вам посмотреть здесь и посмотреть, как происходит поток IAP. Отличный учебник, за которым я следовал сам, и он отлично работает.