This branch contains the code for cocos2d-x v3.x, and was optimized and tested for v3.1. For cocos2d-x v2.x use cocos2dx-v2
branch.
This project is a part of The SOOMLA Project which is a series of open source initiatives with a joint goal to help mobile game developers get better stores and more in-app purchases.
Haven't you always wanted an in-app purchase one liner that looks like this?!
soomla::CCStoreInventory::sharedStoreInventory()->buyItem("[itemId]");
Jul 21, 2014: Improved implementation
May 30, 2014: Support Amazon Billing Provider.
May 29, 2014: Added prebuilt libs.
April 9, 2014: Complete upgrading to cocos2d-x v3.
April 8, 2014: Added new events from native stores.
February 4, 2014: Added support for js-bindings.
December 1, 2013: Android in-app billing has been updated to use Google's in-app billing version 3.
October 28, 2013: iOS server side verification is added. This feature is not activated by default. learn more
October 27, 2013: cocos2dx-store has been updated since its last version. Everything has been rewritten from scratch and is much more Cocos2d-x friendly. cocos2dx-store allows your Cocos2d-x game to use SOOMLA's in app purchasing services and storage. cocos2dx-store has also been updated to use the third version of our economy model: modelV3.
cocos2dx-store currently supports all Cocos2d-x 3.x, and 2.x versions. At the moment code related to v2 is in
cocos2dx-v2
branch, v3 is inmaster
branch.
The current virtual economy model is called modelV3. Want to learn more about it? Try these links:
The cocos2dx-store is the Cocos2d-x flavour of The SOOMLA Project. This project uses android-store and ios-store in order to provide game developers with in-app billing for their cocos2d-x projects.
If you also want to create a Storefront you can do that using SOOMLA's In-App Purchase Store Designer.
There is an example project that show how to use cocos2dx-store:
http://github.com/soomla/cocos2dx-store-example,
The example project is still under development but it already has some important aspects of the framework you can learn and implement in your application.
-
As with all Cocos2d-x projects, you need to clone the Cocos2d-x framework from here or download it from the Cocos2d-x website.
Make sure the version you clone is supported by cocos2dx-store (the tag is the version).
-
Go into your cocos2d-x project and recursively clone cocos2dx-core into the
extensions
directory located at the root of your Cocos2d-x framework.$ git clone --recursive git@github.com:soomla/soomla-cocos2dx-core.git extensions/soomla-cocos2dx-core
-
Perform the same action for cocos2dx-store.
$ git clone --recursive git@github.com:soomla/cocos2dx-store.git extensions/cocos2dx-store
-
We use a fork of the jansson library for json parsing, clone our fork into the
external
directory at the root of your cocos2d-x framework.$ git clone git@github.com:vedi/jansson.git external/jansson
-
Make sure to include the
Cocos2dxStore.h
header whenever you use any of the cocos2dx-store functions:#include "Cocos2dxStore.h"
-
Create your own implementation of
CCStoreAssets
that will represent the assets in your specific game (Refer to cocos2dx-store-example for an example.). -
Implement your
CCStoreEventHandler
in order to be notified about in-app purchasing related events. Refer to the Event Handling section for more information. -
Initialize
CCStoreService
with your assets class (instance ofCCStoreAssets
), and a__Dictionary
containing various parameters for it:__Dictionary *storeParams = __Dictionary::create(); storeParams->setObject(__String::create("ExamplePublicKey"), "androidPublicKey"); storeParams->setObject(__String::create("ExampleCustomSecret"), "customSecret"); soomla::CCStoreService::initShared(assets, storeParams);
- Custom Secret - is an encryption secret you provide that will be used to secure your data.
- Android Public Key - is the public key given to you from Google. (iOS doesn't have a public key).
Choose the secret wisely. You can't change it after you launch your game!
Initialize
CCStoreService
ONLY ONCE when your application loads. -
Add instance of your event handler to
CCStoreEventDispatcher
afterCCStoreService
initialization:soomla::CCStoreEventDispatcher::getInstance()->addEventHandler(handler);
And that's it! You now have storage and in-app purchasing capabilities.
In your XCode project, perform following steps:
-
Add
jansson
(external/jansson/) to sources of your project. -
Add
Cocos2dXCore.xcodeproj
(extensions/soomla-cocos2dx-core/) as linked project to your project. -
Add
Cocos2dXStore.xcodeproj
(extensions/cocos2dx-store/) as linked project to your project. -
Add targets of these projects to Build Phases->Target Dependencies.
-
Add *.a of these projects to Build Phases->Link Binary With Libraries.
-
Add to Build Settings->Header Search Paths:
$(SRCROOT)/../cocos2d/extensions/soomla-cocos2dx-core/Soomla
$(SRCROOT)/../cocos2d/extensions/soomla-cocos2dx-core/soomla-native/compilations/ios/headers
$(SRCROOT)/../cocos2d/extensions/cocos2dx-store/Soomla
$(SRCROOT)/../cocos2d/extensions/cocos2dx-store/soomla-native/compilations/ios/headers
with recursive
option.
-
Register native StoreService, adding:
[[ServiceManager sharedServiceManager] registerService:[StoreService sharedStoreService]];
at the begining of the method application: didFinishLaunchingWithOptions:
of AppController
.
- Make sure you have these 3 Frameworks linked to your XCode project: Security, libsqlite3.0.dylib, StoreKit.
That's it! Now all you have to do is build your XCode project and run your game with cocos2dx-store.
If you're building your application for the Android platform, here are some instructions on how to integrate cocos2dx-store into your Android project:
-
Import the cocos2dx-store library into your project's Android.mk by adding the following lines in their appropriate places.
LOCAL_WHOLE_STATIC_LIBRARIES += cocos2dx_store_static # add this line along with your other LOCAL_WHOLE_STATIC_LIBRARIES $(call import-module, extensions/cocos2dx-store/android/jni) # add this line at the end of the file, along with the other import-module calls
-
Add the following jars from the folder
extensions/cocos2dx-store/soomla-native/compilations/android
to your classpath:- SoomlaAndroidCore.jar
- Cocos2dxAndroidCore.jar
- SoomlaAndroidStore.jar
- Cocos2dxAndroidStore.jar
-
In your main Cocos2dxActivity (The one your Cocos2d-x application runs in), call the following in the
onCreateView
method:public Cocos2dxGLSurfaceView onCreateView() { // initialize services final ServiceManager serviceManager = ServiceManager.getInstance(); serviceManager.setActivity(this); serviceManager.setGlSurfaceView(glSurfaceView); serviceManager.registerService(StoreService.getInstance());
-
Override
onPause
,onResume
:@Override protected void onPause() { super.onPause(); ServiceManager.getInstance().onPause(); }
@Override protected void onResume() { ServiceManager.getInstance().onResume(); super.onResume(); }
-
Update your manifest to include internet premission and SoomlaApp:
<uses-permission android:name="android.permission.INTERNET"/> <uses-permission android:name="com.android.vending.BILLING"/> <application ... android:name="com.soomla.SoomlaApp"> ... </application>
-
Depending on your Billing Service perform the next step
-
Add
AndroidStoreGooglePlay.jar
from the folderextensions/cocos2dx-store/soomla-native/compilations/android
to your classpath: -
Update your manifest:
... <uses-permission android:name="com.android.vending.BILLING"/> <application ... <activity android:name="com.soomla.store.billing.google.GooglePlayIabService$IabActivity" android:theme="@android:style/Theme.Translucent.NoTitleBar.Fullscreen"/> <meta-data android:name="billing.service" android:value="google.GooglePlayIabService" /> </application>
-
Add
in-app-purchasing-1.0.3.jar
andAndroidStoreAmazon.jar
from the folderextensions/cocos2dx-store/soomla-native/compilations/android
to your classpath: -
Update your manifest:
... <receiver android:name = "com.amazon.inapp.purchasing.ResponseReceiver" > <intent-filter> <action android:name = "com.amazon.inapp.purchasing.NOTIFY" android:permission = "com.amazon.inapp.purchasing.Permission.NOTIFY" /> </intent-filter> </receiver> <meta-data android:name="billing.service" android:value="amazon.AmazonIabService" />
That's it! Now all you have to do is run the build_native.sh script and you can begin using cocos2dx-store in your game.
If you have your own storefront implemented inside your game, it's recommended that you open the IAB Service in the background when the store opens and close it when the store is closed.
C++
// Start Iab Service
#if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
CCSoomlaStore::getInstance()->startIabServiceInBg();
#endif
// Stop Iab Service
#if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
CCSoomlaStore::getInstance()->stopIabServiceInBg();
#endif
Don't forget to close the Iab Service when your store is closed. You don't have to do this at all, this is just an optimization.
When we implemented modelV3, we were thinking about ways that people buy things inside apps. We figured out many ways you can let your users purchase items in your game and we designed the new modelV3 to support 2 of them: CCPurchaseWithMarket
and CCPurchaseWithVirtualItem
.
- CCPurchaseWithMarket is a
CCPurchaseType
that allows users to purchase aCCVirtualItem
with Google Play or the App Store. - CCPurchaseWithVirtualItem is a
CCPurchaseType
that lets your users purchase aCCVirtualItem
with anotherCCVirtualItem
. For example: Buying a sword with 100 gems.
In order to define the way your various virtual items are purchased, you'll need to create your implementation of CCStoreAssets
(the same one from step 5 in the Getting Started section above).
Here is an example:
Lets say you have a CCVirtualCurrencyPack
you want to call TEN_COINS_PACK
and a CCVirtualCurrency
you want to call COIN_CURRENCY
(TEN_COINS_PACK
will hold 10 pieces of the currency COIN_CURRENCY
):
#define COIN_CURRENCY_ITEM_ID "coin_currency"
#define TEN_COIN_PACK_ITEM_ID "ten_coin_pack"
#define TEN_COIN_PACK_PRODUCT_ID "10_coins_pack" // this is the product id from the developer console
CCVirtualCurrency *COIN_CURRENCY = CCVirtualCurrency::create(
String::create("COIN_CURRECY"),
String::create(""),
String::create(COIN_CURRENCY_ITEM_ID)
);
CCVirtualCurrencyPack *TEN_COIN_PACK = CCVirtualCurrencyPack::create(
String::create("10 Coins"),
String::create("A pack of 10 coins"),
String::create(TEN_COIN_PACK_ITEM_ID),
Integer::create(10),
String::create(COIN_CURRENCY_ITEM_ID),
CCPurchaseWithMarket::create(String::create(TEN_COIN_PACK_PRODUCT_ID), Double::create(0.99))
);
Now you can use CCStoreInventory
to buy your new currency pack:
soomla::CCStoreInventory::sharedStoreInventory()->buyItem(TEN_COIN_PACK_ITEM_ID);
And that's it! cocos2dx-store knows how to contact Google Play or the App Store for you and will redirect your users to the purchasing system to complete the transaction. Don't forget to subscribe to store events in order to get notified of successful or failed purchases (see Event Handling).
CCStoreInventory
and CCStoreInfo
are important storage and metadata classes you should use when you want to perform all store operations:
CCStoreInventory
is a convenience class to let you perform operations onCCVirtualCurrencies
andCCVirtualGood
s. Use it to fetch/change the balances ofCCVirtualItem
s in your game (using their ItemIds!)CCStoreInfo
is where all meta data information about your specific game can be retrieved. It is initialized with your implementation ofCCStoreAssets
and you can use it to retrieve information about your specific game.
The on-device storage is encrypted and kept in a SQLite database. SOOMLA has a cloud-based storage service (The SOOMLA Highway) that allows this SQLite to be synced to a cloud-based repository that you define.
Example Usages
-
Get all the
CCVirtualCurrencies
:CCArray *vcArray = soomla::CCStoreInfo::sharedStoreInfo()->getVirtualCurrencies();
-
Give the user 10 pieces of a virtual currency with itemId "currency_coin":
soomla::CCStoreInventory::sharedStoreInventory()->giveItem("currency_coin", 10);
-
Take 10 virtual goods with itemId "green_hat":
soomla::CCStoreInventory::sharedStoreInventory()->takeItem("green_hat", 10);
-
Get the current balance of green hats (virtual goods with itemId "green_hat"):
int greenHatsBalance = soomla::CCStoreInventory::sharedStoreInventory()->getItemBalance("green_hat");
SOOMLA lets you subscribe to store events, get notified and implement your own application specific behaviour to them.
Your behaviour is an addition to the default behaviour implemented by SOOMLA. You don't replace SOOMLA's behaviour.
The CCStoreEventDispatcher
class is where all events go through. To handle various events, create your own event handler, a class that implements CCEventHandler
, and add it to the CCStoreEventDispatcher
class:
soomla::CCStoreEventDispatcher::getInstance()->addEventHandler(storeEventHandler);
Since Cocos2d-x doesn't support exceptions, we use a different method to catch and work with exceptions on the native side. All functions that raise an exception on the native side have an additional CCError** parameter to them. In order to know if an exception was raised, send a reference to CCError* to the function, and inspect it after running.
For example, if I want to purchase an item with the ItemID huge_sword
, and check if all went well after the purchase, I would call CCSoomlaStore::buyItem()
, like this:
soomla::CCError *err;
soomla::CCStoreInventory::sharedStoreInventory()->buyItem("huge_sword", &err);
if (err != NULL) {
int code = err->getCode();
switch code {
case SOOMLA_EXCEPTION_ITEM_NOT_FOUND:
// itemNotFoundException was raised
break;
case SOOMLA_EXCEPTION_INSUFFICIENT_FUNDS:
// insufficienFundsException was raised
break;
case SOOMLA_EXCEPTION_NOT_ENOUGH_GOODS:
// notEnoughGoodsException was raised
break;
}
}
You can choose to handle each exception on its own, handle all three at once, or not handle the exceptions at all. The CCError
parameter is entirely optional, you can pass NULL instead if you do not wish to handle errors, but remember, error handling is your responsibility. cocos2dx-store doesn't do any external error handling (i.e. error handling that uses CCError
) for you.
As you probably know, fraud on IAP is pretty common. Hackers can crack their smartphones to think that a purchase is made when payment wasn't actually transferred to you. We want to help you with it so we created our verification server and we let you instantly use it through the framework.
All you need to do is let cocos2dx-store know you want to verify purchases. You can do this by passing an extra parameter to CCSoomlaStore
:
storeParams->setObject(Bool::create(true), "SSV");
soomla::CCStoreService::initShared(assets, storeParams);
You can enable debug logging in cocos2dx-store by setting SOOMLA_DEBUG
in CCStoreUtils.h
to true
. Debug logging can also be enabled at build time by adding -DSOOMLA_DEBUG=1
to APP_CPPFLAGS
in your Application.mk
on Android, or by setting SOOMLA_DEBUG=1
in your Build Settings' Preprocessor Macros
on iOS.
If you want to see debug messages from android-store, set the logDebug
variable in com.soomla.store.StoreConfig
to true
.
To see debug messages on iOS, make sure you have also DEBUG=1
in your Build Settings' Preprocessor Macros
(for Debug only).
We want you!
Fork -> Clone -> Implement -> Insert Comments -> Test -> Pull-Request.
We have great RESPECT for contributors.
android-store follows strict code documentation conventions. If you would like to contribute please read our Documentation Guidelines and follow them. Clear, consistent comments will make our code easy to understand.
MIT License. Copyright (c) 2012 SOOMLA. http://project.soom.la