покупка в приложении в Swift с помощью одного продукта
есть ли простой способ реализовать покупку в приложении в swift для одного продукта?
Я хочу кнопку, которая запускает покупку в приложении, как [ad-removal] или [unlock-premium-content]
Я не могу понять всю логику.
Я пытаюсь следовать и перевести этот учебник из [Techotopia] http://www.techotopia.com/index.php/An_iOS_7_In-App_Purchase_Tutorial
но это мой первый раз с StoreKit Рамки, а также с Swift.
Я просто хочу знать логику транзакции покупки в приложении с помощью платформы Swift StoreKit.
спасибо!
3 ответов
Шаг 0: в вашей учетной записи iTunes Connect создайте покупку в приложении.
для одиночных покупок используйте этот метод:
- импорт
import StoreKit
- соответствует делегату StoreKit
class YOURViewController: UIViewController, SKProductsRequestDelegate, SKPaymentTransactionObserver {
- используйте пользователя по умолчанию для отслеживания транзакций
let defaults = NSUserDefaults.standardUserDefaults()
- код продукта. Это будет так же, как в вашем iTunes Connect в покупке приложения
var product_id: NSString?
override func viewDidLoad() {
product_id = "YOUR_PRODUCT_ID"
super.viewDidLoad()
SKPaymentQueue.defaultQueue().addTransactionObserver(self)
//Check if product is purchased
if (defaults.boolForKey("purchased")){
// Hide a view or show content depends on your requirement
overlayView.hidden = true
} else if (!defaults.boolForKey("stonerPurchased")) {
print("false")
}
}
- Контент Разблокировки. Это действие кнопки, которое инициализирует покупку
@IBAction func unlockAction(sender: AnyObject) {
print("About to fetch the products")
// We check that we are allow to make the purchase.
if (SKPaymentQueue.canMakePayments()) {
var productID:NSSet = NSSet(object: self.product_id!);
var productsRequest:SKProductsRequest = SKProductsRequest(productIdentifiers: productID);
productsRequest.delegate = self;
productsRequest.start();
println("Fetching Products");
} else {
print("can't make purchases");
}
}
- Вспомогательные Методы
func buyProduct(product: SKProduct) {
println("Sending the Payment Request to Apple");
var payment = SKPayment(product: product)
SKPaymentQueue.defaultQueue().addPayment(payment);
}
- методы делегирования для IAP
func productsRequest (request: SKProductsRequest, didReceiveResponse response: SKProductsResponse) {
var count : Int = response.products.count
if (count>0) {
var validProducts = response.products
var validProduct: SKProduct = response.products[0] as SKProduct
if (validProduct.productIdentifier == self.product_id) {
print(validProduct.localizedTitle)
print(validProduct.localizedDescription)
print(validProduct.price)
buyProduct(validProduct);
} else {
print(validProduct.productIdentifier)
}
} else {
print("nothing")
}
}
func request(request: SKRequest!, didFailWithError error: NSError!) {
print("Error Fetching product information");
}
func paymentQueue(queue: SKPaymentQueue!, updatedTransactions transactions: [AnyObject]!) {
print("Received Payment Transaction Response from Apple");
for transaction:AnyObject in transactions {
if let trans:SKPaymentTransaction = transaction as? SKPaymentTransaction{
switch trans.transactionState {
case .Purchased:
print("Product Purchased");
SKPaymentQueue.defaultQueue().finishTransaction(transaction as SKPaymentTransaction)
defaults.setBool(true , forKey: "purchased")
overlayView.hidden = true
break;
case .Failed:
print("Purchased Failed");
SKPaymentQueue.defaultQueue().finishTransaction(transaction as SKPaymentTransaction)
break;
case .Restored:
print("Already Purchased");
SKPaymentQueue.defaultQueue().restoreCompletedTransactions()
default:
break;
}
}
}
}
Swift > 3.0
import StoreKit
class YOURVIEWController:UIViewController, SKProductsRequestDelegate, SKPaymentTransactionObserver {
let product_id: NSString = "com.some.inappid" // <!-- Change it to your inapp id
в своем viewDidLoad
добавить
override func viewDidLoad() {
super.viewDidLoad()
SKPaymentQueue.default().add(self)
в вашем действии кнопки покупки
@IBAction func buyNowAction(_ sender: UIButton) {
if (SKPaymentQueue.canMakePayments()) {
let productID:NSSet = NSSet(array: [self.product_id as NSString]);
let productsRequest:SKProductsRequest = SKProductsRequest(productIdentifiers: productID as! Set<String>);
productsRequest.delegate = self;
productsRequest.start();
print("Fetching Products");
} else {
print("can't make purchases");
}
}
в кнопку Восстановить действие
// MARK: - Restore In App Purchase
@IBAction func restoreAction(_ sender: UIButton) {
if (SKPaymentQueue.canMakePayments()) {
SKPaymentQueue.default().add(self)
SKPaymentQueue.default().restoreCompletedTransactions()
} else {
// show error
}
}
Добавить Делегатов:
// SKProductRequest Delegate
func productsRequest(_ request: SKProductsRequest, didReceive response: SKProductsResponse) {
print(response.products)
let count : Int = response.products.count
if (count>0) {
let validProduct: SKProduct = response.products[0] as SKProduct
if (validProduct.productIdentifier == self.product_id as String) {
print(validProduct.localizedTitle)
print(validProduct.localizedDescription)
print(validProduct.price)
self.buyProduct(product: validProduct)
} else {
print(validProduct.productIdentifier)
}
} else {
print("nothing")
}
}
func paymentQueue(_ queue: SKPaymentQueue, updatedTransactions transactions: [SKPaymentTransaction]) {
for transaction:AnyObject in transactions {
if let trans:SKPaymentTransaction = transaction as? SKPaymentTransaction{
self.dismissPurchaseBtn.isEnabled = true
self.restorePurchaseBtn.isEnabled = true
self.buyNowBtn.isEnabled = true
switch trans.transactionState {
case .purchased:
print("Product Purchased")
//Do unlocking etc stuff here in case of new purchase
SKPaymentQueue.default().finishTransaction(transaction as! SKPaymentTransaction)
break;
case .failed:
print("Purchased Failed");
SKPaymentQueue.default().finishTransaction(transaction as! SKPaymentTransaction)
break;
case .restored:
print("Already Purchased")
//Do unlocking etc stuff here in case of restor
SKPaymentQueue.default().finishTransaction(transaction as! SKPaymentTransaction)
default:
break;
}
}
}
}
//If an error occurs, the code will go to this function
func paymentQueue(_ queue: SKPaymentQueue, restoreCompletedTransactionsFailedWithError error: Error) {
// Show some alert
}
в app Purchas делегировать метод:
func buyInApp()
{
if (SKPaymentQueue.canMakePayments())
{
let productID:NSSet = NSSet(object: "com.valueedge.plane_post");
let productsRequest:SKProductsRequest = SKProductsRequest(productIdentifiers: productID as! Set<String>);
productsRequest.delegate = self;
productsRequest.start();
}
else
{
MBProgressHUD.hideHUDForView(self.view, animated: true)
}
}
func productsRequest (request: SKProductsRequest, didReceiveResponse response: SKProductsResponse)
{
let count : Int = response.products.count
if (count>0)
{
let validProduct: SKProduct = response.products[0] as SKProduct
if (validProduct.productIdentifier == "com.valueedge.plane_post")
{
buyProduct(validProduct);
}
else
{
MBProgressHUD.hideHUDForView(self.view, animated: true)
}
}
else
{
MBProgressHUD.hideHUDForView(self.view, animated: true)
}
}
func buyProduct(product: SKProduct)
{
let payment = SKPayment(product: product)
SKPaymentQueue.defaultQueue().addPayment(payment)
SKPaymentQueue.defaultQueue().addTransactionObserver(self)
}
func paymentQueue(queue: SKPaymentQueue, updatedTransactions transactions: [SKPaymentTransaction])
{
for transaction:AnyObject in transactions
{
if let trans:SKPaymentTransaction = transaction as? SKPaymentTransaction
{
switch trans.transactionState
{
case .Purchased:
self.PaymentSuccess("Apple", details: "Apple(In-App)")
print("In App Payment Success")
SKPaymentQueue.defaultQueue().finishTransaction(transaction as! SKPaymentTransaction)
MBProgressHUD.hideHUDForView(self.view, animated: true)
break
case .Failed:
MBProgressHUD.hideHUDForView(self.view, animated: true)
SKPaymentQueue.defaultQueue().finishTransaction(transaction as! SKPaymentTransaction)
let alert = UIAlertController(title: "Planes Only", message: "Payment failed", preferredStyle: .Alert)
alert.addAction(UIAlertAction(title: "OK", style: .Default, handler: {(action:UIAlertAction!) in
}))
self.presentViewController(alert, animated: true, completion: nil)
break
case .Restored :
SKPaymentQueue.defaultQueue().finishTransaction(transaction as! SKPaymentTransaction)
MBProgressHUD.hideHUDForView(self.view, animated: true)
break
default:
MBProgressHUD.hideHUDForView(self.view, animated: true)
break
}
}
}
}
func restore()
{
SKPaymentQueue.defaultQueue().addTransactionObserver(self)
SKPaymentQueue.defaultQueue().restoreCompletedTransactions()
MBProgressHUD.showHUDAddedTo(self.view, animated: true)
}
func paymentQueueRestoreCompletedTransactionsFinished(queue: SKPaymentQueue)
{
for transact:SKPaymentTransaction in queue.transactions
{
if transact.transactionState == SKPaymentTransactionState.Restored
{
//let t: SKPaymentTransaction = transact as SKPaymentTransaction
//let prodID = t.payment.productIdentifier as String
//restore prodID
SKPaymentQueue .defaultQueue().finishTransaction(transact)
MBProgressHUD.hideHUDForView(self.view, animated: true)
}
}
}
func paymentQueue(queue: SKPaymentQueue, restoreCompletedTransactionsFailedWithError error: NSError)
{
for transaction:SKPaymentTransaction in queue.transactions
{
if transaction.transactionState == SKPaymentTransactionState.Restored
{
SKPaymentQueue.defaultQueue().finishTransaction(transaction)
MBProgressHUD.hideHUDForView(self.view, animated: true)
break
}
}
MBProgressHUD.hideHUDForView(self.view, animated: true)
}
func request(request: SKRequest, didFailWithError error: NSError)
{
MBProgressHUD.hideHUDForView(self.view, animated: true)
}
в приложении кнопка нажала событие:
@IBAction func In_App_Payment_Clicked(sender: AnyObject)
{
MBProgressHUD.showHUDAddedTo(self.view, animated: true)
self.buyInApp()
}
для разработки в App purchase app даже простым способом вам нужно решить далее подзадачи, которые не относятся к кодированию:
- зарегистрироваться как разработчик и создание идентификатора приложения под https://developer.apple.com/account/
- выберите сертификаты, идентификаторы и профили.
- выберите идентификаторы > идентификаторы приложений и нажмите + создать новый идентификатор приложения.
- убедитесь, что Покупка в приложении включена для вашего идентификатора приложения.
- заполните соглашение и информацию о банке. для этого входа в https://itunesconnect.apple.com/
- важный шаг - дважды проверьте соглашения, налоги и банковский раздел. Все они должны быть заполнены. В другом случае в app purchase не будет работать. Вы получите не ответ сервер.
- создание приложения в iTunes Connect. для этого просто нажмите на Мои Приложения кнопка под https://itunesconnect.apple.com/
- Создание Продуктов Покупки В Приложении. В вашем случае расходуемый или не расходуемый. Для управления расходуемым или не расходуемым кодом будет аналогично.
- создать пользователя или пользователей песочницы. создание приложения в iTunes Соединять. Для этого просто зайдите на https://itunesconnect.apple.com/ выберите пользователи и роли кнопка, затем перейдите на вкладку тестеры песочницы. Создайте пользователя - > и перейдите на тестовое письмо и подтвердить пользователя по электронной почте!
- для начала разработки и проверка связи между itunesconnect и вашим приложением вы можете использовать симулятор. Но когда вы попытаетесь купить продукт, который вам нужно использовать реальное устройство.
- обязательно для ios app с in app purchase, чтобы иметь не только купить кнопка на "восстановить покупки" кнопка должна быть разработана тоже. (например, в Android-приложении это не обязательно, Android-приложение проверяет его по-другому). В случае, если пользователь удалит свое приложение и установит его снова, он хочет восстановить уже купленные предметы. За это "восстановить покупки" должны быть разработаны.
- последний шаг разработка и тестирование вы можете использовать следующий учебник, чтобы получить полезный пример реализации кнопка восстановить покупку. - https://www.raywenderlich.com/122144/in-app-purchase-tutorial
- также слушайте видео youtube, чтобы сделать покупку в приложении более ясной для вас - вы можете использовать следующее видео, которое действительно здорово -https://youtu.be/o3hJ0rY1NNw
- вот именно.