2013-11-14 18:18:25 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
#include <functional>
|
|
|
|
#include <memory>
|
|
|
|
|
|
|
|
#import <Cocoa/Cocoa.h>
|
|
|
|
#import <AppKit/AppKit.h>
|
|
|
|
#import <OpenGL/OpenGL.h>
|
|
|
|
|
2013-11-29 20:51:39 +01:00
|
|
|
#include <util/base.h>
|
|
|
|
#include <obs.h>
|
2013-11-14 18:18:25 +01:00
|
|
|
|
|
|
|
static const int cx = 800;
|
|
|
|
static const int cy = 600;
|
|
|
|
|
|
|
|
/* --------------------------------------------------- */
|
|
|
|
|
2013-12-16 01:16:55 +01:00
|
|
|
template <typename T, typename D_T, D_T D>
|
|
|
|
struct OBSUniqueHandle : std::unique_ptr<T, std::function<D_T>>
|
2013-11-14 18:18:25 +01:00
|
|
|
{
|
2013-12-16 01:16:55 +01:00
|
|
|
using base = std::unique_ptr<T, std::function<D_T>>;
|
2014-04-14 21:24:57 +02:00
|
|
|
explicit OBSUniqueHandle(T *obj=nullptr) : base(obj, D) {}
|
2013-12-16 01:16:55 +01:00
|
|
|
operator T*() { return base::get(); }
|
|
|
|
};
|
2013-11-14 18:18:25 +01:00
|
|
|
|
2013-12-16 01:16:55 +01:00
|
|
|
#define DECLARE_DELETER(x) decltype(x), x
|
|
|
|
|
|
|
|
using SourceContext = OBSUniqueHandle<obs_source,
|
|
|
|
DECLARE_DELETER(obs_source_release)>;
|
|
|
|
|
|
|
|
using SceneContext = OBSUniqueHandle<obs_scene,
|
2013-12-29 08:54:06 -07:00
|
|
|
DECLARE_DELETER(obs_scene_release)>;
|
2013-12-16 01:16:55 +01:00
|
|
|
|
|
|
|
#undef DECLARE_DELETER
|
2013-11-14 18:18:25 +01:00
|
|
|
|
|
|
|
/* --------------------------------------------------- */
|
|
|
|
|
2014-04-14 21:24:57 +02:00
|
|
|
static void CreateOBS(NSView *view)
|
2013-11-14 18:18:25 +01:00
|
|
|
{
|
2014-06-25 00:21:16 -07:00
|
|
|
if (!obs_startup("en"))
|
2013-11-14 18:18:25 +01:00
|
|
|
throw "Couldn't create OBS";
|
2013-11-20 15:00:16 -07:00
|
|
|
|
|
|
|
struct obs_video_info ovi;
|
|
|
|
ovi.adapter = 0;
|
|
|
|
ovi.fps_num = 30000;
|
|
|
|
ovi.fps_den = 1001;
|
2015-01-09 20:19:22 +01:00
|
|
|
ovi.graphics_module = DL_OPENGL;
|
2013-11-20 15:00:16 -07:00
|
|
|
ovi.output_format = VIDEO_FORMAT_RGBA;
|
2013-11-26 22:26:14 -07:00
|
|
|
ovi.base_width = cx;
|
|
|
|
ovi.base_height = cy;
|
2013-11-20 15:00:16 -07:00
|
|
|
ovi.output_width = cx;
|
|
|
|
ovi.output_height = cy;
|
2013-11-26 22:26:14 -07:00
|
|
|
ovi.window_width = cx;
|
|
|
|
ovi.window_height = cy;
|
2014-04-14 21:24:57 +02:00
|
|
|
ovi.window.view = view;
|
2013-11-20 15:00:16 -07:00
|
|
|
|
2014-07-20 17:40:57 -07:00
|
|
|
if (obs_reset_video(&ovi) != 0)
|
2013-11-20 15:00:16 -07:00
|
|
|
throw "Couldn't initialize video";
|
2013-11-14 18:18:25 +01:00
|
|
|
}
|
|
|
|
|
2014-04-14 21:24:57 +02:00
|
|
|
static SceneContext SetupScene()
|
2013-11-14 18:18:25 +01:00
|
|
|
{
|
2014-04-14 21:24:57 +02: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();
|
2014-04-14 21:24:57 +02:00
|
|
|
|
|
|
|
/* ------------------------------------------------------ */
|
|
|
|
/* create source */
|
|
|
|
SourceContext source{obs_source_create(OBS_SOURCE_TYPE_INPUT,
|
2014-11-01 21:41:17 +01:00
|
|
|
"random", "a test source", nullptr, nullptr)};
|
2014-04-14 21:24:57 +02:00
|
|
|
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 */
|
2014-08-04 08:41:15 -07:00
|
|
|
obs_set_output_source(0, obs_scene_get_source(scene));
|
2014-04-14 21:24:57 +02:00
|
|
|
|
|
|
|
return scene;
|
2013-11-14 18:18:25 +01:00
|
|
|
}
|
|
|
|
|
2014-04-14 21:24:57 +02:00
|
|
|
@interface OBSTest : NSObject<NSApplicationDelegate, NSWindowDelegate>
|
2013-11-14 18:18:25 +01:00
|
|
|
{
|
2014-04-14 21:24:57 +02:00
|
|
|
NSWindow *win;
|
|
|
|
NSView *view;
|
|
|
|
SceneContext scene;
|
2013-11-14 18:18:25 +01:00
|
|
|
}
|
2014-04-14 21:24:57 +02:00
|
|
|
- (void)applicationDidFinishLaunching:(NSNotification*)notification;
|
|
|
|
- (BOOL)applicationShouldTerminateAfterLastWindowClosed:(NSApplication*)app;
|
|
|
|
- (void)windowWillClose:(NSNotification*)notification;
|
2013-11-14 18:18:25 +01:00
|
|
|
@end
|
|
|
|
|
2014-04-14 21:24:57 +02:00
|
|
|
@implementation OBSTest
|
|
|
|
- (void)applicationDidFinishLaunching:(NSNotification*)notification
|
2013-11-14 18:18:25 +01:00
|
|
|
{
|
2014-04-14 21:24:57 +02:00
|
|
|
UNUSED_PARAMETER(notification);
|
2013-11-14 18:18:25 +01:00
|
|
|
|
|
|
|
try {
|
2014-04-14 21:24:57 +02:00
|
|
|
NSRect content_rect = NSMakeRect(0, 0, cx, cy);
|
|
|
|
win = [[NSWindow alloc]
|
|
|
|
initWithContentRect:content_rect
|
|
|
|
styleMask:NSTitledWindowMask | NSClosableWindowMask
|
|
|
|
backing:NSBackingStoreBuffered
|
|
|
|
defer:NO];
|
2013-11-14 18:18:25 +01:00
|
|
|
if (!win)
|
2014-04-14 21:24:57 +02:00
|
|
|
throw "Could not create window";
|
2013-11-14 18:18:25 +01:00
|
|
|
|
2014-04-14 21:24:57 +02:00
|
|
|
view = [[NSView alloc] initWithFrame:content_rect];
|
|
|
|
if (!view)
|
|
|
|
throw "Could not create view";
|
|
|
|
|
|
|
|
win.title = @"foo";
|
|
|
|
win.delegate = self;
|
|
|
|
win.contentView = view;
|
|
|
|
|
|
|
|
[win orderFrontRegardless];
|
|
|
|
[win center];
|
|
|
|
[win makeMainWindow];
|
|
|
|
|
|
|
|
CreateOBS(view);
|
2013-11-14 18:18:25 +01:00
|
|
|
|
2014-04-14 21:24:57 +02:00
|
|
|
scene = SetupScene();
|
|
|
|
|
|
|
|
obs_add_draw_callback(
|
|
|
|
[](void *, uint32_t, uint32_t) {
|
|
|
|
obs_render_main_view();
|
|
|
|
}, nullptr);
|
2013-11-14 18:18:25 +01:00
|
|
|
|
|
|
|
} catch (char const *error) {
|
2014-04-14 21:24:57 +02:00
|
|
|
NSLog(@"%s\n", error);
|
|
|
|
|
|
|
|
[NSApp terminate:nil];
|
2013-11-14 18:18:25 +01:00
|
|
|
}
|
2014-04-14 21:24:57 +02:00
|
|
|
}
|
2013-11-14 18:18:25 +01:00
|
|
|
|
2014-04-14 21:24:57 +02:00
|
|
|
- (BOOL)applicationShouldTerminateAfterLastWindowClosed:(NSApplication*)app
|
|
|
|
{
|
|
|
|
UNUSED_PARAMETER(app);
|
|
|
|
|
|
|
|
return YES;
|
|
|
|
}
|
2013-11-14 18:18:25 +01:00
|
|
|
|
2014-04-14 21:24:57 +02:00
|
|
|
- (void)windowWillClose:(NSNotification*)notification
|
|
|
|
{
|
|
|
|
UNUSED_PARAMETER(notification);
|
|
|
|
|
|
|
|
obs_set_output_source(0, nullptr);
|
|
|
|
scene.reset();
|
|
|
|
|
|
|
|
obs_shutdown();
|
|
|
|
NSLog(@"Number of memory leaks: %lu", bnum_allocs());
|
2013-11-14 18:18:25 +01:00
|
|
|
}
|
2014-04-14 21:24:57 +02:00
|
|
|
@end
|
2013-11-14 18:18:25 +01:00
|
|
|
|
|
|
|
/* --------------------------------------------------- */
|
|
|
|
|
|
|
|
int main()
|
|
|
|
{
|
|
|
|
@autoreleasepool {
|
2014-08-03 02:20:55 +02:00
|
|
|
NSApplication *app = [NSApplication sharedApplication];
|
2014-04-14 21:24:57 +02:00
|
|
|
OBSTest *test = [[OBSTest alloc] init];
|
2014-08-03 02:20:55 +02:00
|
|
|
app.delegate = test;
|
2014-04-14 21:24:57 +02:00
|
|
|
|
2014-08-03 02:20:55 +02:00
|
|
|
[app run];
|
2013-11-14 18:18:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|