Re: Translation support (wrc)

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Le jeu 23/10/2003 à 23:55, Alexandre Julliard a écrit :
> Vincent Béron <vberon@mecano.gme.usherb.ca> writes:
> 
> > Changelog:
> > Add support for comparing resources to wrc for translations.
> 
> That's a clear case of preprocessor abuse... there has to be a better
> way to do that than with a 100-line macro.

And here it is.

I'll repost the other part (creation of HTML pages) after taking into
account comments from everyone on the layout.

Vincent
Index: tools/wrc/Makefile.in
===================================================================
RCS file: /home/wine/wine/tools/wrc/Makefile.in,v
retrieving revision 1.28
diff -u -r1.28 Makefile.in
--- tools/wrc/Makefile.in	1 Sep 2003 23:59:41 -0000	1.28
+++ tools/wrc/Makefile.in	31 Oct 2003 22:29:12 -0000
@@ -15,6 +15,7 @@
 	genres.c \
 	newstruc.c \
 	readres.c \
+	translation.c \
 	utils.c \
 	wrc.c \
 	writeres.c
Index: tools/wrc/wrc.c
===================================================================
RCS file: /home/wine/wine/tools/wrc/wrc.c,v
retrieving revision 1.36
diff -u -r1.36 wrc.c
--- tools/wrc/wrc.c	18 Aug 2003 19:48:58 -0000	1.36
+++ tools/wrc/wrc.c	31 Oct 2003 22:29:12 -0000
@@ -39,6 +39,7 @@
 #include "utils.h"
 #include "readres.h"
 #include "dumpres.h"
+#include "translation.h"
 #include "genres.h"
 #include "newstruc.h"
 #include "parser.h"
@@ -97,6 +98,7 @@
 	"    * 0x08 Preprocessor messages\n"
 	"    * 0x10 Preprocessor lex messages\n"
 	"    * 0x20 Preprocessor yacc trace\n"
+	"    * 0x40 Verify translations\n"
 	"If no input filename is given and the output name is not overridden\n"
 	"with -o, then the output is written to \"wrc.tab.res\"\n"
 	;
@@ -123,6 +125,7 @@
  * debuglevel & DEBUGLEVEL_PPMSG	Preprocessor messages
  * debuglevel & DEBUGLEVEL_PPLEX	Preprocessor lex trace
  * debuglevel & DEBUGLEVEL_PPTRACE	Preprocessor yacc trace
+ * debuglevel & DEBUGLEVEL_TRANSLATION	Verify translations
  */
 int debuglevel = DEBUGLEVEL_NONE;
 
@@ -453,6 +456,12 @@
 
 	if(debuglevel & DEBUGLEVEL_DUMP)
 		dump_resources(resource_top);
+
+	if(debuglevel & DEBUGLEVEL_TRANSLATION)
+	{
+		verify_translations(resource_top);
+		exit(0);
+	}
 
 	/* Convert the internal lists to binary data */
 	resources2res(resource_top);
Index: tools/wrc/wrc.h
===================================================================
RCS file: /home/wine/wine/tools/wrc/wrc.h,v
retrieving revision 1.31
diff -u -r1.31 wrc.h
--- tools/wrc/wrc.h	5 Sep 2003 23:15:40 -0000	1.31
+++ tools/wrc/wrc.h	31 Oct 2003 22:29:12 -0000
@@ -45,6 +45,7 @@
 #define DEBUGLEVEL_PPMSG	0x0008
 #define DEBUGLEVEL_PPLEX	0x0010
 #define DEBUGLEVEL_PPTRACE	0x0020
+#define DEBUGLEVEL_TRANSLATION	0x0040
 
 extern int win32;
 extern int create_res;
diff -u /dev/null tools/wrc/translation.c
--- /dev/null	2002-08-30 19:31:37.000000000 -0400
+++ tools/wrc/translation.c	2003-10-31 17:32:10.000000000 -0500
@@ -0,0 +1,1138 @@
+/*
+ * Copyright 2003 Vincent Béron
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ */
+
+#include <stdio.h>
+#include <stdlib.h>
+
+#include "translation.h"
+
+static int present_resources[res_usr+1];
+static char *res_names[res_usr+1];
+static int nb_resources[res_usr+1][lang_type_normal+1];
+static resource_t **list_resources[res_usr+1][lang_type_normal+1];
+
+static int get_language_id(resource_t *resource) {
+	switch(resource->type) {
+		case res_acc:
+			return resource->res.acc->lvc.language->id;
+		case res_bmp:
+			return resource->res.bmp->data->lvc.language->id;
+		case res_cur:
+			return resource->res.cur->lvc.language->id;
+		case res_curg:
+			return resource->res.curg->lvc.language->id;
+		case res_dlg:
+			return resource->res.dlg->lvc.language->id;
+		case res_dlgex:
+			return resource->res.dlgex->lvc.language->id;
+		case res_fnt:
+			return resource->res.fnt->data->lvc.language->id;
+		case res_fntdir:
+			return resource->res.fnd->data->lvc.language->id;
+		case res_ico:
+			return resource->res.ico->lvc.language->id;
+		case res_icog:
+			return resource->res.icog->lvc.language->id;
+		case res_men:
+			return resource->res.men->lvc.language->id;
+		case res_menex:
+			return resource->res.menex->lvc.language->id;
+		case res_rdt:
+			return resource->res.rdt->data->lvc.language->id;
+		case res_stt:
+			return resource->res.stt->lvc.language->id;
+		case res_usr:
+			return resource->res.usr->data->lvc.language->id;
+		case res_msg:
+			return resource->res.msg->data->lvc.language->id;
+		case res_ver:
+			return resource->res.ver->lvc.language->id;
+		case res_dlginit:
+			return resource->res.dlgi->data->lvc.language->id;
+		case res_toolbar:
+			return resource->res.tbt->lvc.language->id;
+		case res_anicur:
+		case res_aniico:
+			return resource->res.ani->data->lvc.language->id;
+		default:
+			/* Not supposed to reach here */
+			fprintf(stderr, "Not supposed to reach here (get_language_id())\n");
+			abort();
+			return -1;
+	}
+}
+
+static void add_resource(resource_t *resource) {
+	enum lang_type_e lang_type;
+	enum res_e res_type = resource->type;
+	int lid = get_language_id(resource);
+
+	if(lid == MASTER_LANGUAGE) {
+		lang_type = lang_type_master;
+	} else if(lid == LANG_NEUTRAL) {
+		lang_type = lang_type_neutral;
+	} else {
+		lang_type = lang_type_normal;
+	}
+	nb_resources[res_type][lang_type]++;
+	list_resources[res_type][lang_type] = realloc(list_resources[res_type][lang_type], nb_resources[res_type][lang_type]*sizeof(resource_t *));
+	list_resources[res_type][lang_type][nb_resources[res_type][lang_type]-1] = resource;
+}
+
+static const char *get_language_name(int lid) {
+	PRETTYPRINTLANG(NEUTRAL)
+	PRETTYPRINTLANG(AFRIKAANS)
+	PRETTYPRINTLANG(ALBANIAN)
+	PRETTYPRINTLANG(ARABIC)
+	PRETTYPRINTLANG(ARMENIAN)
+	PRETTYPRINTLANG(ASSAMESE)
+	PRETTYPRINTLANG(AZERI)
+	PRETTYPRINTLANG(BASQUE)
+	PRETTYPRINTLANG(BELARUSIAN)
+	PRETTYPRINTLANG(BENGALI)
+	PRETTYPRINTLANG(BULGARIAN)
+	PRETTYPRINTLANG(CATALAN)
+	PRETTYPRINTLANG(CHINESE)
+	PRETTYPRINTLANG(CROATIAN)
+	PRETTYPRINTLANG(CZECH)
+	PRETTYPRINTLANG(DANISH)
+	PRETTYPRINTLANG(DIVEHI)
+	PRETTYPRINTLANG(DUTCH)
+	PRETTYPRINTLANG(ENGLISH)
+	PRETTYPRINTLANG(ESTONIAN)
+	PRETTYPRINTLANG(FAEROESE)
+	PRETTYPRINTLANG(FARSI)
+	PRETTYPRINTLANG(FINNISH)
+	PRETTYPRINTLANG(FRENCH)
+	PRETTYPRINTLANG(GALICIAN)
+	PRETTYPRINTLANG(GEORGIAN)
+	PRETTYPRINTLANG(GERMAN)
+	PRETTYPRINTLANG(GREEK)
+	PRETTYPRINTLANG(GUJARATI)
+	PRETTYPRINTLANG(HEBREW)
+	PRETTYPRINTLANG(HINDI)
+	PRETTYPRINTLANG(HUNGARIAN)
+	PRETTYPRINTLANG(ICELANDIC)
+	PRETTYPRINTLANG(INDONESIAN)
+	PRETTYPRINTLANG(ITALIAN)
+	PRETTYPRINTLANG(JAPANESE)
+	PRETTYPRINTLANG(KANNADA)
+	PRETTYPRINTLANG(KASHMIRI)
+	PRETTYPRINTLANG(KAZAK)
+	PRETTYPRINTLANG(KONKANI)
+	PRETTYPRINTLANG(KOREAN)
+	PRETTYPRINTLANG(KYRGYZ)
+	PRETTYPRINTLANG(LATVIAN)
+	PRETTYPRINTLANG(LITHUANIAN)
+	PRETTYPRINTLANG(MACEDONIAN)
+	PRETTYPRINTLANG(MALAY)
+	PRETTYPRINTLANG(MALAYALAM)
+	PRETTYPRINTLANG(MANIPURI)
+	PRETTYPRINTLANG(MARATHI)
+	PRETTYPRINTLANG(MONGOLIAN)
+	PRETTYPRINTLANG(NEPALI)
+	PRETTYPRINTLANG(NORWEGIAN)
+	PRETTYPRINTLANG(ORIYA)
+	PRETTYPRINTLANG(POLISH)
+	PRETTYPRINTLANG(PORTUGUESE)
+	PRETTYPRINTLANG(PUNJABI)
+	PRETTYPRINTLANG(ROMANIAN)
+	PRETTYPRINTLANG(RUSSIAN)
+	PRETTYPRINTLANG(SANSKRIT)
+	PRETTYPRINTLANG(SERBIAN)
+	PRETTYPRINTLANG(SINDHI)
+	PRETTYPRINTLANG(SLOVAK)
+	PRETTYPRINTLANG(SLOVENIAN)
+	PRETTYPRINTLANG(SPANISH)
+	PRETTYPRINTLANG(SWAHILI)
+	PRETTYPRINTLANG(SWEDISH)
+	PRETTYPRINTLANG(SYRIAC)
+	PRETTYPRINTLANG(TAMIL)
+	PRETTYPRINTLANG(TATAR)
+	PRETTYPRINTLANG(TELUGU)
+	PRETTYPRINTLANG(THAI)
+	PRETTYPRINTLANG(TURKISH)
+	PRETTYPRINTLANG(UKRAINIAN)
+	PRETTYPRINTLANG(URDU)
+	PRETTYPRINTLANG(UZBEK)
+	PRETTYPRINTLANG(VIETNAMESE)
+	PRETTYPRINTLANG(GAELIC)
+	PRETTYPRINTLANG(MALTESE)
+	PRETTYPRINTLANG(MAORI)
+	PRETTYPRINTLANG(RHAETO_ROMANCE)
+	PRETTYPRINTLANG(SAAMI)
+	PRETTYPRINTLANG(SORBIAN)
+	PRETTYPRINTLANG(SUTU)
+	PRETTYPRINTLANG(TSONGA)
+	PRETTYPRINTLANG(TSWANA)
+	PRETTYPRINTLANG(VENDA)
+	PRETTYPRINTLANG(XHOSA)
+	PRETTYPRINTLANG(ZULU)
+	PRETTYPRINTLANG(ESPERANTO)
+	PRETTYPRINTLANG(WALON)
+	PRETTYPRINTLANG(CORNISH)
+	PRETTYPRINTLANG(WELSH)
+	PRETTYPRINTLANG(BRETON)
+	return "Unkown language";
+}
+
+static int compare_accelerator(accelerator_t *accelerator1, accelerator_t *accelerator2) {
+	int different = 0;
+	event_t *ev1 = NULL, *ev2 = NULL;
+	if(!different &&
+	   ((accelerator1->memopt != accelerator2->memopt) ||
+	   (accelerator1->lvc.version != accelerator2->lvc.version) ||
+	   (accelerator1->lvc.characts != accelerator2->lvc.characts)))
+		different = 1;
+	ev1 = accelerator1->events;
+	ev2 = accelerator2->events;
+	while(!different && ev1 && ev2) {
+		if(!different &&
+		   ((ev1->id != ev2->id) ||
+		   (ev1->flags != ev2->flags)))
+			different = 1;
+		ev1 = ev1->next;
+		ev2 = ev2->next;
+	}
+	if(!different &&
+	   ((ev1 && !ev2) || (!ev1 && ev2)))
+		different = 1;
+	return different;
+}
+
+static int compare_bitmap(bitmap_t *bitmap1, bitmap_t *bitmap2) {
+	int different = 0;
+	if(!different &&
+	   ((bitmap1->memopt != bitmap2->memopt) ||
+	   (bitmap1->data->lvc.version != bitmap2->data->lvc.version) ||
+	   (bitmap1->data->lvc.characts != bitmap2->data->lvc.characts)))
+		different = 1;
+	return different;
+}
+
+static int compare_cursor(cursor_t *cursor1, cursor_t *cursor2) {
+	int different = 0;
+	if(!different &&
+	   ((cursor1->id != cursor2->id) ||
+	   (cursor1->width != cursor2->width) ||
+	   (cursor1->height != cursor2->height) ||
+	   (cursor1->xhot != cursor2->xhot) ||
+	   (cursor1->yhot != cursor2->yhot)))
+		different = 1;
+	if(!different &&
+	   ((cursor1->lvc.version != cursor2->lvc.version) ||
+	   (cursor1->lvc.characts != cursor2->lvc.characts)))
+		different = 1;
+	return different;
+}
+
+static int compare_cursor_group(cursor_group_t *cursor_group1, cursor_group_t *cursor_group2) {
+	int different = 0;
+	cursor_t *cursor1 = NULL, *cursor2 = NULL;
+	if(!different &&
+	   ((cursor_group1->memopt != cursor_group2->memopt) ||
+	   (cursor_group1->lvc.version != cursor_group2->lvc.version) ||
+	   (cursor_group1->lvc.characts != cursor_group2->lvc.characts)))
+		different = 1;
+	if(!different &&
+	   (cursor_group1->ncursor != cursor_group2->ncursor))
+		different = 1;
+	if(!different) {
+		cursor1 = cursor_group1->cursorlist;
+		cursor2 = cursor_group2->cursorlist;
+		while(!different && cursor1 && cursor2) {
+			different = compare_cursor(cursor1, cursor2);
+			cursor1 = cursor1->next;
+			cursor2 = cursor2->next;
+		}
+		if(!different &&
+		   ((cursor1 && !cursor2) ||
+		   (!cursor1 && cursor2)))
+			different = 1;
+	}
+	return different;
+}
+
+static int compare_control(control_t *control1, control_t *control2) {
+	int different = 0;
+	char *nameid = NULL;
+	if(!different &&
+		((control1 && !control2) ||
+		(!control1 && control2)))
+			different = 1;
+	if(different || !control1 || !control2)
+		return different;
+	nameid = strdup(get_nameid_str(control1->ctlclass));
+	if(!different && strcmp(nameid, get_nameid_str(control2->ctlclass)))
+		different = 1;
+	free(nameid);
+	if(!different && 
+	   (control1->id != control2->id))
+		different = 1;
+	if(!different && control1->gotstyle && control2->gotstyle) {
+		if((!control1->style || !control2->style) ||
+		   (control1->style->and_mask || control2->style->and_mask) ||
+		   (control1->style->or_mask != control2->style->or_mask))
+			different = 1;
+	} else if(!different &&
+		  ((control1->gotstyle && !control2->gotstyle) ||
+		  (!control1->gotstyle && control2->gotstyle)))
+			different = 1;
+	if(!different && control1->gotexstyle && control2->gotexstyle) {
+		if((!control1->exstyle || !control2->exstyle) ||
+		   (control1->exstyle->and_mask || control2->exstyle->and_mask) ||
+		   (control1->exstyle->or_mask != control2->exstyle->or_mask))
+			different = 1;
+	} else if(!different &&
+		  ((control1->gotexstyle && !control2->gotexstyle) ||
+		  (!control1->gotexstyle && control2->gotexstyle)))
+			different = 1;
+	if(!different && control1->gothelpid && control2->gothelpid) {
+		if(control1->helpid != control2->helpid)
+			different = 1;
+	} else if(!different &&
+		  ((control1->gothelpid && !control2->gothelpid) ||
+		  (!control1->gothelpid && control2->gothelpid)))
+			different = 1;
+	return different;
+}
+
+static int compare_dialog(dialog_t *dialog1, dialog_t *dialog2) {
+	int different = 0;
+	char *nameid = NULL;
+	if(!different &&
+	   ((dialog1->memopt != dialog2->memopt) ||
+	   (dialog1->lvc.version != dialog2->lvc.version) ||
+	   (dialog1->lvc.characts != dialog2->lvc.characts)))
+		different = 1;
+	if(!different && dialog1->gotstyle && dialog2->gotstyle) {
+		if((!dialog1->style || !dialog2->style) ||
+		   (dialog1->style->and_mask || dialog2->style->and_mask) ||
+		   (dialog1->style->or_mask != dialog2->style->or_mask))
+			different = 1;
+	} else if(!different &&
+		  ((dialog1->gotstyle && !dialog2->gotstyle) ||
+		  (!dialog1->gotstyle && dialog2->gotstyle)))
+			different = 1;
+	if(!different && dialog1->gotexstyle && dialog2->gotexstyle) {
+		if((!dialog1->exstyle || !dialog2->exstyle) ||
+		   (dialog1->exstyle->and_mask || dialog2->exstyle->and_mask) ||
+		   (dialog1->exstyle->or_mask != dialog2->exstyle->or_mask))
+			different = 1;
+	} else if(!different &&
+		  ((dialog1->gotexstyle && !dialog2->gotexstyle) ||
+		  (!dialog1->gotexstyle && dialog2->gotexstyle)))
+			different = 1;
+	nameid = strdup(get_nameid_str(dialog1->menu));
+	if(!different && strcmp(nameid, get_nameid_str(dialog2->menu)))
+		different = 1;
+	free(nameid);
+	nameid = strdup(get_nameid_str(dialog1->dlgclass));
+	if(!different && strcmp(nameid, get_nameid_str(dialog2->dlgclass)))
+		different = 1;
+	free(nameid);
+	if(!different)
+		different = compare_control(dialog1->controls, dialog2->controls);
+	return different;
+}
+
+static int compare_dialogex(dialogex_t *dialogex1, dialogex_t *dialogex2) {
+	int different = 0;
+	char *nameid = NULL;
+	if(!different &&
+	   ((dialogex1->memopt != dialogex2->memopt) ||
+	   (dialogex1->lvc.version != dialogex2->lvc.version) ||
+	   (dialogex1->lvc.characts != dialogex2->lvc.characts)))
+		different = 1;
+	if(!different && dialogex1->gotstyle && dialogex2->gotstyle) {
+		if((!dialogex1->style || !dialogex2->style) ||
+		   (dialogex1->style->and_mask || dialogex2->style->and_mask) ||
+		   (dialogex1->style->or_mask != dialogex2->style->or_mask))
+			different = 1;
+	} else if(!different &&
+		  ((dialogex1->gotstyle && !dialogex2->gotstyle) ||
+		  (!dialogex1->gotstyle && dialogex2->gotstyle)))
+			different = 1;
+	if(!different && dialogex1->gotexstyle && dialogex2->gotexstyle) {
+		if((!dialogex1->exstyle || !dialogex2->exstyle) ||
+		   (dialogex1->exstyle->and_mask || dialogex2->exstyle->and_mask) ||
+		   (dialogex1->exstyle->or_mask != dialogex2->exstyle->or_mask))
+			different = 1;
+	} else if(!different &&
+		  ((dialogex1->gotexstyle && !dialogex2->gotexstyle) ||
+		  (!dialogex1->gotexstyle && dialogex2->gotexstyle)))
+			different = 1;
+	if(!different && dialogex1->gothelpid && dialogex2->gothelpid) {
+		if(dialogex1->helpid != dialogex2->helpid)
+			different = 1;
+	} else if(!different &&
+		  ((dialogex1->gothelpid && !dialogex2->gothelpid) ||
+		  (!dialogex1->gothelpid && dialogex2->gothelpid)))
+			different = 1;
+	nameid = strdup(get_nameid_str(dialogex1->menu));
+	if(!different && strcmp(nameid, get_nameid_str(dialogex2->menu)))
+		different = 1;
+	free(nameid);
+	nameid = strdup(get_nameid_str(dialogex1->dlgclass));
+	if(!different && strcmp(nameid, get_nameid_str(dialogex2->dlgclass)))
+		different = 1;
+	free(nameid);
+	if(!different)
+		different = compare_control(dialogex1->controls, dialogex2->controls);
+	return different;
+}
+
+static int compare_font(font_t *font1, font_t *font2) {
+	int different = 0;
+	if(!different &&
+	   ((font1->memopt != font2->memopt) ||
+	   (font1->data->lvc.version != font2->data->lvc.version) ||
+	   (font1->data->lvc.characts != font2->data->lvc.characts)))
+		different = 1;
+	return different;
+}
+
+static int compare_fontdir(fontdir_t *fontdir1, fontdir_t *fontdir2) {
+	int different = 0;
+	if(!different &&
+	   ((fontdir1->memopt != fontdir2->memopt) ||
+	   (fontdir1->data->lvc.version != fontdir2->data->lvc.version) ||
+	   (fontdir1->data->lvc.characts != fontdir2->data->lvc.characts)))
+		different = 1;
+	return different;
+}
+
+static int compare_icon(icon_t *icon1, icon_t *icon2) {
+	int different = 0;
+	if(!different &&
+	   ((icon1->id != icon2->id) ||
+	   (icon1->width != icon2->width) ||
+	   (icon1->height != icon2->height)))
+		different = 1;
+	if(!different &&
+	   ((icon1->lvc.version != icon2->lvc.version) ||
+	   (icon1->lvc.characts != icon2->lvc.characts)))
+		different = 1;
+	return different;
+}
+
+static int compare_icon_group(icon_group_t *icon_group1, icon_group_t *icon_group2) {
+	int different = 0;
+	icon_t *icon1 = NULL, *icon2 = NULL;
+	if(!different &&
+	   ((icon_group1->memopt != icon_group2->memopt) ||
+	   (icon_group1->lvc.version != icon_group2->lvc.version) ||
+	   (icon_group1->lvc.characts != icon_group2->lvc.characts)))
+		different = 1;
+	if(!different &&
+	   (icon_group1->nicon != icon_group2->nicon))
+		different = 1;
+	if(!different) {
+		icon1 = icon_group1->iconlist;
+		icon2 = icon_group2->iconlist;
+		while(!different && icon1 && icon2) {
+			different = compare_icon(icon1, icon2);
+			icon1 = icon1->next;
+			icon2 = icon2->next;
+		}
+		if(!different &&
+		   ((icon1 && !icon2) ||
+		   (!icon1 && icon2)))
+			different = 1;
+	}
+	return different;
+}
+
+static int compare_menu_item(menu_item_t *menu_item1, menu_item_t *menu_item2) {
+	int different = 0;
+	while(!different && menu_item1 && menu_item2) {
+		if(menu_item1->popup && menu_item2->popup)
+			different = compare_menu_item(menu_item1->popup, menu_item2->popup);
+		else if(!menu_item1->popup && !menu_item2->popup) {
+			if(menu_item1->name && menu_item2->name) {
+				if((menu_item1->id != menu_item2->id) ||
+				   (menu_item1->state != menu_item2->state))
+					different = 1;
+			} else if((menu_item1->name && !menu_item2->name) ||
+				  (!menu_item1->name && menu_item2->name))
+					different = 1;
+		} else
+			different = 1;
+		menu_item1 = menu_item1->next;
+		menu_item2 = menu_item2->next;
+	}
+	if(!different &&
+	   ((menu_item1 && !menu_item2) ||
+	   (!menu_item1 && menu_item2)))
+		different = 1;
+	return different;
+}
+
+static int compare_menu(menu_t *menu1, menu_t *menu2) {
+	int different = 0;
+	if(!different &&
+	   ((menu1->memopt != menu2->memopt) ||
+	   (menu1->lvc.version != menu2->lvc.version) ||
+	   (menu1->lvc.characts != menu2->lvc.characts)))
+		different = 1;
+	if(!different)
+		different = compare_menu_item(menu1->items, menu2->items);
+	return different;
+}
+
+static int compare_menuex_item(menuex_item_t *menuex_item1, menuex_item_t *menuex_item2) {
+	int different = 0;
+	while(!different && menuex_item1 && menuex_item2) {
+		if(menuex_item1->popup && menuex_item2->popup) {
+			if(!different && menuex_item1->gotid && menuex_item2->gotid) {
+				if(menuex_item1->id != menuex_item2->id)
+					different = 1;
+			} else if(!different &&
+				  ((menuex_item1->gotid && !menuex_item2->gotid) ||
+				  (!menuex_item2->gotid && menuex_item2->gotid)))
+					different = 1;
+			if(!different && menuex_item1->gottype && menuex_item2->gottype) {
+				if(menuex_item1->type != menuex_item2->type)
+					different = 1;
+			} else if(!different &&
+				  ((menuex_item1->gottype && !menuex_item2->gottype) ||
+				  (!menuex_item2->gottype && menuex_item2->gottype)))
+					different = 1;
+			if(!different && menuex_item1->gotstate && menuex_item2->gotstate) {
+				if(menuex_item1->state != menuex_item2->state)
+					different = 1;
+			} else if(!different &&
+				  ((menuex_item1->gotstate && !menuex_item2->gotstate) ||
+				  (!menuex_item2->gotstate && menuex_item2->gotstate)))
+					different = 1;
+			if(!different && menuex_item1->gothelpid && menuex_item2->gothelpid) {
+				if(menuex_item1->helpid != menuex_item2->helpid)
+					different = 1;
+			} else if(!different &&
+				  ((menuex_item1->gothelpid && !menuex_item2->gothelpid) ||
+				  (!menuex_item2->gothelpid && menuex_item2->gothelpid)))
+					different = 1;
+			if(!different)
+				different = compare_menuex_item(menuex_item1->popup, menuex_item2->popup);
+		} else if(!menuex_item1->popup && !menuex_item2->popup) {
+			if(menuex_item1->name && menuex_item2->name) {
+				if(!different && menuex_item1->gotid && menuex_item2->gotid) {
+					if(menuex_item1->id != menuex_item2->id)
+						different = 1;
+				} else if(!different &&
+					  ((menuex_item1->gotid && !menuex_item2->gotid) ||
+					  (!menuex_item2->gotid && menuex_item2->gotid)))
+						different = 1;
+				if(!different && menuex_item1->gottype && menuex_item2->gottype) {
+					if(menuex_item1->type != menuex_item2->type)
+						different = 1;
+				} else if(!different &&
+					  ((menuex_item1->gottype && !menuex_item2->gottype) ||
+					  (!menuex_item2->gottype && menuex_item2->gottype)))
+						different = 1;
+				if(!different && menuex_item1->gotstate && menuex_item2->gotstate) {
+					if(menuex_item1->state != menuex_item2->state)
+						different = 1;
+				} else if(!different &&
+					  ((menuex_item1->gotstate && !menuex_item2->gotstate) ||
+					  (!menuex_item2->gotstate && menuex_item2->gotstate)))
+						different = 1;
+				if(!different && menuex_item1->gothelpid && menuex_item2->gothelpid) {
+					if(menuex_item1->helpid != menuex_item2->helpid)
+						different = 1;
+				} else if(!different &&
+					  ((menuex_item1->gothelpid && !menuex_item2->gothelpid) ||
+					  (!menuex_item2->gothelpid && menuex_item2->gothelpid)))
+						different = 1;
+			} else if((menuex_item1->name && !menuex_item2->name) ||
+				  (!menuex_item1->name && menuex_item2->name))
+					different = 1;
+		} else
+			different = 1;
+		menuex_item1 = menuex_item1->next;
+		menuex_item2 = menuex_item2->next;
+	}
+	if(!different &&
+	   ((menuex_item1 && !menuex_item2) ||
+	   (!menuex_item1 && menuex_item2)))
+		different = 1;
+	return different;
+}
+
+static int compare_menuex(menuex_t *menuex1, menuex_t *menuex2) {
+	int different = 0;
+	if(!different &&
+	   ((menuex1->memopt != menuex2->memopt) ||
+	   (menuex1->lvc.version != menuex2->lvc.version) ||
+	   (menuex1->lvc.characts != menuex2->lvc.characts)))
+		different = 1;
+	if(!different)
+		different = compare_menuex_item(menuex1->items, menuex2->items);
+	return different;
+}
+
+static int compare_rcdata(rcdata_t *rcdata1, rcdata_t *rcdata2) {
+	int different = 0;
+	if(!different &&
+	   ((rcdata1->memopt != rcdata2->memopt) ||
+	   (rcdata1->data->lvc.version != rcdata2->data->lvc.version) ||
+	   (rcdata1->data->lvc.characts != rcdata2->data->lvc.characts)))
+		different = 1;
+	return different;
+}
+
+static int compare_stringtable(stringtable_t *stringtable1, stringtable_t *stringtable2) {
+	int different = 0;
+	int i;
+	while(!different && stringtable1 && stringtable2) {
+		if((stringtable1->memopt != stringtable2->memopt) ||
+		   (stringtable1->lvc.version != stringtable2->lvc.version) ||
+		   (stringtable1->lvc.characts != stringtable2->lvc.characts))
+			different = 1;
+		if(!different) {
+			if((stringtable1->nentries != stringtable2->nentries) ||
+			   (stringtable1->idbase != stringtable2->idbase))
+				different = 1;
+			else
+				for(i = 0 ; i < stringtable1->nentries; i++)
+					if((stringtable1->entries[i].id != stringtable2->entries[i].id) ||
+					   (stringtable1->entries[i].memopt != stringtable2->entries[i].memopt) ||
+					   (stringtable1->entries[i].str && !stringtable2->entries[i].str) ||
+					   (!stringtable1->entries[i].str && stringtable2->entries[i].str)) {
+						different = 1;
+						break;
+					}
+		}
+		stringtable1 = stringtable1->next;
+		stringtable2 = stringtable2->next;
+	}
+	return different;
+}
+
+static int compare_user(user_t *user1, user_t *user2) {
+	int different = 0;
+	char *nameid = NULL;
+	if(!different &&
+	   ((user1->memopt != user2->memopt) ||
+	   (user1->data->lvc.version != user2->data->lvc.version) ||
+	   (user1->data->lvc.characts != user2->data->lvc.characts)))
+		different = 1;
+	nameid = strdup(get_nameid_str(user1->type));
+	if(!different && strcmp(nameid, get_nameid_str(user2->type)))
+		different = 1;
+	free(nameid);
+	return different;
+}
+
+static int compare_messagetable(messagetable_t *messagetable1, messagetable_t *messagetable2) {
+	int different = 0;
+	if(!different &&
+	   ((messagetable1->memopt != messagetable2->memopt) ||
+	   (messagetable1->data->lvc.version != messagetable2->data->lvc.version) ||
+	   (messagetable1->data->lvc.characts != messagetable2->data->lvc.characts)))
+		different = 1;
+	return different;
+}
+
+static int compare_string(string_t *string1, string_t *string2) {
+	int different = 0;
+	if(!different &&
+	   ((string1->size != string2->size) ||
+	   (string1->type != string2->type)))
+		different = 1;
+	if(!different) {
+		if(string1->type == str_char)
+			different = memcmp(string1->str.cstr, string2->str.cstr, string1->size);
+		else if(string1->type == str_unicode)
+			different = memcmp(string1->str.wstr, string2->str.wstr, string1->size*sizeof(WCHAR));
+		else
+			different = 1;
+	}
+	return different;
+}
+
+static int compare_ver_block(ver_block_t *ver_block1, ver_block_t *ver_block2);
+
+static int compare_ver_value(ver_value_t *ver_value1, ver_value_t *ver_value2) {
+	int different = 0;
+	int i = 0;
+	if(!different &&
+	   (ver_value1->type == ver_value2->type)) {
+		switch(ver_value1->type) {
+			case val_str:
+				if(!different && ver_value1->key && ver_value2->key)
+					different = compare_string(ver_value1->key, ver_value2->key);
+				else if(!different &&
+					((ver_value1->key && !ver_value2->key) ||
+					(!ver_value1->key && ver_value2->key)))
+						different = 1;
+				break;
+			case val_words:
+				if(!different && ver_value1->key && ver_value2->key)
+					different = compare_string(ver_value1->key, ver_value2->key);
+				else if(!different &&
+					((ver_value1->key && !ver_value2->key) ||
+					(!ver_value1->key && ver_value2->key)))
+						different = 1;
+				if(!different && ver_value1->value.words && ver_value2->value.words) {
+					if(!different &&
+					   (ver_value1->value.words->nwords != ver_value2->value.words->nwords))
+						different = 1;
+					if(!different)
+						for(i = 0; i < ver_value1->value.words->nwords; i++) {
+							if(ver_value1->value.words->words[i] != ver_value2->value.words->words[i]) {
+								different = 1;
+								break;
+							}
+						}
+				} else if(!different &&
+					  ((ver_value1->value.words && !ver_value2->value.words) ||
+					  (!ver_value1->value.words && ver_value2->value.words)))
+						different = 1;
+				break;
+			case val_block:
+				if(!different && ver_value1->value.block && ver_value2->value.block)
+					different = compare_ver_block(ver_value1->value.block, ver_value2->value.block);
+				else if(!different &&
+					((ver_value1->value.block && !ver_value2->value.block) ||
+					(!ver_value1->value.block && ver_value2->value.block)))
+						different = 1;
+				break;
+			default:
+				different = 1;
+		}
+	} else
+		different = 1;
+	return different;
+}
+
+static int compare_ver_block(ver_block_t *ver_block1, ver_block_t *ver_block2) {
+	int different = 0;
+	ver_value_t *ver_value1 = NULL, *ver_value2 = NULL;
+	if(!different) {
+		ver_value1 = ver_block1->values;
+		ver_value2 = ver_block2->values;
+		while(!different && ver_value1 && ver_value2) {
+			different = compare_ver_value(ver_value1, ver_value2);
+			ver_value1 = ver_value1->next;
+			ver_value2 = ver_value2->next;
+		}
+		if(!different &&
+		   ((ver_value1 && !ver_value2) ||
+		   (!ver_value1 && ver_value2)))
+			different = 1;
+	}
+	return different;
+}
+
+static int compare_versioninfo(versioninfo_t *versioninfo1, versioninfo_t *versioninfo2) {
+	int different = 0;
+	ver_block_t *ver_block1 = NULL, *ver_block2 = NULL;
+	if(!different &&
+	   ((versioninfo1->memopt != versioninfo2->memopt) ||
+	   (versioninfo1->lvc.version != versioninfo2->lvc.version) ||
+	   (versioninfo1->lvc.characts != versioninfo2->lvc.characts)))
+		different = 1;
+	if(!different && versioninfo1->gotit.fv && versioninfo2->gotit.fv) {
+		if((versioninfo1->filever_maj1 != versioninfo2->filever_maj1) ||
+		   (versioninfo1->filever_maj2 != versioninfo2->filever_maj2) ||
+		   (versioninfo1->filever_min1 != versioninfo2->filever_min1) ||
+		   (versioninfo1->filever_min2 != versioninfo2->filever_min2))
+			different = 1;
+	} else if(!different &&
+		  ((versioninfo1->gotit.fv && !versioninfo2->gotit.fv) ||
+		  (!versioninfo1->gotit.fv && versioninfo2->gotit.fv)))
+			different = 1;
+	if(!different && versioninfo1->gotit.pv && versioninfo2->gotit.pv) {
+		if((versioninfo1->prodver_maj1 != versioninfo2->prodver_maj1) ||
+		   (versioninfo1->prodver_maj2 != versioninfo2->prodver_maj2) ||
+		   (versioninfo1->prodver_min1 != versioninfo2->prodver_min1) ||
+		   (versioninfo1->prodver_min2 != versioninfo2->prodver_min2))
+			different = 1;
+	} else if(!different &&
+		  ((versioninfo1->gotit.pv && !versioninfo2->gotit.pv) ||
+		  (!versioninfo1->gotit.pv && versioninfo2->gotit.pv)))
+			different = 1;
+	if(!different && versioninfo1->gotit.fo && versioninfo2->gotit.fo) {
+		if(versioninfo1->fileos != versioninfo2->fileos)
+			different = 1;
+	} else if(!different &&
+		  ((versioninfo1->gotit.fo && !versioninfo2->gotit.fo) ||
+		  (!versioninfo1->gotit.fo && versioninfo2->gotit.fo)))
+			different = 1;
+	if(!different && versioninfo1->gotit.ff && versioninfo2->gotit.ff) {
+		if(versioninfo1->fileflags != versioninfo2->fileflags)
+			different = 1;
+	} else if(!different &&
+		  ((versioninfo1->gotit.ff && !versioninfo2->gotit.ff) ||
+		  (!versioninfo1->gotit.ff && versioninfo2->gotit.ff)))
+			different = 1;
+	if(!different && versioninfo1->gotit.ffm && versioninfo2->gotit.ffm) {
+		if(versioninfo1->fileflagsmask != versioninfo2->fileflagsmask)
+			different = 1;
+	} else if(!different &&
+		  ((versioninfo1->gotit.ffm && !versioninfo2->gotit.ffm) ||
+		  (!versioninfo1->gotit.ffm && versioninfo2->gotit.ffm)))
+			different = 1;
+	if(!different && versioninfo1->gotit.ft && versioninfo2->gotit.ft) {
+		if(versioninfo1->filetype != versioninfo2->filetype)
+			different = 1;
+	} else if(!different &&
+		  ((versioninfo1->gotit.ft && !versioninfo2->gotit.ft) ||
+		  (!versioninfo1->gotit.ft && versioninfo2->gotit.ft)))
+			different = 1;
+	if(!different && versioninfo1->gotit.fst && versioninfo2->gotit.fst) {
+		if(versioninfo1->filesubtype != versioninfo2->filesubtype)
+			different = 1;
+	} else if(!different &&
+		  ((versioninfo1->gotit.fst && !versioninfo2->gotit.fst) ||
+		  (!versioninfo1->gotit.fst && versioninfo2->gotit.fst)))
+			different = 1;
+	if(!different) {
+		ver_block1 = versioninfo1->blocks;
+		ver_block2 = versioninfo2->blocks;
+		while(!different && ver_block1 && ver_block2) {
+			different = compare_ver_block(ver_block1, ver_block2);
+			ver_block1 = ver_block1->next;
+			ver_block2 = ver_block2->next;
+		}
+		if(!different &&
+		   ((ver_block1 && !ver_block2) ||
+		   (ver_block1 && !ver_block2)))
+			different = 1;
+	}
+	return different;
+}
+
+static int compare_dlginit(dlginit_t *dlginit1, dlginit_t *dlginit2) {
+	int different = 0;
+	if(!different &&
+	   ((dlginit1->memopt != dlginit2->memopt) ||
+	   (dlginit1->data->lvc.version != dlginit2->data->lvc.version) ||
+	   (dlginit1->data->lvc.characts != dlginit2->data->lvc.characts)))
+		different = 1;
+	return different;
+}
+
+static int compare_toolbar_item(toolbar_item_t *toolbar_item1, toolbar_item_t *toolbar_item2) {
+	int different = 0;
+	while(!different && toolbar_item1 && toolbar_item2) {
+		if((toolbar_item1->id && !toolbar_item2->id) ||
+		   (!toolbar_item1->id && toolbar_item2->id))
+			different = 1;
+		toolbar_item1 = toolbar_item1->next;
+		toolbar_item2 = toolbar_item2->next;
+	}
+	if(!different &&
+	   ((toolbar_item1 && !toolbar_item2) ||
+	   (!toolbar_item1 && toolbar_item2)))
+		different = 1;
+	return different;
+}
+
+static int compare_toolbar(toolbar_t *toolbar1, toolbar_t *toolbar2) {
+	int different = 0;
+	if(!different &&
+	   ((toolbar1->memopt != toolbar2->memopt) ||
+	   (toolbar1->lvc.version != toolbar2->lvc.version) ||
+	   (toolbar1->lvc.characts != toolbar2->lvc.characts)))
+		different = 1;
+	if(!different)
+		different = compare_toolbar_item(toolbar1->items, toolbar2->items);
+	return different;
+}
+
+static int compare_ani_curico(ani_curico_t *ani_curico1, ani_curico_t *ani_curico2) {
+	int different = 0;
+	if(!different &&
+	   ((ani_curico1->memopt != ani_curico2->memopt) ||
+	   (ani_curico1->data->lvc.version != ani_curico2->data->lvc.version) ||
+	   (ani_curico1->data->lvc.characts != ani_curico2->data->lvc.characts)))
+		different = 1;
+	return different;
+}
+
+static int compare(resource_t *resource1, resource_t *resource2) {
+	switch(resource1->type) {
+		case res_acc:
+			return compare_accelerator(resource1->res.acc, resource2->res.acc);
+		case res_bmp:
+			return compare_bitmap(resource1->res.bmp, resource2->res.bmp);
+		case res_cur:
+			return compare_cursor(resource1->res.cur, resource2->res.cur);
+		case res_curg:
+			return compare_cursor_group(resource1->res.curg, resource2->res.curg);
+		case res_dlg:
+			return compare_dialog(resource1->res.dlg, resource2->res.dlg);
+		case res_dlgex:
+			return compare_dialogex(resource1->res.dlgex, resource2->res.dlgex);
+		case res_fnt:
+			return compare_font(resource1->res.fnt, resource2->res.fnt);
+		case res_fntdir:
+			return compare_fontdir(resource1->res.fnd, resource2->res.fnd);
+		case res_ico:
+			return compare_icon(resource1->res.ico, resource2->res.ico);
+		case res_icog:
+			return compare_icon_group(resource1->res.icog, resource2->res.icog);
+		case res_men:
+			return compare_menu(resource1->res.men, resource2->res.men);
+		case res_menex:
+			return compare_menuex(resource1->res.menex, resource2->res.menex);
+		case res_rdt:
+			return compare_rcdata(resource1->res.rdt, resource2->res.rdt);
+		case res_stt:
+			return compare_stringtable(resource1->res.stt, resource2->res.stt);
+		case res_usr:
+			return compare_user(resource1->res.usr, resource2->res.usr);
+		case res_msg:
+			return compare_messagetable(resource1->res.msg, resource2->res.msg);
+		case res_ver:
+			return compare_versioninfo(resource1->res.ver, resource2->res.ver);
+		case res_dlginit:
+			return compare_dlginit(resource1->res.dlgi, resource2->res.dlgi);
+		case res_toolbar:
+			return compare_toolbar(resource1->res.tbt, resource2->res.tbt);
+		case res_anicur:
+		case res_aniico:
+			return compare_ani_curico(resource1->res.ani, resource2->res.ani);
+		default:
+			/* Not supposed to reach here */
+			fprintf(stderr, "Not supposed to reach here (compare())\n");
+			abort();
+			return -1;
+	}
+}
+
+void verify_translations(resource_t *top) {
+	enum lang_type_e lang_type;
+	enum res_e res_type;
+	int **presence;
+	int i, j;
+	char *nameid;
+	char **problems;
+	int nb_problems, last_problem;
+	int complete, needs_work, partial;
+
+	for(res_type = res_0; res_type <= res_usr; res_type++) {
+		present_resources[res_type] = 0;
+		for(lang_type = lang_type_master; lang_type <= lang_type_normal; lang_type++) {
+			nb_resources[res_type][lang_type] = 0;
+			list_resources[res_type][lang_type] = NULL;
+		}
+	}
+
+	resource_t *next = top;
+	while(next) {
+		switch(next->type) {
+			case res_acc:
+			case res_bmp:
+			case res_cur:
+			case res_curg:
+			case res_dlg:
+			case res_dlgex:
+			case res_fnt:
+			case res_fntdir:
+			case res_ico:
+			case res_icog:
+			case res_men:
+			case res_menex:
+			case res_rdt:
+			case res_stt:
+			case res_usr:
+			case res_msg:
+			case res_ver:
+			case res_dlginit:
+			case res_toolbar:
+			case res_anicur:
+			case res_aniico:
+				add_resource(next);
+				break;
+			default:
+				fprintf(stderr, "Report this: unkown resource type parsed %08x\n", next->type);
+		}
+		next = next->next;
+	}
+	present_resources[res_acc] = 1;
+	res_names[res_acc] = strdup("accelerator");
+	present_resources[res_bmp] = 1;
+	res_names[res_bmp] = strdup("bitmap");
+	present_resources[res_cur] = 1;
+	res_names[res_cur] = strdup("cursor");
+	present_resources[res_curg] = 1;
+	res_names[res_curg] = strdup("cursor_group");
+	present_resources[res_dlg] = 1;
+	res_names[res_dlg] = strdup("dialog");
+	present_resources[res_dlgex] = 1;
+	res_names[res_dlgex] = strdup("dialogex");
+	present_resources[res_fnt] = 1;
+	res_names[res_fnt] = strdup("font");
+	present_resources[res_fntdir] = 1;
+	res_names[res_fntdir] = strdup("fontdir");
+	present_resources[res_ico] = 1;
+	res_names[res_ico] = strdup("icon");
+	present_resources[res_icog] = 1;
+	res_names[res_icog] = strdup("icon_group");
+	present_resources[res_men] = 1;
+	res_names[res_men] = strdup("menu");
+	present_resources[res_menex] = 1;
+	res_names[res_menex] = strdup("menuex");
+	present_resources[res_rdt] = 1;
+	res_names[res_rdt] = strdup("rcdata");
+	present_resources[res_stt] = 1;
+	res_names[res_stt] = strdup("stringtable");
+	present_resources[res_usr] = 1;
+	res_names[res_usr] = strdup("user");
+	present_resources[res_msg] = 1;
+	res_names[res_msg] = strdup("messagetable");
+	present_resources[res_ver] = 1;
+	res_names[res_ver] = strdup("versioninfo");
+	present_resources[res_dlginit] = 1;
+	res_names[res_dlginit] = strdup("dlginit");
+	present_resources[res_toolbar] = 1;
+	res_names[res_toolbar] = strdup("toolbar");
+	present_resources[res_anicur] = 1;
+	res_names[res_anicur] = strdup("ani_cursor");
+	present_resources[res_aniico] = 1;
+	res_names[res_aniico] = strdup("ani_icon");
+
+	for(res_type = res_0; res_type <= res_usr; res_type++) {
+		if(!present_resources[res_type]) {
+			continue;
+		}
+		if(nb_resources[res_type][lang_type_normal] > 0) {
+			if(nb_resources[res_type][lang_type_master] && nb_resources[res_type][lang_type_neutral]) {
+				fprintf(stderr, "Type %s:\n", res_names[res_type]);
+				fprintf(stderr, "There are both a NEUTRAL and a MASTER version for %s, along with additional localized versions. The NEUTRAL versions will not be checked against other versions.\n", res_names[res_type]);
+			} else if(nb_resources[res_type][lang_type_neutral]) {
+				fprintf(stderr, "Type %s:\n", res_names[res_type]);
+				fprintf(stderr, "There are no MASTER version, but there are some NEUTRAL versions for %s, so will use those instead of MASTER for comparison.\n", res_names[res_type]);
+				list_resources[res_type][lang_type_master] = list_resources[res_type][lang_type_neutral];
+				nb_resources[res_type][lang_type_master] = nb_resources[res_type][lang_type_neutral];
+			} else if(!nb_resources[res_type][lang_type_master]) {
+				fprintf(stderr, "Type %s:\n", res_names[res_type]);
+				fprintf(stderr, "There are no NEUTRAL nor MASTER versions for %s, but there are some other localized versions. No comparison will be done at all.\n", res_names[res_type]);
+			}
+		} else {
+			if(nb_resources[res_type][lang_type_master] && nb_resources[res_type][lang_type_neutral]) {
+				fprintf(stderr, "Type %s:\n", res_names[res_type]);
+				fprintf(stderr, "There are both a NEUTRAL and a MASTER versions for %s, but no other localized version. No comparison will be done at all.\n", res_names[res_type]);
+			} else if(nb_resources[res_type][lang_type_master]) {
+				fprintf(stderr, "Type %s:\n", res_names[res_type]);
+				fprintf(stderr, "There are only MASTER versions for %s. No comparison will be done at all.\n", res_names[res_type]);
+			} else if(nb_resources[res_type][lang_type_neutral]) {
+				/* fprintf(stderr, "There are only NEUTRAL versions for %s. No comparison will be done at all.\n", res_names[res_type]); */
+			} else {
+				/* fprintf(stderr, "There are no versions at all for %s. No comparison will be done at all.\n", res_names[res_type]); */
+			}
+		}
+
+		presence = malloc(nb_resources[res_type][lang_type_master]*sizeof(int *));
+		for(i = 0; i < nb_resources[res_type][lang_type_master]; i++) {
+			presence[i] = calloc(NB_LANG, sizeof(int));
+			presence[i][MASTER_LANGUAGE] = -1;
+		}
+
+		for(i = 0; i < nb_resources[res_type][lang_type_normal]; i++) {
+			for(j = 0; j < nb_resources[res_type][lang_type_master]; j++) {
+				nameid = strdup(get_nameid_str(list_resources[res_type][lang_type_normal][i]->name));
+				if(!strcmp(nameid, get_nameid_str(list_resources[res_type][lang_type_master][j]->name))) {
+					if(compare(list_resources[res_type][lang_type_normal][i], list_resources[res_type][lang_type_master][j])) {
+						presence[j][get_language_id(list_resources[res_type][lang_type_normal][i])] = 2;
+						/* fprintf(stderr, "Differences in type %s, ID %s, for language %s\n", res_names[res_type], nameid, get_language_name(get_language_id(list_resources[res_type][lang_type_normal][i]))); */
+					} else {
+						presence[j][get_language_id(list_resources[res_type][lang_type_normal][i])] = 1;
+					}
+				}
+				free(nameid);
+			}
+		}
+
+		problems = malloc(sizeof(char *));
+		problems[0] = strdup("");
+		nb_problems = 0;
+		last_problem = -1;
+		for(i = 0; i < NB_LANG; i++) {
+			complete = 1;
+			needs_work = 0;
+			partial = 0;
+			for(j = 0; j < nb_resources[res_type][lang_type_master]; j++) {
+				if(presence[j][i]) {
+					partial = 1;
+					if(presence[j][i] == 2) {
+						needs_work = 1;
+						problems = realloc(problems, (++nb_problems+1)*sizeof(char *));
+						problems[nb_problems] = malloc(strlen(get_nameid_str(list_resources[res_type][lang_type_master][j]->name)) + 9);
+						sprintf(problems[nb_problems], "DIFF %s %02x", get_nameid_str(list_resources[res_type][lang_type_master][j]->name), i);
+						if(last_problem == i) {
+							problems[nb_problems-1] = realloc(problems[nb_problems-1], strlen(problems[nb_problems-1]) + 3);
+							strcat(problems[nb_problems-1], " \\");
+						} else {
+							last_problem = i;
+						}
+					}
+				} else {
+					complete = 0;
+					problems = realloc(problems, (++nb_problems+1)*sizeof(char *));
+					problems[nb_problems] = malloc(strlen(get_nameid_str(list_resources[res_type][lang_type_master][j]->name)) + 8);
+					sprintf(problems[nb_problems], "ABS %s %02x", get_nameid_str(list_resources[res_type][lang_type_master][j]->name), i);
+					if(last_problem == i) {
+						problems[nb_problems-1] = realloc(problems[nb_problems-1], strlen(problems[nb_problems-1]) + 3);
+						strcat(problems[nb_problems-1], " \\");
+					} else {
+						last_problem = i;
+					}
+				}
+			}
+			if(complete && partial && !needs_work) {
+				/* Support is complete, no need to do anything */
+				/* fprintf(stderr, "Support for language %s is complete for %s.\n", get_language_name(i), res_names[res_type]); */
+				printf(".");
+			} else if(complete && partial && needs_work) {
+				/* Support is incomplete (differing resources), needs work */
+				/* fprintf(stderr, "Support for language %s is incomplete (differing resources) for %s.\n", get_language_name(i), res_names[res_type]); */
+				printf("x");
+			} else if(!complete && partial && !needs_work) {
+				/* Support is incomplete (missing resources), needs work */
+				/* fprintf(stderr, "Support for language %s is incomplete (missing resources) for %s.\n", get_language_name(i), res_names[res_type]); */
+				printf("-");
+			} else if(!complete && partial && needs_work) {
+				/* Support is incomplete (missing and differing resources), needs work */
+				/* fprintf(stderr, "Support for language %s is incomplete (missing and differing resources) for %s.\n", get_language_name(i), res_names[res_type]); */
+				printf("+");
+			} else if(!complete && !partial) {
+				/* Support is totally absent, might be interesting to do */
+				/* fprintf(stderr, "Support for language %s is absent for %s.\n", get_language_name(i), res_names[res_type]); */
+				printf(" ");
+			} else {
+				/* Support is not relevant, no need to do anything */
+				/* fprintf(stderr, "Support for language %s is not relevant for %s.\n", get_language_name(i), res_names[res_type]); */
+				printf("n");
+			}
+		}
+		printf("\n");
+		for(i = 1; i <= nb_problems; i++) {
+			printf("%s\n", problems[i]);
+			free(problems[i]);
+		}
+		free(problems[0]);
+		free(problems);
+		for(i = 0; i < nb_resources[res_type][lang_type_master]; i++)
+			free(presence[i]);
+		free(presence);
+	}
+}
diff -u /dev/null tools/wrc/translation.h
--- /dev/null	2002-08-30 19:31:37.000000000 -0400
+++ tools/wrc/translation.h	2003-10-31 17:04:07.000000000 -0500
@@ -0,0 +1,44 @@
+/*
+ * Verification of resources translations prototypes
+ *
+ * Copyright 2003 Vincent Béron
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ */
+
+#ifndef __WRC_TRANSLATION_H
+#define __WRC_TRANSLATION_H
+
+#include <string.h>
+
+#include "dumpres.h"
+
+#define MASTER_LANGUAGE LANG_ENGLISH
+#define NB_LANG 0x94
+
+enum lang_type_e {
+	lang_type_master = 0,
+	lang_type_neutral,
+	lang_type_normal
+};
+
+void verify_translations(resource_t *top);
+
+#define PRETTYPRINTLANG(langid) \
+	if(LANG_##langid == lid) { \
+		return #langid; \
+	}
+
+#endif

[Index of Archives]     [Gimp for Windows]     [Red Hat]     [Samba]     [Yosemite Camping]     [Graphics Cards]     [Wine Home]

  Powered by Linux