2002-07-07 02:00:43 +02:00
|
|
|
|
/*
|
2002-07-12 23:06:03 +02:00
|
|
|
|
* $XFree86: xc/lib/fontconfig/src/fclang.c,v 1.3 2002/07/08 07:31:53 keithp Exp $
|
2002-07-07 02:00:43 +02:00
|
|
|
|
*
|
|
|
|
|
* Copyright <EFBFBD> 2002 Keith Packard, member of The XFree86 Project, Inc.
|
|
|
|
|
*
|
|
|
|
|
* Permission to use, copy, modify, distribute, and sell this software and its
|
|
|
|
|
* documentation for any purpose is hereby granted without fee, provided that
|
|
|
|
|
* the above copyright notice appear in all copies and that both that
|
|
|
|
|
* copyright notice and this permission notice appear in supporting
|
|
|
|
|
* documentation, and that the name of Keith Packard not be used in
|
|
|
|
|
* advertising or publicity pertaining to distribution of the software without
|
|
|
|
|
* specific, written prior permission. Keith Packard makes no
|
|
|
|
|
* representations about the suitability of this software for any purpose. It
|
|
|
|
|
* is provided "as is" without express or implied warranty.
|
|
|
|
|
*
|
|
|
|
|
* KEITH PACKARD DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
|
|
|
|
|
* INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
|
|
|
|
|
* EVENT SHALL KEITH PACKARD BE LIABLE FOR ANY SPECIAL, INDIRECT OR
|
|
|
|
|
* CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
|
|
|
|
|
* DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
|
|
|
|
* TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
|
|
|
* PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "fcint.h"
|
|
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
|
FcChar8 *lang;
|
|
|
|
|
FcCharSet charset;
|
|
|
|
|
} FcLangCharSet;
|
|
|
|
|
|
2002-07-08 09:31:53 +02:00
|
|
|
|
#include "../fc-lang/fclang.h"
|
2002-07-07 02:00:43 +02:00
|
|
|
|
|
|
|
|
|
#define NUM_LANG_CHAR_SET (sizeof (fcLangCharSets) / sizeof (fcLangCharSets[0]))
|
|
|
|
|
|
|
|
|
|
FcBool
|
2002-07-08 09:31:53 +02:00
|
|
|
|
FcFreeTypeSetLang (FcPattern *pattern,
|
|
|
|
|
FcCharSet *charset,
|
|
|
|
|
const FcChar8 *exclusiveLang)
|
2002-07-07 02:00:43 +02:00
|
|
|
|
{
|
2002-07-08 09:31:53 +02:00
|
|
|
|
int i;
|
|
|
|
|
FcChar32 missing;
|
|
|
|
|
FcBool hasLang = FcFalse;
|
|
|
|
|
const FcCharSet *exclusiveCharset = 0;
|
2002-07-07 02:00:43 +02:00
|
|
|
|
|
2002-07-08 09:31:53 +02:00
|
|
|
|
if (exclusiveLang)
|
|
|
|
|
exclusiveCharset = FcCharSetForLang (exclusiveLang);
|
2002-07-07 02:00:43 +02:00
|
|
|
|
for (i = 0; i < NUM_LANG_CHAR_SET; i++)
|
|
|
|
|
{
|
2002-07-08 09:31:53 +02:00
|
|
|
|
/*
|
|
|
|
|
* Check for Han charsets to make fonts
|
|
|
|
|
* which advertise support for a single language
|
|
|
|
|
* not support other Han languages
|
|
|
|
|
*/
|
|
|
|
|
if (exclusiveCharset &&
|
|
|
|
|
FcFreeTypeIsExclusiveLang (fcLangCharSets[i].lang) &&
|
|
|
|
|
fcLangCharSets[i].charset.leaves != exclusiveCharset->leaves)
|
|
|
|
|
{
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2002-07-07 02:00:43 +02:00
|
|
|
|
missing = FcCharSetSubtractCount (&fcLangCharSets[i].charset, charset);
|
|
|
|
|
if (FcDebug() & FC_DBG_SCANV)
|
2002-07-12 23:06:03 +02:00
|
|
|
|
{
|
|
|
|
|
if (missing && missing < 10)
|
|
|
|
|
{
|
|
|
|
|
FcCharSet *missed = FcCharSetSubtract (&fcLangCharSets[i].charset,
|
|
|
|
|
charset);
|
|
|
|
|
FcChar32 ucs4;
|
|
|
|
|
FcChar32 map[FC_CHARSET_MAP_SIZE];
|
|
|
|
|
FcChar32 next;
|
|
|
|
|
|
|
|
|
|
printf ("\n%s(%d) ", fcLangCharSets[i].lang, missing);
|
|
|
|
|
printf ("{");
|
|
|
|
|
for (ucs4 = FcCharSetFirstPage (missed, map, &next);
|
|
|
|
|
ucs4 != FC_CHARSET_DONE;
|
|
|
|
|
ucs4 = FcCharSetNextPage (missed, map, &next))
|
|
|
|
|
{
|
|
|
|
|
int i, j;
|
|
|
|
|
for (i = 0; i < FC_CHARSET_MAP_SIZE; i++)
|
|
|
|
|
if (map[i])
|
|
|
|
|
{
|
|
|
|
|
for (j = 0; j < 32; j++)
|
|
|
|
|
if (map[i] & (1 << j))
|
|
|
|
|
printf (" %04x", ucs4 + i * 32 + j);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
printf (" }\n\t");
|
|
|
|
|
FcCharSetDestroy (missed);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
printf ("%s(%d) ", fcLangCharSets[i].lang, missing);
|
|
|
|
|
}
|
2002-07-08 09:31:53 +02:00
|
|
|
|
if (!missing)
|
|
|
|
|
{
|
2002-07-07 02:00:43 +02:00
|
|
|
|
if (!FcPatternAddString (pattern, FC_LANG, fcLangCharSets[i].lang))
|
|
|
|
|
return FcFalse;
|
2002-07-08 09:31:53 +02:00
|
|
|
|
hasLang = FcTrue;
|
|
|
|
|
}
|
2002-07-07 02:00:43 +02:00
|
|
|
|
}
|
2002-07-08 09:31:53 +02:00
|
|
|
|
/*
|
|
|
|
|
* Make sure it has a lang entry
|
|
|
|
|
*/
|
|
|
|
|
if (!hasLang)
|
|
|
|
|
if (!FcPatternAddString (pattern, FC_LANG, (FcChar8 *) "x-unknown"))
|
|
|
|
|
return FcFalse;
|
|
|
|
|
|
2002-07-07 02:00:43 +02:00
|
|
|
|
if (FcDebug() & FC_DBG_SCANV)
|
|
|
|
|
printf ("\n");
|
|
|
|
|
return FcTrue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
FcLangResult
|
|
|
|
|
FcLangCompare (const FcChar8 *s1, const FcChar8 *s2)
|
|
|
|
|
{
|
|
|
|
|
const FcChar8 *orig_s1 = s1;
|
|
|
|
|
FcChar8 c1, c2;
|
|
|
|
|
FcLangResult result;
|
|
|
|
|
/*
|
|
|
|
|
* Compare ISO 639 language codes
|
|
|
|
|
*/
|
|
|
|
|
for (;;)
|
|
|
|
|
{
|
|
|
|
|
c1 = *s1++;
|
|
|
|
|
c2 = *s2++;
|
|
|
|
|
if (c1 == '\0' || c1 == '-')
|
|
|
|
|
break;
|
|
|
|
|
if (c2 == '\0' || c2 == '-')
|
|
|
|
|
break;
|
|
|
|
|
c1 = FcToLower (c1);
|
|
|
|
|
c2 = FcToLower (c2);
|
|
|
|
|
if (c1 != c2)
|
|
|
|
|
return FcLangDifferentLang; /* mismatching lang code */
|
|
|
|
|
}
|
|
|
|
|
if (!c1 && !c2)
|
|
|
|
|
return FcLangEqual;
|
|
|
|
|
/*
|
|
|
|
|
* Make x-* mismatch as if the lang part didn't match
|
|
|
|
|
*/
|
|
|
|
|
result = FcLangDifferentCountry;
|
|
|
|
|
if (orig_s1[0] == 'x' && (orig_s1[1] == '\0' || orig_s1[1] == '-'))
|
|
|
|
|
result = FcLangDifferentLang;
|
|
|
|
|
|
|
|
|
|
if (c1 == '\0' || c2 == '\0')
|
|
|
|
|
return result;
|
|
|
|
|
/*
|
|
|
|
|
* Compare ISO 3166 country codes
|
|
|
|
|
*/
|
|
|
|
|
for (;;)
|
|
|
|
|
{
|
|
|
|
|
c1 = *s1++;
|
|
|
|
|
c2 = *s2++;
|
|
|
|
|
if (!c1 || !c2)
|
|
|
|
|
break;
|
|
|
|
|
c1 = FcToLower (c1);
|
|
|
|
|
c2 = FcToLower (c2);
|
|
|
|
|
if (c1 != c2)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (c1 == c2)
|
|
|
|
|
return FcLangEqual;
|
|
|
|
|
else
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const FcCharSet *
|
|
|
|
|
FcCharSetForLang (const FcChar8 *lang)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
int country = -1;
|
|
|
|
|
for (i = 0; i < NUM_LANG_CHAR_SET; i++)
|
|
|
|
|
{
|
|
|
|
|
switch (FcLangCompare (lang, fcLangCharSets[i].lang)) {
|
|
|
|
|
case FcLangEqual:
|
|
|
|
|
return &fcLangCharSets[i].charset;
|
|
|
|
|
case FcLangDifferentCountry:
|
|
|
|
|
if (country == -1)
|
|
|
|
|
country = i;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (country == -1)
|
|
|
|
|
return 0;
|
|
|
|
|
return &fcLangCharSets[i].charset;
|
|
|
|
|
}
|