mirror of
https://github.com/defold/extension-iap
synced 2025-06-27 18:37:44 +02:00
Compare commits
19 Commits
Author | SHA1 | Date | |
---|---|---|---|
|
88028fd29f | ||
|
c0e35039f4 | ||
|
2e17b2f413 | ||
|
f398677d7b | ||
|
d1c4e88562 | ||
|
6f06d7f08f | ||
|
0852e42977 | ||
|
33174f25ea | ||
|
ba0e1b645a | ||
|
801179288e | ||
|
1c27130eef | ||
|
195ef400b5 | ||
|
68ef7f4615 | ||
|
1a34582603 | ||
|
09f5060d44 | ||
|
649a8a1ebf | ||
|
5c09447e37 | ||
|
ad06de7b9c | ||
|
d74c97d5c7 |
2
.gitignore
vendored
2
.gitignore
vendored
@ -9,3 +9,5 @@ Thumbs.db
|
|||||||
.cproject
|
.cproject
|
||||||
builtins
|
builtins
|
||||||
_site
|
_site
|
||||||
|
manifest.private.der
|
||||||
|
manifest.public.der
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
---
|
---
|
||||||
title: Defold In-app purchase extension API documentation
|
title: Defold In-app purchase extension API documentation
|
||||||
brief: This manual covers how to setup and use Google Play Game Services in Defold.
|
brief: This manual covers how to setup and use In-App Purchases in Defold.
|
||||||
---
|
---
|
||||||
|
|
||||||
# Defold In-app purchase extension API documentation
|
# Defold In-app purchase extension API documentation
|
||||||
@ -234,6 +234,3 @@ On iOS, the "price_string" field contains '~' characters
|
|||||||
## Source code
|
## Source code
|
||||||
|
|
||||||
The source code is available on [GitHub](https://github.com/defold/extension-iap)
|
The source code is available on [GitHub](https://github.com/defold/extension-iap)
|
||||||
|
|
||||||
|
|
||||||
## API reference
|
|
||||||
|
@ -1,10 +1,6 @@
|
|||||||
name: IAPExt
|
name: IAPExt
|
||||||
|
|
||||||
platforms:
|
platforms:
|
||||||
armv7-ios:
|
|
||||||
context:
|
|
||||||
weakFrameworks: ['StoreKit', 'UIKit', 'Foundation']
|
|
||||||
|
|
||||||
arm64-ios:
|
arm64-ios:
|
||||||
context:
|
context:
|
||||||
weakFrameworks: ['StoreKit', 'UIKit', 'Foundation']
|
weakFrameworks: ['StoreKit', 'UIKit', 'Foundation']
|
||||||
|
@ -71,7 +71,7 @@ var LibraryFacebookIAP = {
|
|||||||
|
|
||||||
if(url_index == product_count-1) {
|
if(url_index == product_count-1) {
|
||||||
var productsJSON = JSON.stringify(products);
|
var productsJSON = JSON.stringify(products);
|
||||||
var res_buf = allocate(intArrayFromString(productsJSON), 'i8', ALLOC_STACK);
|
var res_buf = stringToUTF8OnStack(productsJSON);
|
||||||
{{{ makeDynCall('vii', 'callback')}}}(lua_callback, res_buf);
|
{{{ makeDynCall('vii', 'callback')}}}(lua_callback, res_buf);
|
||||||
} else {
|
} else {
|
||||||
var xmlhttp = new XMLHttpRequest();
|
var xmlhttp = new XMLHttpRequest();
|
||||||
@ -143,7 +143,7 @@ var LibraryFacebookIAP = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var productsJSON = JSON.stringify(result)
|
var productsJSON = JSON.stringify(result)
|
||||||
var res_buf = allocate(intArrayFromString(productsJSON), 'i8', ALLOC_STACK);
|
var res_buf = stringToUTF8OnStack(productsJSON);
|
||||||
{{{ makeDynCall('viii', 'callback')}}}(lua_callback, res_buf, 0);
|
{{{ makeDynCall('viii', 'callback')}}}(lua_callback, res_buf, 0);
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
@ -166,4 +166,4 @@ var LibraryFacebookIAP = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
autoAddDeps(LibraryFacebookIAP, '$FBinner');
|
autoAddDeps(LibraryFacebookIAP, '$FBinner');
|
||||||
mergeInto(LibraryManager.library, LibraryFacebookIAP);
|
addToLibrary(LibraryFacebookIAP);
|
||||||
|
@ -1,3 +1,7 @@
|
|||||||
dependencies {
|
repositories {
|
||||||
implementation 'com.android.billingclient:billing:5.0.0'
|
mavenCentral()
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
implementation 'com.android.billingclient:billing:7.0.0'
|
||||||
}
|
}
|
||||||
|
@ -31,6 +31,7 @@ enum BillingResponse
|
|||||||
BILLING_RESPONSE_RESULT_ERROR = 6,
|
BILLING_RESPONSE_RESULT_ERROR = 6,
|
||||||
BILLING_RESPONSE_RESULT_ITEM_ALREADY_OWNED = 7,
|
BILLING_RESPONSE_RESULT_ITEM_ALREADY_OWNED = 7,
|
||||||
BILLING_RESPONSE_RESULT_ITEM_NOT_OWNED = 8,
|
BILLING_RESPONSE_RESULT_ITEM_NOT_OWNED = 8,
|
||||||
|
BILLING_RESPONSE_RESULT_NETWORK_ERROR = 9,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum ProviderId
|
enum ProviderId
|
||||||
|
@ -21,6 +21,7 @@ import com.android.billingclient.api.BillingClient;
|
|||||||
import com.android.billingclient.api.BillingClient.BillingResponseCode;
|
import com.android.billingclient.api.BillingClient.BillingResponseCode;
|
||||||
import com.android.billingclient.api.BillingClient.ProductType;
|
import com.android.billingclient.api.BillingClient.ProductType;
|
||||||
import com.android.billingclient.api.BillingResult;
|
import com.android.billingclient.api.BillingResult;
|
||||||
|
import com.android.billingclient.api.PendingPurchasesParams;
|
||||||
import com.android.billingclient.api.Purchase;
|
import com.android.billingclient.api.Purchase;
|
||||||
import com.android.billingclient.api.Purchase.PurchaseState;
|
import com.android.billingclient.api.Purchase.PurchaseState;
|
||||||
import com.android.billingclient.api.ProductDetails;
|
import com.android.billingclient.api.ProductDetails;
|
||||||
@ -56,7 +57,8 @@ public class IapGooglePlay implements PurchasesUpdatedListener {
|
|||||||
this.activity = activity;
|
this.activity = activity;
|
||||||
this.autoFinishTransactions = autoFinishTransactions;
|
this.autoFinishTransactions = autoFinishTransactions;
|
||||||
|
|
||||||
billingClient = BillingClient.newBuilder(activity).setListener(this).enablePendingPurchases().build();
|
PendingPurchasesParams pendingPurchasesParams = PendingPurchasesParams.newBuilder().enableOneTimeProducts().build();
|
||||||
|
billingClient = BillingClient.newBuilder(activity).setListener(this).enablePendingPurchases(pendingPurchasesParams).build();
|
||||||
billingClient.startConnection(new BillingClientStateListener() {
|
billingClient.startConnection(new BillingClientStateListener() {
|
||||||
@Override
|
@Override
|
||||||
public void onBillingSetupFinished(BillingResult billingResult) {
|
public void onBillingSetupFinished(BillingResult billingResult) {
|
||||||
@ -221,7 +223,6 @@ public class IapGooglePlay implements PurchasesUpdatedListener {
|
|||||||
case BillingResponseCode.OK:
|
case BillingResponseCode.OK:
|
||||||
defoldResponse = IapJNI.BILLING_RESPONSE_RESULT_OK;
|
defoldResponse = IapJNI.BILLING_RESPONSE_RESULT_OK;
|
||||||
break;
|
break;
|
||||||
case BillingResponseCode.SERVICE_TIMEOUT:
|
|
||||||
case BillingResponseCode.SERVICE_UNAVAILABLE:
|
case BillingResponseCode.SERVICE_UNAVAILABLE:
|
||||||
case BillingResponseCode.SERVICE_DISCONNECTED:
|
case BillingResponseCode.SERVICE_DISCONNECTED:
|
||||||
defoldResponse = IapJNI.BILLING_RESPONSE_RESULT_SERVICE_UNAVAILABLE;
|
defoldResponse = IapJNI.BILLING_RESPONSE_RESULT_SERVICE_UNAVAILABLE;
|
||||||
@ -229,6 +230,9 @@ public class IapGooglePlay implements PurchasesUpdatedListener {
|
|||||||
case BillingResponseCode.USER_CANCELED:
|
case BillingResponseCode.USER_CANCELED:
|
||||||
defoldResponse = IapJNI.BILLING_RESPONSE_RESULT_USER_CANCELED;
|
defoldResponse = IapJNI.BILLING_RESPONSE_RESULT_USER_CANCELED;
|
||||||
break;
|
break;
|
||||||
|
case BillingResponseCode.NETWORK_ERROR: // new in Play Billing Library 6.0.0
|
||||||
|
defoldResponse = IapJNI.BILLING_RESPONSE_RESULT_NETWORK_ERROR;
|
||||||
|
break;
|
||||||
case BillingResponseCode.FEATURE_NOT_SUPPORTED:
|
case BillingResponseCode.FEATURE_NOT_SUPPORTED:
|
||||||
case BillingResponseCode.ERROR:
|
case BillingResponseCode.ERROR:
|
||||||
default:
|
default:
|
||||||
@ -243,6 +247,20 @@ public class IapGooglePlay implements PurchasesUpdatedListener {
|
|||||||
return billingResponseCodeToDefoldResponse(result.getResponseCode());
|
return billingResponseCodeToDefoldResponse(result.getResponseCode());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void invokeOnPurchaseResultListener(IPurchaseListener purchaseListener, int billingResultCode, String purchaseData) {
|
||||||
|
if (purchaseListener == null) {
|
||||||
|
Log.w(TAG, "Received billing result but no listener has been set");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
purchaseListener.onPurchaseResult(billingResultCode, purchaseData);
|
||||||
|
}
|
||||||
|
private void invokeOnPurchaseResultListener(IPurchaseListener purchaseListener, BillingResult billingResult, Purchase purchase) {
|
||||||
|
invokeOnPurchaseResultListener(purchaseListener, billingResultToDefoldResponse(billingResult), convertPurchase(purchase));
|
||||||
|
}
|
||||||
|
private void invokeOnPurchaseResultListener(IPurchaseListener purchaseListener, BillingResult billingResult) {
|
||||||
|
invokeOnPurchaseResultListener(purchaseListener, billingResultToDefoldResponse(billingResult), "");
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method is called either explicitly from Lua or from extension code
|
* This method is called either explicitly from Lua or from extension code
|
||||||
* when "set_listener()" is called from Lua.
|
* when "set_listener()" is called from Lua.
|
||||||
@ -306,7 +324,7 @@ public class IapGooglePlay implements PurchasesUpdatedListener {
|
|||||||
// note: we only call the purchase listener if an error happens
|
// note: we only call the purchase listener if an error happens
|
||||||
if (billingResult.getResponseCode() != BillingResponseCode.OK) {
|
if (billingResult.getResponseCode() != BillingResponseCode.OK) {
|
||||||
Log.e(TAG, "Unable to consume purchase: " + billingResult.getDebugMessage());
|
Log.e(TAG, "Unable to consume purchase: " + billingResult.getDebugMessage());
|
||||||
purchaseListener.onPurchaseResult(billingResultToDefoldResponse(billingResult), "");
|
invokeOnPurchaseResultListener(purchaseListener, billingResult);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -329,7 +347,7 @@ public class IapGooglePlay implements PurchasesUpdatedListener {
|
|||||||
// note: we only call the purchase listener if an error happens
|
// note: we only call the purchase listener if an error happens
|
||||||
if (billingResult.getResponseCode() != BillingResponseCode.OK) {
|
if (billingResult.getResponseCode() != BillingResponseCode.OK) {
|
||||||
Log.e(TAG, "Unable to acknowledge purchase: " + billingResult.getDebugMessage());
|
Log.e(TAG, "Unable to acknowledge purchase: " + billingResult.getDebugMessage());
|
||||||
purchaseListener.onPurchaseResult(billingResultToDefoldResponse(billingResult), "");
|
invokeOnPurchaseResultListener(purchaseListener, billingResult);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -348,12 +366,12 @@ public class IapGooglePlay implements PurchasesUpdatedListener {
|
|||||||
@Override
|
@Override
|
||||||
public void onConsumeResponse(BillingResult billingResult, String purchaseToken) {
|
public void onConsumeResponse(BillingResult billingResult, String purchaseToken) {
|
||||||
Log.d(TAG, "handlePurchase() response code " + billingResult.getResponseCode() + " purchaseToken: " + purchaseToken);
|
Log.d(TAG, "handlePurchase() response code " + billingResult.getResponseCode() + " purchaseToken: " + purchaseToken);
|
||||||
purchaseListener.onPurchaseResult(billingResultToDefoldResponse(billingResult), convertPurchase(purchase));
|
invokeOnPurchaseResultListener(purchaseListener, billingResult, purchase);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
purchaseListener.onPurchaseResult(billingResponseCodeToDefoldResponse(BillingResponseCode.OK), convertPurchase(purchase));
|
invokeOnPurchaseResultListener(purchaseListener, billingResponseCodeToDefoldResponse(BillingResponseCode.OK), convertPurchase(purchase));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -362,13 +380,17 @@ public class IapGooglePlay implements PurchasesUpdatedListener {
|
|||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public void onPurchasesUpdated(BillingResult billingResult, List<Purchase> purchases) {
|
public void onPurchasesUpdated(BillingResult billingResult, List<Purchase> purchases) {
|
||||||
if (billingResult.getResponseCode() == BillingResponseCode.OK && purchases != null) {
|
if (billingResult.getResponseCode() == BillingResponseCode.OK) {
|
||||||
for (Purchase purchase : purchases) {
|
if (purchases != null && !purchases.isEmpty()) {
|
||||||
handlePurchase(purchase, this.purchaseListener);
|
for (Purchase purchase : purchases) {
|
||||||
|
if (purchase != null) {
|
||||||
|
handlePurchase(purchase, this.purchaseListener);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
this.purchaseListener.onPurchaseResult(billingResultToDefoldResponse(billingResult), "");
|
invokeOnPurchaseResultListener(this.purchaseListener, billingResult);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -392,7 +414,7 @@ public class IapGooglePlay implements PurchasesUpdatedListener {
|
|||||||
BillingResult billingResult = billingClient.launchBillingFlow(this.activity, billingFlowParams);
|
BillingResult billingResult = billingClient.launchBillingFlow(this.activity, billingFlowParams);
|
||||||
if (billingResult.getResponseCode() != BillingResponseCode.OK) {
|
if (billingResult.getResponseCode() != BillingResponseCode.OK) {
|
||||||
Log.e(TAG, "Purchase failed: " + billingResult.getDebugMessage());
|
Log.e(TAG, "Purchase failed: " + billingResult.getDebugMessage());
|
||||||
purchaseListener.onPurchaseResult(billingResultToDefoldResponse(billingResult), "");
|
invokeOnPurchaseResultListener(purchaseListener, billingResult);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -413,11 +435,16 @@ public class IapGooglePlay implements PurchasesUpdatedListener {
|
|||||||
@Override
|
@Override
|
||||||
public void onProductDetailsResponse(BillingResult billingResult, List<ProductDetails> productDetailsList) {
|
public void onProductDetailsResponse(BillingResult billingResult, List<ProductDetails> productDetailsList) {
|
||||||
if (billingResult.getResponseCode() == BillingResponseCode.OK && (productDetailsList != null) && !productDetailsList.isEmpty()) {
|
if (billingResult.getResponseCode() == BillingResponseCode.OK && (productDetailsList != null) && !productDetailsList.isEmpty()) {
|
||||||
buyProduct(productDetailsList.get(0), token, purchaseListener);
|
for (ProductDetails productDetails : productDetailsList) {
|
||||||
|
if (productDetails != null) {
|
||||||
|
buyProduct(productDetails, token, purchaseListener);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
Log.e(TAG, "Unable to get product details before buying: " + billingResult.getDebugMessage());
|
Log.e(TAG, "Unable to get product details before buying: " + billingResult.getDebugMessage());
|
||||||
purchaseListener.onPurchaseResult(billingResultToDefoldResponse(billingResult), "");
|
invokeOnPurchaseResultListener(purchaseListener, billingResult);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -435,10 +462,12 @@ public class IapGooglePlay implements PurchasesUpdatedListener {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onProductDetailsResponse(BillingResult billingResult, List<ProductDetails> productDetails) {
|
public void onProductDetailsResponse(BillingResult billingResult, List<ProductDetails> productDetails) {
|
||||||
if (productDetails != null) {
|
if (productDetails != null && !productDetails.isEmpty()) {
|
||||||
// cache products (cache will be used to speed up buying)
|
// cache products (cache will be used to speed up buying)
|
||||||
for (ProductDetails pd : productDetails) {
|
for (ProductDetails pd : productDetails) {
|
||||||
IapGooglePlay.this.products.put(pd.getProductId(), pd);
|
if (pd != null) {
|
||||||
|
IapGooglePlay.this.products.put(pd.getProductId(), pd);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// add to list of all product details
|
// add to list of all product details
|
||||||
allProductDetails.addAll(productDetails);
|
allProductDetails.addAll(productDetails);
|
||||||
@ -485,9 +514,11 @@ public class IapGooglePlay implements PurchasesUpdatedListener {
|
|||||||
@Override
|
@Override
|
||||||
public void onProductDetailsResponse(BillingResult billingResult, List<ProductDetails> productDetails) {
|
public void onProductDetailsResponse(BillingResult billingResult, List<ProductDetails> productDetails) {
|
||||||
JSONArray a = new JSONArray();
|
JSONArray a = new JSONArray();
|
||||||
if (billingResult.getResponseCode() == BillingResponseCode.OK) {
|
if ((billingResult.getResponseCode() == BillingResponseCode.OK) && (productDetails != null) && !productDetails.isEmpty()) {
|
||||||
for (ProductDetails pd : productDetails) {
|
for (ProductDetails pd : productDetails) {
|
||||||
a.put(convertProductDetails(pd));
|
if (pd != null) {
|
||||||
|
a.put(convertProductDetails(pd));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
@ -18,6 +18,7 @@ public class IapJNI implements IListProductsListener, IPurchaseListener {
|
|||||||
public static final int BILLING_RESPONSE_RESULT_ERROR = 6;
|
public static final int BILLING_RESPONSE_RESULT_ERROR = 6;
|
||||||
public static final int BILLING_RESPONSE_RESULT_ITEM_ALREADY_OWNED = 7;
|
public static final int BILLING_RESPONSE_RESULT_ITEM_ALREADY_OWNED = 7;
|
||||||
public static final int BILLING_RESPONSE_RESULT_ITEM_NOT_OWNED = 8;
|
public static final int BILLING_RESPONSE_RESULT_ITEM_NOT_OWNED = 8;
|
||||||
|
public static final int BILLING_RESPONSE_RESULT_NETWORK_ERROR = 9;
|
||||||
|
|
||||||
public IapJNI() {
|
public IapJNI() {
|
||||||
}
|
}
|
||||||
|
@ -11,12 +11,12 @@ height = 1136
|
|||||||
[android]
|
[android]
|
||||||
input_method = HiddenInputField
|
input_method = HiddenInputField
|
||||||
package = com.defold.extension.iap
|
package = com.defold.extension.iap
|
||||||
version_code = 7
|
version_code = 9
|
||||||
target_sdk_version = 29
|
minimum_sdk_version = 21
|
||||||
|
|
||||||
[project]
|
[project]
|
||||||
title = extension-iap
|
title = extension-iap
|
||||||
dependencies = https://github.com/andsve/dirtylarry/archive/master.zip
|
dependencies#0 = https://github.com/andsve/dirtylarry/archive/master.zip
|
||||||
|
|
||||||
[library]
|
[library]
|
||||||
include_dirs = extension-iap
|
include_dirs = extension-iap
|
||||||
|
100
main/main.gui
100
main/main.gui
@ -1,7 +1,7 @@
|
|||||||
script: "/main/main.gui_script"
|
script: "/main/main.gui_script"
|
||||||
fonts {
|
fonts {
|
||||||
name: "system_font"
|
name: "default"
|
||||||
font: "/builtins/fonts/system_font.font"
|
font: "/builtins/fonts/default.font"
|
||||||
}
|
}
|
||||||
background_color {
|
background_color {
|
||||||
x: 0.0
|
x: 0.0
|
||||||
@ -47,6 +47,8 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template: "/dirtylarry/button.gui"
|
template: "/dirtylarry/button.gui"
|
||||||
template_node_child: false
|
template_node_child: false
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -102,6 +104,10 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template_node_child: true
|
template_node_child: true
|
||||||
size_mode: SIZE_MODE_MANUAL
|
size_mode: SIZE_MODE_MANUAL
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -166,6 +172,10 @@ nodes {
|
|||||||
template_node_child: true
|
template_node_child: true
|
||||||
text_leading: 1.0
|
text_leading: 1.0
|
||||||
text_tracking: 0.0
|
text_tracking: 0.0
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -205,6 +215,8 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template: "/dirtylarry/button.gui"
|
template: "/dirtylarry/button.gui"
|
||||||
template_node_child: false
|
template_node_child: false
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -261,6 +273,10 @@ nodes {
|
|||||||
overridden_fields: 4
|
overridden_fields: 4
|
||||||
template_node_child: true
|
template_node_child: true
|
||||||
size_mode: SIZE_MODE_MANUAL
|
size_mode: SIZE_MODE_MANUAL
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -325,6 +341,10 @@ nodes {
|
|||||||
template_node_child: true
|
template_node_child: true
|
||||||
text_leading: 1.0
|
text_leading: 1.0
|
||||||
text_tracking: 0.0
|
text_tracking: 0.0
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -360,7 +380,7 @@ nodes {
|
|||||||
type: TYPE_TEXT
|
type: TYPE_TEXT
|
||||||
blend_mode: BLEND_MODE_ALPHA
|
blend_mode: BLEND_MODE_ALPHA
|
||||||
text: "<text>"
|
text: "<text>"
|
||||||
font: "system_font"
|
font: "default"
|
||||||
id: "log"
|
id: "log"
|
||||||
xanchor: XANCHOR_NONE
|
xanchor: XANCHOR_NONE
|
||||||
yanchor: YANCHOR_NONE
|
yanchor: YANCHOR_NONE
|
||||||
@ -387,6 +407,10 @@ nodes {
|
|||||||
template_node_child: false
|
template_node_child: false
|
||||||
text_leading: 1.0
|
text_leading: 1.0
|
||||||
text_tracking: 0.0
|
text_tracking: 0.0
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -426,6 +450,8 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template: "/dirtylarry/button.gui"
|
template: "/dirtylarry/button.gui"
|
||||||
template_node_child: false
|
template_node_child: false
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -481,6 +507,10 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template_node_child: true
|
template_node_child: true
|
||||||
size_mode: SIZE_MODE_MANUAL
|
size_mode: SIZE_MODE_MANUAL
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -545,6 +575,10 @@ nodes {
|
|||||||
template_node_child: true
|
template_node_child: true
|
||||||
text_leading: 1.0
|
text_leading: 1.0
|
||||||
text_tracking: 0.0
|
text_tracking: 0.0
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -584,6 +618,8 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template: "/dirtylarry/button.gui"
|
template: "/dirtylarry/button.gui"
|
||||||
template_node_child: false
|
template_node_child: false
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -639,6 +675,10 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template_node_child: true
|
template_node_child: true
|
||||||
size_mode: SIZE_MODE_MANUAL
|
size_mode: SIZE_MODE_MANUAL
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -703,6 +743,10 @@ nodes {
|
|||||||
template_node_child: true
|
template_node_child: true
|
||||||
text_leading: 1.0
|
text_leading: 1.0
|
||||||
text_tracking: 0.0
|
text_tracking: 0.0
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -742,6 +786,8 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template: "/dirtylarry/button.gui"
|
template: "/dirtylarry/button.gui"
|
||||||
template_node_child: false
|
template_node_child: false
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -798,6 +844,10 @@ nodes {
|
|||||||
overridden_fields: 4
|
overridden_fields: 4
|
||||||
template_node_child: true
|
template_node_child: true
|
||||||
size_mode: SIZE_MODE_MANUAL
|
size_mode: SIZE_MODE_MANUAL
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -862,6 +912,10 @@ nodes {
|
|||||||
template_node_child: true
|
template_node_child: true
|
||||||
text_leading: 1.0
|
text_leading: 1.0
|
||||||
text_tracking: 0.0
|
text_tracking: 0.0
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -901,6 +955,8 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template: "/dirtylarry/button.gui"
|
template: "/dirtylarry/button.gui"
|
||||||
template_node_child: false
|
template_node_child: false
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -956,6 +1012,10 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template_node_child: true
|
template_node_child: true
|
||||||
size_mode: SIZE_MODE_MANUAL
|
size_mode: SIZE_MODE_MANUAL
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -1020,6 +1080,10 @@ nodes {
|
|||||||
template_node_child: true
|
template_node_child: true
|
||||||
text_leading: 1.0
|
text_leading: 1.0
|
||||||
text_tracking: 0.0
|
text_tracking: 0.0
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -1059,6 +1123,8 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template: "/dirtylarry/button.gui"
|
template: "/dirtylarry/button.gui"
|
||||||
template_node_child: false
|
template_node_child: false
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -1115,6 +1181,10 @@ nodes {
|
|||||||
overridden_fields: 4
|
overridden_fields: 4
|
||||||
template_node_child: true
|
template_node_child: true
|
||||||
size_mode: SIZE_MODE_MANUAL
|
size_mode: SIZE_MODE_MANUAL
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -1179,6 +1249,10 @@ nodes {
|
|||||||
template_node_child: true
|
template_node_child: true
|
||||||
text_leading: 1.0
|
text_leading: 1.0
|
||||||
text_tracking: 0.0
|
text_tracking: 0.0
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -1218,6 +1292,8 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template: "/dirtylarry/checkbox_label.gui"
|
template: "/dirtylarry/checkbox_label.gui"
|
||||||
template_node_child: false
|
template_node_child: false
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -1273,6 +1349,10 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template_node_child: true
|
template_node_child: true
|
||||||
size_mode: SIZE_MODE_MANUAL
|
size_mode: SIZE_MODE_MANUAL
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -1337,6 +1417,10 @@ nodes {
|
|||||||
template_node_child: true
|
template_node_child: true
|
||||||
text_leading: 1.0
|
text_leading: 1.0
|
||||||
text_tracking: 0.0
|
text_tracking: 0.0
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -1376,6 +1460,8 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template: "/dirtylarry/checkbox_label.gui"
|
template: "/dirtylarry/checkbox_label.gui"
|
||||||
template_node_child: false
|
template_node_child: false
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -1431,6 +1517,10 @@ nodes {
|
|||||||
alpha: 1.0
|
alpha: 1.0
|
||||||
template_node_child: true
|
template_node_child: true
|
||||||
size_mode: SIZE_MODE_MANUAL
|
size_mode: SIZE_MODE_MANUAL
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
nodes {
|
nodes {
|
||||||
position {
|
position {
|
||||||
@ -1495,6 +1585,10 @@ nodes {
|
|||||||
template_node_child: true
|
template_node_child: true
|
||||||
text_leading: 1.0
|
text_leading: 1.0
|
||||||
text_tracking: 0.0
|
text_tracking: 0.0
|
||||||
|
custom_type: 0
|
||||||
|
enabled: true
|
||||||
|
visible: true
|
||||||
|
material: ""
|
||||||
}
|
}
|
||||||
material: "/builtins/materials/gui.material"
|
material: "/builtins/materials/gui.material"
|
||||||
adjust_reference: ADJUST_REFERENCE_PARENT
|
adjust_reference: ADJUST_REFERENCE_PARENT
|
||||||
|
@ -44,7 +44,13 @@ end
|
|||||||
|
|
||||||
local function buy(id)
|
local function buy(id)
|
||||||
log("iap.buy() " .. id)
|
log("iap.buy() " .. id)
|
||||||
iap.buy(id)
|
local options = {}
|
||||||
|
local item = available_items[id]
|
||||||
|
if item.subscriptions then
|
||||||
|
local subscription = item.subscriptions[1]
|
||||||
|
options.token = subscription.token
|
||||||
|
end
|
||||||
|
iap.buy(id, options)
|
||||||
end
|
end
|
||||||
|
|
||||||
local function restore()
|
local function restore()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user