3 /// \ingroup Properties
14 #include "m_property.h"
18 #define ROUND(x) ((int)((x)<0 ? (x)-0.5 : (x)+0.5))
20 static int do_action(m_option_t
* prop_list
, const char* name
,
21 int action
, void* arg
, void *ctx
) {
24 m_property_action_t ka
;
26 if((sep
= strchr(name
,'/')) && sep
[1]) {
29 memcpy(base
,name
,len
);
31 prop
= m_option_list_find(prop_list
, base
);
35 action
= M_PROPERTY_KEY_ACTION
;
38 prop
= m_option_list_find(prop_list
, name
);
39 if(!prop
) return M_PROPERTY_UNKNOWN
;
40 r
= ((m_property_ctrl_f
)prop
->p
)(prop
,action
,arg
,ctx
);
41 if(action
== M_PROPERTY_GET_TYPE
&& r
< 0) {
42 if(!arg
) return M_PROPERTY_ERROR
;
43 *(m_option_t
**)arg
= prop
;
49 int m_property_do(m_option_t
* prop_list
, const char* name
,
50 int action
, void* arg
, void *ctx
) {
57 case M_PROPERTY_PRINT
:
58 if((r
= do_action(prop_list
,name
,M_PROPERTY_PRINT
,arg
,ctx
)) >= 0)
60 // fallback on the default print for this type
61 case M_PROPERTY_TO_STRING
:
62 if((r
= do_action(prop_list
,name
,M_PROPERTY_TO_STRING
,arg
,ctx
)) !=
63 M_PROPERTY_NOT_IMPLEMENTED
)
65 // fallback on the options API. Get the type, value and print.
66 if((r
= do_action(prop_list
,name
,M_PROPERTY_GET_TYPE
,&opt
,ctx
)) <= 0)
68 val
= calloc(1,opt
->type
->size
);
69 if((r
= do_action(prop_list
,name
,M_PROPERTY_GET
,val
,ctx
)) <= 0) {
73 if(!arg
) return M_PROPERTY_ERROR
;
74 str
= m_option_print(opt
,val
);
76 *(char**)arg
= str
== (char*)-1 ? NULL
: str
;
77 return str
!= (char*)-1;
78 case M_PROPERTY_PARSE
:
79 // try the property own parsing func
80 if((r
= do_action(prop_list
,name
,M_PROPERTY_PARSE
,arg
,ctx
)) !=
81 M_PROPERTY_NOT_IMPLEMENTED
)
83 // fallback on the options API, get the type and parse.
84 if((r
= do_action(prop_list
,name
,M_PROPERTY_GET_TYPE
,&opt
,ctx
)) <= 0)
86 if(!arg
) return M_PROPERTY_ERROR
;
87 val
= calloc(1,opt
->type
->size
);
88 if((r
= m_option_parse(opt
,opt
->name
,arg
,val
,M_CONFIG_FILE
)) <= 0) {
92 r
= do_action(prop_list
,name
,M_PROPERTY_SET
,val
,ctx
);
93 m_option_free(opt
,val
);
97 return do_action(prop_list
,name
,action
,arg
,ctx
);
100 char* m_properties_expand_string(m_option_t
* prop_list
,char* str
, void *ctx
) {
101 int l
,fr
=0,pos
=0,size
=strlen(str
)+512;
102 char *p
= NULL
,*e
,*ret
= malloc(size
), num_val
;
103 int skip
= 0, lvl
= 0, skip_lvl
= 0;
110 p
= "\x1b", l
= 1; break;
112 p
= "\n", l
= 1; break;
114 p
= "\r", l
= 1; break;
116 p
= "\t", l
= 1; break;
119 char num
[3] = { str
[2], str
[3], 0 };
121 num_val
= strtol(num
,&end
,16);
132 } else if(lvl
> 0 && str
[0] == ')') {
133 if(skip
&& lvl
<= skip_lvl
) skip
= 0;
135 } else if(str
[0] == '$' && str
[1] == '{' && (e
= strchr(str
+2,'}'))) {
138 memcpy(pname
,str
+2,pl
);
140 if(m_property_do(prop_list
, pname
,
141 M_PROPERTY_PRINT
, &p
, ctx
) >= 0 && p
)
142 l
= strlen(p
), fr
= 1;
146 } else if(str
[0] == '?' && str
[1] == '(' && (e
= strchr(str
+2,':'))) {
151 memcpy(pname
,str
+2,pl
);
153 if(m_property_do(prop_list
,pname
,M_PROPERTY_GET
,NULL
,ctx
) < 0)
154 skip
= 1, skip_lvl
= lvl
;
158 p
= str
, l
= 1, str
++;
160 if(skip
|| l
<= 0) continue;
164 ret
= realloc(ret
,size
);
168 if(fr
) free(p
), fr
= 0;
175 void m_properties_print_help_list(m_option_t
* list
) {
176 char min
[50],max
[50];
179 mp_msg(MSGT_CFGPARSER
, MSGL_INFO
, MSGTR_PropertyListHeader
);
180 for(i
= 0 ; list
[i
].name
; i
++) {
181 m_option_t
* opt
= &list
[i
];
182 if(opt
->flags
& M_OPT_MIN
)
183 sprintf(min
,"%-8.0f",opt
->min
);
186 if(opt
->flags
& M_OPT_MAX
)
187 sprintf(max
,"%-8.0f",opt
->max
);
190 mp_msg(MSGT_CFGPARSER
, MSGL_INFO
, " %-20.20s %-15.15s %-10.10s %-10.10s\n",
197 mp_msg(MSGT_CFGPARSER
, MSGL_INFO
, MSGTR_TotalProperties
, count
);
200 // Some generic property implementations
202 int m_property_int_ro(m_option_t
* prop
,int action
,
210 return M_PROPERTY_NOT_IMPLEMENTED
;
213 int m_property_int_range(m_option_t
* prop
,int action
,
214 void* arg
,int* var
) {
218 M_PROPERTY_CLAMP(prop
,*(int*)arg
);
221 case M_PROPERTY_STEP_UP
:
222 case M_PROPERTY_STEP_DOWN
:
223 *var
+= (arg
? *(int*)arg
: 1) *
224 (action
== M_PROPERTY_STEP_DOWN
? -1 : 1);
225 M_PROPERTY_CLAMP(prop
,*var
);
228 return m_property_int_ro(prop
,action
,arg
,*var
);
231 int m_property_choice(m_option_t
* prop
,int action
,
232 void* arg
,int* var
) {
234 case M_PROPERTY_STEP_UP
:
235 case M_PROPERTY_STEP_DOWN
:
236 *var
+= action
== M_PROPERTY_STEP_UP
? 1 : prop
->max
;
237 *var
%= (int)prop
->max
+1;
240 return m_property_int_range(prop
,action
,arg
,var
);
243 int m_property_flag(m_option_t
* prop
,int action
,
244 void* arg
,int* var
) {
246 case M_PROPERTY_STEP_UP
:
247 case M_PROPERTY_STEP_DOWN
:
248 *var
= *var
== prop
->min
? prop
->max
: prop
->min
;
250 case M_PROPERTY_PRINT
:
252 *(char**)arg
= strdup((*var
> prop
->min
) ? MSGTR_Enabled
: MSGTR_Disabled
);
255 return m_property_int_range(prop
,action
,arg
,var
);
258 int m_property_float_ro(m_option_t
* prop
,int action
,
259 void* arg
,float var
) {
265 case M_PROPERTY_PRINT
:
267 *(char**)arg
= malloc(20);
268 sprintf(*(char**)arg
,"%.2f",var
);
271 return M_PROPERTY_NOT_IMPLEMENTED
;
274 int m_property_float_range(m_option_t
* prop
,int action
,
275 void* arg
,float* var
) {
279 M_PROPERTY_CLAMP(prop
,*(float*)arg
);
282 case M_PROPERTY_STEP_UP
:
283 case M_PROPERTY_STEP_DOWN
:
284 *var
+= (arg
? *(float*)arg
: 0.1) *
285 (action
== M_PROPERTY_STEP_DOWN
? -1 : 1);
286 M_PROPERTY_CLAMP(prop
,*var
);
289 return m_property_float_ro(prop
,action
,arg
,*var
);
292 int m_property_delay(m_option_t
* prop
,int action
,
293 void* arg
,float* var
) {
295 case M_PROPERTY_PRINT
:
297 *(char**)arg
= malloc(20);
298 sprintf(*(char**)arg
,"%d ms",ROUND((*var
)*1000));
301 return m_property_float_range(prop
,action
,arg
,var
);
305 int m_property_double_ro(m_option_t
* prop
,int action
,
306 void* arg
,double var
) {
312 case M_PROPERTY_PRINT
:
314 *(char**)arg
= malloc(20);
315 sprintf(*(char**)arg
,"%.2f",var
);
318 return M_PROPERTY_NOT_IMPLEMENTED
;
321 int m_property_time_ro(m_option_t
* prop
,int action
,
322 void* arg
,double var
) {
324 case M_PROPERTY_PRINT
:
326 return M_PROPERTY_ERROR
;
333 *(char **) arg
= malloc(20);
335 sprintf(*(char **) arg
, "%d:%02d:%02d", h
, m
, s
);
337 sprintf(*(char **) arg
, "%d:%02d", m
, s
);
339 sprintf(*(char **) arg
, "%d", s
);
340 return M_PROPERTY_OK
;
343 return m_property_double_ro(prop
,action
,arg
,var
);
346 int m_property_string_ro(m_option_t
* prop
,int action
,void* arg
,char* str
) {
352 case M_PROPERTY_PRINT
:
354 *(char**)arg
= str
? strdup(str
) : NULL
;
357 return M_PROPERTY_NOT_IMPLEMENTED
;
360 int m_property_bitrate(m_option_t
* prop
,int action
,void* arg
,int rate
) {
362 case M_PROPERTY_PRINT
:
364 return M_PROPERTY_ERROR
;
365 *(char**)arg
= malloc (16);
366 sprintf(*(char**)arg
, "%d kbps", rate
*8/1000);
367 return M_PROPERTY_OK
;
369 return m_property_int_ro(prop
, action
, arg
, rate
);