summaryrefslogtreecommitdiffabout
path: root/content
authorMichael Krelin <hacker@klever.net>2006-10-01 23:53:31 (UTC)
committer Michael Krelin <hacker@klever.net>2006-10-01 23:53:31 (UTC)
commitb6a122d536aef024b69d40cdf2ac0332d7c14607 (patch) (unidiff)
tree5b6f01eb73f858e6caaf22301e23d7b6b996ffea /content
parent455df70118b045c6053b33ae2f0318a70e4f96b8 (diff)
downloadfireflix-b6a122d536aef024b69d40cdf2ac0332d7c14607.zip
fireflix-b6a122d536aef024b69d40cdf2ac0332d7c14607.tar.gz
fireflix-b6a122d536aef024b69d40cdf2ac0332d7c14607.tar.bz2
paginating through search results
git-svn-id: http://svn.klever.net/kin/fireflix/trunk@179 fe716a7a-6dde-0310-88d9-d003556173a8
Diffstat (limited to 'content') (more/less context) (ignore whitespace changes)
-rw-r--r--content/fireflix-panel.xul13
-rw-r--r--content/fireflix.css10
-rw-r--r--content/fireflix.js50
-rw-r--r--content/util.js7
4 files changed, 75 insertions, 5 deletions
diff --git a/content/fireflix-panel.xul b/content/fireflix-panel.xul
index 398dd78..bf6df73 100644
--- a/content/fireflix-panel.xul
+++ b/content/fireflix-panel.xul
@@ -1,274 +1,287 @@
1<?xml version="1.0"?> 1<?xml version="1.0"?>
2<?xml-stylesheet href="chrome://global/skin/" type="text/css"?> 2<?xml-stylesheet href="chrome://global/skin/" type="text/css"?>
3<?xml-stylesheet href="fireflix.css" type="text/css"?> 3<?xml-stylesheet href="fireflix.css" type="text/css"?>
4<!DOCTYPE page SYSTEM "chrome://fireflix/locale/fireflix.dtd"> 4<!DOCTYPE page SYSTEM "chrome://fireflix/locale/fireflix.dtd">
5<page 5<page
6 xmlns="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul" 6 xmlns="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"
7 id="fireflixwindow" title="Fireflix" 7 id="fireflixwindow" title="Fireflix"
8 onload="fireflix.init()" 8 onload="fireflix.init()"
9 orient="vertical" 9 orient="vertical"
10 ondragover="nsDragAndDrop.dragOver(event,fireflix.uploadObserver)" 10 ondragover="nsDragAndDrop.dragOver(event,fireflix.uploadObserver)"
11 ondragdrop="nsDragAndDrop.drop(event,fireflix.uploadObserver)" 11 ondragdrop="nsDragAndDrop.drop(event,fireflix.uploadObserver)"
12 > 12 >
13 13
14 <script src="chrome://global/content/nsDragAndDrop.js"/> 14 <script src="chrome://global/content/nsDragAndDrop.js"/>
15 <script src="chrome://global/content/nsTransferable.js"/> 15 <script src="chrome://global/content/nsTransferable.js"/>
16 <script type="application/x-javascript" src="md5.js" /> 16 <script type="application/x-javascript" src="md5.js" />
17 <script type="application/x-javascript" src="util.js" /> 17 <script type="application/x-javascript" src="util.js" />
18 <script type="application/x-javascript" src="flickr.js" /> 18 <script type="application/x-javascript" src="flickr.js" />
19 <script type="application/x-javascript" src="fireflix.js" /> 19 <script type="application/x-javascript" src="fireflix.js" />
20 20
21 <stringbundleset> 21 <stringbundleset>
22 <stringbundle id="loc_strings" src="chrome://fireflix/locale/fireflix.properties" /> 22 <stringbundle id="loc_strings" src="chrome://fireflix/locale/fireflix.properties" />
23 </stringbundleset> 23 </stringbundleset>
24 24
25 <commandset> 25 <commandset>
26 <command id="cmd_auth_auth" label="&panel.auth.auth.label;" 26 <command id="cmd_auth_auth" label="&panel.auth.auth.label;"
27 oncommand="fireflix.on_cmd_auth()"/> 27 oncommand="fireflix.on_cmd_auth()"/>
28 <command id="cmd_auth_done" label="&panel.auth.done.label;" 28 <command id="cmd_auth_done" label="&panel.auth.done.label;"
29 oncommand="fireflix.on_cmd_auth_done()" disabled="true"/> 29 oncommand="fireflix.on_cmd_auth_done()" disabled="true"/>
30 <command id="cmd_auth_open_flickr" label="&panel.auth.flickr.label;" 30 <command id="cmd_auth_open_flickr" label="&panel.auth.flickr.label;"
31 oncommand="fireflix.openTab('htp://www.flickr.com/')" /> 31 oncommand="fireflix.openTab('htp://www.flickr.com/')" />
32 <command id="cmd_auth_unauth" label="&panel.auth.unauth.label;" 32 <command id="cmd_auth_unauth" label="&panel.auth.unauth.label;"
33 oncommand="fireflix.on_cmd_auth_unauth()" /> 33 oncommand="fireflix.on_cmd_auth_unauth()" />
34 </commandset> 34 </commandset>
35 35
36 <popupset> 36 <popupset>
37 <popup id="auth_menu"> 37 <popup id="auth_menu">
38 <menuitem command="cmd_auth_auth"/> 38 <menuitem command="cmd_auth_auth"/>
39 <menuitem command="cmd_auth_done" hidden="true" id="menu_auth_done"/> 39 <menuitem command="cmd_auth_done" hidden="true" id="menu_auth_done"/>
40 <menuitem command="cmd_auth_unauth" /> 40 <menuitem command="cmd_auth_unauth" />
41 <menuseparator/> 41 <menuseparator/>
42 <menuitem command="cmd_auth_open_flickr"/> 42 <menuitem command="cmd_auth_open_flickr"/>
43 </popup> 43 </popup>
44 </popupset> 44 </popupset>
45 45
46 <commandset id="cmdset_search"> 46 <commandset id="cmdset_search">
47 <command id="cmd_search" label="&panel.search.cmd_search.label;" 47 <command id="cmd_search" label="&panel.search.cmd_search.label;"
48 oncommand="fireflix.foundphotos.search_photos()"/> 48 oncommand="fireflix.foundphotos.search_photos()"/>
49 <command id="cmd_search_open" label="&panel.search.cmd_search_open.label;" 49 <command id="cmd_search_open" label="&panel.search.cmd_search_open.label;"
50 oncommand="fireflix.foundphotos.on_cmd_open(event)" /> 50 oncommand="fireflix.foundphotos.on_cmd_open(event)" />
51 <command id="cmd_search_prev_page"
52 label="&panel.search.cmd_search_prev_page.label;"
53 oncommand="fireflix.foundphotos.on_cmd_prev(event)" disabled="true"/>
54 <command id="cmd_search_next_page"
55 label="&panel.search.cmd_search_next_page.label;"
56 oncommand="fireflix.foundphotos.on_cmd_next(event)" disabled="true"/>
51 </commandset> 57 </commandset>
52 58
53 <commandset id="cmdset_sets"> 59 <commandset id="cmdset_sets">
54 <command id="cmd_refresh_sets" label="&panel.sets.cmd_refresh_sets;" 60 <command id="cmd_refresh_sets" label="&panel.sets.cmd_refresh_sets;"
55 oncommand="fireflix.on_refresh_sets()" /> 61 oncommand="fireflix.on_refresh_sets()" />
56 <command id="cmd_set_props" label="&panel.sets.cmd_properties;" 62 <command id="cmd_set_props" label="&panel.sets.cmd_properties;"
57 oncommand="fireflix.on_set_props()" disabled="true" /> 63 oncommand="fireflix.on_set_props()" disabled="true" />
58 </commandset> 64 </commandset>
59 65
60 <popupset> 66 <popupset>
61 <popup id="sets_menu"> 67 <popup id="sets_menu">
62 <menuitem command="cmd_set_props"/> 68 <menuitem command="cmd_set_props"/>
63 <menuitem command="cmd_refresh_sets"/> 69 <menuitem command="cmd_refresh_sets"/>
64 <menuseparator/> 70 <menuseparator/>
65 <menu label="&panel.sets.generate_html;" id="sets_html_menu"/> 71 <menu label="&panel.sets.generate_html;" id="sets_html_menu"/>
66 </popup> 72 </popup>
67 </popupset> 73 </popupset>
68 74
69 <commandset id="cmdset_uploads"> 75 <commandset id="cmdset_uploads">
70 <command id="cmd_uploads_clear" label="&panel.uploads.clear.label;" 76 <command id="cmd_uploads_clear" label="&panel.uploads.clear.label;"
71 oncommand="fireflix.uploads.on_clear()" /> 77 oncommand="fireflix.uploads.on_clear()" />
72 <command id="cmd_uploads_upload" label="&panel.uploads.upload.label;" 78 <command id="cmd_uploads_upload" label="&panel.uploads.upload.label;"
73 oncommand="fireflix.uploads.on_upload()" /> 79 oncommand="fireflix.uploads.on_upload()" />
74 <command id="cmd_uploads_remove" label="&panel.uploads.remove.label;" 80 <command id="cmd_uploads_remove" label="&panel.uploads.remove.label;"
75 oncommand="fireflix.uploads.on_remove()" /> 81 oncommand="fireflix.uploads.on_remove()" />
76 <command id="cmd_uploads_add" label="&panel.uploads.add.label;" 82 <command id="cmd_uploads_add" label="&panel.uploads.add.label;"
77 oncommand="fireflix.uploads.on_add()" /> 83 oncommand="fireflix.uploads.on_add()" />
78 </commandset> 84 </commandset>
79 85
80 <popupset> 86 <popupset>
81 <popup id="uploads_menu"> 87 <popup id="uploads_menu">
82 <menuitem command="cmd_uploads_add"/> 88 <menuitem command="cmd_uploads_add"/>
83 <menuitem command="cmd_uploads_clear"/> 89 <menuitem command="cmd_uploads_clear"/>
84 <menuitem command="cmd_uploads_remove"/> 90 <menuitem command="cmd_uploads_remove"/>
85 <menuseparator/> 91 <menuseparator/>
86 <menuitem command="cmd_uploads_upload"/> 92 <menuitem command="cmd_uploads_upload"/>
87 <menuseparator/> 93 <menuseparator/>
88 <menu label="&panel.uploads.generate_html;" id="uploads_html_menu"/> 94 <menu label="&panel.uploads.generate_html;" id="uploads_html_menu"/>
89 </popup> 95 </popup>
90 </popupset> 96 </popupset>
91 97
92 <vbox class="wholething" flex="1"> 98 <vbox class="wholething" flex="1">
93 99
94 <groupbox context="auth_menu"> 100 <groupbox context="auth_menu">
95 <caption label="&panel.auth_info;"/> 101 <caption label="&panel.auth_info;"/>
96 <hbox> 102 <hbox>
97 <label id="auth_info" value="&panel.no_auth_info;" flex="1" disabled="true" crop="end"/> 103 <label id="auth_info" value="&panel.no_auth_info;" flex="1" disabled="true" crop="end"/>
98 <button id="b_auth" command="cmd_auth_auth"/> 104 <button id="b_auth" command="cmd_auth_auth"/>
99 <button id="b_auth_done" command="cmd_auth_done" hidden="true"/> 105 <button id="b_auth_done" command="cmd_auth_done" hidden="true"/>
100 <button command="cmd_auth_open_flickr" 106 <button command="cmd_auth_open_flickr"
101 tooltiptext="&panel.auth.flickr.tip;"/> 107 tooltiptext="&panel.auth.flickr.tip;"/>
102 </hbox> 108 </hbox>
103 </groupbox> 109 </groupbox>
104 110
105 <tabbox flex="1" id="fireflix_tabs"> 111 <tabbox flex="1" id="fireflix_tabs">
106 112
107 <tabs> 113 <tabs>
108 <tab label="&panel.tabs.search;"/> 114 <tab label="&panel.tabs.search;"/>
109 <tab label="&panel.tabs.sets;"/> 115 <tab label="&panel.tabs.sets;"/>
110 <tab label="&panel.tabs.tags;" hidden="true"/> <!-- TODO: --> 116 <tab label="&panel.tabs.tags;" hidden="true"/> <!-- TODO: -->
111 <tab id="tab_upload" label="&panel.tabs.upload;"/> 117 <tab id="tab_upload" label="&panel.tabs.upload;"/>
112 </tabs> 118 </tabs>
113 119
114 <tabpanels flex="1"> 120 <tabpanels flex="1">
115 121
116 <tabpanel id="tabpanel_search" flex="1"> 122 <tabpanel id="tabpanel_search" flex="1">
117 <vbox flex="1"> 123 <vbox flex="1">
118 <groupbox class="search_params" orient="vertical" onkeypress="if(event.keyCode==event.DOM_VK_RETURN) fireflix.foundphotos.search_photos()"> 124 <groupbox class="search_params" orient="vertical" onkeypress="if(event.keyCode==event.DOM_VK_RETURN) fireflix.foundphotos.search_photos()">
119 <hbox> 125 <hbox>
120 <label control="search_for" value="&panel.search.search_for.label;" 126 <label control="search_for" value="&panel.search.search_for.label;"
121 accesskey="s"/> 127 accesskey="s"/>
122 <textbox id="search_for" flex="1"/> 128 <textbox id="search_for" flex="1"/>
123 </hbox> 129 </hbox>
124 <hbox> 130 <hbox>
125 <checkbox id="search_tags" label="&panel.search.mode.tagsonly.label;" 131 <checkbox id="search_tags" label="&panel.search.mode.tagsonly.label;"
126 tooltiptext="&panel.search.mode.tagsonly.tip;" checked="false" 132 tooltiptext="&panel.search.mode.tagsonly.tip;" checked="false"
127 accesskey="t" /> 133 accesskey="t" />
128 <checkbox id="search_mine" label="&panel.search.mode.mine.label;" checked="true" accesskey="m"/> 134 <checkbox id="search_mine" label="&panel.search.mode.mine.label;" checked="true" accesskey="m"/>
129 <spacer flex="1"/> 135 <spacer flex="1"/>
130 <button command="cmd_search"/> 136 <button command="cmd_search"/>
131 </hbox> 137 </hbox>
132 </groupbox> 138 </groupbox>
139 <hbox>
140 <button command="cmd_search_prev_page"/>
141 <spacer flex="1"/>
142 <label id="search_page" hidden="true"/>
143 <spacer flex="1"/>
144 <button command="cmd_search_next_page"/>
145 </hbox>
133 <tree id="searchresults" rows="2" flex="1" 146 <tree id="searchresults" rows="2" flex="1"
134 onselect="fireflix.foundphotos.on_select()" 147 onselect="fireflix.foundphotos.on_select()"
135 ondblclick="fireflix.foundphotos.on_cmd_open(event)" 148 ondblclick="fireflix.foundphotos.on_cmd_open(event)"
136 onkeypress="if(event.keyCode==event.DOM_VK_RETURN) 149 onkeypress="if(event.keyCode==event.DOM_VK_RETURN)
137 fireflix.foundphotos.on_cmd_open(event)"> 150 fireflix.foundphotos.on_cmd_open(event)">
138 <treecols> 151 <treecols>
139 <treecol id="sr_title" label="&panel.search.col.title.label;" flex="2" crop="end" align="start" /> 152 <treecol id="sr_title" label="&panel.search.col.title.label;" flex="2" crop="end" align="start" />
140 </treecols> 153 </treecols>
141 <treechildren/> 154 <treechildren/>
142 </tree> 155 </tree>
143 <groupbox id="searchresult_props" orient="horizontal" hidden="true"> 156 <groupbox id="searchresult_props" orient="horizontal" hidden="true">
144 <vbox width="100" pack="center"> 157 <vbox width="100" pack="center">
145 <hbox pack="center"> 158 <hbox pack="center">
146 <image id="search_photo"/> 159 <image id="search_photo"/>
147 </hbox> 160 </hbox>
148 </vbox> 161 </vbox>
149 <vbox flex="1"> 162 <vbox flex="1">
150 <label id="searchresult_title" crop="end"/> 163 <label id="searchresult_title" crop="end"/>
151 <hbox flex="1" pack="center"> 164 <hbox flex="1" pack="center">
152 <div flex="1" id="searchresult_description" xmlns="http://www.w3.org/1999/xhtml"/> 165 <div flex="1" id="searchresult_description" xmlns="http://www.w3.org/1999/xhtml"/>
153 </hbox> 166 </hbox>
154 <hbox pack="end"> 167 <hbox pack="end">
155 <button command="cmd_search_open"/> 168 <button command="cmd_search_open"/>
156 </hbox> 169 </hbox>
157 </vbox> 170 </vbox>
158 </groupbox> 171 </groupbox>
159 </vbox> 172 </vbox>
160 </tabpanel> 173 </tabpanel>
161 174
162 <tabpanel id="tabpanel_sets" flex="1" 175 <tabpanel id="tabpanel_sets" flex="1"
163 onkeypress="if(event.keyCode==event.DOM_VK_RETURN) 176 onkeypress="if(event.keyCode==event.DOM_VK_RETURN)
164 document.getElementById('setphotos').focus()"> 177 document.getElementById('setphotos').focus()">
165 <vbox flex="1"> 178 <vbox flex="1">
166 <tree id="setslist" rows="2" onselect="fireflix.photosets.on_select()" 179 <tree id="setslist" rows="2" onselect="fireflix.photosets.on_select()"
167 flex="1" context="sets_menu" 180 flex="1" context="sets_menu"
168 > 181 >
169 <treecols> 182 <treecols>
170 <treecol id="sl_name" label="&panel.sets.name.label;" flex="4" crop="end" align="start" tooltiptext="&panel.sets.name.tip;"/> 183 <treecol id="sl_name" label="&panel.sets.name.label;" flex="4" crop="end" align="start" tooltiptext="&panel.sets.name.tip;"/>
171 <splitter class="tree-splitter" /> 184 <splitter class="tree-splitter" />
172 <treecol id="sl_photos" label="&panel.sets.photos.label;" flex="1" align="end" tooltiptext="&panel.sets.photos.tip;" /> 185 <treecol id="sl_photos" label="&panel.sets.photos.label;" flex="1" align="end" tooltiptext="&panel.sets.photos.tip;" />
173 </treecols> 186 </treecols>
174 <treechildren/> 187 <treechildren/>
175 </tree> 188 </tree>
176 <hbox> 189 <hbox>
177 <button command="cmd_refresh_sets" /> 190 <button command="cmd_refresh_sets" />
178 <button command="cmd_set_props" /> 191 <button command="cmd_set_props" />
179 </hbox> 192 </hbox>
180 <tree id="setphotos" rows="2" onselect="fireflix.photoset.on_select()" 193 <tree id="setphotos" rows="2" onselect="fireflix.photoset.on_select()"
181 flex="1"> 194 flex="1">
182 <treecols> 195 <treecols>
183 <treecol id="sp_title" label="&panel.setphotos.title.label;" flex="1" crop="end" align="start" tooltiptext="&panel.setphotos.title.tip;" /> 196 <treecol id="sp_title" label="&panel.setphotos.title.label;" flex="1" crop="end" align="start" tooltiptext="&panel.setphotos.title.tip;" />
184 <splitter class="tree-splitter" /> 197 <splitter class="tree-splitter" />
185 <treecol id="sp_taken" label="&panel.setphotos.taken.label;" crop="end" align="start" tooltiptext="&panel.setphotos.taken.tip;" hidden="true" /> 198 <treecol id="sp_taken" label="&panel.setphotos.taken.label;" crop="end" align="start" tooltiptext="&panel.setphotos.taken.tip;" hidden="true" />
186 <treecol id="sp_upload" label="&panel.setphotos.upload.label;" crop="end" align="start" tooltiptext="&panel.setphotos.upload.tip;" hidden="true" /> 199 <treecol id="sp_upload" label="&panel.setphotos.upload.label;" crop="end" align="start" tooltiptext="&panel.setphotos.upload.tip;" hidden="true" />
187 </treecols> 200 </treecols>
188 <treechildren/> 201 <treechildren/>
189 </tree> 202 </tree>
190 <groupbox id="set_photo_props" orient="horizontal"> 203 <groupbox id="set_photo_props" orient="horizontal">
191 <vbox width="100" pack="center"> 204 <vbox width="100" pack="center">
192 <hbox pack="center"> 205 <hbox pack="center">
193 <image id="set_photo" hidden="true"/> 206 <image id="set_photo" hidden="true"/>
194 </hbox> 207 </hbox>
195 </vbox> 208 </vbox>
196 <spacer flex="1"/> 209 <spacer flex="1"/>
197 </groupbox> 210 </groupbox>
198 </vbox> 211 </vbox>
199 </tabpanel> 212 </tabpanel>
200 213
201 <tabpanel id="tabpanel_tags"> 214 <tabpanel id="tabpanel_tags">
202 <listbox id="tagslist" rows="8" flex="1"> 215 <listbox id="tagslist" rows="8" flex="1">
203 <listhead> 216 <listhead>
204 <listheader label="&panel.tagslist.tag.label;"/> 217 <listheader label="&panel.tagslist.tag.label;"/>
205 </listhead> 218 </listhead>
206 <listcols> 219 <listcols>
207 <listcol flex="1"/> 220 <listcol flex="1"/>
208 </listcols> 221 </listcols>
209 </listbox> 222 </listbox>
210 </tabpanel> 223 </tabpanel>
211 224
212 <tabpanel id="tabpanel_upload"> 225 <tabpanel id="tabpanel_upload">
213 <vbox flex="1"> 226 <vbox flex="1">
214 <tree id="uploadlist" rows="2" flex="1" 227 <tree id="uploadlist" rows="2" flex="1"
215 onselect="fireflix.uploads.selectionChanged()" 228 onselect="fireflix.uploads.selectionChanged()"
216 context="uploads_menu"> 229 context="uploads_menu">
217 <treecols> 230 <treecols>
218 <treecol id="up_file" label="&panel.uploadlist.file.label;" flex="4" crop="start" align="start"/> 231 <treecol id="up_file" label="&panel.uploadlist.file.label;" flex="4" crop="start" align="start"/>
219 <splitter class="tree-splitter" /> 232 <splitter class="tree-splitter" />
220 <treecol id="up_title" label="&panel.uploadlist.title.label;" flex="5" crop="end" align="start" /> 233 <treecol id="up_title" label="&panel.uploadlist.title.label;" flex="5" crop="end" align="start" />
221 <splitter class="tree-splitter" /> 234 <splitter class="tree-splitter" />
222 <treecol id="up_status" label="&panel.uploadlist.status.label;" flex="1" crop="end" align="start" /> 235 <treecol id="up_status" label="&panel.uploadlist.status.label;" flex="1" crop="end" align="start" />
223 </treecols> 236 </treecols>
224 <treechildren/> 237 <treechildren/>
225 </tree> 238 </tree>
226 <progressmeter id="upload_progress" mode="undetermined" hidden="true" /> 239 <progressmeter id="upload_progress" mode="undetermined" hidden="true" />
227 <groupbox id="upload_file_props" orient="horizontal" hidden="true"> 240 <groupbox id="upload_file_props" orient="horizontal" hidden="true">
228 <image id="upload_file_preview" width="100" height="100" /> 241 <image id="upload_file_preview" width="100" height="100" />
229 <grid flex="1"> 242 <grid flex="1">
230 <columns> 243 <columns>
231 <column/> 244 <column/>
232 <column flex="1"/> 245 <column flex="1"/>
233 </columns> 246 </columns>
234 <rows> 247 <rows>
235 <row> 248 <row>
236 <label control="upload_filename" 249 <label control="upload_filename"
237 value="&panel.upload_props.filename.label;" /> 250 value="&panel.upload_props.filename.label;" />
238 <textbox id="upload_filename" 251 <textbox id="upload_filename"
239 oninput="fireflix.uploads.propsToSel('filename')"/> 252 oninput="fireflix.uploads.propsToSel('filename')"/>
240 </row> 253 </row>
241 <row> 254 <row>
242 <label control="upload_title" value="&panel.upload_props.title.label;" /> 255 <label control="upload_title" value="&panel.upload_props.title.label;" />
243 <textbox id="upload_title" 256 <textbox id="upload_title"
244 oninput="fireflix.uploads.propsToSel('title')"/> 257 oninput="fireflix.uploads.propsToSel('title')"/>
245 </row> 258 </row>
246 <row> 259 <row>
247 <label control="uplod_tags" value="&panel.upload_props.tags.label;" /> 260 <label control="uplod_tags" value="&panel.upload_props.tags.label;" />
248 <textbox id="upload_tags" 261 <textbox id="upload_tags"
249 oninput="fireflix.uploads.propsToSel('tags')"/> 262 oninput="fireflix.uploads.propsToSel('tags')"/>
250 </row> 263 </row>
251 <!-- TODO: description, public, friend, family --> 264 <!-- TODO: description, public, friend, family -->
252 </rows> 265 </rows>
253 </grid> 266 </grid>
254 </groupbox> 267 </groupbox>
255 <hbox> 268 <hbox>
256 <button command="cmd_uploads_add" /> 269 <button command="cmd_uploads_add" />
257 <spacer flex="1"/> 270 <spacer flex="1"/>
258 <button command="cmd_uploads_remove" /> 271 <button command="cmd_uploads_remove" />
259 <spacer flex="1"/> 272 <spacer flex="1"/>
260 <button command="cmd_uploads_clear" /> 273 <button command="cmd_uploads_clear" />
261 </hbox> 274 </hbox>
262 <hbox pack="center"> 275 <hbox pack="center">
263 <button command="cmd_uploads_upload" flex="1"/> 276 <button command="cmd_uploads_upload" flex="1"/>
264 </hbox> 277 </hbox>
265 </vbox> 278 </vbox>
266 </tabpanel> 279 </tabpanel>
267 280
268 </tabpanels> 281 </tabpanels>
269 282
270 </tabbox> 283 </tabbox>
271 284
272 </vbox> 285 </vbox>
273 286
274</page> 287</page>
diff --git a/content/fireflix.css b/content/fireflix.css
index 71080d1..9e7bf6b 100644
--- a/content/fireflix.css
+++ b/content/fireflix.css
@@ -1,89 +1,99 @@
1.generated.wholething, 1.generated.wholething,
2.about.wholething, 2.about.wholething,
3tabbox, tabpanels, tabpanel { 3tabbox, tabpanels, tabpanel {
4 background: url("background.jpeg"); 4 background: url("background.jpeg");
5} 5}
6tabpanels { 6tabpanels {
7 padding: 0px; 7 padding: 0px;
8} 8}
9 9
10tree { 10tree {
11 margin-top: 2px; 11 margin-top: 2px;
12 background: rgb(12,167,0); 12 background: rgb(12,167,0);
13 color: rgb(255,255,0); 13 color: rgb(255,255,0);
14 font-size: 90%; 14 font-size: 90%;
15} 15}
16tree treechildren { /* for windows */ 16tree treechildren { /* for windows */
17 background: rgb(12,167,0); 17 background: rgb(12,167,0);
18} 18}
19 19
20tree#uploadlist treechildren::-moz-tree-cell-text(pending) { 20tree#uploadlist treechildren::-moz-tree-cell-text(pending) {
21} 21}
22tree#uploadlist treechildren::-moz-tree-cell-text(completed) { 22tree#uploadlist treechildren::-moz-tree-cell-text(completed) {
23 color: white; 23 color: white;
24} 24}
25tree#uploadlist treechildren::-moz-tree-row(failed) { 25tree#uploadlist treechildren::-moz-tree-row(failed) {
26 background: yellow; 26 background: yellow;
27} 27}
28tree#uploadlist treechildren::-moz-tree-cell-text(failed) { 28tree#uploadlist treechildren::-moz-tree-cell-text(failed) {
29 color: red; 29 color: red;
30} 30}
31tree#uploadlist treechildren::-moz-tree-cell-text(uploading) { 31tree#uploadlist treechildren::-moz-tree-cell-text(uploading) {
32 font-weight: bold; 32 font-weight: bold;
33} 33}
34 34
35groupbox#searchresult_props, 35groupbox#searchresult_props,
36groupbox.search_params, 36groupbox.search_params,
37groupbox#upload_file_props, 37groupbox#upload_file_props,
38groupbox#set_props, 38groupbox#set_props,
39groupbox#set_photo_props { 39groupbox#set_photo_props {
40 background: white; 40 background: white;
41} 41}
42 42
43groupbox#upload_file_props label { 43groupbox#upload_file_props label {
44 text-align: right; 44 text-align: right;
45} 45}
46 46
47image#search_photo, image#set_photo, image#set_primary { 47image#search_photo, image#set_photo, image#set_primary {
48 border: black 1px solid; 48 border: black 1px solid;
49} 49}
50 50
51.about .insides { 51.about .insides {
52 margin: 1ex; 52 margin: 1ex;
53} 53}
54.about .text { 54.about .text {
55 border: yellow solid 1px; 55 border: yellow solid 1px;
56 background: green; 56 background: green;
57} 57}
58.about .title { 58.about .title {
59 font-size: 300%; 59 font-size: 300%;
60 font-weight: bold; 60 font-weight: bold;
61 color: yellow; 61 color: yellow;
62} 62}
63.about .link { 63.about .link {
64 text-decoration: underline; 64 text-decoration: underline;
65 color: white; 65 color: white;
66 cursor: pointer; 66 cursor: pointer;
67} 67}
68 68
69menuitem.menuhead { 69menuitem.menuhead {
70 background: gray; 70 background: gray;
71 color: black; 71 color: black;
72 font-weight: bold; 72 font-weight: bold;
73} 73}
74 74
75div#searchresult_description { 75div#searchresult_description {
76 min-height: 5em; max-height: 8em; 76 min-height: 5em; max-height: 8em;
77 overflow: auto; 77 overflow: auto;
78 font-size: 90%; 78 font-size: 90%;
79 margin: 0.1ex 0.5ex; 79 margin: 0.1ex 0.5ex;
80} 80}
81 81
82#copying div { 82#copying div {
83 margin: 1ex 1em; 83 margin: 1ex 1em;
84 font-family: courier, monospace; 84 font-family: courier, monospace;
85 font-size: 9pt; 85 font-size: 9pt;
86 padding: 2px; 86 padding: 2px;
87 border: dotted 1px gray; 87 border: dotted 1px gray;
88 background: white; 88 background: white;
89} 89}
90
91label#search_page {
92 font-weight: bold;
93 background: white; color: #404040;
94 border-color: #c0c0c0 #404040 #404040 #c0c0c0;
95 border-style: solid;
96 border-width: 1px;
97 -moz-border-radius: 1em;
98 padding: 0.5ex 1ex;
99}
diff --git a/content/fireflix.js b/content/fireflix.js
index e144aae..92c42f4 100644
--- a/content/fireflix.js
+++ b/content/fireflix.js
@@ -1,907 +1,947 @@
1function splitascii(s) { 1function splitascii(s) {
2 var rv=''; 2 var rv='';
3 for(var i=0;i<s.length;++i) { 3 for(var i=0;i<s.length;++i) {
4 var w = s.charCodeAt(i); 4 var w = s.charCodeAt(i);
5 rv += String.fromCharCode( 5 rv += String.fromCharCode(
6 w&0xff, (w>>8)&0xff ); 6 w&0xff, (w>>8)&0xff );
7 } 7 }
8 return rv; 8 return rv;
9} 9}
10 10
11 11
12var fireflix = { 12var fireflix = {
13 flickr: new Flickr(), 13 flickr: new Flickr(),
14 init: function() { 14 init: function() {
15 pull_elements(this,document,[ 15 pull_elements(this,document,[
16 'cmd_auth_auth','cmd_auth_done','cmd_auth_unauth', 16 'cmd_auth_auth','cmd_auth_done','cmd_auth_unauth',
17 'menu_auth_done','b_auth','b_auth_done','auth_info', 17 'menu_auth_done','b_auth','b_auth_done','auth_info',
18 'loc_strings','cmd_set_props' 18 'loc_strings','cmd_set_props'
19 ]); 19 ]);
20 this.build_menus(); 20 this.build_menus();
21 this.foundphotos.init(this); 21 this.foundphotos.init(this);
22 this.photosets.init(this); 22 this.photosets.init(this);
23 this.photoset.init(this); 23 this.photoset.init(this);
24 this.uploads.init(this); 24 this.uploads.init(this);
25 this.uploadObserver.init(this); 25 this.uploadObserver.init(this);
26 this.flickr.api_key = '9c43cd66947a57e6f29db1a9da3f72e3'; 26 this.flickr.api_key = '9c43cd66947a57e6f29db1a9da3f72e3';
27 this.flickr.api_shs = '9c33c9e2f0f0cfd5'; 27 this.flickr.api_shs = '9c33c9e2f0f0cfd5';
28 this.flickr.prefs_root = 'net.klever.kin.fireflix'; 28 this.flickr.prefs_root = 'net.klever.kin.fireflix';
29 this.flickr.load_token(); 29 this.flickr.load_token();
30 document.getElementById('setslist').view = this.photosets; 30 document.getElementById('setslist').view = this.photosets;
31 document.getElementById('setphotos').view = this.photoset; 31 document.getElementById('setphotos').view = this.photoset;
32 document.getElementById('uploadlist').view = this.uploads; 32 document.getElementById('uploadlist').view = this.uploads;
33 this.no_auth_info_label = this.auth_info.value; 33 this.no_auth_info_label = this.auth_info.value;
34 this.set_auth_state(this.flickr.token,false); 34 this.set_auth_state(this.flickr.token,false);
35 if(this.flickr.token) { 35 if(this.flickr.token) {
36 this.refresh_stuff(); 36 this.refresh_stuff();
37 }else{ 37 }else{
38 this.on_cmd_auth(); 38 this.on_cmd_auth();
39 } 39 }
40 }, 40 },
41 set_auth_state: function(au,inp) { /* authorized, in progress */ 41 set_auth_state: function(au,inp) { /* authorized, in progress */
42 this.cmd_auth_unauth.disabled = !au; 42 this.cmd_auth_unauth.disabled = !au;
43 this.b_auth.hidden = au || inp; 43 this.b_auth.hidden = au || inp;
44 this.b_auth_done.hidden = !inp; 44 this.b_auth_done.hidden = !inp;
45 this.menu_auth_done.hidden = !inp; 45 this.menu_auth_done.hidden = !inp;
46 this.cmd_auth_done.setAttribute('disabled',!inp); 46 this.cmd_auth_done.setAttribute('disabled',!inp);
47 this.auth_info.disabled = !au; 47 this.auth_info.disabled = !au;
48 if(au) { 48 if(au) {
49 this.auth_info.value = this.flickr.user.fullname+' ['+this.flickr.user.username+']'; /* TODO: move to locale */ 49 this.auth_info.value = this.flickr.user.fullname+' ['+this.flickr.user.username+']'; /* TODO: move to locale */
50 }else{ 50 }else{
51 this.auth_info.value = this.no_auth_info_label; 51 this.auth_info.value = this.no_auth_info_label;
52 } 52 }
53 }, 53 },
54 on_cmd_auth: function() { 54 on_cmd_auth: function() {
55 var _this = this; 55 var _this = this;
56 this.flickr.authorize_0( 56 this.flickr.authorize_0(
57 'delete', 57 'delete',
58 function(x,f,u) { 58 function(x,f,u) {
59 _this.openTab(u); 59 _this.openTab(u);
60 _this.set_auth_state(_this.flickr.token,true); 60 _this.set_auth_state(_this.flickr.token,true);
61 }, function(x,s,c,m) { 61 }, function(x,s,c,m) {
62 _this.flickr_failure(x,s,c,m); 62 _this.flickr_failure(x,s,c,m);
63 } 63 }
64 ); 64 );
65 }, 65 },
66 on_cmd_auth_done: function() { 66 on_cmd_auth_done: function() {
67 this.set_auth_state(this.flickr.token,false); 67 this.set_auth_state(this.flickr.token,false);
68 var _this = this; 68 var _this = this;
69 this.flickr.authorize_1( 69 this.flickr.authorize_1(
70 function() { 70 function() {
71 _this.flickr.save_token(); 71 _this.flickr.save_token();
72 _this.refresh_stuff(); 72 _this.refresh_stuff();
73 _this.set_auth_state(_this.flickr.token,false); 73 _this.set_auth_state(_this.flickr.token,false);
74 _this.auth_info.value = 74 _this.auth_info.value =
75 _this.flickr.user.fullname+' ['+_this.flickr.user.username+']'; 75 _this.flickr.user.fullname+' ['+_this.flickr.user.username+']';
76 }, function(x,s,c,m) { 76 }, function(x,s,c,m) {
77 _this.set_auth_state(_this.flickr.token,false); /* XXX: no reset token? */ 77 _this.set_auth_state(_this.flickr.token,false); /* XXX: no reset token? */
78 _this.flickr_failure(x,s,c,m); 78 _this.flickr_failure(x,s,c,m);
79 } 79 }
80 ); 80 );
81 }, 81 },
82 on_cmd_auth_unauth: function() { 82 on_cmd_auth_unauth: function() {
83 this.flickr.reset_token(); 83 this.flickr.reset_token();
84 this.set_auth_state(false,false); 84 this.set_auth_state(false,false);
85 }, 85 },
86 86
87 refresh_sets: function() { this.photosets.refresh_sets(); }, 87 refresh_sets: function() { this.photosets.refresh_sets(); },
88 refresh_stuff: function() { 88 refresh_stuff: function() {
89 this.refresh_sets(); 89 this.refresh_sets();
90 this.refresh_user_tags(); 90 this.refresh_user_tags();
91 }, 91 },
92 92
93 /* photoset treeview */ 93 /* photoset treeview */
94 photoset: { 94 photoset: {
95 photos: new Array(), 95 photos: new Array(),
96 fireflix: null, 96 fireflix: null,
97 init: function(f) { 97 init: function(f) {
98 this.fireflix = f; 98 this.fireflix = f;
99 pull_elements(this,document,[ 'set_photo' ]); 99 pull_elements(this,document,[ 'set_photo' ]);
100 }, 100 },
101 rowCount: 0, 101 rowCount: 0,
102 getCellText: function(r,c) { 102 getCellText: function(r,c) {
103 var p = this.photos[r]; 103 var p = this.photos[r];
104 if(c.id=='sp_title') return p.title; 104 if(c.id=='sp_title') return p.title;
105 if(c.id=='sp_taken') return p.datetaken; 105 if(c.id=='sp_taken') return p.datetaken;
106 if(c.id=='sp_upload') return p.dateupload; /* TODO: unixtime conversion */ 106 if(c.id=='sp_upload') return p.dateupload; /* TODO: unixtime conversion */
107 return c.id; 107 return c.id;
108 }, 108 },
109 setTree: function(t) { this.tree = t }, 109 setTree: function(t) { this.tree = t },
110 isContainer: function(r) { return false; }, 110 isContainer: function(r) { return false; },
111 isSeparator: function(r) { return false; }, 111 isSeparator: function(r) { return false; },
112 isSorted: function(r) { return false; }, 112 isSorted: function(r) { return false; },
113 getLevel: function(r) { return 0; }, 113 getLevel: function(r) { return 0; },
114 getImageSrc: function(r,c) { return null }, 114 getImageSrc: function(r,c) { return null },
115 getRowProperties: function(r,p) {}, 115 getRowProperties: function(r,p) {},
116 getCellProperties: function(cid,cel,p) {}, 116 getCellProperties: function(cid,cel,p) {},
117 getColumnProperties: function(cid,cel,p) { }, 117 getColumnProperties: function(cid,cel,p) { },
118 cycleHeader: function(cid,e) { }, 118 cycleHeader: function(cid,e) { },
119 getParentIndex: function(r) { return -1; }, 119 getParentIndex: function(r) { return -1; },
120 drop: function(r,o) { }, 120 drop: function(r,o) { },
121 canDropBeforeAfter: function(r,b) { return false }, 121 canDropBeforeAfter: function(r,b) { return false },
122 122
123 importXPR: function(xp) { 123 importXPR: function(xp) {
124 this.tree.beginUpdateBatch(); 124 this.tree.beginUpdateBatch();
125 this.photos = new Array(); 125 this.photos = new Array();
126 var n; while(n=xp.iterateNext()) { 126 var n; while(n=xp.iterateNext()) {
127 this.photos.push(new Photo(n)); 127 this.photos.push(new Photo(n));
128 } 128 }
129 this.rowCount = this.photos.length; 129 this.rowCount = this.photos.length;
130 this.tree.endUpdateBatch(); 130 this.tree.endUpdateBatch();
131 }, 131 },
132 load_photos: function(psid) { 132 load_photos: function(psid) {
133 var _this = this; 133 var _this = this;
134 this.fireflix.flickr.api_call( 134 this.fireflix.flickr.api_call(
135 { 135 {
136 method: 'flickr.photosets.getPhotos', 136 method: 'flickr.photosets.getPhotos',
137 auth_token: 'default', 137 auth_token: 'default',
138 photoset_id: psid, 138 photoset_id: psid,
139 extras: 'license,date_upload,date_taken,owner_name,icon_server,original_format,last_update' 139 extras: 'license,date_upload,date_taken,owner_name,icon_server,original_format,last_update'
140 }, function(xr) { 140 }, function(xr) {
141 var x = xr.responseXML; 141 var x = xr.responseXML;
142 var xp = x.evaluate( 142 var xp = x.evaluate(
143 '/rsp/photoset/photo', x, null, 143 '/rsp/photoset/photo', x, null,
144 XPathResult.ORDERED_NODE_ITERATOR_TYPE, null ); 144 XPathResult.ORDERED_NODE_ITERATOR_TYPE, null );
145 _this.importXPR(xp); 145 _this.importXPR(xp);
146 }, function(x,s,c,m) { 146 }, function(x,s,c,m) {
147 _this.fireflix.flickr_failure(x,s,c,m); 147 _this.fireflix.flickr_failure(x,s,c,m);
148 } 148 }
149 ); 149 );
150 }, 150 },
151 on_select: function() { 151 on_select: function() {
152 if(this.selection.count==1) { 152 if(this.selection.count==1) {
153 var p = this.photos[this.selection.currentIndex]; 153 var p = this.photos[this.selection.currentIndex];
154 this.set_photo.src = 154 this.set_photo.src =
155 this.fireflix.flickr.get_photo_url(p.server,p.id,p.secret,'t'); 155 this.fireflix.flickr.get_photo_url(p.server,p.id,p.secret,'t');
156 this.set_photo.hidden = false; 156 this.set_photo.hidden = false;
157 }else{ 157 }else{
158 this.set_photo.hidden = true; 158 this.set_photo.hidden = true;
159 } 159 }
160 } 160 }
161 }, 161 },
162 162
163 /* photosets treeview */ 163 /* photosets treeview */
164 photosets: { 164 photosets: {
165 sets: new Array(), 165 sets: new Array(),
166 fireflix: null, 166 fireflix: null,
167 init: function(f) { 167 init: function(f) {
168 this.fireflix = f; 168 this.fireflix = f;
169 }, 169 },
170 rowCount: 0, 170 rowCount: 0,
171 getCellText: function(r,c) { 171 getCellText: function(r,c) {
172 var s = this.sets[r]; 172 var s = this.sets[r];
173 if(c.id=='sl_name') return s.title; 173 if(c.id=='sl_name') return s.title;
174 if(c.id=='sl_photos') return s.photos; 174 if(c.id=='sl_photos') return s.photos;
175 return c.id; 175 return c.id;
176 }, 176 },
177 setTree: function(t) { this.tree = t }, 177 setTree: function(t) { this.tree = t },
178 isContainer: function(r) { return false; }, 178 isContainer: function(r) { return false; },
179 isSeparator: function(r) { return false; }, 179 isSeparator: function(r) { return false; },
180 isSorted: function() { return false; }, 180 isSorted: function() { return false; },
181 getLevel: function(r) { return 0; }, 181 getLevel: function(r) { return 0; },
182 getImageSrc: function(r,c) { return null }, 182 getImageSrc: function(r,c) { return null },
183 getRowProperties: function(r,p) {}, 183 getRowProperties: function(r,p) {},
184 getCellProperties: function(cid,cel,p) { }, 184 getCellProperties: function(cid,cel,p) { },
185 getColumnProperties: function(cid,cel,p) { }, 185 getColumnProperties: function(cid,cel,p) { },
186 cycleHeader: function(cid,e) { }, 186 cycleHeader: function(cid,e) { },
187 getParentIndex: function(r) { return -1; }, 187 getParentIndex: function(r) { return -1; },
188 drop: function(r,o) { }, 188 drop: function(r,o) { },
189 canDropBeforeAfter: function(r,b) { return false }, 189 canDropBeforeAfter: function(r,b) { return false },
190 190
191 importXPR: function(xp) { 191 importXPR: function(xp) {
192 this.tree.beginUpdateBatch(); 192 this.tree.beginUpdateBatch();
193 this.sets = new Array(); 193 this.sets = new Array();
194 var n; while(n=xp.iterateNext()) { 194 var n; while(n=xp.iterateNext()) {
195 this.sets.push(new Photoset(n)); 195 this.sets.push(new Photoset(n));
196 } 196 }
197 this.rowCount = this.sets.length; 197 this.rowCount = this.sets.length;
198 this.tree.endUpdateBatch(); 198 this.tree.endUpdateBatch();
199 }, 199 },
200 refresh_sets: function() { 200 refresh_sets: function() {
201 var _this = this; 201 var _this = this;
202 this.fireflix.flickr.api_call( 202 this.fireflix.flickr.api_call(
203 { 203 {
204 method: 'flickr.photosets.getList', 204 method: 'flickr.photosets.getList',
205 auth_token: 'default' 205 auth_token: 'default'
206 }, function(xr) { 206 }, function(xr) {
207 var x = xr.responseXML; 207 var x = xr.responseXML;
208 var xp = x.evaluate( 208 var xp = x.evaluate(
209 '/rsp/photosets/photoset', x, null, 209 '/rsp/photosets/photoset', x, null,
210 XPathResult.ORDERED_NODE_ITERATOR_TYPE, null ); 210 XPathResult.ORDERED_NODE_ITERATOR_TYPE, null );
211 _this.importXPR(xp); 211 _this.importXPR(xp);
212 }, function(x,s,c,m) { 212 }, function(x,s,c,m) {
213 _this.fireflix.flickr_failure(x,s,c,m); 213 _this.fireflix.flickr_failure(x,s,c,m);
214 } 214 }
215 ); 215 );
216 }, 216 },
217 on_select: function() { 217 on_select: function() {
218 if(this.selection.count==1) { 218 if(this.selection.count==1) {
219 this.fireflix.cmd_set_props.setAttribute('disabled','false'); 219 this.fireflix.cmd_set_props.setAttribute('disabled','false');
220 var s = this.sets[this.selection.currentIndex]; 220 var s = this.sets[this.selection.currentIndex];
221 this.fireflix.photoset.load_photos(s.id); 221 this.fireflix.photoset.load_photos(s.id);
222 }else{ 222 }else{
223 this.fireflix.cmd_set_props.setAttribute('disabled','true'); 223 this.fireflix.cmd_set_props.setAttribute('disabled','true');
224 } 224 }
225 } 225 }
226 }, 226 },
227 227
228 refresh_user_tags: function() { 228 refresh_user_tags: function() {
229 var lb = document.getElementById('tagslist'); 229 var lb = document.getElementById('tagslist');
230 var _this = this; 230 var _this = this;
231 this.flickr.api_call( 231 this.flickr.api_call(
232 { 232 {
233 method: 'flickr.tags.getListUser', 233 method: 'flickr.tags.getListUser',
234 auth_token: 'default', 234 auth_token: 'default',
235 }, function(xr) { 235 }, function(xr) {
236 var x = xr.responseXML; 236 var x = xr.responseXML;
237 var xp = x.evaluate( 237 var xp = x.evaluate(
238 '/rsp/who/tags/tag', x, null, 238 '/rsp/who/tags/tag', x, null,
239 XPathResult.ORDERED_NODE_ITERATOR_TYPE, null ); 239 XPathResult.ORDERED_NODE_ITERATOR_TYPE, null );
240 // TODO: clear list 240 // TODO: clear list
241 var n; while(n=xp.iterateNext()) { 241 var n; while(n=xp.iterateNext()) {
242 lb.appendItem(n.firstChild.nodeValue); 242 lb.appendItem(n.firstChild.nodeValue);
243 } 243 }
244 }, function(x,s,c,m) { 244 }, function(x,s,c,m) {
245 _this.flickr_failure(x,s,c,m); 245 _this.flickr_failure(x,s,c,m);
246 } 246 }
247 ); 247 );
248 }, 248 },
249 249
250 uploadObserver: { 250 uploadObserver: {
251 fireflix: null, 251 fireflix: null,
252 init: function(f) { 252 init: function(f) {
253 this.fireflix = f; 253 this.fireflix = f;
254 }, 254 },
255 getSupportedFlavours: function() { 255 getSupportedFlavours: function() {
256 var rv = new FlavourSet(); 256 var rv = new FlavourSet();
257 rv.appendFlavour('application/x-moz-file','nsIFile'); 257 rv.appendFlavour('application/x-moz-file','nsIFile');
258 rv.appendFlavour('application/x-moz-url'); 258 rv.appendFlavour('application/x-moz-url');
259 rv.appendFlavour('text/uri-list'); 259 rv.appendFlavour('text/uri-list');
260 rv.appendFlavour('text/unicode'); 260 rv.appendFlavour('text/unicode');
261 return rv; 261 return rv;
262 }, 262 },
263 canHandleMultipleItems: true, 263 canHandleMultipleItems: true,
264 onDragOver: function(ev,fl,sess) { 264 onDragOver: function(ev,fl,sess) {
265 return true; 265 return true;
266 }, 266 },
267 onDrop: function(ev,dd,s) { 267 onDrop: function(ev,dd,s) {
268 var ldf = null; 268 var ldf = null;
269 for(var i in dd.dataList) { 269 for(var i in dd.dataList) {
270 var di = dd.dataList[i]; 270 var di = dd.dataList[i];
271 var dif = di.first; 271 var dif = di.first;
272 if( 272 if(
273 ldf==null 273 ldf==null
274 || ldf.flavour.contentType!=dif.flavour.contentType 274 || ldf.flavour.contentType!=dif.flavour.contentType
275 || ldf.contentLength!=dif.contentLength 275 || ldf.contentLength!=dif.contentLength
276 || ldf.data!=dif.data ) 276 || ldf.data!=dif.data )
277 this.drop_item(ev,di,s); 277 this.drop_item(ev,di,s);
278 ldf = dif; 278 ldf = dif;
279 } 279 }
280 }, 280 },
281 drop_item: function(ev,di,s) { 281 drop_item: function(ev,di,s) {
282 var d = di.first; 282 var d = di.first;
283 switch(d.flavour.contentType) { 283 switch(d.flavour.contentType) {
284 case 'text/unicode': 284 case 'text/unicode':
285 this.drop_urilist(ev,d.data,s); 285 this.drop_urilist(ev,d.data,s);
286 break; 286 break;
287 case 'application/x-moz-file': 287 case 'application/x-moz-file':
288 this.fireflix.uploads.add(d.data.path); 288 this.fireflix.uploads.add(d.data.path);
289 document.getElementById('fireflix_tabs').selectedTab 289 document.getElementById('fireflix_tabs').selectedTab
290 = document.getElementById('tab_upload'); 290 = document.getElementById('tab_upload');
291 break; 291 break;
292 case 'text/uri-list': 292 case 'text/uri-list':
293 // is it ascii or could it be utf8? 293 // is it ascii or could it be utf8?
294 this.drop_urilist(ev,splitascii(d.data),s); 294 this.drop_urilist(ev,splitascii(d.data),s);
295 break; 295 break;
296 default: alert(d.flavour.contentType+':'+d.data); break; 296 default: alert(d.flavour.contentType+':'+d.data); break;
297 }; 297 };
298 }, 298 },
299 drop_urilist: function(ev,ul,s) { 299 drop_urilist: function(ev,ul,s) {
300 // TODO: check for being a file? 300 // TODO: check for being a file?
301 var us = decodeURIComponent(ul).split(/[\r\n]/); 301 var us = decodeURIComponent(ul).split(/[\r\n]/);
302 for(var ui in us) 302 for(var ui in us)
303 if(/\S/.test(us[ui])) 303 if(/\S/.test(us[ui]))
304 this.fireflix.uploads.add(us[ui]); 304 this.fireflix.uploads.add(us[ui]);
305 document.getElementById('fireflix_tabs').selectedTab 305 document.getElementById('fireflix_tabs').selectedTab
306 = document.getElementById('tab_upload'); 306 = document.getElementById('tab_upload');
307 } 307 }
308 }, 308 },
309 309
310 uploads: { 310 uploads: {
311 fireflix: null, 311 fireflix: null,
312 init: function(f) { 312 init: function(f) {
313 this.fireflix=f; 313 this.fireflix=f;
314 pull_elements(this,document,[ 314 pull_elements(this,document,[
315 'upload_filename','upload_title','upload_file_preview', 315 'upload_filename','upload_title','upload_file_preview',
316 'upload_file_props','upload_progress','upload_tags', 316 'upload_file_props','upload_progress','upload_tags',
317 'cmd_uploads_upload' 317 'cmd_uploads_upload'
318 ]); 318 ]);
319 }, 319 },
320 files: new Array(), 320 files: new Array(),
321 rowCount: 0, 321 rowCount: 0,
322 getCellText: function(r,c) { 322 getCellText: function(r,c) {
323 var f = this.files[r]; 323 var f = this.files[r];
324 if(c.id=='up_file') return f.file; 324 if(c.id=='up_file') return f.file;
325 if(c.id=='up_title') return f.title; 325 if(c.id=='up_title') return f.title;
326 if(c.id=='up_status') return f.state; 326 if(c.id=='up_status') return f.state;
327 return c.id; 327 return c.id;
328 }, 328 },
329 setTree: function(t) { this.tree = t }, 329 setTree: function(t) { this.tree = t },
330 isContainer: function(r) { return false; }, 330 isContainer: function(r) { return false; },
331 isSeparator: function(r) { return false; }, 331 isSeparator: function(r) { return false; },
332 isSorted: function(r) { return false; }, 332 isSorted: function(r) { return false; },
333 getLevel: function(r) { return 0; }, 333 getLevel: function(r) { return 0; },
334 getImageSrc: function(r,c) { return null }, 334 getImageSrc: function(r,c) { return null },
335 getRowProperties: function(r,p) { 335 getRowProperties: function(r,p) {
336 try { 336 try {
337 if(!Components) return; 337 if(!Components) return;
338 }catch(e) { return } 338 }catch(e) { return }
339 var f = this.files[r]; 339 var f = this.files[r];
340 var as = Components.classes['@mozilla.org/atom-service;1']. 340 var as = Components.classes['@mozilla.org/atom-service;1'].
341 getService(Components.interfaces.nsIAtomService); 341 getService(Components.interfaces.nsIAtomService);
342 p.AppendElement(as.getAtom(f.state)); 342 p.AppendElement(as.getAtom(f.state));
343 }, 343 },
344 getCellProperties: function(r,c,p) { this.getRowProperties(r,p); }, 344 getCellProperties: function(r,c,p) { this.getRowProperties(r,p); },
345 getColumnProperties: function(c,p) { }, 345 getColumnProperties: function(c,p) { },
346 cycleHeader: function(cid,e) { }, 346 cycleHeader: function(cid,e) { },
347 getParentIndex: function(r) { return -1; }, 347 getParentIndex: function(r) { return -1; },
348 drop: function(r,o) { }, 348 drop: function(r,o) { },
349 canDropBeforeAfter: function(r,b) { return false }, 349 canDropBeforeAfter: function(r,b) { return false },
350 350
351 add: function(f) { 351 add: function(f) {
352 if(f.indexOf('file:/')==0) { 352 if(f.indexOf('file:/')==0) {
353 f = f.substr(5); 353 f = f.substr(5);
354 while(f.substr(0,2)=='//') { // XXX: not very performant, is it? ;-) 354 while(f.substr(0,2)=='//') { // XXX: not very performant, is it? ;-)
355 f = f.substr(1); 355 f = f.substr(1);
356 } 356 }
357 } 357 }
358 var t = f; 358 var t = f;
359 var ls = t.lastIndexOf('/'); 359 var ls = t.lastIndexOf('/');
360 if(ls>0) t = t.substr(ls+1); 360 if(ls>0) t = t.substr(ls+1);
361 ls = t.lastIndexOf('\\'); 361 ls = t.lastIndexOf('\\');
362 if(ls>0) t = t.substr(ls+1); 362 if(ls>0) t = t.substr(ls+1);
363 var ld = t.lastIndexOf('.'); 363 var ld = t.lastIndexOf('.');
364 if(ld>0) t = t.substr(0,ld); 364 if(ld>0) t = t.substr(0,ld);
365 this.files.push( { 365 this.files.push( {
366 file: f, 366 file: f,
367 title: t, 367 title: t,
368 tags: '', 368 tags: '',
369 state: 'pending' 369 state: 'pending'
370 } ); 370 } );
371 this.rowCount = this.files.length; 371 this.rowCount = this.files.length;
372 this.tree.rowCountChanged(this.rowCount-1,1); 372 this.tree.rowCountChanged(this.rowCount-1,1);
373 }, 373 },
374 374
375 upload_worker: function() { 375 upload_worker: function() {
376 for(var f in this.files) { 376 for(var f in this.files) {
377 if(this.files[f].state=='pending') { 377 if(this.files[f].state=='pending') {
378 var ff = this.files[f]; 378 var ff = this.files[f];
379 dump('upload '+ff.file+'\n'); 379 dump('upload '+ff.file+'\n');
380 this.on_file_upload(ff); 380 this.on_file_upload(ff);
381 ff.state='uploading'; 381 ff.state='uploading';
382 this.tree.invalidate(); 382 this.tree.invalidate();
383 var _this = this; 383 var _this = this;
384 this.fireflix.flickr.upload_file( 384 this.fireflix.flickr.upload_file(
385 ff.file, { title: ff.title, tags: ff.tags }, 385 ff.file, { title: ff.title, tags: ff.tags },
386 function(x,p) { 386 function(x,p) {
387 ff.photoid = p; 387 ff.photoid = p;
388 _this.batch_ids.push(p); 388 _this.batch_ids.push(p);
389 ff.state='completed'; 389 ff.state='completed';
390 _this.tree.invalidate(); 390 _this.tree.invalidate();
391 window.setTimeout(_this.upload_to,0,_this); 391 window.setTimeout(_this.upload_to,0,_this);
392 }, function(x,s,c,m) { 392 }, function(x,s,c,m) {
393 ff.state='failed'; 393 ff.state='failed';
394 ff.flickr_errcode = c; 394 ff.flickr_errcode = c;
395 ff.flickr_errmsg = m; 395 ff.flickr_errmsg = m;
396 _this.tree.invalidate(); 396 _this.tree.invalidate();
397 window.setTimeout(_this.upload_to,0,_this); 397 window.setTimeout(_this.upload_to,0,_this);
398 } 398 }
399 ); 399 );
400 return; 400 return;
401 } 401 }
402 } 402 }
403 dump('uploading done\n'); 403 dump('uploading done\n');
404 this.on_finish_upload(); 404 this.on_finish_upload();
405 }, 405 },
406 upload_to: function(_this) { _this.upload_worker(); }, 406 upload_to: function(_this) { _this.upload_worker(); },
407 on_file_upload: function(f) { 407 on_file_upload: function(f) {
408 this.cmd_uploads_upload.setAttribute('disabled','true'); 408 this.cmd_uploads_upload.setAttribute('disabled','true');
409 for(var fi in this.files) { 409 for(var fi in this.files) {
410 if(this.files[fi].file==f.file) { 410 if(this.files[fi].file==f.file) {
411 this.tree.ensureRowIsVisible(fi); 411 this.tree.ensureRowIsVisible(fi);
412 this.selection.rangedSelect(fi,fi,false); 412 this.selection.rangedSelect(fi,fi,false);
413 this.selection.currentIndex = fi; 413 this.selection.currentIndex = fi;
414 this.selToProps(); 414 this.selToProps();
415 break; 415 break;
416 } 416 }
417 } 417 }
418 }, 418 },
419 on_finish_upload: function() { 419 on_finish_upload: function() {
420 if(this.batch_ids.length) { 420 if(this.batch_ids.length) {
421 var psn = prompt(this.fireflix.loc_strings.getString('postUploadPhotoset')); 421 var psn = prompt(this.fireflix.loc_strings.getString('postUploadPhotoset'));
422 if(psn!=null) { 422 if(psn!=null) {
423 var pids = this.batch_ids.join(','); 423 var pids = this.batch_ids.join(',');
424 var ppid = this.batch_ids[0]; 424 var ppid = this.batch_ids[0];
425 var _this = this; 425 var _this = this;
426 this.fireflix.flickr.api_call( 426 this.fireflix.flickr.api_call(
427 { 427 {
428 method: 'flickr.photosets.create', 428 method: 'flickr.photosets.create',
429 auth_token: 'default', 429 auth_token: 'default',
430 title: psn, 430 title: psn,
431 primary_photo_id: ppid 431 primary_photo_id: ppid
432 }, function(x) { 432 }, function(x) {
433 var npid = 433 var npid =
434 x.responseXML.getElementsByTagName('photoset').item(0).getAttribute('id'); 434 x.responseXML.getElementsByTagName('photoset').item(0).getAttribute('id');
435 _this.fireflix.flickr.api_call( 435 _this.fireflix.flickr.api_call(
436 { 436 {
437 method: 'flickr.photosets.editPhotos', 437 method: 'flickr.photosets.editPhotos',
438 auth_token: 'default', 438 auth_token: 'default',
439 photoset_id: npid, 439 photoset_id: npid,
440 primary_photo_id: ppid, 440 primary_photo_id: ppid,
441 photo_ids: pids 441 photo_ids: pids
442 }, function(x) { 442 }, function(x) {
443 _this.fireflix.refresh_sets(); 443 _this.fireflix.refresh_sets();
444 }, function(x,s,c,m) { 444 }, function(x,s,c,m) {
445 _this.fireflix.flickr_failure(x,s,c,m); 445 _this.fireflix.flickr_failure(x,s,c,m);
446 } 446 }
447 ); 447 );
448 }, function(x,s,c,m) { 448 }, function(x,s,c,m) {
449 _this.fireflix.flickr_failure(x,s,c,m); 449 _this.fireflix.flickr_failure(x,s,c,m);
450 } 450 }
451 ); 451 );
452 } 452 }
453 } 453 }
454 this.selection.clearSelection(); 454 this.selection.clearSelection();
455 this.cmd_uploads_upload.setAttribute('disabled','false'); 455 this.cmd_uploads_upload.setAttribute('disabled','false');
456 this.upload_progress.setAttribute('hidden','true'); 456 this.upload_progress.setAttribute('hidden','true');
457 }, 457 },
458 458
459 clear_list: function() { 459 clear_list: function() {
460 this.tree.beginUpdateBatch(); 460 this.tree.beginUpdateBatch();
461 this.rowCount = 0; 461 this.rowCount = 0;
462 this.files = new Array(); 462 this.files = new Array();
463 this.tree.endUpdateBatch(); 463 this.tree.endUpdateBatch();
464 this.selToProps(); 464 this.selToProps();
465 }, 465 },
466 selectionChanged: function() { 466 selectionChanged: function() {
467 this.selToProps(); 467 this.selToProps();
468 }, 468 },
469 disableProps: function() { 469 disableProps: function() {
470 this.upload_filename.value=''; 470 this.upload_filename.value='';
471 this.upload_filename.disabled = true; 471 this.upload_filename.disabled = true;
472 this.upload_title.value=''; 472 this.upload_title.value='';
473 this.upload_title.disabled = true; 473 this.upload_title.disabled = true;
474 this.upload_file_preview.src = null; 474 this.upload_file_preview.src = null;
475 this.upload_file_props.hidden = true; 475 this.upload_file_props.hidden = true;
476 this.upload_tags.value=''; 476 this.upload_tags.value='';
477 this.upload_tags.disabled = true; 477 this.upload_tags.disabled = true;
478 }, 478 },
479 selToProps: function() { 479 selToProps: function() {
480 if(!this.selection.count) { 480 if(!this.selection.count) {
481 this.disableProps(); 481 this.disableProps();
482 }else if(this.selection.count==1) { 482 }else if(this.selection.count==1) {
483 var f=this.files[this.selection.currentIndex]; 483 var f=this.files[this.selection.currentIndex];
484 if(f==null || f.state!='pending') { 484 if(f==null || f.state!='pending') {
485 this.disableProps(); 485 this.disableProps();
486 }else{ 486 }else{
487 this.upload_filename.value = f.file; 487 this.upload_filename.value = f.file;
488 this.upload_filename.disabled = false; 488 this.upload_filename.disabled = false;
489 this.upload_title.value = f.title; 489 this.upload_title.value = f.title;
490 this.upload_title.disabled = false; 490 this.upload_title.disabled = false;
491 this.upload_file_preview.src = 'file:///'+f.file; 491 this.upload_file_preview.src = 'file:///'+f.file;
492 this.upload_file_props.hidden = false; 492 this.upload_file_props.hidden = false;
493 this.upload_tags.value = f.tags; 493 this.upload_tags.value = f.tags;
494 this.upload_tags.disabled = false; 494 this.upload_tags.disabled = false;
495 } 495 }
496 }else{ 496 }else{
497 var ftitle = null; var onetitle = true; 497 var ftitle = null; var onetitle = true;
498 var ftags = null; var onetag = true; 498 var ftags = null; var onetag = true;
499 var fs = 0; 499 var fs = 0;
500 for(var ff in this.files) { 500 for(var ff in this.files) {
501 if(this.selection.isSelected(ff) && this.files[ff].state=='pending' ) { 501 if(this.selection.isSelected(ff) && this.files[ff].state=='pending' ) {
502 ++fs; 502 ++fs;
503 if(ftitle==null) { 503 if(ftitle==null) {
504 ftitle = this.files[ff].title; 504 ftitle = this.files[ff].title;
505 }else if(ftitle!=this.files[ff].title) { 505 }else if(ftitle!=this.files[ff].title) {
506 onetitle = false; 506 onetitle = false;
507 } 507 }
508 if(ftags==null) { 508 if(ftags==null) {
509 ftags = this.files[ff].tags; 509 ftags = this.files[ff].tags;
510 }else if(ftags!=this.files[ff].tags) { 510 }else if(ftags!=this.files[ff].tags) {
511 onetag = false; 511 onetag = false;
512 } 512 }
513 } 513 }
514 } 514 }
515 if(fs) { 515 if(fs) {
516 this.upload_filename.value=''; 516 this.upload_filename.value='';
517 this.upload_filename.disabled = true; 517 this.upload_filename.disabled = true;
518 if(onetitle) 518 if(onetitle)
519 this.upload_title.value = ftitle; 519 this.upload_title.value = ftitle;
520 this.upload_title.disabled = false; 520 this.upload_title.disabled = false;
521 if(onetag) 521 if(onetag)
522 this.upload_tags.value = ftags; 522 this.upload_tags.value = ftags;
523 this.upload_tags.disabled = false; 523 this.upload_tags.disabled = false;
524 this.upload_file_preview.src = null; 524 this.upload_file_preview.src = null;
525 this.upload_file_props.hidden = false; 525 this.upload_file_props.hidden = false;
526 }else 526 }else
527 this.disableProps(); 527 this.disableProps();
528 } 528 }
529 }, 529 },
530 propsToSel: function(prop) { 530 propsToSel: function(prop) {
531 if(this.selection.count<=0) return; 531 if(this.selection.count<=0) return;
532 for(var ff in this.files) { 532 for(var ff in this.files) {
533 if(this.selection.isSelected(ff) && this.files[ff].state=='pending') { 533 if(this.selection.isSelected(ff) && this.files[ff].state=='pending') {
534 if(prop=='filename') 534 if(prop=='filename')
535 this.files[ff].file = this.upload_filename.value; 535 this.files[ff].file = this.upload_filename.value;
536 if(prop=='title') 536 if(prop=='title')
537 this.files[ff].title = this.upload_title.value; 537 this.files[ff].title = this.upload_title.value;
538 if(prop=='tags') 538 if(prop=='tags')
539 this.files[ff].tags = this.upload_tags.value; 539 this.files[ff].tags = this.upload_tags.value;
540 this.tree.invalidateRow(ff); 540 this.tree.invalidateRow(ff);
541 } 541 }
542 } 542 }
543 }, 543 },
544 544
545 on_upload: function() { 545 on_upload: function() {
546 this.selToProps(); 546 this.selToProps();
547 this.batch_ids = new Array(); 547 this.batch_ids = new Array();
548 this.upload_progress.value=0; 548 this.upload_progress.value=0;
549 this.upload_progress.setAttribute('hidden','false'); 549 this.upload_progress.setAttribute('hidden','false');
550 this.upload_worker(); 550 this.upload_worker();
551 }, 551 },
552 on_clear: function() { 552 on_clear: function() {
553 this.clear_list(); 553 this.clear_list();
554 }, 554 },
555 on_remove: function() { 555 on_remove: function() {
556 if(this.selection.count) { 556 if(this.selection.count) {
557 this.tree.beginUpdateBatch(); 557 this.tree.beginUpdateBatch();
558 for(var i=this.files.length-1;i>=0;--i) { 558 for(var i=this.files.length-1;i>=0;--i) {
559 if(this.selection.isSelected(i)) { 559 if(this.selection.isSelected(i)) {
560 this.files.splice(i,1); 560 this.files.splice(i,1);
561 this.rowCount--; 561 this.rowCount--;
562 } 562 }
563 } 563 }
564 this.tree.endUpdateBatch(); 564 this.tree.endUpdateBatch();
565 this.selection.clearSelection(); 565 this.selection.clearSelection();
566 } 566 }
567 }, 567 },
568 on_add: function() { 568 on_add: function() {
569 var ifp = Components.interfaces.nsIFilePicker; 569 var ifp = Components.interfaces.nsIFilePicker;
570 var fp = Components.classes["@mozilla.org/filepicker;1"].createInstance(ifp); 570 var fp = Components.classes["@mozilla.org/filepicker;1"].createInstance(ifp);
571 fp.init(window, "Select a File", ifp.modeOpenMultiple); 571 fp.init(window, "Select a File", ifp.modeOpenMultiple);
572 fp.appendFilters(ifp.filterImages); 572 fp.appendFilters(ifp.filterImages);
573 var rv = fp.show(); 573 var rv = fp.show();
574 if(rv==ifp.returnOK) { 574 if(rv==ifp.returnOK) {
575 var ff = fp.files; 575 var ff = fp.files;
576 while(ff.hasMoreElements()) { 576 while(ff.hasMoreElements()) {
577 var f = ff.getNext(); 577 var f = ff.getNext();
578 f.QueryInterface(Components.interfaces.nsIFile); 578 f.QueryInterface(Components.interfaces.nsIFile);
579 this.add(f.path); 579 this.add(f.path);
580 } 580 }
581 } 581 }
582 } 582 }
583 }, 583 },
584 584
585 on_set_props: function() { 585 on_set_props: function() {
586 var pset = this.photosets.sets[this.photosets.selection.currentIndex]; 586 var pset = this.photosets.sets[this.photosets.selection.currentIndex];
587 window.openDialog( 587 window.openDialog(
588 "chrome://fireflix/content/photoset-props.xul", 588 "chrome://fireflix/content/photoset-props.xul",
589 null, "dependent,modal,dialog,chrome", this, 589 null, "dependent,modal,dialog,chrome", this,
590 pset ); 590 pset );
591 if(pset.dirty) { 591 if(pset.dirty) {
592 var _this = this; 592 var _this = this;
593 this.flickr.api_call( 593 this.flickr.api_call(
594 { 594 {
595 method: 'flickr.photosets.editMeta', 595 method: 'flickr.photosets.editMeta',
596 auth_token: 'default', 596 auth_token: 'default',
597 photoset_id: pset.id, 597 photoset_id: pset.id,
598 title: pset.title, 598 title: pset.title,
599 description: pset.description 599 description: pset.description
600 }, function(xr) { 600 }, function(xr) {
601 pset.dirty = false; 601 pset.dirty = false;
602 _this.flickr.api_call( 602 _this.flickr.api_call(
603 { 603 {
604 method: 'flickr.photosets.getPhotos', 604 method: 'flickr.photosets.getPhotos',
605 auth_token: 'default', 605 auth_token: 'default',
606 photoset_id: pset.id 606 photoset_id: pset.id
607 }, function(xr) { 607 }, function(xr) {
608 var x = xr.responseXML; 608 var x = xr.responseXML;
609 var xp = x.evaluate( 609 var xp = x.evaluate(
610 '/rsp/photoset/photo', x, null, 610 '/rsp/photoset/photo', x, null,
611 XPathResult.ORDERED_NODE_ITERATOR_TYPE, null ); 611 XPathResult.ORDERED_NODE_ITERATOR_TYPE, null );
612 var phids = new Array(); 612 var phids = new Array();
613 var priph = null; 613 var priph = null;
614 var n; while(n=xp.iterateNext()) { 614 var n; while(n=xp.iterateNext()) {
615 var pid = n.getAttribute('id'); 615 var pid = n.getAttribute('id');
616 phids.push( pid ); 616 phids.push( pid );
617 if(pid==pset.primary && n.getAttribute('isprimary')!='1') 617 if(pid==pset.primary && n.getAttribute('isprimary')!='1')
618 priph = pid; 618 priph = pid;
619 } 619 }
620 if(priph) { 620 if(priph) {
621 _this.flickr.api_call( 621 _this.flickr.api_call(
622 { 622 {
623 method: 'flickr.photosets.editPhotos', 623 method: 'flickr.photosets.editPhotos',
624 auth_token: 'default', 624 auth_token: 'default',
625 photoset_id: pset.id, 625 photoset_id: pset.id,
626 primary_photo_id: priph, 626 primary_photo_id: priph,
627 photo_ids: phids.join(',') 627 photo_ids: phids.join(',')
628 }, function() { }, function(x,s,c,m) { /* flickr.photosets.editPhotos */ 628 }, function() { }, function(x,s,c,m) { /* flickr.photosets.editPhotos */
629 _this.flickr_failure(x,s,c,m); 629 _this.flickr_failure(x,s,c,m);
630 } 630 }
631 ); 631 );
632 } 632 }
633 }, function(x,s,c,m) { /* flickr.photosets.getPhotos */ 633 }, function(x,s,c,m) { /* flickr.photosets.getPhotos */
634 _this.flickr_failure(x,s,c,m); 634 _this.flickr_failure(x,s,c,m);
635 } 635 }
636 ); 636 );
637 }, function(x,s,c,m) { /* flickr.photosets.editMeta */ 637 }, function(x,s,c,m) { /* flickr.photosets.editMeta */
638 _this.flickr_failure(x,s,c,m); 638 _this.flickr_failure(x,s,c,m);
639 } 639 }
640 ); 640 );
641 } 641 }
642 }, 642 },
643 on_refresh_sets: function() { 643 on_refresh_sets: function() {
644 this.refresh_sets(); 644 this.refresh_sets();
645 }, 645 },
646 on_cmd_sets_html: function(csfx,ev) { 646 on_cmd_sets_html: function(csfx,ev) {
647 var uti = csfx.charAt(0); var utl = csfx.charAt(1); 647 var uti = csfx.charAt(0); var utl = csfx.charAt(1);
648 var rv = this.build_html(this.photoset.photos,uti,utl); 648 var rv = this.build_html(this.photoset.photos,uti,utl);
649 this.popup_content(rv); 649 this.popup_content(rv);
650 }, 650 },
651 651
652 on_cmd_uploads_html: function(csfx,ev) { 652 on_cmd_uploads_html: function(csfx,ev) {
653 var uti = csfx.charAt(0); var utl = csfx.charAt(1); 653 var uti = csfx.charAt(0); var utl = csfx.charAt(1);
654 var pids = new Array(); 654 var pids = new Array();
655 for(var f in this.uploads.files) { 655 for(var f in this.uploads.files) {
656 if(this.uploads.selection.isSelected(f)) 656 if(this.uploads.selection.isSelected(f))
657 if(this.uploads.files[f].photoid) 657 if(this.uploads.files[f].photoid)
658 pids.push(this.uploads.files[f].photoid); 658 pids.push(this.uploads.files[f].photoid);
659 } 659 }
660 var pp = this.uploads.rowCount*2; if(pp>500) pp = 500; 660 var pp = this.uploads.rowCount*2; if(pp>500) pp = 500;
661 var _this = this; 661 var _this = this;
662 this.flickr.api_call( 662 this.flickr.api_call(
663 { 663 {
664 method: 'flickr.photos.search', 664 method: 'flickr.photos.search',
665 auth_token: 'default', 665 auth_token: 'default',
666 extras: 'original_format', 666 extras: 'original_format',
667 user_id: 'me', 667 user_id: 'me',
668 per_page: pp 668 per_page: pp
669 }, 669 },
670 function(xr) { 670 function(xr) {
671 var x = xr.responseXML; 671 var x = xr.responseXML;
672 var rv = ''; 672 var rv = '';
673 for(var pn in pids) { 673 for(var pn in pids) {
674 var p = pids[pn]; 674 var p = pids[pn];
675 var pp = new Photo(xp_node('/rsp/photos/photo[@id='+p+']',x)); 675 var pp = new Photo(xp_node('/rsp/photos/photo[@id='+p+']',x));
676 rv += _this.photo_html(pp,uti,utl)+'\n'; 676 rv += _this.photo_html(pp,uti,utl)+'\n';
677 } 677 }
678 _this.popup_content(rv); 678 _this.popup_content(rv);
679 }, function(x,s,c,m) { 679 }, function(x,s,c,m) {
680 _this.flickr_failure(x,s,c,m); 680 _this.flickr_failure(x,s,c,m);
681 } 681 }
682 ); 682 );
683 }, 683 },
684 684
685 /* 685 /*
686 * 686 *
687 */ 687 */
688 foundphotos: { 688 foundphotos: {
689 fireflix: null, 689 fireflix: null,
690 init: function(f) { 690 init: function(f) {
691 this.fireflix = f; 691 this.fireflix = f;
692 pull_elements(this,document,[ 692 pull_elements(this,document,[
693 'search_for','search_tags','search_mine', 693 'search_for','search_tags','search_mine',
694 'searchresult_props','search_photo', 694 'searchresult_props','search_photo',
695 'searchresult_title','searchresult_description' 695 'searchresult_title','searchresult_description',
696 'search_page','cmd_search_prev_page','cmd_search_next_page'
696 ]); 697 ]);
697 document.getElementById('searchresults').view = this; 698 document.getElementById('searchresults').view = this;
698 }, 699 },
699 photos: new Array(), 700 photos: new Array(),
700 rowCount: 0, 701 rowCount: 0,
701 getCellText: function(r,c) { 702 getCellText: function(r,c) {
702 var p = this.photos[r]; 703 var p = this.photos[r];
703 if(c.id=='sr_title') return p.title; 704 if(c.id=='sr_title') return p.title;
704 return c.id; 705 return c.id;
705 }, 706 },
706 setTree: function(t) { this.tree = t }, 707 setTree: function(t) { this.tree = t },
707 isContainer: function(r) { return false }, 708 isContainer: function(r) { return false },
708 isSeparator: function(r) { return false }, 709 isSeparator: function(r) { return false },
709 isSorted: function(r) { return false }, 710 isSorted: function(r) { return false },
710 getLevel: function(r) { return 0 }, 711 getLevel: function(r) { return 0 },
711 getImageSrc: function(r,c) { return null }, 712 getImageSrc: function(r,c) { return null },
712 getRowProperties: function(r,p) { }, 713 getRowProperties: function(r,p) { },
713 getCellProperties: function(cid,cel,p) { }, 714 getCellProperties: function(cid,cel,p) { },
714 getColumnProperties: function(cid,cel,p) { }, 715 getColumnProperties: function(cid,cel,p) { },
715 cycleHeader: function(cid,e) { }, 716 cycleHeader: function(cid,e) { },
716 getParentIndex: function(r) { return -1 }, 717 getParentIndex: function(r) { return -1 },
717 drop: function(r,o) { }, 718 drop: function(r,o) { },
718 canDropBeforeAfter: function(r,b) { return false }, 719 canDropBeforeAfter: function(r,b) { return false },
719 720
720 importXPR: function(xp) { 721 importXPR: function(xp) {
721 this.selection.clearSelection(); 722 this.selection.clearSelection();
722 this.selection.currentIndex = -1; 723 this.selection.currentIndex = -1;
723 this.searchresult_props.hidden = true; 724 this.searchresult_props.hidden = true;
724 this.tree.beginUpdateBatch(); 725 this.tree.beginUpdateBatch();
725 this.photos = new Array(); 726 this.photos = new Array();
726 var n; while(n=xp.iterateNext()) { 727 var n; while(n=xp.iterateNext()) {
727 this.photos.push(new Photo(n)); 728 this.photos.push(new Photo(n));
728 } 729 }
729 this.rowCount = this.photos.length; 730 this.rowCount = this.photos.length;
730 this.tree.endUpdateBatch(); 731 this.tree.endUpdateBatch();
731 }, 732 },
733 paging: {
734 pars: null,
735 page: null, pages: null, perpage: null, total: null
736 },
732 search_photos: function() { 737 search_photos: function() {
733 var pars = { 738 var pars = {
734 method: 'flickr.photos.search', 739 method: 'flickr.photos.search',
735 auth_token: 'default', 740 auth_token: 'default',
736 extras: 'license,date_upload,date_taken,owner_name,icon_server,original_format,last_update,geo' 741 extras: 'license,date_upload,date_taken,owner_name,icon_server,original_format,last_update,geo'
737 }; 742 };
738 if(this.search_mine.checked) 743 if(this.search_mine.checked)
739 pars.user_id='me'; 744 pars.user_id='me';
740 if(this.search_tags.checked) { 745 if(this.search_tags.checked) {
741 pars.tags=this.search_for.value.split(/ +/).join(','); 746 pars.tags=this.search_for.value.split(/ +/).join(',');
742 }else{ 747 }else{
743 pars.text=this.search_for.value; 748 pars.text=this.search_for.value;
744 } 749 }
750 this.paging.pars = new Object();
751 this.paging.page = null; this.paging.pages = null;
752 this.paging.perpage = null; this.paging.total = null;
753 for(var p in pars) this.paging.pars[p] = pars[p];
754 this.perform_search(pars);
755 },
756 perform_search: function(p) {
745 var _this = this; 757 var _this = this;
746 this.fireflix.flickr.api_call( pars, 758 this.fireflix.flickr.api_call( p,
747 function(xr) { 759 function(xr) {
748 var x = xr.responseXML; 760 var x = xr.responseXML;
749 var xp = x.evaluate( 761 var xp = xp_nodes('/rsp/photos/photo',x);
750 '/rsp/photos/photo', x, null,
751 XPathResult.ORDERED_NODE_ITERATOR_TYPE, null );
752 _this.importXPR(xp); 762 _this.importXPR(xp);
753 _this.tree.ensureRowIsVisible(0); 763 _this.tree.ensureRowIsVisible(0);
764 xp = xp_node('/rsp/photos',x);
765 _this.paging.page = parseInt(xp.getAttribute('page'));
766 _this.paging.pages = parseInt(xp.getAttribute('pages'));
767 _this.paging.perpage = parseInt(xp.getAttribute('perpage'));
768 _this.paging.total = parseInt(xp.getAttribute('total'));
769 _this.update_paging();
754 _this.on_select(); 770 _this.on_select();
755 }, function(x,s,c,m) { 771 }, function(x,s,c,m) {
756 _this.fireflix.flickr_failure(x,s,c,m); 772 _this.fireflix.flickr_failure(x,s,c,m);
757 } 773 }
758 ); 774 );
759 }, 775 },
776 on_cmd_prev: function(ev) {
777 var pars = new Object();
778 for(var p in this.paging.pars) pars[p] = this.paging.pars[p];
779 pars.page=this.paging.page-1; pars.per_page=this.paging.perpage;
780 this.perform_search(pars);
781 },
782 on_cmd_next: function(ev) {
783 var pars = new Object();
784 for(var p in this.paging.pars) pars[p] = this.paging.pars[p];
785 pars.page=this.paging.page+1; pars.per_page=this.paging.perpage;
786 this.perform_search(pars);
787 },
788 update_paging: function() {
789 if(! (this.paging.pars && this.paging.page && this.paging.pages) ) {
790 this.search_page.value=''; this.search_page.hidden = true;
791 this.cmd_search_prev_page.setAttribute('disabled','true');
792 this.cmd_search_next_page.setAttribute('disabled','true');
793 }else{
794 this.search_page.value=this.fireflix.loc_strings.getFormattedString('search_page',[this.paging.page,this.paging.pages]);
795 this.search_page.hidden=false;
796 this.cmd_search_prev_page.setAttribute('disabled',(this.paging.page>1)?'false':'true');
797 this.cmd_search_next_page.setAttribute('disabled',(this.paging.page<this.paging.pages)?'false':'true');
798 }
799 },
760 render_description_frame: function(content) { 800 render_description_frame: function(content) {
761 if(!content) { 801 if(!content) {
762 this.searchresult_description.innerHTML = ''; 802 this.searchresult_description.innerHTML = '';
763 }else{ 803 }else{
764 this.searchresult_description.innerHTML = content?content:''; 804 this.searchresult_description.innerHTML = content?content:'';
765 /* of all linking elements flickr only allows a */ 805 /* of all linking elements flickr only allows a */
766 var as = this.searchresult_description.getElementsByTagName('a'); 806 var as = this.searchresult_description.getElementsByTagName('a');
767 for(var a=0;a<as.length;++a) 807 for(var a=0;a<as.length;++a)
768 as.item(a).setAttribute('target','_blank'); 808 as.item(a).setAttribute('target','_blank');
769 } 809 }
770 }, 810 },
771 on_select: function() { 811 on_select: function() {
772 if(this.selection.currentIndex<0) { 812 if(this.selection.currentIndex<0) {
773 this.searchresult_props.hidden = true; 813 this.searchresult_props.hidden = true;
774 }else{ 814 }else{
775 var p = this.photos[this.selection.currentIndex]; 815 var p = this.photos[this.selection.currentIndex];
776 if(!p) { 816 if(!p) {
777 this.searchresult_props.hidden = true; 817 this.searchresult_props.hidden = true;
778 }else{ 818 }else{
779 this.search_photo.src = this.fireflix.flickr.make_photo_url(p,'t'); 819 this.search_photo.src = this.fireflix.flickr.make_photo_url(p,'t');
780 this.searchresult_title.value = p.title; 820 this.searchresult_title.value = p.title;
781 this.searchresult_title.tooltipText = p.title; 821 this.searchresult_title.tooltipText = p.title;
782 this.render_description_frame(null); 822 this.render_description_frame(null);
783 if(p.description==null && p.description==undefined) { 823 if(p.description==null && p.description==undefined) {
784 var pid = p.id; 824 var pid = p.id;
785 var ci = this.selection.currentIndex; 825 var ci = this.selection.currentIndex;
786 var _this = this; 826 var _this = this;
787 this.fireflix.flickr.api_call( 827 this.fireflix.flickr.api_call(
788 { 828 {
789 method: 'flickr.photos.getInfo', 829 method: 'flickr.photos.getInfo',
790 auth_token: 'default', 830 auth_token: 'default',
791 photo_id: p.id, 831 photo_id: p.id,
792 secret: p.secret 832 secret: p.secret
793 }, function(xr) { 833 }, function(xr) {
794 var pp = _this.photos[ci]; 834 var pp = _this.photos[ci];
795 if(ci==_this.selection.currentIndex && pp.id==pid) { 835 if(ci==_this.selection.currentIndex && pp.id==pid) {
796 var n = xp_node('/rsp/photo',xr.responseXML); 836 var n = xp_node('/rsp/photo',xr.responseXML);
797 pp.fromNode_(n); 837 pp.fromNode_(n);
798 _this.render_description_frame(pp.description); 838 _this.render_description_frame(pp.description);
799 } 839 }
800 }, function(x,s,c,m) { 840 }, function(x,s,c,m) {
801 _this.fireflix.flickr_failure(x,s,c,m); 841 _this.fireflix.flickr_failure(x,s,c,m);
802 } 842 }
803 ); 843 );
804 this.searchresult_props.hidden = false; 844 this.searchresult_props.hidden = false;
805 }else{ 845 }else{
806 this.render_description_frame(p.description); 846 this.render_description_frame(p.description);
807 } 847 }
808 } 848 }
809 } 849 }
810 }, 850 },
811 on_cmd_open: function(ev) { 851 on_cmd_open: function(ev) {
812 if(this.selection.currentIndex<0) 852 if(this.selection.currentIndex<0)
813 return; 853 return;
814 var p = this.photos[this.selection.currentIndex]; 854 var p = this.photos[this.selection.currentIndex];
815 if(!p.id) 855 if(!p.id)
816 return; 856 return;
817 this.fireflix.openTab(this.fireflix.flickr.make_photo_url(p,'p')); 857 this.fireflix.openTab(this.fireflix.flickr.make_photo_url(p,'p'));
818 } 858 }
819 }, 859 },
820 860
821 photo_html: function(p,i,l) { 861 photo_html: function(p,i,l) {
822 // TODO: add alt/title when possible 862 // TODO: add alt/title when possible
823 var rv = 863 var rv =
824 '<a href="'+this.flickr.make_photo_url(p,l)+'">' + 864 '<a href="'+this.flickr.make_photo_url(p,l)+'">' +
825 '<img src="'+this.flickr.make_photo_url(p,i)+'" />'+ 865 '<img src="'+this.flickr.make_photo_url(p,i)+'" />'+
826 '</a>'; 866 '</a>';
827 return rv; 867 return rv;
828 }, 868 },
829 build_html: function(photos,uti,utl) { 869 build_html: function(photos,uti,utl) {
830 var rv = ''; 870 var rv = '';
831 for(var i in photos) { 871 for(var i in photos) {
832 var p = photos[i]; 872 var p = photos[i];
833 rv += this.photo_html(p,utl,uti)+'\n'; 873 rv += this.photo_html(p,utl,uti)+'\n';
834 } 874 }
835 return rv; 875 return rv;
836 }, 876 },
837 877
838 popup_content: function(s) { 878 popup_content: function(s) {
839 window.openDialog( 879 window.openDialog(
840 "chrome://fireflix/content/generated-content.xul", 880 "chrome://fireflix/content/generated-content.xul",
841 null, "dialog,chrome", this, s ); 881 null, "dialog,chrome", this, s );
842 }, 882 },
843 copy_to_clipboard: function(s) { 883 copy_to_clipboard: function(s) {
844 var ch = Components.classes["@mozilla.org/widget/clipboardhelper;1"] 884 var ch = Components.classes["@mozilla.org/widget/clipboardhelper;1"]
845 .getService(Components.interfaces.nsIClipboardHelper); 885 .getService(Components.interfaces.nsIClipboardHelper);
846 ch.copyString(s); 886 ch.copyString(s);
847 }, 887 },
848 openTab: function(l) { 888 openTab: function(l) {
849 var wm = Components.classes['@mozilla.org/appshell/window-mediator;1'].getService( 889 var wm = Components.classes['@mozilla.org/appshell/window-mediator;1'].getService(
850 Components.interfaces.nsIWindowMediator ); 890 Components.interfaces.nsIWindowMediator );
851 var bw = wm.getMostRecentWindow('navigator:browser'); 891 var bw = wm.getMostRecentWindow('navigator:browser');
852 var b = bw.getBrowser(); 892 var b = bw.getBrowser();
853 var t = b.addTab(l); 893 var t = b.addTab(l);
854 b.selectedTab = t; 894 b.selectedTab = t;
855 }, 895 },
856 896
857 build_menus: function() { 897 build_menus: function() {
858 this.append_html_menu( 898 this.append_html_menu(
859 document.getElementById('sets_html_menu'), 899 document.getElementById('sets_html_menu'),
860 'stm_','m_bop','cmdset_sets','cmd_sets_html' 900 'stm_','m_bop','cmdset_sets','cmd_sets_html'
861 ); 901 );
862 this.append_html_menu( 902 this.append_html_menu(
863 document.getElementById('uploads_html_menu'), 903 document.getElementById('uploads_html_menu'),
864 'stm_','m_bop','cmdset_uploads','cmd_uploads_html' 904 'stm_','m_bop','cmdset_uploads','cmd_uploads_html'
865 ); 905 );
866 return; 906 return;
867 }, 907 },
868 append_html_menu: function(m,imgt,lnkt,csid,cpfx) { 908 append_html_menu: function(m,imgt,lnkt,csid,cpfx) {
869 var mp = m.appendChild(document.createElement('menupopup')); 909 var mp = m.appendChild(document.createElement('menupopup'));
870 var t; 910 var t;
871 t=mp.appendChild(document.createElement('menuitem')); 911 t=mp.appendChild(document.createElement('menuitem'));
872 t.setAttribute('label',this.loc_strings.getString('menutitle_Images')); 912 t.setAttribute('label',this.loc_strings.getString('menutitle_Images'));
873 t.setAttribute('class','menuhead');t.setAttribute('disabled','true'); 913 t.setAttribute('class','menuhead');t.setAttribute('disabled','true');
874 mp.appendChild(document.createElement('menuseparator')); 914 mp.appendChild(document.createElement('menuseparator'));
875 var cs = document.getElementById(csid); 915 var cs = document.getElementById(csid);
876 for(var iti=0;iti<imgt.length;++iti) { 916 for(var iti=0;iti<imgt.length;++iti) {
877 t = mp.appendChild(document.createElement('menu')); 917 t = mp.appendChild(document.createElement('menu'));
878 t.setAttribute('label',this.loc_strings.getString('urltype_'+imgt.charAt(iti))); 918 t.setAttribute('label',this.loc_strings.getString('urltype_'+imgt.charAt(iti)));
879 var smp = t.appendChild(document.createElement('menupopup')); 919 var smp = t.appendChild(document.createElement('menupopup'));
880 t=smp.appendChild(document.createElement('menuitem')); 920 t=smp.appendChild(document.createElement('menuitem'));
881 t.setAttribute('label',this.loc_strings.getString('menutitle_Links')); 921 t.setAttribute('label',this.loc_strings.getString('menutitle_Links'));
882 t.setAttribute('class','menuhead');t.setAttribute('disabled','true'); 922 t.setAttribute('class','menuhead');t.setAttribute('disabled','true');
883 smp.appendChild(document.createElement('menuseparator')); 923 smp.appendChild(document.createElement('menuseparator'));
884 for(var lti=0;lti<lnkt.length;++lti) { 924 for(var lti=0;lti<lnkt.length;++lti) {
885 var csfx = imgt.charAt(iti)+lnkt.charAt(lti); 925 var csfx = imgt.charAt(iti)+lnkt.charAt(lti);
886 t=smp.appendChild(document.createElement('menuitem')); 926 t=smp.appendChild(document.createElement('menuitem'));
887 t.setAttribute('label',this.loc_strings.getString('urltype_'+lnkt.charAt(lti))); 927 t.setAttribute('label',this.loc_strings.getString('urltype_'+lnkt.charAt(lti)));
888 t.setAttribute('command',cpfx+'_'+csfx); 928 t.setAttribute('command',cpfx+'_'+csfx);
889 t=cs.appendChild(document.createElement('command')); 929 t=cs.appendChild(document.createElement('command'));
890 t.setAttribute('id',cpfx+'_'+csfx); 930 t.setAttribute('id',cpfx+'_'+csfx);
891 t.setAttribute('oncommand','fireflix.on_'+cpfx+"('"+csfx+"',event)"); 931 t.setAttribute('oncommand','fireflix.on_'+cpfx+"('"+csfx+"',event)");
892 } 932 }
893 } 933 }
894 return mp; 934 return mp;
895 }, 935 },
896 936
897 flickr_failure: function(x,s,c,m) { 937 flickr_failure: function(x,s,c,m) {
898 if(c==98) { // Invalid auth token 938 if(c==98) { // Invalid auth token
899 this.flickr.reset_token(); 939 this.flickr.reset_token();
900 this.set_auth_state(false,false); 940 this.set_auth_state(false,false);
901 return; 941 return;
902 } 942 }
903 // TODO: is that beauty? 943 // TODO: is that beauty?
904 alert('flickr api call failed\n'+c+' '+m); 944 alert('flickr api call failed\n'+c+' '+m);
905 } 945 }
906 946
907}; 947};
diff --git a/content/util.js b/content/util.js
index c4af09e..ccc61cf 100644
--- a/content/util.js
+++ b/content/util.js
@@ -1,70 +1,77 @@
1/* 1/*
2 * convert unicode string to utf-8 representation. 2 * convert unicode string to utf-8 representation.
3 * needed for correct md5 hash calculation. 3 * needed for correct md5 hash calculation.
4 */ 4 */
5function toutf8(ucode) { 5function toutf8(ucode) {
6 var rv = ''; 6 var rv = '';
7 for(var i=0;i<ucode.length;++i) { 7 for(var i=0;i<ucode.length;++i) {
8 var cc = ucode.charCodeAt(i); 8 var cc = ucode.charCodeAt(i);
9 if(cc<=0x7F) 9 if(cc<=0x7F)
10 rv += ucode.charAt(i); 10 rv += ucode.charAt(i);
11 else if(cc<=0x7ff) 11 else if(cc<=0x7ff)
12 rv += String.fromCharCode( 12 rv += String.fromCharCode(
13 0xc0|((cc>> 6)&0x1f), 13 0xc0|((cc>> 6)&0x1f),
14 0x80|( cc &0x3f) ); 14 0x80|( cc &0x3f) );
15 else if(cc<=0xffff) 15 else if(cc<=0xffff)
16 rv += String.fromCharCode( 16 rv += String.fromCharCode(
17 0xe0|((cc>>12)&0x0f), 17 0xe0|((cc>>12)&0x0f),
18 0x80|((cc>> 6)&0x3f), 18 0x80|((cc>> 6)&0x3f),
19 0x80|( cc &0x3f) ); 19 0x80|( cc &0x3f) );
20 else if(cc<=0x1fffff) 20 else if(cc<=0x1fffff)
21 rv += String.fromCharCode( 21 rv += String.fromCharCode(
22 0xf0|((cc>>18)&0x07), 22 0xf0|((cc>>18)&0x07),
23 0x80|((cc>>12)&0x3f), 23 0x80|((cc>>12)&0x3f),
24 0x80|((cc>> 6)&0x3f), 24 0x80|((cc>> 6)&0x3f),
25 0x80|( cc &0x3f) ); 25 0x80|( cc &0x3f) );
26 else if(cc<=0x03ffffff) 26 else if(cc<=0x03ffffff)
27 rv += String.fromCharCode( 27 rv += String.fromCharCode(
28 0xf8|((cc>>24)&0x03), 28 0xf8|((cc>>24)&0x03),
29 0x80|((cc>>18)&0x3f), 29 0x80|((cc>>18)&0x3f),
30 0x80|((cc>>12)&0x3f), 30 0x80|((cc>>12)&0x3f),
31 0x80|((cc>> 6)&0x3f), 31 0x80|((cc>> 6)&0x3f),
32 0x80|( cc &0x3f) ); 32 0x80|( cc &0x3f) );
33 else if(cc<=0x7fffffff) 33 else if(cc<=0x7fffffff)
34 rv += String.fromCharCode( 34 rv += String.fromCharCode(
35 0xfc|((cc>>30)&0x01), 35 0xfc|((cc>>30)&0x01),
36 0x80|((cc>>24)&0x3f), 36 0x80|((cc>>24)&0x3f),
37 0x80|((cc>>18)&0x3f), 37 0x80|((cc>>18)&0x3f),
38 0x80|((cc>>12)&0x3f), 38 0x80|((cc>>12)&0x3f),
39 0x80|((cc>> 6)&0x3f), 39 0x80|((cc>> 6)&0x3f),
40 0x80|( cc &0x3f) ); 40 0x80|( cc &0x3f) );
41 } 41 }
42 return rv; 42 return rv;
43} 43}
44 44
45/* 45/*
46 * extract xpath-specified string value 46 * extract xpath-specified string value
47 */ 47 */
48function xp_str(xp,x) { 48function xp_str(xp,x) {
49 var rv = x.evaluate( 49 var rv = x.evaluate(
50 xp, x, null, XPathResult.STRING_TYPE, null ); 50 xp, x, null, XPathResult.STRING_TYPE, null );
51 return rv.stringValue; 51 return rv.stringValue;
52} 52}
53/* 53/*
54 * extract xpath-specified node 54 * extract xpath-specified node
55 */ 55 */
56function xp_node(xp,x) { 56function xp_node(xp,x) {
57 var rv = x.evaluate( 57 var rv = x.evaluate(
58 xp, x, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null ); 58 xp, x, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null );
59 return rv.singleNodeValue; 59 return rv.singleNodeValue;
60} 60}
61/*
62 * extract xpath-specified ordered set of nodes
63 */
64function xp_nodes(xp,x) {
65 return x.evaluate(
66 xp, x, null, XPathResult.ORDERED_NODE_ITERATOR_TYPE, null);
67}
61 68
62/* 69/*
63 * pull in elements into documents as a member variables 70 * pull in elements into documents as a member variables
64 */ 71 */
65function pull_elements(th,d,els) { 72function pull_elements(th,d,els) {
66 for(var e in els) { 73 for(var e in els) {
67 var en=els[e]; 74 var en=els[e];
68 th[en] = d.getElementById(en); 75 th[en] = d.getElementById(en);
69 } 76 }
70} 77}