/* vim:expandtab:ts=2 sw=2: */ /* Grafx2 - The Ultimate 256-color bitmap paint program Copyright 2008 Yves Rizoud Copyright 2008 Franck Charlet Copyright 2008 Adrien Destugues Copyright 1996-2001 Sunset Design (Guillaume Dorme & Karl Maritaud) Grafx2 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; version 2 of the License. Grafx2 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 Grafx2; if not, see */ // Pour désactiver le support TrueType, définir NOTTF // To disable TrueType support, define NOTTF #include #include #include // tolower() // TrueType #ifndef NOTTF #if defined(__macosx__) #include #else #include #endif #if defined(__linux__) #if defined(__macosx__) #include #import #import #else #include #endif #endif #endif #include // SFont #include "SFont.h" #include "struct.h" #include "global.h" #include "sdlscreen.h" #include "io.h" #include "errors.h" typedef struct T_Font { char * Name; int Is_truetype; int Is_bitmap; char Label[22]; // Liste chainée simple struct T_Font * Next; struct T_Font * Previous; } T_Font; // Liste chainée des polices de texte T_Font * font_list_start; int Nb_fonts; // Inspiré par Allegro #define EXTID(a,b,c) ((((a)&255)<<16) | (((b)&255)<<8) | (((c)&255))) #define EXTID4(a,b,c,d) ((((a)&255)<<24) | (((b)&255)<<16) | (((c)&255)<<8) | (((d)&255))) int Compare_fonts(T_Font * font_1, T_Font * font_2) { if (font_1->Is_bitmap && !font_2->Is_bitmap) return -1; if (font_2->Is_bitmap && !font_1->Is_bitmap) return 1; return strcmp(font_1->Label, font_2->Label); } // Ajout d'une fonte à la liste. void Add_font(const char *name) { char * font_name; T_Font * font; int size=strlen(name)+1; int index; // Détermination du type: #if defined(__macosx__) if (size < 6) return; char strFontName[512]; CFStringRef CFSFontName;// = CFSTR(name); CFSFontName = CFStringCreateWithBytes(NULL, (UInt8 *) name, size - 1, kCFStringEncodingASCII, false); // Fix some funny names CFStringGetCString(CFSFontName, strFontName, 512, kCFStringEncodingASCII); // Now we have a printable font name, use it name = strFontName; #else if (size<5 || name[size-5]!='.') return; #endif font = (T_Font *)malloc(sizeof(T_Font)); switch (EXTID(tolower(name[size-4]), tolower(name[size-3]), tolower(name[size-2]))) { case EXTID('t','t','f'): case EXTID('f','o','n'): case EXTID('o','t','f'): case EXTID('p','f','b'): font->Is_truetype = 1; font->Is_bitmap = 0; break; case EXTID('b','m','p'): case EXTID('g','i','f'): case EXTID('j','p','g'): case EXTID('l','b','m'): case EXTID('p','c','x'): case EXTID('p','n','g'): case EXTID('t','g','a'): case EXTID('t','i','f'): case EXTID('x','c','f'): case EXTID('x','p','m'): case EXTID('.','x','v'): font->Is_truetype = 0; font->Is_bitmap = 1; break; default: #if defined(__macosx__) if(strcasecmp(&name[size-6], "dfont") == 0) { font->Is_truetype = 1; font->Is_bitmap = 0; } else { free(font); return; } #else free(font); return; #endif } font->Name = (char *)malloc(size); strcpy(font->Name, name); // Label strcpy(font->Label, " "); if (font->Is_truetype) font->Label[17]=font->Label[18]='T'; // Logo TT font_name=Find_last_slash(font->Name); if (font_name==NULL) font_name=font->Name; else font_name++; for (index=0; index < 17 && font_name[index]!='\0' && font_name[index]!='.'; index++) font->Label[index]=font_name[index]; // Gestion Liste font->Next = NULL; font->Previous = NULL; if (font_list_start==NULL) { // Premiere (liste vide) font_list_start = font; Nb_fonts++; } else { int compare; compare = Compare_fonts(font, font_list_start); if (compare<=0) { if (compare==0 && !strcmp(font->Name, font_list_start->Name)) { // Doublon free(font->Name); free(font); return; } // Avant la premiere font->Next=font_list_start; font_list_start=font; Nb_fonts++; } else { T_Font *searched_font; searched_font=font_list_start; while (searched_font->Next && (compare=Compare_fonts(font, searched_font->Next))>0) searched_font=searched_font->Next; // Après searched_font if (compare==0 && strcmp(font->Name, searched_font->Next->Name)==0) { // Doublon free(font->Name); free(font); return; } font->Next=searched_font->Next; searched_font->Next=font; Nb_fonts++; } } } // Trouve le nom d'une fonte par son numéro char * Font_name(int index) { T_Font *font = font_list_start; if (index<0 ||index>=Nb_fonts) return ""; while (index--) font = font->Next; return font->Name; } // Trouve le libellé d'affichage d'une fonte par son numéro // Renvoie un pointeur sur un buffer statique de 20 caracteres. char * Font_label(int index) { T_Font *font; static char label[20]; strcpy(label, " "); // Recherche de la fonte font = font_list_start; if (index<0 ||index>=Nb_fonts) return label; while (index--) font = font->Next; // Libellé strcpy(label, font->Label); return label; } // Vérifie si une fonte donnée est TrueType int TrueType_font(int index) { T_Font *font = font_list_start; if (index<0 ||index>=Nb_fonts) return 0; while (index--) font = font->Next; return font->Is_truetype; } // Initialisation à faire une fois au début du programme void Init_text(void) { char directory_name[MAX_PATH_CHARACTERS]; #ifndef NOTTF // Initialisation de TTF TTF_Init(); #endif // Initialisation des fontes font_list_start = NULL; Nb_fonts=0; // Parcours du répertoire "fonts" strcpy(directory_name, Data_directory); strcat(directory_name, "fonts"); For_each_file(directory_name, Add_font); #if defined(__WIN32__) // Parcours du répertoire systeme windows "fonts" #ifndef NOTTF { char * WindowsPath=getenv("windir"); if (WindowsPath) { sprintf(directory_name, "%s\\FONTS", WindowsPath); For_each_file(directory_name, Add_font); } } #endif #elif defined(__macosx__) // Récupération de la liste des fonts avec fontconfig #ifndef NOTTF int i,number; char home_dir[MAXPATHLEN]; char *font_path_list[3] = { "/System/Library/Fonts", "/Library/Fonts" }; number = 3; // Make sure we also search into the user's fonts directory CFURLRef url = (CFURLRef) CFCopyHomeDirectoryURLForUser(NULL); CFURLGetFileSystemRepresentation(url, true, (UInt8 *) home_dir, MAXPATHLEN); strcat(home_dir, "/Library/Fonts"); font_path_list[2] = home_dir; for(i=0;ifg, black->bg for (index=0; index < Texte8Bit->w * Texte8Bit->h; index++) { int sum; sum = Main_palette[*(new_brush+index)].R + Main_palette[*(new_brush+index)].G + Main_palette[*(new_brush+index)].B; if (sum < 128 * 3) *(new_brush+index)=Back_color; else *(new_brush+index)=Fore_color; } } *width=Texte8Bit->w; *height=Texte8Bit->h; SDL_FreeSurface(Texte8Bit); TTF_CloseFont(font); return new_brush; } #endif byte *Render_text_SFont(const char *str, int font_number, int *width, int *height) { SFont_Font *font; SDL_Surface * TexteColore; SDL_Surface * Texte8Bit; SDL_Surface *Surface_fonte; byte * new_brush; SDL_Rect rectangle; // Chargement de la fonte Surface_fonte=IMG_Load(Font_name(font_number)); if (!Surface_fonte) { DEBUG("Font loading failed",0); return NULL; } font=SFont_InitFont(Surface_fonte); if (!font) { DEBUG("Font init failed",1); return NULL; } // Calcul des dimensions *height=SFont_TextHeight(font); *width=SFont_TextWidth(font, str); // Allocation d'une surface SDL TexteColore=SDL_CreateRGBSurface(SDL_SWSURFACE, *width, *height, 24, 0, 0, 0, 0); // Fill with backcolor rectangle.x=0; rectangle.y=0; rectangle.w=*width; rectangle.h=*height; SDL_FillRect(TexteColore, &rectangle, SDL_MapRGB( TexteColore->format, Main_palette[Back_color].R, Main_palette[Back_color].G, Main_palette[Back_color].B )); // Rendu du texte SFont_Write(TexteColore, font, 0, 0, str); if (!TexteColore) { DEBUG("Rendering failed",2); SFont_FreeFont(font); return NULL; } Texte8Bit=SDL_DisplayFormat(TexteColore); SDL_FreeSurface(TexteColore); new_brush=Surface_to_bytefield(Texte8Bit, NULL); if (!new_brush) { DEBUG("Converting failed",3); SDL_FreeSurface(TexteColore); SDL_FreeSurface(Texte8Bit); SFont_FreeFont(font); return NULL; } SDL_FreeSurface(Texte8Bit); SFont_FreeFont(font); return new_brush; } #ifdef NOTTF #define TTFONLY __attribute__((unused)) #else #define TTFONLY #endif // Crée une brosse à partir des paramètres de texte demandés. // Si cela réussit, la fonction place les dimensions dans width et height, // et retourne l'adresse du bloc d'octets. byte *Render_text(const char *str, int font_number, TTFONLY int size, int TTFONLY antialias, TTFONLY int bold, TTFONLY int italic, int *width, int *height) { T_Font *font = font_list_start; int index=font_number; // Verification type de la fonte if (font_number<0 ||font_number>=Nb_fonts) return NULL; while (index--) font = font->Next; if (font->Is_truetype) { #ifndef NOTTF return Render_text_TTF(str, font_number, size, antialias, bold, italic, width, height); #else return NULL; #endif } else { return Render_text_SFont(str, font_number, width, height); } }