Skip to content

OLC PGE Mobile 2.0

John Galvin edited this page Oct 6, 2024 · 14 revisions

Overview

This wiki gives a full overview of every function and method used in the OLC PGE Mobile 2.0.

Definitions

TODO : Each function and method within the olcPixelGameEngine_Mobile.cpp will have a link back to this page, so that the developer can access more information on what the function/method does. Add more verbiage

main.cpp

This is the main entry point for any application that uses 'olcPixelGameEngine_Mobile.cpp'. It is from this class you start and create all our your amazing PGE mobile games.

Headers:

To get our OLC PGE Engine up and running on both Android & iOS we need a few headers and some #define. These work together to allow you to just development your application/game using the amazing PGE 2.0 and C++ without having to handle all complicitly under the hood to get your code to work smoothly on both operating systems.

We like to keep things super simple here at the One Lone Coder Community so this is all done and dusted for you :)

  • pch.h Pre-Complied Header, this is used to setup and complied required headers, makes comply times faster, and helps us to ensure the correct sub headers for Android & iOS are complied and ready for you to use. NOTE the iOS pch.h file is handled within the Objective C
  • #include "../_ProjectName_/pch.h" the ProjectName is auto generated, do not change, well you can, but I wouldn't ;)
  • #if defined (__ANDROID__) Complier Talk Tells the complier to use this block of code to for Android Devices
#if defined (__ANDROID__)

#include "../<ProjectName>/pch.h"

#endif //__ANDROID__
  • #if defined (__APPLE__) Complier Talk Tells the complier to use this block of code to for APPLE iOS Devices
  • ios_native_app_glue.h the is a GLUE interface between the PGE Mobile 2.0 and the iOS application.
  • #include <memory> there are a few Smart Pointers needed to get thing running and linking, therefore we include the memory.h here
#if defined (__APPLE__)

#include "ios_native_app_glue.h"
#include <memory>

#endif
  • SIMD (SSE2) is enabled by default for all x86/64 CPUs, and is auto disabled when an ARM/ARM64 CPU is detected
  • #define STBI_NO_SIMD Complier Talk Tells the complier not to use SIMD. This is commented out by default.
  • SIMD/NEON greatly improves the speed of your game
  • NOTE Mobile devices CPU/GPU are very weak compared to their big sisters in computers. SIMD/NEON technology is used to greatly reduce this gap.
//#define STBI_NO_SIMD // Removes SIMD Support
// SIMD greatly improves the speed of your game
  • #if defined(__arm__)||(__aarch64__) Complier Talk Tells the complier to use this block of code when an ARM or ARM64 CPU is detected
  • #define STBI_NEON for ARM/ARM64, Complier Talk Tells the complier to use use Advance SIMD NEON when loading and processing images.
  • Nearly every mobile device today use ARM/ARM64, except of course for the Intel Atom used in most Chrome Books and some Tablets
  • NOTE Mobile devices CPU/GPU are very weak compared to their big sisters in Computers. SIMD/NEON technology is used to greatly reduce this gap.
#if defined(__arm__)||(__aarch64__)

#define STBI_NEON

#endif
  • #define OLC_PGE_APPLICATION Complier Talk Tells the complier to use the olcPixelGameEngine.h as a header-only library, where the header both declares and also defines the bodies of the methods.
  • #define OLC_IMAGE_STB Complier Talk Tells the complier to use STBImage.h for image processing, we need to use this over libpng.h as it supports SIMD/NEON, otherwise, as stated above, things would be too slow.
  • Nearly every mobile device today use ARM/ARM64, except of course for the Intel Atom used in most Chrome Books and some Tablets
  • #include "olcPixelGameEngine_Mobile.h" IMPORTANT: we must include olcPixelGameEngine_Mobile.h not olcPixelGameEngine.h , the mobile edition ; althought contains the same functionality; is very different under the hood.
  • The OLC PGE 2.0 Mobile only supports Android and iOS only.
  • NOTE OLC PGE 2.0 Mobile has support for iOS Apps downloaded from the Apple Store to a MAC only.
#define OLC_PGE_APPLICATION
#define OLC_IMAGE_STB
#include "olcPixelGameEngine_Mobile.h"
  • #define OLC_PGEX_MINIAUDIO Complier Talk Tells the complier to use OLC PEGX Mini Audio for sounds
  • Checkout OLC PGEX MiniAudio Thanks Moro1138
#define OLC_PGEX_MINIAUDIO
#include "olcPGEX_MiniAudio.h"  // Checkout https://github.com/Moros1138/olcPGEX_MiniAudio Thanks Moros1138
  • #include <fstream> Used for saving the SaveState to a file, see add link to OnSaveStateRequested
#include <fstream> // Used for saving the SaveState to a file

Class

To ensure proper cross platform support between Android and iOS it is recommended to keep the class name as PGE_Mobile, This will ensure the iOS can launch the engine correctly, if you change it make the required changes in <TODO: Link> GameViewController.mm in the iOS app to suit.

  • class PGE_Mobile : public olc::PixelGameEngine creates an instance of our game class with a public interface of olc::PixelGameEngine, in short gives us access to all the cool and wonderful functionalities of the Pixel Game Engine

  • PGE_Mobile() { ... } Class constructor. NOTE Never load an image/object from the constructor, always use OnUserCreate() (TODO: Add link to why this can break on iOS)

  • sAppName = "Android/iOS Demo"; Stores the name of your game / application

  • For information on OnUserCreate() , OnUserUpdated() and OnUserDestroy() please see the OLC PGE 2.0 wiki as these are standard PGE 2.0 methods

  • void OnSaveStateRequested() override fires when your application as lost focus and might be destoryed. i.e. An incoming phone call, Text Message, An Alert, anything where the user has moved away from your application.

    • The void OnDestory() override maybe not always fire, as it depends on the current actions of the user. However once the void OnDestory() override is fired it means the OS is about to destroy your application. It is important you save what details you need.
    • void OnSaveStateRequested() override will always be the last method fired, i.e. it will always fire after an OnUserDestory() if requested from the OS.
    • You have limited time to save your game state, save only important information i.e. Player Location, Game Level etc. Do not try to store any memory objects, like Smart Points, Vectors etc, as once an application is destroyed all memory associated with the application is also destroyed.
  • void OnRestoreStateRequested() override { fires when your application loads.

    • Fires when your application loads regardless if the application was destroyed previously.
    • Fires when your application regains focus, i.e. after an incoming call, text message etc
    • Fires before the OnUserCreate()
    • Use this method to reload your game state
  • void OnLowMemoryWarning() override { , fires when the OS detects it is low on memory

    • This method is fired on all currently active applications
    • Use this method to clean up any unused, leftover or not needed memory (Sprites, Decals, vectors)
    • Use this method to clear any vectors or blocks of memory that can be reloaded later
    • Use this method to save memory to a file
    • If the OS cannot gain back enough memory from its applications it will force close these applications
    • Please follow the steps above to ensure your application is not force closed
    • BTW Most Application Developers never implement this method, and then wonder why their app is closed at random, don't be one them please!
class PGE_Mobile : public olc::PixelGameEngine {

public:

    PGE_Mobile() {
        sAppName = "Android/iOS Demo";
    }

    bool OnUserCreate() override {
         // Load images, objects, sounds etc etc
    }

    bool OnUserUpdate(float fElapsedTime) override {
        // Execute Game Logic fires once a frame
        return true;
    }

    bool OnUserDestroy() override {
        // Fires when the app is about to be destroyed
        return true;
    }

    void OnSaveStateRequested() override {
        // Fires when the app is about to be destroyed
        
    }

   void OnRestoreStateRequested() override {
        // Fires when the app is restored
        
    }

    void OnLowMemoryWarning() override {
        // Fires when the OS is low on memory, reduce your memory usage or risk your app been closed
        
    }



}

Entry points

There are two entry points for Android and iOS respectfully. in order to maintain cross platform compatibility the entry points for both operating systems are different, as both Operation Systems work completely different.

The good news is once you have your entry point setup correctly, as detailed below, everything after this is the same for both Operating Systems.

We like to keep things super simple here at the One Lone Coder Community


Android:

  • #if defined (__ANDROID__) Complier Talk Tells the complier to use this block of code to for Android Devices
#if defined (__ANDROID__)
/*
* Code stuff
*/

#endif //__ANDROID__
  • void android_main() this is the main entry point of a native application
  • It runs in its own thread, with its own event loop for receiving input events and doing other things.
  • This is now what drives the engine, the thread is controlled from the OS
void android_main(struct android_app* initialstate) {
/*
* Code stuff
*/
}
  • struct android_app* initialstate initalstate allows you to make some more edits to your app before the PGE Engine starts
  • Recommended just to leave it at its defaults, but change it at your own risk
  • to access the Android directly in your code android_app* pMyAndroid = this->pOsEngine.app;
void android_main(struct android_app* initialstate) {
/*
* initialstate stuff
* <TODO> : Create a section for explaining intialstate
*/
}
  • PGE_Mobile demo class instantiation, starts up the PGE Engine
  • demo.Construct(1280, 720, 2, 2, true, false, false); Note it is best to use HD(1280, 720, ? X ? pixel, Fullscreen = true) the engine can scale this best for all screen sizes, without affecting performance... well it will have a very small affect, it will depend on your pixel size. Note: cohesion is currently not working.
  • demo.Start(); Start the application
 PGE_Mobile demo;
 demo.Construct(1280, 720, 2, 2, true, false, false);
 demo.Start(); // Lets get the party started
  • Example:
#if defined (__ANDROID__)

void android_main(struct android_app* initialstate) {

    PGE_Mobile demo;

    demo.Construct(1280, 720, 2, 2, true, false, false);

    demo.Start(); // Lets get the party started

}

#endif

iOS:

  • #if defined (__APPLE__) Complier Talk Tells the complier to use this block of code to for APPLE iOS Devices
#if defined (__APPLE__)
/*
* Code stuff
*/

#endif //__APPLE__
  • int ios_main(IOSNativeApp* pIOSNatvieApp) The is the entry point from the iOS Objective C, called during the start-up of your application
  • Use the objects defined in IOSNativeApp to pass data to the Objective C
  • By Default you must at minimum pass the game construct vars, pIOSNatvieApp->SetPGEConstruct
  • iOS runs in its own threads, with its own event loop for receiving input events and doing other things.
  • This is now what drives the engine, the thread is controlled from the OS
int ios_main(IOSNativeApp* pIOSNatvieApp) {
/*
* Code Stuff
*/
}
  • int ios_main(IOSNativeApp* pIOSNatvieApp) pIOSNatvieApp allows access to the OS App options
  • The iOS will instance your app differently to how Android does it
  • In the iOS it will automatically create the required classes and pointers to get the PGE up and running successfully.
  • IMPORTANT: You must set your class name to PGE_Mobile (see above) always for iOS
  • Don't worry it will not conflict with any other apps that use the same base class name of PGE_Mobile
  • To access the iOS directly in your code auto* pMyApple = this->pOsEngine.app;
int ios_main(IOSNativeApp* pIOSNatvieApp) {
/*
* pIOSNatvieApp-> <TODO> Create section explaining options
*/
}
  • pIOSNatvieApp->SetPGEConstruct(0, 0, 2, 2, true, true, false); Note it is best to use HD(0, 0, ? X ? pixel, Fullscreen = true) the engine can scale this best for all screen sizes, without affecting performance... well it will have a very small affect, it will depend on your pixel size Note: cohesion is currently not working.
  • return EXIT_SUCCESS; You must return EXIT_SUCCESS in order to tell the iOS to start up the engine, you can return EXIT_FAILURE, error codes etc will stop the startup of the engine and the app will exit gracefully
pIOSNatvieApp->SetPGEConstruct(0, 0, 2, 2, true, true, false);
return EXIT_SUCCESS;
  • Example:
#if defined(__APPLE__)

int ios_main(IOSNativeApp* pIOSNatvieApp)
{
    
    pIOSNatvieApp->SetPGEConstruct(0, 0, 2, 2, true, true, false);
    return EXIT_SUCCESS;
}

#endif

Please Note: Although this is considered to be the main entry point, under the hood it is not, the main entry point comes from the phone OS and is used to start up all the threads, memory, smart pointers etc in order for the PGE Engine to start up. The reason we state that main.cpp is the main entry point is to allow cross compatibility with the OLC PGE 2.0 engine for Windows, MAC, Linux, PlayStation and the list goes on.

main.cpp Example:

//////////////////////////////////////////////////////////////////
// Pixel Game Engine Mobile Release 2.2.8                      //
// John Galvin aka Johnngy63: 18-Jun-2024                       //
// iOS Sensor NOT supported, coming soon                        //
// Please report all bugs to https://discord.com/invite/WhwHUMV //
// Or on Github: https://github.com/Johnnyg63					//
//////////////////////////////////////////////////////////////////

//
// Base Project
//


// Set up headers for the different platforms
#if defined (__ANDROID__)

#include "../ForWiki/pch.h"
//#include "pch.h"

#endif

#if defined (__APPLE__)

#include "ios_native_app_glue.h"
#include <memory>

#endif

//#define STBI_NO_SIMD // Removes SIMD Support
// SIMD greatly improves the speed of your game
#if defined(__arm__)||(__aarch64__)

// Use Advance SIMD NEON when loading images for STB Default is SSE2 (x86)
#define STBI_NEON

#endif

#define OLC_PGE_APPLICATION
#define OLC_IMAGE_STB
#include "olcPixelGameEngine_Mobile.h"

#define OLC_PGEX_MINIAUDIO
#include "olcPGEX_MiniAudio.h"  // Checkout https://github.com/Moros1138/olcPGEX_MiniAudio Thanks Moros1138

#include <fstream> // Used for saving the savestate to a file

/// <summary>
/// To ensure proper cross platform support keep the class name as PGE_Mobile
/// This will ensure the iOS can launch the engine correctly
/// If you change it make the required changes in GameViewController.mm in the iOS app to suit
/// </summary>
class PGE_Mobile : public olc::PixelGameEngine {

public:

    PGE_Mobile() {
        sAppName = "Android/iOS Demo";
    }

    /* Vectors */
    std::vector<std::string> vecMessages;
    /* END Vectors*/

    int nFrameCount = 0;
    int nStep = 20;

    /* Sprites */
    olc::Sprite* sprTouchTester = nullptr;
    olc::Sprite* sprOLCPGEMobLogo = nullptr;
    /* END Sprites*/

    /* Decals */
    olc::Decal* decTouchTester = nullptr;
    olc::Decal* decOLCPGEMobLogo = nullptr;
    /* End Decals */


    /* Sensors */
    std::vector<olc::SensorInformation> vecSensorInfos;
    /*End Sensors*/

    // To keep track of our sample ID
    // Ensure that all sound IDs are set to -1 stop memory leak when Android/iOS takes
    // the app out of focus
    int32_t song1 = -1;

    // For demonstration controls, with sensible default values
    float pan = 0.0f;
    float pitch = 1.0f;
    float seek = 0.0f;
    float volume = 1.0f;

    // The instance of the audio engine, no fancy config required.
    olc::MiniAudio ma;


public:
    //Example Save State Struct and Vector for when your app is paused
    struct MySaveState {
        std::string key;
        int value;
    };

    std::vector<MySaveState> vecLastState;

    std::string sampleAFullPath; // Holds the full path to sampleA.wav

public:
    bool OnUserCreate() override {
        //NOTE: To access the features with your phone device use:
#if defined(__ANDROID__)
        // Access android directly
        //android_app* pMyAndroid = this->pOsEngine.app;

        // USE OF SOUND olcPGE_MiniAudio
        /*
         * For Android you cannot play the sounds directly from the assets as you would
         * on a Windows/Mac/Linux system. Android compress your assets in a compress file to
         * save on valuable phone storage. AndroidAudio (AAudio), miniAudio, and most others need
         * to be able to stream the music data, in short they are not very good at streaming for
         * a compress file.
         * Therefore you will need to extract these sound files to internal storage so the sounds
         * can be played.
         *
         * In short, as I know you didn't read the above, you cannot stream from an asset in Android
         *
         */

        std::string songFullPath = (std::string)app_GetInternalAppStorage() + "/sounds/song1.mp3";
        olc::rcode fileRes = olc::filehandler->ExtractFileFromAssets("sounds/song1.mp3", songFullPath);

        switch (fileRes) {

        case olc::rcode::NO_FILE:
        { break; }
        case olc::rcode::FAIL:
        { break; }
        case olc::rcode::OK:
        {
            // only load the song if it is not already loaded
            song1 = ma.LoadSound(songFullPath);

            break;
        }
        }

        sampleAFullPath = (std::string)app_GetInternalAppStorage() + "/sounds/SampleA.wav";
        olc::filehandler->ExtractFileFromAssets("sounds/SampleA.wav", sampleAFullPath);


#endif

#if defined(__APPLE__)
        // Access iOS directly
        //apple_app* pMyApple = this->pOsEngine.app;

        // USE OF SOUND olcPGE_MiniAudio

        std::string songFullPath = (std::string)app_GetInternalAppStorage() + "/sounds/song1.mp3";
        olc::rcode fileRes = olc::filehandler->ExtractFileFromAssets("sounds/song1.mp3", songFullPath);

        switch (fileRes) {

        case olc::rcode::NO_FILE:
        { break; }
        case olc::rcode::FAIL:
        { break; }
        case olc::rcode::OK:
        {
            if (song1 < 0)
            {
                song1 = ma.LoadSound(songFullPath);
            }

            break;
        }
        }

        sampleAFullPath = (std::string)app_GetInternalAppStorage() + "/sounds/SampleA.wav";
        olc::filehandler->ExtractFileFromAssets("sounds/SampleA.wav", sampleAFullPath);

#endif

        sprTouchTester = new olc::Sprite("images/north_south_east_west_logo.png");
        decTouchTester = new olc::Decal(sprTouchTester);

        sprOLCPGEMobLogo = new olc::Sprite("images/olcpgemobilelogo.png");
        decOLCPGEMobLogo = new olc::Decal(sprOLCPGEMobLogo);


        return true;
    }

    // <summary>
    /// Draws a Target Pointer at the center position of Center Point
    /// </summary>
    /// <param name="vCenterPoint">Center Position of the target</param>
    /// <param name="nLineLenght">Length of lines</param>
    /// <param name="nCircleRadus">Center Circle radius</param>
    void DrawTargetPointer(olc::vi2d vCenterPoint, int32_t nLineLenght, int32_t nCircleRadus, olc::Pixel p = olc::WHITE)
    {
        /*
                        |
                        |
                    ----O----
                        |
                        |


        */
        FillCircle(vCenterPoint, nCircleRadus, p);
        DrawLine(vCenterPoint, { vCenterPoint.x, vCenterPoint.y + nLineLenght }, p);
        DrawLine(vCenterPoint, { vCenterPoint.x, vCenterPoint.y - nLineLenght }, p);
        DrawLine(vCenterPoint, { vCenterPoint.x + nLineLenght, vCenterPoint.y }, p);
        DrawLine(vCenterPoint, { vCenterPoint.x - nLineLenght, vCenterPoint.y }, p);

    }

    bool OnUserUpdate(float fElapsedTime) override {

        SetDrawTarget(nullptr);

        Clear(olc::BLUE);

        nFrameCount = GetFPS();

        std::string sLineBreak = "-------------------------";

        std::string sMessage = "OneLoneCoder.com";
        vecMessages.push_back(sMessage);

        sMessage = "PGE Mobile Release 2.2.8";
        vecMessages.push_back(sMessage);

        sMessage = "Now With iOS Support";
        vecMessages.push_back(sMessage);

        sMessage = "NOTE: Android FPS = CPU FPS, iOS = GPU FPS";
        vecMessages.push_back(sMessage);

        sMessage = sAppName + " - FPS: " + std::to_string(nFrameCount);
        vecMessages.push_back(sMessage);

        sMessage = "---";
        vecMessages.push_back(sMessage);


        sMessage = "Volume <" + std::to_string(volume) + "> Btn Up, Btn Down";
        vecMessages.push_back(sMessage);

        if (ma.IsPlaying(song1))
        {
            sMessage = "Touch Screen: Pause";
            vecMessages.push_back(sMessage);
        }
        else
        {
            sMessage = "Touch Screen: Play";
            vecMessages.push_back(sMessage);
        }

        sMessage = "---";
        vecMessages.push_back(sMessage);

        sMessage = "Music: Joy Ride [Full version] by MusicLFiles";
        vecMessages.push_back(sMessage);
        sMessage = "Free download:";
        vecMessages.push_back(sMessage);
        sMessage = "https://filmmusic.io/song/11627-joy-ride-full-version";
        vecMessages.push_back(sMessage);
        sMessage = "Licensed under CC BY 4.0:";
        vecMessages.push_back(sMessage);
        sMessage = "https://filmmusic.io/standard-license";
        vecMessages.push_back(sMessage);
        vecMessages.push_back(sLineBreak);

        std::string sTouchScreen = "Touch the screen with two fingers";
        vecMessages.push_back(sTouchScreen);

        vecMessages.push_back(sLineBreak);

        olc::vi2d centreScreenPos = GetScreenSize();
        centreScreenPos.x = centreScreenPos.x / 2;
        centreScreenPos.y = centreScreenPos.y / 2;
        DrawTargetPointer(centreScreenPos, 50, 10);

        // Get the default touch point
        // This is alway Index 0 and first touch piont
        olc::vi2d defautTouchPos = GetTouchPos();
        std::string defautTouch = "Default Touch 0:  X: " + std::to_string(defautTouchPos.x) + " Y: " + std::to_string(defautTouchPos.y);
        vecMessages.push_back(defautTouch);

        if (GetTouch().bHeld)
        {
            DrawLine(centreScreenPos, defautTouchPos, olc::YELLOW, 0xF0F0F0F0);
            DrawTargetPointer(defautTouchPos, 50, 10, olc::YELLOW);
        }

        /*
            You asked for Multi-touch... you got it!
            You can support up to 126 touch points, however most phones and tablets can only handle 5

            As always with touch sensors it is an approximate and always will be
            I would recommend no more that 3 points

            When you are using lots of touch points it is best to run ClearTouchPoints();
            every so often to ensure lost touch points are cleared

        */

        olc::vi2d touchPos;
        // The more touch points the harder to manage
        for (int i = 1; i < 5; i++)
        {
            if (GetTouch(i).bHeld)
            {

                touchPos = GetTouchPos(i);
                std::string TouchID = "Touch ID: " + std::to_string(i) + " X: " + std::to_string(touchPos.x) + " Y: " + std::to_string(touchPos.y);
                vecMessages.push_back(TouchID);
                DrawLine(centreScreenPos, touchPos, olc::WHITE, 0xF0F0F0F0);
                DrawTargetPointer(touchPos, 50, 10);

            }
        }

        // Called once per frame, draws random coloured pixels
        // Uncomment me if you dare
        /*for (int x = 0; x < ScreenWidth(); x++)
            for (int y = 0; y < ScreenHeight(); y++)
                Draw(x, y, olc::Pixel(rand() % 256, rand() % 256, rand() % 256));
        */

        nStep = 10;
        for (auto& s : vecMessages)
        {
            DrawString(20, nStep, s);
            nStep += 10;
        }
        vecMessages.clear();




        if (GetTouch(0).bPressed) {
            // Toggle takes a sample ID (int) and either starts playback or pauses playback
            // depending on whether the sample is currently playing, or not.
            ma.Toggle(song1);
        }

        if (GetTouch(1).bPressed) {
            ma.Play(sampleAFullPath);
        }

        if (GetKey(olc::A).bHeld)
        {
            volume += 1.0f * fElapsedTime;
            if (volume > 1.0f) volume = 1.0f;
        }

        if (GetKey(olc::B).bHeld)
        {
            volume -= 1.0f * fElapsedTime;
            if (volume < 0.0f) volume = 0.0f;
        }

        if (GetKey(olc::VOLUME_DOWN).bHeld) {
            // NOTE: Android volume buttons can be read but cannot be captured
            // NOTE: iOS volume buttons cannot be read and cannot be captured
            volume -= 1.0f * fElapsedTime;
            if (volume < 0.0f) volume = 0.0f;
        }

        if (GetKey(olc::VOLUME_UP).bHeld) {
            // NOTE: Android volume buttons can be read but cannot be captured
            // NOTE: iOS volume buttons cannot be read and cannot be captured
            volume += 1.0f * fElapsedTime;
            if (volume > 1.0f) volume = 1.0f;
        }

        // Set volume, takes a sample ID (int), and a float
        // 0.0 to 1.0 where 1.0 is full volume
        ma.SetVolume(song1, volume);

        // Gets the current playback position in the provided sample ID (int),
        // returns float 0.0 to 1.0, nearer 1.0 is near the end
        seek = ma.GetCursorFloat(song1);


        // Draw Logo
        DrawDecal({ 5.0f, (float)ScreenHeight() - 100 }, decOLCPGEMobLogo, { 0.5f, 0.5f });

        // Draw The Playback Cursor (aka the position in the sound file)
        FillRect({ 0, ScreenHeight() - 10 }, { (int)(ScreenWidth() * seek), 10 }, olc::YELLOW);

        return true;
    }

    bool OnUserDestroy() override {
        return true;
    }

    void OnSaveStateRequested() override
    {
        // Fires when the OS is about to put your game into pause mode
        // You have, at best 30 Seconds before your game will be fully shutdown
        // It depends on why the OS is pausing your game though, Phone call, etc
        // It is best to save a simple Struct of your settings, i.e. current level, player position etc
        // NOTE: The OS can terminate all of your data, pointers, sprites, layers can be freed
        // Therefore do not save sprites, pointers etc 

        // Example 1: vector
        vecLastState.clear();
        vecLastState.push_back({ "MouseX", 55 });
        vecLastState.push_back({ "MouseY", 25 });
        vecLastState.push_back({ "GameLevel", 5 });

#if defined(__ANDROID__)
        // You can save files in the android Internal app storage
        const char* internalPath = app_GetInternalAppStorage(); //Android protected storage
#endif
#if defined(__APPLE__)
        // For iOS the internal app storage is read only, therefore we use External App Storage
        const char* internalPath = app_GetExternalAppStorage(); // iOS protected storage AKA /Library
#endif

        std::string dataPath(internalPath);

        // internalDataPath points directly to the files/ directory                                  
        std::string lastStateFile = dataPath + "/lastStateFile.bin";

        std::ofstream file(lastStateFile, std::ios::out | std::ios::binary);

        if (file)
        {
            float fVecSize = vecLastState.size();
            file.write((char*)&fVecSize, sizeof(long));
            for (auto& vSS : vecLastState)
            {
                file.write((char*)&vSS, sizeof(MySaveState));
            }

            file.close();
        }


    }

    void OnRestoreStateRequested() override
    {
        // This will fire every time your game launches 
        // OnUserCreate will be fired again as the OS may have terminated all your data

#if defined(__ANDROID__)
        // You can save files in the android Internal app storage
        const char* internalPath = app_GetInternalAppStorage(); //Android protected storage
#endif
#if defined(__APPLE__)
        // For iOS the internal app storage is read only, therefore we use External App Storage
        const char* internalPath = app_GetExternalAppStorage(); // iOS protected storage AKA /Library
#endif

        std::string dataPath(internalPath);
        std::string lastStateFile = dataPath + "/lastStateFile.bin";

        vecLastState.clear();

        std::ifstream file(lastStateFile, std::ios::in | std::ios::binary);

        MySaveState saveState;

        if (file)
        {
            float fVecSize = 0.0f;
            file.read((char*)&fVecSize, sizeof(long));
            for (long i = 0; i < fVecSize; i++)
            {
                file.read((char*)&saveState, sizeof(MySaveState));
                vecLastState.push_back(saveState);
            }

            file.close();
            // Note this is a temp file, we must delete it
            std::remove(lastStateFile.c_str());

        }


    }

};


#if defined (__ANDROID__)
/**
* This is the main entry point of a native application that is using
* android_native_app_glue.  It runs in its own thread, with its own
* event loop for receiving input events and doing other things.
* This is now what drives the engine, the thread is controlled from the OS
*/
void android_main(struct android_app* initialstate) {

    /*
        initalstate allows you to make some more edits
        to your app before the PGE Engine starts
        Recommended just to leave it at its defaults
        but change it at your own risk
        to access the Android/iOS directly in your code
        android_app* pMyAndroid = this->pOsEngine.app;;

    */

   
    PGE_Mobile demo;

    /*
        Note it is best to use HD(1280, 720, ? X ? pixel, Fullscreen = true) the engine can scale this best for all screen sizes,
        without affecting performance... well it will have a very small affect, it will depend on your pixel size
        Note: cohesion is currently not working
    */
    demo.Construct(1280, 720, 2, 2, true, false, false);

    demo.Start(); // Lets get the party started


}

#endif

#if defined(__APPLE__)

/*
* The is the calling point from the iOS Objective C, called during the start-up of your application
* Use the objects definded in IOSNativeApp to pass data to the Objective C
* By Default you must at minmum pass the game construct vars, pIOSNatvieApp->SetPGEConstruct
*
* iOS runs in its own threads, with its own
* event loop for receiving input events and doing other things.
* This is now what drives the engine, the thread is controlled from the OS
*/
int ios_main(IOSNativeApp* pIOSNatvieApp)
{
    // The iOS will instance your app differently to how Android does it
    // In the iOS it will automatically create the required classes and pointers
    // to get the PGE up and running successfully.

    // IMPORTANT: You must set your class name to PGE_Mobile (see above) always for iOS
    // Don't worry it will not conflict with any other apps that use the same base class name of PGE_Mobile
    // I got your back

    // Finally just like the Android you can access any available OS options using pIOSNatvieApp
    // Please note options will NOT be the same across both platforms
    // It is best to use the build in functions for File handling, Mouse/Touch events, Key events, Joypad etc

    //
    // To access the iOS directly in your code
    // auto* pMyApple = this->pOsEngine.app;
    //

    /*
        Note it is best to use HD(0, 0, ? X ? pixel, Fullscreen = true) the engine can scale this best for all screen sizes,
        without affecting performance... well it will have a very small affect, it will depend on your pixel size
        Note: cohesion is currently not working
        Note: It is best to set maintain_aspect_ratio to false, Fullscreen to true and use the olcPGEX_TransformView.h to manage your world-view
        in short iOS does not want to play nice, the screen ratios and renta displays make maintaining a full screen with aspect radio a pain to manage
    */
    pIOSNatvieApp->SetPGEConstruct(0, 0, 2, 2, true, true, false);


    // We now need to return SUCCESS or FAILURE to get the party stated!!!!
    return EXIT_SUCCESS;
}

#endif


Clone this wiki locally