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 game engine. Yandex.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.
You can use it in your own project by adding this project as a Defold library dependency. Open your game.project
file and in the dependencies field add a link to the ZIP file of a specific release.
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, which are necessary to support older browsers and which break the Yandex.Games SDK.
- 🎓 Tutorial: "Releasing HTML5 games on Yandex.Games" - 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.
- 💬 The official Telegram chat where you can talk with representatives from Yandex. Feel free to ask questions in English!
- 💬 The Telegram chat about Defold for Russian-speaking users.
- 💬 The Defold forum topic about the YaGames extension.
- Sign up as a developer.
- Translate your game to the Russian language (tip: translate your game title into Russian too!). English and Turkish are optional (more info).
- Prepare assets for the catalogue:
- Icon 512 x 512 px.
- Cover 800 x 470 px.
- Screenshots.
- (Optional) Videos and GIF.
- Add the extension as a Defold library dependency to your project.
- 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.
- You can publish your game on Yandex.Games from this moment. It fully meets the requirements.
- 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.
- 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. - You don't need to set up any cache-busting techniques, since Yandex.Games hosts each version of your game in separate paths.
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 to make sure that you implemented SDK in the right way.
To get started, you need to initialize the SDK using the init
method.
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
end
function init(self)
yagames.init(init_handler)
end
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 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 thewas_shown
argument (typeboolean
), 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.
local yagames = require("yagames.yagames")
local function adv_open(self)
-- You should switch off all sounds!
end
local function adv_close(self, was_shown)
-- You can switch sounds back!
end
local function adv_offline(self)
-- Internet is offline
end
local function adv_error(self, err)
-- Something wrong happened :(
end
function 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
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
.
local yagames = require("yagames.yagames")
local function rewarded_open(self)
-- You should switch off all sounds!
end
local function rewarded_rewarded(self)
-- Add coins!
end
local function rewarded_close(self)
-- You can switch sounds back!
end
local function rewarded_error(self, err)
-- Something wrong happened :(
end
function 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
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.
Yandex's Native Cache lets users use games offline. Currently, it's available only in Yandex Browser or the Yandex app on smartphones.
- Set the path to the file
yandex-manifest.json
in thegame.project
settings. - Copy the
yagames/manifests/web/yandex-manifest.json
file to the root directory of your release build. - Edit the list of all game files inside your
yandex-manifest.json
, and update the path to the icon. Omitsw.js
andyandex-manifest.json
.
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.
- Set the path to the file
sw.js
in thegame.project
settings. - Copy the
yagames/manifests/web/sw.js
file to the root directory of your release build. - Edit the list of all game files inside your
sw.js
. Omitsw.js
andyandex-manifest.json
. - You should increment the version inside
sw.js
on every update of your game on Yandex.Games.
[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 JSYaGames.init
function. Example:{ orientation: { value: "landscape", lock: true } }
.sdk_init_snippet
- JavaScript code that is passed as-is and called when theysdk
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'ssw.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.
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.
The best way to integrate SDK into your game is to read the official documentation 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 |
Advertisement (docs) | |
ysdk.adv.showFullscreenAdv({callbacks:{}}) |
yagames.adv_show_fullscreen_adv(callbacks) Example |
ysdk.adv.showRewardedVideo({callbacks:{}}) |
yagames.adv_show_rewarded_video(callbacks) Example |
Advertisement - Sticky Banners (docs) | |
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) | |
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() |
yagames.player_get_id() |
player.getUniqueID() |
yagames.player_get_unique_id() |
player.getIDsPerGame() |
yagames.player_get_ids_per_game(callback) |
player.getMode() |
yagames.player_get_mode() (more info) |
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) | |
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) | |
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) | |
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) | |
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) | |
ysdk.clipboard.writeText(text) |
yagames.clipboard_write_text(text, [callback]) |
Device Info (docs) | |
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) | |
ysdk.environment |
yagames.environment() Returns Lua table { app = { id = ... }, ... } |
Screen (docs) | |
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) | 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) | |
ysdk.onEvent(eventName, listener) |
yagames.event_on(event_name, listener) |
ysdk.dispatchEvent(eventName) |
yagames.event_dispatch(event_name) |
Remote Config (docs) | |
ysdk.getFlags(options) |
yagames.flags_get(options, callback) Options is optional. The callback result is a table like { flagName = "value" } |
Sitelock (docs) | |
sitelock.add_domain(domain) |
|
sitelock.verify_domain() |
|
sitelock.get_current_domain() |
|
sitelock.is_release_build() |
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).
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
Artsiom Trubchyk (@aglitchman) is the current YaGames owner within Indiesoft and is responsible for the open source repository.
This project uses the source code of JsToDef.
MIT license.