Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/indiesoftby/defold-yagames
YaGames is the Yandex.Games SDK implementation for the Defold game engine.
https://github.com/indiesoftby/defold-yagames
browser-games defold defold-game-engine defold-library defold-native-extension game-development gamedev html5-game-development yandex-games
Last synced: 5 days ago
JSON representation
YaGames is the Yandex.Games SDK implementation for the Defold game engine.
- Host: GitHub
- URL: https://github.com/indiesoftby/defold-yagames
- Owner: indiesoftby
- License: mit
- Created: 2020-07-22T14:01:29.000Z (over 4 years ago)
- Default Branch: master
- Last Pushed: 2024-10-19T20:49:44.000Z (3 months ago)
- Last Synced: 2025-01-09T02:39:15.776Z (13 days ago)
- Topics: browser-games, defold, defold-game-engine, defold-library, defold-native-extension, game-development, gamedev, html5-game-development, yandex-games
- Language: Lua
- Homepage: https://defold.com/2021/04/21/Releasing-games-on-Yandex/
- Size: 86.8 MB
- Stars: 64
- Watchers: 7
- Forks: 8
- Open Issues: 7
-
Metadata Files:
- Readme: README.md
- License: LICENSE.md
Awesome Lists containing this project
- awesome-defold - YaGames
README
[![YaGames Logo](cover.png)](https://github.com/indiesoftby/defold-yagames)
# YaGames - Yandex.Games for Defold
*This is an open-source project, and it's not affiliated with Yandex LLC.*
YaGames is the Yandex.Games SDK native extension for the [Defold](https://www.defold.com/) game engine. [Yandex.Games](https://yandex.com/games/) is a collection of browser HTML5 games for smartphones, computers, tablets, and TVs. The games are available in Yandex Browser and the Yandex app. Games from the catalog are displayed in Yandex recommendation systems, which have a total audience of more than 50 million users per month.
## Installation
You can use it in your own project by adding this project as a [Defold library dependency](http://www.defold.com/manuals/libraries/). Open your `game.project` file and in the dependencies field add **a link to the ZIP file of a [specific release](https://github.com/indiesoftby/defold-yagames/releases).**
> [!CAUTION]
> For Yandex.Games use only WASM variant when creating HTML5 build of your game. Asm.js variant won't work at all because of [polyfills](https://github.com/emscripten-core/emscripten/blob/10cb9d46cdd17e7a96de68137c9649d9a630fbc7/src/shell.js#L45), which are necessary to support older browsers and which break the Yandex.Games SDK.## Getting Started
* **[π Tutorial: "Releasing HTML5 games on Yandex.Games"](https://defold.com/2021/04/21/Releasing-games-on-Yandex/) - How to add the Yandex.Games SDK to a Defold game and how to submit your game to the Yandex.Games catalog.**
* [π The official documentation](https://yandex.ru/dev/games/doc/dg/concepts/about.html?lang=en).
* [π¬ The official Telegram chat](https://t.me/yagamedev) where you can talk with representatives from Yandex. Feel free to ask questions in English!
* [π¬ The Telegram chat about Defold](https://t.me/DefoldEngine) for Russian-speaking users.
* [π¬ The Defold forum topic](https://forum.defold.com/t/yagames-yandex-games-sdk-for-defold/66810) about the YaGames extension.### Checklist For Releasing Game
1. [Sign up as a developer](https://yandex.ru/dev/games/doc/dg/concepts/about.html?lang=en).
2. Translate your game to the Russian language (*tip:* translate your game title into Russian too!). English and Turkish are optional [(more info)](https://yandex.ru/dev/games/doc/dg/concepts/languages-and-domains.html?lang=en).
3. Prepare assets for the catalogue:
- Icon 512 x 512 px.
- Cover 800 x 470 px.
- Screenshots.
- *(Optional)* Videos and GIF.
4. Add [the extension](https://github.com/indiesoftby/defold-yagames/archive/master.zip) as a Defold library dependency to your project.
5. Enable monetization and earn revenue from placing ad blocks in your game. Ad blocks are available in the following formats:
- **Interstitial blocks**: ad blocks that completely cover the app background and show up at certain points (for example, when accessing the next game level). *Important: Mute sounds before showing the ad!*
- **Rewarded videos**: blocks with video ads that the user can choose to view and earn a reward or in-game currency. *Important: Mute sounds before showing the ad!*
- **Sticky banners**: banner ads, super easy to setup.
- **In-game purchases**: earn revenue by providing paid services to your users.
5. You can [publish your game on Yandex.Games](https://games.yandex.ru/console/) from this moment. It fully meets [the requirements](https://yandex.ru/dev/games/doc/dg/concepts/requirements.html?lang=en).### Best Practices & Tips
1. The YaGames extension imitates a real API on *non-HTML5* platforms. The idea is to allow to you quickly implement API on your favourite platform (macOS, Windows, Linux) and don't spend time on slowly rebuilding/uploading the game to the Yandex.
2. The code from `yagames/manifests/web/engine_template.html` is always added to your HTML5 template. This behaviour can't be disabled. Tip: use Git-branching for every platform and do not mix platform-specific code between them.
3. You don't need to set up any cache-busting techniques, since Yandex.Games hosts each version of your game in separate paths.## Code Examples
Take a look at the demo project inside `example` directory. It has quite a few buttons to test all APIs. You can use it in your game as a debug screen or simply [download/upload a pre-built .zip archive](https://github.com/indiesoftby/defold-yagames/blob/gh-pages/demo_no-sw_no-native-cache.zip?raw=true) to make sure that you implemented SDK in the right way.
![YaGames Demo](screenshot.png)
### 1. Initialization
To get started, you need to initialize the SDK using the `init` method.
```lua
local yagames = require("yagames.yagames")local function init_handler(self, err)
if err then
print("Something bad happened :(", err)
else
--
-- SDK is ready!
-- From this moment, you can use all available functions, i.e. invoke ads, get player data, etc.
---- For example, signal that the game has loaded all resources and is ready for user interaction:
yagames.features_loadingapi_ready()-- Do something else!
end
endfunction init(self)
yagames.init(init_handler)
end
```### 2. Interstitial Ad
Interstitial ads are ad blocks that completely cover the app background and show up before a user gets the data requested (for example, accessing the next game level).
***Note:** Yandex.Games [recommends that developers call the display of full-screen ads in the game as often as possible](https://yandex.ru/blog/gamesfordevelopers/obnovlenie-algoritmov-pokaza-fulskrinov) but in suitable places in the game β so that the user understands that this is not a part of the game, but an ad unit. Do this in logical pauses in the game, for example: before starting the game, when moving to the next level, after losing.For example, inserting an ad unit is appropriate after going to the next level by pressing a button, and not appropriate in the middle of a level, when an ad suddenly appears under the playerΚΌs finger.*
* `open` - Called when an ad is opened successfully.
* `close` - Called when an ad is closed, an error occurred, or on ad failed to open due to too frequent calls. Used with the `was_shown` argument (type `boolean`), the value of which indicates whether an ad was shown.
* `offline` - Called when the network connection is lost (when offline mode is enabled).
* `error` - Called when an error occurrs. The error object is passed to the callback function.**The `close` callback is called in any situation, even if there was an error.**
```lua
local yagames = require("yagames.yagames")local function adv_open(self)
-- You should switch off all sounds!
endlocal function adv_close(self, was_shown)
-- You can switch sounds back!
endlocal function adv_offline(self)
-- Internet is offline
endlocal function adv_error(self, err)
-- Something wrong happened :(
endfunction on_message(self, message_id, message)
if message_id == hash("show_fullscreen_adv") then
yagames.adv_show_fullscreen_adv({
open = adv_open,
close = adv_close,
offline = adv_offline,
error = adv_error
})
end
end
```### 3. Rewarded Videos
Rewarded videos are video ad blocks used to monetize games and earn a reward or in-game currency.
* `open` - Called when a video ad is displayed on the screen.
* `rewarded` - Called when a video ad impression is counted. Use this function to specify a reward for viewing the video ad.
* `close` - Called when a user closes a video ad or an error happens.
* `error` - Called when an error occurrs. The error object is passed to the callback function.**The `close` callback is called in any situation, even if there was an error.** The `rewarded` callback is called before `close`, and you should update your in-game UI only after `close`.
```lua
local yagames = require("yagames.yagames")local function rewarded_open(self)
-- You should switch off all sounds!
endlocal function rewarded_rewarded(self)
-- Add coins!
endlocal function rewarded_close(self)
-- You can switch sounds back!
endlocal function rewarded_error(self, err)
-- Something wrong happened :(
endfunction on_message(self, message_id, message)
if message_id == hash("show_rewarded_video") then
yagames.adv_show_rewarded_video({
open = rewarded_open,
rewarded = rewarded_rewarded,
close = rewarded_close,
error = rewarded_error
})
end
end
```### Misc
> [!TIP]
> We don't use thes features in our games as we don't see any improvements in our games metrics, and the complexity of its integration and support is quite high.#### Native Cache How-To
Yandex's [Native Cache](https://yandex.ru/dev/games/doc/dg/concepts/native-cache-settings.html?lang=en) lets users use games offline. Currently, it's available only in Yandex Browser or the Yandex app on smartphones.
1. Set the path to the file `yandex-manifest.json` in the `game.project` settings.
2. Copy the `yagames/manifests/web/yandex-manifest.json` file to the root directory of your release build.
3. Edit the list of all game files inside your `yandex-manifest.json`, and update the path to the icon. Omit `sw.js` and `yandex-manifest.json`.#### Service Worker How-To
Yandex dropped the Service Worker description page in their docs, but it still allows to integrate Service Worker into your game to be able to run both offline and online.
1. Set the path to the file `sw.js` in the `game.project` settings.
2. Copy the `yagames/manifests/web/sw.js` file to the root directory of your release build.
3. Edit the list of all game files inside your `sw.js`. Omit `sw.js` and `yandex-manifest.json`.
4. You should increment the version inside `sw.js` on every update of your game on Yandex.Games.## The `game.project` Settings (Optional!)
```ini
[yagames]
sdk_url = /sdk.js
sdk_init_options = {}
sdk_init_snippet = console.log("Yandex Games SDK is ready!");
service_worker_url = sw.js
manifest_url = yandex-manifest.json
```* `sdk_url` - Sets the URL of the Yandex.Games SDK. In July 2024 the platform changed the URL of its SDK and now it can be of two kinds. First is the local `/sdk.js` for games you upload as an archive (default, **suitable for 99% of games**). The second is for iFrame games - `https://sdk.games.s3.yandex.net/sdk.js`.
* `sdk_init_options` - JavaScript Object that is passed as-is into the Yandex Games SDK initialization options for [the JS `YaGames.init` function](https://yandex.ru/dev/games/doc/dg/sdk/sdk-about.html?lang=en). Example: `{ orientation: { value: "landscape", lock: true } }`.
* `sdk_init_snippet` - JavaScript code that is passed as-is and called when the `ysdk` variable becomes available. Example: `console.log(ysdk);`. **Use with care, and don't forget to put a semicolon `;` at the end.**
* `service_worker_url` - Relative URL to the Service Worker file. Usually it's `sw.js`. Set the URL to enable Service Worker.
* `manifest_url` - URL to the Web App Manifest file. Set the URL to enable support of Yandex Native Cache.## Lua API
Yandex.Games JavaScript SDK uses ES6 Promise for asynchronous operations. For Lua API promises were replaced with callback functions with arguments `(self, err, result)`, where
- `self` userdata - Script self reference.
- `err` string - Error code if something went wrong.
- `result` - Data if the operation should return something.### Lua <-> JS
The best way to integrate SDK into your game is to read [the official documentation](https://yandex.ru/dev/games/doc/dg/concepts/about.html?lang=en) and to use corresponding Lua API functions.
And it's also a good idea to upload a demo build of YaGames to your game's draft and click on the buttons to understand what the arguments are and what each function returns.
| Yandex.Games JS SDK | YaGames Lua API |
| ------------------- | --------------- |
| `YaGames.init(options)` | `yagames.init(callback)`
The `options` is a JavaScript object `{}`, and it can be set in the `yagames.sdk_init_options` setting. |
| `ysdk.isAvailableMethod(name)` | `yagames.is_available_method(name, callback)` [Example](https://github.com/indiesoftby/defold-yagames/blob/master/example/ysdkdebug/pg_methods.lua) |
| **Advertisement** [(docs)](https://yandex.ru/dev/games/doc/en/sdk/sdk-adv) | |
| `ysdk.adv.showFullscreenAdv({callbacks:{}})` | `yagames.adv_show_fullscreen_adv(callbacks)` [Example](#2-interstitial-ad) |
| `ysdk.adv.showRewardedVideo({callbacks:{}})` | `yagames.adv_show_rewarded_video(callbacks)` [Example](#3-rewarded-videos) |
| **Advertisement - Sticky Banners** [(docs)](https://yandex.ru/dev/games/doc/en/sdk/sdk-adv#sticky-banner) | |
| `ysdk.adv.getBannerAdvStatus()` | `yagames.adv_get_banner_adv_status(callback)` |
| `ysdk.adv.showBannerAdv()` | `yagames.adv_show_banner_adv([callback])` |
| `ysdk.adv.hideBannerAdv()` | `yagames.adv_hide_banner_adv([callback])` |
| **Authentication + Player** [(docs)](https://yandex.ru/dev/games/doc/en/sdk/sdk-player) | |
| `ysdk.auth.openAuthDialog()` | `yagames.auth_open_auth_dialog(callback)` |
| `ysdk.getPlayer(options)` | `yagames.player_init(options, callback)`
The argument `options` is a Lua table `{ signed = boolean, scopes = boolean }`. |
| `player._personalInfo` | `yagames.player_get_personal_info()`
The result is `table` or `nil` if the `_personalInfo` object is not available. |
| `player.signature` | `yagames.player_get_signature()`
The result is string if player's object is initialized with `options.signed = true`. Otherwise, `nil`. |
| `player.setData(data, flush)` | `yagames.player_set_data(data, flush, callback)` |
| `player.getData(keys)` | `yagames.player_get_data(keys, callback)` |
| `player.setStats(stats)` | `yagames.player_set_stats(stats, callback)` |
| `player.incrementStats(increments)` | `yagames.player_increment_stats(increments, callback)` |
| `player.getStats(keys)` | `yagames.player_get_stats(keys, callback)` |
| ~~`player.getID()`~~ Deprecated | ~~`yagames.player_get_id()`~~ Deprecated |
| `player.getUniqueID()` | `yagames.player_get_unique_id()` |
| `player.getIDsPerGame()` | `yagames.player_get_ids_per_game(callback)` |
| `player.getMode()` | `yagames.player_get_mode()`
[(more info)](https://yandex.ru/blog/gamesfordevelopers/novye-vozmozhnosti-dlya-neavtorizovannykh-polzovateley) |
| `player.getName()` | `yagames.player_get_name()` |
| `player.getPhoto(size)` | `yagames.player_get_photo(size)` |
| `player.getPayingStatus()` | `yagames.player_get_paying_status()` |
| **In-Game Purchases** [(docs)](https://yandex.ru/dev/games/doc/en/sdk/sdk-purchases) | |
| `ysdk.getPayments(options)` | `yagames.payments_init(options, callback)` |
| `payments.purchase(options)` | `yagames.payments_purchase(options, callback)` |
| `payments.getPurchases()` | `yagames.payments_get_purchases(callback)`
The result has the format `{ purchases = { ... }, signature = "..." }` |
| `payments.getCatalog()` | `yagames.payments_get_catalog([options], callback)`
The argument `options` is an optional Lua table `{ getPriceCurrencyImage = "size" }`, where `size` (string) can be `medium`, `small` and `svg`, the currency image url will be injected to the `getPriceCurrencyImage` field of each product. |
| `payments.consumePurchase(purchaseToken)` | `yagames.payments_consume_purchase(purchase_token, callback)` |
| **Leaderboards** [(docs)](https://yandex.ru/dev/games/doc/en/sdk/sdk-leaderboard) | |
| `ysdk.getLeaderboards()` | `yagames.leaderboards_init(callback)` |
| `lb.getLeaderboardDescription(leaderboardName)` | `yagames.leaderboards_get_description(leaderboard_name, callback)` |
| `lb.getLeaderboardPlayerEntry(leaderboardName)` | `yagames.leaderboards_get_player_entry(leaderboard_name, [options], callback)`
If the player doesn't have any score, you get the error `FetchError: Player is not present in leaderboard`.
The argument `options` is an optional Lua table `{ getAvatarSrc = "size", getAvatarSrcSet = "size" }`, where `size` (string) can be `small`, `medium`, `large`. |
| `lb.getLeaderboardEntries(leaderboardName, options)` | `yagames.leaderboards_get_entries(leaderboard_name, [options], callback)`
The argument `options` is an optional Lua table `{ includeUser = boolean, quantityAround = number, quantityTop = number, getAvatarSrc = "size", getAvatarSrcSet = "size" }`, where `size` (string) can be `small`, `medium`, `large`. |
| `lb.setLeaderboardScore(leaderboardName, score, extraData)` | `yagames.leaderboards_set_score(leaderboard_name, score, [extra_data], [callback])` |
| **Features** [(docs)](https://yandex.com/dev/games/doc/en/sdk/sdk-game-events) | |
| `ysdk.features.LoadingAPI?.ready()` | `yagames.features_loadingapi_ready()` |
| `ysdk.features.GameplayAPI?.start()` | `yagames.features_gameplayapi_start()` |
| `ysdk.features.GameplayAPI?.stop()` | `yagames.features_gameplayapi_stop()` |
| `ysdk.features.GamesAPI?.getAllGames()` | `yagames.features_gamesapi_get_all_games(callback)`
The callback result is a table `{ games = { ... }, developerURL = "string" }` |
| `ysdk.features.GamesAPI?.getGameByID(appID)` | `yagames.features_gamesapi_get_game_by_id(app_id, callback)`
The callback result is a table `{ isAvailable = true/false, game = { appID = "string", title = "string", url = "string", coverURL = "string", iconURL = "string" } }` |
| **Feedback** [(docs)](https://yandex.ru/dev/games/doc/en/sdk/sdk-review) | |
| `ysdk.feedback.canReview()` | `yagames.feedback_can_review(callback)`
The callback result is a table `{ value = true/false, reason = "string" }` |
| `ysdk.feedback.requestReview()` | `yagames.feedback_request_review(callback)`
The callback result is a table `{ feedbackSent = true/false }` |
| **Clipboard** [(docs)](https://yandex.ru/dev/games/doc/en/sdk/sdk-params) | |
| `ysdk.clipboard.writeText(text)` | `yagames.clipboard_write_text(text, [callback])` |
| **Device Info** [(docs)](https://yandex.ru/dev/games/doc/en/sdk/sdk-params) | |
| `ysdk.deviceInfo.type` | `yagames.device_info_type()`
Returns `"desktop"`, `"mobile"`, `"tablet"` or `"tv"` |
| `ysdk.deviceInfo.isDesktop()` | `yagames.device_info_is_desktop()` |
| `ysdk.deviceInfo.isMobile()` | `yagames.device_info_is_mobile()` |
| `ysdk.deviceInfo.isTablet()` | `yagames.device_info_is_tablet()` |
| `ysdk.deviceInfo.isTV()` | `yagames.device_info_is_tv()` |
| **Environment** [(docs)](https://yandex.ru/dev/games/doc/en/sdk/sdk-environment) | |
| `ysdk.environment` | `yagames.environment()`
Returns Lua table `{ app = { id = ... }, ... }` |
| **Screen** [(docs)](https://yandex.ru/dev/games/doc/en/sdk/sdk-params) | |
| `ysdk.screen.fullscreen.status` | `yagames.screen_fullscreen_status()`
Returns `"on"` or `"off"` |
| `ysdk.screen.fullscreen.request()` | `yagames.screen_fullscreen_request([callback])` |
| `ysdk.screen.fullscreen.exit()` | `yagames.screen_fullscreen_exit([callback])` |
| **Safe Storage** [(docs)](https://yandex.ru/dev/games/doc/en/sdk/sdk-player#progress-loss) | *Note: `key` and `value` should be valid UTF-8 strings. Storing strings with zero bytes aren't supported.* |
| `ysdk.getStorage()` | `yagames.storage_init(callback)` |
| - `safeStorage.getItem(key)` | `yagames.storage_get_item(key)`
Returns that key's value or `nil`. |
| - `safeStorage.setItem(key, value)` | `yagames.storage_set_item(key, value)`
Adds that key to the storage, or update that key's value if it already exists. |
| - `safeStorage.removeItem(key)` | `yagames.storage_remove_item(key)`
Removes that key from the storage. |
| - `safeStorage.clear()` | `yagames.storage_clear()`
Empties all keys out of the storage. |
| - `safeStorage.key(n)` | `yagames.storage_key(n)`
Returns the name of the nth key in the storage or `nil`. *Note: the n index is zero-based.* |
| - `safeStorage.length` | `yagames.storage_length()`
Returns the number of data items stored in the storage. |
| **Events** [(docs)](https://yandex.ru/dev/games/doc/en/sdk/sdk-events) | |
| `ysdk.onEvent(eventName, listener)` | `yagames.event_on(event_name, listener)` |
| `ysdk.dispatchEvent(eventName)` | `yagames.event_dispatch(event_name)` |
| **Remote Config** [(docs)](https://yandex.ru/dev/games/doc/en/sdk/sdk-config) | |
| `ysdk.getFlags(options)` | `yagames.flags_get(options, callback)`
Options is optional. The callback result is a table like `{ flagName = "value" }` |
| **Sitelock** [(docs)](#sitelock) | |
| | `sitelock.add_domain(domain)` |
| | `sitelock.verify_domain()` |
| | `sitelock.get_current_domain()` |
| | `sitelock.is_release_build()` |## Sitelock
It's a good idea to protect your HTML5 game from simple copy-pasting to another website. The YaGames extension has Sitelock API for that purpose. It's simple, but it's better than nothing.
By default, it checks hostnames `yandex.net` (CDN of the Yandex.Games) and `localhost` (for local debugging).
```lua
local sitelock = require("yagames.sitelock")-- Also you can add your domains:
-- sitelock.add_domain("yourdomainname.com")function init(self)
if html5 and sitelock.is_release_build() then
if not sitelock.verify_domain() then
-- Show warning and pause the game
end
end
end
```## Credits
Artsiom Trubchyk ([@aglitchman](https://github.com/aglitchman)) is the current YaGames owner within Indiesoft and is responsible for the open source repository.
This project uses the source code of [JsToDef](https://github.com/AGulev/jstodef).
### License
MIT license.