obs-studio/test/osx/test.mm

194 lines
4.3 KiB
Plaintext
Raw Normal View History

2013-11-14 09:18:25 -08:00
#include <stdio.h>
#include <time.h>
#include <functional>
#include <memory>
#import <Cocoa/Cocoa.h>
#import <AppKit/AppKit.h>
#import <OpenGL/OpenGL.h>
#include <util/base.h>
#include <obs.h>
2013-11-14 09:18:25 -08:00
static const int cx = 800;
static const int cy = 600;
/* --------------------------------------------------- */
2019-07-09 11:29:39 -07:00
template<typename T, typename D_T, D_T D>
struct OBSUniqueHandle : std::unique_ptr<T, std::function<D_T>> {
using base = std::unique_ptr<T, std::function<D_T>>;
2019-07-09 11:29:39 -07:00
explicit OBSUniqueHandle(T *obj = nullptr) : base(obj, D) {}
operator T *() { return base::get(); }
};
2013-11-14 09:18:25 -08:00
#define DECLARE_DELETER(x) decltype(x), x
2019-07-09 11:29:39 -07:00
using SourceContext =
OBSUniqueHandle<obs_source, DECLARE_DELETER(obs_source_release)>;
2019-07-09 11:29:39 -07:00
using SceneContext =
OBSUniqueHandle<obs_scene, DECLARE_DELETER(obs_scene_release)>;
2019-07-09 11:29:39 -07:00
using DisplayContext =
OBSUniqueHandle<obs_display, DECLARE_DELETER(obs_display_destroy)>;
#undef DECLARE_DELETER
2013-11-14 09:18:25 -08:00
/* --------------------------------------------------- */
static void CreateOBS()
2013-11-14 09:18:25 -08:00
{
if (!obs_startup("en", nullptr))
2013-11-14 09:18:25 -08:00
throw "Couldn't create OBS";
struct obs_video_info ovi;
2019-07-09 11:29:39 -07:00
ovi.adapter = 0;
ovi.fps_num = 30000;
ovi.fps_den = 1001;
ovi.graphics_module = DL_OPENGL;
2019-07-09 11:29:39 -07:00
ovi.output_format = VIDEO_FORMAT_RGBA;
ovi.base_width = cx;
ovi.base_height = cy;
ovi.output_width = cx;
ovi.output_height = cy;
if (obs_reset_video(&ovi) != 0)
throw "Couldn't initialize video";
2013-11-14 09:18:25 -08:00
}
static DisplayContext CreateDisplay(NSView *view)
{
2019-07-09 11:29:39 -07:00
gs_init_data info = {};
info.cx = cx;
info.cy = cy;
info.format = GS_RGBA;
info.zsformat = GS_ZS_NONE;
info.window.view = view;
return DisplayContext{obs_display_create(&info)};
}
static SceneContext SetupScene()
2013-11-14 09:18:25 -08:00
{
/* ------------------------------------------------------ */
(API Change) Refactor module handling Changed API: - char *obs_find_plugin_file(const char *sub_path); Changed to: char *obs_module_file(const char *file); Cahnge it so you no longer need to specify a sub-path such as: obs_find_plugin_file("module_name/file.ext") Instead, now automatically handle the module data path so all you need to do is: obs_module_file("file.ext") - int obs_load_module(const char *name); Changed to: int obs_open_module(obs_module_t *module, const char *path, const char *data_path); bool obs_init_module(obs_module_t module); Change the module loading API so that if the front-end chooses, it can load modules directly from a specified path, and associate a data directory with it on the spot. The module will not be initialized immediately; obs_init_module must be called on the module pointer in order to fully initialize the module. This is done so a module can be disabled by the front-end if the it so chooses. New API: - void obs_add_module_path(const char *bin, const char *data); These functions allow you to specify new module search paths to add, and allow you to search through them, or optionally just load all modules from them. If the string %module% is included, it will replace it with the module's name when that string is used as a lookup. Data paths are now directly added to the module's internal storage structure, and when obs_find_module_file is used, it will look up the pointer to the obs_module structure and get its data directory that way. Example: obs_add_module_path("/opt/obs/my-modules/%module%/bin", "/opt/obs/my-modules/%module%/data"); This would cause it to additionally look for the binary of a hypthetical module named "foo" at /opt/obs/my-modules/foo/bin/foo.so (or libfoo.so), and then look for the data in /opt/obs/my-modules/foo/data. This gives the front-end more flexibility for handling third-party plugin modules, or handling all plugin modules in a custom way. - void obs_find_modules(obs_find_module_callback_t callback, void *param); This searches the existing paths for modules and calls the callback function when any are found. Useful for plugin management and custom handling of the paths by the front-end if desired. - void obs_load_all_modules(void); Search through the paths and both loads and initializes all modules automatically without custom handling. - void obs_enum_modules(obs_enum_module_callback_t callback, void *param); Enumerates currently opened modules.
2014-07-27 12:00:11 -07:00
/* load modules */
obs_load_all_modules();
/* ------------------------------------------------------ */
/* create source */
2019-07-09 11:29:39 -07:00
SourceContext source{
obs_source_create("random", "a test source", nullptr, nullptr)};
if (!source)
throw "Couldn't create random test source";
/* ------------------------------------------------------ */
/* create scene and add source to scene */
SceneContext scene{obs_scene_create("test scene")};
if (!scene)
throw "Couldn't create scene";
obs_scene_add(scene, source);
/* ------------------------------------------------------ */
/* set the scene as the primary draw source and go */
obs_set_output_source(0, obs_scene_get_source(scene));
return scene;
2013-11-14 09:18:25 -08:00
}
2019-07-09 11:29:39 -07:00
@interface OBSTest : NSObject <NSApplicationDelegate, NSWindowDelegate> {
NSWindow *win;
NSView *view;
DisplayContext display;
SceneContext scene;
2013-11-14 09:18:25 -08:00
}
2019-07-09 11:29:39 -07:00
- (void)applicationDidFinishLaunching:(NSNotification *)notification;
- (BOOL)applicationShouldTerminateAfterLastWindowClosed:(NSApplication *)app;
- (void)windowWillClose:(NSNotification *)notification;
2013-11-14 09:18:25 -08:00
@end
@implementation OBSTest
2019-07-09 11:29:39 -07:00
- (void)applicationDidFinishLaunching:(NSNotification *)notification
2013-11-14 09:18:25 -08:00
{
UNUSED_PARAMETER(notification);
2013-11-14 09:18:25 -08:00
try {
NSRect content_rect = NSMakeRect(0, 0, cx, cy);
win = [[NSWindow alloc]
initWithContentRect:content_rect
2019-07-09 11:29:39 -07:00
styleMask:NSTitledWindowMask |
NSClosableWindowMask
backing:NSBackingStoreBuffered
defer:NO];
2013-11-14 09:18:25 -08:00
if (!win)
throw "Could not create window";
2013-11-14 09:18:25 -08:00
view = [[NSView alloc] initWithFrame:content_rect];
if (!view)
throw "Could not create view";
CreateOBS();
win.title = @"foo";
win.delegate = self;
win.contentView = view;
[win orderFrontRegardless];
[win center];
[win makeMainWindow];
display = CreateDisplay(view);
2013-11-14 09:18:25 -08:00
scene = SetupScene();
2019-07-09 11:29:39 -07:00
obs_display_add_draw_callback(
display.get(),
[](void *, uint32_t, uint32_t) {
obs_render_main_texture();
},
nullptr);
2013-11-14 09:18:25 -08:00
} catch (char const *error) {
NSLog(@"%s\n", error);
[NSApp terminate:nil];
2013-11-14 09:18:25 -08:00
}
}
2013-11-14 09:18:25 -08:00
2019-07-09 11:29:39 -07:00
- (BOOL)applicationShouldTerminateAfterLastWindowClosed:(NSApplication *)app
{
UNUSED_PARAMETER(app);
return YES;
}
2013-11-14 09:18:25 -08:00
2019-07-09 11:29:39 -07:00
- (void)windowWillClose:(NSNotification *)notification
{
UNUSED_PARAMETER(notification);
obs_set_output_source(0, nullptr);
scene.reset();
display.reset();
obs_shutdown();
NSLog(@"Number of memory leaks: %lu", bnum_allocs());
2013-11-14 09:18:25 -08:00
}
@end
2013-11-14 09:18:25 -08:00
/* --------------------------------------------------- */
int main()
{
@autoreleasepool {
NSApplication *app = [NSApplication sharedApplication];
OBSTest *test = [[OBSTest alloc] init];
app.delegate = test;
[app run];
2013-11-14 09:18:25 -08:00
}
return 0;
}