]> git.wh0rd.org - fontconfig.git/blobdiff - src/fcdir.c
Reimplement FcConfigAppFontAddDir; function was lost in 2.4.0.
[fontconfig.git] / src / fcdir.c
index 6db0b8bddf3c15b96de1be0b3476817fe85de5d4..d08299269f0f85bcf937c2d224cab9f4c91c3f3e 100644 (file)
@@ -35,26 +35,17 @@ FcFileIsDir (const FcChar8 *file)
     return S_ISDIR(statb.st_mode);
 }
 
-FcBool
-FcFileScanConfig (FcFontSet    *set,
-                 FcStrSet      *dirs,
-                 FcGlobalCache *cache,
-                 FcBlanks      *blanks,
-                 const FcChar8 *file,
-                 FcBool        force,
-                 FcConfig      *config)
+static FcBool
+FcFileScanFontConfig (FcFontSet                *set,
+                     FcBlanks          *blanks,
+                     const FcChar8     *file,
+                     FcConfig          *config)
 {
-    int                        id;
-    FcPattern          *font;
-    FcBool             ret = FcTrue;
-    FcBool             isDir;
-    int                        count = 0;
+    FcPattern  *font;
+    FcBool     ret = FcTrue;
+    int                id;
+    int                count = 0;
     
-    if (config && !FcConfigAcceptFilename (config, file))
-       return FcTrue;
-
-    if (force)
-       cache = 0;
     id = 0;
     do
     {
@@ -70,21 +61,31 @@ FcFileScanConfig (FcFontSet *set,
        font = FcFreeTypeQuery (file, id, blanks, &count);
        if (FcDebug () & FC_DBG_SCAN)
            printf ("done\n");
-       isDir = FcFalse;
-       if (!font && FcFileIsDir (file))
+
+       /*
+        * Edit pattern with user-defined rules
+        */
+       if (config && !FcConfigSubstituteWithPat (config, font, NULL, FcMatchScan))
        {
-           isDir = FcTrue;
-           ret = FcStrSetAdd (dirs, file);
+           FcPatternDestroy (font);
+           font = NULL;
+           ret = FcFalse;
        }
+
        /*
         * Add the font
         */
        if (font && (!config || FcConfigAcceptFont (config, font)))
        {
+           if (FcDebug() & FC_DBG_SCANV)
+           {
+               printf ("Final font pattern:\n");
+               FcPatternPrint (font);
+           }
            if (!FcFontSetAdd (set, font))
            {
                FcPatternDestroy (font);
-               font = 0;
+               font = NULL;
                ret = FcFalse;
            }
        }
@@ -95,61 +96,66 @@ FcFileScanConfig (FcFontSet *set,
     return ret;
 }
 
+FcBool
+FcFileScanConfig (FcFontSet    *set,
+                 FcStrSet      *dirs,
+                 FcBlanks      *blanks,
+                 const FcChar8 *file,
+                 FcConfig      *config)
+{
+    if (FcFileIsDir (file))
+       return FcStrSetAdd (dirs, file);
+    else
+       return FcFileScanFontConfig (set, blanks, file, config);
+}
+
 FcBool
 FcFileScan (FcFontSet      *set,
            FcStrSet        *dirs,
-           FcGlobalCache   *cache,
+           FcFileCache     *cache, /* XXX unused */
            FcBlanks        *blanks,
            const FcChar8   *file,
            FcBool          force)
 {
-    return FcFileScanConfig (set, dirs, cache, blanks, file, force, 0);
+    return FcFileScanConfig (set, dirs, blanks, file, NULL);
 }
 
 /*
- * Scan 'dir', adding font files to 'set' and
- * subdirectories to 'dirs'
+ * Strcmp helper that takes pointers to pointers, copied from qsort(3) manpage
  */
+static int
+cmpstringp(const void *p1, const void *p2)
+{
+    return strcmp(* (char **) p1, * (char **) p2);
+}
 
-FcBool
-FcDirScanConfig (FcFontSet     *set,
-                FcStrSet       *dirs,
-                FcGlobalCache  *cache,
-                FcBlanks       *blanks,
-                const FcChar8  *dir,
-                FcBool         force,
-                FcConfig       *config)
+/*
+ * Scan the specified directory and construct a cache of its contents
+ */
+FcCache *
+FcDirCacheScan (const FcChar8 *dir, FcConfig *config)
 {
     DIR                        *d;
     struct dirent      *e;
+    FcStrSet           *files;
+    FcStrSet           *dirs;
     FcChar8            *file;
     FcChar8            *base;
     FcBool             ret = FcTrue;
-    FcFontSet          *tmpSet;
+    FcFontSet          *set;
     int                        i;
+    FcBlanks           *blanks = FcConfigGetBlanks (config);
+    FcCache            *cache = NULL;
 
-    if (config && !FcConfigAcceptFilename (config, dir))
-       return FcTrue;
+    if (FcDebug () & FC_DBG_FONTSET)
+       printf ("cache scan dir %s\n", dir);
 
-    if (config)
-        FcConfigAddFontDir (config, dir);
-
-    if (!force)
-    {
-       /*
-        * Check ~/.fonts.cache-<version> file
-        */
-       if (cache && FcGlobalCacheReadDir (set, dirs, cache, (char *)dir, config))
-           return FcTrue;
-
-       if (FcDirCacheValid (dir) && FcDirCacheRead (set, dirs, dir, config))
-           return FcTrue;
-    }
-    
     /* freed below */
     file = (FcChar8 *) malloc (strlen ((char *) dir) + 1 + FC_MAX_FILE_LEN + 1);
-    if (!file)
-       return FcFalse;
+    if (!file) {
+       ret = FcFalse;
+       goto bail;
+    }
 
     strcpy ((char *) file, (char *) dir);
     strcat ((char *) file, "/");
@@ -161,64 +167,131 @@ FcDirScanConfig (FcFontSet       *set,
     d = opendir ((char *) dir);
     if (!d)
     {
-       free (file);
        /* Don't complain about missing directories */
        if (errno == ENOENT)
-           return FcTrue;
-       return FcFalse;
+           ret = FcTrue;
+       else
+           ret = FcFalse;
+       goto bail_1;
     }
 
-    tmpSet = FcFontSetCreate();
-    if (!tmpSet)
-    {  
-       free (file);
-       return FcFalse;
+    set = FcFontSetCreate();
+    if (!set) 
+    {
+       ret = FcFalse;
+       goto bail0;
     }
 
-    while (ret && (e = readdir (d)))
+    files = FcStrSetCreate ();
+    if (!files)
+    {
+       ret = FcFalse;
+       goto bail1;
+    }
+    while ((e = readdir (d)))
     {
        if (e->d_name[0] != '.' && strlen (e->d_name) < FC_MAX_FILE_LEN)
        {
            strcpy ((char *) base, (char *) e->d_name);
-           ret = FcFileScanConfig (tmpSet, dirs, cache, blanks, file, force, config);
+           if (!FcStrSetAdd (files, file)) {
+               ret = FcFalse;
+               goto bail2;
+           }
        }
     }
-    free (file);
-    closedir (d);
+
     /*
-     * Now that the directory has been scanned,
-     * add the cache entry 
+     * Sort files to make things prettier
      */
-    if (ret && cache)
-       FcGlobalCacheUpdate (cache, (char *)dir, tmpSet, config);
+    qsort(files->strs, files->num, sizeof(FcChar8 *), cmpstringp);
+    
+    dirs = FcStrSetCreate ();
+    if (!dirs)
+       goto bail2;
+    
+    /*
+     * Scan file files to build font patterns
+     */
+    for (i = 0; i < files->num; i++)
+       FcFileScanConfig (set, dirs, blanks, files->strs[i], config);
+    
+    /*
+     * Build the cache object
+     */
+    cache = FcDirCacheBuild (set, dir, dirs);
+    if (!cache)
+       goto bail3;
+    
+    /*
+     * Write out the cache file, ignoring any troubles
+     */
+    FcDirCacheWrite (cache, config);
+    
+ bail3:
+    FcStrSetDestroy (dirs);
+ bail2:
+    FcStrSetDestroy (files);
+ bail1:
+    FcFontSetDestroy (set);
+    
+ bail0:
+    closedir (d);
+    
+ bail_1:
+    free (file);
+ bail:
+    return cache;
+}
 
-    for (i = 0; i < tmpSet->nfont; i++)
-       FcFontSetAdd (set, tmpSet->fonts[i]);
+/*
+ * Read (or construct) the cache for a directory
+ */
+FcCache *
+FcDirCacheRead (const FcChar8 *dir, FcBool force, FcConfig *config)
+{
+    FcCache            *cache = NULL;
 
-    if (tmpSet->fonts)
-    {
-       FcMemFree (FC_MEM_FONTPTR, tmpSet->sfont * sizeof (FcPattern *));
-       free (tmpSet->fonts);
-    }
-    FcMemFree (FC_MEM_FONTSET, sizeof (FcFontSet));
-    free (tmpSet);
-       
-    return ret;
+    if (config && !FcConfigAcceptFilename (config, dir))
+       return NULL;
+
+    /* Try to use existing cache file */
+    if (!force)
+       cache = FcDirCacheLoad (dir, config, NULL);
+    
+    /* Not using existing cache file, construct new cache */
+    if (!cache)
+       cache = FcDirCacheScan (dir, config);
+    
+    return cache;
+}
+
+FcBool
+FcDirScanConfig (FcFontSet     *set,
+                FcStrSet       *dirs,
+                FcBlanks       *blanks,
+                const FcChar8  *dir,
+                FcBool         force,
+                FcConfig       *config)
+{
+    return FcFalse; /* XXX deprecated */
 }
 
 FcBool
 FcDirScan (FcFontSet       *set,
           FcStrSet         *dirs,
-          FcGlobalCache    *cache,
+          FcFileCache      *cache, /* XXX unused */
           FcBlanks         *blanks,
           const FcChar8    *dir,
           FcBool           force)
 {
-    return FcDirScanConfig (set, dirs, cache, blanks, dir, force, 0);
+    return FcFalse; /* XXX deprecated */
 }
 
 FcBool
 FcDirSave (FcFontSet *set, FcStrSet * dirs, const FcChar8 *dir)
 {
-    return FcDirCacheWrite (set, dirs, dir);
+    return FcFalse; /* XXX deprecated */
 }
+#define __fcdir__
+#include "fcaliastail.h"
+#undef __fcdir__