Add copyright header to those files missing one
[prosody.git] / util / dataforms.lua
1 -- Prosody IM
2 -- Copyright (C) 2008-2009 Matthew Wild
3 -- Copyright (C) 2008-2009 Waqas Hussain
4 -- 
5 -- This project is MIT/X11 licensed. Please see the
6 -- COPYING file in the source package for more information.
7 --
8
9 local setmetatable = setmetatable;
10 local pairs, ipairs = pairs, ipairs;
11 local tostring, type = tostring, type;
12 local t_concat = table.concat;
13
14 local st = require "util.stanza";
15
16 module "dataforms"
17
18 local xmlns_forms = 'jabber:x:data';
19
20 local form_t = {};
21 local form_mt = { __index = form_t };
22
23 function new(layout)
24         return setmetatable(layout, form_mt);
25 end
26
27 function form_t.form(layout, data)
28         local form = st.stanza("x", { xmlns = xmlns_forms, type = "form" });
29         if layout.title then
30                 form:tag("title"):text(layout.title):up();
31         end
32         if layout.instructions then
33                 form:tag("instructions"):text(layout.instructions):up();
34         end
35         for n, field in ipairs(layout) do
36                 local field_type = field.type or "text-single";
37                 -- Add field tag
38                 form:tag("field", { type = field_type, var = field.name, label = field.label });
39
40                 local value = data[field.name] or field.value;
41                 
42                 -- Add value, depending on type
43                 if field_type == "hidden" then
44                         if type(value) == "table" then
45                                 -- Assume an XML snippet
46                                 form:tag("value")
47                                         :add_child(value)
48                                         :up();
49                         elseif value then
50                                 form:tag("value"):text(tostring(value)):up();
51                         end
52                 elseif field_type == "boolean" then
53                         form:tag("value"):text((value and "1") or "0"):up();
54                 elseif field_type == "fixed" then
55                         
56                 elseif field_type == "jid-multi" then
57                         for _, jid in ipairs(value) do
58                                 form:tag("value"):text(jid):up();
59                         end
60                 elseif field_type == "jid-single" then
61                         form:tag("value"):text(value):up();
62                 elseif field_type == "text-single" or field_type == "text-private" then
63                         form:tag("value"):text(value):up();
64                 elseif field_type == "text-multi" then
65                         -- Split into multiple <value> tags, one for each line
66                         for line in value:gmatch("([^\r\n]+)\r?\n*") do
67                                 form:tag("value"):text(line):up();
68                         end
69                 end
70                 
71                 if field.required then
72                         form:tag("required"):up();
73                 end
74                 
75                 -- Jump back up to list of fields
76                 form:up();
77         end
78         return form;
79 end
80
81 local field_readers = {};
82
83 function form_t.data(layout, stanza)
84         local data = {};
85         
86         for field_tag in stanza:childtags() do
87                 local field_type = field_tag.attr.type;
88                 
89                 local reader = field_readers[field_type];
90                 if reader then
91                         data[field_tag.attr.var] = reader(field_tag);
92                 end
93                 
94         end
95         return data;
96 end
97
98 field_readers["text-single"] = 
99         function (field_tag)
100                 local value = field_tag:child_with_name("value");
101                 if value then
102                         return value[1];
103                 end
104         end
105
106 field_readers["text-private"] = 
107         field_readers["text-single"];
108
109 field_readers["text-multi"] = 
110         function (field_tag)
111                 local result = {};
112                 for value_tag in field_tag:childtags() do
113                         if value_tag.name == "value" then
114                                 result[#result+1] = value_tag[1];
115                         end
116                 end
117                 return t_concat(result, "\n");
118         end
119
120 field_readers["boolean"] = 
121         function (field_tag)
122                 local value = field_tag:child_with_name("value");
123                 if value then
124                         if value[1] == "1" or value[1] == "true" then
125                                 return true;
126                         else
127                                 return false;
128                         end
129                 end             
130         end
131
132 field_readers["hidden"] = 
133         function (field_tag)
134                 local value = field_tag:child_with_name("value");
135                 if value then
136                         return value[1];
137                 end
138         end
139         
140 return _M;
141
142
143 --[=[
144
145 Layout:
146 {
147
148         title = "MUC Configuration",
149         instructions = [[Use this form to configure options for this MUC room.]],
150
151         { name = "FORM_TYPE", type = "hidden", required = true };
152         { name = "field-name", type = "field-type", required = false };
153 }
154
155
156 --]=]