/defold-liveupdate-reszip

The complete solution to use Live Update to improve the load speed of your HTML5 game.

Primary LanguageLuaCreative Commons Zero v1.0 UniversalCC0-1.0

ResZip Cover

ResZip

A ready-to-use Lua module and example project on how to use LiveUpdate to download extra content in your game. In the world of HTML5 games, we use this to speed up the delivery of the game, as every second counts!

Detailed Explanation

It's important to deliver something meaningful to the user as soon as possible — the longer they wait for the game to load, the bigger the chance they will leave before waiting for everything to finish. Why? The video explains that well: https://vimeo.com/350139974

You can do the following things:

  • Split your Defold game resources into two parts: resources required for the first level plus everything else. The game lazily loads more content or loads it on demand while players are playing the first level.
  • Or make SD/HD versions of your atlases to lazily load HD graphics on game start.

Defold has the Live Update feature that we can use to implement these ideas, and the project aims to demonstrate the usage of it. The project contains:

  • The Lua liveupdate_reszip.reszip module that downloads (with progress!) and mounts the missing resources.
  • The Bash script (example_build_script.sh) shows you how to automatically build your game for the web and move the resources.zip file to the build result folder.

Check out the online demos:

  1. Demo 1 - this project. Tap anywhere to load level 2.
  2. Demo 2 - the same but with an alternative resources.zip file to test that it can handle upgrade of the game.
  3. Demo 3 - same thing, but using the old LiveUpdate API. To test how Defold will handle this if we next open demo 1 or 2..

Current Status

💬 Feel free to ask questions: the topic about this asset is on the Defold forum.

Asset Version Defold Version Status
1.5.0 1.6.2 Tested ✅

Showcase

This is a list of some games that have used ResZip:

Game Links Extra
Duo Vikings Play it on Poki
Duo Vikings 2 Play it on Poki
Duo Vikings 3 Play it on Poki
Fish Eat Fish Play it on Poki The total size of the game is only 4 MB! After its start, it downloads the HD version of the graphics (~10MB) and applies it depending on the hardware capabilities.
Monkey Mart Play it on Poki In the zip archive the game stores and downloads the resources needed for the next levels (shops).
Puffy Cat Play it on Poki The game loads only 750 KB of data for the first three levels. Everything else (5 MB) is lazily downloaded from the resources.zip file.
Puffy Cat 2 Play it on Poki Only music and some sounds have been cut out from the game data into the resources.zip file.

Installation

1. Add dependency

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 under project add:

https://github.com/indiesoftby/defold-liveupdate-reszip/archive/main.zip

2. Prepare your project

Follow the Live Update tutorial on the Defold website and exclude chosen collections in proxies.

Look at the example/main.script to learn how to check for the missing resources and how to download and mount the .zip resources file:

-- Paths to the resources files
local zip_filename = sys.get_config("liveupdate_reszip.filename", "resources.zip")
local zip_file_location = zip_filename
if not html5 then
    -- You must host your resource file on any hosting service:
    "http://localhost:8080/" .. zip_filename
end

-- URL of the excluded proxy:
local excluded_proxy_url = "/level2#collectionproxy"

-- We check if resources are missing and also check the version of the currently
-- mounted resources using the resource file name.
local missing_resources = collectionproxy.missing_resources(excluded_proxy_url)
if liveupdate and (not reszip.version_match(zip_filename) or next(missing_resources) ~= nil) then
    print("Some resources are missing, so download and mount the resources archive...")

    reszip.load_and_mount_zip(zip_file_location, {
        filename = zip_filename,
        delete_old_file = true,
        on_finish = function (self, err)
            if not err then
                -- All resources are loaded, finally load the level:
                print("Everything is OK, load level 2!")
                msg.post(excluded_proxy_url, hash("load"))
            else
                -- Try again?...
                print("ERROR: " .. err)
            end
        end,
        on_progress = function (self, loaded, total)
            -- Update progress in your GUI:
            -- local progress = string.format("%dKB / %dKB", loaded / 1024, total / 1024)
            -- label.set_text("#loading_progress", progress)
        end
    })
else
    -- LiveUpdate is not enabled, i.e. we test the game from IDE. Or all resources exist, so load the level:
    print("Resources are already loaded. Let's load level 2!")
    msg.post(excluded_proxy_url, hash("load"))
end

Important

The example above assumes that you name the resource file differently for each version of the game (resource_v1.zip, resources_v2.zip), because reszip uses the file name to determine whether resources need to be updated.

3. Build your project

Open Project / Live Update Settings and enable the Zip mode for Live Update. Publish the Live Update content through Project / Bundle... or by using Bob the builder bob.jar (the most important arg is --liveupdate yes). Move the resulting .zip file with resources into your production build folder.

Important

The included Bash script (example_build_script.sh) shows you how to automatically build your game for the web and move the resources.zip file to the build result folder.

4. Summary

  • Put the content that will be downloaded later into the collection proxy.
  • Exclude this collection proxy so that Defold will put it in the Live Update zip archive with resources.
  • For every unique version of the project, set a unique name for the resource file. ResZip uses resources filename to check if the current version of resources is compatible with what the player already has. This is important for updating your game between versions!
  • Use simple-to-use ResZip API to download and mount the resource file.

Tips

The easiest way to use ResZip in your project is to move some of your audio files (i.e. sound components) to a proxied collection and exclude the collection for the release build. To play these sounds, you should make an external script that acts as a sound manager of all your in-game audio and knows when proxied sounds are loaded from the resources.zip file.

Advanced Usage (HTML5 only)

Preload resources

ResZip can start preloading the resources.zip file as soon as game loading is finished. It's highly recommended to enable this option because the engine initialisation takes some time, during which we can already start loading resources:

[liveupdate_reszip]
preload_file = your_resources_file_name.zip

If you suspect a bug in Live Update or ResZip, the following option can be used to force Live Update data to be completely cleared before the game starts:

[liveupdate_reszip]
wipe_on_start = 1

Credits

This project is licensed under the terms of the CC0 1.0 Universal license. It's developed and supported by @aglitchman.

The demo contains third-party music files which require attribution:

Ethernight Club by Kevin MacLeod
Link: https://incompetech.filmmusic.io/song/7612-ethernight-club
License: https://filmmusic.io/standard-license