summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMike Frysinger <vapier@gentoo.org>2003-09-13 03:46:59 +0000
committerMike Frysinger <vapier@gentoo.org>2003-09-13 03:46:59 +0000
commitc06f32cde6f99a539fc177ccf1f2680d89468f2a (patch)
tree05413012831e5a3f2520279e95fc42436dadf11d /media-sound/xmms/files
parentwhite space fixes (diff)
downloadgentoo-2-c06f32cde6f99a539fc177ccf1f2680d89468f2a.tar.gz
gentoo-2-c06f32cde6f99a539fc177ccf1f2680d89468f2a.tar.bz2
gentoo-2-c06f32cde6f99a539fc177ccf1f2680d89468f2a.zip
DTD patch #27967
Diffstat (limited to 'media-sound/xmms/files')
-rw-r--r--media-sound/xmms/files/digest-xmms-1.2.8-r12
-rw-r--r--media-sound/xmms/files/xmms-1.2.8-dtd.patch1985
-rw-r--r--media-sound/xmms/files/xmms-1.2.8-jump.patch76
-rw-r--r--media-sound/xmms/files/xmms-1.2.8-russian-charset.patch2174
-rw-r--r--media-sound/xmms/files/xmms-1.2.8-sigterm.patch31
5 files changed, 4268 insertions, 0 deletions
diff --git a/media-sound/xmms/files/digest-xmms-1.2.8-r1 b/media-sound/xmms/files/digest-xmms-1.2.8-r1
new file mode 100644
index 000000000000..55451e9e64ea
--- /dev/null
+++ b/media-sound/xmms/files/digest-xmms-1.2.8-r1
@@ -0,0 +1,2 @@
+MD5 bf1d7ea568b01d72e99d75ae9d0d1594 xmms-1.2.8.tar.gz 3884444
+MD5 a68edd5b03d650c366f09852a8414ee2 gentoo_ice.zip 68458
diff --git a/media-sound/xmms/files/xmms-1.2.8-dtd.patch b/media-sound/xmms/files/xmms-1.2.8-dtd.patch
new file mode 100644
index 000000000000..94b342568b0c
--- /dev/null
+++ b/media-sound/xmms/files/xmms-1.2.8-dtd.patch
@@ -0,0 +1,1985 @@
+diff -ruN dtd-old/AUTHORS dtd-new/AUTHORS
+--- dtd-old/AUTHORS 2003-01-07 15:50:08.000000000 +0100
++++ dtd-new/AUTHORS 2003-09-08 13:46:19.000000000 +0200
+@@ -42,6 +42,7 @@
+ Chris Wilson
+ Dave Yearke
+ Stephan K. Zitz
++ Johan Walles (Dynamic Taste Detection)
+
+ Default skin: Leonard "Blayde" Tan
+ Robin Sylvestre (Equalizer and Playlist)
+diff -ruN dtd-old/README dtd-new/README
+--- dtd-old/README 2003-09-04 23:00:18.000000000 +0200
++++ dtd-new/README 2003-09-08 13:46:19.000000000 +0200
+@@ -57,6 +57,7 @@
+ 5. Features
+ 5.1 Supported File formats
+ 5.2 Supported Features
++ 5.2.1 Dynamic Taste Detection
+ 6. Obtaining XMMS
+ 7. Misc
+ 7.1 Shoutcast support
+@@ -1121,6 +1122,34 @@
+ Compiles and works on other UNIX's
+ Proxy authentication support
+
++5.2.1 Dynamic Taste Detection
++-----------------------------
++Basically, Dynamic Taste Detection collects statistics on how you
++listen to songs, and then adapts the shuffle play and randomize
++playlist functions to better follow your taste. It takes care of
++itself entirely, so that's all you really need to know about it ;-).
++
++For those of you who are curious, here is how it works. Two kinds of
++statistics are collected:
++
++- What songs you don't like
++- What songs you like to hear next to each other
++
++The first one is easy. When you skip a song, it looses a point.
++Songs with low scores are placed at the end of the playlist.
++
++What songs you want to hear next to each other are a bit more
++intricate. Let's say you have two songs, A and B. A (or any song
++above it in the playlist) is playing. You move song B to the position
++right after A. A finishes playing, and B starts. This is interpreted
++as "you want to hear A and B next to each other".
++
++In this example, A and B will then *tend* to (i.e. not always) end up
++next to each other in the playlist (though not necessarily in that
++order).
++
++To explicitly change your opinion of a song, or disconnect one song
++from another, you can do so from the right-click menu in the playlist.
+
+ 6. Obtaining XMMS
+ -------------------
+diff -ruN dtd-old/xmms/Makefile.am dtd-new/xmms/Makefile.am
+--- dtd-old/xmms/Makefile.am 2003-09-08 13:31:40.000000000 +0200
++++ dtd-new/xmms/Makefile.am 2003-09-08 13:46:19.000000000 +0200
+@@ -34,6 +34,7 @@
+ menurow.c menurow.h \
+ hslider.c hslider.h \
+ monostereo.c monostereo.h \
++dtd.c dtd.h \
+ vis.c vis.h \
+ svis.c svis.h \
+ number.c number.h \
+diff -ruN dtd-old/xmms/dtd.c dtd-new/xmms/dtd.c
+--- dtd-old/xmms/dtd.c 1970-01-01 01:00:00.000000000 +0100
++++ dtd-new/xmms/dtd.c 2003-09-08 14:37:21.000000000 +0200
+@@ -0,0 +1,624 @@
++/* XMMS - Cross-platform multimedia player
++ * Copyright (C) 2001 Johan Walles, d92-jwa@nada.kth.se
++ *
++ * This program is free software; you can redistribute it and/or modify
++ * it under the terms of the GNU General Public License as published by
++ * the Free Software Foundation; either version 2 of the License, or
++ * (at your option) any later version.
++ *
++ * This program is distributed in the hope that it will be useful,
++ * but WITHOUT ANY WARRANTY; without even the implied warranty of
++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
++ * GNU General Public License for more details.
++ *
++ * You should have received a copy of the GNU General Public License
++ * along with this program; if not, write to the Free Software
++ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
++ */
++
++/* Hint: DTD = Dynamic Taste Detection */
++
++#include <assert.h>
++
++#include "xmms.h"
++
++static GList *dtd_recommendations_list;
++
++static DtdNextSong *dtd_find_next_song(DtdStartSong *start_song,
++ const gchar *filename)
++{
++ GList *iterator;
++
++ for (iterator = start_song->next_song;
++ iterator != NULL;
++ iterator = g_list_next(iterator))
++ {
++ if (strcmp(((DtdNextSong *)(iterator->data))->filename,
++ filename) == 0)
++ {
++ return (DtdNextSong *)(iterator->data);
++ }
++ }
++
++ return NULL;
++}
++
++static DtdStartSong *dtd_find_start_song(const gchar *filename)
++{
++ GList *iterator;
++
++ if (filename == NULL)
++ {
++ return NULL;
++ }
++
++ for (iterator = dtd_recommendations_list;
++ iterator != NULL;
++ iterator = g_list_next(iterator))
++ {
++ if (strcmp(((DtdStartSong *)(iterator->data))->filename,
++ filename) == 0)
++ {
++ return (DtdStartSong *)(iterator->data);
++ }
++ }
++
++ return NULL;
++}
++
++/*
++ * Memorize that after playing the file named start, the user might
++ * want to hear the file named follow.
++ */
++static void __dtd_recommend_next(int weight, const gchar *start, const gchar *follow)
++{
++ DtdStartSong *start_song;
++ DtdNextSong *next_song;
++
++ /* Check if the start song already has recommendations. */
++ if ((start_song = dtd_find_start_song(start)) ==
++ NULL)
++ {
++ /* Nope, create a new recommendation start node */
++ start_song = g_new(DtdStartSong, 1);
++
++ start_song->filename = g_strdup(start);
++ start_song->next_song = NULL;
++ start_song->score = 0;
++
++ /* Add the recommendation start node to the
++ recommendation start node list. */
++ dtd_recommendations_list =
++ g_list_append(dtd_recommendations_list,
++ start_song);
++ }
++
++ /* Check if the next song is already recommended for this
++ start song */
++ if ((next_song = dtd_find_next_song(start_song,
++ follow)) == NULL)
++ {
++ /* It's not. Create a recommendation next node */
++ next_song = g_new(DtdNextSong, 1);
++
++ next_song->filename = g_strdup(follow);
++
++ next_song->weight = 0;
++
++ /* Add the new recommendation to the start node */
++ start_song->next_song =
++ g_list_append(start_song->next_song,
++ next_song);
++ }
++
++ next_song->weight += weight;
++}
++
++/*
++ * Memorize that after playing the file named start, the user might
++ * want to hear the file named follow.
++ */
++void dtd_recommend_next(const gchar *start, const gchar *follow)
++{
++ /* Recommend symmetrically */
++ __dtd_recommend_next(1, start, follow);
++ __dtd_recommend_next(1, follow, start);
++}
++
++void __dtd_dissociate(const gchar *file1, const gchar *file2)
++{
++ DtdStartSong *startSong;
++ DtdNextSong *nextSong = NULL;
++ GList *nextSongIterator;
++
++ // Find the start pointer for file1
++ startSong = dtd_find_start_song(file1);
++ if (startSong == NULL)
++ {
++ // No such recommendation exists
++ return;
++ }
++
++ // Find its next pointer for file2
++ for (nextSongIterator = startSong->next_song;
++ nextSongIterator != NULL;
++ nextSongIterator = g_list_next(nextSongIterator))
++ {
++ nextSong = (DtdNextSong *)(nextSongIterator->data);
++ if (strcmp(file2, nextSong->filename) == 0)
++ {
++ break;
++ }
++ }
++ if (nextSong == NULL)
++ {
++ // No such recommendation exists
++ return;
++ }
++
++ // Remove the next pointer
++ startSong->next_song =
++ g_list_remove_link(startSong->next_song, nextSongIterator);
++ g_free(nextSong->filename);
++ g_free(nextSong);
++ g_list_free_1(nextSongIterator);
++}
++
++void dtd_dissociate(const gchar *file1, const gchar *file2)
++{
++ // The slave's chain is heavy in both ends, so let's set 'em
++ // both free :-)
++ __dtd_dissociate(file1, file2);
++ __dtd_dissociate(file2, file1);
++}
++
++/*
++ * Modify a song's score.
++ */
++void dtd_set_score(const gchar *filename, gint score)
++{
++ DtdStartSong *start_song = dtd_find_start_song(filename);
++
++ if (start_song == NULL)
++ {
++ start_song = g_new(DtdStartSong, 1);
++
++ start_song->filename = g_strdup(filename);
++ start_song->next_song = NULL;
++ start_song->score = 0;
++
++ /* Add the new recommendation start node to the
++ recommendation start node list. */
++ dtd_recommendations_list =
++ g_list_append(dtd_recommendations_list,
++ start_song);
++ }
++
++ start_song->score = score;
++ /* Scores > 0 means "always put this song at the start of the
++ list". Probably we don't want that.
++ */
++ if (start_song->score > 0)
++ {
++ start_song->score = 0;
++ }
++}
++
++void dtd_change_score(const gchar *filename, gint delta)
++{
++ DtdStartSong *start_song = dtd_find_start_song(filename);
++
++ if (start_song == NULL)
++ {
++ start_song = g_new(DtdStartSong, 1);
++
++ start_song->filename = g_strdup(filename);
++ start_song->next_song = NULL;
++ start_song->score = 0;
++
++ /* Add the new recommendation start node to the
++ recommendation start node list. */
++ dtd_recommendations_list =
++ g_list_append(dtd_recommendations_list,
++ start_song);
++ }
++
++ /* Scores > 0 means "always put this song at the start of the
++ list". Probably we don't want that.
++ */
++ start_song->score += delta;
++ if (start_song->score > 0)
++ {
++ start_song->score = 0;
++ }
++}
++
++/*
++ * Retrieve a song's score.
++ */
++gint dtd_get_score(const gchar *filename)
++{
++ DtdStartSong *start_song = dtd_find_start_song(filename);
++
++ if (start_song == NULL)
++ {
++ return 0;
++ }
++ else
++ {
++ return start_song->score;
++ }
++}
++
++const GList *dtd_get_recommendations(const gchar *start)
++{
++ DtdStartSong *dtdStartSong = dtd_find_start_song(start);
++
++ return dtdStartSong ? dtdStartSong->next_song : NULL;
++}
++
++static void __dtd_set_score(gchar *filename, gint score)
++{
++ DtdStartSong *start_song = dtd_find_start_song(filename);
++
++ if (start_song == NULL)
++ {
++ start_song = g_new(DtdStartSong, 1);
++
++ start_song->filename = g_strdup(filename);
++ start_song->next_song = NULL;
++ start_song->score = 0;
++
++ /* Add the new recommendation start node to the
++ recommendation start node list. */
++ dtd_recommendations_list =
++ g_list_append(dtd_recommendations_list,
++ start_song);
++ }
++
++ start_song->score = score;
++}
++
++static guint dtd_sum_recommendations(DtdStartSong *start_song)
++{
++ GList *iterator;
++ guint sum = 0;
++
++ assert(start_song != NULL);
++
++ for (iterator = start_song->next_song;
++ iterator != NULL;
++ iterator = g_list_next(iterator))
++ {
++ sum += ((DtdNextSong *)(iterator->data))->weight;
++ }
++
++ assert(sum != 0);
++
++ return sum;
++}
++
++static const gchar *dtd_get_nth_recommendation(DtdStartSong *start_song,
++ guint n)
++{
++ GList *iterator;
++ guint sum = 0;
++
++ assert(start_song != NULL);
++
++ if (n == 0)
++ return NULL;
++
++ for (iterator = start_song->next_song;
++ iterator != NULL;
++ iterator = g_list_next(iterator))
++ {
++ sum += ((DtdNextSong *)(iterator->data))->weight;
++
++ if (sum >= n)
++ return ((DtdNextSong *)(iterator->data))->filename;
++ }
++
++ /* If we get here then the n passed to this function was too
++ large. It can be no larger than the number calculated by
++ dtd_sum_recommendations(). */
++ assert(FALSE);
++
++ return NULL;
++}
++
++/*
++ Try to recommend a song to play after first_filename. NULL means
++ that no recommendation is given (for whatever reason).
++
++ FIXME: This method should accept a list of songs *not* to choose
++ between. If the user has asked for a recommendation and received an
++ answer that is not in the current playlist, they must be able to try
++ again.
++*/
++const gchar *dtd_get_recommendation(const gchar *first_filename)
++{
++ DtdStartSong *start_song;
++ guint recommendation_total;
++
++ if (first_filename == NULL)
++ {
++ return NULL;
++ }
++
++ if ((start_song = dtd_find_start_song(first_filename)) ==
++ NULL)
++ {
++ /* There are no recommendations for first_filename */
++ return NULL;
++ }
++
++ if (start_song->next_song == NULL)
++ {
++ return NULL;
++ }
++
++ recommendation_total = dtd_sum_recommendations(start_song);
++
++ return dtd_get_nth_recommendation(start_song,
++ rand() % (recommendation_total + 1));
++}
++
++static void dtd_clear(void)
++{
++ GList *start_iterator;
++ GList *next_iterator;
++
++ if (dtd_recommendations_list == NULL)
++ {
++ return;
++ }
++
++ /* Free all memory used by the recommendations */
++ for (start_iterator = dtd_recommendations_list;
++ start_iterator != NULL;
++ start_iterator = g_list_next(start_iterator))
++ {
++ /* Free all file names in the next list */
++ for (next_iterator = ((DtdStartSong *)(start_iterator->data))->next_song;
++ next_iterator != NULL;
++ next_iterator = g_list_next(next_iterator))
++ {
++ free(((DtdNextSong *)(next_iterator->data))->filename);
++ ((DtdNextSong *)(next_iterator->data))->filename = NULL;
++
++ free(next_iterator->data);
++ next_iterator->data = NULL;
++ }
++ /* Free the next list */
++ g_list_free(((DtdStartSong *)(start_iterator->data))->next_song);
++ ((DtdStartSong *)(start_iterator->data))->next_song = NULL;
++
++ /* Free the file name */
++ free(((DtdStartSong *)(start_iterator->data))->filename);
++ ((DtdStartSong *)(start_iterator->data))->filename = NULL;
++
++ /* Free the struct */
++ free(start_iterator->data);
++ start_iterator->data = NULL;
++ }
++ g_list_free(dtd_recommendations_list);
++ dtd_recommendations_list = NULL;
++}
++
++void dtd_init(void)
++{
++ gchar *dtd_data_file_name;
++ FILE *dtd_data_file;
++
++ gchar start[950], follow[950];
++ gchar line[999];
++ int weight;
++
++ enum { NEXT, SCORES } section = NEXT;
++
++ gboolean failure = FALSE;
++
++ /* We have no recommendations to begin with */
++ dtd_recommendations_list = NULL;
++
++ /* Make up a filename for the dtd data file */
++ dtd_data_file_name = g_strconcat(g_get_home_dir(),
++ "/.xmms/",
++ DTD_DATA_FILE_NAME,
++ NULL);
++
++ /* Open the dtd data file for input */
++ dtd_data_file = fopen(dtd_data_file_name,
++ "r");
++ if (dtd_data_file == NULL)
++ {
++ return;
++ }
++
++ /* Parse the dtd data file */
++ start[0] = '\0';
++ while (!feof(dtd_data_file))
++ {
++ if (fgets(line, 990, dtd_data_file) != NULL)
++ {
++ if (sscanf(line, "%d %940[^\n]\n", &weight, follow) == 2)
++ {
++ /* Found a line with both number and name */
++
++ switch (section)
++ {
++ case NEXT:
++ if (start[0] != '\0')
++ {
++ if (weight < 0)
++ {
++ failure= TRUE;
++ }
++ else if (weight > 0)
++ {
++ __dtd_recommend_next(weight, start, follow);
++ }
++ }
++ else
++ {
++ /* Parse error! */
++ failure = TRUE;
++ }
++ break;
++
++ case SCORES:
++ if (weight != 0)
++ {
++ __dtd_set_score(follow, weight);
++ }
++ break;
++ }
++
++ if (failure) break;
++ }
++ else if (sscanf(line, "%940[^\n]\n", start) == 1)
++ {
++ if (strcmp(start, "scores") == 0)
++ {
++ /* The scores section starts here */
++ section = SCORES;
++ start[0] = '\0';
++ }
++ else
++ {
++ /* New start song read OK */
++ section = NEXT;
++ }
++ }
++ else
++ {
++ /* Parse error! */
++ failure = TRUE;
++ break;
++ }
++ }
++ }
++
++ if (failure)
++ {
++ /* Free all space occupied by the recommendations list
++ in favour of remembering a broken list. */
++ dtd_clear();
++ }
++
++ fclose(dtd_data_file);
++ dtd_data_file = NULL;
++}
++
++void dtd_persist(void)
++{
++ gchar *dtd_data_file_name;
++ FILE *dtd_data_file;
++
++ GList *start_iterator;
++ GList *next_iterator;
++
++ gboolean failure = FALSE;
++
++ /* Are there any recommendations to store */
++ if (dtd_recommendations_list == NULL)
++ {
++ /* Nope. */
++ return;
++ }
++
++ /* Make up a filename for the dtd data file */
++ dtd_data_file_name = g_strconcat(g_get_home_dir(),
++ "/.xmms/",
++ DTD_DATA_FILE_NAME,
++ NULL);
++
++ /* Open the dtd data file for output */
++ dtd_data_file = fopen(dtd_data_file_name,
++ "w");
++ if (dtd_data_file == NULL)
++ {
++ /* FIXME: We fail silently. Should we output a
++ * warning somehow? */
++ return;
++ }
++
++ /* Loop through all recommendation starts, storing their
++ recommended followers */
++ for (start_iterator = dtd_recommendations_list;
++ start_iterator != NULL;
++ start_iterator = g_list_next(start_iterator))
++ {
++ if (((DtdStartSong *)(start_iterator->data))->next_song == NULL)
++ {
++ /* Don't store songs with only a score (yet). */
++ continue;
++ }
++
++ if (failure ||
++ (fprintf(dtd_data_file,
++ "%s\n",
++ ((DtdStartSong *)(start_iterator->data))->filename) == 0))
++ {
++ failure = TRUE;
++ break;
++ }
++
++ for (next_iterator = ((DtdStartSong *)(start_iterator->data))->next_song;
++ next_iterator != NULL;
++ next_iterator = g_list_next(next_iterator))
++ {
++ if (fprintf(dtd_data_file, "%d %s\n",
++ ((DtdNextSong *)(next_iterator->data))->weight,
++ ((DtdNextSong *)(next_iterator->data))->filename) == 0)
++ {
++ failure = TRUE;
++ break;
++ }
++ }
++ }
++
++ if (!failure)
++ {
++ /* Output the "here starts the scoring section" marker. */
++ if (fprintf(dtd_data_file, "scores\n") == 0)
++ {
++ failure = TRUE;
++ }
++
++ /* Output all (non-zero) scores */
++ for (start_iterator = dtd_recommendations_list;
++ start_iterator != NULL;
++ start_iterator = g_list_next(start_iterator))
++ {
++ if (((DtdStartSong *)(start_iterator->data))->score == 0)
++ {
++ /* Don't store songs with zero score. */
++ continue;
++ }
++
++ if (failure ||
++ (fprintf(dtd_data_file,
++ "%d %s\n",
++ ((DtdStartSong *)(start_iterator->data))->score,
++ ((DtdStartSong *)(start_iterator->data))->filename) == 0))
++ {
++ failure = TRUE;
++ break;
++ }
++ }
++ }
++
++ fclose(dtd_data_file);
++ dtd_data_file = NULL;
++
++ if (failure)
++ {
++ /* If writing of the data file failed, attempt to
++ remove it rather than leaving a broken data file
++ behind. */
++
++ unlink(dtd_data_file_name);
++ }
++}
+diff -ruN dtd-old/xmms/dtd.h dtd-new/xmms/dtd.h
+--- dtd-old/xmms/dtd.h 1970-01-01 01:00:00.000000000 +0100
++++ dtd-new/xmms/dtd.h 2003-09-08 14:38:32.000000000 +0200
+@@ -0,0 +1,73 @@
++/* XMMS - Cross-platform multimedia player
++ * Copyright (C) 2001 Johan Walles, d92-jwa@nada.kth.se
++ *
++ * This program is free software; you can redistribute it and/or modify
++ * it under the terms of the GNU General Public License as published by
++ * the Free Software Foundation; either version 2 of the License, or
++ * (at your option) any later version.
++ *
++ * This program is distributed in the hope that it will be useful,
++ * but WITHOUT ANY WARRANTY; without even the implied warranty of
++ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
++ * GNU General Public License for more details.
++ *
++ * You should have received a copy of the GNU General Public License
++ * along with this program; if not, write to the Free Software
++ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
++ */
++
++/* Hint: DTD = Dynamic Taste Detection */
++
++/* FIXME: Should lists prove to be too slow we could use balanced
++ trees for the DtdStartSong collection. The DtdNextSong is scanned
++ through however, so tree-ifying will do no good. */
++
++#ifndef DTD_H
++#define DTD_H
++
++#define DTD_DATA_FILE_NAME "dtd-stats"
++
++typedef struct
++{
++ gchar *filename;
++ guint weight;
++}
++DtdNextSong;
++
++typedef struct
++{
++ gchar *filename;
++ gint score;
++ GList *next_song;
++}
++DtdStartSong;
++
++void dtd_init(void);
++void dtd_persist(void);
++
++// Tell DTD that after start, the user tends to want to hear follow
++void dtd_recommend_next(const gchar *start,
++ const gchar *follow);
++
++// Tell DTD to break the association between file1 and file2 (if one
++// exists)
++void dtd_dissociate(const gchar *file1,
++ const gchar *file2);
++
++
++// Let DTD recommend what to play after start. NULL = No
++// recommendation = anything goes.
++const gchar *dtd_get_recommendation(const gchar *start);
++
++// Change the absolute score for a file
++void dtd_change_score(const gchar *filename, gint delta);
++void dtd_set_score(const gchar *filename, gint score);
++
++// Get the score for a file. Unknown file = score 0.
++gint dtd_get_score(const gchar *filename);
++
++// Get the list of recommendations for a given file. NULL = no
++// recommendations for that file.
++const GList *dtd_get_recommendations(const gchar *start);
++
++#endif
+diff -ruN dtd-old/xmms/main.c dtd-new/xmms/main.c
+--- dtd-old/xmms/main.c 2003-09-08 13:31:40.000000000 +0200
++++ dtd-new/xmms/main.c 2003-09-08 13:49:16.000000000 +0200
+@@ -476,6 +476,8 @@
+ }
+ }
+ xmms_cfg_read_string(cfgfile, "xmms", "generic_title_format", &cfg.gentitle_format);
++
++ xmms_cfg_read_boolean(cfgfile, "xmms", "dynamic_taste_detection", &cfg.dynamic_taste_detection);
+
+ xmms_cfg_free(cfgfile);
+ }
+@@ -674,6 +676,8 @@
+ g_free(str);
+ }
+ xmms_cfg_write_string(cfgfile, "xmms", "generic_title_format", cfg.gentitle_format);
++
++ xmms_cfg_write_boolean(cfgfile, "xmms", "dynamic_taste_detection", cfg.dynamic_taste_detection);
+
+ xmms_cfg_write_file(cfgfile, filename);
+ xmms_cfg_free(cfgfile);
+@@ -903,6 +907,7 @@
+ playlist_clear();
+ cleanup_plugins();
+ sm_cleanup();
++ dtd_persist();
+ gtk_exit(0);
+ }
+
+@@ -3130,6 +3135,7 @@
+ GDK_THREADS_LEAVE();
+ }
+
++ cfg.dynamic_taste_detection = TRUE;
+
+ GDK_THREADS_ENTER();
+ check_ctrlsocket();
+@@ -3608,6 +3614,10 @@
+ mainwin_timeout_tag = gtk_timeout_add(10, idle_func, NULL);
+ playlist_start_get_info_thread();
+
++
++ /* Initialize dynamic taste detection */
++ dtd_init();
++
+ enable_x11r5_session_management(argc, argv);
+ sm_init(argc, argv, options.previous_session_id);
+ gtk_main();
+diff -ruN dtd-old/xmms/main.h dtd-new/xmms/main.h
+--- dtd-old/xmms/main.h 2003-09-08 13:31:40.000000000 +0200
++++ dtd-new/xmms/main.h 2003-09-08 13:51:48.000000000 +0200
+@@ -64,6 +64,7 @@
+ gchar *charset_id3, *charset_output, *charset_fs, *charset_pl;
+ gint autocharset;
+ gchar *shade_font;
++ gboolean dynamic_taste_detection;
+ }
+ Config;
+
+diff -ruN dtd-old/xmms/playlist.c dtd-new/xmms/playlist.c
+--- dtd-old/xmms/playlist.c 2003-09-08 13:31:40.000000000 +0200
++++ dtd-new/xmms/playlist.c 2003-09-08 14:12:58.000000000 +0200
+@@ -23,6 +23,7 @@
+ #include "charset.h"
+ #include <sys/stat.h>
+ #include <unistd.h>
++#include <assert.h>
+
+ GList *playlist = NULL;
+ GList *shuffle_list = NULL;
+@@ -653,6 +654,17 @@
+
+ plist_pos_list = find_playlist_position_list();
+
++ if ((plist_pos_list != NULL) &&
++ (cfg.repeat || (g_list_next(plist_pos_list) != NULL)))
++ {
++ /*
++ The user is skipping a song. We interpret this as
++ meaning "I don't like this song".
++ */
++ dtd_change_score(((PlaylistEntry *)(plist_pos_list->data))->filename,
++ -1);
++ }
++
+ if (!cfg.repeat && !g_list_next(plist_pos_list))
+ {
+ PL_UNLOCK();
+@@ -725,7 +737,18 @@
+
+ plist_pos_list = find_playlist_position_list();
+ if (g_list_previous(plist_pos_list))
+- playlist_position = plist_pos_list->prev->data;
++ {
++ playlist_position =
++ g_list_previous(plist_pos_list)->data;
++
++ /*
++ The user has gone back to a song. We interpret that
++ as meaning "I like this song so much that I want to
++ hear it again".
++ */
++ dtd_change_score(playlist_position->filename,
++ 1);
++ }
+ else if (cfg.repeat)
+ {
+ GList *node;
+@@ -740,7 +763,7 @@
+ }
+ PL_UNLOCK();
+ playlist_check_pos_current();
+-
++
+ if (restart_playing)
+ playlist_play();
+ else
+@@ -856,6 +879,7 @@
+
+ playlist_position = node->data;
+ PL_UNLOCK();
++ ((PlaylistEntry *)playlist_position)->moved = FALSE;
+ playlist_check_pos_current();
+
+ if (restart_playing)
+@@ -875,12 +899,19 @@
+
+ void playlist_eof_reached(void)
+ {
++ /*
++ * This function is called whenever a song ends and the next
++ * one should start playing.
++ */
++
+ GList *plist_pos_list;
++ GList *previous_plist_pos;
+
+ input_stop();
+
+ PL_LOCK();
+ plist_pos_list = find_playlist_position_list();
++ previous_plist_pos = plist_pos_list->data;
+
+ if (cfg.no_playlist_advance)
+ {
+@@ -912,6 +943,25 @@
+ else
+ playlist_position = plist_pos_list->next->data;
+ PL_UNLOCK();
++
++ if (!cfg.shuffle && (((PlaylistEntry *)playlist_position)->moved))
++ {
++ /*
++ * Song A has stopped playing. Song B will start
++ * playing now. Song B had moved. Thus, we conclude
++ * that the user wants to hear Song B after song A.
++ * Therefore, Song B should be added to Song A's
++ * next-song-preferences.
++ *
++ * Song A is previous_plist_pos. Song B is
++ * playlist_position. */
++
++ dtd_recommend_next(((PlaylistEntry *)previous_plist_pos)->filename,
++ ((PlaylistEntry *)playlist_position)->filename);
++
++ ((PlaylistEntry *)playlist_position)->moved = FALSE;
++ }
++
+ playlist_check_pos_current();
+ playlist_play();
+ mainwin_set_info_text();
+@@ -1146,6 +1196,7 @@
+ ext = strrchr(filename, '.');
+ if (ext && !strcasecmp(ext, ".pls"))
+ {
++ /* It's a playlist, let's parse it. */
+ int noe, i;
+ char key[10];
+
+@@ -1283,6 +1334,11 @@
+ char *ret;
+ PlaylistEntry *entry;
+ GList *node;
++
++ if (pos < 0)
++ {
++ return NULL;
++ }
+
+ PL_LOCK();
+ if (!playlist)
+@@ -1290,6 +1346,11 @@
+ PL_UNLOCK();
+ return NULL;
+ }
++ if (pos >= __get_playlist_length())
++ {
++ PL_UNLOCK();
++ return NULL;
++ }
+ node = g_list_nth(playlist, pos);
+ if (!node)
+ {
+@@ -1355,6 +1416,84 @@
+ return title;
+ }
+
++int playlist_get_score(gint pos)
++{
++ PlaylistEntry *entry;
++ GList *node;
++ gint score;
++
++ PL_LOCK();
++ if (!playlist)
++ {
++ PL_UNLOCK();
++ return 0;
++ }
++ node = g_list_nth(playlist, pos);
++ if (!node)
++ {
++ PL_UNLOCK();
++ return 0;
++ }
++ entry = node->data;
++ score = dtd_get_score(entry->filename);
++ PL_UNLOCK();
++
++ return score;
++}
++
++void playlist_set_score(gint pos, gint score)
++{
++ PlaylistEntry *entry;
++ GList *node;
++
++ PL_LOCK();
++ if (!playlist)
++ {
++ PL_UNLOCK();
++ return;
++ }
++ node = g_list_nth(playlist, pos);
++ if (!node)
++ {
++ PL_UNLOCK();
++ return;
++ }
++ entry = node->data;
++ dtd_set_score(entry->filename, score);
++ PL_UNLOCK();
++}
++
++const gchar* playlist_filename2songtitle(const gchar *filename)
++{
++ GList *iterator;
++
++ if (filename == NULL)
++ {
++ return NULL;
++ }
++
++ // Scan loaded songs for filename
++ PL_LOCK();
++ for (iterator = get_playlist();
++ iterator != NULL;
++ iterator = g_list_next(iterator))
++ {
++ PlaylistEntry *currentEntry =
++ (PlaylistEntry *)(iterator->data);
++
++ if (strcmp(currentEntry->filename, filename) == 0)
++ {
++ // FIXME: Race condition
++ PL_UNLOCK();
++ return currentEntry->title;
++ }
++ }
++ PL_UNLOCK();
++
++ // Not found / don't know
++ return NULL;
++}
++
+ gint playlist_get_songtime(gint pos)
+ {
+ int retval = -1;
+@@ -1616,6 +1755,26 @@
+ PL_UNLOCK();
+ }
+
++static guint playlist_filename_to_index(PlaylistEntry *playlist_entries[],
++ guint n_entries,
++ const gchar *filename)
++{
++ guint i;
++
++ for (i = 0; i < n_entries; i++)
++ {
++ if (strcmp(playlist_entries[i]->filename,
++ filename) == 0)
++ {
++ return i;
++ }
++ }
++
++ return -1;
++}
++
++/* This function is used by the qsort() call in
++ smart_playlist_shuffle_list() (below). */
+ void playlist_sort_selected_by_date(void)
+ {
+ PL_LOCK();
+@@ -1630,7 +1789,20 @@
+ PL_UNLOCK();
+ }
+
+-static GList *playlist_shuffle_list(GList *list)
++static int playlist_entry_score_comparator(const void *a,
++ const void *b)
++{
++
++ int score_a = dtd_get_score((*((PlaylistEntry **)a))->filename);
++ int score_b = dtd_get_score((*((PlaylistEntry **)b))->filename);
++
++ /* We want high scores before low, so if score_a > score_b we
++ want to return something negative. */
++
++ return score_b - score_a;
++}
++
++static GList *smart_playlist_shuffle_list(GList *list)
+ {
+ /* Caller should hold playlist mutex */
+ /*
+@@ -1639,43 +1811,213 @@
+ * fuction is run.
+ */
+ int len = g_list_length(list);
+- int i, j;
+- GList *node, **ptrs;
++ gint i;
++ gint next_score_section = -1;
++ GList *iterator;
++ PlaylistEntry **ptrs;
+
+- if (!len)
++ if (len == 0)
+ return NULL;
+
+- ptrs = g_new(GList *, len);
++ ptrs = g_new(PlaylistEntry *, len);
+
+- for (node = list, i = 0; i < len; node = g_list_next(node), i++)
+- ptrs[i] = node;
++ /* Convert the list into an array of pointers */
++ for (iterator = list, i = 0; i < len; iterator = g_list_next(iterator), i++)
++ {
++ ptrs[i] = (PlaylistEntry *)(iterator->data);
++
++ // Shuffling songs voids information about user listening preferences
++ ptrs[i]->moved = FALSE;
++ }
++ g_list_free(list);
++ list = NULL;
++
++ /* Sort the array by score */
++ qsort(ptrs, len, sizeof(PlaylistEntry *), playlist_entry_score_comparator);
++
++ /* Shuffle the pointer array */
++ for (i = 0; i < len; i++)
++ {
++ PlaylistEntry *swap_ptr;
++ gint j;
++
++ const gchar *previous_filename;
++ const gchar *new_filename;
++
++ if (i >= next_score_section)
++ {
++ /* We are in a new score section; find out
++ where the next one starts */
++ gint current_score =
++ dtd_get_score(ptrs[i]->filename);
++
++ for (next_score_section = i;
++ next_score_section < len;
++ next_score_section++)
++ {
++ if (dtd_get_score(ptrs[next_score_section]->filename) !=
++ current_score)
++ {
++ break;
++ }
++ }
++ }
++
++ if (i == 0)
++ {
++ previous_filename = NULL;
++ }
++ else
++ {
++ previous_filename = ptrs[i - 1]->filename;
++ }
++
++ /* Find out from which index we should get the next song */
++ do {
++ if ((new_filename = dtd_get_recommendation(previous_filename)) !=
++ NULL)
++ {
++ j = playlist_filename_to_index(&(ptrs[i]),
++ len - i,
++ new_filename);
++ if (j != -1)
++ j += i;
++ }
++ else
++ {
++ /* Only find new songs in our current
++ score section */
++ j = (rand() % (next_score_section - i)) + i;
++
++ assert(j < next_score_section);
++ }
++ } while (j == -1);
++
++ assert(j < len);
++ assert(j >= i);
++
++ if (j < next_score_section)
++ {
++ /* Swap pointer #i and pointer #j */
++ swap_ptr = ptrs[i];
++ ptrs[i] = ptrs[j];
++ ptrs[j] = swap_ptr;
++ }
++ else
++ {
++ /*
++ We need to preserve the score section
++ ordering, so we do it the slow way if i and
++ j are in different scoring sections.
++ */
++ gint k;
++ swap_ptr = ptrs[i];
++ ptrs[i] = ptrs[j];
++
++ for (k = j; k >= (i + 2); k--)
++ {
++ ptrs[k] = ptrs[k - 1];
++ }
++
++ ptrs[i + 1] = swap_ptr;
++ }
++ }
++
++ /* Create a new list from the pointer array */
++ for (i = 0; i < len; i++)
++ {
++ list = g_list_append(list, ptrs[i]);
++ }
+
+- j = random() % len;
+- list = ptrs[j];
+- ptrs[j]->next = NULL;
+- ptrs[j] = ptrs[0];
++ g_free(ptrs);
+
+- for (i = 1; i < len; i++)
+- {
+- j = random() % (len - i);
+- list->prev = ptrs[i + j];
+- ptrs[i + j]->next = list;
+- list = ptrs[i + j];
+- ptrs[i + j] = ptrs[i];
+- }
+- list->prev = NULL;
++ return list;
++}
++
++static GList *old_playlist_shuffle_list(GList *list)
++{
++ /* Caller should hold playlist mutex */
++ /*
++ * Note that this doesn't make a copy of the original list.
++ * The pointer to the original list is not valid after this
++ * fuction is run.
++ */
++ int len = g_list_length(list);
++ gint i;
++ GList *iterator;
++ PlaylistEntry **ptrs;
++
++ if (len == 0)
++ return NULL;
++
++ ptrs = g_new(PlaylistEntry *, len);
++
++ /* Convert the list into an array of pointers */
++ for (iterator = list, i = 0; i < len; iterator = g_list_next(iterator), i++)
++ {
++ ptrs[i] = (PlaylistEntry *)(iterator->data);
++
++ // Shuffling songs voids information about user listening preferences
++ ptrs[i]->moved = FALSE;
++ }
++ g_list_free(list);
++ list = NULL;
++
++ /* Shuffle the pointer array */
++ for (i = 0; i < len; i++)
++ {
++ PlaylistEntry *swap_ptr;
++ gint j;
++
++ /* Pick a random song among the ones that are left */
++ j = (rand() % (len - i)) + i;
++
++ /* Swap pointer #i and pointer #j */
++ swap_ptr = ptrs[i];
++ ptrs[i] = ptrs[j];
++ ptrs[j] = swap_ptr;
++ }
++
++ /* Create a new list from the pointer array */
++ for (i = 0; i < len; i++)
++ {
++ list = g_list_append(list, ptrs[i]);
++ }
+
+ g_free(ptrs);
+
+ return list;
+ }
+
++static GList *playlist_shuffle_list(GList *list)
++{
++ if (cfg.dynamic_taste_detection)
++ {
++ return smart_playlist_shuffle_list(list);
++ }
++ else
++ {
++ return old_playlist_shuffle_list(list);
++ }
++}
++
+ void playlist_random(void)
+ {
++ GList *for_each;
++
+ PL_LOCK();
+
+ playlist = playlist_shuffle_list(playlist);
+
++ /* Remove all moved-marks from the playlist */
++
++ for (for_each = playlist;
++ for_each != NULL;
++ for_each = g_list_next(for_each))
++ {
++ ((PlaylistEntry *) for_each->data)->moved = FALSE;
++ }
++
+ PL_UNLOCK();
+ }
+
+@@ -1710,7 +2052,52 @@
+ PL_UNLOCK();
+ return num;
+ }
+-
++
++gint playlist_get_single_selection(void)
++{
++ // Returns the playlist position of the current selection. If
++ // 0 or > 1 songs are selected, this method returns -1.
++ GList *iterator;
++ gint currentPos;
++
++ gint selectedPos = -1;
++
++ for (iterator = get_playlist(), currentPos = 0;
++ iterator != NULL;
++ iterator = g_list_next(iterator), currentPos++)
++ {
++ if (((PlaylistEntry *)(iterator->data))->selected)
++ {
++ if (selectedPos == -1)
++ {
++ // First selected song found
++ selectedPos = currentPos;
++ }
++ else
++ {
++ // > 1 selected song found
++ return -1;
++ }
++ }
++ }
++
++ return selectedPos;
++}
++
++gboolean playlist_is_selected(gint pos)
++{
++ GList *node;
++ gboolean isSelected = 0;
++
++ PL_LOCK();
++ if ((node = g_list_nth(get_playlist(), pos)) != NULL)
++ {
++ isSelected = ((PlaylistEntry *)(node->data))->selected;
++ }
++ PL_UNLOCK();
++
++ return isSelected;
++}
+
+ static void playlist_generate_shuffle_list(void)
+ {
+diff -ruN dtd-old/xmms/playlist.h dtd-new/xmms/playlist.h
+--- dtd-old/xmms/playlist.h 2003-09-08 13:31:40.000000000 +0200
++++ dtd-new/xmms/playlist.h 2003-09-08 14:14:13.000000000 +0200
+@@ -26,7 +26,8 @@
+ gchar *title;
+ gchar *sort;
+ gint length;
+- gboolean selected;
++ gboolean selected;
++ gboolean moved;
+ }
+ PlaylistEntry;
+
+@@ -81,10 +82,15 @@
+ void playlist_delete_filenames(GList *filenames);
+ gchar* playlist_get_filename(gint pos);
+ gchar* playlist_get_songtitle(gint pos);
++gint playlist_get_score(gint pos);
++void playlist_set_score(gint pos, gint score);
++const gchar* playlist_filename2songtitle(const gchar *filename);
+ gint playlist_get_songtime(gint pos);
+ GList * playlist_get_selected(void);
+ GList * playlist_get_selected_list(void);
+ int playlist_get_num_selected(void);
++gboolean playlist_is_selected(gint pos);
++gint playlist_get_single_selection(void);
+ void playlist_get_total_time(gulong *total_time, gulong *selection_time, gboolean *total_more, gboolean *selection_more);
+ void playlist_select_all(gboolean set);
+ void playlist_select_range(int min, int max, gboolean sel);
+diff -ruN dtd-old/xmms/playlist_list.c dtd-new/xmms/playlist_list.c
+--- dtd-old/xmms/playlist_list.c 2003-09-08 13:31:40.000000000 +0200
++++ dtd-new/xmms/playlist_list.c 2003-09-08 13:46:19.000000000 +0200
+@@ -24,6 +24,8 @@
+ #endif
+ #include <X11/Xatom.h>
+
++static gint playlist_move_delta = 0;
++static PlaylistEntry *playlist_last_moved_song = NULL;
+ static GdkFont *playlist_list_font = NULL;
+
+ static int playlist_list_auto_drag_down_func(gpointer data)
+@@ -61,24 +63,35 @@
+ GList *list;
+
+ PL_LOCK();
++
++ /* If the first song is selected... */
+ if ((list = get_playlist()) == NULL)
+ {
+ PL_UNLOCK();
+ return;
+ }
++
+ if (((PlaylistEntry *) list->data)->selected)
+ {
+- /* We are at the top */
++ /* ... don't move. */
+ PL_UNLOCK();
+ return;
+ }
+- while (list)
++
++ while (list)
+ {
+- if (((PlaylistEntry *) list->data)->selected)
++ if (((PlaylistEntry *) list->data)->selected)
++ {
++ playlist_last_moved_song =
++ (PlaylistEntry *) list->data;
+ glist_moveup(list);
++ }
+ list = g_list_next(list);
+ }
+ PL_UNLOCK();
++
++ playlist_move_delta--;
++
+ if (pl->pl_prev_selected != -1)
+ pl->pl_prev_selected--;
+ if (pl->pl_prev_min != -1)
+@@ -92,24 +105,36 @@
+ GList *list;
+
+ PL_LOCK();
++
++ /* If the last song is selected... */
+ if ((list = g_list_last(get_playlist())) == NULL)
+ {
+ PL_UNLOCK();
+ return;
+ }
++
+ if (((PlaylistEntry *) list->data)->selected)
+ {
+- /* We are at the bottom */
++ /* ... don't move. */
+ PL_UNLOCK();
+ return;
+ }
++
+ while (list)
+ {
+ if (((PlaylistEntry *) list->data)->selected)
++ {
++ playlist_last_moved_song =
++ (PlaylistEntry *) list->data;
+ glist_movedown(list);
++ }
+ list = g_list_previous(list);
+ }
++
+ PL_UNLOCK();
++
++ playlist_move_delta++;
++
+ if (pl->pl_prev_selected != -1)
+ pl->pl_prev_selected++;
+ if (pl->pl_prev_min != -1)
+@@ -177,6 +202,7 @@
+ playlist_play();
+ }
+ pl->pl_dragging = TRUE;
++ playlist_move_delta = 0;
+ playlistwin_update_list();
+ }
+ }
+@@ -244,11 +270,36 @@
+ }
+ }
+
++int playlist_n_selected_songs(void)
++{
++ int n_selected = 0;
++ GList *list;
++
++ for (list = get_playlist();
++ list != NULL;
++ list = g_list_next(list))
++ {
++ if (((PlaylistEntry *) list->data)->selected)
++ n_selected++;
++ }
++
++ return n_selected;
++}
++
+ void playlist_list_button_release_cb(GtkWidget * widget, GdkEventButton * event, PlayList_List * pl)
+ {
+ pl->pl_dragging = FALSE;
+ pl->pl_auto_drag_down = FALSE;
+ pl->pl_auto_drag_up = FALSE;
++
++ if ((playlist_move_delta < 0) && (playlist_n_selected_songs() == 1))
++ {
++ /* Exactly one song has been moved upwards, mark it as moved. */
++ playlist_last_moved_song->moved = TRUE;
++ }
++
++ playlist_move_delta = 0;
++ playlist_last_moved_song = NULL;
+ }
+
+ #ifdef HAVE_WCHAR_H
+diff -ruN dtd-old/xmms/playlistwin.c dtd-new/xmms/playlistwin.c
+--- dtd-old/xmms/playlistwin.c 2003-09-08 13:31:40.000000000 +0200
++++ dtd-new/xmms/playlistwin.c 2003-09-08 14:16:39.000000000 +0200
+@@ -17,6 +17,7 @@
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+ */
++#include <assert.h>
+ #include "xmms.h"
+ #include "libxmms/dirbrowser.h"
+ #include "libxmms/util.h"
+@@ -24,7 +25,7 @@
+ GtkWidget *playlistwin;
+ static GtkWidget *playlistwin_url_window = NULL;
+ static GtkItemFactory *playlistwin_sort_menu, *playlistwin_sub_menu;
+-static GtkItemFactory *playlistwin_popup_menu, *playlistwin_save_menu;
++static GtkItemFactory *playlistwin_popup_menu, *playlistwin_save_menu, *playlistwin_dtd_menu;
+
+ static GdkPixmap *playlistwin_bg;
+ static GdkBitmap *playlistwin_mask;
+@@ -68,13 +69,15 @@
+ SEL_INV, SEL_ZERO, SEL_ALL,
+ MISC_SORT, MISC_FILEINFO, MISC_MISCOPTS,
+ PLIST_NEW, PLIST_SAVE, PLIST_LOAD,
+- SEL_LOOKUP,
++ SEL_LOOKUP, DTD_GOODSONG, DTD_BADSONG,
++ DTD_DISSOCIATE
+ };
+
+ void playlistwin_sort_menu_callback(gpointer cb_data, guint action, GtkWidget * w);
+ void playlistwin_sub_menu_callback(gpointer cb_data, guint action, GtkWidget * w);
+ void playlistwin_save_type_cb(gpointer cb_data, guint action, GtkWidget * w);
+ void playlistwin_set_hints(void);
++void playlistwin_dissociate_callback(GtkMenuItem *menuitem, gpointer ignored);
+
+ enum
+ {
+@@ -151,7 +154,11 @@
+ GtkItemFactoryEntry playlistwin_popup_menu_entries[] =
+ {
+ {N_("/View File Info"), NULL, playlistwin_popup_menu_callback, MISC_FILEINFO, "<Item>"},
++
+ {N_("/-"), NULL, NULL, 0, "<Separator>"},
++
++ {N_("/Dynamic Taste Detection"), NULL, NULL, 0, "<Item>"},
++
+ {N_("/Add"), NULL, NULL, 0, "<Branch>"},
+ {N_("/Add/File"), NULL, playlistwin_popup_menu_callback, ADD_FILE, "<Item>"},
+ {N_("/Add/Directory"), NULL, playlistwin_popup_menu_callback, ADD_DIR, "<Item>"},
+@@ -182,6 +189,21 @@
+ sizeof(playlistwin_popup_menu_entries) /
+ sizeof(playlistwin_popup_menu_entries[0]);
+
++GtkItemFactoryEntry playlistwin_dtd_menu_entries[] =
++{
++ {N_("/I Like this Song"),
++ NULL, playlistwin_popup_menu_callback, DTD_GOODSONG,
++ "<RadioItem>"},
++ {N_("/I Don't Like this Song"),
++ NULL, playlistwin_popup_menu_callback, DTD_BADSONG,
++ /* <RadioItem> */ "/I Like this Song"},
++ {N_("/Dissociate From"), NULL, NULL, 0, "<Item>"}
++};
++
++static const int playlistwin_dtd_menu_entries_num =
++ sizeof(playlistwin_dtd_menu_entries) /
++ sizeof(playlistwin_dtd_menu_entries[0]);
++
+ void playlistwin_draw_frame(void);
+
+ static void playlistwin_update_info(void)
+@@ -1030,6 +1052,35 @@
+ case PLIST_LOAD:
+ playlistwin_show_load_filesel();
+ break;
++
++ /* DTD button */
++ case DTD_GOODSONG: {
++ gint score;
++ gint selected_playlist_pos = playlist_get_single_selection();
++
++ assert(selected_playlist_pos > -1);
++ score = playlist_get_score(selected_playlist_pos);
++ if (score < 0)
++ {
++ playlist_set_score(selected_playlist_pos, 0);
++ }
++
++ break;
++ }
++
++ case DTD_BADSONG: {
++ gint score;
++ gint selected_playlist_pos = playlist_get_single_selection();
++
++ assert(selected_playlist_pos > -1);
++ score = playlist_get_score(selected_playlist_pos);
++ if (score >= 0)
++ {
++ playlist_set_score(selected_playlist_pos, -1);
++ }
++
++ break;
++ }
+ }
+ }
+
+@@ -1053,6 +1104,133 @@
+ inside_widget(x, y, playlistwin_sscroll_down));
+ }
+
++gchar *playlistwin_fileName2songName(const gchar *fileName)
++{
++ const gchar *afterLastSlash, *beforeLastDot, *c;
++ gchar *returnMe;
++ gint i;
++ const gchar *loadedTitle =
++ playlist_filename2songtitle(fileName);
++
++ if (loadedTitle != NULL)
++ {
++ return g_strdup(loadedTitle);
++ }
++
++ // Ditch the last slash and everything before it, as well as
++ // the last dot and everything after it before returning it.
++ afterLastSlash = strrchr(fileName, '/');
++ if (afterLastSlash != NULL)
++ {
++ afterLastSlash++;
++ }
++ else
++ {
++ afterLastSlash = fileName;
++ }
++
++ beforeLastDot = strrchr(fileName, '.');
++ if ((beforeLastDot != NULL) && (beforeLastDot > afterLastSlash))
++ {
++ beforeLastDot--;
++ }
++ else
++ {
++ beforeLastDot = fileName + strlen(fileName) - 1;
++ }
++
++ if ((afterLastSlash >= beforeLastDot) ||
++ (beforeLastDot <= fileName) ||
++ ((afterLastSlash - fileName) >= strlen(fileName)))
++ {
++ return g_strdup(fileName);
++ }
++
++ // Copy from after last slash to before last dot
++ returnMe = g_new(gchar, (beforeLastDot - afterLastSlash) + 1);
++ i = 0;
++ for (c = afterLastSlash; c <= beforeLastDot; c++)
++ {
++ returnMe[i++] = *c;
++ }
++ returnMe[i] = '\0';
++
++ return returnMe;
++}
++
++void playlistwin_remove_dissociation_menu(GtkMenuItem *attachPoint)
++{
++ // Nuke any previous dissociations menu
++
++ // FIXME: I have absolutely no clue to how GTK+ refcounting
++ // works. This implementation may or may not leak memory.
++ // Candidates for memory leaks are the user_data fields of the
++ // labels (they should be freed).
++ // /Johan Walles - jan 05 / 2002
++ gtk_menu_item_remove_submenu(attachPoint);
++}
++
++void playlistwin_setup_dissociation_menu(GtkMenuItem *attachPoint,
++ gint selected_playlist_pos)
++{
++ const GList *recommendations = NULL;
++ const gchar *filename = playlist_get_filename(selected_playlist_pos);
++
++ // Ditch the previous dissociations menu (if any)
++ playlistwin_remove_dissociation_menu(attachPoint);
++
++ recommendations =
++ dtd_get_recommendations(filename);
++
++ if (recommendations != NULL)
++ {
++ // Create a new menu
++ GtkWidget *dissociateMenu = gtk_menu_new();
++ const GList *iterator;
++
++ // Associate it with the selected song
++ gtk_object_set_user_data(GTK_OBJECT(dissociateMenu), g_strdup(filename));
++
++ // For all recommendations...
++ for (iterator = recommendations;
++ iterator != NULL;
++ iterator = g_list_next(iterator))
++ {
++ const gchar *filename = ((DtdNextSong *)(iterator->data))->filename;
++ gchar *songName = playlistwin_fileName2songName(filename);
++ GtkWidget *newMenuItem;
++
++ // ... create a new menu entry...
++ newMenuItem = gtk_menu_item_new_with_label(songName);
++ g_free(songName);
++
++ // ... with its user_data set to point
++ // to the associated file name...
++ gtk_object_set_user_data(GTK_OBJECT(newMenuItem), g_strdup(filename));
++
++ // ... tell it what to do on receiving a click...
++ gtk_signal_connect(GTK_OBJECT(newMenuItem),
++ "activate",
++ GTK_SIGNAL_FUNC(playlistwin_dissociate_callback),
++ NULL);
++
++ // ... and add it to the new menu.
++ gtk_menu_append(GTK_MENU(dissociateMenu), newMenuItem);
++ }
++
++ // Enable our new dissociations menu
++ gtk_widget_show_all(dissociateMenu);
++ gtk_menu_item_set_submenu(attachPoint, dissociateMenu);
++ gtk_widget_set_sensitive(GTK_WIDGET(attachPoint), 1);
++ }
++ else
++ {
++ // ... or disable the dissociation menu heading
++ // if there is nothing to dissociate from.
++ gtk_widget_set_sensitive(GTK_WIDGET(attachPoint), 0);
++ }
++}
++
+ #define REGION_L(x1,x2,y1,y2) \
+ (event->x >= (x1) && event->x < (x2) && \
+ event->y >= cfg.playlist_height - (y1) && \
+@@ -1227,18 +1405,101 @@
+ else if (event->button == 3 &&
+ inside_widget(event->x, event->y, playlistwin_list))
+ {
+- int pos, sensitive;
++ // FIXME: This block is much too long, put it in its
++ // own function
++
++ gint clicked_playlist_pos, selected_playlist_pos;
++ gboolean sensitive;
++
+ GtkWidget *w;
+- pos = playlist_list_get_playlist_position(playlistwin_list,
+- event->x, event->y);
+- sensitive = pos != -1;
++
++ clicked_playlist_pos =
++ playlist_list_get_playlist_position(playlistwin_list,
++ event->x, event->y);
++
++ // Unless the clicked song is part of a (multiple)
++ // selection...
++ if (!playlist_is_selected(clicked_playlist_pos))
++ {
++ // ... select just the current song before
++ // displaying the menu
++ playlist_select_all(0);
++ playlist_select_range(clicked_playlist_pos, clicked_playlist_pos, 1);
++ playlistwin_update_list();
++ }
++
++ selected_playlist_pos = playlist_get_single_selection();
++ sensitive = (selected_playlist_pos != -1);
++
++ // Disable "View File Info" if not exactly one song is selected
+ w = gtk_item_factory_get_widget(playlistwin_popup_menu,
+ "/View File Info");
+ gtk_widget_set_sensitive(w, sensitive);
++
++ // Disable the DTD menu hierarchy if not exactly one
++ // song is selected
++ // FIXME: or if it's disabled in the prefs
++ w = gtk_item_factory_get_widget(playlistwin_popup_menu,
++ "/Dynamic Taste Detection");
++ gtk_widget_set_sensitive(w, sensitive);
++
++ if (sensitive)
++ {
++ GtkWidget *playlistwin_dtd_menu_widget =
++ gtk_item_factory_get_widget(playlistwin_dtd_menu, "<Main>");
++
++ gtk_widget_show_all(playlistwin_dtd_menu_widget);
++ gtk_menu_item_set_submenu(GTK_MENU_ITEM(w), playlistwin_dtd_menu_widget);
++ }
++ else
++ {
++ gtk_menu_item_remove_submenu(GTK_MENU_ITEM(w));
++ gtk_widget_hide_all(gtk_item_factory_get_widget(playlistwin_dtd_menu, "<Main>"));
++ }
++
++ if (selected_playlist_pos != -1)
++ {
++ // Enable either "/I Like this Song" (if the
++ // song scores >= 0) or "Bad Song" (if the
++ // song scores < 0).
++ gint score = dtd_get_score(playlist_get_filename(selected_playlist_pos));
++
++ w = gtk_item_factory_get_widget(playlistwin_dtd_menu,
++ "/I Like this Song");
++ if (score >= 0)
++ {
++ gtk_check_menu_item_set_state(GTK_CHECK_MENU_ITEM(w), 1);
++ }
++ gtk_widget_set_sensitive(w, 1);
++ w = gtk_item_factory_get_widget(playlistwin_dtd_menu,
++ "/I Don't Like this Song");
++ if (score < 0)
++ {
++ gtk_check_menu_item_set_state(GTK_CHECK_MENU_ITEM(w), 1);
++ }
++ gtk_widget_set_sensitive(w, 1);
++ }
++ else
++ {
++ // We don't do ratings of multiple songs at once
++ w = gtk_item_factory_get_widget(playlistwin_dtd_menu,
++ "/I Like this Song");
++ gtk_widget_set_sensitive(w, 0);
++ gtk_check_menu_item_set_state(GTK_CHECK_MENU_ITEM(w), 0);
++ w = gtk_item_factory_get_widget(playlistwin_dtd_menu,
++ "/I Don't Like this Song");
++ gtk_widget_set_sensitive(w, 0);
++ gtk_check_menu_item_set_state(GTK_CHECK_MENU_ITEM(w), 0);
++ }
+
++ // Create the dissociation menu
++ w = gtk_item_factory_get_widget(playlistwin_dtd_menu,
++ "/Dissociate From");
++ playlistwin_setup_dissociation_menu(GTK_MENU_ITEM(w), selected_playlist_pos);
++
+ playlistwin_set_sensitive_sortmenu();
+ util_item_factory_popup_with_data(playlistwin_popup_menu,
+- GINT_TO_POINTER(pos), NULL,
++ GINT_TO_POINTER(selected_playlist_pos), NULL,
+ event->x_root,
+ event->y_root + 5,
+ 3, event->time);
+@@ -1946,6 +2207,11 @@
+ cfg.playlist_height,
+ gdk_rgb_get_visual()->depth);
+
++ playlistwin_dtd_menu = gtk_item_factory_new(GTK_TYPE_MENU, "<Main>", NULL);
++ gtk_item_factory_create_items(GTK_ITEM_FACTORY(playlistwin_dtd_menu),
++ playlistwin_dtd_menu_entries_num,
++ playlistwin_dtd_menu_entries, NULL);
++
+ playlistwin_popup_menu =
+ gtk_item_factory_new(GTK_TYPE_MENU, "<Main>", NULL);
+ gtk_item_factory_set_translate_func(playlistwin_popup_menu,
+@@ -2022,6 +2288,23 @@
+ tbutton_set_toggled(mainwin_pl, FALSE);
+ }
+
++void playlistwin_dissociate_callback(GtkMenuItem *menuitem,
++ gpointer ignored)
++{
++ gchar *first, *second;
++
++ // Find out "first" by checking the user data of the menuitem
++ // that was activated
++ first = (gchar *)gtk_object_get_user_data(GTK_OBJECT(menuitem));
++
++ // Find out "second" by checking the user data of the menu
++ // that is the parent of the menuitem that was activated
++ second = (gchar *)gtk_object_get_user_data(GTK_OBJECT(GTK_WIDGET(menuitem)->parent));
++
++ dtd_dissociate(first, second);
++}
++
++
+ void playlistwin_popup_menu_callback(gpointer cb_data, guint action, GtkWidget * w)
+ {
+ int pos = GPOINTER_TO_INT(gtk_item_factory_popup_data_from_widget(w));
+@@ -2034,6 +2317,17 @@
+ case SEL_LOOKUP:
+ playlist_read_info_selection();
+ break;
++ case DTD_DISSOCIATE:
++
++ break;
++ case DTD_GOODSONG:
++ case DTD_BADSONG:
++ // Act only upon the active item
++ if (GTK_CHECK_MENU_ITEM(w)->active)
++ {
++ playlistwin_popup_handler(action);
++ }
++ break;
+ default:
+ playlistwin_popup_handler(action);
+ }
+diff -ruN dtd-old/xmms/prefswin.c dtd-new/xmms/prefswin.c
+--- dtd-old/xmms/prefswin.c 2003-09-08 13:31:40.000000000 +0200
++++ dtd-new/xmms/prefswin.c 2003-09-08 13:46:19.000000000 +0200
+@@ -1042,6 +1042,8 @@
+
+ gtk_notebook_append_page(GTK_NOTEBOOK(prefswin_notebook), prefswin_options_vbox, gtk_label_new(_("Options")));
+
++ prefswin_option_new_with_label_to_table(&cfg.dynamic_taste_detection, _("Dynamic taste detection"), GTK_TABLE(options_table), 1, 10);
++
+ /*
+ * Fonts page
+ */
+diff -ruN dtd-old/xmms/xmms.h dtd-new/xmms/xmms.h
+--- dtd-old/xmms/xmms.h 2001-03-14 15:06:39.000000000 +0100
++++ dtd-new/xmms/xmms.h 2003-09-08 13:46:19.000000000 +0200
+@@ -81,6 +81,7 @@
+ #include "sm.h"
+ #include "dnd.h"
+ #include "urldecode.h"
++#include "dtd.h"
+
+ #include "config.h"
+
diff --git a/media-sound/xmms/files/xmms-1.2.8-jump.patch b/media-sound/xmms/files/xmms-1.2.8-jump.patch
new file mode 100644
index 000000000000..5ebd59b3d08a
--- /dev/null
+++ b/media-sound/xmms/files/xmms-1.2.8-jump.patch
@@ -0,0 +1,76 @@
+diff -Naur xmms-1.2.8/libxmms/xmmsctrl.c xmms-1.2.8-new/libxmms/xmmsctrl.c
+--- xmms-1.2.8/libxmms/xmmsctrl.c 2003-09-04 15:48:34.000000000 +0200
++++ xmms-1.2.8-new/libxmms/xmmsctrl.c 2003-09-06 18:18:53.000000000 +0200
+@@ -381,6 +381,11 @@
+ remote_cmd(session, CMD_PLAY_PAUSE);
+ }
+
++void xmms_remote_show_jump_box(gint session)
++{
++ remote_cmd(session, CMD_SHOW_JUMP_BOX);
++}
++
+ gboolean xmms_remote_is_playing(gint session)
+ {
+ return remote_get_gboolean(session, CMD_IS_PLAYING);
+diff -Naur xmms-1.2.8/libxmms/xmmsctrl.h xmms-1.2.8-new/libxmms/xmmsctrl.h
+--- xmms-1.2.8/libxmms/xmmsctrl.h 2003-06-09 15:22:10.000000000 +0200
++++ xmms-1.2.8-new/libxmms/xmmsctrl.h 2003-09-06 18:18:53.000000000 +0200
+@@ -61,6 +61,7 @@
+ gboolean xmms_remote_is_pl_win(gint session);
+ gboolean xmms_remote_is_eq_win(gint session);
+ void xmms_remote_show_prefs_box(gint session);
++void xmms_remote_show_jump_box(gint session);
+ void xmms_remote_toggle_aot(gint session, gboolean ontop);
+ void xmms_remote_eject(gint session);
+ void xmms_remote_playlist_prev(gint session);
+diff -Naur xmms-1.2.8/xmms/controlsocket.c xmms-1.2.8-new/xmms/controlsocket.c
+--- xmms-1.2.8/xmms/controlsocket.c 2003-09-04 15:48:34.000000000 +0200
++++ xmms-1.2.8-new/xmms/controlsocket.c 2003-09-06 18:18:53.000000000 +0200
+@@ -506,6 +506,9 @@
+ case CMD_SET_SKIN:
+ load_skin(data);
+ break;
++ case CMD_SHOW_JUMP_BOX:
++ show_jump_to_file();
++ break;
+ case CMD_PL_WIN_TOGGLE:
+ tbool = *((gboolean *) data);
+ playlistwin_show(!!tbool);
+diff -Naur xmms-1.2.8/xmms/controlsocket.h xmms-1.2.8-new/xmms/controlsocket.h
+--- xmms-1.2.8/xmms/controlsocket.h 2003-06-09 15:22:10.000000000 +0200
++++ xmms-1.2.8-new/xmms/controlsocket.h 2003-09-06 18:18:53.000000000 +0200
+@@ -46,6 +46,7 @@
+ CMD_GET_EQ, CMD_GET_EQ_PREAMP, CMD_GET_EQ_BAND,
+ CMD_SET_EQ, CMD_SET_EQ_PREAMP, CMD_SET_EQ_BAND,
+ CMD_QUIT, CMD_PLAYLIST_INS_URL_STRING, CMD_PLAYLIST_INS, CMD_PLAY_PAUSE,
++ CMD_SHOW_JUMP_BOX
+ };
+
+ typedef struct
+diff -Naur xmms-1.2.8/xmms/main.c xmms-1.2.8-new/xmms/main.c
+--- xmms-1.2.8/xmms/main.c 2003-09-02 15:01:40.000000000 +0200
++++ xmms-1.2.8-new/xmms/main.c 2003-09-06 18:18:53.000000000 +0200
+@@ -1847,6 +1847,11 @@
+ GTK_CLIST(clist)->focus_row = GPOINTER_TO_INT(GTK_CLIST(clist)->selection->data);
+ }
+
++void show_jump_to_file(void)
++{
++ mainwin_jump_to_file();
++}
++
+ static gboolean mainwin_configure(GtkWidget * window, GdkEventConfigure *event, gpointer data)
+ {
+ if (!GTK_WIDGET_VISIBLE(window))
+diff -Naur xmms-1.2.8/xmms/main.h xmms-1.2.8-new/xmms/main.h
+--- xmms-1.2.8/xmms/main.h 2002-10-06 18:35:27.000000000 +0200
++++ xmms-1.2.8-new/xmms/main.h 2003-09-06 18:18:53.000000000 +0200
+@@ -117,6 +117,7 @@
+ void mainwin_set_shade_menu_cb(gboolean shaded);
+ void mainwin_set_shade(gboolean shaded);
+ void mainwin_shade_toggle(void);
++void show_jump_to_file(void);
+
+ #define PLAYER_HEIGHT ((cfg.player_shaded ? 14 : 116) * (cfg.doublesize + 1))
+ #define PLAYER_WIDTH (275 * (cfg.doublesize + 1))
diff --git a/media-sound/xmms/files/xmms-1.2.8-russian-charset.patch b/media-sound/xmms/files/xmms-1.2.8-russian-charset.patch
new file mode 100644
index 000000000000..57ffb8612f89
--- /dev/null
+++ b/media-sound/xmms/files/xmms-1.2.8-russian-charset.patch
@@ -0,0 +1,2174 @@
+diff -Naur xmms-1.2.8/Input/mpg123/fileinfo.c xmms-1.2.8-new/Input/mpg123/fileinfo.c
+--- xmms-1.2.8/Input/mpg123/fileinfo.c 2003-07-14 15:24:28.000000000 +0200
++++ xmms-1.2.8-new/Input/mpg123/fileinfo.c 2003-09-06 19:42:29.000000000 +0200
+@@ -26,6 +26,10 @@
+ #include <libxmms/xentry.h>
+ #include "mpg123.h"
+
++#include "libxmms/charset.h"
++#include <gdk/gdkkeysyms.h>
++#include "xmms/xmms.h"
++
+ static GtkWidget *window = NULL;
+ static GtkWidget *filename_entry, *id3_frame;
+ static GtkWidget *title_entry, *artist_entry, *album_entry, *year_entry, *tracknum_entry, *comment_entry;
+@@ -46,20 +50,35 @@
+ {
+ gint stripped_len;
+ gchar *text;
++ gchar *ctag;
++ gint clength;
+
++ ctag=xmms_charset_recode_id3(tag,length,&clength);
++ if (ctag) {
++ tag=ctag;
++ length=clength;
++ }
++
+ stripped_len = mpg123_strip_spaces(tag, length);
+ text = g_strdup_printf("%-*.*s", stripped_len, stripped_len, tag);
+ gtk_entry_set_text(entry, text);
+ g_free(text);
++ if (ctag) g_free(ctag);
+ }
+
+ static void get_entry_tag(GtkEntry * entry, gchar * tag, gint length)
+ {
+ gchar *text;
++ gchar *ctext;
+
+ text = gtk_entry_get_text(entry);
++ ctext = xmms_charset_recode_output(text,length,NULL);
++ if (ctext) text=ctext;
++
+ memset(tag, ' ', length);
+ memcpy(tag, text, strlen(text) > length ? length : strlen(text));
++
++ if (ctext) g_free(ctext);
+ }
+
+ static gint find_genre_id(gchar * text)
+@@ -225,6 +244,31 @@
+ }
+ }
+
++static int restore_focus;
++
++gboolean mpg123_keypress_cb(GtkWidget * w, GdkEventKey * event, gpointer save) {
++ switch(event->keyval) {
++ case GDK_Return:
++ restore_focus=1;
++ gtk_signal_emit_by_name(GTK_OBJECT(save), "clicked", NULL);
++ break;
++ case GDK_Escape:
++ restore_focus=1;
++ break;
++ }
++ return TRUE;
++}
++
++void gtk_widget_destroyed_focus(GtkWidget *widget, GtkWidget **widget_pointer) {
++ gtk_widget_destroyed(widget,widget_pointer);
++ if (restore_focus) {
++ gtk_widget_hide(playlistwin);
++ gtk_widget_show(playlistwin);
++// playlistwin_real_hide();
++// playlistwin_real_show();
++ }
++}
++
+ void mpg123_file_info_box(char *filename)
+ {
+ gint i;
+@@ -233,7 +277,10 @@
+ gchar *tmp, *title;
+ const gchar *emphasis[4];
+ const gchar *bool_label[2];
++ gchar *cfilename;
+
++ restore_focus=0;
++
+ emphasis[0] = _("None");
+ emphasis[1] = _("50/15 ms");
+ emphasis[2] = "";
+@@ -247,10 +294,11 @@
+ GtkWidget *mpeg_frame, *mpeg_box;
+ GtkWidget *label, *filename_hbox;
+ GtkWidget *bbox, *save, *remove_id3, *cancel;
++ GtkAccelGroup *ag;
+
+ window = gtk_window_new(GTK_WINDOW_DIALOG);
+ gtk_window_set_policy(GTK_WINDOW(window), FALSE, FALSE, FALSE);
+- gtk_signal_connect(GTK_OBJECT(window), "destroy", GTK_SIGNAL_FUNC(gtk_widget_destroyed), &window);
++ gtk_signal_connect(GTK_OBJECT(window), "destroy", GTK_SIGNAL_FUNC(gtk_widget_destroyed_focus), &window);
+ gtk_container_set_border_width(GTK_CONTAINER(window), 10);
+
+ vbox = gtk_vbox_new(FALSE, 10);
+@@ -392,6 +440,18 @@
+ gtk_label_set_justify(GTK_LABEL(mpeg_fileinfo), GTK_JUSTIFY_LEFT);
+ gtk_box_pack_start(GTK_BOX(mpeg_box), mpeg_fileinfo, FALSE, FALSE, 0);
+
++ ag = gtk_accel_group_new();
++ gtk_accel_group_add(ag, GDK_Escape, 0, 0, GTK_OBJECT(cancel), "clicked");
++ gtk_accel_group_add(ag, GDK_Return, 0, 0, GTK_OBJECT(save), "clicked");
++ gtk_window_add_accel_group(GTK_WINDOW(window), ag);
++/* gtk_signal_connect(GTK_OBJECT(title_entry), "key_press_event", GTK_SIGNAL_FUNC(mpg123_keypress_cb), save);
++ gtk_signal_connect(GTK_OBJECT(artist_entry), "key_press_event", GTK_SIGNAL_FUNC(mpg123_keypress_cb), save);
++ gtk_signal_connect(GTK_OBJECT(album_entry), "key_press_event", GTK_SIGNAL_FUNC(mpg123_keypress_cb), save);
++ gtk_signal_connect(GTK_OBJECT(comment_entry), "key_press_event", GTK_SIGNAL_FUNC(mpg123_keypress_cb), save);
++ gtk_signal_connect(GTK_OBJECT(year_entry), "key_press_event", GTK_SIGNAL_FUNC(mpg123_keypress_cb), save);
++ gtk_signal_connect(GTK_OBJECT(tracknum_entry), "key_press_event", GTK_SIGNAL_FUNC(mpg123_keypress_cb), save);*/
++ gtk_signal_connect(GTK_OBJECT(window),"key_press_event",GTK_SIGNAL_FUNC(mpg123_keypress_cb),save);
++
+ gtk_widget_show_all(window);
+ }
+
+@@ -399,6 +459,9 @@
+ g_free(current_filename);
+ current_filename = g_strdup(filename);
+
++ cfilename=xmms_charset_recode_fsout(filename,0,NULL);
++ if (cfilename) filename=cfilename;
++
+ title = g_strdup_printf(_("File Info - %s"), g_basename(filename));
+ gtk_window_set_title(GTK_WINDOW(window), title);
+ g_free(title);
+@@ -412,6 +475,8 @@
+ gtk_entry_set_text(GTK_ENTRY(title_entry), title);
+ g_free(title);
+
++ if (cfilename) g_free(cfilename);
++
+ gtk_entry_set_text(GTK_ENTRY(artist_entry), "");
+ gtk_entry_set_text(GTK_ENTRY(album_entry), "");
+ gtk_entry_set_text(GTK_ENTRY(year_entry), "");
+diff -Naur xmms-1.2.8/Input/mpg123/mpg123.c xmms-1.2.8-new/Input/mpg123/mpg123.c
+--- xmms-1.2.8/Input/mpg123/mpg123.c 2003-07-10 15:12:19.000000000 +0200
++++ xmms-1.2.8-new/Input/mpg123/mpg123.c 2003-09-06 19:42:29.000000000 +0200
+@@ -551,6 +551,7 @@
+ {
+ gchar *ret = NULL, *path, *temp;
+ TitleInput *input;
++ gint len;
+
+ XMMS_NEW_TITLEINPUT(input);
+
+@@ -582,9 +583,18 @@
+ /*
+ * Format according to filename.
+ */
++ ret = xmms_charset_recode_fs(g_basename(filename),0,&len);
++ if (ret) {
++ for (;len>=0;len--)
++ if (ret[len]=='.') {
++ ret[len]=0;
++ break;
++ }
++ } else {
+ ret = g_strdup(g_basename(filename));
+ if (extname(ret) != NULL)
+ *(extname(ret) - 1) = '\0'; /* removes period */
++ }
+ }
+
+ return ret;
+diff -Naur xmms-1.2.8/Input/vorbis/vorbis.c xmms-1.2.8-new/Input/vorbis/vorbis.c
+--- xmms-1.2.8/Input/vorbis/vorbis.c 2003-07-14 15:24:28.000000000 +0200
++++ xmms-1.2.8-new/Input/vorbis/vorbis.c 2003-09-06 19:42:29.000000000 +0200
+@@ -47,6 +47,7 @@
+ #include "libxmms/util.h"
+ #include "libxmms/configfile.h"
+ #include "libxmms/titlestring.h"
++#include "libxmms/charset.h"
+ #include <xmms/i18n.h>
+
+ #include "vorbis.h"
+@@ -668,6 +669,7 @@
+ gchar *displaytitle = NULL, *tmp, *path, *temp;
+ vorbis_comment *comment;
+ TitleInput *input;
++ gint len;
+
+ XMMS_NEW_TITLEINPUT(input);
+ path = g_strdup(fn);
+@@ -705,10 +707,20 @@
+
+ if (!displaytitle)
+ {
+- if (!vorbis_is_streaming)
+- displaytitle = g_strdup_printf("%s", g_basename(fn));
+- else
++ if (!vorbis_is_streaming) {
++ displaytitle = xmms_charset_recode_fs(g_basename(fn),0,&len);
++ if (displaytitle) {
++ for (;len>=0;len--)
++ if (displaytitle[len]=='.') {
++ displaytitle[len]=0;
++ break;
++ }
++ } else {
++ displaytitle = g_strdup_printf("%s", g_basename(fn));
++ }
++ } else {
+ displaytitle = vorbis_http_get_title(fn);
++ }
+ }
+
+ return displaytitle;
+diff -Naur xmms-1.2.8/libxmms/charset_auto_russian.h xmms-1.2.8-new/libxmms/charset_auto_russian.h
+--- xmms-1.2.8/libxmms/charset_auto_russian.h 1970-01-01 01:00:00.000000000 +0100
++++ xmms-1.2.8-new/libxmms/charset_auto_russian.h 2003-09-06 19:42:29.000000000 +0200
+@@ -0,0 +1,293 @@
++/* Stealed from GPPL's (http://gppl.terminal.ru) autorecode-xmms1.2.7 patch,
++ changed to support limited size strings */
++
++typedef struct lng_stat {
++ unsigned char a;
++ unsigned char b;
++ int rate;
++} lng_stat;
++
++//statistics calculated from "Moskva-Petushki"
++//1292 elems
++static const lng_stat win[]={
++{'À','À',1}, {'À','Â',1}, {'À','Ç',1}, {'À','È',1}, {'À','Ê',1}, {'À','Ë',1}, {'À','Ì',8}, {'À','Í',2}, {'À','Ï',2}, {'À','Ð',1}, {'À','Ñ',4}, {'À','Ò',2}, {'À','Ô',1}, {'À','Õ',1}, {'À','×',1}, {'À','Ø',1},
++{'À','Ý',1}, {'À','à',1}, {'À','á',6}, {'À','â',60}, {'À','ã',5}, {'À','ä',7}, {'À','å',23}, {'À','æ',2}, {'À','ç',6}, {'À','è',4}, {'À','é',1}, {'À','ê',27}, {'À','ë',15}, {'À','ì',10}, {'À','í',25}, {'À','î',21},
++{'À','ï',51}, {'À','ð',12}, {'À','ñ',13}, {'À','ò',31}, {'À','ó',15}, {'À','ô',1}, {'À','õ',6}, {'À','÷',17}, {'À','ý',8}, {'À','ÿ',20}, {'Á','Â',1}, {'Á','Ä',1}, {'Á','Í',2}, {'Á','Ò',1}, {'Á','à',3}, {'Á','â',2},
++{'Á','å',16}, {'Á','è',1}, {'Á','ë',4}, {'Á','í',3}, {'Á','î',25}, {'Á','ð',10}, {'Á','ó',9}, {'Á','û',8}, {'Â','À',1}, {'Â','Ê',2}, {'Â','Ì',1}, {'Â','Î',3}, {'Â','Ï',8}, {'Â','Ñ',2}, {'Â','Õ',1}, {'Â','Ø',2},
++{'Â','à',15}, {'Â','â',2}, {'Â','ã',3}, {'Â','ä',5}, {'Â','å',98}, {'Â','ç',5}, {'Â','è',7}, {'Â','ê',5}, {'Â','ë',8}, {'Â','ì',7}, {'Â','í',6}, {'Â','î',107}, {'Â','ï',18}, {'Â','ð',2}, {'Â','ñ',74}, {'Â','ò',6},
++{'Â','ó',1}, {'Â','÷',3}, {'Â','û',39}, {'Â','ý',8}, {'Ã','Î',2}, {'Ã','à',6}, {'Ã','ä',16}, {'Ã','å',26}, {'Ã','è',4}, {'Ã','ë',3}, {'Ã','ì',1}, {'Ã','î',31}, {'Ã','ð',7}, {'Ã','ó',2}, {'Ä','à',167}, {'Ä','â',6},
++{'Ä','å',17}, {'Ä','è',3}, {'Ä','ë',7}, {'Ä','ì',3}, {'Ä','í',1}, {'Ä','î',20}, {'Ä','ð',9}, {'Ä','ó',19}, {'Å','Ã',2}, {'Å','Õ',1}, {'Å','Ø',1}, {'Å','â',9}, {'Å','ä',1}, {'Å','å',1}, {'Å','é',1}, {'Å','ë',5},
++{'Å','ì',2}, {'Å','ð',19}, {'Å','ñ',69}, {'Å','ô',1}, {'Å','õ',1}, {'Å','ù',5}, {'Æ','à',5}, {'Æ','ä',1}, {'Æ','å',14}, {'Æ','è',5}, {'Æ','î',2}, {'Æ','ð',1}, {'Ç','È',1}, {'Ç','à',31}, {'Ç','â',3}, {'Ç','å',1},
++{'Ç','è',2}, {'Ç','í',23}, {'Ç','ð',2}, {'Ç','ó',12}, {'È','À',1}, {'È','Â',1}, {'È','Ë',2}, {'È','Ì',2}, {'È','Ñ',2}, {'È','à',2}, {'È','á',8}, {'È','â',78}, {'È','ã',7}, {'È','ä',20}, {'È','å',7}, {'È','æ',1},
++{'È','ç',17}, {'È','è',3}, {'È','ê',27}, {'È','ë',15}, {'È','ì',11}, {'È','í',21}, {'È','î',26}, {'È','ï',21}, {'È','ð',4}, {'È','ñ',21}, {'È','ò',37}, {'È','ó',5}, {'È','ô',2}, {'È','õ',3}, {'È','÷',8}, {'È','ø',1},
++{'È','ý',2}, {'È','ÿ',12}, {'Ê','Ï',3}, {'Ê','à',100}, {'Ê','è',4}, {'Ê','ê',1}, {'Ê','ë',14}, {'Ê','í',4}, {'Ê','î',62}, {'Ê','ï',2}, {'Ê','ð',24}, {'Ê','ò',10}, {'Ê','ó',62}, {'Ë','À',1}, {'Ë','È',1}, {'Ë','Î',2},
++{'Ë','à',7}, {'Ë','å',11}, {'Ë','è',4}, {'Ë','î',12}, {'Ë','ó',15}, {'Ë','þ',3}, {'Ë','ÿ',1}, {'Ì','À',1}, {'Ì','È',1}, {'Ì','Î',1}, {'Ì','Ï',1}, {'Ì','à',31}, {'Ì','â',3}, {'Ì','ã',1}, {'Ì','å',10}, {'Ì','è',44},
++{'Ì','ë',1}, {'Ì','í',19}, {'Ì','î',93}, {'Ì','ð',1}, {'Ì','ó',6}, {'Ì','û',29}, {'Í','À',2}, {'Í','Ç',1}, {'Í','È',3}, {'Í','à',56}, {'Í','å',111}, {'Í','è',47}, {'Í','í',2}, {'Í','î',98}, {'Í','ó',100}, {'Î','Â',1},
++{'Î','Å',2}, {'Î','Í',6}, {'Î','Ð',1}, {'Î','Ò',1}, {'Î','Ó',1}, {'Î','ß',1}, {'Î','á',18}, {'Î','â',1}, {'Î','ã',4}, {'Î','ä',20}, {'Î','ç',2}, {'Î','è',2}, {'Î','ê',7}, {'Î','ë',9}, {'Î','ì',2}, {'Î','í',96},
++{'Î','î',5}, {'Î','ï',17}, {'Î','ð',16}, {'Î','ñ',21}, {'Î','ò',35}, {'Î','÷',13}, {'Î','ù',1}, {'Î','ý',6}, {'Ï','Ñ',3}, {'Ï','à',18}, {'Ï','å',148}, {'Ï','è',11}, {'Ï','ë',9}, {'Ï','î',186}, {'Ï','ð',43}, {'Ï','ó',38},
++{'Ï','ü',4}, {'Ð','Ë',1}, {'Ð','à',22}, {'Ð','å',13}, {'Ð','è',5}, {'Ð','î',13}, {'Ð','ð',1}, {'Ð','ó',1}, {'Ñ','À',2}, {'Ñ','Â',1}, {'Ñ','Ñ',3}, {'Ñ','à',19}, {'Ñ','á',1}, {'Ñ','â',4}, {'Ñ','ä',1}, {'Ñ','å',56},
++{'Ñ','è',16}, {'Ñ','ê',22}, {'Ñ','ë',11}, {'Ñ','ì',12}, {'Ñ','í',6}, {'Ñ','î',26}, {'Ñ','ï',6}, {'Ñ','ð',6}, {'Ñ','ñ',3}, {'Ñ','ò',35}, {'Ñ','ó',4}, {'Ñ','õ',2}, {'Ñ','÷',2}, {'Ñ','û',1}, {'Ñ','ý',1}, {'Ñ','ÿ',1},
++{'Ò','À',4}, {'Ò','Å',1}, {'Ò','È',1}, {'Ò','Î',1}, {'Ò','à',79}, {'Ò','â',1}, {'Ò','å',18}, {'Ò','è',17}, {'Ò','î',34}, {'Ò','ð',11}, {'Ò','ó',31}, {'Ò','û',55}, {'Ò','ü',2}, {'Ò','ÿ',3}, {'Ó','á',2}, {'Ó','â',3},
++{'Ó','ã',1}, {'Ó','ä',3}, {'Ó','å',2}, {'Ó','æ',4}, {'Ó','ç',4}, {'Ó','è',1}, {'Ó','é',2}, {'Ó','ê',1}, {'Ó','ë',2}, {'Ó','ì',9}, {'Ó','í',5}, {'Ó','î',1}, {'Ó','ï',1}, {'Ó','ñ',9}, {'Ó','ò',8}, {'Ó','õ',2},
++{'Ó','÷',1}, {'Ó','ø',1}, {'Ô','À',1}, {'Ô','à',7}, {'Ô','å',2}, {'Ô','î',1}, {'Ô','ð',16}, {'Ô','ô',2}, {'Õ','Ô',1}, {'Õ','à',8}, {'Õ','â',1}, {'Õ','å',3}, {'Õ','î',20}, {'Õ','ð',14}, {'Ö','à',7}, {'Ö','è',3},
++{'Ö','ñ',1}, {'Ö','ò',1}, {'Ö','û',1}, {'×','å',46}, {'×','ò',60}, {'×','ó',8}, {'×','ü',1}, {'Ø','È',1}, {'Ø','à',3}, {'Ø','å',7}, {'Ø','è',8}, {'Ø','ò',8}, {'Ø','ó',7}, {'Ø','ø',1}, {'Ý','Ò',1}, {'Ý','á',5},
++{'Ý','â',1}, {'Ý','é',1}, {'Ý','ë',4}, {'Ý','í',1}, {'Ý','ð',10}, {'Ý','ñ',1}, {'Ý','ò',61}, {'Ý','õ',1}, {'Þ','ç',2}, {'Þ','ð',1}, {'ß','Í',1}, {'ß','á',4}, {'ß','â',49}, {'ß','ã',5}, {'ß','ä',8}, {'ß','å',9},
++{'ß','æ',2}, {'ß','ç',12}, {'ß','è',4}, {'ß','ê',10}, {'ß','ë',5}, {'ß','ì',6}, {'ß','í',28}, {'ß','î',24}, {'ß','ï',39}, {'ß','ð',4}, {'ß','ñ',30}, {'ß','ò',7}, {'ß','ó',5}, {'ß','õ',3}, {'ß','÷',4}, {'ß','ø',3},
++{'ß','ý',3}, {'à','À',56}, {'à','Á',18}, {'à','Â',48}, {'à','Ã',9}, {'à','Ä',26}, {'à','Å',12}, {'à','Æ',7}, {'à','Ç',11}, {'à','È',53}, {'à','Ê',59}, {'à','Ë',9}, {'à','Ì',21}, {'à','Í',32}, {'à','Î',27}, {'à','Ï',52},
++{'à','Ð',7}, {'à','Ñ',21}, {'à','Ò',42}, {'à','Ó',10}, {'à','Ô',1}, {'à','Õ',4}, {'à','×',16}, {'à','Ø',2}, {'à','Ý',7}, {'à','ß',22}, {'à','à',69}, {'à','á',257}, {'à','â',701}, {'à','ã',218}, {'à','ä',548}, {'à','å',328},
++{'à','æ',262}, {'à','ç',626}, {'à','è',225}, {'à','é',194}, {'à','ê',1285}, {'à','ë',1288}, {'à','ì',771}, {'à','í',911}, {'à','î',166}, {'à','ï',421}, {'à','ð',368}, {'à','ñ',802}, {'à','ò',874}, {'à','ó',98}, {'à','ô',38}, {'à','õ',228},
++{'à','ö',24}, {'à','÷',300}, {'à','ø',121}, {'à','ù',27}, {'à','ý',63}, {'à','þ',238}, {'à','ÿ',384}, {'á','À',1}, {'á','Ï',1}, {'á','Ô',1}, {'á','à',199}, {'á','á',6}, {'á','â',13}, {'á','ä',6}, {'á','å',420}, {'á','ç',2},
++{'á','è',142}, {'á','ê',7}, {'á','ë',137}, {'á','ì',7}, {'á','í',31}, {'á','î',327}, {'á','ï',4}, {'á','ð',172}, {'á','ñ',18}, {'á','ò',1}, {'á','ó',288}, {'á','õ',11}, {'á','ö',3}, {'á','÷',2}, {'á','ø',1}, {'á','ù',32},
++{'á','û',561}, {'á','ü',10}, {'á','ý',7}, {'á','þ',1}, {'á','ÿ',134}, {'â','À',11}, {'â','Á',1}, {'â','Â',6}, {'â','Ã',7}, {'â','Ä',5}, {'â','Å',5}, {'â','È',6}, {'â','Ê',14}, {'â','Ë',2}, {'â','Ì',9}, {'â','Í',13},
++{'â','Î',17}, {'â','Ï',51}, {'â','Ð',7}, {'â','Ñ',18}, {'â','Ò',4}, {'â','Ó',2}, {'â','Õ',8}, {'â','Ö',1}, {'â','×',2}, {'â','Ø',7}, {'â','Ý',2}, {'â','ß',4}, {'â','à',989}, {'â','á',26}, {'â','â',62}, {'â','ã',43},
++{'â','ä',74}, {'â','å',911}, {'â','æ',26}, {'â','ç',88}, {'â','è',387}, {'â','é',1}, {'â','ê',105}, {'â','ë',122}, {'â','ì',65}, {'â','í',231}, {'â','î',1220}, {'â','ï',118}, {'â','ð',97}, {'â','ñ',851}, {'â','ò',123}, {'â','ó',116},
++{'â','ô',1}, {'â','õ',9}, {'â','ö',11}, {'â','÷',47}, {'â','ø',53}, {'â','ù',2}, {'â','û',518}, {'â','ü',33}, {'â','ý',44}, {'â','þ',3}, {'â','ÿ',48}, {'ã','À',3}, {'ã','Á',1}, {'ã','Â',3}, {'ã','Ä',1}, {'ã','Ç',3},
++{'ã','È',3}, {'ã','Ë',4}, {'ã','Í',1}, {'ã','Î',4}, {'ã','Ï',3}, {'ã','Ð',1}, {'ã','Ñ',2}, {'ã','Ò',2}, {'ã','×',1}, {'ã','Ø',1}, {'ã','ß',2}, {'ã','à',210}, {'ã','á',18}, {'ã','â',6}, {'ã','ã',3}, {'ã','ä',191},
++{'ã','å',83}, {'ã','ç',2}, {'ã','è',103}, {'ã','ê',20}, {'ã','ë',205}, {'ã','ì',3}, {'ã','í',62}, {'ã','î',1340}, {'ã','ï',7}, {'ã','ð',180}, {'ã','ñ',13}, {'ã','ò',4}, {'ã','ó',88}, {'ã','õ',1}, {'ã','÷',15}, {'ã','ý',1},
++{'ã','ÿ',4}, {'ä','À',1}, {'ä','Â',2}, {'ä','Ä',2}, {'ä','Ç',1}, {'ä','È',2}, {'ä','Ê',1}, {'ä','Ì',1}, {'ä','Í',2}, {'ä','Ï',2}, {'ä','Ò',1}, {'ä','Ó',1}, {'ä','Ô',1}, {'ä','×',2}, {'ä','ß',2}, {'ä','à',795},
++{'ä','á',7}, {'ä','â',165}, {'ä','ã',6}, {'ä','ä',14}, {'ä','å',839}, {'ä','æ',8}, {'ä','ç',7}, {'ä','è',487}, {'ä','é',1}, {'ä','ê',79}, {'ä','ë',78}, {'ä','ì',10}, {'ä','í',289}, {'ä','î',613}, {'ä','ï',17}, {'ä','ð',195},
++{'ä','ñ',62}, {'ä','ò',41}, {'ä','ó',417}, {'ä','ô',1}, {'ä','õ',11}, {'ä','ö',138}, {'ä','÷',6}, {'ä','û',111}, {'ä','ü',200}, {'ä','ý',6}, {'ä','ÿ',68}, {'å','À',30}, {'å','Á',8}, {'å','Â',33}, {'å','Ã',7}, {'å','Ä',34},
++{'å','Å',13}, {'å','Æ',1}, {'å','Ç',6}, {'å','È',21}, {'å','Ê',21}, {'å','Ë',4}, {'å','Ì',13}, {'å','Í',20}, {'å','Î',21}, {'å','Ï',31}, {'å','Ð',2}, {'å','Ñ',13}, {'å','Ò',10}, {'å','Ó',7}, {'å','Ô',4}, {'å','Õ',2},
++{'å','Ö',1}, {'å','×',9}, {'å','Ø',2}, {'å','Ý',11}, {'å','Þ',1}, {'å','ß',25}, {'å','à',44}, {'å','á',463}, {'å','â',704}, {'å','ã',592}, {'å','ä',613}, {'å','å',281}, {'å','æ',136}, {'å','ç',444}, {'å','è',177}, {'å','é',297},
++{'å','ê',398}, {'å','ë',926}, {'å','ì',982}, {'å','í',1512}, {'å','î',206}, {'å','ï',524}, {'å','ð',1231}, {'å','ñ',1082}, {'å','ò',1607}, {'å','ó',103}, {'å','ô',15}, {'å','õ',182}, {'å','ö',66}, {'å','÷',308}, {'å','ø',235}, {'å','ù',137},
++{'å','ý',79}, {'å','þ',24}, {'å','ÿ',87}, {'æ','Â',1}, {'æ','È',2}, {'æ','Í',1}, {'æ','Î',1}, {'æ','Ï',3}, {'æ','à',187}, {'æ','á',4}, {'æ','â',9}, {'æ','ã',1}, {'æ','ä',145}, {'æ','å',690}, {'æ','æ',4}, {'æ','ç',1},
++{'æ','è',214}, {'æ','ê',35}, {'æ','ë',1}, {'æ','ì',8}, {'æ','í',96}, {'æ','î',13}, {'æ','ï',14}, {'æ','ð',7}, {'æ','ñ',4}, {'æ','ò',8}, {'æ','ó',69}, {'æ','õ',1}, {'æ','÷',1}, {'æ','ü',6}, {'æ','ý',2}, {'æ','ÿ',7},
++{'ç','À',1}, {'ç','Â',1}, {'ç','Æ',1}, {'ç','È',2}, {'ç','Ê',1}, {'ç','Ì',4}, {'ç','Í',3}, {'ç','Î',1}, {'ç','Ï',8}, {'ç','Ñ',4}, {'ç','Ò',2}, {'ç','Ö',1}, {'ç','×',2}, {'ç','ß',2}, {'ç','à',889}, {'ç','á',39},
++{'ç','â',147}, {'ç','ã',67}, {'ç','ä',174}, {'ç','å',56}, {'ç','æ',18}, {'ç','ç',11}, {'ç','è',113}, {'ç','ê',16}, {'ç','ë',33}, {'ç','ì',64}, {'ç','í',302}, {'ç','î',109}, {'ç','ï',31}, {'ç','ð',53}, {'ç','ñ',21}, {'ç','ò',7},
++{'ç','ó',109}, {'ç','õ',3}, {'ç','÷',11}, {'ç','ø',4}, {'ç','û',71}, {'ç','ü',20}, {'ç','ý',4}, {'ç','ÿ',43}, {'è','À',39}, {'è','Á',10}, {'è','Â',46}, {'è','Ã',5}, {'è','Ä',25}, {'è','Å',11}, {'è','Æ',2}, {'è','Ç',1},
++{'è','È',30}, {'è','Ê',24}, {'è','Ë',5}, {'è','Ì',36}, {'è','Í',39}, {'è','Î',25}, {'è','Ï',49}, {'è','Ð',3}, {'è','Ñ',22}, {'è','Ò',25}, {'è','Ó',8}, {'è','Ô',3}, {'è','Õ',4}, {'è','Ö',2}, {'è','×',4}, {'è','Ø',3},
++{'è','Ý',7}, {'è','ß',28}, {'è','à',90}, {'è','á',242}, {'è','â',658}, {'è','ã',153}, {'è','ä',435}, {'è','å',280}, {'è','æ',87}, {'è','ç',388}, {'è','è',301}, {'è','é',216}, {'è','ê',492}, {'è','ë',888}, {'è','ì',573}, {'è','í',958},
++{'è','î',216}, {'è','ï',335}, {'è','ð',275}, {'è','ñ',668}, {'è','ò',949}, {'è','ó',108}, {'è','ô',17}, {'è','õ',324}, {'è','ö',123}, {'è','÷',363}, {'è','ø',134}, {'è','ù',36}, {'è','ý',39}, {'è','þ',22}, {'è','ÿ',234}, {'é','À',20},
++{'é','Á',6}, {'é','Â',20}, {'é','Ã',6}, {'é','Ä',16}, {'é','Å',6}, {'é','Ç',2}, {'é','È',22}, {'é','Ê',18}, {'é','Ë',2}, {'é','Ì',12}, {'é','Í',13}, {'é','Î',15}, {'é','Ï',13}, {'é','Ð',7}, {'é','Ñ',9}, {'é','Ò',15},
++{'é','Ó',3}, {'é','Ô',1}, {'é','Õ',1}, {'é','×',1}, {'é','Ý',7}, {'é','ß',8}, {'é','à',21}, {'é','á',36}, {'é','â',112}, {'é','ã',26}, {'é','ä',100}, {'é','å',15}, {'é','æ',21}, {'é','ç',30}, {'é','è',88}, {'é','é',1},
++{'é','ê',87}, {'é','ë',25}, {'é','ì',47}, {'é','í',117}, {'é','î',55}, {'é','ï',133}, {'é','ð',42}, {'é','ñ',162}, {'é','ò',131}, {'é','ó',17}, {'é','ô',6}, {'é','õ',7}, {'é','ö',11}, {'é','÷',69}, {'é','ø',20}, {'é','ù',2},
++{'é','ý',15}, {'é','ÿ',16}, {'ê','À',11}, {'ê','Â',13}, {'ê','Ã',7}, {'ê','Ä',4}, {'ê','Å',2}, {'ê','Æ',1}, {'ê','Ç',5}, {'ê','È',8}, {'ê','Ê',9}, {'ê','Ë',2}, {'ê','Ì',5}, {'ê','Í',20}, {'ê','Î',7}, {'ê','Ï',9},
++{'ê','Ñ',7}, {'ê','Ò',9}, {'ê','Ó',4}, {'ê','Õ',1}, {'ê','Ö',1}, {'ê','×',4}, {'ê','Ý',2}, {'ê','ß',8}, {'ê','à',1358}, {'ê','á',49}, {'ê','â',147}, {'ê','ã',15}, {'ê','ä',48}, {'ê','å',93}, {'ê','æ',22}, {'ê','ç',55},
++{'ê','è',678}, {'ê','ê',28}, {'ê','ë',81}, {'ê','ì',22}, {'ê','í',151}, {'ê','î',1337}, {'ê','ï',64}, {'ê','ð',273}, {'ê','ñ',131}, {'ê','ò',238}, {'ê','ó',384}, {'ê','ô',7}, {'ê','õ',10}, {'ê','ö',5}, {'ê','÷',31}, {'ê','ø',2},
++{'ê','û',2}, {'ê','ý',13}, {'ê','ÿ',25}, {'ë','À',20}, {'ë','Á',2}, {'ë','Â',25}, {'ë','Ã',5}, {'ë','Ä',14}, {'ë','Å',3}, {'ë','Æ',1}, {'ë','È',19}, {'ë','Ê',12}, {'ë','Ë',2}, {'ë','Ì',14}, {'ë','Í',31}, {'ë','Î',20},
++{'ë','Ï',20}, {'ë','Ð',1}, {'ë','Ñ',20}, {'ë','Ò',10}, {'ë','Ó',1}, {'ë','Ô',1}, {'ë','Õ',1}, {'ë','Ö',1}, {'ë','×',2}, {'ë','Ý',4}, {'ë','ß',19}, {'ë','à',758}, {'ë','á',34}, {'ë','â',102}, {'ë','ã',35}, {'ë','ä',55},
++{'ë','å',675}, {'ë','æ',33}, {'ë','ç',23}, {'ë','è',1300}, {'ë','é',1}, {'ë','ê',114}, {'ë','ë',21}, {'ë','ì',45}, {'ë','í',111}, {'ë','î',947}, {'ë','ï',83}, {'ë','ð',18}, {'ë','ñ',320}, {'ë','ò',61}, {'ë','ó',264}, {'ë','ô',5},
++{'ë','õ',7}, {'ë','ö',2}, {'ë','÷',66}, {'ë','ø',1}, {'ë','û',146}, {'ë','ü',642}, {'ë','ý',21}, {'ë','þ',159}, {'ë','ÿ',299}, {'ì','À',11}, {'ì','Á',2}, {'ì','Â',14}, {'ì','Ã',8}, {'ì','Ä',10}, {'ì','Å',1}, {'ì','Ç',5},
++{'ì','È',12}, {'ì','Ê',10}, {'ì','Ì',6}, {'ì','Í',23}, {'ì','Î',12}, {'ì','Ï',12}, {'ì','Ð',1}, {'ì','Ñ',9}, {'ì','Ò',16}, {'ì','Ó',2}, {'ì','Õ',2}, {'ì','×',3}, {'ì','Ø',1}, {'ì','Ý',1}, {'ì','ß',10}, {'ì','à',447},
++{'ì','á',69}, {'ì','â',119}, {'ì','ã',33}, {'ì','ä',41}, {'ì','å',838}, {'ì','æ',27}, {'ì','ç',38}, {'ì','è',484}, {'ì','é',2}, {'ì','ê',71}, {'ì','ë',74}, {'ì','ì',96}, {'ì','í',497}, {'ì','î',761}, {'ì','ï',132}, {'ì','ð',41},
++{'ì','ñ',137}, {'ì','ò',74}, {'ì','ó',411}, {'ì','ô',5}, {'ì','õ',13}, {'ì','ö',4}, {'ì','÷',46}, {'ì','ø',3}, {'ì','ù',3}, {'ì','û',207}, {'ì','ü',33}, {'ì','ý',17}, {'ì','ÿ',82}, {'í','À',11}, {'í','Â',5}, {'í','Ã',8},
++{'í','Ä',3}, {'í','Å',4}, {'í','È',7}, {'í','Ê',10}, {'í','Ë',1}, {'í','Ì',2}, {'í','Í',6}, {'í','Î',2}, {'í','Ï',9}, {'í','Ð',2}, {'í','Ñ',2}, {'í','Ò',6}, {'í','Ó',1}, {'í','Ö',1}, {'í','×',6}, {'í','Ø',1},
++{'í','Ý',2}, {'í','ß',2}, {'í','à',1706}, {'í','á',28}, {'í','â',55}, {'í','ã',48}, {'í','ä',97}, {'í','å',1870}, {'í','æ',5}, {'í','ç',20}, {'í','è',1319}, {'í','ê',84}, {'í','ë',7}, {'í','ì',11}, {'í','í',331}, {'í','î',1483},
++{'í','ï',54}, {'í','ð',27}, {'í','ñ',133}, {'í','ò',165}, {'í','ó',480}, {'í','ô',14}, {'í','õ',4}, {'í','ö',65}, {'í','÷',55}, {'í','ø',3}, {'í','ù',31}, {'í','û',385}, {'í','ü',196}, {'í','ý',7}, {'í','þ',30}, {'í','ÿ',396},
++{'î','À',34}, {'î','Á',5}, {'î','Â',44}, {'î','Ã',9}, {'î','Ä',16}, {'î','Å',16}, {'î','Æ',4}, {'î','Ç',17}, {'î','È',43}, {'î','Ê',17}, {'î','Ë',3}, {'î','Ì',23}, {'î','Í',49}, {'î','Î',23}, {'î','Ï',40}, {'î','Ð',6},
++{'î','Ñ',14}, {'î','Ò',19}, {'î','Ó',4}, {'î','Ô',7}, {'î','Õ',4}, {'î','Ö',2}, {'î','×',12}, {'î','Ý',5}, {'î','ß',27}, {'î','à',34}, {'î','á',652}, {'î','â',1590}, {'î','ã',915}, {'î','ä',946}, {'î','å',457}, {'î','æ',437},
++{'î','ç',279}, {'î','è',291}, {'î','é',602}, {'î','ê',498}, {'î','ë',923}, {'î','ì',1073}, {'î','í',1281}, {'î','î',241}, {'î','ï',588}, {'î','ð',921}, {'î','ñ',1264}, {'î','ò',1536}, {'î','ó',152}, {'î','ô',35}, {'î','õ',137}, {'î','ö',51},
++{'î','÷',459}, {'î','ø',223}, {'î','ù',56}, {'î','ý',89}, {'î','þ',81}, {'î','ÿ',154}, {'ï','Â',2}, {'ï','È',1}, {'ï','Ê',1}, {'ï','à',252}, {'ï','â',1}, {'ï','ä',1}, {'ï','å',364}, {'ï','è',356}, {'ï','ê',8}, {'ï','ë',189},
++{'ï','í',11}, {'ï','î',1543}, {'ï','ï',6}, {'ï','ð',845}, {'ï','ñ',6}, {'ï','ò',23}, {'ï','ó',202}, {'ï','÷',2}, {'ï','û',33}, {'ï','ü',122}, {'ï','ÿ',149}, {'ð','À',3}, {'ð','Á',4}, {'ð','Â',3}, {'ð','Ä',6}, {'ð','Å',2},
++{'ð','Ç',1}, {'ð','Ê',2}, {'ð','Ì',1}, {'ð','Í',5}, {'ð','Î',8}, {'ð','Ï',10}, {'ð','Ñ',2}, {'ð','Ò',4}, {'ð','Ô',2}, {'ð','Õ',1}, {'ð','×',2}, {'ð','Ý',3}, {'ð','ß',6}, {'ð','à',1264}, {'ð','á',29}, {'ð','â',98},
++{'ð','ã',53}, {'ð','ä',129}, {'ð','å',936}, {'ð','æ',41}, {'ð','ç',20}, {'ð','è',777}, {'ð','é',1}, {'ð','ê',54}, {'ð','ë',29}, {'ð','ì',39}, {'ð','í',181}, {'ð','î',1126}, {'ð','ï',38}, {'ð','ð',34}, {'ð','ñ',76}, {'ð','ò',122},
++{'ð','ó',355}, {'ð','ô',8}, {'ð','õ',36}, {'ð','ö',23}, {'ð','÷',9}, {'ð','ø',49}, {'ð','ù',5}, {'ð','û',169}, {'ð','ü',98}, {'ð','ý',7}, {'ð','þ',59}, {'ð','ÿ',159}, {'ñ','À',6}, {'ñ','Â',4}, {'ñ','Ã',2}, {'ñ','Ä',2},
++{'ñ','Æ',1}, {'ñ','Ç',1}, {'ñ','È',4}, {'ñ','Ê',2}, {'ñ','Ë',3}, {'ñ','Ì',4}, {'ñ','Í',5}, {'ñ','Î',3}, {'ñ','Ï',3}, {'ñ','Ð',1}, {'ñ','Ñ',2}, {'ñ','Ò',1}, {'ñ','×',1}, {'ñ','Ý',3}, {'ñ','ß',1}, {'ñ','à',306},
++{'ñ','á',33}, {'ñ','â',259}, {'ñ','ã',18}, {'ñ','ä',41}, {'ñ','å',973}, {'ñ','æ',8}, {'ñ','ç',11}, {'ñ','è',295}, {'ñ','ê',691}, {'ñ','ë',519}, {'ñ','ì',222}, {'ñ','í',236}, {'ñ','î',403}, {'ñ','ï',274}, {'ñ','ð',58}, {'ñ','ñ',222},
++{'ñ','ò',1721}, {'ñ','ó',146}, {'ñ','ô',18}, {'ñ','õ',34}, {'ñ','ö',6}, {'ñ','÷',48}, {'ñ','ø',20}, {'ñ','ù',2}, {'ñ','û',79}, {'ñ','ü',401}, {'ñ','ý',10}, {'ñ','þ',27}, {'ñ','ÿ',680}, {'ò','À',29}, {'ò','Á',5}, {'ò','Â',25},
++{'ò','Ã',3}, {'ò','Ä',11}, {'ò','Å',9}, {'ò','Æ',2}, {'ò','Ç',5}, {'ò','È',16}, {'ò','Ê',21}, {'ò','Ì',20}, {'ò','Í',24}, {'ò','Î',10}, {'ò','Ï',23}, {'ò','Ð',2}, {'ò','Ñ',10}, {'ò','Ò',13}, {'ò','Ó',2}, {'ò','Ô',2},
++{'ò','Õ',3}, {'ò','×',11}, {'ò','Ø',1}, {'ò','Ý',3}, {'ò','Þ',1}, {'ò','ß',16}, {'ò','à',1225}, {'ò','á',56}, {'ò','â',436}, {'ò','ã',25}, {'ò','ä',67}, {'ò','å',864}, {'ò','æ',17}, {'ò','ç',33}, {'ò','è',758}, {'ò','é',1},
++{'ò','ê',180}, {'ò','ë',52}, {'ò','ì',65}, {'ò','í',329}, {'ò','î',2625}, {'ò','ï',119}, {'ò','ð',616}, {'ò','ñ',337}, {'ò','ò',117}, {'ò','ó',398}, {'ò','ô',1}, {'ò','õ',24}, {'ò','ö',10}, {'ò','÷',87}, {'ò','ø',3}, {'ò','ù',4},
++{'ò','û',398}, {'ò','ü',1292}, {'ò','ý',37}, {'ò','þ',13}, {'ò','ÿ',103}, {'ó','À',22}, {'ó','Á',4}, {'ó','Â',19}, {'ó','Ã',4}, {'ó','Ä',20}, {'ó','Å',3}, {'ó','Ç',2}, {'ó','È',25}, {'ó','Ê',9}, {'ó','Ë',6}, {'ó','Ì',12},
++{'ó','Í',18}, {'ó','Î',13}, {'ó','Ï',18}, {'ó','Ð',1}, {'ó','Ñ',18}, {'ó','Ò',12}, {'ó','Ó',2}, {'ó','Ô',2}, {'ó','Õ',5}, {'ó','×',6}, {'ó','Ø',3}, {'ó','Ý',5}, {'ó','ß',17}, {'ó','à',45}, {'ó','á',142}, {'ó','â',184},
++{'ó','ã',161}, {'ó','ä',445}, {'ó','å',83}, {'ó','æ',269}, {'ó','ç',75}, {'ó','è',111}, {'ó','é',40}, {'ó','ê',169}, {'ó','ë',251}, {'ó','ì',322}, {'ó','í',195}, {'ó','î',50}, {'ó','ï',258}, {'ó','ð',203}, {'ó','ñ',380}, {'ó','ò',442},
++{'ó','ó',24}, {'ó','ô',5}, {'ó','õ',97}, {'ó','÷',241}, {'ó','ø',302}, {'ó','ù',37}, {'ó','ý',19}, {'ó','þ',156}, {'ó','ÿ',31}, {'ô','Ï',1}, {'ô','Ö',2}, {'ô','à',21}, {'ô','å',45}, {'ô','è',47}, {'ô','ë',14}, {'ô','í',2},
++{'ô','î',24}, {'ô','ð',20}, {'ô','ñ',4}, {'ô','ò',5}, {'ô','ó',8}, {'ô','ô',6}, {'ô','û',4}, {'ô','ü',1}, {'ô','þ',2}, {'õ','À',7}, {'õ','Â',7}, {'õ','Ä',4}, {'õ','Å',3}, {'õ','Æ',3}, {'õ','È',5}, {'õ','Ê',4},
++{'õ','Ë',2}, {'õ','Ì',1}, {'õ','Í',10}, {'õ','Î',5}, {'õ','Ï',7}, {'õ','Ñ',4}, {'õ','Ò',5}, {'õ','×',3}, {'õ','Ø',5}, {'õ','Ý',2}, {'õ','ß',5}, {'õ','à',135}, {'õ','á',21}, {'õ','â',72}, {'õ','ã',27}, {'õ','ä',21},
++{'õ','å',25}, {'õ','æ',4}, {'õ','ç',13}, {'õ','è',75}, {'õ','ê',16}, {'õ','ë',43}, {'õ','ì',49}, {'õ','í',83}, {'õ','î',450}, {'õ','ï',70}, {'õ','ð',19}, {'õ','ñ',39}, {'õ','ò',27}, {'õ','ó',42}, {'õ','ô',3}, {'õ','õ',4},
++{'õ','ö',2}, {'õ','÷',18}, {'õ','ø',8}, {'õ','û',1}, {'õ','ü',6}, {'õ','ý',7}, {'õ','ÿ',9}, {'ö','À',1}, {'ö','Ã',1}, {'ö','È',2}, {'ö','Ê',1}, {'ö','Ë',1}, {'ö','Ì',1}, {'ö','Ï',1}, {'ö','Ò',1}, {'ö','ß',1},
++{'ö','à',179}, {'ö','á',1}, {'ö','â',11}, {'ö','å',162}, {'ö','ç',5}, {'ö','è',75}, {'ö','ê',6}, {'ö','ë',5}, {'ö','ì',1}, {'ö','í',2}, {'ö','î',29}, {'ö','ï',4}, {'ö','ð',2}, {'ö','ñ',4}, {'ö','ò',4}, {'ö','ó',41},
++{'ö','÷',2}, {'ö','û',41}, {'÷','Â',1}, {'÷','Ã',1}, {'÷','Å',1}, {'÷','È',1}, {'÷','Í',3}, {'÷','Î',1}, {'÷','Ñ',3}, {'÷','Ò',2}, {'÷','×',1}, {'÷','à',294}, {'÷','á',1}, {'÷','â',6}, {'÷','å',874}, {'÷','è',319},
++{'÷','ê',76}, {'÷','ë',9}, {'÷','ì',4}, {'÷','í',158}, {'÷','î',10}, {'÷','ï',3}, {'÷','ð',8}, {'÷','ñ',2}, {'÷','ò',536}, {'÷','ó',101}, {'÷','÷',7}, {'÷','ø',37}, {'÷','ü',49}, {'÷','ÿ',1}, {'ø','ß',1}, {'ø','à',150},
++{'ø','â',5}, {'ø','å',388}, {'ø','è',188}, {'ø','ê',201}, {'ø','ë',68}, {'ø','ì',4}, {'ø','í',78}, {'ø','î',53}, {'ø','ï',8}, {'ø','ð',3}, {'ø','ñ',2}, {'ø','ò',17}, {'ø','ó',45}, {'ø','õ',1}, {'ø','ö',1}, {'ø','ü',200},
++{'ø','ý',1}, {'ù','Ï',1}, {'ù','Ò',1}, {'ù','à',68}, {'ù','å',242}, {'ù','è',96}, {'ù','í',7}, {'ù','î',1}, {'ù','ð',1}, {'ù','ñ',3}, {'ù','ó',20}, {'ù','ü',2}, {'û','À',14}, {'û','Á',6}, {'û','Â',28}, {'û','Ã',4},
++{'û','Ä',8}, {'û','Å',6}, {'û','Æ',1}, {'û','Ç',4}, {'û','È',12}, {'û','Ê',10}, {'û','Ë',2}, {'û','Ì',15}, {'û','Í',16}, {'û','Î',13}, {'û','Ï',6}, {'û','Ð',3}, {'û','Ñ',7}, {'û','Ò',10}, {'û','Ó',1}, {'û','Ô',1},
++{'û','×',4}, {'û','Ý',5}, {'û','Þ',1}, {'û','ß',10}, {'û','à',10}, {'û','á',66}, {'û','â',248}, {'û','ã',37}, {'û','ä',65}, {'û','å',138}, {'û','æ',21}, {'û','ç',45}, {'û','è',74}, {'û','é',264}, {'û','ê',83}, {'û','ë',305},
++{'û','ì',164}, {'û','í',141}, {'û','î',34}, {'û','ï',231}, {'û','ð',61}, {'û','ñ',189}, {'û','ò',148}, {'û','ó',35}, {'û','ô',1}, {'û','õ',146}, {'û','ö',3}, {'û','÷',63}, {'û','ø',80}, {'û','ù',1}, {'û','ý',9}, {'û','ÿ',10},
++{'ü','À',44}, {'ü','Á',8}, {'ü','Â',45}, {'ü','Ã',7}, {'ü','Ä',18}, {'ü','Å',12}, {'ü','Æ',1}, {'ü','Ç',4}, {'ü','È',29}, {'ü','Ê',19}, {'ü','Ì',18}, {'ü','Í',36}, {'ü','Î',25}, {'ü','Ï',42}, {'ü','Ð',4}, {'ü','Ñ',25},
++{'ü','Ò',24}, {'ü','Ó',10}, {'ü','Ô',1}, {'ü','Õ',3}, {'ü','×',14}, {'ü','Ø',1}, {'ü','Ý',4}, {'ü','ß',21}, {'ü','à',38}, {'ü','á',63}, {'ü','â',196}, {'ü','ã',35}, {'ü','ä',110}, {'ü','å',154}, {'ü','æ',11}, {'ü','ç',58},
++{'ü','è',194}, {'ü','é',1}, {'ü','ê',324}, {'ü','ë',36}, {'ü','ì',124}, {'ü','í',321}, {'ü','î',92}, {'ü','ï',164}, {'ü','ð',41}, {'ü','ñ',302}, {'ü','ò',152}, {'ü','ó',48}, {'ü','ô',5}, {'ü','õ',20}, {'ü','ö',15}, {'ü','÷',88},
++{'ü','ø',85}, {'ü','ù',1}, {'ü','ý',27}, {'ü','þ',88}, {'ü','ÿ',110}, {'ý','È',1}, {'ý','á',1}, {'ý','ã',1}, {'ý','ä',2}, {'ý','ç',1}, {'ý','ê',6}, {'ý','ë',9}, {'ý','ì',6}, {'ý','í',3}, {'ý','ð',8}, {'ý','ñ',4},
++{'ý','ò',531}, {'ý','ô',2}, {'ý','õ',1}, {'ý','ý',2}, {'þ','À',10}, {'þ','Á',1}, {'þ','Â',10}, {'þ','Ã',1}, {'þ','Ä',3}, {'þ','Æ',1}, {'þ','Ç',2}, {'þ','È',17}, {'þ','Ê',4}, {'þ','Ë',2}, {'þ','Ì',3}, {'þ','Í',16},
++{'þ','Î',4}, {'þ','Ï',5}, {'þ','Ð',1}, {'þ','Ñ',3}, {'þ','Ò',2}, {'þ','Ó',2}, {'þ','Õ',3}, {'þ','×',1}, {'þ','Ø',2}, {'þ','Ý',2}, {'þ','ß',9}, {'þ','à',15}, {'þ','á',73}, {'þ','â',37}, {'þ','ã',12}, {'þ','ä',72},
++{'þ','å',12}, {'þ','æ',6}, {'þ','ç',11}, {'þ','è',45}, {'þ','é',1}, {'þ','ê',30}, {'þ','ë',9}, {'þ','ì',28}, {'þ','í',58}, {'þ','î',17}, {'þ','ï',58}, {'þ','ð',14}, {'þ','ñ',71}, {'þ','ò',139}, {'þ','ó',10}, {'þ','ô',5},
++{'þ','õ',5}, {'þ','ö',3}, {'þ','÷',42}, {'þ','ø',11}, {'þ','ù',27}, {'þ','ý',2}, {'þ','þ',10}, {'þ','ÿ',20}, {'ÿ','À',18}, {'ÿ','Á',5}, {'ÿ','Â',28}, {'ÿ','Ã',2}, {'ÿ','Ä',23}, {'ÿ','Å',6}, {'ÿ','Æ',2}, {'ÿ','Ç',3},
++{'ÿ','È',15}, {'ÿ','Ê',14}, {'ÿ','Ë',3}, {'ÿ','Ì',10}, {'ÿ','Í',23}, {'ÿ','Î',22}, {'ÿ','Ï',26}, {'ÿ','Ð',5}, {'ÿ','Ñ',12}, {'ÿ','Ò',17}, {'ÿ','Ó',1}, {'ÿ','Õ',2}, {'ÿ','Ö',1}, {'ÿ','×',7}, {'ÿ','Ø',2}, {'ÿ','Ý',10},
++{'ÿ','ß',18}, {'ÿ','à',23}, {'ÿ','á',85}, {'ÿ','â',260}, {'ÿ','ã',44}, {'ÿ','ä',170}, {'ÿ','å',70}, {'ÿ','æ',63}, {'ÿ','ç',114}, {'ÿ','è',146}, {'ÿ','é',7}, {'ÿ','ê',145}, {'ÿ','ë',137}, {'ÿ','ì',108}, {'ÿ','í',275}, {'ÿ','î',110},
++{'ÿ','ï',204}, {'ÿ','ð',43}, {'ÿ','ñ',259}, {'ÿ','ò',457}, {'ÿ','ó',64}, {'ÿ','ô',2}, {'ÿ','õ',37}, {'ÿ','ö',18}, {'ÿ','÷',75}, {'ÿ','ø',13}, {'ÿ','ù',33}, {'ÿ','ý',22}, {'ÿ','þ',31}, {'ÿ','ÿ',47}
++};
++
++static const lng_stat koi[]={
++{'À','À',10}, {'À','Á',15}, {'À','Â',73}, {'À','Ã',3}, {'À','Ä',72}, {'À','Å',12}, {'À','Æ',5}, {'À','Ç',12}, {'À','È',5}, {'À','É',45}, {'À','Ê',1}, {'À','Ë',30}, {'À','Ì',9}, {'À','Í',28}, {'À','Î',58}, {'À','Ï',17},
++{'À','Ð',58}, {'À','Ñ',20}, {'À','Ò',14}, {'À','Ó',71}, {'À','Ô',139}, {'À','Õ',10}, {'À','Ö',6}, {'À','×',37}, {'À','Ú',11}, {'À','Û',11}, {'À','Ü',2}, {'À','Ý',27}, {'À','Þ',42}, {'À','á',10}, {'À','â',1}, {'À','ä',3},
++{'À','ç',1}, {'À','è',3}, {'À','é',17}, {'À','ë',4}, {'À','ì',2}, {'À','í',3}, {'À','î',16}, {'À','ï',4}, {'À','ð',5}, {'À','ñ',9}, {'À','ò',1}, {'À','ó',3}, {'À','ô',2}, {'À','õ',2}, {'À','ö',1}, {'À','÷',10},
++{'À','ú',2}, {'À','û',2}, {'À','ü',2}, {'À','þ',1}, {'Á','À',238}, {'Á','Á',69}, {'Á','Â',257}, {'Á','Ã',24}, {'Á','Ä',548}, {'Á','Å',328}, {'Á','Æ',38}, {'Á','Ç',218}, {'Á','È',228}, {'Á','É',225}, {'Á','Ê',194}, {'Á','Ë',1285},
++{'Á','Ì',1288}, {'Á','Í',771}, {'Á','Î',911}, {'Á','Ï',166}, {'Á','Ð',421}, {'Á','Ñ',384}, {'Á','Ò',368}, {'Á','Ó',802}, {'Á','Ô',874}, {'Á','Õ',98}, {'Á','Ö',262}, {'Á','×',701}, {'Á','Ú',626}, {'Á','Û',121}, {'Á','Ü',63}, {'Á','Ý',27},
++{'Á','Þ',300}, {'Á','á',56}, {'Á','â',18}, {'Á','ä',26}, {'Á','å',12}, {'Á','æ',1}, {'Á','ç',9}, {'Á','è',4}, {'Á','é',53}, {'Á','ë',59}, {'Á','ì',9}, {'Á','í',21}, {'Á','î',32}, {'Á','ï',27}, {'Á','ð',52}, {'Á','ñ',22},
++{'Á','ò',7}, {'Á','ó',21}, {'Á','ô',42}, {'Á','õ',10}, {'Á','ö',7}, {'Á','÷',48}, {'Á','ú',11}, {'Á','û',2}, {'Á','ü',7}, {'Á','þ',16}, {'Â','À',1}, {'Â','Á',199}, {'Â','Â',6}, {'Â','Ã',3}, {'Â','Ä',6}, {'Â','Å',420},
++{'Â','È',11}, {'Â','É',142}, {'Â','Ë',7}, {'Â','Ì',137}, {'Â','Í',7}, {'Â','Î',31}, {'Â','Ï',327}, {'Â','Ð',4}, {'Â','Ñ',134}, {'Â','Ò',172}, {'Â','Ó',18}, {'Â','Ô',1}, {'Â','Õ',288}, {'Â','×',13}, {'Â','Ø',10}, {'Â','Ù',561},
++{'Â','Ú',2}, {'Â','Û',1}, {'Â','Ü',7}, {'Â','Ý',32}, {'Â','Þ',2}, {'Â','á',1}, {'Â','æ',1}, {'Â','ð',1}, {'Ã','Á',179}, {'Ã','Â',1}, {'Ã','Å',162}, {'Ã','É',75}, {'Ã','Ë',6}, {'Ã','Ì',5}, {'Ã','Í',1}, {'Ã','Î',2},
++{'Ã','Ï',29}, {'Ã','Ð',4}, {'Ã','Ò',2}, {'Ã','Ó',4}, {'Ã','Ô',4}, {'Ã','Õ',41}, {'Ã','×',11}, {'Ã','Ù',41}, {'Ã','Ú',5}, {'Ã','Þ',2}, {'Ã','á',1}, {'Ã','ç',1}, {'Ã','é',2}, {'Ã','ë',1}, {'Ã','ì',1}, {'Ã','í',1},
++{'Ã','ð',1}, {'Ã','ñ',1}, {'Ã','ô',1}, {'Ä','Á',795}, {'Ä','Â',7}, {'Ä','Ã',138}, {'Ä','Ä',14}, {'Ä','Å',839}, {'Ä','Æ',1}, {'Ä','Ç',6}, {'Ä','È',11}, {'Ä','É',487}, {'Ä','Ê',1}, {'Ä','Ë',79}, {'Ä','Ì',78}, {'Ä','Í',10},
++{'Ä','Î',289}, {'Ä','Ï',613}, {'Ä','Ð',17}, {'Ä','Ñ',68}, {'Ä','Ò',195}, {'Ä','Ó',62}, {'Ä','Ô',41}, {'Ä','Õ',417}, {'Ä','Ö',8}, {'Ä','×',165}, {'Ä','Ø',200}, {'Ä','Ù',111}, {'Ä','Ú',7}, {'Ä','Ü',6}, {'Ä','Þ',6}, {'Ä','á',1},
++{'Ä','ä',2}, {'Ä','æ',1}, {'Ä','é',2}, {'Ä','ë',1}, {'Ä','í',1}, {'Ä','î',2}, {'Ä','ð',2}, {'Ä','ñ',2}, {'Ä','ô',1}, {'Ä','õ',1}, {'Ä','÷',2}, {'Ä','ú',1}, {'Ä','þ',2}, {'Å','À',24}, {'Å','Á',44}, {'Å','Â',463},
++{'Å','Ã',66}, {'Å','Ä',613}, {'Å','Å',281}, {'Å','Æ',15}, {'Å','Ç',592}, {'Å','È',182}, {'Å','É',177}, {'Å','Ê',297}, {'Å','Ë',398}, {'Å','Ì',926}, {'Å','Í',982}, {'Å','Î',1512}, {'Å','Ï',206}, {'Å','Ð',524}, {'Å','Ñ',87}, {'Å','Ò',1231},
++{'Å','Ó',1082}, {'Å','Ô',1607}, {'Å','Õ',103}, {'Å','Ö',136}, {'Å','×',704}, {'Å','Ú',444}, {'Å','Û',235}, {'Å','Ü',79}, {'Å','Ý',137}, {'Å','Þ',308}, {'Å','à',1}, {'Å','á',30}, {'Å','â',8}, {'Å','ã',1}, {'Å','ä',34}, {'Å','å',13},
++{'Å','æ',4}, {'Å','ç',7}, {'Å','è',2}, {'Å','é',21}, {'Å','ë',21}, {'Å','ì',4}, {'Å','í',13}, {'Å','î',20}, {'Å','ï',21}, {'Å','ð',31}, {'Å','ñ',25}, {'Å','ò',2}, {'Å','ó',13}, {'Å','ô',10}, {'Å','õ',7}, {'Å','ö',1},
++{'Å','÷',33}, {'Å','ú',6}, {'Å','û',2}, {'Å','ü',11}, {'Å','þ',9}, {'Æ','À',2}, {'Æ','Á',21}, {'Æ','Å',45}, {'Æ','Æ',6}, {'Æ','É',47}, {'Æ','Ì',14}, {'Æ','Î',2}, {'Æ','Ï',24}, {'Æ','Ò',20}, {'Æ','Ó',4}, {'Æ','Ô',5},
++{'Æ','Õ',8}, {'Æ','Ø',1}, {'Æ','Ù',4}, {'Æ','ã',2}, {'Æ','ð',1}, {'Ç','Á',210}, {'Ç','Â',18}, {'Ç','Ä',191}, {'Ç','Å',83}, {'Ç','Ç',3}, {'Ç','È',1}, {'Ç','É',103}, {'Ç','Ë',20}, {'Ç','Ì',205}, {'Ç','Í',3}, {'Ç','Î',62},
++{'Ç','Ï',1340}, {'Ç','Ð',7}, {'Ç','Ñ',4}, {'Ç','Ò',180}, {'Ç','Ó',13}, {'Ç','Ô',4}, {'Ç','Õ',88}, {'Ç','×',6}, {'Ç','Ú',2}, {'Ç','Ü',1}, {'Ç','Þ',15}, {'Ç','á',3}, {'Ç','â',1}, {'Ç','ä',1}, {'Ç','é',3}, {'Ç','ì',4},
++{'Ç','î',1}, {'Ç','ï',4}, {'Ç','ð',3}, {'Ç','ñ',2}, {'Ç','ò',1}, {'Ç','ó',2}, {'Ç','ô',2}, {'Ç','÷',3}, {'Ç','ú',3}, {'Ç','û',1}, {'Ç','þ',1}, {'È','Á',135}, {'È','Â',21}, {'È','Ã',2}, {'È','Ä',21}, {'È','Å',25},
++{'È','Æ',3}, {'È','Ç',27}, {'È','È',4}, {'È','É',75}, {'È','Ë',16}, {'È','Ì',43}, {'È','Í',49}, {'È','Î',83}, {'È','Ï',450}, {'È','Ð',70}, {'È','Ñ',9}, {'È','Ò',19}, {'È','Ó',39}, {'È','Ô',27}, {'È','Õ',42}, {'È','Ö',4},
++{'È','×',72}, {'È','Ø',6}, {'È','Ù',1}, {'È','Ú',13}, {'È','Û',8}, {'È','Ü',7}, {'È','Þ',18}, {'È','á',7}, {'È','ä',4}, {'È','å',3}, {'È','é',5}, {'È','ë',4}, {'È','ì',2}, {'È','í',1}, {'È','î',10}, {'È','ï',5},
++{'È','ð',7}, {'È','ñ',5}, {'È','ó',4}, {'È','ô',5}, {'È','ö',3}, {'È','÷',7}, {'È','û',5}, {'È','ü',2}, {'È','þ',3}, {'É','À',22}, {'É','Á',90}, {'É','Â',242}, {'É','Ã',123}, {'É','Ä',435}, {'É','Å',280}, {'É','Æ',17},
++{'É','Ç',153}, {'É','È',324}, {'É','É',301}, {'É','Ê',216}, {'É','Ë',492}, {'É','Ì',888}, {'É','Í',573}, {'É','Î',958}, {'É','Ï',216}, {'É','Ð',335}, {'É','Ñ',234}, {'É','Ò',275}, {'É','Ó',668}, {'É','Ô',949}, {'É','Õ',108}, {'É','Ö',87},
++{'É','×',658}, {'É','Ú',388}, {'É','Û',134}, {'É','Ü',39}, {'É','Ý',36}, {'É','Þ',363}, {'É','á',39}, {'É','â',10}, {'É','ã',2}, {'É','ä',25}, {'É','å',11}, {'É','æ',3}, {'É','ç',5}, {'É','è',4}, {'É','é',30}, {'É','ë',24},
++{'É','ì',5}, {'É','í',36}, {'É','î',39}, {'É','ï',25}, {'É','ð',49}, {'É','ñ',28}, {'É','ò',3}, {'É','ó',22}, {'É','ô',25}, {'É','õ',8}, {'É','ö',2}, {'É','÷',46}, {'É','ú',1}, {'É','û',3}, {'É','ü',7}, {'É','þ',4},
++{'Ê','Á',21}, {'Ê','Â',36}, {'Ê','Ã',11}, {'Ê','Ä',100}, {'Ê','Å',15}, {'Ê','Æ',6}, {'Ê','Ç',26}, {'Ê','È',7}, {'Ê','É',88}, {'Ê','Ê',1}, {'Ê','Ë',87}, {'Ê','Ì',25}, {'Ê','Í',47}, {'Ê','Î',117}, {'Ê','Ï',55}, {'Ê','Ð',133},
++{'Ê','Ñ',16}, {'Ê','Ò',42}, {'Ê','Ó',162}, {'Ê','Ô',131}, {'Ê','Õ',17}, {'Ê','Ö',21}, {'Ê','×',112}, {'Ê','Ú',30}, {'Ê','Û',20}, {'Ê','Ü',15}, {'Ê','Ý',2}, {'Ê','Þ',69}, {'Ê','á',20}, {'Ê','â',6}, {'Ê','ä',16}, {'Ê','å',6},
++{'Ê','æ',1}, {'Ê','ç',6}, {'Ê','è',1}, {'Ê','é',22}, {'Ê','ë',18}, {'Ê','ì',2}, {'Ê','í',12}, {'Ê','î',13}, {'Ê','ï',15}, {'Ê','ð',13}, {'Ê','ñ',8}, {'Ê','ò',7}, {'Ê','ó',9}, {'Ê','ô',15}, {'Ê','õ',3}, {'Ê','÷',20},
++{'Ê','ú',2}, {'Ê','ü',7}, {'Ê','þ',1}, {'Ë','Á',1358}, {'Ë','Â',49}, {'Ë','Ã',5}, {'Ë','Ä',48}, {'Ë','Å',93}, {'Ë','Æ',7}, {'Ë','Ç',15}, {'Ë','È',10}, {'Ë','É',678}, {'Ë','Ë',28}, {'Ë','Ì',81}, {'Ë','Í',22}, {'Ë','Î',151},
++{'Ë','Ï',1337}, {'Ë','Ð',64}, {'Ë','Ñ',25}, {'Ë','Ò',273}, {'Ë','Ó',131}, {'Ë','Ô',238}, {'Ë','Õ',384}, {'Ë','Ö',22}, {'Ë','×',147}, {'Ë','Ù',2}, {'Ë','Ú',55}, {'Ë','Û',2}, {'Ë','Ü',13}, {'Ë','Þ',31}, {'Ë','á',11}, {'Ë','ã',1},
++{'Ë','ä',4}, {'Ë','å',2}, {'Ë','ç',7}, {'Ë','è',1}, {'Ë','é',8}, {'Ë','ë',9}, {'Ë','ì',2}, {'Ë','í',5}, {'Ë','î',20}, {'Ë','ï',7}, {'Ë','ð',9}, {'Ë','ñ',8}, {'Ë','ó',7}, {'Ë','ô',9}, {'Ë','õ',4}, {'Ë','ö',1},
++{'Ë','÷',13}, {'Ë','ú',5}, {'Ë','ü',2}, {'Ë','þ',4}, {'Ì','À',159}, {'Ì','Á',758}, {'Ì','Â',34}, {'Ì','Ã',2}, {'Ì','Ä',55}, {'Ì','Å',675}, {'Ì','Æ',5}, {'Ì','Ç',35}, {'Ì','È',7}, {'Ì','É',1300}, {'Ì','Ê',1}, {'Ì','Ë',114},
++{'Ì','Ì',21}, {'Ì','Í',45}, {'Ì','Î',111}, {'Ì','Ï',947}, {'Ì','Ð',83}, {'Ì','Ñ',299}, {'Ì','Ò',18}, {'Ì','Ó',320}, {'Ì','Ô',61}, {'Ì','Õ',264}, {'Ì','Ö',33}, {'Ì','×',102}, {'Ì','Ø',642}, {'Ì','Ù',146}, {'Ì','Ú',23}, {'Ì','Û',1},
++{'Ì','Ü',21}, {'Ì','Þ',66}, {'Ì','á',20}, {'Ì','â',2}, {'Ì','ã',1}, {'Ì','ä',14}, {'Ì','å',3}, {'Ì','æ',1}, {'Ì','ç',5}, {'Ì','è',1}, {'Ì','é',19}, {'Ì','ë',12}, {'Ì','ì',2}, {'Ì','í',14}, {'Ì','î',31}, {'Ì','ï',20},
++{'Ì','ð',20}, {'Ì','ñ',19}, {'Ì','ò',1}, {'Ì','ó',20}, {'Ì','ô',10}, {'Ì','õ',1}, {'Ì','ö',1}, {'Ì','÷',25}, {'Ì','ü',4}, {'Ì','þ',2}, {'Í','Á',447}, {'Í','Â',69}, {'Í','Ã',4}, {'Í','Ä',41}, {'Í','Å',838}, {'Í','Æ',5},
++{'Í','Ç',33}, {'Í','È',13}, {'Í','É',484}, {'Í','Ê',2}, {'Í','Ë',71}, {'Í','Ì',74}, {'Í','Í',96}, {'Í','Î',497}, {'Í','Ï',761}, {'Í','Ð',132}, {'Í','Ñ',82}, {'Í','Ò',41}, {'Í','Ó',137}, {'Í','Ô',74}, {'Í','Õ',411}, {'Í','Ö',27},
++{'Í','×',119}, {'Í','Ø',33}, {'Í','Ù',207}, {'Í','Ú',38}, {'Í','Û',3}, {'Í','Ü',17}, {'Í','Ý',3}, {'Í','Þ',46}, {'Í','á',11}, {'Í','â',2}, {'Í','ä',10}, {'Í','å',1}, {'Í','ç',8}, {'Í','è',2}, {'Í','é',12}, {'Í','ë',10},
++{'Í','í',6}, {'Í','î',23}, {'Í','ï',12}, {'Í','ð',12}, {'Í','ñ',10}, {'Í','ò',1}, {'Í','ó',9}, {'Í','ô',16}, {'Í','õ',2}, {'Í','÷',14}, {'Í','ú',5}, {'Í','û',1}, {'Í','ü',1}, {'Í','þ',3}, {'Î','À',30}, {'Î','Á',1706},
++{'Î','Â',28}, {'Î','Ã',65}, {'Î','Ä',97}, {'Î','Å',1870}, {'Î','Æ',14}, {'Î','Ç',48}, {'Î','È',4}, {'Î','É',1319}, {'Î','Ë',84}, {'Î','Ì',7}, {'Î','Í',11}, {'Î','Î',331}, {'Î','Ï',1483}, {'Î','Ð',54}, {'Î','Ñ',396}, {'Î','Ò',27},
++{'Î','Ó',133}, {'Î','Ô',165}, {'Î','Õ',480}, {'Î','Ö',5}, {'Î','×',55}, {'Î','Ø',196}, {'Î','Ù',385}, {'Î','Ú',20}, {'Î','Û',3}, {'Î','Ü',7}, {'Î','Ý',31}, {'Î','Þ',55}, {'Î','á',11}, {'Î','ã',1}, {'Î','ä',3}, {'Î','å',4},
++{'Î','ç',8}, {'Î','é',7}, {'Î','ë',10}, {'Î','ì',1}, {'Î','í',2}, {'Î','î',6}, {'Î','ï',2}, {'Î','ð',9}, {'Î','ñ',2}, {'Î','ò',2}, {'Î','ó',2}, {'Î','ô',6}, {'Î','õ',1}, {'Î','÷',5}, {'Î','û',1}, {'Î','ü',2},
++{'Î','þ',6}, {'Ï','À',81}, {'Ï','Á',34}, {'Ï','Â',652}, {'Ï','Ã',51}, {'Ï','Ä',946}, {'Ï','Å',457}, {'Ï','Æ',35}, {'Ï','Ç',915}, {'Ï','È',137}, {'Ï','É',291}, {'Ï','Ê',602}, {'Ï','Ë',498}, {'Ï','Ì',923}, {'Ï','Í',1073}, {'Ï','Î',1281},
++{'Ï','Ï',241}, {'Ï','Ð',588}, {'Ï','Ñ',154}, {'Ï','Ò',921}, {'Ï','Ó',1264}, {'Ï','Ô',1536}, {'Ï','Õ',152}, {'Ï','Ö',437}, {'Ï','×',1590}, {'Ï','Ú',279}, {'Ï','Û',223}, {'Ï','Ü',89}, {'Ï','Ý',56}, {'Ï','Þ',459}, {'Ï','á',34}, {'Ï','â',5},
++{'Ï','ã',2}, {'Ï','ä',16}, {'Ï','å',16}, {'Ï','æ',7}, {'Ï','ç',9}, {'Ï','è',4}, {'Ï','é',43}, {'Ï','ë',17}, {'Ï','ì',3}, {'Ï','í',23}, {'Ï','î',49}, {'Ï','ï',23}, {'Ï','ð',40}, {'Ï','ñ',27}, {'Ï','ò',6}, {'Ï','ó',14},
++{'Ï','ô',19}, {'Ï','õ',4}, {'Ï','ö',4}, {'Ï','÷',44}, {'Ï','ú',17}, {'Ï','ü',5}, {'Ï','þ',12}, {'Ð','Á',252}, {'Ð','Ä',1}, {'Ð','Å',364}, {'Ð','É',356}, {'Ð','Ë',8}, {'Ð','Ì',189}, {'Ð','Î',11}, {'Ð','Ï',1543}, {'Ð','Ð',6},
++{'Ð','Ñ',149}, {'Ð','Ò',845}, {'Ð','Ó',6}, {'Ð','Ô',23}, {'Ð','Õ',202}, {'Ð','×',1}, {'Ð','Ø',122}, {'Ð','Ù',33}, {'Ð','Þ',2}, {'Ð','é',1}, {'Ð','ë',1}, {'Ð','÷',2}, {'Ñ','À',31}, {'Ñ','Á',23}, {'Ñ','Â',85}, {'Ñ','Ã',18},
++{'Ñ','Ä',170}, {'Ñ','Å',70}, {'Ñ','Æ',2}, {'Ñ','Ç',44}, {'Ñ','È',37}, {'Ñ','É',146}, {'Ñ','Ê',7}, {'Ñ','Ë',145}, {'Ñ','Ì',137}, {'Ñ','Í',108}, {'Ñ','Î',275}, {'Ñ','Ï',110}, {'Ñ','Ð',204}, {'Ñ','Ñ',47}, {'Ñ','Ò',43}, {'Ñ','Ó',259},
++{'Ñ','Ô',457}, {'Ñ','Õ',64}, {'Ñ','Ö',63}, {'Ñ','×',260}, {'Ñ','Ú',114}, {'Ñ','Û',13}, {'Ñ','Ü',22}, {'Ñ','Ý',33}, {'Ñ','Þ',75}, {'Ñ','á',18}, {'Ñ','â',5}, {'Ñ','ã',1}, {'Ñ','ä',23}, {'Ñ','å',6}, {'Ñ','ç',2}, {'Ñ','è',2},
++{'Ñ','é',15}, {'Ñ','ë',14}, {'Ñ','ì',3}, {'Ñ','í',10}, {'Ñ','î',23}, {'Ñ','ï',22}, {'Ñ','ð',26}, {'Ñ','ñ',18}, {'Ñ','ò',5}, {'Ñ','ó',12}, {'Ñ','ô',17}, {'Ñ','õ',1}, {'Ñ','ö',2}, {'Ñ','÷',28}, {'Ñ','ú',3}, {'Ñ','û',2},
++{'Ñ','ü',10}, {'Ñ','þ',7}, {'Ò','À',59}, {'Ò','Á',1264}, {'Ò','Â',29}, {'Ò','Ã',23}, {'Ò','Ä',129}, {'Ò','Å',936}, {'Ò','Æ',8}, {'Ò','Ç',53}, {'Ò','È',36}, {'Ò','É',777}, {'Ò','Ê',1}, {'Ò','Ë',54}, {'Ò','Ì',29}, {'Ò','Í',39},
++{'Ò','Î',181}, {'Ò','Ï',1126}, {'Ò','Ð',38}, {'Ò','Ñ',159}, {'Ò','Ò',34}, {'Ò','Ó',76}, {'Ò','Ô',122}, {'Ò','Õ',355}, {'Ò','Ö',41}, {'Ò','×',98}, {'Ò','Ø',98}, {'Ò','Ù',169}, {'Ò','Ú',20}, {'Ò','Û',49}, {'Ò','Ü',7}, {'Ò','Ý',5},
++{'Ò','Þ',9}, {'Ò','á',3}, {'Ò','â',4}, {'Ò','ä',6}, {'Ò','å',2}, {'Ò','æ',2}, {'Ò','è',1}, {'Ò','ë',2}, {'Ò','í',1}, {'Ò','î',5}, {'Ò','ï',8}, {'Ò','ð',10}, {'Ò','ñ',6}, {'Ò','ó',2}, {'Ò','ô',4}, {'Ò','÷',3},
++{'Ò','ú',1}, {'Ò','ü',3}, {'Ò','þ',2}, {'Ó','À',27}, {'Ó','Á',306}, {'Ó','Â',33}, {'Ó','Ã',6}, {'Ó','Ä',41}, {'Ó','Å',973}, {'Ó','Æ',18}, {'Ó','Ç',18}, {'Ó','È',34}, {'Ó','É',295}, {'Ó','Ë',691}, {'Ó','Ì',519}, {'Ó','Í',222},
++{'Ó','Î',236}, {'Ó','Ï',403}, {'Ó','Ð',274}, {'Ó','Ñ',680}, {'Ó','Ò',58}, {'Ó','Ó',222}, {'Ó','Ô',1721}, {'Ó','Õ',146}, {'Ó','Ö',8}, {'Ó','×',259}, {'Ó','Ø',401}, {'Ó','Ù',79}, {'Ó','Ú',11}, {'Ó','Û',20}, {'Ó','Ü',10}, {'Ó','Ý',2},
++{'Ó','Þ',48}, {'Ó','á',6}, {'Ó','ä',2}, {'Ó','ç',2}, {'Ó','é',4}, {'Ó','ë',2}, {'Ó','ì',3}, {'Ó','í',4}, {'Ó','î',5}, {'Ó','ï',3}, {'Ó','ð',3}, {'Ó','ñ',1}, {'Ó','ò',1}, {'Ó','ó',2}, {'Ó','ô',1}, {'Ó','ö',1},
++{'Ó','÷',4}, {'Ó','ú',1}, {'Ó','ü',3}, {'Ó','þ',1}, {'Ô','À',13}, {'Ô','Á',1225}, {'Ô','Â',56}, {'Ô','Ã',10}, {'Ô','Ä',67}, {'Ô','Å',864}, {'Ô','Æ',1}, {'Ô','Ç',25}, {'Ô','È',24}, {'Ô','É',758}, {'Ô','Ê',1}, {'Ô','Ë',180},
++{'Ô','Ì',52}, {'Ô','Í',65}, {'Ô','Î',329}, {'Ô','Ï',2625}, {'Ô','Ð',119}, {'Ô','Ñ',103}, {'Ô','Ò',616}, {'Ô','Ó',337}, {'Ô','Ô',117}, {'Ô','Õ',398}, {'Ô','Ö',17}, {'Ô','×',436}, {'Ô','Ø',1292}, {'Ô','Ù',398}, {'Ô','Ú',33}, {'Ô','Û',3},
++{'Ô','Ü',37}, {'Ô','Ý',4}, {'Ô','Þ',87}, {'Ô','à',1}, {'Ô','á',29}, {'Ô','â',5}, {'Ô','ä',11}, {'Ô','å',9}, {'Ô','æ',2}, {'Ô','ç',3}, {'Ô','è',3}, {'Ô','é',16}, {'Ô','ë',21}, {'Ô','í',20}, {'Ô','î',24}, {'Ô','ï',10},
++{'Ô','ð',23}, {'Ô','ñ',16}, {'Ô','ò',2}, {'Ô','ó',10}, {'Ô','ô',13}, {'Ô','õ',2}, {'Ô','ö',2}, {'Ô','÷',25}, {'Ô','ú',5}, {'Ô','û',1}, {'Ô','ü',3}, {'Ô','þ',11}, {'Õ','À',156}, {'Õ','Á',45}, {'Õ','Â',142}, {'Õ','Ä',445},
++{'Õ','Å',83}, {'Õ','Æ',5}, {'Õ','Ç',161}, {'Õ','È',97}, {'Õ','É',111}, {'Õ','Ê',40}, {'Õ','Ë',169}, {'Õ','Ì',251}, {'Õ','Í',322}, {'Õ','Î',195}, {'Õ','Ï',50}, {'Õ','Ð',258}, {'Õ','Ñ',31}, {'Õ','Ò',203}, {'Õ','Ó',380}, {'Õ','Ô',442},
++{'Õ','Õ',24}, {'Õ','Ö',269}, {'Õ','×',184}, {'Õ','Ú',75}, {'Õ','Û',302}, {'Õ','Ü',19}, {'Õ','Ý',37}, {'Õ','Þ',241}, {'Õ','á',22}, {'Õ','â',4}, {'Õ','ä',20}, {'Õ','å',3}, {'Õ','æ',2}, {'Õ','ç',4}, {'Õ','è',5}, {'Õ','é',25},
++{'Õ','ë',9}, {'Õ','ì',6}, {'Õ','í',12}, {'Õ','î',18}, {'Õ','ï',13}, {'Õ','ð',18}, {'Õ','ñ',17}, {'Õ','ò',1}, {'Õ','ó',18}, {'Õ','ô',12}, {'Õ','õ',2}, {'Õ','÷',19}, {'Õ','ú',2}, {'Õ','û',3}, {'Õ','ü',5}, {'Õ','þ',6},
++{'Ö','Á',187}, {'Ö','Â',4}, {'Ö','Ä',145}, {'Ö','Å',690}, {'Ö','Ç',1}, {'Ö','È',1}, {'Ö','É',214}, {'Ö','Ë',35}, {'Ö','Ì',1}, {'Ö','Í',8}, {'Ö','Î',96}, {'Ö','Ï',13}, {'Ö','Ð',14}, {'Ö','Ñ',7}, {'Ö','Ò',7}, {'Ö','Ó',4},
++{'Ö','Ô',8}, {'Ö','Õ',69}, {'Ö','Ö',4}, {'Ö','×',9}, {'Ö','Ø',6}, {'Ö','Ú',1}, {'Ö','Ü',2}, {'Ö','Þ',1}, {'Ö','é',2}, {'Ö','î',1}, {'Ö','ï',1}, {'Ö','ð',3}, {'Ö','÷',1}, {'×','À',3}, {'×','Á',989}, {'×','Â',26},
++{'×','Ã',11}, {'×','Ä',74}, {'×','Å',911}, {'×','Æ',1}, {'×','Ç',43}, {'×','È',9}, {'×','É',387}, {'×','Ê',1}, {'×','Ë',105}, {'×','Ì',122}, {'×','Í',65}, {'×','Î',231}, {'×','Ï',1220}, {'×','Ð',118}, {'×','Ñ',48}, {'×','Ò',97},
++{'×','Ó',851}, {'×','Ô',123}, {'×','Õ',116}, {'×','Ö',26}, {'×','×',62}, {'×','Ø',33}, {'×','Ù',518}, {'×','Ú',88}, {'×','Û',53}, {'×','Ü',44}, {'×','Ý',2}, {'×','Þ',47}, {'×','á',11}, {'×','â',1}, {'×','ã',1}, {'×','ä',5},
++{'×','å',5}, {'×','ç',7}, {'×','è',8}, {'×','é',6}, {'×','ë',14}, {'×','ì',2}, {'×','í',9}, {'×','î',13}, {'×','ï',17}, {'×','ð',51}, {'×','ñ',4}, {'×','ò',7}, {'×','ó',18}, {'×','ô',4}, {'×','õ',2}, {'×','÷',6},
++{'×','û',7}, {'×','ü',2}, {'×','þ',2}, {'Ø','À',88}, {'Ø','Á',38}, {'Ø','Â',63}, {'Ø','Ã',15}, {'Ø','Ä',110}, {'Ø','Å',154}, {'Ø','Æ',5}, {'Ø','Ç',35}, {'Ø','È',20}, {'Ø','É',194}, {'Ø','Ê',1}, {'Ø','Ë',324}, {'Ø','Ì',36},
++{'Ø','Í',124}, {'Ø','Î',321}, {'Ø','Ï',92}, {'Ø','Ð',164}, {'Ø','Ñ',110}, {'Ø','Ò',41}, {'Ø','Ó',302}, {'Ø','Ô',152}, {'Ø','Õ',48}, {'Ø','Ö',11}, {'Ø','×',196}, {'Ø','Ú',58}, {'Ø','Û',85}, {'Ø','Ü',27}, {'Ø','Ý',1}, {'Ø','Þ',88},
++{'Ø','á',44}, {'Ø','â',8}, {'Ø','ä',18}, {'Ø','å',12}, {'Ø','æ',1}, {'Ø','ç',7}, {'Ø','è',3}, {'Ø','é',29}, {'Ø','ë',19}, {'Ø','í',18}, {'Ø','î',36}, {'Ø','ï',25}, {'Ø','ð',42}, {'Ø','ñ',21}, {'Ø','ò',4}, {'Ø','ó',25},
++{'Ø','ô',24}, {'Ø','õ',10}, {'Ø','ö',1}, {'Ø','÷',45}, {'Ø','ú',4}, {'Ø','û',1}, {'Ø','ü',4}, {'Ø','þ',14}, {'Ù','Á',10}, {'Ù','Â',66}, {'Ù','Ã',3}, {'Ù','Ä',65}, {'Ù','Å',138}, {'Ù','Æ',1}, {'Ù','Ç',37}, {'Ù','È',146},
++{'Ù','É',74}, {'Ù','Ê',264}, {'Ù','Ë',83}, {'Ù','Ì',305}, {'Ù','Í',164}, {'Ù','Î',141}, {'Ù','Ï',34}, {'Ù','Ð',231}, {'Ù','Ñ',10}, {'Ù','Ò',61}, {'Ù','Ó',189}, {'Ù','Ô',148}, {'Ù','Õ',35}, {'Ù','Ö',21}, {'Ù','×',248}, {'Ù','Ú',45},
++{'Ù','Û',80}, {'Ù','Ü',9}, {'Ù','Ý',1}, {'Ù','Þ',63}, {'Ù','à',1}, {'Ù','á',14}, {'Ù','â',6}, {'Ù','ä',8}, {'Ù','å',6}, {'Ù','æ',1}, {'Ù','ç',4}, {'Ù','é',12}, {'Ù','ë',10}, {'Ù','ì',2}, {'Ù','í',15}, {'Ù','î',16},
++{'Ù','ï',13}, {'Ù','ð',6}, {'Ù','ñ',10}, {'Ù','ò',3}, {'Ù','ó',7}, {'Ù','ô',10}, {'Ù','õ',1}, {'Ù','ö',1}, {'Ù','÷',28}, {'Ù','ú',4}, {'Ù','ü',5}, {'Ù','þ',4}, {'Ú','Á',889}, {'Ú','Â',39}, {'Ú','Ä',174}, {'Ú','Å',56},
++{'Ú','Ç',67}, {'Ú','È',3}, {'Ú','É',113}, {'Ú','Ë',16}, {'Ú','Ì',33}, {'Ú','Í',64}, {'Ú','Î',302}, {'Ú','Ï',109}, {'Ú','Ð',31}, {'Ú','Ñ',43}, {'Ú','Ò',53}, {'Ú','Ó',21}, {'Ú','Ô',7}, {'Ú','Õ',109}, {'Ú','Ö',18}, {'Ú','×',147},
++{'Ú','Ø',20}, {'Ú','Ù',71}, {'Ú','Ú',11}, {'Ú','Û',4}, {'Ú','Ü',4}, {'Ú','Þ',11}, {'Ú','á',1}, {'Ú','ã',1}, {'Ú','é',2}, {'Ú','ë',1}, {'Ú','í',4}, {'Ú','î',3}, {'Ú','ï',1}, {'Ú','ð',8}, {'Ú','ñ',2}, {'Ú','ó',4},
++{'Ú','ô',2}, {'Ú','ö',1}, {'Ú','÷',1}, {'Ú','þ',2}, {'Û','Á',150}, {'Û','Ã',1}, {'Û','Å',388}, {'Û','È',1}, {'Û','É',188}, {'Û','Ë',201}, {'Û','Ì',68}, {'Û','Í',4}, {'Û','Î',78}, {'Û','Ï',53}, {'Û','Ð',8}, {'Û','Ò',3},
++{'Û','Ó',2}, {'Û','Ô',17}, {'Û','Õ',45}, {'Û','×',5}, {'Û','Ø',200}, {'Û','Ü',1}, {'Û','ñ',1}, {'Ü','Â',1}, {'Ü','Ä',2}, {'Ü','Æ',2}, {'Ü','Ç',1}, {'Ü','È',1}, {'Ü','Ë',6}, {'Ü','Ì',9}, {'Ü','Í',6}, {'Ü','Î',3},
++{'Ü','Ò',8}, {'Ü','Ó',4}, {'Ü','Ô',531}, {'Ü','Ú',1}, {'Ü','Ü',2}, {'Ü','é',1}, {'Ý','Á',68}, {'Ý','Å',242}, {'Ý','É',96}, {'Ý','Î',7}, {'Ý','Ï',1}, {'Ý','Ò',1}, {'Ý','Ó',3}, {'Ý','Õ',20}, {'Ý','Ø',2}, {'Ý','ð',1},
++{'Ý','ô',1}, {'Þ','Á',294}, {'Þ','Â',1}, {'Þ','Å',874}, {'Þ','É',319}, {'Þ','Ë',76}, {'Þ','Ì',9}, {'Þ','Í',4}, {'Þ','Î',158}, {'Þ','Ï',10}, {'Þ','Ð',3}, {'Þ','Ñ',1}, {'Þ','Ò',8}, {'Þ','Ó',2}, {'Þ','Ô',536}, {'Þ','Õ',101},
++{'Þ','×',6}, {'Þ','Ø',49}, {'Þ','Û',37}, {'Þ','Þ',7}, {'Þ','å',1}, {'Þ','ç',1}, {'Þ','é',1}, {'Þ','î',3}, {'Þ','ï',1}, {'Þ','ó',3}, {'Þ','ô',2}, {'Þ','÷',1}, {'Þ','þ',1}, {'à','Ò',1}, {'à','Ú',2}, {'á','Á',1},
++{'á','Â',6}, {'á','Ä',7}, {'á','Å',23}, {'á','Æ',1}, {'á','Ç',5}, {'á','È',6}, {'á','É',4}, {'á','Ê',1}, {'á','Ë',27}, {'á','Ì',15}, {'á','Í',10}, {'á','Î',25}, {'á','Ï',21}, {'á','Ð',51}, {'á','Ñ',20}, {'á','Ò',12},
++{'á','Ó',13}, {'á','Ô',31}, {'á','Õ',15}, {'á','Ö',2}, {'á','×',60}, {'á','Ú',6}, {'á','Ü',8}, {'á','Þ',17}, {'á','á',1}, {'á','æ',1}, {'á','è',1}, {'á','é',1}, {'á','ë',1}, {'á','ì',1}, {'á','í',8}, {'á','î',2},
++{'á','ð',2}, {'á','ò',1}, {'á','ó',4}, {'á','ô',2}, {'á','÷',1}, {'á','ú',1}, {'á','û',1}, {'á','ü',1}, {'á','þ',1}, {'â','Á',3}, {'â','Å',16}, {'â','É',1}, {'â','Ì',4}, {'â','Î',3}, {'â','Ï',25}, {'â','Ò',10},
++{'â','Õ',9}, {'â','×',2}, {'â','Ù',8}, {'â','ä',1}, {'â','î',2}, {'â','ô',1}, {'â','÷',1}, {'ã','Á',7}, {'ã','É',3}, {'ã','Ó',1}, {'ã','Ô',1}, {'ã','Ù',1}, {'ä','Á',167}, {'ä','Å',17}, {'ä','É',3}, {'ä','Ì',7},
++{'ä','Í',3}, {'ä','Î',1}, {'ä','Ï',20}, {'ä','Ò',9}, {'ä','Õ',19}, {'ä','×',6}, {'å','Ä',1}, {'å','Å',1}, {'å','Æ',1}, {'å','È',1}, {'å','Ê',1}, {'å','Ì',5}, {'å','Í',2}, {'å','Ò',19}, {'å','Ó',69}, {'å','×',9},
++{'å','Ý',5}, {'å','ç',2}, {'å','è',1}, {'å','û',1}, {'æ','Á',7}, {'æ','Å',2}, {'æ','Æ',2}, {'æ','Ï',1}, {'æ','Ò',16}, {'æ','á',1}, {'ç','Á',6}, {'ç','Ä',16}, {'ç','Å',26}, {'ç','É',4}, {'ç','Ì',3}, {'ç','Í',1},
++{'ç','Ï',31}, {'ç','Ò',7}, {'ç','Õ',2}, {'ç','ï',2}, {'è','Á',8}, {'è','Å',3}, {'è','Ï',20}, {'è','Ò',14}, {'è','×',1}, {'è','æ',1}, {'é','Á',2}, {'é','Â',8}, {'é','Ä',20}, {'é','Å',7}, {'é','Æ',2}, {'é','Ç',7},
++{'é','È',3}, {'é','É',3}, {'é','Ë',27}, {'é','Ì',15}, {'é','Í',11}, {'é','Î',21}, {'é','Ï',26}, {'é','Ð',21}, {'é','Ñ',12}, {'é','Ò',4}, {'é','Ó',21}, {'é','Ô',37}, {'é','Õ',5}, {'é','Ö',1}, {'é','×',78}, {'é','Ú',17},
++{'é','Û',1}, {'é','Ü',2}, {'é','Þ',8}, {'é','á',1}, {'é','ì',2}, {'é','í',2}, {'é','ó',2}, {'é','÷',1}, {'ë','Á',100}, {'ë','É',4}, {'ë','Ë',1}, {'ë','Ì',14}, {'ë','Î',4}, {'ë','Ï',62}, {'ë','Ð',2}, {'ë','Ò',24},
++{'ë','Ô',10}, {'ë','Õ',62}, {'ë','ð',3}, {'ì','À',3}, {'ì','Á',7}, {'ì','Å',11}, {'ì','É',4}, {'ì','Ï',12}, {'ì','Ñ',1}, {'ì','Õ',15}, {'ì','á',1}, {'ì','é',1}, {'ì','ï',2}, {'í','Á',31}, {'í','Å',10}, {'í','Ç',1},
++{'í','É',44}, {'í','Ì',1}, {'í','Î',19}, {'í','Ï',93}, {'í','Ò',1}, {'í','Õ',6}, {'í','×',3}, {'í','Ù',29}, {'í','á',1}, {'í','é',1}, {'í','ï',1}, {'í','ð',1}, {'î','Á',56}, {'î','Å',111}, {'î','É',47}, {'î','Î',2},
++{'î','Ï',98}, {'î','Õ',100}, {'î','á',2}, {'î','é',3}, {'î','ú',1}, {'ï','Â',18}, {'ï','Ä',20}, {'ï','Ç',4}, {'ï','É',2}, {'ï','Ë',7}, {'ï','Ì',9}, {'ï','Í',2}, {'ï','Î',96}, {'ï','Ï',5}, {'ï','Ð',17}, {'ï','Ò',16},
++{'ï','Ó',21}, {'ï','Ô',35}, {'ï','×',1}, {'ï','Ú',2}, {'ï','Ü',6}, {'ï','Ý',1}, {'ï','Þ',13}, {'ï','å',2}, {'ï','î',6}, {'ï','ñ',1}, {'ï','ò',1}, {'ï','ô',1}, {'ï','õ',1}, {'ï','÷',1}, {'ð','Á',18}, {'ð','Å',148},
++{'ð','É',11}, {'ð','Ì',9}, {'ð','Ï',186}, {'ð','Ò',43}, {'ð','Õ',38}, {'ð','Ø',4}, {'ð','ó',3}, {'ñ','Â',4}, {'ñ','Ä',8}, {'ñ','Å',9}, {'ñ','Ç',5}, {'ñ','È',3}, {'ñ','É',4}, {'ñ','Ë',10}, {'ñ','Ì',5}, {'ñ','Í',6},
++{'ñ','Î',28}, {'ñ','Ï',24}, {'ñ','Ð',39}, {'ñ','Ò',4}, {'ñ','Ó',30}, {'ñ','Ô',7}, {'ñ','Õ',5}, {'ñ','Ö',2}, {'ñ','×',49}, {'ñ','Ú',12}, {'ñ','Û',3}, {'ñ','Ü',3}, {'ñ','Þ',4}, {'ñ','î',1}, {'ò','Á',22}, {'ò','Å',13},
++{'ò','É',5}, {'ò','Ï',13}, {'ò','Ò',1}, {'ò','Õ',1}, {'ò','ì',1}, {'ó','Á',19}, {'ó','Â',1}, {'ó','Ä',1}, {'ó','Å',56}, {'ó','È',2}, {'ó','É',16}, {'ó','Ë',22}, {'ó','Ì',11}, {'ó','Í',12}, {'ó','Î',6}, {'ó','Ï',26},
++{'ó','Ð',6}, {'ó','Ñ',1}, {'ó','Ò',6}, {'ó','Ó',3}, {'ó','Ô',35}, {'ó','Õ',4}, {'ó','×',4}, {'ó','Ù',1}, {'ó','Ü',1}, {'ó','Þ',2}, {'ó','á',2}, {'ó','ó',3}, {'ó','÷',1}, {'ô','Á',79}, {'ô','Å',18}, {'ô','É',17},
++{'ô','Ï',34}, {'ô','Ñ',3}, {'ô','Ò',11}, {'ô','Õ',31}, {'ô','×',1}, {'ô','Ø',2}, {'ô','Ù',55}, {'ô','á',4}, {'ô','å',1}, {'ô','é',1}, {'ô','ï',1}, {'õ','Â',2}, {'õ','Ä',3}, {'õ','Å',2}, {'õ','Ç',1}, {'õ','È',2},
++{'õ','É',1}, {'õ','Ê',2}, {'õ','Ë',1}, {'õ','Ì',2}, {'õ','Í',9}, {'õ','Î',5}, {'õ','Ï',1}, {'õ','Ð',1}, {'õ','Ó',9}, {'õ','Ô',8}, {'õ','Ö',4}, {'õ','×',3}, {'õ','Ú',4}, {'õ','Û',1}, {'õ','Þ',1}, {'ö','Á',5},
++{'ö','Ä',1}, {'ö','Å',14}, {'ö','É',5}, {'ö','Ï',2}, {'ö','Ò',1}, {'÷','Á',15}, {'÷','Ä',5}, {'÷','Å',98}, {'÷','Ç',3}, {'÷','É',7}, {'÷','Ë',5}, {'÷','Ì',8}, {'÷','Í',7}, {'÷','Î',6}, {'÷','Ï',107}, {'÷','Ð',18},
++{'÷','Ò',2}, {'÷','Ó',74}, {'÷','Ô',6}, {'÷','Õ',1}, {'÷','×',2}, {'÷','Ù',39}, {'÷','Ú',5}, {'÷','Ü',8}, {'÷','Þ',3}, {'÷','á',1}, {'÷','è',1}, {'÷','ë',2}, {'÷','í',1}, {'÷','ï',3}, {'÷','ð',8}, {'÷','ó',2},
++{'÷','û',2}, {'ú','Á',31}, {'ú','Å',1}, {'ú','É',2}, {'ú','Î',23}, {'ú','Ò',2}, {'ú','Õ',12}, {'ú','×',3}, {'ú','é',1}, {'û','Á',3}, {'û','Å',7}, {'û','É',8}, {'û','Ô',8}, {'û','Õ',7}, {'û','Û',1}, {'û','é',1},
++{'ü','Â',5}, {'ü','È',1}, {'ü','Ê',1}, {'ü','Ì',4}, {'ü','Î',1}, {'ü','Ò',10}, {'ü','Ó',1}, {'ü','Ô',61}, {'ü','×',1}, {'ü','ô',1}, {'þ','Å',46}, {'þ','Ô',60}, {'þ','Õ',8}, {'þ','Ø',1}
++};
++
++static int
++is_win_charset(char *txt, int len){
++ int j,ki=0,wi=0,d=839,l,koi_stat=0,win_stat=0,ws,ks;
++ unsigned char a,b;
++ l=strlen(txt);
++ if ((len)&&(l>len)) l=len;
++ for(j=0;j<l-1;j++){
++ a=txt[j];
++ b=txt[j+1];
++ //skip bottom half of table
++ if(a<128 || b<128)
++ continue;
++ wi=839;
++ ki=839;
++ ws=0;
++ ks=0;
++ d=839;
++ //binary search through lng_stats
++ do{
++ d>>=1;
++ if(!ws){
++ if(a==win[wi].a){
++ if(b==win[wi].b){
++ ws=1;
++ win_stat+=win[wi].rate;
++ }else if(b<win[wi].b){
++ wi-=d;
++ }else{ //b>win[wi].b
++ wi+=d;
++ }
++ }else if(a<win[wi].a){
++ wi-=d;
++ }else{ //a>win[wi].a
++ wi+=d;
++ }
++ }
++ if(!ks){
++ if(a==koi[ki].a){
++ if(b==koi[ki].b){
++ ks=1;
++ koi_stat+=koi[ki].rate;
++ }else if(b<koi[ki].b){
++ ki-=d;
++ }else{ //b>win[wi].b
++ ki+=d;
++ }
++ }else if(a<koi[ki].a){
++ ki-=d;
++ }else{ //a>win[wi].a
++ ki+=d;
++ }
++ }
++ }while(d);
++ }
++// fprintf(stderr,"\nwin %i, koi %i\n",win_stat,koi_stat);
++ if(win_stat>koi_stat)
++ return 1;
++ else
++ return 0;
++}
++
++/* Only CP1251 and KOI8-R still */
++int autocharset_russian(char *buf,int len) {
++ if (is_win_charset(buf,len)) return 0;
++ return 1;
++}
+diff -Naur xmms-1.2.8/libxmms/charset.c xmms-1.2.8-new/libxmms/charset.c
+--- xmms-1.2.8/libxmms/charset.c 1970-01-01 01:00:00.000000000 +0100
++++ xmms-1.2.8-new/libxmms/charset.c 2003-09-06 19:42:29.000000000 +0200
+@@ -0,0 +1,512 @@
++#include <stdio.h>
++#include <stdlib.h>
++#include <string.h>
++#include <errno.h>
++#include <glib.h>
++#include <iconv.h>
++#include <assert.h>
++
++#include "charset.h"
++#include "charset_config.h"
++
++#define LOCALE_STRING_LENGTH 64
++static char charset_default[LOCALE_STRING_LENGTH];
++static char xmms_charset_tmp[CHARSET_MAX_STRING_SIZE];
++
++static int xmms_charset_changed = -1;
++
++static int xmms_charset_id3;
++static int xmms_charset_output;
++static int xmms_charset_fs;
++static int xmms_charset_pl;
++
++static iconv_t iconv_id3=(iconv_t)-1; /* id3 to output */
++static iconv_t iconv_output=(iconv_t)-1; /* output to id3 */
++static iconv_t iconv_fs=(iconv_t)-1; /* fs to output */
++static iconv_t iconv_utf=(iconv_t)-1; /* id3 to UTF-8 */
++static iconv_t iconv_pl2fs=(iconv_t)-1; /* playlist to fs */
++static iconv_t iconv_fs2pl=(iconv_t)-1; /* fs to playlist */
++
++static int xmms_autocharset_mode = 0;
++static int xmms_autocharset_changed = 0;
++static iconv_t *xmms_autocharset = NULL;
++static iconv_t *xmms_autocharset_id2id = NULL;
++static iconv_t *xmms_autocharset_id2fs = NULL;
++static int xmms_autocharset_number = -1;
++
++static int recode_fsout;
++static int recode_fs2utf;
++#include "charset_auto_russian.h"
++
++extern int errno;
++
++/*****************************************************************************
++************************ Auto Recoding ***************************************
++*****************************************************************************/
++static int xmms_autocharset_number_charsets(int m) {
++ const char **encodings;
++ int i;
++
++ if (!m) m=xmms_autocharset_mode;
++ encodings=autocharset_list[m].encodings;
++ for (i=0;encodings[i];i++);
++ return i;
++}
++
++
++static void xmms_autocharset_iconv_close() {
++ int i;
++
++ for (i=0;i<xmms_autocharset_number;i++) {
++ if ((xmms_autocharset)&&(xmms_autocharset[i] != (iconv_t)-1)&&(xmms_autocharset[i] != (iconv_t)-2))
++ iconv_close(xmms_autocharset[i]);
++ if ((xmms_autocharset_id2id)&&(xmms_autocharset_id2id[i] != (iconv_t)-1)&&(xmms_autocharset_id2id[i] != (iconv_t)-2))
++ iconv_close(xmms_autocharset_id2id[i]);
++ if ((xmms_autocharset_id2fs)&&(xmms_autocharset_id2fs[i] != (iconv_t)-1)&&(xmms_autocharset_id2fs[i] != (iconv_t)-2))
++ iconv_close(xmms_autocharset_id2fs[i]);
++ }
++ if (xmms_autocharset) free(xmms_autocharset);
++ if (xmms_autocharset_id2id) free(xmms_autocharset_id2id);
++ if (xmms_autocharset_id2fs) free(xmms_autocharset_id2fs);
++
++ xmms_autocharset=NULL;
++ xmms_autocharset_id2id=NULL;
++ xmms_autocharset_id2fs=NULL;
++ xmms_autocharset_number = -1;
++}
++
++void xmms_autocharset_new_mode(int m) {
++ xmms_autocharset_mode=m;
++ xmms_autocharset_changed=1;
++}
++
++void xmms_autocharset_init(int m) {
++ xmms_autocharset_mode=m;
++ xmms_autocharset_changed=1;
++}
++
++void xmms_autocharset_free() {
++ xmms_autocharset_iconv_close();
++}
++
++static void xmms_autocharset_iconv_open() {
++ const char **encodings;
++ const char *output, *id3, *fs;
++ int i;
++
++ if ((!xmms_autocharset_changed)&&(!xmms_charset_changed)) return;
++ xmms_autocharset_iconv_close();
++ xmms_autocharset_changed=0;
++
++ xmms_autocharset_number = xmms_autocharset_number_charsets(0);
++ if (!xmms_autocharset_number) return;
++
++ xmms_autocharset = (iconv_t*)malloc((xmms_autocharset_number+1)*sizeof(iconv_t));
++ xmms_autocharset_id2id = (iconv_t*)malloc((xmms_autocharset_number+1)*sizeof(iconv_t));
++ xmms_autocharset_id2fs = (iconv_t*)malloc((xmms_autocharset_number+1)*sizeof(iconv_t));
++ if ((!xmms_autocharset)||(!xmms_autocharset_id2id)||(!xmms_autocharset_id2fs)) {
++ if (xmms_autocharset) {
++ free(xmms_autocharset);
++ xmms_autocharset=NULL;
++ }
++ if (xmms_autocharset_id2id) {
++ free(xmms_autocharset_id2id);
++ xmms_autocharset_id2id=NULL;
++ }
++ if (xmms_autocharset_id2fs) {
++ free(xmms_autocharset_id2fs);
++ xmms_autocharset_id2fs=NULL;
++ }
++ xmms_autocharset_number = -1;
++ perror("iconv: allocate memory");
++ return;
++ }
++
++
++ if (xmms_charset_output) output=charset_list[xmms_charset_output];
++ else output=charset_default;
++ if (xmms_charset_id3) id3=charset_list[xmms_charset_id3];
++ else id3=charset_default;
++ if (xmms_charset_fs) fs=charset_list[xmms_charset_fs];
++ else fs=output;
++
++ encodings = autocharset_list[xmms_autocharset_mode].encodings;
++
++ for (i=0;encodings[i];i++) {
++ if (strcmp(output,encodings[i]))
++ xmms_autocharset[i] = iconv_open(output,encodings[i]);
++ else
++ xmms_autocharset[i] = (iconv_t)-2;
++
++ if (strcmp(id3,encodings[i]))
++ xmms_autocharset_id2id[i] = iconv_open(id3,encodings[i]);
++ else
++ xmms_autocharset_id2id[i] = (iconv_t)-2;
++
++ if (strcmp(fs,encodings[i]))
++ xmms_autocharset_id2fs[i] = iconv_open(fs,encodings[i]);
++ else
++ xmms_autocharset_id2fs[i] = (iconv_t)-2;
++ }
++}
++
++
++static iconv_t xmms_autocharset_get(char *buf, int len) {
++ int i;
++ xmms_autocharset_iconv_open();
++ if ((xmms_autocharset)&&(xmms_autocharset_number>0)) {
++ i=autocharset_list[xmms_autocharset_mode].func(buf,len);
++ if (i<xmms_autocharset_number) return xmms_autocharset[i];
++ }
++ return (iconv_t)-1;
++}
++
++static iconv_t xmms_autocharset_get_id2id(char *buf, int len) {
++ int i;
++ xmms_autocharset_iconv_open();
++ if ((xmms_autocharset_id2id)&&(xmms_autocharset_number>0)) {
++ i=autocharset_list[xmms_autocharset_mode].func(buf,len);
++ if (i<xmms_autocharset_number) return xmms_autocharset_id2id[i];
++ }
++ return (iconv_t)-1;
++}
++
++static iconv_t xmms_autocharset_get_id2fs(char *buf, int len) {
++ int i;
++ xmms_autocharset_iconv_open();
++ if ((xmms_autocharset_id2fs)&&(xmms_autocharset_number>0)) {
++ i=autocharset_list[xmms_autocharset_mode].func(buf,len);
++ if (i<xmms_autocharset_number) return xmms_autocharset_id2fs[i];
++ }
++ return (iconv_t)-1;
++}
++
++/*****************************************************************************
++************************ Recoding ********************************************
++*****************************************************************************/
++
++static void xmms_charset_iconv_close() {
++ if (iconv_id3 != (iconv_t)-1) {
++ if (iconv_close(iconv_id3)<0) perror("iconv close: id3");
++ iconv_id3 = (iconv_t)-1;
++ }
++ if (iconv_output != (iconv_t)-1) {
++ if (iconv_close(iconv_output)<0) perror("iconv close: output");
++ iconv_output = (iconv_t)-1;
++ }
++ if (iconv_fs != (iconv_t)-1) {
++ if (iconv_close(iconv_fs)<0) perror("iconv close: fs");
++ iconv_fs = (iconv_t)-1;
++ }
++ if (iconv_fs2pl != (iconv_t)-1) {
++ if (iconv_close(iconv_fs2pl)<0) perror("iconv close: fs2pl");
++ iconv_fs2pl = (iconv_t)-1;
++ }
++ if (iconv_pl2fs != (iconv_t)-1) {
++ if (iconv_close(iconv_pl2fs)<0) perror("iconv close: pl2fs");
++ iconv_pl2fs = (iconv_t)-1;
++ }
++ if (iconv_utf != (iconv_t)-1) {
++ if (iconv_close(iconv_utf)<0) perror("iconv close: utf");
++ iconv_utf=(iconv_t)-1;
++ }
++}
++
++static void xmms_charset_iconv_open() {
++ const char *id3, *output, *fs, *pl;
++ assert(xmms_charset_changed>=0);
++ if (!xmms_charset_changed) return;
++
++ xmms_charset_iconv_close();
++ if (xmms_charset_id3) id3=charset_list[xmms_charset_id3];
++ else id3=charset_default;
++ if (xmms_charset_output) output=charset_list[xmms_charset_output];
++ else output=charset_default;
++ if (xmms_charset_fs) fs=charset_list[xmms_charset_fs];
++ else fs=output;
++ if (xmms_charset_pl) pl=charset_list[xmms_charset_pl];
++ else pl=id3;
++
++
++ if ((xmms_charset_id3 != xmms_charset_output)&&(strcmp(id3,output))) {
++ iconv_id3 = iconv_open(output, id3);
++ iconv_output = iconv_open(id3, output);
++ if (!xmms_charset_fs) iconv_fs = iconv_open(id3, fs);
++ }
++
++ if (strcmp(fs,id3)) {
++ iconv_fs = iconv_open(id3, fs);
++ }
++
++ if (strcmp(fs,pl)) {
++ iconv_fs2pl = iconv_open(pl, fs);
++ iconv_pl2fs = iconv_open(fs,pl);
++ }
++
++ if (strcmp(id3, "UTF-8")) {
++ iconv_utf = iconv_open("UTF-8",id3);
++ }
++
++ if ((xmms_charset_fs)&&(xmms_charset_fs!=xmms_charset_output)) {
++ recode_fsout=1;
++ } else {
++ recode_fsout=0;
++ }
++
++ if (strcmp(fs, "UTF-8")) {
++ recode_fs2utf=1;
++ } else {
++ recode_fs2utf=0;
++ }
++
++ xmms_charset_changed = 0;
++}
++
++
++static int xmms_charset_number(char *c) {
++ int i;
++ if (!c) return 0;
++ for (i=0;charset_list[i];i++)
++ if (strcmp(charset_list[i], c)==0) return i;
++ return 0; /* Default, - if not found */
++}
++
++static void xmms_charset_setdefault() {
++ char *str1, *str2;
++ str1 = getenv(ENV_VARIABLE);
++ if (str1) {
++ str2 = strrchr(str1,'.');
++ if (str2) strncpy(charset_default,str2+1,LOCALE_STRING_LENGTH);
++ else strncpy(charset_default,str1,LOCALE_STRING_LENGTH);
++ charset_default[LOCALE_STRING_LENGTH-1]=0;
++ str2 = strchr(charset_default,'@');
++ if (str2) *str2 = 0;
++ } else {
++ strcpy(charset_default,"UTF-8");
++ }
++}
++
++void xmms_charset_init(char *id3, char *output, char *fs, char *pl) {
++ if (xmms_charset_changed<0) xmms_charset_setdefault();
++
++ xmms_charset_changed = 1;
++ xmms_charset_id3 = xmms_charset_number(id3);
++ xmms_charset_output = xmms_charset_number(output);
++ xmms_charset_fs = xmms_charset_number(fs);
++ xmms_charset_pl = xmms_charset_number(pl);
++}
++
++void xmms_charset_free() {
++ xmms_charset_iconv_close();
++}
++
++void xmms_charset_new_charsets(int id3, int output, int fs, int pl) {
++ if (xmms_charset_changed<0)
++ xmms_charset_setdefault();
++ else
++ if ((xmms_charset_id3==id3)&&(xmms_charset_output==output)&&(xmms_charset_fs==fs)&&(xmms_charset_pl==pl)) return;
++
++ xmms_charset_changed = 1;
++ xmms_charset_id3 = id3;
++ xmms_charset_output = output;
++ xmms_charset_fs = fs;
++ xmms_charset_pl = pl;
++}
++
++// Multi-byte strings still unsupported
++static void iconv_copysymbol(char **in_buf, int *in_left, char **out_buf, int *out_left) {
++ if ((out_left>0)&&(in_left>0)) {
++ (**out_buf)=(**in_buf);
++ (*out_buf)++;
++ (*in_buf)++;
++ (*in_left)--;
++ (*out_left)--;
++ }
++}
++
++static char *xmms_charset_recode(iconv_t icnv, char *buf, int len, int *rlen) {
++ char *in_buf, *out_buf, *res, err;
++ int in_left, out_left, olen;
++ int errors=0;
++
++ assert(buf);
++
++ if (!len) {
++ len=strlen(buf);
++ } else {
++ olen=strlen(buf);
++ if (olen<len) len=olen;
++ }
++
++ if (iconv(icnv, NULL, NULL, NULL, NULL) == -1) {
++ perror( "iconv convert: initialize" );
++ return NULL;
++ }
++
++ in_buf = buf;
++ in_left = len;
++ out_buf = &xmms_charset_tmp;
++ out_left = CHARSET_MAX_STRING_SIZE;
++
++loop:
++ err=iconv(icnv, &in_buf, &in_left, &out_buf, &out_left);
++// printf("%s, %i %u\n",buf,err,errno);
++ if (err<0) {
++ if (errno==E2BIG) {
++ *(int*)(xmms_charset_tmp+(CHARSET_MAX_STRING_SIZE-sizeof(int)))=0;
++ } else if (errno==EILSEQ) {
++ if (errors<CHARSET_MAX_ERRORS) {
++ errors++;
++ iconv_copysymbol(&in_buf, &in_left, &out_buf, &out_left);
++ if (in_left>0) goto loop;
++ } else {
++ perror("iconv convert: invalid encoding?");
++ return NULL;
++ }
++ } else {
++ perror("iconv convert: convert");
++// perror(buf);
++ return NULL;
++ }
++ }
++
++ olen = CHARSET_MAX_STRING_SIZE - out_left;
++ res = g_malloc(olen+1);
++ if (!res) {
++ perror( "iconv convert: malloc" );
++ return NULL;
++ }
++
++ memcpy(res,xmms_charset_tmp,olen);
++ res[olen]=0;
++ if (rlen) *rlen=olen;
++ return res;
++}
++
++char *xmms_charset_recode_id3(char *buf, int len, int *rlen) {
++ iconv_t icnv;
++
++ xmms_charset_iconv_open();
++ icnv = xmms_autocharset_get(buf,len);
++
++ if (icnv == (iconv_t)-2) return NULL;
++
++ if (icnv != (iconv_t)-1)
++ return xmms_charset_recode(icnv, buf, len, rlen);
++
++ if (iconv_id3 != (iconv_t)-1)
++ return xmms_charset_recode(iconv_id3, buf, len, rlen);
++ return NULL;
++}
++
++char *xmms_charset_recode_output(char *buf, int len, int *rlen) {
++ xmms_charset_iconv_open();
++ if (iconv_output != (iconv_t)-1)
++ return xmms_charset_recode(iconv_output, buf, len, rlen);
++ return NULL;
++}
++
++char *xmms_charset_recode_fs(char *buf, int len, int *rlen) {
++ xmms_charset_iconv_open();
++ if (iconv_fs != (iconv_t)-1)
++ return xmms_charset_recode(iconv_fs, buf, len, rlen);
++ return NULL;
++}
++
++char *xmms_charset_recode_fsout(char *buf, int len, int *rlen) {
++ int olen;
++ char *str = NULL;
++
++ xmms_charset_iconv_open();
++ if (recode_fsout) {
++ if (iconv_fs != (iconv_t)-1) {
++ str = xmms_charset_recode(iconv_fs, buf, len, &olen);
++ if (!str) return NULL;
++ len=olen;
++ }
++ if (iconv_id3 != (iconv_t)-1) {
++ buf = xmms_charset_recode(iconv_id3, str ? str : buf, len, &olen);
++ if (str) free(str);
++ } else if (str) {
++ buf = str;
++ } else {
++ buf = NULL;
++ }
++ if (rlen) *rlen=olen;
++ return buf;
++ }
++ return NULL;
++}
++
++char *xmms_charset_recode_fs2pl(char *buf, int len, int *rlen) {
++ xmms_charset_iconv_open();
++ if (iconv_fs2pl != (iconv_t)-1)
++ return xmms_charset_recode(iconv_fs2pl, buf, len, rlen);
++ return NULL;
++}
++
++char *xmms_charset_recode_pl2fs(char *buf, int len, int *rlen) {
++ iconv_t icnv;
++
++ xmms_charset_iconv_open();
++ icnv = xmms_autocharset_get_id2fs(buf,len);
++
++ if (icnv==(iconv_t)-2) return NULL;
++
++ if (icnv != (iconv_t)-1)
++ return xmms_charset_recode(icnv, buf, len, rlen);
++
++ if (iconv_pl2fs != (iconv_t)-1)
++ return xmms_charset_recode(iconv_pl2fs, buf, len, rlen);
++ return NULL;
++}
++
++
++char *xmms_charset_recode_id2utf(char *buf, int len, int *rlen) {
++ char *str = NULL;
++ int olen;
++ iconv_t icnv;
++
++ xmms_charset_iconv_open();
++
++ icnv = xmms_autocharset_get_id2id(buf,len);
++ if ((icnv != (iconv_t)-1)&&(icnv != (iconv_t)-2)) {
++ str = xmms_charset_recode(icnv, buf, len, &olen);
++ if (!str) return NULL;
++ len = olen;
++ }
++
++ xmms_charset_iconv_open();
++ if (iconv_utf != (iconv_t)-1) {
++ buf=xmms_charset_recode(iconv_utf, str ? str : buf, len, rlen);
++ if (str) free(str);
++ return buf;
++ }
++
++ return NULL;
++}
++
++char *xmms_charset_recode_fs2utf(char *buf, int len, int *rlen) {
++ int olen;
++ char *str = NULL;
++
++ xmms_charset_iconv_open();
++ if (recode_fs2utf) {
++ if (iconv_fs != (iconv_t)-1) {
++ str = xmms_charset_recode(iconv_fs, buf, len, &olen);
++ if (!str) return NULL;
++ len=olen;
++ }
++ if (iconv_utf != (iconv_t)-1) {
++ buf = xmms_charset_recode(iconv_utf, str ? str : buf, len, &olen);
++ if (str) free(str);
++ } else if (str) {
++ buf = str;
++ } else {
++ buf = NULL;
++ }
++ if (rlen) *rlen=olen;
++ return buf;
++ }
++ return NULL;
++}
+diff -Naur xmms-1.2.8/libxmms/charset_config.h xmms-1.2.8-new/libxmms/charset_config.h
+--- xmms-1.2.8/libxmms/charset_config.h 1970-01-01 01:00:00.000000000 +0100
++++ xmms-1.2.8-new/libxmms/charset_config.h 2003-09-06 19:42:29.000000000 +0200
+@@ -0,0 +1,13 @@
++/* Misc configuration options */
++
++#define ENV_VARIABLE "LANG"
++#define CHARSET_MAX_STRING_SIZE 1024 /* 256 symbols in UTF-32 */
++#define CHARSET_MAX_ERRORS 3
++
++const char *charset_list[] = { "Default", "KOI8-R", "CP1251", "CP866", "UTF-8", NULL };
++
++const int autocharset_list_ni=2;
++const struct autocharset_list_t autocharset_list[] = {
++ {"Off", NULL, {NULL}},
++ {"Russian", &autocharset_russian, {"CP1251","KOI8-R"} },
++};
+diff -Naur xmms-1.2.8/libxmms/charset.h xmms-1.2.8-new/libxmms/charset.h
+--- xmms-1.2.8/libxmms/charset.h 1970-01-01 01:00:00.000000000 +0100
++++ xmms-1.2.8-new/libxmms/charset.h 2003-09-06 19:42:29.000000000 +0200
+@@ -0,0 +1,36 @@
++#ifndef XMMS_CHARSET_H
++#define XMMS_CHARSET_H
++
++#define AUTOCHARSET_MAX_ENCODINGS 5
++int autocharset_russian(char *buf,int len);
++
++struct autocharset_list_t {
++ char *title;
++ int (*func)(char *buf, int len);
++ char *encodings[AUTOCHARSET_MAX_ENCODINGS];
++};
++
++extern const char *charset_list[];
++extern const int autocharset_list_ni;
++extern const struct autocharset_list_t autocharset_list[];
++
++/* Initialization functions */
++void xmms_autocharset_new_mode(int m);
++void xmms_autocharset_init(int m);
++void xmms_autocharset_free();
++
++void xmms_charset_init(char *id3, char *output, char *fs, char *pl);
++void xmms_charset_free();
++void xmms_charset_new_charsets(int id3, int output, int fs, int pl);
++
++/* Recoding functions */
++char *xmms_charset_recode_id3(char *buf, int len, int *rlen);
++char *xmms_charset_recode_output(char *buf, int len, int *rlen);
++char *xmms_charset_recode_fs(char *buf, int len, int *rlen);
++char *xmms_charset_recode_fsout(char *buf, int len, int *rlen);
++char *xmms_charset_recode_id2utf(char *buf, int len, int *rlen);
++char *xmms_charset_recode_fs2utf(char *buf, int len, int *rlen);
++char *xmms_charset_recode_fs2pl(char *buf, int len, int *rlen);
++char *xmms_charset_recode_pl2fs(char *buf, int len, int *rlen);
++
++#endif /* XMMS_CHARSET_H */
+diff -Naur xmms-1.2.8/libxmms/Makefile.am xmms-1.2.8-new/libxmms/Makefile.am
+--- xmms-1.2.8/libxmms/Makefile.am 2003-09-04 23:00:18.000000000 +0200
++++ xmms-1.2.8-new/libxmms/Makefile.am 2003-09-06 20:11:31.000000000 +0200
+@@ -15,7 +15,9 @@
+ formatter.c formatter.h \
+ titlestring.c titlestring.h \
+ xentry.c xentry.h \
+-xconvert.c xconvert.h
++xconvert.c xconvert.h \
++charset.c charset.h \
++charset_auto_russian.h charset_config.h
+
+-xmmsinclude_HEADERS = configfile.h xmmsctrl.h dirbrowser.h util.h formatter.h titlestring.h
++xmmsinclude_HEADERS = configfile.h xmmsctrl.h dirbrowser.h util.h formatter.h titlestring.h charset.h charset_auto_russian.h
+
+diff -Naur xmms-1.2.8/libxmms/Makefile.in xmms-1.2.8-new/libxmms/Makefile.in
+--- xmms-1.2.8/libxmms/Makefile.in 2003-09-04 23:01:19.000000000 +0200
++++ xmms-1.2.8-new/libxmms/Makefile.in 2003-09-06 19:42:29.000000000 +0200
+@@ -101,10 +101,12 @@
+ formatter.c formatter.h \
+ titlestring.c titlestring.h \
+ xentry.c xentry.h \
+-xconvert.c xconvert.h
++xconvert.c xconvert.h \
++charset.c charset.h \
++charset_auto_russian.h charset_config.h
+
++xmmsinclude_HEADERS = configfile.h xmmsctrl.h dirbrowser.h util.h formatter.h titlestring.h charset.h charset_auto_russian.h
+
+-xmmsinclude_HEADERS = configfile.h xmmsctrl.h dirbrowser.h util.h formatter.h titlestring.h
+ ACLOCAL_M4 = $(top_srcdir)/aclocal.m4
+ mkinstalldirs = $(SHELL) $(top_srcdir)/../mkinstalldirs
+ CONFIG_HEADER = config.h
+@@ -118,7 +120,7 @@
+ LIBS = @LIBS@
+ libxmms_la_DEPENDENCIES =
+ libxmms_la_OBJECTS = configfile.lo xmmsctrl.lo dirbrowser.lo util.lo \
+-formatter.lo titlestring.lo xentry.lo xconvert.lo
++formatter.lo titlestring.lo xentry.lo xconvert.lo charset.lo
+ CFLAGS = @CFLAGS@
+ COMPILE = $(CC) $(DEFS) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CFLAGS) $(CFLAGS)
+ LTCOMPILE = $(LIBTOOL) --mode=compile $(CC) $(DEFS) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CFLAGS) $(CFLAGS)
+diff -Naur xmms-1.2.8/libxmms/titlestring.c xmms-1.2.8-new/libxmms/titlestring.c
+--- xmms-1.2.8/libxmms/titlestring.c 2002-01-08 15:09:08.000000000 +0100
++++ xmms-1.2.8-new/libxmms/titlestring.c 2003-09-06 19:42:29.000000000 +0200
+@@ -40,6 +40,8 @@
+ gchar *xmms_get_titlestring(gchar *fmt, TitleInput *input)
+ {
+ gchar c, *p, outbuf[256], convert[16], *string;
++ gchar *cstring = NULL;
++ gint stringlen;
+ gint numdigits, numpr, val, size, i;
+ gint f_left, f_space, f_zero, someflag, width, precision;
+ gboolean did_output = FALSE;
+@@ -164,9 +166,17 @@
+ goto Print_string;
+ case 'f':
+ string = VS(input, file_name);
++ if (string) {
++ cstring=xmms_charset_recode_fs(string,0,&stringlen);
++ if (cstring) string=cstring;
++ }
+ goto Print_string;
+ case 'F':
+ string = VS(input, file_path);
++ if (string) {
++ cstring=xmms_charset_recode_fs(string,0,&stringlen);
++ if (cstring) string=cstring;
++ }
+ goto Print_string;
+ case 'g':
+ string = VS(input, genre);
+@@ -208,6 +218,10 @@
+ while ((c = *string++) != '\0')
+ PUTCH(c);
+ }
++ if (cstring) {
++ g_free(cstring);
++ cstring=NULL;
++ }
+
+ RIGHTPAD(width - numpr);
+ break;
+diff -Naur xmms-1.2.8/xmms/charset.c xmms-1.2.8-new/xmms/charset.c
+--- xmms-1.2.8/xmms/charset.c 1970-01-01 01:00:00.000000000 +0100
++++ xmms-1.2.8-new/xmms/charset.c 2003-09-06 19:42:29.000000000 +0200
+@@ -0,0 +1,393 @@
++#include <stdio.h>
++#include <locale.h>
++#include <string.h>
++#include <gtk/gtk.h>
++#include "xmms.h"
++#include "libxmms/charset.h"
++#include "charset.h"
++
++#define CHARSET_MAX_STRING_SIZE 1024 /* 256 symbols in UTF-32 */
++static char xmms_charset_tmp[CHARSET_MAX_STRING_SIZE];
++
++extern TextBox *playlistwin_sinfo;
++
++static gint selected_id3_charset, selected_output_charset, selected_fs_charset;
++static gint selected_pl_charset;
++static gint selected_autocharset;
++static GtkWidget *prefswin_options_font_shade_entry, *prefswin_options_font_shade_browse;
++
++static void prefswin_font_shade_browse_ok(GtkWidget * w, gpointer data)
++{
++ GtkFontSelectionDialog *fontsel = GTK_FONT_SELECTION_DIALOG(data);
++ gchar *fontname;
++
++ fontname = gtk_font_selection_dialog_get_font_name(fontsel);
++
++ if (fontname)
++ gtk_entry_set_text(GTK_ENTRY(prefswin_options_font_shade_entry), fontname);
++
++ gtk_widget_destroy(GTK_WIDGET(fontsel));
++}
++
++static void prefswin_font_shade_browse_cb(GtkWidget * w, gpointer data)
++{
++ static GtkWidget *fontsel;
++
++ if (fontsel != NULL)
++ return;
++
++ fontsel = gtk_font_selection_dialog_new(_("Select playlist font:"));
++ gtk_font_selection_dialog_set_font_name(GTK_FONT_SELECTION_DIALOG(fontsel), gtk_entry_get_text(GTK_ENTRY(prefswin_options_font_shade_entry)));
++ gtk_signal_connect(GTK_OBJECT(GTK_FONT_SELECTION_DIALOG(fontsel)->ok_button), "clicked", GTK_SIGNAL_FUNC(prefswin_font_shade_browse_ok), fontsel);
++ gtk_signal_connect_object(GTK_OBJECT(GTK_FONT_SELECTION_DIALOG(fontsel)->cancel_button), "clicked", GTK_SIGNAL_FUNC(gtk_widget_destroy), GTK_OBJECT(fontsel));
++ gtk_signal_connect(GTK_OBJECT(fontsel), "destroy", GTK_SIGNAL_FUNC(gtk_widget_destroyed), &fontsel);
++ gtk_widget_show(fontsel);
++}
++
++
++static void prefswin_charset_cb(GtkWidget * w, gpointer item)
++{
++ int i;
++ i = GPOINTER_TO_INT(item);
++ if (i>=0x3FFFF) selected_pl_charset=i-0x3FFFF;
++ else if (i>=0x2FFFF) selected_autocharset=i-0x2FFFF;
++ else if (i>=0x1FFFF) selected_fs_charset=i-0x1FFFF;
++ else if (i>=0xFFFF) selected_output_charset=i-0xFFFF;
++ else selected_id3_charset=i;
++}
++
++static void charset_options_add(Config *cfg, GtkOptionMenu *cmenu1, GtkOptionMenu *cmenu2, GtkOptionMenu *cmenu3, GtkOptionMenu *cmenu4)
++{
++ int i;
++ GtkWidget *menu1, *item1;
++ GtkWidget *menu2, *item2;
++ GtkWidget *menu3, *item3;
++ GtkWidget *menu4, *item4;
++
++ menu1 = gtk_menu_new();
++ gtk_option_menu_remove_menu(cmenu1);
++ gtk_option_menu_set_menu(cmenu1, menu1);
++
++ menu2 = gtk_menu_new();
++ gtk_option_menu_remove_menu(cmenu2);
++ gtk_option_menu_set_menu(cmenu2, menu2);
++
++ menu3 = gtk_menu_new();
++ gtk_option_menu_remove_menu(cmenu3);
++ gtk_option_menu_set_menu(cmenu3, menu3);
++
++ menu4 = gtk_menu_new();
++ gtk_option_menu_remove_menu(cmenu4);
++ gtk_option_menu_set_menu(cmenu4, menu4);
++
++ selected_id3_charset=0;
++ selected_output_charset=0;
++ selected_fs_charset=0;
++ selected_pl_charset=0;
++
++ for (i=0;charset_list[i];i++) {
++ item1 = gtk_menu_item_new_with_label(charset_list[i]);
++ gtk_signal_connect(GTK_OBJECT(item1), "activate", GTK_SIGNAL_FUNC(prefswin_charset_cb), GINT_TO_POINTER(i));
++ gtk_widget_show(item1);
++ gtk_menu_append(GTK_MENU(menu1), item1);
++ item2 = gtk_menu_item_new_with_label(charset_list[i]);
++ gtk_signal_connect(GTK_OBJECT(item2), "activate", GTK_SIGNAL_FUNC(prefswin_charset_cb), GINT_TO_POINTER(i+0xFFFF));
++ gtk_widget_show(item2);
++ gtk_menu_append(GTK_MENU(menu2), item2);
++ item3 = gtk_menu_item_new_with_label(charset_list[i]);
++ gtk_signal_connect(GTK_OBJECT(item3), "activate", GTK_SIGNAL_FUNC(prefswin_charset_cb), GINT_TO_POINTER(i+0x1FFFF));
++ gtk_widget_show(item3);
++ gtk_menu_append(GTK_MENU(menu3), item3);
++ item4 = gtk_menu_item_new_with_label(charset_list[i]);
++ gtk_signal_connect(GTK_OBJECT(item4), "activate", GTK_SIGNAL_FUNC(prefswin_charset_cb), GINT_TO_POINTER(i+0x3FFFF));
++ gtk_widget_show(item4);
++ gtk_menu_append(GTK_MENU(menu4), item4);
++ if ((cfg->charset_id3)&&(!strcmp(cfg->charset_id3,charset_list[i]))) {
++ selected_id3_charset=i;
++ }
++ if ((cfg->charset_output)&&(!strcmp(cfg->charset_output,charset_list[i]))) {
++ selected_output_charset=i;
++ }
++ if ((cfg->charset_fs)&&(!strcmp(cfg->charset_fs,charset_list[i]))) {
++ selected_fs_charset=i;
++ }
++ if ((cfg->charset_pl)&&(!strcmp(cfg->charset_pl,charset_list[i]))) {
++ selected_pl_charset=i;
++ }
++ }
++
++ gtk_option_menu_set_history(cmenu1, selected_id3_charset);
++ gtk_option_menu_set_history(cmenu2, selected_output_charset);
++ gtk_option_menu_set_history(cmenu3, selected_fs_charset);
++ gtk_option_menu_set_history(cmenu4, selected_pl_charset);
++}
++
++static void autocharset_options_add(Config *cfg, GtkOptionMenu *cmenu)
++{
++ int i;
++ GtkWidget *menu, *item;
++
++ menu = gtk_menu_new();
++ gtk_option_menu_remove_menu(cmenu);
++ gtk_option_menu_set_menu(cmenu, menu);
++
++ selected_autocharset=0;
++
++
++ for (i=0;i<autocharset_list_ni;i++) {
++ item = gtk_menu_item_new_with_label(autocharset_list[i].title);
++ gtk_signal_connect(GTK_OBJECT(item), "activate", GTK_SIGNAL_FUNC(prefswin_charset_cb), GINT_TO_POINTER(i+0x2FFFF));
++ gtk_widget_show(item);
++ gtk_menu_append(GTK_MENU(menu), item);
++ if ((cfg->autocharset)&&(cfg->autocharset==i)) {
++ selected_autocharset=i;
++ }
++ }
++
++ gtk_option_menu_set_history(cmenu, selected_autocharset);
++}
++
++void charset_create_prefs_window(Config *cfg, GtkNotebook *prefswin_notebook, GtkWidget *prefswin_fonts_vbox) {
++ GtkWidget *prefswin_encodings_vbox;
++ GtkWidget *prefswin_fonts_shade_frame;
++ GtkWidget *options_font_shade_hbox, *options_font_shade_vbox;
++ GtkWidget *prefswin_charset_frame, *prefswin_charset_vbox;
++ GtkWidget *prefswin_charset_list1, *prefswin_charset_list2, *prefswin_charset_list3, *prefswin_charset_list4;
++ GtkWidget *prefswin_charset_hbox1, *prefswin_charset_hbox2, *prefswin_charset_hbox3, *prefswin_charset_hbox4;
++ GtkWidget *prefswin_charset_label1, *prefswin_charset_label2, *prefswin_charset_label3, *prefswin_charset_label4;
++ GtkWidget *prefswin_autocharset_frame, *prefswin_autocharset_vbox;
++ GtkWidget *prefswin_autocharset_list;
++ GtkWidget *prefswin_autocharset_hbox;
++ GtkWidget *prefswin_autocharset_label;
++
++ prefswin_fonts_shade_frame = gtk_frame_new(_("Shade"));
++ gtk_container_set_border_width(GTK_CONTAINER(prefswin_fonts_shade_frame), 5);
++ gtk_box_pack_start(GTK_BOX(prefswin_fonts_vbox), prefswin_fonts_shade_frame, FALSE, FALSE, 0);
++ options_font_shade_vbox = gtk_vbox_new(FALSE, 5);
++ gtk_container_border_width(GTK_CONTAINER(options_font_shade_vbox), 5);
++ gtk_container_add(GTK_CONTAINER(prefswin_fonts_shade_frame), options_font_shade_vbox);
++ options_font_shade_hbox = gtk_hbox_new(FALSE, 5);
++ gtk_box_pack_start_defaults(GTK_BOX(options_font_shade_vbox), options_font_shade_hbox);
++ prefswin_options_font_shade_entry = gtk_entry_new();
++ gtk_box_pack_start(GTK_BOX(options_font_shade_hbox), prefswin_options_font_shade_entry, TRUE, TRUE, 0);
++ prefswin_options_font_shade_browse = gtk_button_new_with_label(_("Browse"));
++ gtk_signal_connect(GTK_OBJECT(prefswin_options_font_shade_browse), "clicked", GTK_SIGNAL_FUNC(prefswin_font_shade_browse_cb), NULL);
++ gtk_widget_set_usize(prefswin_options_font_shade_browse, 85, 17);
++ gtk_box_pack_start(GTK_BOX(options_font_shade_hbox), prefswin_options_font_shade_browse, FALSE, TRUE, 0);
++
++ prefswin_encodings_vbox = gtk_vbox_new(FALSE, 0);
++
++ prefswin_charset_frame = gtk_frame_new(_("Encodings"));
++ gtk_box_pack_start(GTK_BOX(prefswin_encodings_vbox), prefswin_charset_frame, FALSE, FALSE, 0);
++ gtk_container_border_width(GTK_CONTAINER(prefswin_charset_frame), 5);
++ prefswin_charset_vbox = gtk_vbox_new(FALSE, 3);
++ gtk_container_add(GTK_CONTAINER(prefswin_charset_frame), prefswin_charset_vbox);
++ gtk_container_border_width(GTK_CONTAINER(prefswin_charset_vbox), 5);
++
++ prefswin_charset_hbox1 = gtk_hbox_new(FALSE, 10);
++ gtk_container_add(GTK_CONTAINER(prefswin_charset_vbox), prefswin_charset_hbox1);
++ gtk_container_border_width(GTK_CONTAINER(prefswin_charset_hbox1), 5);
++ prefswin_charset_label1 = gtk_label_new(_("ID3 Encoding"));
++ gtk_widget_set_usize(prefswin_charset_label1, 120, 17);
++ gtk_box_pack_start(GTK_BOX(prefswin_charset_hbox1), prefswin_charset_label1, FALSE, FALSE, 0);
++ prefswin_charset_list1 = gtk_option_menu_new();
++ gtk_box_pack_start(GTK_BOX(prefswin_charset_hbox1), prefswin_charset_list1, TRUE, TRUE, 0);
++
++ prefswin_charset_hbox3 = gtk_hbox_new(FALSE, 10);
++ gtk_container_add(GTK_CONTAINER(prefswin_charset_vbox), prefswin_charset_hbox3);
++ gtk_container_border_width(GTK_CONTAINER(prefswin_charset_hbox3), 5);
++ prefswin_charset_label3 = gtk_label_new(_("FileSystem Encoding"));
++ gtk_widget_set_usize(prefswin_charset_label3, 120, 17);
++ gtk_box_pack_start(GTK_BOX(prefswin_charset_hbox3), prefswin_charset_label3, FALSE, FALSE, 0);
++ prefswin_charset_list3 = gtk_option_menu_new();
++ gtk_box_pack_start(GTK_BOX(prefswin_charset_hbox3), prefswin_charset_list3, TRUE, TRUE, 0);
++
++ prefswin_charset_hbox4 = gtk_hbox_new(FALSE, 10);
++ gtk_container_add(GTK_CONTAINER(prefswin_charset_vbox), prefswin_charset_hbox4);
++ gtk_container_border_width(GTK_CONTAINER(prefswin_charset_hbox4), 5);
++ prefswin_charset_label4 = gtk_label_new(_("Playlist Encoding"));
++ gtk_widget_set_usize(prefswin_charset_label4, 120, 17);
++ gtk_box_pack_start(GTK_BOX(prefswin_charset_hbox4), prefswin_charset_label4, FALSE, FALSE, 0);
++ prefswin_charset_list4 = gtk_option_menu_new();
++ gtk_box_pack_start(GTK_BOX(prefswin_charset_hbox4), prefswin_charset_list4, TRUE, TRUE, 0);
++
++ prefswin_charset_hbox2 = gtk_hbox_new(FALSE, 10);
++ gtk_container_add(GTK_CONTAINER(prefswin_charset_vbox), prefswin_charset_hbox2);
++ gtk_container_border_width(GTK_CONTAINER(prefswin_charset_hbox2), 5);
++ prefswin_charset_label2 = gtk_label_new(_("Output Encoding"));
++ gtk_widget_set_usize(prefswin_charset_label2, 120, 17);
++ gtk_box_pack_start(GTK_BOX(prefswin_charset_hbox2), prefswin_charset_label2, FALSE, FALSE, 0);
++ prefswin_charset_list2 = gtk_option_menu_new();
++ gtk_box_pack_start(GTK_BOX(prefswin_charset_hbox2), prefswin_charset_list2, TRUE, TRUE, 0);
++
++ charset_options_add(cfg, GTK_OPTION_MENU(prefswin_charset_list1),GTK_OPTION_MENU(prefswin_charset_list2),GTK_OPTION_MENU(prefswin_charset_list3), GTK_OPTION_MENU(prefswin_charset_list4));
++
++ prefswin_autocharset_frame = gtk_frame_new(_("Encoding Autoselection"));
++ gtk_box_pack_start(GTK_BOX(prefswin_encodings_vbox), prefswin_autocharset_frame, FALSE, FALSE, 0);
++ gtk_container_border_width(GTK_CONTAINER(prefswin_autocharset_frame), 5);
++ prefswin_autocharset_vbox = gtk_vbox_new(FALSE, 3);
++ gtk_container_add(GTK_CONTAINER(prefswin_autocharset_frame), prefswin_autocharset_vbox);
++ gtk_container_border_width(GTK_CONTAINER(prefswin_autocharset_vbox), 5);
++
++ prefswin_autocharset_hbox = gtk_hbox_new(FALSE, 10);
++ gtk_container_add(GTK_CONTAINER(prefswin_autocharset_vbox), prefswin_autocharset_hbox);
++ gtk_container_border_width(GTK_CONTAINER(prefswin_autocharset_hbox), 5);
++ prefswin_autocharset_label = gtk_label_new(_("Language"));
++ gtk_widget_set_usize(prefswin_autocharset_label, 120, 17);
++ gtk_box_pack_start(GTK_BOX(prefswin_autocharset_hbox), prefswin_autocharset_label, FALSE, FALSE, 0);
++ prefswin_autocharset_list = gtk_option_menu_new();
++ gtk_box_pack_start(GTK_BOX(prefswin_autocharset_hbox), prefswin_autocharset_list, TRUE, TRUE, 0);
++
++ autocharset_options_add(cfg, GTK_OPTION_MENU(prefswin_autocharset_list));
++
++ gtk_notebook_append_page(prefswin_notebook, prefswin_encodings_vbox, gtk_label_new(_("Encodings")));
++}
++
++void charset_prefswin_apply_changes(Config *cfg) {
++ if (cfg->charset_id3) g_free(cfg->charset_id3);
++ if (cfg->charset_output) g_free(cfg->charset_output);
++ if (cfg->charset_fs) g_free(cfg->charset_fs);
++ if (cfg->charset_pl) g_free(cfg->charset_pl);
++ cfg->charset_id3=g_strdup(charset_list[selected_id3_charset]);
++ cfg->charset_output=g_strdup(charset_list[selected_output_charset]);
++ cfg->charset_fs=g_strdup(charset_list[selected_fs_charset]);
++ cfg->charset_pl=g_strdup(charset_list[selected_pl_charset]);
++ xmms_charset_new_charsets(selected_id3_charset, selected_output_charset, selected_fs_charset, selected_pl_charset);
++ cfg->autocharset=selected_autocharset;
++ xmms_autocharset_new_mode(selected_autocharset);
++
++ if (cfg->shade_font) g_free(cfg->shade_font);
++ cfg->shade_font = g_strdup(gtk_entry_get_text(GTK_ENTRY(prefswin_options_font_shade_entry)));
++ textbox_set_xfont(playlistwin_sinfo, cfg->mainwin_use_xfont, cfg->shade_font);
++}
++
++void charset_show_prefs_window(Config *cfg) {
++ gtk_entry_set_text(GTK_ENTRY(prefswin_options_font_shade_entry), cfg->shade_font);
++}
++
++void charset_create_playlist_window(Config *cfg) {
++ textbox_set_xfont(playlistwin_sinfo, cfg->mainwin_use_xfont, cfg->shade_font);
++}
++/*****************************************************************************
++************************ Initialization / Deinitilization ********************
++*****************************************************************************/
++void charset_free(Config *cfg) {
++ xmms_charset_free();
++ xmms_autocharset_free();
++}
++
++void charset_init(Config *cfg) {
++ xmms_charset_init(cfg->charset_id3, cfg->charset_output, cfg->charset_fs, cfg->charset_pl);
++ xmms_autocharset_init(cfg->autocharset);
++}
++
++void charset_read_config(ConfigFile *cfgfile, Config *cfg) {
++ cfg->autocharset=0;
++ xmms_cfg_read_string(cfgfile, "xmms", "shade_font", &cfg->shade_font);
++ xmms_cfg_read_string(cfgfile, "xmms", "charset_id3", &cfg->charset_id3);
++ xmms_cfg_read_string(cfgfile, "xmms", "charset_output", &cfg->charset_output);
++ xmms_cfg_read_string(cfgfile, "xmms", "charset_fs", &cfg->charset_fs);
++ xmms_cfg_read_string(cfgfile, "xmms", "charset_pl", &cfg->charset_pl);
++ xmms_cfg_read_int(cfgfile,"xmms","autocharset",&cfg->autocharset);
++}
++
++void charset_write_config(ConfigFile *cfgfile, Config *cfg) {
++ xmms_cfg_write_string(cfgfile, "xmms", "shade_font", cfg->shade_font);
++ xmms_cfg_write_string(cfgfile, "xmms", "charset_id3", cfg->charset_id3);
++ xmms_cfg_write_string(cfgfile, "xmms", "charset_output", cfg->charset_output);
++ xmms_cfg_write_string(cfgfile, "xmms", "charset_fs", cfg->charset_fs);
++ xmms_cfg_write_string(cfgfile, "xmms", "charset_pl", cfg->charset_pl);
++ xmms_cfg_write_int(cfgfile,"xmms","autocharset",cfg->autocharset);
++}
++
++void charset_default_config(Config *cfg) {
++ if (cfg->shade_font && strlen(cfg->shade_font) == 0)
++ {
++ g_free(cfg->shade_font);
++ cfg->shade_font = NULL;
++ }
++ if (cfg->shade_font == NULL)
++ cfg->shade_font = g_strdup("-misc-fixed-medium-r-*-*-7-*");
++}
++
++/*****************************************************************************
++*********************************** Sort *************************************
++*****************************************************************************/
++gchar *saved_locale;
++
++static void charset_sort_null_entry(gpointer entry, gpointer data) {
++ ((PlaylistEntry*)entry)->sort=NULL;
++}
++
++static void charset_sort_free_entry(gpointer entry, gpointer data) {
++ if (((PlaylistEntry*)entry)->sort)
++ g_free(((PlaylistEntry*)entry)->sort);
++}
++
++void charset_sort_prepare(GList *playlist) {
++ g_list_foreach(playlist, charset_sort_null_entry, NULL);
++ saved_locale=g_strdup(setlocale(LC_COLLATE,NULL));
++ setlocale(LC_COLLATE,"UTF-8");
++}
++
++void charset_sort_free(GList *playlist) {
++ g_list_foreach(playlist, charset_sort_free_entry, NULL);
++ if (saved_locale) {
++ setlocale(LC_COLLATE,saved_locale);
++ g_free(saved_locale);
++ }
++}
++
++char *charset_sort_convert(PlaylistEntry *entry, gchar *title, int fs) {
++ char *str;
++ if (entry->sort) return entry->sort;
++ if (fs)
++ str=xmms_charset_recode_fs2utf(title,0,NULL);
++ else
++ str=xmms_charset_recode_id2utf(title,0,NULL);
++
++ if (!str)
++ entry->sort=g_strdup(title);
++ else {
++ strxfrm(xmms_charset_tmp,str,CHARSET_MAX_STRING_SIZE);
++// *(int*)(xmms_charset_tmp+(CHARSET_MAX_STRING_SIZE-sizeof(int)))=0;
++ xmms_charset_tmp[CHARSET_MAX_STRING_SIZE-1]=0;
++ entry->sort=g_strdup(xmms_charset_tmp);
++ g_free(str);
++ }
++
++ if (entry->sort)
++ return entry->sort;
++
++ return title;
++}
++
++/*****************************************************************************
++*********************************** Misc *************************************
++*****************************************************************************/
++
++void textbox_set_ctext(TextBox * tb, gchar * text)
++{
++ gchar *rtext;
++
++ lock_widget(tb);
++
++ rtext=xmms_charset_recode_id3(text,0,NULL);
++ if (rtext) text=rtext;
++ if (tb->tb_text)
++ {
++ if (!strcmp(text, tb->tb_text))
++ {
++ if (rtext) free(rtext);
++ unlock_widget(tb);
++ return;
++ }
++ g_free(tb->tb_text);
++ }
++
++ tb->tb_text = g_strdup(text);
++ if (rtext) free(rtext);
++
++ unlock_widget(tb);
++ draw_widget(tb);
++}
+diff -Naur xmms-1.2.8/xmms/charset.h xmms-1.2.8-new/xmms/charset.h
+--- xmms-1.2.8/xmms/charset.h 1970-01-01 01:00:00.000000000 +0100
++++ xmms-1.2.8-new/xmms/charset.h 2003-09-06 19:42:29.000000000 +0200
+@@ -0,0 +1,19 @@
++#include "libxmms/configfile.h"
++#include "libxmms/charset.h"
++void charset_create_prefs_window(Config *cfg, GtkNotebook *prefswin_notebook, GtkWidget *prefswin_fonts_vbox);
++void charset_show_prefs_window(Config *cfg);
++void charset_prefswin_apply_changes(Config *cfg);
++void charset_create_playlist_window(Config *cfg);
++
++void charset_init(Config *cfg);
++void charset_free(Config *cfg);
++
++void charset_read_config(ConfigFile *cfgfile, Config *cfg);
++void charset_default_config(Config *cfg);
++void charset_write_config(ConfigFile *cfgfile, Config *cfg);
++
++void charset_sort_prepare(GList *playlist);
++void charset_sort_free(GList *playlist);
++char *charset_sort_convert(PlaylistEntry *entry, gchar *title, int fs);
++
++void textbox_set_ctext(TextBox * tb, gchar * text);
+diff -Naur xmms-1.2.8/xmms/input.c xmms-1.2.8-new/xmms/input.c
+--- xmms-1.2.8/xmms/input.c 2003-07-14 15:24:28.000000000 +0200
++++ xmms-1.2.8-new/xmms/input.c 2003-09-06 19:42:29.000000000 +0200
+@@ -22,6 +22,7 @@
+ #include "libxmms/titlestring.h"
+ #include "libxmms/util.h"
+ #include "libxmms/xentry.h"
++#include "libxmms/charset.h"
+
+ static pthread_mutex_t vis_mutex = PTHREAD_MUTEX_INITIALIZER;
+
+@@ -343,7 +344,7 @@
+ {
+ GList *node;
+ InputPlugin *ip = NULL;
+-
++
+ node = get_input_list();
+ while (node)
+ {
+@@ -371,8 +372,11 @@
+
+ (*title) = xmms_get_titlestring(xmms_get_gentitle_format(),
+ input);
+- if ( (*title) == NULL )
+- (*title) = g_strdup(input->file_name);
++ if ( (*title) == NULL ) {
++ (*title) = xmms_charset_recode_fs(input->file_name,0,NULL);
++ if (!*title) (*title) = g_strdup(input->file_name);
++ }
++
+ (*length) = -1;
+ g_free(temp);
+ g_free(input);
+diff -Naur xmms-1.2.8/xmms/main.c xmms-1.2.8-new/xmms/main.c
+--- xmms-1.2.8/xmms/main.c 2003-09-02 15:01:40.000000000 +0200
++++ xmms-1.2.8-new/xmms/main.c 2003-09-06 19:42:29.000000000 +0200
+@@ -35,6 +35,7 @@
+ #include "libxmms/xmmsctrl.h"
+ #include "libxmms/util.h"
+ #include "libxmms/dirbrowser.h"
++#include "charset.h"
+ #include "xmms_mini.xpm"
+
+ GtkWidget *mainwin, *mainwin_url_window = NULL, *mainwin_dir_browser = NULL;
+@@ -377,7 +378,6 @@
+
+ cfg.gentitle_format = NULL;
+
+-
+ filename = g_strconcat(g_get_home_dir(), "/.xmms/config", NULL);
+ cfgfile = xmms_cfg_open_file(filename);
+ if (cfgfile)
+@@ -428,6 +428,7 @@
+ xmms_cfg_read_boolean(cfgfile, "xmms", "use_fontsets", &cfg.use_fontsets);
+ xmms_cfg_read_boolean(cfgfile, "xmms", "mainwin_use_xfont", &cfg.mainwin_use_xfont);
+ xmms_cfg_read_string(cfgfile, "xmms", "mainwin_font", &cfg.mainwin_font);
++ charset_read_config(cfgfile,&cfg);
+ xmms_cfg_read_int(cfgfile, "xmms", "playlist_position", &cfg.playlist_position);
+ xmms_cfg_read_int(cfgfile, "xmms", "equalizer_x", &cfg.equalizer_x);
+ xmms_cfg_read_int(cfgfile, "xmms", "equalizer_y", &cfg.equalizer_y);
+@@ -514,6 +515,8 @@
+ if (cfg.eqpreset_extension == NULL)
+ cfg.eqpreset_extension = g_strdup("preset");
+
++ charset_default_config(&cfg);
++
+ g_free(filename);
+ }
+
+@@ -594,6 +597,7 @@
+ xmms_cfg_write_boolean(cfgfile, "xmms", "use_fontsets", cfg.use_fontsets);
+ xmms_cfg_write_boolean(cfgfile, "xmms", "mainwin_use_xfont", cfg.mainwin_use_xfont);
+ xmms_cfg_write_string(cfgfile, "xmms", "mainwin_font", cfg.mainwin_font);
++ charset_write_config(cfgfile,&cfg);
+ xmms_cfg_write_int(cfgfile, "xmms", "playlist_position", get_playlist_position());
+ /* dock_get_widget_pos(equalizerwin, &cfg.equalizer_x, &cfg.equalizer_y); */
+ xmms_cfg_write_int(cfgfile, "xmms", "equalizer_x", cfg.equalizer_x);
+@@ -893,6 +897,7 @@
+ gtk_timeout_remove(mainwin_timeout_tag);
+ util_set_cursor(NULL);
+ save_config();
++ charset_free(&cfg);
+ cleanup_ctrlsocket();
+ playlist_stop_get_info_thread();
+ playlist_clear();
+@@ -991,11 +996,11 @@
+
+ if ((text = input_get_info_text()) != NULL)
+ {
+- textbox_set_text(mainwin_info, text);
++ textbox_set_ctext(mainwin_info, text);
+ }
+ else if ((text = playlist_get_info_text()) != NULL)
+ {
+- textbox_set_text(mainwin_info, text);
++ textbox_set_ctext(mainwin_info, text);
+ g_free(text);
+ }
+ else
+@@ -3491,6 +3496,7 @@
+ #endif
+
+ read_config();
++ charset_init(&cfg);
+
+ if (geteuid() == 0)
+ {
+diff -Naur xmms-1.2.8/xmms/main.h xmms-1.2.8-new/xmms/main.h
+--- xmms-1.2.8/xmms/main.h 2002-10-06 18:35:27.000000000 +0200
++++ xmms-1.2.8-new/xmms/main.h 2003-09-06 19:42:29.000000000 +0200
+@@ -19,6 +19,7 @@
+ */
+ #ifndef MAIN_H
+ #define MAIN_H
++#include "libxmms/charset.h"
+
+ typedef enum
+ {
+@@ -60,6 +61,9 @@
+ gint mouse_change;
+ gboolean playlist_transparent;
+ gchar *gentitle_format;
++ gchar *charset_id3, *charset_output, *charset_fs, *charset_pl;
++ gint autocharset;
++ gchar *shade_font;
+ }
+ Config;
+
+diff -Naur xmms-1.2.8/xmms/Makefile.am xmms-1.2.8-new/xmms/Makefile.am
+--- xmms-1.2.8/xmms/Makefile.am 2003-08-11 15:47:41.000000000 +0200
++++ xmms-1.2.8-new/xmms/Makefile.am 2003-09-06 19:42:29.000000000 +0200
+@@ -56,7 +56,8 @@
+ getopt.c getopt1.c getopt.h \
+ urldecode.c urldecode.h \
+ dnd.h \
+-mkdtemp.c
++mkdtemp.c \
++charset.c charset.h
+
+ EXTRA_DIST= xmms_logo.xpm xmms_mini.xpm xmms.desktop xmms.wmconfig
+
+diff -Naur xmms-1.2.8/xmms/Makefile.in xmms-1.2.8-new/xmms/Makefile.in
+--- xmms-1.2.8/xmms/Makefile.in 2003-09-04 23:01:20.000000000 +0200
++++ xmms-1.2.8-new/xmms/Makefile.in 2003-09-06 19:42:29.000000000 +0200
+@@ -228,8 +228,8 @@
+ getopt.c getopt1.c getopt.h \
+ urldecode.c urldecode.h \
+ dnd.h \
+-mkdtemp.c
+-
++mkdtemp.c \
++charset.c charset.h
+
+ EXTRA_DIST = xmms_logo.xpm xmms_mini.xpm xmms.desktop xmms.wmconfig
+
+@@ -257,7 +257,7 @@
+ main.$(OBJEXT) skinwin.$(OBJEXT) prefswin.$(OBJEXT) \
+ playlistwin.$(OBJEXT) equalizer.$(OBJEXT) hints.$(OBJEXT) \
+ about.$(OBJEXT) sm.$(OBJEXT) getopt.$(OBJEXT) getopt1.$(OBJEXT) \
+-urldecode.$(OBJEXT) mkdtemp.$(OBJEXT)
++urldecode.$(OBJEXT) mkdtemp.$(OBJEXT) charset.$(OBJEXT)
+ xmms_DEPENDENCIES = $(top_builddir)/libxmms/libxmms.la
+ CFLAGS = @CFLAGS@
+ COMPILE = $(CC) $(DEFS) $(INCLUDES) $(AM_CPPFLAGS) $(CPPFLAGS) $(AM_CFLAGS) $(CFLAGS)
+diff -Naur xmms-1.2.8/xmms/playlist.c xmms-1.2.8-new/xmms/playlist.c
+--- xmms-1.2.8/xmms/playlist.c 2003-09-02 15:01:40.000000000 +0200
++++ xmms-1.2.8-new/xmms/playlist.c 2003-09-06 19:42:29.000000000 +0200
+@@ -20,6 +20,7 @@
+ #include "xmms.h"
+ #include <time.h>
+ #include "libxmms/util.h"
++#include "charset.h"
+ #include <sys/stat.h>
+ #include <unistd.h>
+
+@@ -939,6 +940,7 @@
+ char *playlist_get_info_text(void)
+ {
+ char *text, *title, *tmp, *numbers, *length;
++ char *ctitle;
+
+ PL_LOCK();
+ if (!playlist_position)
+@@ -947,10 +949,17 @@
+ return NULL;
+ }
+
+- if (playlist_position->title)
++ if (playlist_position->title) {
++ ctitle = xmms_charset_recode_id3(playlist_position->title,0,NULL);
++ if (ctitle) title=ctitle;
++ else
+ title = playlist_position->title;
+- else
++ } else {
++ ctitle = xmms_charset_recode_fsout(g_basename(playlist_position->filename),0,NULL);
++ if (ctitle) title=ctitle;
++ else
+ title = g_basename(playlist_position->filename);
++ }
+
+ /*
+ * If the user don't want numbers in the playlist, don't
+@@ -972,6 +981,7 @@
+ text = g_strdup_printf("%s%s%s", numbers, title, length);
+ g_free(numbers);
+ g_free(length);
++ if (ctitle) g_free(ctitle);
+
+ PL_UNLOCK();
+
+@@ -1007,6 +1017,7 @@
+ {
+ GList *node;
+ FILE *file;
++ gchar *fn, *cfn;
+
+ if ((file = fopen(filename, "w")) == NULL)
+ return FALSE;
+@@ -1025,10 +1036,13 @@
+ while (node)
+ {
+ PlaylistEntry *entry = node->data;
++ cfn=xmms_charset_recode_fs2pl(entry->filename,0,NULL);
++ if (cfn) fn=cfn;
++ else fn=entry->filename;
+ if (is_pls)
+ fprintf(file, "File%d=%s\n",
+ g_list_position(playlist, node) + 1,
+- entry->filename);
++ fn);
+ else
+ {
+ if (entry->title && cfg.use_pl_metadata)
+@@ -1043,8 +1057,9 @@
+ fprintf(file, "#EXTINF:%d,%s\n",
+ seconds, entry->title);
+ }
+- fprintf(file, "%s\n", entry->filename);
++ fprintf(file, "%s\n", fn);
+ }
++ if (cfn) g_free(cfn);
+ node = g_list_next(node);
+ }
+ PL_UNLOCK();
+@@ -1121,6 +1136,7 @@
+ {
+ FILE *file;
+ char *line, *ext;
++ char *cline;
+ guint entries = 0;
+ int linelen = 1024;
+ gboolean extm3u = FALSE;
+@@ -1147,8 +1163,10 @@
+ "playlist", key);
+ if (line != NULL)
+ {
+- playlist_load_ins_file(line, filename, pos,
++ cline = xmms_charset_recode_pl2fs(line,0,NULL);
++ playlist_load_ins_file(cline ? cline : line, filename, pos,
+ NULL, -1);
++ if (cline) g_free(cline);
+ entries++;
+ if (pos >= 0)
+ pos++;
+@@ -1215,7 +1233,9 @@
+ ext_info = NULL;
+ }
+
+- playlist_load_ins_file(line, filename, pos, ext_title, ext_len);
++ cline = xmms_charset_recode_pl2fs(line,linelen,NULL);
++ playlist_load_ins_file(cline ? cline : line, filename, pos, ext_title, ext_len);
++ if (cline) g_free(cline);
+
+ g_free(ext_title);
+ ext_title = NULL;
+@@ -1308,19 +1328,27 @@
+
+ if (entry->title == NULL && entry->length == -1)
+ {
+- if (playlist_get_info_entry(entry))
++ if (playlist_get_info_entry(entry)) {
++ title = xmms_charset_recode_id3(entry->title,0,NULL);
++ if (!title)
+ title = g_strdup(entry->title);
++ }
+
+ PL_UNLOCK();
+ }
+ else
+ {
++ title = xmms_charset_recode_id3(entry->title,0,NULL);
++ if (!title)
+ title = g_strdup(entry->title);
+ PL_UNLOCK();
+ }
+
+- if (title == NULL)
++ if (title == NULL) {
++ title = xmms_charset_recode_fsout(g_basename(filename),0,NULL);
++ if (!title)
+ title = g_strdup(g_basename(filename));
++ }
+
+ g_free(filename);
+
+@@ -1366,6 +1394,7 @@
+ static int playlist_sort_by_title_cmpfunc(PlaylistEntry * a, PlaylistEntry * b)
+ {
+ char *a_title, *b_title;
++ int a_fs=0, b_fs=0;
+
+ if (a->title)
+ a_title = a->title;
+@@ -1375,6 +1404,7 @@
+ a_title = strrchr(a->filename, '/') + 1;
+ else
+ a_title = a->filename;
++ a_fs=1;
+ }
+
+ if (b->title)
+@@ -1385,15 +1415,21 @@
+ b_title = strrchr(b->filename, '/') + 1;
+ else
+ b_title = b->filename;
+-
++ b_fs=1;
+ }
++
++ a_title=charset_sort_convert(a,a_title,a_fs);
++ b_title=charset_sort_convert(b,b_title,b_fs);
++
+ return strcasecmp(a_title, b_title);
+ }
+
+ void playlist_sort_by_title(void)
+ {
+ PL_LOCK();
++ charset_sort_prepare(playlist);
+ playlist = g_list_sort(playlist, (GCompareFunc) playlist_sort_by_title_cmpfunc);
++ charset_sort_free(playlist);
+ PL_UNLOCK();
+ }
+
+@@ -1411,13 +1447,18 @@
+ else
+ b_filename = b->filename;
+
++ a_filename=charset_sort_convert(a,a_filename,1);
++ b_filename=charset_sort_convert(b,b_filename,1);
++
+ return strcasecmp(a_filename, b_filename);
+ }
+
+ void playlist_sort_by_filename(void)
+ {
+ PL_LOCK();
++ charset_sort_prepare(playlist);
+ playlist = g_list_sort(playlist, (GCompareFunc) playlist_sort_by_filename_cmpfunc);
++ charset_sort_free(playlist);
+ PL_UNLOCK();
+ }
+
+@@ -1453,13 +1494,15 @@
+
+ static int playlist_sort_by_path_cmpfunc(PlaylistEntry * a, PlaylistEntry * b)
+ {
+- return playlist_sort_str_by_path_cmpfunc(a->filename, b->filename);
++ return playlist_sort_str_by_path_cmpfunc(charset_sort_convert(a,a->filename,1), charset_sort_convert(b,b->filename,1));
+ }
+
+ void playlist_sort_by_path(void)
+ {
+ PL_LOCK();
++ charset_sort_prepare(playlist);
+ playlist = g_list_sort(playlist, (GCompareFunc) playlist_sort_by_path_cmpfunc);
++ charset_sort_free(playlist);
+ PL_UNLOCK();
+ }
+
+@@ -1549,21 +1592,27 @@
+ void playlist_sort_selected_by_title(void)
+ {
+ PL_LOCK();
++ charset_sort_prepare(playlist);
+ playlist = playlist_sort_selected(playlist, (GCompareFunc) playlist_sort_by_title_cmpfunc);
++ charset_sort_free(playlist);
+ PL_UNLOCK();
+ }
+
+ void playlist_sort_selected_by_filename(void)
+ {
+ PL_LOCK();
++ charset_sort_prepare(playlist);
+ playlist = playlist_sort_selected(playlist, (GCompareFunc) playlist_sort_by_filename_cmpfunc);
++ charset_sort_free(playlist);
+ PL_UNLOCK();
+ }
+
+ void playlist_sort_selected_by_path(void)
+ {
+ PL_LOCK();
++ charset_sort_prepare(playlist);
+ playlist = playlist_sort_selected(playlist, (GCompareFunc) playlist_sort_by_path_cmpfunc);
++ charset_sort_free(playlist);
+ PL_UNLOCK();
+ }
+
+diff -Naur xmms-1.2.8/xmms/playlist.h xmms-1.2.8-new/xmms/playlist.h
+--- xmms-1.2.8/xmms/playlist.h 2003-06-09 15:22:10.000000000 +0200
++++ xmms-1.2.8-new/xmms/playlist.h 2003-09-06 19:42:29.000000000 +0200
+@@ -24,6 +24,7 @@
+ {
+ gchar *filename;
+ gchar *title;
++ gchar *sort;
+ gint length;
+ gboolean selected;
+ }
+diff -Naur xmms-1.2.8/xmms/playlist_list.c xmms-1.2.8-new/xmms/playlist_list.c
+--- xmms-1.2.8/xmms/playlist_list.c 2003-06-09 15:22:10.000000000 +0200
++++ xmms-1.2.8-new/xmms/playlist_list.c 2003-09-06 19:42:29.000000000 +0200
+@@ -287,6 +287,7 @@
+ {
+ GdkWChar *wtext;
+ int len, newlen;
++ gchar *rtext;
+ /*
+ * Convert the string to a wide character string to avoid
+ * destroying multibyte strings, when converting underscores,
+@@ -296,6 +297,10 @@
+ * Allocate some extra space, we might extend it by one
+ * character below
+ */
++
++ rtext = xmms_charset_recode_id3(text,0,NULL);
++ if (rtext) text = rtext;
++
+ wtext = g_malloc((strlen(text) + 3) * sizeof(GdkWChar));
+ len = gdk_mbstowcs(wtext, text, strlen(text) + 1);
+ if (len == -1)
+@@ -312,6 +317,7 @@
+ if (wtext[i] == L'_')
+ wtext[i] = L' ';
+ }
++ if (rtext) g_free(rtext);
+
+ if (cfg.convert_twenty && len > 2)
+ {
+@@ -361,6 +367,7 @@
+ {
+ int len;
+ char *tmp;
++ gchar *rtext;
+
+ if (cfg.convert_underscore)
+ while ((tmp = strchr(text, '_')) != NULL)
+@@ -384,7 +391,10 @@
+ text[len] = '\0';
+ }
+
++ rtext = xmms_charset_recode_id3(text,0,NULL);
++ if (rtext) text = rtext;
+ gdk_draw_text(pl->pl_widget.parent, font, pl->pl_widget.gc, pl->pl_widget.x, pl->pl_widget.y + line * pl->pl_fheight + font->ascent, text, len);
++ if (rtext) g_free(rtext);
+ }
+
+ void playlist_list_draw(Widget * w)
+@@ -396,6 +406,7 @@
+ int width, height;
+ char *text, *title;
+ int i, tw, max_first;
++ gchar *ctitle = NULL;
+
+ gc = pl->pl_widget.gc;
+ width = pl->pl_widget.width;
+@@ -453,8 +464,11 @@
+
+ if (entry->title)
+ title = entry->title;
+- else
+- title = g_basename(entry->filename);
++ else {
++ ctitle = xmms_charset_recode_fs(g_basename(entry->filename),0,NULL);
++ if (ctitle) title = ctitle;
++ else title = g_basename(entry->filename);
++ }
+
+ pos = playlist_get_queue_position(entry);
+
+@@ -497,6 +511,10 @@
+ playlist_list_draw_string(pl, playlist_list_font,
+ i - pl->pl_first, tw, text);
+ g_free(text);
++ if (ctitle) {
++ g_free(ctitle);
++ ctitle=NULL;
++ }
+ }
+ PL_UNLOCK();
+ }
+diff -Naur xmms-1.2.8/xmms/playlistwin.c xmms-1.2.8-new/xmms/playlistwin.c
+--- xmms-1.2.8/xmms/playlistwin.c 2003-04-04 19:42:51.000000000 +0200
++++ xmms-1.2.8-new/xmms/playlistwin.c 2003-09-06 20:24:02.000000000 +0200
+@@ -38,7 +38,8 @@
+ PButton *playlistwin_shade, *playlistwin_close;
+ static PlaylistSlider *playlistwin_slider = NULL;
+ static TextBox *playlistwin_time_min, *playlistwin_time_sec;
+-static TextBox *playlistwin_info, *playlistwin_sinfo;
++static TextBox *playlistwin_info;
++ TextBox *playlistwin_sinfo;
+ static SButton *playlistwin_srew, *playlistwin_splay;
+ static SButton *playlistwin_spause, *playlistwin_sstop;
+ static SButton *playlistwin_sfwd, *playlistwin_seject;
+@@ -1846,6 +1847,7 @@
+ static void playlistwin_create_widgets(void)
+ {
+ playlistwin_sinfo = create_textbox(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, 4, 4, cfg.playlist_width - 35, FALSE, SKIN_TEXT);
++ charset_create_playlist_window(&cfg);
+ if (!cfg.playlist_shaded)
+ hide_widget(playlistwin_sinfo);
+ if (cfg.playlist_shaded)
+diff -Naur xmms-1.2.8/xmms/prefswin.c xmms-1.2.8-new/xmms/prefswin.c
+--- xmms-1.2.8/xmms/prefswin.c 2003-07-16 15:17:47.000000000 +0200
++++ xmms-1.2.8-new/xmms/prefswin.c 2003-09-06 19:42:29.000000000 +0200
+@@ -1,4 +1,4 @@
+-/* XMMS - Cross-platform multimedia player
++/* XMMS - Cdross-platform multimedia player
+ * Copyright (C) 1998-2001 Peter Alm, Mikael Alm, Olle Hallnas,
+ * Thomas Nilsson and 4Front Technologies
+ * Copyright (C) 1999-2001 Haavard Kvaalen
+@@ -20,6 +20,7 @@
+ #include "xmms.h"
+ #include "libxmms/util.h"
+ #include "libxmms/titlestring.h"
++#include "charset.h"
+
+ static GtkWidget *prefswin, *prefswin_notebook, *prefswin_ok;
+ static GtkWidget *prefswin_audio_ie_cbox;
+@@ -47,6 +48,7 @@
+ extern PButton *playlistwin_shade, *playlistwin_close, *equalizerwin_close;
+ extern PButton *mainwin_menubtn, *mainwin_minimize, *mainwin_shade, *mainwin_close;
+ extern TextBox *mainwin_info;
++extern TextBox *playlistwin_sinfo;
+ extern gboolean mainwin_focus, equalizerwin_focus, playlistwin_focus;
+
+ static gboolean is_opening = FALSE;
+@@ -351,6 +353,8 @@
+ cfg.pause_between_songs_time = CLAMP(atoi(gtk_entry_get_text(GTK_ENTRY(prefswin_options_pbs_entry))), 0, 1000);
+ cfg.mouse_change = gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(prefswin_options_mouse_spin));
+
++ charset_prefswin_apply_changes(&cfg);
++
+ set_current_output_plugin(selected_oplugin);
+
+ equalizerwin_set_doublesize(cfg.doublesize && cfg.eq_doublesize_linked);
+@@ -384,7 +388,6 @@
+ textbox_set_scroll(mainwin_info, FALSE);
+ textbox_set_scroll(mainwin_info, TRUE);
+ }
+-
+ if (show_wm_old != cfg.show_wm_decorations)
+ prefswin_toggle_wm_decorations();
+
+@@ -1089,6 +1092,8 @@
+
+ gtk_notebook_append_page(GTK_NOTEBOOK(prefswin_notebook), prefswin_fonts_vbox, gtk_label_new(_("Fonts")));
+
++ charset_create_prefs_window(&cfg,GTK_NOTEBOOK(prefswin_notebook), prefswin_fonts_vbox);
++
+ /*
+ * Title page
+ */
+@@ -1358,6 +1363,8 @@
+ gtk_entry_set_text(GTK_ENTRY(prefswin_options_pbs_entry), temp);
+ gtk_spin_button_set_value(GTK_SPIN_BUTTON(prefswin_options_mouse_spin), cfg.mouse_change);
+
++ charset_show_prefs_window(&cfg);
++
+ gtk_widget_show_all(prefswin);
+ gtk_widget_grab_default(prefswin_ok);
+
diff --git a/media-sound/xmms/files/xmms-1.2.8-sigterm.patch b/media-sound/xmms/files/xmms-1.2.8-sigterm.patch
new file mode 100644
index 000000000000..001b3df13209
--- /dev/null
+++ b/media-sound/xmms/files/xmms-1.2.8-sigterm.patch
@@ -0,0 +1,31 @@
+diff -Naur xmms-1.2.8/xmms/main.c xmms-1.2.8-new/xmms/main.c
+--- xmms-1.2.8/xmms/main.c 2003-09-02 15:01:40.000000000 +0200
++++ xmms-1.2.8-new/xmms/main.c 2003-09-06 18:15:15.000000000 +0200
+@@ -3343,6 +3343,18 @@
+ exit(1);
+ }
+
++/* Try to exit nicely when receiving a nice exit signal */
++void sigterm_handler(int sig)
++{
++ /* Original author of patch said that you should not use mainwin_quit_cb(),
++ * but his way deadlock xmms, and it anyhow just calls ctrlsocket stuff with
++ * with CMD_QUIT, which anyhow calls mainwin_quit_cb() in turn. This is not
++ * entirely clean, but works.
++ * <azarah@gentoo.org> (19 Jan 2003)
++ xmms_remote_quit(ctrlsocket_get_session_id()); */
++ mainwin_quit_cb();
++}
++
+ static gboolean pposition_configure(GtkWidget *w, GdkEventConfigure *event, gpointer data)
+ {
+ gint x,y;
+@@ -3473,6 +3485,8 @@
+ #endif
+
+ signal(SIGPIPE, SIG_IGN); /* for controlsocket.c */
++ signal(SIGTERM, sigterm_handler);
++ signal(SIGINT, sigterm_handler);
+ signal(SIGSEGV, segfault_handler);
+ g_thread_init(NULL);
+ if (!g_thread_supported())