Mercurial > audlegacy
annotate audacious/util.c @ 1649:f3934d790a2e trunk
[svn] - abstractionalize a lot of stuff, build still broken, live with it
author | nenolod |
---|---|
date | Thu, 07 Sep 2006 21:44:05 -0700 |
parents | 6e477dd65024 |
children | a6e6d3500c13 |
rev | line source |
---|---|
0 | 1 /* BMP - Cross-platform multimedia player |
2 * Copyright (C) 2003-2004 BMP development team. | |
3 * | |
4 * Based on XMMS: | |
5 * Copyright (C) 1998-2003 XMMS development team. | |
6 * | |
7 * This program is free software; you can redistribute it and/or modify | |
8 * it under the terms of the GNU General Public License as published by | |
9 * the Free Software Foundation; either version 2 of the License, or | |
10 * (at your option) any later version. | |
11 * | |
12 * This program is distributed in the hope that it will be useful, | |
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
15 * GNU General Public License for more details. | |
16 * | |
17 * You should have received a copy of the GNU General Public License | |
18 * along with this program; if not, write to the Free Software | |
1459 | 19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. |
0 | 20 */ |
21 | |
245
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
22 #define WEIRD_UTF_16_PLAYLIST_ENCODING |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
23 |
0 | 24 #ifdef HAVE_CONFIG_H |
25 # include "config.h" | |
26 #endif | |
27 | |
28 #define NEED_GLADE | |
29 #include "util.h" | |
30 | |
31 #include <glib.h> | |
32 #include <glib/gi18n.h> | |
33 #include <glade/glade.h> | |
34 #include <gtk/gtk.h> | |
35 #include <stdio.h> | |
36 #include <stdlib.h> | |
37 #include <string.h> | |
38 #include <ctype.h> | |
39 | |
1640 | 40 #include "platform/smartinclude.h" |
0 | 41 #include <gdk/gdkkeysyms.h> |
42 #include <X11/Xlib.h> | |
287
2d919c756941
[svn] - attach -Iaudacious to libaudcore build crap.
nenolod
parents:
284
diff
changeset
|
43 //#include <sys/ipc.h> |
0 | 44 #include <unistd.h> |
45 #include <errno.h> | |
46 | |
47 #ifdef HAVE_FTS_H | |
48 # include <fts.h> | |
49 #endif | |
50 | |
51 #include "glade.h" | |
52 #include "input.h" | |
53 #include "main.h" | |
538
e4e897d20791
[svn] remove libaudcore, we never did anything with it
nenolod
parents:
383
diff
changeset
|
54 #include "playback.h" |
0 | 55 #include "playlist.h" |
56 | |
1105
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
57 #ifdef USE_CHARDET |
1106 | 58 #include "../libguess/libguess.h" |
1613
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
59 #include "../librcd/librcd.h" |
1106 | 60 #ifdef HAVE_UDET |
1105
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
61 #include <libudet_c.h> |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
62 #endif |
1106 | 63 #endif |
0 | 64 |
65 static GQuark quark_popup_data; | |
66 | |
67 | |
68 /* | |
69 * escape_shell_chars() | |
70 * | |
71 * Escapes characters that are special to the shell inside double quotes. | |
72 */ | |
73 | |
74 gchar * | |
75 escape_shell_chars(const gchar * string) | |
76 { | |
77 const gchar *special = "$`\"\\"; /* Characters to escape */ | |
78 const gchar *in = string; | |
79 gchar *out, *escaped; | |
80 gint num = 0; | |
81 | |
82 while (*in != '\0') | |
83 if (strchr(special, *in++)) | |
84 num++; | |
85 | |
86 escaped = g_malloc(strlen(string) + num + 1); | |
87 | |
88 in = string; | |
89 out = escaped; | |
90 | |
91 while (*in != '\0') { | |
92 if (strchr(special, *in)) | |
93 *out++ = '\\'; | |
94 *out++ = *in++; | |
95 } | |
96 *out = '\0'; | |
97 | |
98 return escaped; | |
99 } | |
100 | |
101 | |
102 /* | |
103 * find <file> in directory <dirname> or subdirectories. return | |
104 * pointer to complete filename which has to be freed by calling | |
105 * "g_free()" after use. Returns NULL if file could not be found. | |
106 */ | |
107 | |
108 typedef struct { | |
109 const gchar *to_match; | |
110 gchar *match; | |
111 gboolean found; | |
112 } FindFileContext; | |
113 | |
114 static gboolean | |
115 find_file_func(const gchar * path, const gchar * basename, gpointer data) | |
116 { | |
117 FindFileContext *context = data; | |
118 | |
119 if (strlen(path) > FILENAME_MAX) { | |
120 g_warning("Ignoring path: name too long (%s)", path); | |
121 return TRUE; | |
122 } | |
123 | |
124 if (g_file_test(path, G_FILE_TEST_IS_REGULAR)) { | |
125 if (!strcasecmp(basename, context->to_match)) { | |
126 context->match = g_strdup(path); | |
127 context->found = TRUE; | |
128 return TRUE; | |
129 } | |
130 } | |
131 else if (g_file_test(path, G_FILE_TEST_IS_DIR)) { | |
132 dir_foreach(path, find_file_func, context, NULL); | |
133 if (context->found) | |
134 return TRUE; | |
135 } | |
136 | |
137 return FALSE; | |
138 } | |
139 | |
140 gchar * | |
141 find_file_recursively(const gchar * path, const gchar * filename) | |
142 { | |
143 FindFileContext context; | |
144 | |
145 context.to_match = filename; | |
146 context.match = NULL; | |
147 context.found = FALSE; | |
148 | |
149 dir_foreach(path, find_file_func, &context, NULL); | |
150 return context.match; | |
151 } | |
152 | |
153 | |
154 typedef enum { | |
155 ARCHIVE_UNKNOWN = 0, | |
156 ARCHIVE_DIR, | |
157 ARCHIVE_TAR, | |
158 ARCHIVE_TGZ, | |
159 ARCHIVE_ZIP, | |
160 ARCHIVE_TBZ2 | |
161 } ArchiveType; | |
162 | |
163 typedef gchar *(*ArchiveExtractFunc) (const gchar *, const gchar *); | |
164 | |
165 typedef struct { | |
166 ArchiveType type; | |
167 const gchar *ext; | |
168 } ArchiveExtensionType; | |
169 | |
170 static ArchiveExtensionType archive_extensions[] = { | |
171 {ARCHIVE_TAR, ".tar"}, | |
172 {ARCHIVE_ZIP, ".wsz"}, | |
173 {ARCHIVE_ZIP, ".zip"}, | |
174 {ARCHIVE_TGZ, ".tar.gz"}, | |
175 {ARCHIVE_TGZ, ".tgz"}, | |
176 {ARCHIVE_TBZ2, ".tar.bz2"}, | |
177 {ARCHIVE_TBZ2, ".bz2"}, | |
178 {ARCHIVE_UNKNOWN, NULL} | |
179 }; | |
180 | |
181 static gchar *archive_extract_tar(const gchar * archive, const gchar * dest); | |
182 static gchar *archive_extract_zip(const gchar * archive, const gchar * dest); | |
183 static gchar *archive_extract_tgz(const gchar * archive, const gchar * dest); | |
184 static gchar *archive_extract_tbz2(const gchar * archive, const gchar * dest); | |
185 | |
186 static ArchiveExtractFunc archive_extract_funcs[] = { | |
187 NULL, | |
188 NULL, | |
189 archive_extract_tar, | |
190 archive_extract_tgz, | |
191 archive_extract_zip, | |
192 archive_extract_tbz2 | |
193 }; | |
194 | |
195 | |
196 /* FIXME: these functions can be generalised into a function using a | |
197 * command lookup table */ | |
198 | |
199 static const gchar * | |
200 get_tar_command(void) | |
201 { | |
202 static const gchar *command = NULL; | |
203 | |
204 if (!command) { | |
205 if (!(command = getenv("TARCMD"))) | |
206 command = "tar"; | |
207 } | |
208 | |
209 return command; | |
210 } | |
211 | |
212 static const gchar * | |
213 get_unzip_command(void) | |
214 { | |
215 static const gchar *command = NULL; | |
216 | |
217 if (!command) { | |
218 if (!(command = getenv("UNZIPCMD"))) | |
219 command = "unzip"; | |
220 } | |
221 | |
222 return command; | |
223 } | |
224 | |
225 | |
226 static gchar * | |
227 archive_extract_tar(const gchar * archive, const gchar * dest) | |
228 { | |
229 return g_strdup_printf("%s >/dev/null xf \"%s\" -C %s", | |
230 get_tar_command(), archive, dest); | |
231 } | |
232 | |
233 static gchar * | |
234 archive_extract_zip(const gchar * archive, const gchar * dest) | |
235 { | |
236 return g_strdup_printf("%s >/dev/null -o -j \"%s\" -d %s", | |
237 get_unzip_command(), archive, dest); | |
238 } | |
239 | |
240 static gchar * | |
241 archive_extract_tgz(const gchar * archive, const gchar * dest) | |
242 { | |
243 return g_strdup_printf("%s >/dev/null xzf \"%s\" -C %s", | |
244 get_tar_command(), archive, dest); | |
245 } | |
246 | |
247 static gchar * | |
248 archive_extract_tbz2(const gchar * archive, const gchar * dest) | |
249 { | |
250 return g_strdup_printf("bzip2 -dc \"%s\" | %s >/dev/null xf - -C %s", | |
251 archive, get_tar_command(), dest); | |
252 } | |
253 | |
254 | |
255 ArchiveType | |
256 archive_get_type(const gchar * filename) | |
257 { | |
258 gint i = 0; | |
259 | |
260 if (g_file_test(filename, G_FILE_TEST_IS_DIR)) | |
261 return ARCHIVE_DIR; | |
262 | |
263 while (archive_extensions[i].ext) { | |
264 if (g_str_has_suffix(filename, archive_extensions[i].ext)) { | |
265 return archive_extensions[i].type; | |
266 } | |
267 i++; | |
268 } | |
269 | |
270 return ARCHIVE_UNKNOWN; | |
271 } | |
272 | |
273 gboolean | |
274 file_is_archive(const gchar * filename) | |
275 { | |
276 return (archive_get_type(filename) > ARCHIVE_DIR); | |
277 } | |
278 | |
279 gchar * | |
280 archive_basename(const gchar * str) | |
281 { | |
282 gint i = 0; | |
283 | |
284 while (archive_extensions[i].ext) { | |
285 if (str_has_suffix_nocase(str, archive_extensions[i].ext)) { | |
286 const gchar *end = g_strrstr(str, archive_extensions[i].ext); | |
287 if (end) { | |
288 return g_strndup(str, end - str); | |
289 } | |
290 break; | |
291 } | |
292 i++; | |
293 } | |
294 | |
295 return NULL; | |
296 } | |
297 | |
298 /* | |
299 decompress_archive | |
300 | |
301 Decompresses the archive "filename" to a temporary directory, | |
302 returns the path to the temp dir, or NULL if failed, | |
303 watch out tho, doesn't actually check if the system command succeeds :-| | |
304 */ | |
305 | |
306 gchar * | |
307 archive_decompress(const gchar * filename) | |
308 { | |
309 gchar *tmpdir, *cmd, *escaped_filename; | |
310 ArchiveType type; | |
311 | |
312 if ((type = archive_get_type(filename)) <= ARCHIVE_DIR) | |
313 return NULL; | |
314 | |
315 tmpdir = g_build_filename(g_get_tmp_dir(), "bmp.XXXXXXXX", NULL); | |
316 if (!mkdtemp(tmpdir)) { | |
317 g_free(tmpdir); | |
318 g_message("Unable to load skin: Failed to create temporary " | |
319 "directory: %s", g_strerror(errno)); | |
320 return NULL; | |
321 } | |
322 | |
323 escaped_filename = escape_shell_chars(filename); | |
324 cmd = archive_extract_funcs[type] (escaped_filename, tmpdir); | |
325 g_free(escaped_filename); | |
326 | |
327 if (!cmd) { | |
328 g_message("extraction function is NULL!"); | |
329 g_free(tmpdir); | |
330 return NULL; | |
331 } | |
332 | |
333 system(cmd); | |
334 g_free(cmd); | |
335 | |
336 return tmpdir; | |
337 } | |
338 | |
339 | |
340 #ifdef HAVE_FTS_H | |
341 | |
342 void | |
343 del_directory(const gchar * dirname) | |
344 { | |
345 gchar *const argv[2] = { (gchar *) dirname, NULL }; | |
346 FTS *fts; | |
347 FTSENT *p; | |
348 | |
349 fts = fts_open(argv, FTS_PHYSICAL, (gint(*)())NULL); | |
350 while ((p = fts_read(fts))) { | |
351 switch (p->fts_info) { | |
352 case FTS_D: | |
353 break; | |
354 case FTS_DNR: | |
355 case FTS_ERR: | |
356 break; | |
357 case FTS_DP: | |
358 rmdir(p->fts_accpath); | |
359 break; | |
360 default: | |
361 unlink(p->fts_accpath); | |
362 break; | |
363 } | |
364 } | |
365 fts_close(fts); | |
366 } | |
367 | |
368 #else /* !HAVE_FTS */ | |
369 | |
370 gboolean | |
371 del_directory_func(const gchar * path, const gchar * basename, | |
372 gpointer params) | |
373 { | |
374 if (!strcmp(basename, ".") || !strcmp(path, "..")) | |
375 return FALSE; | |
376 | |
377 if (g_file_test(path, G_FILE_TEST_IS_DIR)) { | |
378 dir_foreach(path, del_directory_func, NULL, NULL); | |
379 rmdir(path); | |
380 return FALSE; | |
381 } | |
382 | |
383 unlink(path); | |
384 | |
385 return FALSE; | |
386 } | |
387 | |
388 void | |
389 del_directory(const gchar * path) | |
390 { | |
391 dir_foreach(path, del_directory_func, NULL, NULL); | |
392 rmdir(path); | |
393 } | |
394 | |
395 #endif /* ifdef HAVE_FTS */ | |
396 | |
397 gchar * | |
398 read_ini_string(const gchar * filename, const gchar * section, | |
399 const gchar * key) | |
400 { | |
1166
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
401 static gchar *buffer = NULL; |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
402 static gchar *open_buffer = NULL; |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
403 gchar *ret_buffer = NULL; |
625
0a73d1faeb4e
[svn] GCC 4.1 warning fixes by Diego 'Flameeyes' Petteno from Gentoo.
chainsaw
parents:
538
diff
changeset
|
404 gint found_section = 0, len = 0; |
1193
03414b9d2507
[svn] filesize must be cached, otherwise lookup loop will crash.
yaz
parents:
1166
diff
changeset
|
405 static gsize filesize = 0; |
03414b9d2507
[svn] filesize must be cached, otherwise lookup loop will crash.
yaz
parents:
1166
diff
changeset
|
406 gsize off = 0; |
245
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
407 gchar *outbuf; |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
408 unsigned char x[] = { 0xff, 0xfe, 0x00 }; |
625
0a73d1faeb4e
[svn] GCC 4.1 warning fixes by Diego 'Flameeyes' Petteno from Gentoo.
chainsaw
parents:
538
diff
changeset
|
409 guint counter; |
0 | 410 |
411 if (!filename) | |
412 return NULL; | |
413 | |
1166
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
414 /* |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
415 * We optimise for the condition that we may be reading from the |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
416 * same ini-file multiple times. This is fairly common; it happens |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
417 * on .pls playlist loads. To do otherwise would take entirely too |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
418 * long, as fstat() can be very slow when done 21,000 times too many. |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
419 * |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
420 * Therefore, we optimise by keeping the last ini file in memory. |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
421 * Yes, this is a memory leak, but it is not too bad, hopefully. |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
422 * - nenolod |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
423 */ |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
424 if (open_buffer == NULL || strcasecmp(filename, open_buffer)) |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
425 { |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
426 if (buffer != NULL) |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
427 { |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
428 g_free(buffer); |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
429 buffer = NULL; |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
430 } |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
431 |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
432 if (open_buffer != NULL) |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
433 { |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
434 g_free(open_buffer); |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
435 open_buffer = NULL; |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
436 } |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
437 |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
438 if (!g_file_get_contents(filename, &buffer, &filesize, NULL)) |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
439 return NULL; |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
440 |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
441 open_buffer = g_strdup(filename); |
36e8431d8e19
[svn] - optimise for multiple lookups of a single INI file, by caching the contents of the INI file.
nenolod
parents:
1123
diff
changeset
|
442 } |
0 | 443 |
245
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
444 /* |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
445 * Convert UTF-16 into something useful. Original implementation |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
446 * by incomp@#audacious. Cleanups \nenolod |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
447 */ |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
448 if (!memcmp(&buffer[0],&x,2)) { |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
449 outbuf = g_malloc (filesize); /* it's safe to waste memory. */ |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
450 |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
451 for (counter = 2; counter < filesize; counter += 2) |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
452 if (!memcmp(&buffer[counter+1], &x[2], 1)) |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
453 outbuf[(counter-2)/2] = buffer[counter]; |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
454 else |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
455 return NULL; |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
456 |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
457 outbuf[(counter-2)/2] = '\0'; |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
458 |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
459 if ((filesize - 2) / 2 == (counter - 2) / 2) { |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
460 g_free(buffer); |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
461 buffer = outbuf; |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
462 } else { |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
463 g_free(outbuf); |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
464 return NULL; /* XXX wrong encoding */ |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
465 } |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
466 } |
1d8012046612
[svn] Convert UTF-16 (windows-1252) into something useful. Original patch by incomp@#audacious,
nenolod
parents:
86
diff
changeset
|
467 |
0 | 468 while (!ret_buffer && off < filesize) { |
469 while (off < filesize && | |
470 (buffer[off] == '\r' || buffer[off] == '\n' || | |
471 buffer[off] == ' ' || buffer[off] == '\t')) | |
472 off++; | |
473 if (off >= filesize) | |
474 break; | |
475 if (buffer[off] == '[') { | |
476 gint slen = strlen(section); | |
477 off++; | |
478 found_section = 0; | |
479 if (off + slen + 1 < filesize && | |
480 !strncasecmp(section, &buffer[off], slen)) { | |
481 off += slen; | |
482 if (buffer[off] == ']') { | |
483 off++; | |
484 found_section = 1; | |
485 } | |
486 } | |
487 } | |
488 else if (found_section && off + strlen(key) < filesize && | |
489 !strncasecmp(key, &buffer[off], strlen(key))) { | |
490 off += strlen(key); | |
491 while (off < filesize && | |
492 (buffer[off] == ' ' || buffer[off] == '\t')) | |
493 off++; | |
494 if (off >= filesize) | |
495 break; | |
496 if (buffer[off] == '=') { | |
497 off++; | |
498 while (off < filesize && | |
499 (buffer[off] == ' ' || buffer[off] == '\t')) | |
500 off++; | |
501 if (off >= filesize) | |
502 break; | |
503 len = 0; | |
504 while (off + len < filesize && | |
505 buffer[off + len] != '\r' && | |
506 buffer[off + len] != '\n' && buffer[off + len] != ';') | |
507 len++; | |
508 ret_buffer = g_strndup(&buffer[off], len); | |
509 off += len; | |
510 } | |
511 } | |
512 while (off < filesize && buffer[off] != '\r' && buffer[off] != '\n') | |
513 off++; | |
514 } | |
515 | |
516 return ret_buffer; | |
517 } | |
518 | |
519 GArray * | |
520 string_to_garray(const gchar * str) | |
521 { | |
522 GArray *array; | |
523 gint temp; | |
524 const gchar *ptr = str; | |
525 gchar *endptr; | |
526 | |
527 array = g_array_new(FALSE, TRUE, sizeof(gint)); | |
528 for (;;) { | |
529 temp = strtol(ptr, &endptr, 10); | |
530 if (ptr == endptr) | |
531 break; | |
532 g_array_append_val(array, temp); | |
533 ptr = endptr; | |
534 while (!isdigit(*ptr) && (*ptr) != '\0') | |
535 ptr++; | |
536 if (*ptr == '\0') | |
537 break; | |
538 } | |
539 return (array); | |
540 } | |
541 | |
542 GArray * | |
543 read_ini_array(const gchar * filename, const gchar * section, | |
544 const gchar * key) | |
545 { | |
546 gchar *temp; | |
547 GArray *a; | |
548 | |
549 if ((temp = read_ini_string(filename, section, key)) == NULL) { | |
550 g_free(temp); | |
551 return NULL; | |
552 } | |
553 a = string_to_garray(temp); | |
554 g_free(temp); | |
555 return a; | |
556 } | |
557 | |
558 void | |
559 glist_movedown(GList * list) | |
560 { | |
561 gpointer temp; | |
562 | |
563 if (g_list_next(list)) { | |
564 temp = list->data; | |
565 list->data = list->next->data; | |
566 list->next->data = temp; | |
567 } | |
568 } | |
569 | |
570 void | |
571 glist_moveup(GList * list) | |
572 { | |
573 gpointer temp; | |
574 | |
575 if (g_list_previous(list)) { | |
576 temp = list->data; | |
577 list->data = list->prev->data; | |
578 list->prev->data = temp; | |
579 } | |
580 } | |
581 | |
582 | |
583 void | |
584 util_menu_position(GtkMenu * menu, gint * x, gint * y, | |
585 gboolean * push_in, gpointer data) | |
586 { | |
587 GtkRequisition requisition; | |
588 gint screen_width; | |
589 gint screen_height; | |
590 MenuPos *pos = data; | |
591 | |
592 gtk_widget_size_request(GTK_WIDGET(menu), &requisition); | |
593 | |
594 screen_width = gdk_screen_width(); | |
595 screen_height = gdk_screen_height(); | |
596 | |
597 *x = CLAMP(pos->x - 2, 0, MAX(0, screen_width - requisition.width)); | |
598 *y = CLAMP(pos->y - 2, 0, MAX(0, screen_height - requisition.height)); | |
599 } | |
600 | |
601 static void | |
602 util_menu_delete_popup_data(GtkObject * object, GtkItemFactory * ifactory) | |
603 { | |
604 gtk_signal_disconnect_by_func(object, | |
605 GTK_SIGNAL_FUNC | |
606 (util_menu_delete_popup_data), ifactory); | |
607 gtk_object_remove_data_by_id(GTK_OBJECT(ifactory), quark_popup_data); | |
608 } | |
609 | |
610 | |
611 /* | |
612 * util_item_factory_popup[_with_data]() is a replacement for | |
613 * gtk_item_factory_popup[_with_data](). | |
614 * | |
615 * The difference is that the menu is always popped up whithin the | |
616 * screen. This means it does not neccesarily pop up at (x,y). | |
617 */ | |
618 | |
619 void | |
620 util_item_factory_popup_with_data(GtkItemFactory * ifactory, | |
621 gpointer data, | |
622 GtkDestroyNotify destroy, guint x, | |
623 guint y, guint mouse_button, guint32 time) | |
624 { | |
625 static GQuark quark_user_menu_pos = 0; | |
626 MenuPos *pos; | |
627 | |
628 if (!quark_user_menu_pos) | |
629 quark_user_menu_pos = g_quark_from_static_string("user_menu_pos"); | |
630 | |
631 if (!quark_popup_data) | |
632 quark_popup_data = | |
633 g_quark_from_static_string("GtkItemFactory-popup-data"); | |
634 | |
635 pos = g_object_get_qdata(G_OBJECT(ifactory), quark_user_menu_pos); | |
636 if (!pos) { | |
637 pos = g_new0(MenuPos, 1); | |
638 | |
639 g_object_set_qdata_full(G_OBJECT(ifactory->widget), | |
640 quark_user_menu_pos, pos, g_free); | |
641 } | |
642 pos->x = x; | |
643 pos->y = y; | |
644 | |
645 | |
646 if (data != NULL) { | |
647 g_object_set_qdata_full(G_OBJECT(ifactory), | |
648 quark_popup_data, data, destroy); | |
649 g_signal_connect(G_OBJECT(ifactory->widget), | |
650 "selection-done", | |
651 G_CALLBACK(util_menu_delete_popup_data), ifactory); | |
652 } | |
653 | |
654 gtk_menu_popup(GTK_MENU(ifactory->widget), NULL, NULL, | |
655 (GtkMenuPositionFunc) util_menu_position, | |
656 pos, mouse_button, time); | |
657 } | |
658 | |
659 void | |
660 util_item_factory_popup(GtkItemFactory * ifactory, guint x, guint y, | |
661 guint mouse_button, guint32 time) | |
662 { | |
663 util_item_factory_popup_with_data(ifactory, NULL, NULL, x, y, | |
664 mouse_button, time); | |
665 } | |
666 | |
667 | |
668 #define URL_HISTORY_MAX_SIZE 30 | |
669 | |
670 static void | |
671 util_add_url_callback(GtkWidget * widget, | |
672 GtkEntry * entry) | |
673 { | |
674 const gchar *text; | |
675 | |
676 text = gtk_entry_get_text(entry); | |
677 if (g_list_find_custom(cfg.url_history, text, (GCompareFunc) strcasecmp)) | |
678 return; | |
679 | |
680 cfg.url_history = g_list_prepend(cfg.url_history, g_strdup(text)); | |
681 | |
682 while (g_list_length(cfg.url_history) > URL_HISTORY_MAX_SIZE) { | |
683 GList *node = g_list_last(cfg.url_history); | |
684 g_free(node->data); | |
685 cfg.url_history = g_list_delete_link(cfg.url_history, node); | |
686 } | |
687 } | |
688 | |
689 GtkWidget * | |
86 | 690 util_add_url_dialog_new(const gchar * caption, GCallback ok_func, |
691 GCallback enqueue_func) | |
0 | 692 { |
86 | 693 GtkWidget *win, *vbox, *bbox, *enqueue, *ok, *cancel, *combo, *entry; |
0 | 694 GList *url; |
695 | |
696 win = gtk_window_new(GTK_WINDOW_TOPLEVEL); | |
697 gtk_window_set_title(GTK_WINDOW(win), caption); | |
698 gtk_window_set_type_hint(GTK_WINDOW(win), GDK_WINDOW_TYPE_HINT_DIALOG); | |
699 gtk_window_set_position(GTK_WINDOW(win), GTK_WIN_POS_CENTER); | |
700 gtk_window_set_default_size(GTK_WINDOW(win), 400, -1); | |
701 gtk_container_set_border_width(GTK_CONTAINER(win), 12); | |
702 | |
703 vbox = gtk_vbox_new(FALSE, 10); | |
704 gtk_container_add(GTK_CONTAINER(win), vbox); | |
705 | |
706 combo = gtk_combo_box_entry_new_text(); | |
707 gtk_box_pack_start(GTK_BOX(vbox), combo, FALSE, FALSE, 0); | |
708 | |
709 entry = gtk_bin_get_child(GTK_BIN(combo)); | |
710 gtk_window_set_focus(GTK_WINDOW(win), entry); | |
711 gtk_entry_set_text(GTK_ENTRY(entry), ""); | |
712 | |
713 for (url = cfg.url_history; url; url = g_list_next(url)) | |
714 gtk_combo_box_append_text(GTK_COMBO_BOX(combo), (const gchar *) url->data); | |
715 | |
716 g_signal_connect(entry, "activate", | |
717 G_CALLBACK(util_add_url_callback), | |
718 entry); | |
719 g_signal_connect(entry, "activate", | |
86 | 720 G_CALLBACK(ok_func), |
0 | 721 entry); |
722 g_signal_connect_swapped(entry, "activate", | |
723 G_CALLBACK(gtk_widget_destroy), | |
724 win); | |
725 | |
726 bbox = gtk_hbutton_box_new(); | |
727 gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox), GTK_BUTTONBOX_END); | |
728 gtk_button_box_set_spacing(GTK_BUTTON_BOX(bbox), 5); | |
729 gtk_box_pack_start(GTK_BOX(vbox), bbox, FALSE, FALSE, 0); | |
730 | |
86 | 731 ok = gtk_button_new_from_stock(GTK_STOCK_OPEN); |
732 g_signal_connect(ok, "clicked", | |
733 G_CALLBACK(util_add_url_callback), entry); | |
734 g_signal_connect(ok, "clicked", | |
735 G_CALLBACK(ok_func), entry); | |
736 g_signal_connect_swapped(ok, "clicked", | |
737 G_CALLBACK(gtk_widget_destroy), | |
738 win); | |
739 gtk_box_pack_start(GTK_BOX(bbox), ok, FALSE, FALSE, 0); | |
740 | |
0 | 741 enqueue = gtk_button_new_from_stock(GTK_STOCK_ADD); |
742 gtk_box_pack_start(GTK_BOX(bbox), enqueue, FALSE, FALSE, 0); | |
743 | |
744 g_signal_connect(enqueue, "clicked", | |
745 G_CALLBACK(util_add_url_callback), | |
746 entry); | |
747 g_signal_connect(enqueue, "clicked", | |
748 G_CALLBACK(enqueue_func), | |
749 entry); | |
750 g_signal_connect_swapped(enqueue, "clicked", | |
751 G_CALLBACK(gtk_widget_destroy), | |
752 win); | |
753 | |
754 cancel = gtk_button_new_from_stock(GTK_STOCK_CLOSE); | |
755 gtk_box_pack_start(GTK_BOX(bbox), cancel, FALSE, FALSE, 0); | |
756 | |
757 g_signal_connect_swapped(cancel, "clicked", | |
758 G_CALLBACK(gtk_widget_destroy), | |
759 win); | |
760 | |
761 gtk_widget_show_all(vbox); | |
762 | |
763 return win; | |
764 } | |
765 | |
1649
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
766 /* *** TO WA2GUI *** */ |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
767 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
768 #if 0 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
769 /* text_get_extents() taken from The GIMP (C) Spencer Kimball, Peter |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
770 * Mattis et al */ |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
771 gboolean |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
772 text_get_extents(const gchar * fontname, |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
773 const gchar * text, |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
774 gint * width, gint * height, gint * ascent, gint * descent) |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
775 { |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
776 PangoFontDescription *font_desc; |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
777 PangoLayout *layout; |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
778 PangoRectangle rect; |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
779 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
780 g_return_val_if_fail(fontname != NULL, FALSE); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
781 g_return_val_if_fail(text != NULL, FALSE); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
782 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
783 /* FIXME: resolution */ |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
784 layout = gtk_widget_create_pango_layout(GTK_WIDGET(mainwin), text); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
785 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
786 font_desc = pango_font_description_from_string(fontname); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
787 pango_layout_set_font_description(layout, font_desc); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
788 pango_font_description_free(font_desc); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
789 pango_layout_get_pixel_extents(layout, NULL, &rect); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
790 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
791 if (width) |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
792 *width = rect.width; |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
793 if (height) |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
794 *height = rect.height; |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
795 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
796 if (ascent || descent) { |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
797 PangoLayoutIter *iter; |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
798 PangoLayoutLine *line; |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
799 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
800 iter = pango_layout_get_iter(layout); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
801 line = pango_layout_iter_get_line(iter); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
802 pango_layout_iter_free(iter); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
803 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
804 pango_layout_line_get_pixel_extents(line, NULL, &rect); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
805 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
806 if (ascent) |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
807 *ascent = PANGO_ASCENT(rect); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
808 if (descent) |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
809 *descent = -PANGO_DESCENT(rect); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
810 } |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
811 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
812 g_object_unref(layout); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
813 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
814 return TRUE; |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
815 } |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
816 |
0 | 817 static void |
818 filebrowser_add_files(GtkFileChooser * browser, | |
819 GSList * files) | |
820 { | |
821 GSList *cur; | |
822 gchar *ptr; | |
823 guint ctr = 0; | |
824 | |
825 if (GTK_IS_WIDGET(mainwin_jtf)) | |
826 gtk_widget_set_sensitive(mainwin_jtf, FALSE); | |
827 | |
828 for (cur = files; cur; cur = g_slist_next(cur)) { | |
829 | |
830 if (g_file_test(cur->data,G_FILE_TEST_IS_DIR)) { | |
831 playlist_add_dir((const gchar *) cur->data); | |
832 } else { | |
833 playlist_add((const gchar *) cur->data); | |
834 } | |
835 | |
836 if (++ctr == 20) { | |
837 playlistwin_update_list(); | |
838 ctr = 0; | |
839 while (gtk_events_pending() ) gtk_main_iteration(); | |
840 } | |
841 } | |
842 | |
843 playlistwin_update_list(); | |
844 | |
845 if (GTK_IS_WIDGET(mainwin_jtf)) | |
846 gtk_widget_set_sensitive(mainwin_jtf, TRUE); | |
847 | |
848 #ifdef HAVE_GNOME_VFS | |
849 ptr = gtk_file_chooser_get_current_folder_uri(GTK_FILE_CHOOSER(browser)); | |
850 #else | |
851 ptr = gtk_file_chooser_get_current_folder(GTK_FILE_CHOOSER(browser)); | |
852 #endif | |
853 | |
854 g_free(cfg.filesel_path); | |
855 cfg.filesel_path = ptr; | |
856 } | |
857 | |
858 static void | |
859 filebrowser_add(GtkFileChooser * browser) | |
860 { | |
861 GSList *files; | |
862 | |
863 #ifdef HAVE_GNOME_VFS | |
864 files = gtk_file_chooser_get_uris(GTK_FILE_CHOOSER(browser)); | |
865 #else | |
866 files = gtk_file_chooser_get_filenames(GTK_FILE_CHOOSER(browser)); | |
867 #endif | |
868 | |
869 if (!files) { | |
870 return; | |
871 } | |
872 | |
873 filebrowser_add_files(browser, files); | |
874 g_slist_foreach(files, (GFunc) g_free, NULL); | |
875 g_slist_free(files); | |
876 } | |
877 | |
878 static void | |
879 filebrowser_play(GtkFileChooser * browser) | |
880 { | |
881 GSList *files; | |
882 | |
883 #ifdef HAVE_GNOME_VFS | |
884 files = gtk_file_chooser_get_uris(GTK_FILE_CHOOSER(browser)); | |
885 #else | |
886 files = gtk_file_chooser_get_filenames(GTK_FILE_CHOOSER(browser)); | |
887 #endif | |
888 | |
889 if (!files) return; | |
890 | |
891 playlist_clear(); | |
892 | |
893 filebrowser_add_files(browser, files); | |
894 g_slist_foreach(files, (GFunc) g_free, NULL); | |
895 g_slist_free(files); | |
896 | |
897 bmp_playback_initiate(); | |
898 } | |
899 | |
900 | |
901 static void | |
902 _filebrowser_add(GtkWidget *widget, | |
903 gpointer data) | |
904 { | |
905 filebrowser_add(data); | |
906 gtk_file_chooser_unselect_all(GTK_FILE_CHOOSER(data)); | |
907 } | |
908 | |
909 static void | |
910 _filebrowser_play(GtkWidget *widget, gpointer data) | |
911 { | |
912 filebrowser_play(data); | |
913 gtk_file_chooser_unselect_all(data); | |
914 } | |
915 | |
916 #if 0 | |
917 static void | |
918 filebrowser_on_response(GtkFileChooser * browser, | |
919 gint response, | |
920 gpointer data) | |
921 { | |
922 gtk_widget_hide(GTK_WIDGET(browser)); | |
923 switch (response) { | |
924 case GTK_RESPONSE_OK: | |
925 break; | |
926 case GTK_RESPONSE_ACCEPT: | |
927 break; | |
928 case GTK_RESPONSE_CLOSE: | |
929 break; | |
930 } | |
931 gtk_widget_destroy(GTK_WIDGET(browser)); | |
932 | |
933 } | |
934 | |
935 #endif | |
936 | |
937 static void | |
938 _filebrowser_check_hide_add(GtkWidget * widget, | |
939 gpointer data) | |
940 { | |
941 cfg.close_dialog_add = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget)); | |
942 } | |
943 | |
944 static void | |
945 _filebrowser_check_hide_open(GtkWidget * widget, | |
946 gpointer data) | |
947 { | |
948 cfg.close_dialog_open = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget)); | |
949 } | |
950 | |
951 | |
952 | |
953 static gboolean | |
954 filebrowser_on_keypress(GtkWidget * browser, | |
955 GdkEventKey * event, | |
956 gpointer data) | |
957 { | |
958 if (event->keyval == GDK_Escape) { | |
959 /* FIXME: this crashes BMP for some reason */ | |
960 /* g_signal_emit_by_name(browser, "delete-event"); */ | |
961 gtk_widget_hide(browser); | |
962 return TRUE; | |
963 } | |
964 | |
965 return FALSE; | |
966 } | |
967 | |
968 static void | |
969 _filebrowser_do_hide_add(GtkWidget *widget, | |
970 gpointer data) | |
971 { | |
972 if (cfg.close_dialog_add) | |
973 gtk_widget_hide(data); | |
974 } | |
975 | |
976 static void | |
977 _filebrowser_do_hide_open(GtkWidget *widget, | |
978 gpointer data) | |
979 { | |
980 if (cfg.close_dialog_open) | |
981 gtk_widget_hide(data); | |
982 } | |
983 | |
984 void | |
985 util_run_filebrowser(gboolean play_button) | |
986 { | |
987 static GladeXML *xml = NULL; | |
988 static GtkWidget *dialog = NULL; | |
989 static GtkWidget *chooser = NULL; | |
990 | |
991 static GtkWidget *button_add; | |
992 static GtkWidget *button_select_all, *button_deselect_all; | |
993 static GtkWidget *toggle; | |
994 | |
995 static gulong handlerid, handlerid_check, handlerid_do; | |
996 static gulong handlerid_activate, handlerid_do_activate; | |
997 | |
998 if (!xml) { | |
999 /* FIXME: Creating a file chooser dialog manually using | |
1000 libglade because there's no way to stop | |
1001 GtkFileChooserDialog from resizing the buttons to the same | |
1002 size. The long toggle button title causes the buttons to | |
1003 turn unnecessarily elongated and fugly. */ | |
1004 | |
1005 GtkWidget *alignment; | |
1006 | |
1007 xml = glade_xml_new_or_die(_("Add/Open Files dialog"), | |
1008 DATA_DIR "/glade/addfiles.glade", | |
1009 NULL, NULL); | |
1010 glade_xml_signal_autoconnect(xml); | |
1011 | |
1012 dialog = glade_xml_get_widget(xml, "add_files_dialog"); | |
1013 | |
1014 /* FIXME: Creating file chooser widget here because libglade <= 2.4.0 does | |
1015 not support GtkFileChooserWidget */ | |
1016 | |
1017 chooser = gtk_file_chooser_widget_new(GTK_FILE_CHOOSER_ACTION_OPEN); | |
1018 gtk_file_chooser_set_select_multiple(GTK_FILE_CHOOSER(chooser), TRUE); | |
1019 | |
1020 #ifdef HAVE_GNOME_VFS | |
1021 gtk_file_chooser_set_local_only(GTK_FILE_CHOOSER(chooser), FALSE); | |
1022 #endif | |
1023 | |
1024 if (cfg.filesel_path) | |
1025 #ifdef HAVE_GNOME_VFS | |
1026 gtk_file_chooser_set_current_folder_uri(GTK_FILE_CHOOSER(chooser), | |
1027 cfg.filesel_path); | |
1028 #else | |
1029 gtk_file_chooser_set_current_folder(GTK_FILE_CHOOSER(chooser), | |
1030 cfg.filesel_path); | |
1031 #endif | |
1032 | |
1033 alignment = glade_xml_get_widget(xml, "alignment2"); | |
1034 gtk_container_add(GTK_CONTAINER(alignment), chooser); | |
1035 | |
1036 toggle = glade_xml_get_widget(xml, "close_on_action"); | |
1037 button_select_all = glade_xml_get_widget(xml, "select_all"); | |
1038 button_deselect_all = glade_xml_get_widget(xml, "deselect_all"); | |
1039 button_add = glade_xml_get_widget(xml, "action"); | |
1040 | |
1041 g_signal_connect_swapped(button_select_all, "clicked", | |
1042 G_CALLBACK(gtk_file_chooser_select_all), | |
1043 chooser); | |
1044 g_signal_connect_swapped(button_deselect_all, "clicked", | |
1045 G_CALLBACK(gtk_file_chooser_unselect_all), | |
1046 chooser); | |
1047 | |
1048 g_signal_connect(dialog, "key_press_event", | |
1049 G_CALLBACK(filebrowser_on_keypress), | |
1050 NULL); | |
1051 | |
1052 gtk_widget_show_all(dialog); | |
1053 } /* !xml */ | |
1054 else { | |
1055 g_signal_handler_disconnect(button_add, handlerid); | |
1056 g_signal_handler_disconnect(toggle, handlerid_check); | |
1057 g_signal_handler_disconnect(chooser, handlerid_activate); | |
1058 g_signal_handler_disconnect(button_add, handlerid_do); | |
1059 g_signal_handler_disconnect(chooser, handlerid_do_activate); | |
893
6afdd0d7e1e1
[svn] Make refreshing optional, default to disabled if Gnome VFS available.
nemo
parents:
879
diff
changeset
|
1060 if (cfg.refresh_file_list) { |
6afdd0d7e1e1
[svn] Make refreshing optional, default to disabled if Gnome VFS available.
nemo
parents:
879
diff
changeset
|
1061 // *sigh* force a refresh |
6afdd0d7e1e1
[svn] Make refreshing optional, default to disabled if Gnome VFS available.
nemo
parents:
879
diff
changeset
|
1062 gtk_file_chooser_set_current_folder(GTK_FILE_CHOOSER(chooser),gtk_file_chooser_get_current_folder(GTK_FILE_CHOOSER(chooser))); |
6afdd0d7e1e1
[svn] Make refreshing optional, default to disabled if Gnome VFS available.
nemo
parents:
879
diff
changeset
|
1063 } |
0 | 1064 } |
1065 | |
1066 if (play_button) { | |
1067 gtk_window_set_title(GTK_WINDOW(dialog), _("Open Files")); | |
1068 | |
1069 gtk_button_set_label(GTK_BUTTON(button_add), GTK_STOCK_OPEN); | |
1070 | |
1071 gtk_button_set_label(GTK_BUTTON(toggle), _("Close dialog on Open")); | |
1072 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(toggle), cfg.close_dialog_open); | |
1073 | |
1074 handlerid = g_signal_connect(button_add, "clicked", G_CALLBACK(_filebrowser_play), chooser); | |
1075 handlerid_check = g_signal_connect(toggle, "toggled", G_CALLBACK(_filebrowser_check_hide_open), NULL); | |
1076 handlerid_do = g_signal_connect_after(button_add, "clicked", G_CALLBACK(_filebrowser_do_hide_open), dialog); | |
1077 handlerid_activate = g_signal_connect(chooser, "file-activated", G_CALLBACK(_filebrowser_play), chooser); | |
1078 handlerid_do_activate = g_signal_connect_after(chooser,"file_activated", G_CALLBACK(_filebrowser_do_hide_open), dialog); | |
1079 } | |
1080 else { | |
1081 gtk_window_set_title(GTK_WINDOW(dialog), _("Add Files")); | |
1082 | |
1083 gtk_button_set_label(GTK_BUTTON(button_add), GTK_STOCK_ADD); | |
1084 | |
1085 gtk_button_set_label(GTK_BUTTON(toggle), _("Close dialog on Add")); | |
1086 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(toggle), cfg.close_dialog_add); | |
1087 | |
1088 handlerid = g_signal_connect(button_add, "clicked", G_CALLBACK(_filebrowser_add), chooser); | |
1089 handlerid_check = g_signal_connect(toggle, "toggled", G_CALLBACK(_filebrowser_check_hide_add), NULL); | |
1090 handlerid_do = g_signal_connect_after(button_add, "clicked", G_CALLBACK(_filebrowser_do_hide_add), dialog); | |
1091 handlerid_activate = g_signal_connect(chooser, "file-activated", G_CALLBACK(_filebrowser_add), chooser); | |
1092 handlerid_do_activate = g_signal_connect_after(chooser,"file_activated", G_CALLBACK(_filebrowser_do_hide_add), dialog); | |
1093 } | |
1094 | |
1095 gtk_window_present(GTK_WINDOW(dialog)); | |
1096 } | |
1097 | |
1649
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1098 #endif |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1099 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1100 /******************************************************************** keep in util.c */ |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1101 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1102 GtkWidget * |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1103 make_filebrowser(const gchar * title, |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1104 GtkWidget *parent, |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1105 gboolean save) |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1106 { |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1107 GtkWidget *dialog; |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1108 GtkWidget *button; |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1109 GtkWidget *button_close; |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1110 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1111 g_return_val_if_fail(title != NULL, NULL); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1112 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1113 dialog = gtk_file_chooser_dialog_new(title, GTK_WINDOW(parent), |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1114 GTK_FILE_CHOOSER_ACTION_OPEN, NULL, NULL); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1115 if (save) |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1116 gtk_file_chooser_set_action(GTK_FILE_CHOOSER(dialog), |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1117 GTK_FILE_CHOOSER_ACTION_SAVE); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1118 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1119 if (!save) |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1120 gtk_file_chooser_set_select_multiple(GTK_FILE_CHOOSER(dialog), TRUE); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1121 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1122 g_signal_connect(dialog, "destroy", |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1123 G_CALLBACK(gtk_widget_destroyed), &dialog); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1124 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1125 #ifdef HAVE_GNOME_VFS |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1126 gtk_file_chooser_set_local_only(GTK_FILE_CHOOSER(dialog), FALSE); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1127 #endif |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1128 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1129 button_close = gtk_dialog_add_button(GTK_DIALOG(dialog), GTK_STOCK_CANCEL, |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1130 GTK_RESPONSE_REJECT); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1131 gtk_button_set_use_stock(GTK_BUTTON(button_close), TRUE); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1132 GTK_WIDGET_SET_FLAGS(button_close, GTK_CAN_DEFAULT); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1133 g_signal_connect_swapped(button_close, "clicked", |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1134 G_CALLBACK(gtk_widget_destroy), dialog); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1135 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1136 button = gtk_dialog_add_button(GTK_DIALOG(dialog), save ? |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1137 GTK_STOCK_SAVE : GTK_STOCK_OPEN, |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1138 GTK_RESPONSE_ACCEPT); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1139 gtk_button_set_use_stock(GTK_BUTTON(button), TRUE); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1140 GTK_WIDGET_SET_FLAGS(button, GTK_CAN_DEFAULT); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1141 gtk_window_set_default(GTK_WINDOW(dialog), button); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1142 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1143 gtk_widget_show(dialog); |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1144 |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1145 return dialog; |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1146 } |
f3934d790a2e
[svn] - abstractionalize a lot of stuff, build still broken, live with it
nenolod
parents:
1640
diff
changeset
|
1147 |
0 | 1148 GdkFont * |
1149 util_font_load(const gchar * name) | |
1150 { | |
1151 GdkFont *font; | |
1152 PangoFontDescription *desc; | |
1153 | |
1154 desc = pango_font_description_from_string(name); | |
1155 font = gdk_font_from_description(desc); | |
1156 | |
1157 return font; | |
1158 } | |
1159 | |
1160 #ifdef ENABLE_NLS | |
1161 gchar * | |
1162 bmp_menu_translate(const gchar * path, gpointer func_data) | |
1163 { | |
1164 gchar *translation = gettext(path); | |
1165 | |
1166 if (!translation || *translation != '/') { | |
1167 g_warning("Bad translation for menupath: %s", path); | |
1168 translation = (gchar *) path; | |
1169 } | |
1170 | |
1171 return translation; | |
1172 } | |
1173 #endif | |
1174 | |
1175 void | |
1176 util_set_cursor(GtkWidget * window) | |
1177 { | |
1178 static GdkCursor *cursor = NULL; | |
1179 | |
1180 if (!window) { | |
1181 if (cursor) { | |
813
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
625
diff
changeset
|
1182 gdk_cursor_unref(cursor); |
0 | 1183 cursor = NULL; |
1184 } | |
813
c8cf439179b8
[svn] - Fix a ton and a half of memory leaks, via the wonderful Leonardo Boshell <leonardop -at- gentoo.org>.
nenolod
parents:
625
diff
changeset
|
1185 |
0 | 1186 return; |
1187 } | |
1188 | |
1189 if (!cursor) | |
1190 cursor = gdk_cursor_new(GDK_LEFT_PTR); | |
1191 | |
1192 gdk_window_set_cursor(window->window, cursor); | |
1193 } | |
1194 | |
1195 /* counts number of digits in a gint */ | |
1196 guint | |
1197 gint_count_digits(gint n) | |
1198 { | |
1199 guint count = 0; | |
1200 | |
1201 n = ABS(n); | |
1202 do { | |
1203 count++; | |
1204 n /= 10; | |
1205 } while (n > 0); | |
1206 | |
1207 return count; | |
1208 } | |
1209 | |
1210 static gchar * | |
1211 str_twenty_to_space(gchar * str) | |
1212 { | |
1213 gchar *match, *match_end; | |
1214 | |
1215 g_return_val_if_fail(str != NULL, NULL); | |
1216 | |
1217 while ((match = strstr(str, "%20"))) { | |
1218 match_end = match + 3; | |
1219 *match++ = ' '; | |
1220 while (*match_end) | |
1221 *match++ = *match_end++; | |
1222 *match = 0; | |
1223 } | |
1224 | |
1225 return str; | |
1226 } | |
1227 | |
1228 static gchar * | |
1229 str_replace_char(gchar * str, gchar old, gchar new) | |
1230 { | |
1231 gchar *match; | |
1232 | |
1233 g_return_val_if_fail(str != NULL, NULL); | |
1234 | |
1235 match = str; | |
1236 while ((match = strchr(match, old))) | |
1237 *match = new; | |
1238 | |
1239 return str; | |
1240 } | |
1241 | |
1242 gchar * | |
1243 str_append(gchar * str, const gchar * add_str) | |
1244 { | |
1245 return str_replace(str, g_strconcat(str, add_str, NULL)); | |
1246 } | |
1247 | |
1248 gchar * | |
1249 str_replace(gchar * str, gchar * new_str) | |
1250 { | |
1251 g_free(str); | |
1252 return new_str; | |
1253 } | |
1254 | |
1255 void | |
1256 str_replace_in(gchar ** str, gchar * new_str) | |
1257 { | |
1258 *str = str_replace(*str, new_str); | |
1259 } | |
1260 | |
1261 | |
1262 gboolean | |
1263 str_has_prefix_nocase(const gchar * str, const gchar * prefix) | |
1264 { | |
1265 return (strncasecmp(str, prefix, strlen(prefix)) == 0); | |
1266 } | |
1267 | |
1268 gboolean | |
1269 str_has_suffix_nocase(const gchar * str, const gchar * suffix) | |
1270 { | |
1271 return (strcasecmp(str + strlen(str) - strlen(suffix), suffix) == 0); | |
1272 } | |
1273 | |
1274 gboolean | |
1275 str_has_suffixes_nocase(const gchar * str, gchar * const *suffixes) | |
1276 { | |
1277 gchar *const *suffix; | |
1278 | |
1279 g_return_val_if_fail(str != NULL, FALSE); | |
1280 g_return_val_if_fail(suffixes != NULL, FALSE); | |
1281 | |
1282 for (suffix = suffixes; *suffix; suffix++) | |
1283 if (str_has_suffix_nocase(str, *suffix)) | |
1284 return TRUE; | |
1285 | |
1286 return FALSE; | |
1287 } | |
1288 | |
1289 gchar * | |
1290 str_to_utf8_fallback(const gchar * str) | |
1291 { | |
1292 gchar *out_str, *convert_str, *chr; | |
1293 | |
1294 /* NULL in NULL out */ | |
1295 if (!str) | |
1296 return NULL; | |
1297 | |
1298 convert_str = g_strdup(str); | |
1299 for (chr = convert_str; *chr; chr++) { | |
1300 if (*chr & 0x80) | |
1301 *chr = '?'; | |
1302 } | |
1303 | |
1304 out_str = g_strconcat(convert_str, _(" (invalid UTF-8)"), NULL); | |
1305 g_free(convert_str); | |
1306 | |
1307 return out_str; | |
1308 } | |
1309 | |
1310 gchar * | |
1311 filename_to_utf8(const gchar * filename) | |
1312 { | |
1313 gchar *out_str; | |
1314 | |
1315 /* NULL in NULL out */ | |
1316 if (!filename) | |
1317 return NULL; | |
1318 | |
1319 if ((out_str = g_filename_to_utf8(filename, -1, NULL, NULL, NULL))) | |
1320 return out_str; | |
1321 | |
1322 return str_to_utf8_fallback(filename); | |
1323 } | |
1324 | |
1325 gchar * | |
1326 str_to_utf8(const gchar * str) | |
1327 { | |
1328 gchar *out_str; | |
1329 | |
1330 /* NULL in NULL out */ | |
1331 if (!str) | |
1332 return NULL; | |
1333 | |
1324
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1334 /* Note: Currently, playlist calls this function repeatedly, even |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1335 * if the string is already converted into utf-8. |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1336 * chardet_to_utf8() would convert a valid utf-8 string into a |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1337 * different utf-8 string, if fallback encodings were supplied and |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1338 * the given string could be treated as a string in one of fallback |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1339 * encodings. To avoid this, the order of evaluation has been |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1340 * changed. (It might cause a drawback?) |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1341 */ |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1342 /* already UTF-8? */ |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1343 if (g_utf8_validate(str, -1, NULL)) |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1344 return g_strdup(str); |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1345 |
1105
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1346 /* chardet encoding detector */ |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1347 if ((out_str = chardet_to_utf8(str, strlen(str), NULL, NULL, NULL))) |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1348 return out_str; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1349 |
0 | 1350 /* assume encoding associated with locale */ |
1351 if ((out_str = g_locale_to_utf8(str, -1, NULL, NULL, NULL))) | |
1352 return out_str; | |
1353 | |
1354 /* all else fails, we mask off character codes >= 128, | |
1355 replace with '?' */ | |
1356 return str_to_utf8_fallback(str); | |
1357 } | |
1358 | |
1359 | |
1360 const gchar * | |
1361 str_skip_chars(const gchar * str, const gchar * chars) | |
1362 { | |
1363 while (strchr(chars, *str)) | |
1364 str++; | |
1365 return str; | |
1366 } | |
1367 | |
1368 gchar * | |
1369 convert_title_text(gchar * title) | |
1370 { | |
1371 g_return_val_if_fail(title != NULL, NULL); | |
1372 | |
1373 if (cfg.convert_underscore) | |
1374 str_replace_char(title, '_', ' '); | |
1375 | |
1376 if (cfg.convert_twenty) | |
1377 str_twenty_to_space(title); | |
1378 | |
1379 return title; | |
1380 } | |
1381 | |
1382 | |
1383 gboolean | |
1384 dir_foreach(const gchar * path, DirForeachFunc function, | |
1385 gpointer user_data, GError ** error) | |
1386 { | |
1387 GError *error_out = NULL; | |
1388 GDir *dir; | |
1389 const gchar *entry; | |
1390 gchar *entry_fullpath; | |
1391 | |
1392 if (!(dir = g_dir_open(path, 0, &error_out))) { | |
1393 g_propagate_error(error, error_out); | |
1394 return FALSE; | |
1395 } | |
1396 | |
1397 while ((entry = g_dir_read_name(dir))) { | |
1398 entry_fullpath = g_build_filename(path, entry, NULL); | |
1399 | |
1400 if ((*function) (entry_fullpath, entry, user_data)) { | |
1401 g_free(entry_fullpath); | |
1402 break; | |
1403 } | |
1404 | |
1405 g_free(entry_fullpath); | |
1406 } | |
1407 | |
1408 g_dir_close(dir); | |
1409 | |
1410 return TRUE; | |
1411 } | |
1412 | |
1413 GtkItemFactory * | |
1414 create_menu(GtkItemFactoryEntry *entries, | |
1415 guint n_entries, | |
1416 GtkAccelGroup *accel) | |
1417 { | |
1418 GtkItemFactory *menu; | |
1419 | |
1420 menu = gtk_item_factory_new(GTK_TYPE_MENU, "<Main>", accel); | |
1421 gtk_item_factory_set_translate_func(menu, bmp_menu_translate, NULL, | |
1422 NULL); | |
1423 gtk_item_factory_create_items(menu, n_entries, entries, NULL); | |
1424 | |
1425 return menu; | |
1426 } | |
1427 | |
1428 | |
1429 void | |
1430 make_submenu(GtkItemFactory *menu, | |
1431 const gchar *item_path, | |
1432 GtkItemFactory *submenu) | |
1433 { | |
1434 GtkWidget *item, *menu_; | |
1435 | |
1436 item = gtk_item_factory_get_widget(menu, item_path); | |
1437 menu_ = gtk_item_factory_get_widget(submenu, ""); | |
1438 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), menu_); | |
1439 } | |
1440 | |
1441 | |
1105
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1442 gchar *chardet_to_utf8(const gchar *str, gssize len, |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1443 gsize *arg_bytes_read, gsize *arg_bytes_write, GError **arg_error) |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1444 { |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1445 #ifdef USE_CHARDET |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1446 char *det = NULL, *encoding = NULL; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1447 #endif |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1448 gchar *ret = NULL; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1449 gsize *bytes_read, *bytes_write; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1450 GError **error; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1451 gsize my_bytes_read, my_bytes_write; |
0 | 1452 |
1105
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1453 bytes_read = arg_bytes_read ? arg_bytes_read : &my_bytes_read; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1454 bytes_write = arg_bytes_write ? arg_bytes_write : &my_bytes_write; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1455 error = arg_error ? arg_error : NULL; |
0 | 1456 |
1105
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1457 #ifdef USE_CHARDET |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1458 if(cfg.chardet_detector) |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1459 det = cfg.chardet_detector; |
0 | 1460 |
1105
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1461 if(det){ |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1462 if(!strncasecmp("japanese", det, sizeof("japanese"))) { |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1463 encoding = (char *)guess_jp(str, strlen(str)); |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1464 if (!encoding) |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1465 goto fallback; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1466 } else if(!strncasecmp("taiwanese", det, sizeof("taiwanese"))) { |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1467 encoding = (char *)guess_tw(str, strlen(str)); |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1468 if (!encoding) |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1469 goto fallback; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1470 } else if(!strncasecmp("chinese", det, sizeof("chinese"))) { |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1471 encoding = (char *)guess_cn(str, strlen(str)); |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1472 if (!encoding) |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1473 goto fallback; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1474 } else if(!strncasecmp("korean", det, sizeof("korean"))) { |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1475 encoding = (char *)guess_kr(str, strlen(str)); |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1476 if (!encoding) |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1477 goto fallback; |
1613
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1478 } else if(!strncasecmp("russian", det, sizeof("russian"))) { |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1479 rcd_russian_charset res = rcdGetRussianCharset(str, strlen(str)); |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1480 switch(res) { |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1481 case RUSSIAN_CHARSET_WIN: |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1482 encoding = "CP1251"; |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1483 break; |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1484 case RUSSIAN_CHARSET_ALT: |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1485 encoding = "CP866"; |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1486 break; |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1487 case RUSSIAN_CHARSET_KOI: |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1488 encoding = "KOI8-R"; |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1489 break; |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1490 case RUSSIAN_CHARSET_UTF8: |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1491 encoding = "UTF-8"; |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1492 break; |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1493 } |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1494 if (!encoding) |
9becbe564217
[svn] - chardet patch for Russian language witten by Valentine Sinitsyn.
yaz
parents:
1459
diff
changeset
|
1495 goto fallback; |
1105
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1496 #ifdef HAVE_UDET |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1497 } else if (!strncasecmp("universal", det, sizeof("universal"))) { |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1498 encoding = (char *)detectCharset((char *)str, strlen(str)); |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1499 if (!encoding) |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1500 goto fallback; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1501 #endif |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1502 } else /* none, invalid */ |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1503 goto fallback; |
0 | 1504 |
1105
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1505 ret = g_convert(str, len, "UTF-8", encoding, bytes_read, bytes_write, error); |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1506 } |
0 | 1507 |
1105
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1508 fallback: |
1123
170f9b68dd03
[svn] The fallback label is not used unless detection is defined. Move endif down to reflect that.
chainsaw
parents:
1106
diff
changeset
|
1509 #endif |
1105
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1510 if(!ret && cfg.chardet_fallback){ |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1511 gchar **encs=NULL, **enc=NULL; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1512 encs = g_strsplit_set(cfg.chardet_fallback, " ,:;|/", 0); |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1513 |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1514 if(encs){ |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1515 enc = encs; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1516 for(enc=encs; *enc ; enc++){ |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1517 ret = g_convert(str, len, "UTF-8", *enc, bytes_read, bytes_write, error); |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1518 if(len == *bytes_read){ |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1519 break; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1520 } |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1521 } |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1522 g_strfreev(encs); |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1523 } |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1524 } |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1525 |
1324
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1526 #ifdef USE_CHARDET |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1527 /* many tag libraries return 2byte latin1 utf8 character as |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1528 converted 8bit iso-8859-1 character, if they are asked to return |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1529 latin1 string. |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1530 */ |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1531 if(!ret){ |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1532 ret = g_convert(str, len, "UTF-8", "ISO-8859-1", bytes_read, bytes_write, error); |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1533 } |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1534 #endif |
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1535 |
1105
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1536 if(ret){ |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1537 if(g_utf8_validate(ret, -1, NULL)) |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1538 return ret; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1539 else { |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1540 g_free(ret); |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1541 ret = NULL; |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1542 } |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1543 } |
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1544 |
1324
99454d030722
[svn] try to avoid malconversion of latin1 character when chardet has been enabled.
yaz
parents:
1193
diff
changeset
|
1545 return NULL; /* if I have no idea, return NULL. */ |
1105
4be4d74db123
[svn] automatic character encoding detector for id3 metadata. --enable-chardet enables this feature.
yaz
parents:
1067
diff
changeset
|
1546 } |