obs-properties.c 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512
  1. /******************************************************************************
  2. Copyright (C) 2014 by Hugh Bailey <[email protected]>
  3. This program is free software: you can redistribute it and/or modify
  4. it under the terms of the GNU General Public License as published by
  5. the Free Software Foundation, either version 2 of the License, or
  6. (at your option) any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with this program. If not, see <http://www.gnu.org/licenses/>.
  13. ******************************************************************************/
  14. #include "util/bmem.h"
  15. #include "util/darray.h"
  16. #include "obs-internal.h"
  17. #include "obs-properties.h"
  18. static inline void *get_property_data(struct obs_property *prop);
  19. /* ------------------------------------------------------------------------- */
  20. struct float_data {
  21. double min, max, step;
  22. enum obs_number_type type;
  23. char *suffix;
  24. };
  25. struct int_data {
  26. int min, max, step;
  27. enum obs_number_type type;
  28. char *suffix;
  29. };
  30. struct list_item {
  31. char *name;
  32. bool disabled;
  33. union {
  34. char *str;
  35. long long ll;
  36. double d;
  37. };
  38. };
  39. struct path_data {
  40. char *filter;
  41. char *default_path;
  42. enum obs_path_type type;
  43. };
  44. struct text_data {
  45. enum obs_text_type type;
  46. bool monospace;
  47. enum obs_text_info_type info_type;
  48. bool info_word_wrap;
  49. };
  50. struct list_data {
  51. DARRAY(struct list_item) items;
  52. enum obs_combo_type type;
  53. enum obs_combo_format format;
  54. };
  55. struct editable_list_data {
  56. enum obs_editable_list_type type;
  57. char *filter;
  58. char *default_path;
  59. };
  60. struct button_data {
  61. obs_property_clicked_t callback;
  62. enum obs_button_type type;
  63. char *url;
  64. };
  65. struct frame_rate_option {
  66. char *name;
  67. char *description;
  68. };
  69. struct frame_rate_range {
  70. struct media_frames_per_second min_time;
  71. struct media_frames_per_second max_time;
  72. };
  73. struct frame_rate_data {
  74. DARRAY(struct frame_rate_option) extra_options;
  75. DARRAY(struct frame_rate_range) ranges;
  76. };
  77. struct group_data {
  78. enum obs_group_type type;
  79. obs_properties_t *content;
  80. };
  81. static inline void path_data_free(struct path_data *data)
  82. {
  83. bfree(data->default_path);
  84. if (data->type == OBS_PATH_FILE)
  85. bfree(data->filter);
  86. }
  87. static inline void editable_list_data_free(struct editable_list_data *data)
  88. {
  89. bfree(data->default_path);
  90. bfree(data->filter);
  91. }
  92. static inline void list_item_free(struct list_data *data,
  93. struct list_item *item)
  94. {
  95. bfree(item->name);
  96. if (data->format == OBS_COMBO_FORMAT_STRING)
  97. bfree(item->str);
  98. }
  99. static inline void list_data_free(struct list_data *data)
  100. {
  101. for (size_t i = 0; i < data->items.num; i++)
  102. list_item_free(data, data->items.array + i);
  103. da_free(data->items);
  104. }
  105. static inline void frame_rate_data_options_free(struct frame_rate_data *data)
  106. {
  107. for (size_t i = 0; i < data->extra_options.num; i++) {
  108. struct frame_rate_option *opt = &data->extra_options.array[i];
  109. bfree(opt->name);
  110. bfree(opt->description);
  111. }
  112. da_resize(data->extra_options, 0);
  113. }
  114. static inline void frame_rate_data_ranges_free(struct frame_rate_data *data)
  115. {
  116. da_resize(data->ranges, 0);
  117. }
  118. static inline void frame_rate_data_free(struct frame_rate_data *data)
  119. {
  120. frame_rate_data_options_free(data);
  121. frame_rate_data_ranges_free(data);
  122. da_free(data->extra_options);
  123. da_free(data->ranges);
  124. }
  125. static inline void group_data_free(struct group_data *data)
  126. {
  127. obs_properties_destroy(data->content);
  128. }
  129. static inline void int_data_free(struct int_data *data)
  130. {
  131. if (data->suffix)
  132. bfree(data->suffix);
  133. }
  134. static inline void float_data_free(struct float_data *data)
  135. {
  136. if (data->suffix)
  137. bfree(data->suffix);
  138. }
  139. static inline void button_data_free(struct button_data *data)
  140. {
  141. if (data->url)
  142. bfree(data->url);
  143. }
  144. struct obs_properties;
  145. struct obs_property {
  146. char *name;
  147. char *desc;
  148. char *long_desc;
  149. void *priv;
  150. enum obs_property_type type;
  151. bool visible;
  152. bool enabled;
  153. struct obs_properties *parent;
  154. obs_property_modified_t modified;
  155. obs_property_modified2_t modified2;
  156. struct obs_property *next;
  157. };
  158. struct obs_properties {
  159. void *param;
  160. void (*destroy)(void *param);
  161. uint32_t flags;
  162. struct obs_property *first_property;
  163. struct obs_property **last;
  164. struct obs_property *parent;
  165. };
  166. obs_properties_t *obs_properties_create(void)
  167. {
  168. struct obs_properties *props;
  169. props = bzalloc(sizeof(struct obs_properties));
  170. props->last = &props->first_property;
  171. return props;
  172. }
  173. void obs_properties_set_param(obs_properties_t *props, void *param,
  174. void (*destroy)(void *param))
  175. {
  176. if (!props)
  177. return;
  178. if (props->param && props->destroy)
  179. props->destroy(props->param);
  180. props->param = param;
  181. props->destroy = destroy;
  182. }
  183. void obs_properties_set_flags(obs_properties_t *props, uint32_t flags)
  184. {
  185. if (props)
  186. props->flags = flags;
  187. }
  188. uint32_t obs_properties_get_flags(obs_properties_t *props)
  189. {
  190. return props ? props->flags : 0;
  191. }
  192. void *obs_properties_get_param(obs_properties_t *props)
  193. {
  194. return props ? props->param : NULL;
  195. }
  196. obs_properties_t *obs_properties_create_param(void *param,
  197. void (*destroy)(void *param))
  198. {
  199. struct obs_properties *props = obs_properties_create();
  200. obs_properties_set_param(props, param, destroy);
  201. return props;
  202. }
  203. static void obs_property_destroy(struct obs_property *property)
  204. {
  205. if (property->type == OBS_PROPERTY_LIST)
  206. list_data_free(get_property_data(property));
  207. else if (property->type == OBS_PROPERTY_PATH)
  208. path_data_free(get_property_data(property));
  209. else if (property->type == OBS_PROPERTY_EDITABLE_LIST)
  210. editable_list_data_free(get_property_data(property));
  211. else if (property->type == OBS_PROPERTY_FRAME_RATE)
  212. frame_rate_data_free(get_property_data(property));
  213. else if (property->type == OBS_PROPERTY_GROUP)
  214. group_data_free(get_property_data(property));
  215. else if (property->type == OBS_PROPERTY_INT)
  216. int_data_free(get_property_data(property));
  217. else if (property->type == OBS_PROPERTY_FLOAT)
  218. float_data_free(get_property_data(property));
  219. else if (property->type == OBS_PROPERTY_BUTTON)
  220. button_data_free(get_property_data(property));
  221. bfree(property->name);
  222. bfree(property->desc);
  223. bfree(property->long_desc);
  224. bfree(property);
  225. }
  226. void obs_properties_destroy(obs_properties_t *props)
  227. {
  228. if (props) {
  229. struct obs_property *p = props->first_property;
  230. if (props->destroy && props->param)
  231. props->destroy(props->param);
  232. while (p) {
  233. struct obs_property *next = p->next;
  234. obs_property_destroy(p);
  235. p = next;
  236. }
  237. bfree(props);
  238. }
  239. }
  240. obs_property_t *obs_properties_first(obs_properties_t *props)
  241. {
  242. return (props != NULL) ? props->first_property : NULL;
  243. }
  244. obs_property_t *obs_properties_get(obs_properties_t *props, const char *name)
  245. {
  246. struct obs_property *property;
  247. if (!props)
  248. return NULL;
  249. property = props->first_property;
  250. while (property) {
  251. if (strcmp(property->name, name) == 0)
  252. return property;
  253. if (property->type == OBS_PROPERTY_GROUP) {
  254. obs_properties_t *group =
  255. obs_property_group_content(property);
  256. obs_property_t *found = obs_properties_get(group, name);
  257. if (found != NULL) {
  258. return found;
  259. }
  260. }
  261. property = property->next;
  262. }
  263. return NULL;
  264. }
  265. obs_properties_t *obs_properties_get_parent(obs_properties_t *props)
  266. {
  267. return props->parent ? props->parent->parent : NULL;
  268. }
  269. void obs_properties_remove_by_name(obs_properties_t *props, const char *name)
  270. {
  271. if (!props)
  272. return;
  273. /* obs_properties_t is a forward-linked-list, so we need to keep both
  274. * previous and current pointers around. That way we can fix up the
  275. * pointers for the previous element if we find a match.
  276. */
  277. struct obs_property *cur = props->first_property;
  278. struct obs_property *prev = props->first_property;
  279. while (cur) {
  280. if (strcmp(cur->name, name) == 0) {
  281. // Fix props->last pointer.
  282. if (props->last == &cur->next) {
  283. if (cur == prev) {
  284. // If we are the last entry and there
  285. // is no previous entry, reset.
  286. props->last = &props->first_property;
  287. } else {
  288. // If we are the last entry and there
  289. // is a previous entry, update.
  290. props->last = &prev->next;
  291. }
  292. }
  293. // Fix props->first_property.
  294. if (props->first_property == cur)
  295. props->first_property = cur->next;
  296. // Update the previous element next pointer with our
  297. // next pointer. This is an automatic no-op if both
  298. // elements alias the same memory.
  299. prev->next = cur->next;
  300. // Finally clear our own next pointer and destroy.
  301. cur->next = NULL;
  302. obs_property_destroy(cur);
  303. break;
  304. }
  305. if (cur->type == OBS_PROPERTY_GROUP) {
  306. obs_properties_remove_by_name(
  307. obs_property_group_content(cur), name);
  308. }
  309. prev = cur;
  310. cur = cur->next;
  311. }
  312. }
  313. void obs_properties_apply_settings_internal(obs_properties_t *props,
  314. obs_data_t *settings,
  315. obs_properties_t *realprops)
  316. {
  317. struct obs_property *p;
  318. p = props->first_property;
  319. while (p) {
  320. if (p->type == OBS_PROPERTY_GROUP) {
  321. obs_properties_apply_settings_internal(
  322. obs_property_group_content(p), settings,
  323. realprops);
  324. }
  325. if (p->modified)
  326. p->modified(realprops, p, settings);
  327. else if (p->modified2)
  328. p->modified2(p->priv, realprops, p, settings);
  329. p = p->next;
  330. }
  331. }
  332. void obs_properties_apply_settings(obs_properties_t *props,
  333. obs_data_t *settings)
  334. {
  335. if (!props)
  336. return;
  337. obs_properties_apply_settings_internal(props, settings, props);
  338. }
  339. /* ------------------------------------------------------------------------- */
  340. static inline void propertes_add(struct obs_properties *props,
  341. struct obs_property *p)
  342. {
  343. *props->last = p;
  344. props->last = &p->next;
  345. }
  346. static inline size_t get_property_size(enum obs_property_type type)
  347. {
  348. switch (type) {
  349. case OBS_PROPERTY_INVALID:
  350. return 0;
  351. case OBS_PROPERTY_BOOL:
  352. return 0;
  353. case OBS_PROPERTY_INT:
  354. return sizeof(struct int_data);
  355. case OBS_PROPERTY_FLOAT:
  356. return sizeof(struct float_data);
  357. case OBS_PROPERTY_TEXT:
  358. return sizeof(struct text_data);
  359. case OBS_PROPERTY_PATH:
  360. return sizeof(struct path_data);
  361. case OBS_PROPERTY_LIST:
  362. return sizeof(struct list_data);
  363. case OBS_PROPERTY_COLOR:
  364. return 0;
  365. case OBS_PROPERTY_BUTTON:
  366. return sizeof(struct button_data);
  367. case OBS_PROPERTY_FONT:
  368. return 0;
  369. case OBS_PROPERTY_EDITABLE_LIST:
  370. return sizeof(struct editable_list_data);
  371. case OBS_PROPERTY_FRAME_RATE:
  372. return sizeof(struct frame_rate_data);
  373. case OBS_PROPERTY_GROUP:
  374. return sizeof(struct group_data);
  375. case OBS_PROPERTY_COLOR_ALPHA:
  376. return 0;
  377. }
  378. return 0;
  379. }
  380. static inline struct obs_property *new_prop(struct obs_properties *props,
  381. const char *name, const char *desc,
  382. enum obs_property_type type)
  383. {
  384. size_t data_size = get_property_size(type);
  385. struct obs_property *p;
  386. p = bzalloc(sizeof(struct obs_property) + data_size);
  387. p->parent = props;
  388. p->enabled = true;
  389. p->visible = true;
  390. p->type = type;
  391. p->name = bstrdup(name);
  392. p->desc = bstrdup(desc);
  393. propertes_add(props, p);
  394. return p;
  395. }
  396. static inline obs_properties_t *get_topmost_parent(obs_properties_t *props)
  397. {
  398. obs_properties_t *parent = props;
  399. obs_properties_t *last_parent = parent;
  400. while (parent) {
  401. last_parent = parent;
  402. parent = obs_properties_get_parent(parent);
  403. }
  404. return last_parent;
  405. }
  406. static inline bool contains_prop(struct obs_properties *props, const char *name)
  407. {
  408. struct obs_property *p = props->first_property;
  409. while (p) {
  410. if (strcmp(p->name, name) == 0) {
  411. blog(LOG_WARNING, "Property '%s' exists", name);
  412. return true;
  413. }
  414. if (p->type == OBS_PROPERTY_GROUP) {
  415. if (contains_prop(obs_property_group_content(p),
  416. name)) {
  417. return true;
  418. }
  419. }
  420. p = p->next;
  421. }
  422. return false;
  423. }
  424. static inline bool has_prop(struct obs_properties *props, const char *name)
  425. {
  426. return contains_prop(get_topmost_parent(props), name);
  427. }
  428. static inline void *get_property_data(struct obs_property *prop)
  429. {
  430. return (uint8_t *)prop + sizeof(struct obs_property);
  431. }
  432. static inline void *get_type_data(struct obs_property *prop,
  433. enum obs_property_type type)
  434. {
  435. if (!prop || prop->type != type)
  436. return NULL;
  437. return get_property_data(prop);
  438. }
  439. obs_property_t *obs_properties_add_bool(obs_properties_t *props,
  440. const char *name, const char *desc)
  441. {
  442. if (!props || has_prop(props, name))
  443. return NULL;
  444. return new_prop(props, name, desc, OBS_PROPERTY_BOOL);
  445. }
  446. static obs_property_t *add_int(obs_properties_t *props, const char *name,
  447. const char *desc, int min, int max, int step,
  448. enum obs_number_type type)
  449. {
  450. if (!props || has_prop(props, name))
  451. return NULL;
  452. struct obs_property *p = new_prop(props, name, desc, OBS_PROPERTY_INT);
  453. struct int_data *data = get_property_data(p);
  454. data->min = min;
  455. data->max = max;
  456. data->step = step;
  457. data->type = type;
  458. return p;
  459. }
  460. static obs_property_t *add_flt(obs_properties_t *props, const char *name,
  461. const char *desc, double min, double max,
  462. double step, enum obs_number_type type)
  463. {
  464. if (!props || has_prop(props, name))
  465. return NULL;
  466. struct obs_property *p =
  467. new_prop(props, name, desc, OBS_PROPERTY_FLOAT);
  468. struct float_data *data = get_property_data(p);
  469. data->min = min;
  470. data->max = max;
  471. data->step = step;
  472. data->type = type;
  473. return p;
  474. }
  475. obs_property_t *obs_properties_add_int(obs_properties_t *props,
  476. const char *name, const char *desc,
  477. int min, int max, int step)
  478. {
  479. return add_int(props, name, desc, min, max, step, OBS_NUMBER_SCROLLER);
  480. }
  481. obs_property_t *obs_properties_add_float(obs_properties_t *props,
  482. const char *name, const char *desc,
  483. double min, double max, double step)
  484. {
  485. return add_flt(props, name, desc, min, max, step, OBS_NUMBER_SCROLLER);
  486. }
  487. obs_property_t *obs_properties_add_int_slider(obs_properties_t *props,
  488. const char *name,
  489. const char *desc, int min,
  490. int max, int step)
  491. {
  492. return add_int(props, name, desc, min, max, step, OBS_NUMBER_SLIDER);
  493. }
  494. obs_property_t *obs_properties_add_float_slider(obs_properties_t *props,
  495. const char *name,
  496. const char *desc, double min,
  497. double max, double step)
  498. {
  499. return add_flt(props, name, desc, min, max, step, OBS_NUMBER_SLIDER);
  500. }
  501. obs_property_t *obs_properties_add_text(obs_properties_t *props,
  502. const char *name, const char *desc,
  503. enum obs_text_type type)
  504. {
  505. if (!props || has_prop(props, name))
  506. return NULL;
  507. struct obs_property *p = new_prop(props, name, desc, OBS_PROPERTY_TEXT);
  508. struct text_data *data = get_property_data(p);
  509. data->type = type;
  510. data->info_type = OBS_TEXT_INFO_NORMAL;
  511. data->info_word_wrap = true;
  512. return p;
  513. }
  514. obs_property_t *obs_properties_add_path(obs_properties_t *props,
  515. const char *name, const char *desc,
  516. enum obs_path_type type,
  517. const char *filter,
  518. const char *default_path)
  519. {
  520. if (!props || has_prop(props, name))
  521. return NULL;
  522. struct obs_property *p = new_prop(props, name, desc, OBS_PROPERTY_PATH);
  523. struct path_data *data = get_property_data(p);
  524. data->type = type;
  525. data->default_path = bstrdup(default_path);
  526. if (data->type == OBS_PATH_FILE)
  527. data->filter = bstrdup(filter);
  528. return p;
  529. }
  530. obs_property_t *obs_properties_add_list(obs_properties_t *props,
  531. const char *name, const char *desc,
  532. enum obs_combo_type type,
  533. enum obs_combo_format format)
  534. {
  535. if (!props || has_prop(props, name))
  536. return NULL;
  537. if (type == OBS_COMBO_TYPE_EDITABLE &&
  538. format != OBS_COMBO_FORMAT_STRING) {
  539. blog(LOG_WARNING,
  540. "List '%s', error: Editable combo boxes "
  541. "must be of the 'string' type",
  542. name);
  543. return NULL;
  544. }
  545. struct obs_property *p = new_prop(props, name, desc, OBS_PROPERTY_LIST);
  546. struct list_data *data = get_property_data(p);
  547. data->format = format;
  548. data->type = type;
  549. return p;
  550. }
  551. obs_property_t *obs_properties_add_color(obs_properties_t *props,
  552. const char *name, const char *desc)
  553. {
  554. if (!props || has_prop(props, name))
  555. return NULL;
  556. return new_prop(props, name, desc, OBS_PROPERTY_COLOR);
  557. }
  558. obs_property_t *obs_properties_add_color_alpha(obs_properties_t *props,
  559. const char *name,
  560. const char *desc)
  561. {
  562. if (!props || has_prop(props, name))
  563. return NULL;
  564. return new_prop(props, name, desc, OBS_PROPERTY_COLOR_ALPHA);
  565. }
  566. obs_property_t *obs_properties_add_button(obs_properties_t *props,
  567. const char *name, const char *text,
  568. obs_property_clicked_t callback)
  569. {
  570. if (!props || has_prop(props, name))
  571. return NULL;
  572. struct obs_property *p =
  573. new_prop(props, name, text, OBS_PROPERTY_BUTTON);
  574. struct button_data *data = get_property_data(p);
  575. data->callback = callback;
  576. return p;
  577. }
  578. obs_property_t *obs_properties_add_button2(obs_properties_t *props,
  579. const char *name, const char *text,
  580. obs_property_clicked_t callback,
  581. void *priv)
  582. {
  583. if (!props || has_prop(props, name))
  584. return NULL;
  585. struct obs_property *p =
  586. new_prop(props, name, text, OBS_PROPERTY_BUTTON);
  587. struct button_data *data = get_property_data(p);
  588. data->callback = callback;
  589. p->priv = priv;
  590. return p;
  591. }
  592. obs_property_t *obs_properties_add_font(obs_properties_t *props,
  593. const char *name, const char *desc)
  594. {
  595. if (!props || has_prop(props, name))
  596. return NULL;
  597. return new_prop(props, name, desc, OBS_PROPERTY_FONT);
  598. }
  599. obs_property_t *
  600. obs_properties_add_editable_list(obs_properties_t *props, const char *name,
  601. const char *desc,
  602. enum obs_editable_list_type type,
  603. const char *filter, const char *default_path)
  604. {
  605. if (!props || has_prop(props, name))
  606. return NULL;
  607. struct obs_property *p =
  608. new_prop(props, name, desc, OBS_PROPERTY_EDITABLE_LIST);
  609. struct editable_list_data *data = get_property_data(p);
  610. data->type = type;
  611. data->filter = bstrdup(filter);
  612. data->default_path = bstrdup(default_path);
  613. return p;
  614. }
  615. obs_property_t *obs_properties_add_frame_rate(obs_properties_t *props,
  616. const char *name,
  617. const char *desc)
  618. {
  619. if (!props || has_prop(props, name))
  620. return NULL;
  621. struct obs_property *p =
  622. new_prop(props, name, desc, OBS_PROPERTY_FRAME_RATE);
  623. struct frame_rate_data *data = get_property_data(p);
  624. da_init(data->extra_options);
  625. da_init(data->ranges);
  626. return p;
  627. }
  628. static bool check_property_group_recursion(obs_properties_t *parent,
  629. obs_properties_t *group)
  630. {
  631. /* Scan the group for the parent. */
  632. obs_property_t *current_property = group->first_property;
  633. while (current_property) {
  634. if (current_property->type == OBS_PROPERTY_GROUP) {
  635. obs_properties_t *cprops =
  636. obs_property_group_content(current_property);
  637. if (cprops == parent) {
  638. /* Contains find_props */
  639. return true;
  640. } else if (cprops == group) {
  641. /* Contains self, shouldn't be possible but
  642. * lets verify anyway. */
  643. return true;
  644. }
  645. if (check_property_group_recursion(parent, cprops))
  646. return true;
  647. }
  648. current_property = current_property->next;
  649. }
  650. return false;
  651. }
  652. static bool check_property_group_duplicates(obs_properties_t *parent,
  653. obs_properties_t *group)
  654. {
  655. obs_property_t *current_property = group->first_property;
  656. while (current_property) {
  657. if (has_prop(parent, current_property->name)) {
  658. return true;
  659. }
  660. current_property = current_property->next;
  661. }
  662. return false;
  663. }
  664. obs_property_t *obs_properties_add_group(obs_properties_t *props,
  665. const char *name, const char *desc,
  666. enum obs_group_type type,
  667. obs_properties_t *group)
  668. {
  669. if (!props || has_prop(props, name))
  670. return NULL;
  671. if (!group)
  672. return NULL;
  673. /* Prevent recursion. */
  674. if (props == group)
  675. return NULL;
  676. if (check_property_group_recursion(props, group))
  677. return NULL;
  678. /* Prevent duplicate properties */
  679. if (check_property_group_duplicates(props, group))
  680. return NULL;
  681. obs_property_t *p = new_prop(props, name, desc, OBS_PROPERTY_GROUP);
  682. group->parent = p;
  683. struct group_data *data = get_property_data(p);
  684. data->type = type;
  685. data->content = group;
  686. return p;
  687. }
  688. /* ------------------------------------------------------------------------- */
  689. static inline bool is_combo(struct obs_property *p)
  690. {
  691. return p->type == OBS_PROPERTY_LIST;
  692. }
  693. static inline struct list_data *get_list_data(struct obs_property *p)
  694. {
  695. if (!p || !is_combo(p))
  696. return NULL;
  697. return get_property_data(p);
  698. }
  699. static inline struct list_data *get_list_fmt_data(struct obs_property *p,
  700. enum obs_combo_format format)
  701. {
  702. struct list_data *data = get_list_data(p);
  703. return (data && data->format == format) ? data : NULL;
  704. }
  705. /* ------------------------------------------------------------------------- */
  706. bool obs_property_next(obs_property_t **p)
  707. {
  708. if (!p || !*p)
  709. return false;
  710. *p = (*p)->next;
  711. return *p != NULL;
  712. }
  713. void obs_property_set_modified_callback(obs_property_t *p,
  714. obs_property_modified_t modified)
  715. {
  716. if (p)
  717. p->modified = modified;
  718. }
  719. void obs_property_set_modified_callback2(obs_property_t *p,
  720. obs_property_modified2_t modified2,
  721. void *priv)
  722. {
  723. if (p) {
  724. p->modified2 = modified2;
  725. p->priv = priv;
  726. }
  727. }
  728. bool obs_property_modified(obs_property_t *p, obs_data_t *settings)
  729. {
  730. if (p) {
  731. if (p->modified) {
  732. obs_properties_t *top = get_topmost_parent(p->parent);
  733. return p->modified(top, p, settings);
  734. } else if (p->modified2) {
  735. obs_properties_t *top = get_topmost_parent(p->parent);
  736. return p->modified2(p->priv, top, p, settings);
  737. }
  738. }
  739. return false;
  740. }
  741. bool obs_property_button_clicked(obs_property_t *p, void *obj)
  742. {
  743. struct obs_context_data *context = obj;
  744. if (p) {
  745. struct button_data *data =
  746. get_type_data(p, OBS_PROPERTY_BUTTON);
  747. if (data && data->callback) {
  748. obs_properties_t *top = get_topmost_parent(p->parent);
  749. if (p->priv)
  750. return data->callback(top, p, p->priv);
  751. return data->callback(top, p,
  752. (context ? context->data : NULL));
  753. }
  754. }
  755. return false;
  756. }
  757. void obs_property_set_visible(obs_property_t *p, bool visible)
  758. {
  759. if (p)
  760. p->visible = visible;
  761. }
  762. void obs_property_set_enabled(obs_property_t *p, bool enabled)
  763. {
  764. if (p)
  765. p->enabled = enabled;
  766. }
  767. void obs_property_set_description(obs_property_t *p, const char *description)
  768. {
  769. if (p) {
  770. bfree(p->desc);
  771. p->desc = description && *description ? bstrdup(description)
  772. : NULL;
  773. }
  774. }
  775. void obs_property_set_long_description(obs_property_t *p, const char *long_desc)
  776. {
  777. if (p) {
  778. bfree(p->long_desc);
  779. p->long_desc = long_desc && *long_desc ? bstrdup(long_desc)
  780. : NULL;
  781. }
  782. }
  783. const char *obs_property_name(obs_property_t *p)
  784. {
  785. return p ? p->name : NULL;
  786. }
  787. const char *obs_property_description(obs_property_t *p)
  788. {
  789. return p ? p->desc : NULL;
  790. }
  791. const char *obs_property_long_description(obs_property_t *p)
  792. {
  793. return p ? p->long_desc : NULL;
  794. }
  795. enum obs_property_type obs_property_get_type(obs_property_t *p)
  796. {
  797. return p ? p->type : OBS_PROPERTY_INVALID;
  798. }
  799. bool obs_property_enabled(obs_property_t *p)
  800. {
  801. return p ? p->enabled : false;
  802. }
  803. bool obs_property_visible(obs_property_t *p)
  804. {
  805. return p ? p->visible : false;
  806. }
  807. int obs_property_int_min(obs_property_t *p)
  808. {
  809. struct int_data *data = get_type_data(p, OBS_PROPERTY_INT);
  810. return data ? data->min : 0;
  811. }
  812. int obs_property_int_max(obs_property_t *p)
  813. {
  814. struct int_data *data = get_type_data(p, OBS_PROPERTY_INT);
  815. return data ? data->max : 0;
  816. }
  817. int obs_property_int_step(obs_property_t *p)
  818. {
  819. struct int_data *data = get_type_data(p, OBS_PROPERTY_INT);
  820. return data ? data->step : 0;
  821. }
  822. enum obs_number_type obs_property_int_type(obs_property_t *p)
  823. {
  824. struct int_data *data = get_type_data(p, OBS_PROPERTY_INT);
  825. return data ? data->type : OBS_NUMBER_SCROLLER;
  826. }
  827. const char *obs_property_int_suffix(obs_property_t *p)
  828. {
  829. struct int_data *data = get_type_data(p, OBS_PROPERTY_INT);
  830. return data ? data->suffix : NULL;
  831. }
  832. double obs_property_float_min(obs_property_t *p)
  833. {
  834. struct float_data *data = get_type_data(p, OBS_PROPERTY_FLOAT);
  835. return data ? data->min : 0;
  836. }
  837. double obs_property_float_max(obs_property_t *p)
  838. {
  839. struct float_data *data = get_type_data(p, OBS_PROPERTY_FLOAT);
  840. return data ? data->max : 0;
  841. }
  842. double obs_property_float_step(obs_property_t *p)
  843. {
  844. struct float_data *data = get_type_data(p, OBS_PROPERTY_FLOAT);
  845. return data ? data->step : 0;
  846. }
  847. const char *obs_property_float_suffix(obs_property_t *p)
  848. {
  849. struct float_data *data = get_type_data(p, OBS_PROPERTY_FLOAT);
  850. return data ? data->suffix : NULL;
  851. }
  852. enum obs_number_type obs_property_float_type(obs_property_t *p)
  853. {
  854. struct float_data *data = get_type_data(p, OBS_PROPERTY_FLOAT);
  855. return data ? data->type : OBS_NUMBER_SCROLLER;
  856. }
  857. enum obs_text_type obs_property_text_type(obs_property_t *p)
  858. {
  859. struct text_data *data = get_type_data(p, OBS_PROPERTY_TEXT);
  860. return data ? data->type : OBS_TEXT_DEFAULT;
  861. }
  862. bool obs_property_text_monospace(obs_property_t *p)
  863. {
  864. struct text_data *data = get_type_data(p, OBS_PROPERTY_TEXT);
  865. return data ? data->monospace : false;
  866. }
  867. enum obs_text_info_type obs_property_text_info_type(obs_property_t *p)
  868. {
  869. struct text_data *data = get_type_data(p, OBS_PROPERTY_TEXT);
  870. return data ? data->info_type : OBS_TEXT_INFO_NORMAL;
  871. }
  872. bool obs_property_text_info_word_wrap(obs_property_t *p)
  873. {
  874. struct text_data *data = get_type_data(p, OBS_PROPERTY_TEXT);
  875. return data ? data->info_word_wrap : true;
  876. }
  877. enum obs_path_type obs_property_path_type(obs_property_t *p)
  878. {
  879. struct path_data *data = get_type_data(p, OBS_PROPERTY_PATH);
  880. return data ? data->type : OBS_PATH_DIRECTORY;
  881. }
  882. const char *obs_property_path_filter(obs_property_t *p)
  883. {
  884. struct path_data *data = get_type_data(p, OBS_PROPERTY_PATH);
  885. return data ? data->filter : NULL;
  886. }
  887. const char *obs_property_path_default_path(obs_property_t *p)
  888. {
  889. struct path_data *data = get_type_data(p, OBS_PROPERTY_PATH);
  890. return data ? data->default_path : NULL;
  891. }
  892. enum obs_combo_type obs_property_list_type(obs_property_t *p)
  893. {
  894. struct list_data *data = get_list_data(p);
  895. return data ? data->type : OBS_COMBO_TYPE_INVALID;
  896. }
  897. enum obs_combo_format obs_property_list_format(obs_property_t *p)
  898. {
  899. struct list_data *data = get_list_data(p);
  900. return data ? data->format : OBS_COMBO_FORMAT_INVALID;
  901. }
  902. void obs_property_int_set_limits(obs_property_t *p, int min, int max, int step)
  903. {
  904. struct int_data *data = get_type_data(p, OBS_PROPERTY_INT);
  905. if (!data)
  906. return;
  907. data->min = min;
  908. data->max = max;
  909. data->step = step;
  910. }
  911. void obs_property_float_set_limits(obs_property_t *p, double min, double max,
  912. double step)
  913. {
  914. struct float_data *data = get_type_data(p, OBS_PROPERTY_FLOAT);
  915. if (!data)
  916. return;
  917. data->min = min;
  918. data->max = max;
  919. data->step = step;
  920. }
  921. void obs_property_int_set_suffix(obs_property_t *p, const char *suffix)
  922. {
  923. struct int_data *data = get_type_data(p, OBS_PROPERTY_INT);
  924. if (!data)
  925. return;
  926. bfree(data->suffix);
  927. data->suffix = bstrdup(suffix);
  928. }
  929. void obs_property_float_set_suffix(obs_property_t *p, const char *suffix)
  930. {
  931. struct float_data *data = get_type_data(p, OBS_PROPERTY_FLOAT);
  932. if (!data)
  933. return;
  934. bfree(data->suffix);
  935. data->suffix = bstrdup(suffix);
  936. }
  937. void obs_property_text_set_monospace(obs_property_t *p, bool monospace)
  938. {
  939. struct text_data *data = get_type_data(p, OBS_PROPERTY_TEXT);
  940. if (!data)
  941. return;
  942. data->monospace = monospace;
  943. }
  944. void obs_property_text_set_info_type(obs_property_t *p,
  945. enum obs_text_info_type type)
  946. {
  947. struct text_data *data = get_type_data(p, OBS_PROPERTY_TEXT);
  948. if (!data)
  949. return;
  950. data->info_type = type;
  951. }
  952. void obs_property_text_set_info_word_wrap(obs_property_t *p, bool word_wrap)
  953. {
  954. struct text_data *data = get_type_data(p, OBS_PROPERTY_TEXT);
  955. if (!data)
  956. return;
  957. data->info_word_wrap = word_wrap;
  958. }
  959. void obs_property_button_set_type(obs_property_t *p, enum obs_button_type type)
  960. {
  961. struct button_data *data = get_type_data(p, OBS_PROPERTY_BUTTON);
  962. if (!data)
  963. return;
  964. data->type = type;
  965. }
  966. void obs_property_button_set_url(obs_property_t *p, char *url)
  967. {
  968. struct button_data *data = get_type_data(p, OBS_PROPERTY_BUTTON);
  969. if (!data)
  970. return;
  971. data->url = bstrdup(url);
  972. }
  973. void obs_property_list_clear(obs_property_t *p)
  974. {
  975. struct list_data *data = get_list_data(p);
  976. if (data)
  977. list_data_free(data);
  978. }
  979. static size_t add_item(struct list_data *data, const char *name,
  980. const void *val)
  981. {
  982. struct list_item item = {NULL};
  983. item.name = bstrdup(name);
  984. if (data->format == OBS_COMBO_FORMAT_INT)
  985. item.ll = *(const long long *)val;
  986. else if (data->format == OBS_COMBO_FORMAT_FLOAT)
  987. item.d = *(const double *)val;
  988. else
  989. item.str = bstrdup(val);
  990. return da_push_back(data->items, &item);
  991. }
  992. static void insert_item(struct list_data *data, size_t idx, const char *name,
  993. const void *val)
  994. {
  995. struct list_item item = {NULL};
  996. item.name = bstrdup(name);
  997. if (data->format == OBS_COMBO_FORMAT_INT)
  998. item.ll = *(const long long *)val;
  999. else if (data->format == OBS_COMBO_FORMAT_FLOAT)
  1000. item.d = *(const double *)val;
  1001. else
  1002. item.str = bstrdup(val);
  1003. da_insert(data->items, idx, &item);
  1004. }
  1005. size_t obs_property_list_add_string(obs_property_t *p, const char *name,
  1006. const char *val)
  1007. {
  1008. struct list_data *data = get_list_data(p);
  1009. if (data && data->format == OBS_COMBO_FORMAT_STRING)
  1010. return add_item(data, name, val);
  1011. return 0;
  1012. }
  1013. size_t obs_property_list_add_int(obs_property_t *p, const char *name,
  1014. long long val)
  1015. {
  1016. struct list_data *data = get_list_data(p);
  1017. if (data && data->format == OBS_COMBO_FORMAT_INT)
  1018. return add_item(data, name, &val);
  1019. return 0;
  1020. }
  1021. size_t obs_property_list_add_float(obs_property_t *p, const char *name,
  1022. double val)
  1023. {
  1024. struct list_data *data = get_list_data(p);
  1025. if (data && data->format == OBS_COMBO_FORMAT_FLOAT)
  1026. return add_item(data, name, &val);
  1027. return 0;
  1028. }
  1029. void obs_property_list_insert_string(obs_property_t *p, size_t idx,
  1030. const char *name, const char *val)
  1031. {
  1032. struct list_data *data = get_list_data(p);
  1033. if (data && data->format == OBS_COMBO_FORMAT_STRING)
  1034. insert_item(data, idx, name, val);
  1035. }
  1036. void obs_property_list_insert_int(obs_property_t *p, size_t idx,
  1037. const char *name, long long val)
  1038. {
  1039. struct list_data *data = get_list_data(p);
  1040. if (data && data->format == OBS_COMBO_FORMAT_INT)
  1041. insert_item(data, idx, name, &val);
  1042. }
  1043. void obs_property_list_insert_float(obs_property_t *p, size_t idx,
  1044. const char *name, double val)
  1045. {
  1046. struct list_data *data = get_list_data(p);
  1047. if (data && data->format == OBS_COMBO_FORMAT_FLOAT)
  1048. insert_item(data, idx, name, &val);
  1049. }
  1050. void obs_property_list_item_remove(obs_property_t *p, size_t idx)
  1051. {
  1052. struct list_data *data = get_list_data(p);
  1053. if (data && idx < data->items.num) {
  1054. list_item_free(data, data->items.array + idx);
  1055. da_erase(data->items, idx);
  1056. }
  1057. }
  1058. size_t obs_property_list_item_count(obs_property_t *p)
  1059. {
  1060. struct list_data *data = get_list_data(p);
  1061. return data ? data->items.num : 0;
  1062. }
  1063. bool obs_property_list_item_disabled(obs_property_t *p, size_t idx)
  1064. {
  1065. struct list_data *data = get_list_data(p);
  1066. return (data && idx < data->items.num) ? data->items.array[idx].disabled
  1067. : false;
  1068. }
  1069. void obs_property_list_item_disable(obs_property_t *p, size_t idx,
  1070. bool disabled)
  1071. {
  1072. struct list_data *data = get_list_data(p);
  1073. if (!data || idx >= data->items.num)
  1074. return;
  1075. data->items.array[idx].disabled = disabled;
  1076. }
  1077. const char *obs_property_list_item_name(obs_property_t *p, size_t idx)
  1078. {
  1079. struct list_data *data = get_list_data(p);
  1080. return (data && idx < data->items.num) ? data->items.array[idx].name
  1081. : NULL;
  1082. }
  1083. const char *obs_property_list_item_string(obs_property_t *p, size_t idx)
  1084. {
  1085. struct list_data *data = get_list_fmt_data(p, OBS_COMBO_FORMAT_STRING);
  1086. return (data && idx < data->items.num) ? data->items.array[idx].str
  1087. : NULL;
  1088. }
  1089. long long obs_property_list_item_int(obs_property_t *p, size_t idx)
  1090. {
  1091. struct list_data *data = get_list_fmt_data(p, OBS_COMBO_FORMAT_INT);
  1092. return (data && idx < data->items.num) ? data->items.array[idx].ll : 0;
  1093. }
  1094. double obs_property_list_item_float(obs_property_t *p, size_t idx)
  1095. {
  1096. struct list_data *data = get_list_fmt_data(p, OBS_COMBO_FORMAT_FLOAT);
  1097. return (data && idx < data->items.num) ? data->items.array[idx].d : 0.0;
  1098. }
  1099. enum obs_editable_list_type obs_property_editable_list_type(obs_property_t *p)
  1100. {
  1101. struct editable_list_data *data =
  1102. get_type_data(p, OBS_PROPERTY_EDITABLE_LIST);
  1103. return data ? data->type : OBS_EDITABLE_LIST_TYPE_STRINGS;
  1104. }
  1105. const char *obs_property_editable_list_filter(obs_property_t *p)
  1106. {
  1107. struct editable_list_data *data =
  1108. get_type_data(p, OBS_PROPERTY_EDITABLE_LIST);
  1109. return data ? data->filter : NULL;
  1110. }
  1111. const char *obs_property_editable_list_default_path(obs_property_t *p)
  1112. {
  1113. struct editable_list_data *data =
  1114. get_type_data(p, OBS_PROPERTY_EDITABLE_LIST);
  1115. return data ? data->default_path : NULL;
  1116. }
  1117. /* ------------------------------------------------------------------------- */
  1118. /* OBS_PROPERTY_FRAME_RATE */
  1119. void obs_property_frame_rate_clear(obs_property_t *p)
  1120. {
  1121. struct frame_rate_data *data =
  1122. get_type_data(p, OBS_PROPERTY_FRAME_RATE);
  1123. if (!data)
  1124. return;
  1125. frame_rate_data_options_free(data);
  1126. frame_rate_data_ranges_free(data);
  1127. }
  1128. void obs_property_frame_rate_options_clear(obs_property_t *p)
  1129. {
  1130. struct frame_rate_data *data =
  1131. get_type_data(p, OBS_PROPERTY_FRAME_RATE);
  1132. if (!data)
  1133. return;
  1134. frame_rate_data_options_free(data);
  1135. }
  1136. void obs_property_frame_rate_fps_ranges_clear(obs_property_t *p)
  1137. {
  1138. struct frame_rate_data *data =
  1139. get_type_data(p, OBS_PROPERTY_FRAME_RATE);
  1140. if (!data)
  1141. return;
  1142. frame_rate_data_ranges_free(data);
  1143. }
  1144. size_t obs_property_frame_rate_option_add(obs_property_t *p, const char *name,
  1145. const char *description)
  1146. {
  1147. struct frame_rate_data *data =
  1148. get_type_data(p, OBS_PROPERTY_FRAME_RATE);
  1149. if (!data)
  1150. return DARRAY_INVALID;
  1151. struct frame_rate_option *opt = da_push_back_new(data->extra_options);
  1152. opt->name = bstrdup(name);
  1153. opt->description = bstrdup(description);
  1154. return data->extra_options.num - 1;
  1155. }
  1156. size_t obs_property_frame_rate_fps_range_add(obs_property_t *p,
  1157. struct media_frames_per_second min,
  1158. struct media_frames_per_second max)
  1159. {
  1160. struct frame_rate_data *data =
  1161. get_type_data(p, OBS_PROPERTY_FRAME_RATE);
  1162. if (!data)
  1163. return DARRAY_INVALID;
  1164. struct frame_rate_range *rng = da_push_back_new(data->ranges);
  1165. rng->min_time = min;
  1166. rng->max_time = max;
  1167. return data->ranges.num - 1;
  1168. }
  1169. void obs_property_frame_rate_option_insert(obs_property_t *p, size_t idx,
  1170. const char *name,
  1171. const char *description)
  1172. {
  1173. struct frame_rate_data *data =
  1174. get_type_data(p, OBS_PROPERTY_FRAME_RATE);
  1175. if (!data)
  1176. return;
  1177. struct frame_rate_option *opt = da_insert_new(data->extra_options, idx);
  1178. opt->name = bstrdup(name);
  1179. opt->description = bstrdup(description);
  1180. }
  1181. void obs_property_frame_rate_fps_range_insert(
  1182. obs_property_t *p, size_t idx, struct media_frames_per_second min,
  1183. struct media_frames_per_second max)
  1184. {
  1185. struct frame_rate_data *data =
  1186. get_type_data(p, OBS_PROPERTY_FRAME_RATE);
  1187. if (!data)
  1188. return;
  1189. struct frame_rate_range *rng = da_insert_new(data->ranges, idx);
  1190. rng->min_time = min;
  1191. rng->max_time = max;
  1192. }
  1193. size_t obs_property_frame_rate_options_count(obs_property_t *p)
  1194. {
  1195. struct frame_rate_data *data =
  1196. get_type_data(p, OBS_PROPERTY_FRAME_RATE);
  1197. return data ? data->extra_options.num : 0;
  1198. }
  1199. const char *obs_property_frame_rate_option_name(obs_property_t *p, size_t idx)
  1200. {
  1201. struct frame_rate_data *data =
  1202. get_type_data(p, OBS_PROPERTY_FRAME_RATE);
  1203. return data && data->extra_options.num > idx
  1204. ? data->extra_options.array[idx].name
  1205. : NULL;
  1206. }
  1207. const char *obs_property_frame_rate_option_description(obs_property_t *p,
  1208. size_t idx)
  1209. {
  1210. struct frame_rate_data *data =
  1211. get_type_data(p, OBS_PROPERTY_FRAME_RATE);
  1212. return data && data->extra_options.num > idx
  1213. ? data->extra_options.array[idx].description
  1214. : NULL;
  1215. }
  1216. size_t obs_property_frame_rate_fps_ranges_count(obs_property_t *p)
  1217. {
  1218. struct frame_rate_data *data =
  1219. get_type_data(p, OBS_PROPERTY_FRAME_RATE);
  1220. return data ? data->ranges.num : 0;
  1221. }
  1222. struct media_frames_per_second
  1223. obs_property_frame_rate_fps_range_min(obs_property_t *p, size_t idx)
  1224. {
  1225. struct frame_rate_data *data =
  1226. get_type_data(p, OBS_PROPERTY_FRAME_RATE);
  1227. return data && data->ranges.num > idx
  1228. ? data->ranges.array[idx].min_time
  1229. : (struct media_frames_per_second){0};
  1230. }
  1231. struct media_frames_per_second
  1232. obs_property_frame_rate_fps_range_max(obs_property_t *p, size_t idx)
  1233. {
  1234. struct frame_rate_data *data =
  1235. get_type_data(p, OBS_PROPERTY_FRAME_RATE);
  1236. return data && data->ranges.num > idx
  1237. ? data->ranges.array[idx].max_time
  1238. : (struct media_frames_per_second){0};
  1239. }
  1240. enum obs_text_type obs_proprety_text_type(obs_property_t *p)
  1241. {
  1242. return obs_property_text_type(p);
  1243. }
  1244. enum obs_group_type obs_property_group_type(obs_property_t *p)
  1245. {
  1246. struct group_data *data = get_type_data(p, OBS_PROPERTY_GROUP);
  1247. return data ? data->type : OBS_COMBO_INVALID;
  1248. }
  1249. obs_properties_t *obs_property_group_content(obs_property_t *p)
  1250. {
  1251. struct group_data *data = get_type_data(p, OBS_PROPERTY_GROUP);
  1252. return data ? data->content : NULL;
  1253. }
  1254. enum obs_button_type obs_property_button_type(obs_property_t *p)
  1255. {
  1256. struct button_data *data = get_type_data(p, OBS_PROPERTY_BUTTON);
  1257. return data ? data->type : OBS_BUTTON_DEFAULT;
  1258. }
  1259. const char *obs_property_button_url(obs_property_t *p)
  1260. {
  1261. struct button_data *data = get_type_data(p, OBS_PROPERTY_BUTTON);
  1262. return data ? data->url : "";
  1263. }