Skip to content

Latest commit

 

History

History
560 lines (380 loc) · 27.7 KB

app.md

File metadata and controls

560 lines (380 loc) · 27.7 KB

app.h

Small cross-platform base framework for graphical apps.

Library: app.h

Example

Here's a basic sample program which starts a windowed app and plots random pixels.

#define  APP_IMPLEMENTATION
#define  APP_WINDOWS
#include "app.h"

#include <stdlib.h> // for rand and __argc/__argv
#include <string.h> // for memset

int app_proc( app_t* app, void* user_data ) {
    APP_U32 canvas[ 320 * 200 ]; // a place for us to draw stuff
    memset( canvas, 0xC0, sizeof( canvas ) ); // clear to grey
    app_screenmode( app, APP_SCREENMODE_WINDOW );

    // keep running until the user close the window
    while( app_yield( app ) != APP_STATE_EXIT_REQUESTED ) {
        // plot a random pixel on the canvas
        int x = rand() % 320;
        int y = rand() % 200;
        APP_U32 color = rand() | ( (APP_U32) rand() << 16 );
        canvas[ x + y * 320 ] = color;

        // display the canvas
        app_present( app, canvas, 320, 200, 0xffffff, 0x000000 );
    }
    return 0;
}

int main( int argc, char** argv ) {
    (void) argc, argv;
    return app_run( app_proc, NULL, NULL, NULL, NULL );
}

// pass-through so the program will build with either /SUBSYSTEM:WINDOWS or /SUBSYSTEN:CONSOLE
extern "C" int __stdcall WinMain( struct HINSTANCE__*, struct HINSTANCE__*, char*, int ) { return main( __argc, __argv ); }

API Documentation

app.h is a single-header library, and does not need any .lib files or other binaries, or any build scripts. To use it, you just include app.h to get the API declarations. To get the definitions, you must include app.h from one single C or C++ file, and #define the symbol APP_IMPLEMENTATION before you do.

As app.h is a cross platform library, you must also define which platform you are running on, like this for Windows:

#define APP_IMPLEMENTATION
#define APP_WINDOWS
#include "app.h"

Or like this for other platforms: #define APP_IMPLEMENTATION #define APP_SDL #include "app.h"

Customization

There are a few different things in app.h which are configurable by #defines. Most of the API use the int data type, for integer values where the exact size is not important. However, for some functions, it specifically makes use of 16, 32 and 64 bit data types. These default to using short, unsigned int and unsigned long long by default, but can be redefined by #defining APP_S16, APP_U32 and APP_U64 respectively, before including app.h. This is useful if you, for example, use the types from <stdint.h> in the rest of your program, and you want app.h to use compatible types. In this case, you would include app.h using the following code:

#define APP_S16 int16_t
#define APP_U32 uint32_t
#define APP_U64 uint64_t
#include "app.h"

Note that when customizing the data types, you need to use the same definition in every place where you include app.h, as they affect the declarations as well as the definitions.

The rest of the customizations only affect the implementation, so will only need to be defined in the file where you have the #define APP_IMPLEMENTATION.

Custom memory allocators

Even though app.h attempts to minimize the memory use and number of allocations, it still needs to make some use of dynamic allocation by calling malloc. Programs might want to keep track of allocations done, or use custom defined pools to allocate memory from. app.h allows for specifying custom memory allocation functions for malloc and free. This is done with the following code:

#define APP_IMPLEMENTATION
#define APP_MALLOC( ctx, size ) ( my_custom_malloc( ctx, size ) )
#define APP_FREE( ctx, ptr ) ( my_custom_free( ctx, ptr ) )
#include "app.h"

where my_custom_malloc and my_custom_free are your own memory allocation/deallocation functions. The ctx parameter is an optional parameter of type void*. When app_run is called, you can pass in a memctx parameter, which can be a pointer to anything you like, and which will be passed through as the ctx parameter to every APP_MALLOC/APP_FREE call. For example, if you are doing memory tracking, you can pass a pointer to your tracking data as memctx, and in your custom allocation/deallocation function, you can cast the ctx param back to the right type, and access the tracking data.

If no custom allocator is defined, app.h will default to malloc and free from the C runtime library.

Custom logging function

There's a bunch of things being logged when app.h runs. It will log an informational entry with the date and time for when the app is started and stopped, it will log warnings when non-essential initialization fails, and it will log error messages when things go wrong. By default, logging is done by a simple printf to stdout. As some applications may need a different behavior, such as writing out a log file, it is possible to override the default logging behavior through defines like this:

#define APP_IMPLEMENTATION
#define APP_LOG( ctx, level, message ) ( my_log_func( ctx, level, message ) )
#include "app.h"

where my_log_func is your own logging function. Just like for the memory allocators, the ctx parameter is optional, and is just a void* value which is passed through. But in the case of logging, it will be passed through as the value off the logctx parameter passed into app_run. The level parameter specifies the severity level of the logging, and can be used to direct different types of messages to different logging systems, or filter out messages of certain severity level, e.g. supressing informational messages.

Custom fatal error function

As the app.h library works on the lowest level of your program, interfacing directly with the operating system, there might occur errors which it can not recover from. In these cases, a fatal error will be reported. By default, when a fatal error happens, app.h will print a message to stdout, show a messagebox to the user, and force exit the program.

It is possible to change this behaviour using the following define:

#define APP_IMPLEMENTATION
#define APP_FATAL_ERROR( ctx, message ) ( my_custom_fatal_error_func( ctx, message ) )
#include "app.h"

where my_custom_fatal_error_func is your own error reporting function. The ctx parameter fills the same purpose as for the allocator and logging functions, but here it is the fatalctx parameter to app_run which is passed through.

app_run

int app_run( int (*app_proc)( app_t*, void* ), void* user_data, void* memctx, void* logctx, void* fatalctx )

Creates a new app instance, calls the given app_proc and waits for it to return. Then it destroys the app instance.

  • app_proc - function pointer to the user defined starting point of the app. The parameters to that function are: app_t* a pointer to the app instance. This is an opaque type, and it is passed to all other functions in the API. void* pointer to the user defined data that was passed as the user_data parameter to app_run.
  • user_data - pointer to user defined data which will be passed through to app_proc. May be NULL.
  • memctx - pointer to user defined data which will be passed through to custom APP_MALLOC/APP_FREE calls. May be NULL.
  • logctx - pointer to user defined data to be passed through to custom APP_LOG calls. May be NULL.
  • fatalctx - pointer to user defined data to be passed through to custom APP_FATAL_ERROR calls. May be NULL.

When app_run is called, it will perform all the initialization needed by app.h, and create an app_t* instance. It will then call the user-specified app_proc, and wait for it to return. The app_t* instance will be passed to app_proc, and can be used to call other functions in the API. When returning from app_proc, a return value is specified, and app_run will perform termination and cleanup, and destroy the app_t* instance, and then return the same value it got from the app_proc. After app_run returns, the app_t* value is no longer valid for use in any API calls.

app_yield

app_state_t app_yield( app_t* app )

Allows for app.h and the operating system to perform internal house keeping and updates. It should be called on each iteration of your main loop.

The return value can be either APP_STATE_NORMAL or APP_STATE_EXIT_REQUESTED. APP_STATE_EXIT_REQUESTED means that the user have requested the app to terminate, e.g. by pressing the close button on the window, and the user defined app_proc needs to handle this, by either returning (to signal that the app should terminate) or by calling app_cancel_exit to ignore the request. A typical pattern is to display a message to the user to confirm that the app should exit. In the case of APP_STATE_NORMAL, there is no need to do anything.

app_cancel_exit

void app_cancel_exit( app_t* app )

Used to reset the APP_STATE_EXIT_REQUESTED state. See app_yield for details.

app_title

void app_title( app_t* app, char const* title )

Sets the name of the application, which is displayed in the task switcher and in the title bar of the window.

app_cmdline

char const* app_cmdline( app_t* app )

Returns the command line string used to launch the executable. This can be parsed to get command line arguments.

app_filename

char const* app_filename( app_t* app )

Returns the full filename and path of the executable. The first part of app_cmdline usually contains the name of the executable, but not necessarily the full path, depending on how it was launched. app_filename, however, always returns the full path.

app_userdata

char const* app_userdata( app_t* app )

Returns the full path to a directory where a users personal files can be stored. Depending on the access rights of the user, it may or may not be possible to write data to the same location as the executable, and instead it must be stored in a specific area designated by the operating system. app_userdata returns the path to the root if that directory. A typical use for this is to store the users savegame files, by creating a subfolder corresponding to your app, and save the data there.

app_appdata

char const* app_appdata( app_t* app )

Returns the full path to a directory where application specific files can be stored. Similar to the location returned by app_userdata, but suitable for application data shared between users. Typical use for this is to store the result of cached calculations or temporary files.

app_time_count

APP_U64 app_time_count( app_t* app )

Returns the current value of the high precision clock. The epoch is undefined, and the resolution can vary between systems. Use app_time_freq to convert to seconds. Typical use is to make two calls to app_time_count and calculate the difference, to measure the time elapsed between the two calls.

app_time_freq

APP_U64 app_time_freq( app_t* app )

Returns the number of clock ticks per second of the high precision clock. An example use case could be:

APP_U64 current_count = app_time_count( app );
APP_U64 delta_count = current_count - previous_count;
double delta_time = ( (double) delta_count ) / ( (double) app_time_freq( app ) );
previous_count = current_count;

to measure the time between two iterations through your main loop.

app_log

void app_log( app_t* app, app_log_level_t level, char const* message )

app.h will do logging on certain events, e.q when the app starts and ends or when something goes wrong. As the logging can be customized (see section on customization), it might be desirable for the program to do its own logging the same way as app.h does it. By calling app_log, logging will be done the same way as it is done inside app.h, whether custom logging or default logging is being used.

app_fatal_error

void app_fatal_error( app_t* app, char const* message )

Same as with app_log, but for reporting fatal errors, app_fatal_error will report an error the same way as is done internally in app.h, whether custom or default fatal error reporting is being used.

app_pointer

void app_pointer( app_t* app, int width, int height, APP_U32* pixels_abgr, int hotspot_x, int hotspot_y )

Sets the appearence current mouse pointer. app_pointer is called with the following parameters:

  • width, height - the horizontal and vertical dimensions of the mouse pointer bitmap.
  • pixels_abgr - width x height number of pixels making up the pointer bitmap, each pixel being a 32-bit unsigned integer where the highest 8 bits are the alpha channel, and the following 8-bit groups are blue, green and red channels.
  • hotspot_x, hotspot_y - offset into the bitmap of the pointer origin, the center point it will be drawn at.

app_pointer_default

void app_pointer_default( app_t* app, int* width, int* height, APP_U32* pixels_abgr, int* hotspot_x, int* hotspot_y )

Retrieves the width, height, pixel data and hotspot for the default mouse pointer. Useful for restoring the default pointer after using app_pointer, or for doing software rendered pointers. Called with the following parameters:

  • width, height - pointers to integer values that are to receive the width and height of the pointer. May be NULL.
  • pixels_abgr - width x height number of pixels to receive the pointer bitmap. May be NULL
  • hotspot_x, hotspot_y - pointers to integer values that are to receive the hotspot coordinates. May be NULL.

A typical pattern for calling app_pointer_default is to first call it with pixels_abgr as NULL, to query the bitmaps dimensions, and then call it again after preparing a large enough memory area.

app_pointer_pos

void app_pointer_pos( app_t* app, int x, int y )

Set the position of the mouse pointer, in window coordinates. The function app_coordinates_bitmap_to_window can be used to convert between the coordinate system of the currently displayed bitmap and that of the window.

app_pointer_limit

void app_pointer_limit( app_t* app, int x, int y, int width, int height )

Locks the mouse pointer movements to stay within the specified area, in window coordinates. The function app_coordinates_bitmap_to_window can be used to convert between the coordinate system of the currently displayed bitmap and that of the window.

app_pointer_limit_off

void app_pointer_limit_off( app_t* app )

Turns of the mouse pointer movement restriction, allowing the pointer to be moved freely again.

app_interpolation

void app_interpolation( app_t* app, app_interpolation_t interpolation )

app.h supports two different modes of displaying a bitmap. When using APP_INTERPOLATION_LINEAR, the bitmap will be drawn with bilinear interpolations, stretching it to fill the window (maintaining aspect ratio), giving it a smooth, if somwhat blurry, look. With APP_INTERPOLATION_NONE, scaling will only be done on whole pixel ratios, using no interpolation, which is particularly suitable to maintain the clean, precise look of pixel art. APP_INTERPOLATION_LINEAR is the default setting.

app_screenmode

void app_screenmode( app_t* app, app_screenmode_t screenmode )

Switch between windowed mode and fullscreen mode. APP_SCREENMODE_WINDOW is used to select windowed mode, and APP_SCREENMODE_FULLSCREEN is used to switch to fullscreen mode. APP_SCREENMODE_FULLSCREEN is the default. Note that the app.h fullscreenmode is of the "borderless windowed" type, meaning that fullscreen mode just means that the window is set to cover the entire screen, and its borders are hidden. It does not imply any exclusive locking of GPU resources. When switching from windowed to fullscreen mode on a multi-display system, the app will go fullscreen on the display that the window is currently on.

app_window_size

void app_window_size( app_t* app, int width, int height )

Sets the size of the window. If currently in APP_SCREENMODE_FULLSCREEN screen mode, the setting will not take effect until switching to APP_SCREENMODE_WINDOW. width and height specifies the size of the windows client area, not counting borders, title bar or decorations.

app_window_width/app_window_height

int app_window_width( app_t* app )
int app_window_height( app_t* app )

Returns the current dimensions of the window (which might have been resized by the user). Regardless of whether the app is currently in fullscreen or windowed mode, app_window_width and app_window_height returns the dimension the window would have in windowed mode. Width and height specifies the size of the windows client area, not counting borders, title bar or decorations.

app_window_pos

void app_window_pos( app_t* app, int x, int y )

Sets the position of the top left corner of the window. If currently in APP_SCREENMODE_FULLSCREEN screen mode, the setting will not take effect until switching to APP_SCREENMODE_WINDOW.

app_window_x/app_window_y

int app_window_x( app_t* app )
int app_window_y( app_t* app )

Returns the current position of the windows top left corner. Regardless of whether the app is currently in fullscreen or windowed mode, app_window_x and app_window_y returns the position the window would have in windowed mode.

app_displays

app_displays_t app_displays( app_t* app )

Returns a list of all displays connected to the system. For each display, the following fields are reported:

  • id - a platform specific string used to identify the display. Useful for saving which display was in use.
  • x, y - position of the top left corner of the display, relative to the primary dispay which is always at 0,0.
  • width, height - size of the display, in pixels.

app_present

void app_present( app_t* app, APP_U32 const* pixels_xbgr, int width, int height, APP_U32 mod_xbgr, APP_U32 border_xbgr )

app.h provides a very minimal API for drawing - the only thing you can really do, is provide it with a bitmap for it to display on the screen. It is then up to the rest of your program to implement code for drawing shapes or sprites onto that bitmap. When all your drawing is done, you call app_present passing it the bitmap, and it will be displayed on the screen. app_present takes the following parameters:

  • pixels_xbgr - width x height number of pixels making up the bitmap to be presented, each pixel being a 32-bit unsigned integer where the highest 8 bits are not used, and the following 8-bit groups are blue, green and red channels. This parameter may be NULL, in which case no bitmap is drawn, to allow for custom rendering. See below for details.
  • width, height - the horizontal and vertical dimensions of the bitmap
  • mod_xbgr - an rgb color value which will be automatically multiplied with each pixel, component by component, before it is displayed. Can be used to for example fade the bitmap to/from black. Set to 0xffffff for no effect.
  • border_xbgr - an rgb color value to be used as border color. The borders are the areas outside of the bitmap, which are visible when the window aspect ratio does not match that of the bitmap, so you get bars above or below it.

Since app.h uses opengl, you can also opt to not pass a bitmap to app_present, by passing NULL as the pixels_xbgr parameter (in which case the rest of the parameters are ignored). When doing this, it is up to your program to perform drawing using opengl calls. In this case app_present will work as a call to SwapBuffers only. Note that glSetViewPort will be automatically called whenever the window is resized.

app_sound_buffer_size

void app_sound_buffer_size( app_t* app, int sample_pairs_count )

The api for playing sound samples is just as minimal as that for drawing. app.h provides a single, looping sound stream, and it is up to your program to handle sound formats, voices and mixing. By calling app_sound_buffer_size, a sound stream of the specified size is initialized, and playback is started. If a sample_pairs_count of 0 is given, sound playback will be stopped.

The sound buffer is in 44100hz, signed 16-bit stereo format, and the sample_pairs_count specified how many left/right pairs of 16-bit samples the buffer will contain. As an example, to specify a 1 second stream buffer, you would give the value 44100 for the sample_pairs_count parameter, which would internally create a sound buffer of 176,400 bytes, from 44100 samples x 2 channels x 2 bytes per sample. However, since the bits per sample and number of channels are fixed, the exact byte size of the sound buffer is not important in the app.h API.

app_sound_position

int app_sound_position( app_t* app )

Returns the current playback position of the sound stream, given in the number of sample pairs from the start of the buffer. Typical use of a streaming sound buffer is to fill the buffer with data, wait for the playback position to get passed the mid point of the buffer, and then write the next bit of data over the part that has already been played, and so on.

app_sound_write

void app_sound_write( app_t* app, int sample_pairs_offset, int sample_pairs_count, APP_S16 const* sample_pairs )

Writes sample data to the sound buffer. It takes the following parameters:

  • sample_pairs_offset - the offset into the buffer of where to start writing, in number of sample pairs from the start.
  • sample_pairs_count - the number of sample pairs to write. Must be less than the buffer size.
  • sample_pairs - an array of sound samples, as signed 16-bit integers. Must be at least sample_pairs_count in length.

The sample_pairs parameter can be NULL, in which case the corresponding part of the buffer is cleared.

app_sound_volume

void app_sound_volume( app_t* app, float volume )

Sets the output volume level of the sound stream, as a normalized linear value in the range 0.0f to 1.0f, inclusive.

app_input

app_input_t app_input( app_t* app )

Returns a list of input events which occured since the last call to app_input. Each input event can be of one of a list of types, and the type field of the app_input_event_t struct specifies which type the event is. The data struct is a union of fields, where only one of them is valid, depending on the value of type:

  • APP_INPUT_KEY_DOWN, APP_INPUT_KEY_UP, APP_INPUT_DOUBLE_CLICK - use the key field of the data union, which contains one of the keyboard key identifiers from the app_key_t enumeration. APP_INPUT_KEY_DOWN means a key was pressed, or that it was held long enough for the key repeat to kick in. APP_INPUT_KEY_UP means a key was released, and is not sent on key repeats. For both these events, a key may also mean a mouse button, as those are listed in the app_key_t enum. APP_INPUT_DOUBLE_CLICK means a mouse button have been double clicked, and is not sent for keyboard keys.
  • APP_INPUT_CHAR - use the char_code field of the data union, which contains the ASCII value of the key that was pressed. This is used to read text input, and will handle things like upper/lower case, and characters which requires multiple keys to be pressed in sequence to generate one input. This means that generally, when a key is pressed, you will get both an APP_INPUT_KEY_DOWN event and an APP_INPUT_CHAR event - just use the one you are interested in, and ignore the other.
  • APP_INPUT_MOUSE_MOVE - use the mouse_pos field of the data union, which contains the x and y position of the mouse pointer, in window coordinates. The function app_coordinates_window_to_bitmap can be used to convert between the coordinate system of the window and that of the currently displayed bitmap. The APP_INPUT_MOUSE_MOVE event is sent whenever the user moves the mouse, as long as the window has focus and the pointer is inside its client area.
  • APP_INPUT_MOUSE_DELTA - use the mouse_delta field of the data union, which contains the horizontal and vertical offset which the mouse has been moved by. Ideally, these values should be in normalized -1.0f to 1.0f range, but as there is no standardisation on the hardware and os level for this, it is not possible to do, so instead the value have been scaled to give roughly normalized -1.0f to 1.0f values on a typical setup. For serious use, sensitivity settings and/or user calibration is recommended. The APP_INPUT_MOUSE_DELTA event is sent whenever the user moves the mouse, regardless of whether the window has focus or whether the pointer is inside the window or not. The APP_INPUT_MOUSE_DELTA event is a better option for reading relative mouse movements than using the APP_INPUT_MOUSE_MOVE event together with app_pointer_pos to re-center the pointer on every update.
  • APP_INPUT_SCROLL_WHEEL - use the wheel_delta field of the data union, which contains the number of clicks by which the scroll wheel on the mouse was turned, where positive values indicate that the wheel have been rotated away from the user, and negative values means it has turned towards the user. The APP_INPUT_SCROLL_WHEEL is sent every time the user turns the scroll wheel, as long as the window has focus.
  • APP_INPUT_TABLET - use the tablet field of the data union, which contains details about the pen used with a graphical tablet, if connected and installed. The x and y fields are the horizontal and vertical positions of the pen on the tablet, scaled to the coordinate system of the window. The function app_coordinates_window_to_bitmap can be used to convert between the coordinate system of the window and that of the currently displayed bitmap. The pressure field is the current pressure of the pen against the tablet, in normalized 0.0f to 1.0f range, inclusive, where 0.0f means no pressure and 1.0f means full pressure. The tip field is set to APP_PRESSED if the tip of the pen is touching the tablet at all, and to APP_NOT_PRESSED otherwise. The upper and lower fields indicate the current state of the buttons on the side of the pen. The "eraser" part of the pen is not currently supported.

app_coordinates_window_to_bitmap

void app_coordinates_window_to_bitmap( app_t* app, int width, int height, int* x, int* y )

Functions in the app.h API expects and returns coordinates in the windows coordinate system, where 0, 0 is the top left corner of the windows client area (the area inside of the window borders, excluding title bar and decorations), and width, height is the dimension, in pixels, of the client area. It is often desirable to translate a position given in window coordinates into a position on the bitmap that is being displayed - taking into account whether the window is in fullscreen or windowed mode, and how the bitmap is stretched or padded depending on which interpolation mode is being used: APP_INTERPOLATION_NONE or APP_INTERPOLATION_LINEAR. app_coordinates_window_to_bitmap performs this translation, and is called with the following parameters:

  • width, height - dimensions of the bitmap being presented, the same as the ones passed to app_present.
  • x, y - pointers to integer values containing the coordinate, in the coordinate system of the window, to be translated. When the function returns, their values will have been updated with the corresponding position in the coordinate system of the bitmap.

app_coordinates_bitmap_to_window

void app_coordinates_bitmap_to_window( app_t* app, int width, int height, int* x, int* y )

This performs the opposite translation to app_coordinates_window_to_bitmap - it converts a position given in the coordinate system of the bitmap into the coordinate system of the window. See app_coordinates_window_to_bitmap for details.