The Battle for Wesnoth  1.19.10+dev
font_config.cpp
Go to the documentation of this file.
1 /*
2  Copyright (C) 2016 - 2025
3  by Chris Beck<render787@gmail.com>
4  Part of the Battle for Wesnoth Project https://www.wesnoth.org/
5 
6  This program is free software; you can redistribute it and/or modify
7  it under the terms of the GNU General Public License as published by
8  the Free Software Foundation; either version 2 of the License, or
9  (at your option) any later version.
10  This program is distributed in the hope that it will be useful,
11  but WITHOUT ANY WARRANTY.
12 
13  See the COPYING file for more details.
14 */
15 
16 #include "font/font_config.hpp"
17 #include "font/error.hpp"
18 
19 #include "config.hpp"
20 #include "log.hpp"
21 #include "tstring.hpp"
22 
23 #include "filesystem.hpp"
24 
25 #include "serialization/parser.hpp"
27 
28 #include <sstream>
29 
30 
31 #include <fontconfig/fontconfig.h>
32 
33 static lg::log_domain log_font("font");
34 #define DBG_FT LOG_STREAM(debug, log_font)
35 #define LOG_FT LOG_STREAM(info, log_font)
36 #define WRN_FT LOG_STREAM(warn, log_font)
37 #define ERR_FT LOG_STREAM(err, log_font)
38 
39 namespace font
40 {
41 namespace
42 {
43 
44 /** Records the game's families for sans serif, script, and monospace fonts */
45 struct font_families
46 {
47  font_families() = default;
48 
49  explicit font_families(const config& cfg)
50  : sans(cfg["family_order"].t_str())
51  , mono(cfg["family_order_monospace"].t_str())
52  , script(cfg["family_order_script"].t_str())
53  {
54  if(mono.empty()) {
55  ERR_FT << "No monospace font family defined, falling back to sans serif";
56  mono = sans;
57  }
58 
59  if(script.empty()) {
60  ERR_FT << "No script font family defined, falling back to sans serif";
61  script = sans;
62  }
63  }
64 
68 };
69 
70 font_families families;
71 
72 } //namespace
73 
74 /***
75  * Public interface
76  */
77 
79 try {
80  auto stream = preprocess_file(filesystem::get_wml_location("hardwired/fonts.cfg").value());
81  families = font_families{ io::read(*stream).mandatory_child("fonts") };
82  return true;
83 
84 } catch(const utils::bad_optional_access&) {
85  ERR_FT << "could not resolve path to fonts.cfg, file not found";
86  return false;
87 
88 } catch(const config::error& e) {
89  ERR_FT << "could not read fonts.cfg:\n" << e.message;
90  return false;
91 }
92 
94 {
95  switch(fclass) {
97  return families.mono;
99  return families.script;
100  default:
101  return families.sans;
102  }
103 }
104 
105 /***
106  * Manager member functions
107  */
108 
110 {
111  std::string font_path = game_config::path + "/fonts";
112  if (!FcConfigAppFontAddDir(FcConfigGetCurrent(),
113  reinterpret_cast<const FcChar8 *>(font_path.c_str())))
114  {
115  ERR_FT << "Could not load the true type fonts";
116  throw font::error("font config lib failed to add the font path: '" + font_path + "'");
117  }
118 
119  std::string font_file = font_path + "/fonts.conf";
120  std::string font_file_contents = filesystem::read_file(font_file);
121 
122 // msys2 crosscompiling for windows for whatever reason makes the cache directory prefer using drives other than C:
123 // ie - D:\a\msys64\var\cache\fontconfig
124 // fontconfig also does not seem to provide a way to set the cachedir for a specific platform
125 // so load the fonts.conf file into memory and only for windows insert the cachedir configuration
126 #ifdef _WIN32
127  font_file_contents.insert(font_file_contents.find("</fontconfig>"), "<cachedir>"+filesystem::get_cache_dir()+"</cachedir>\n");
128 #endif
129 
130  if(!FcConfigParseAndLoadFromMemory(FcConfigGetCurrent(),
131  reinterpret_cast<const FcChar8*>(font_file_contents.c_str()),
132  FcFalse))
133  {
134  ERR_FT << "Could not load local font configuration";
135  throw font::error("font config lib failed to find font.conf: '" + font_file + "'");
136  }
137  else
138  {
139  LOG_FT << "Local font configuration loaded";
140  }
141 }
142 
144 {
145  FcConfigAppFontClear(FcConfigGetCurrent());
146 }
147 
148 } // end namespace font
A config object defines a single node in a WML file, with access to child nodes.
Definition: config.hpp:158
config & mandatory_child(config_key_type key, int n=0)
Returns the nth child with the given key, or throws an error if there is none.
Definition: config.cpp:362
bool empty() const
Definition: tstring.hpp:195
Definitions for the interface to Wesnoth Markup Language (WML).
Declarations for File-IO.
t_string mono
Definition: font_config.cpp:66
#define LOG_FT
Definition: font_config.cpp:35
t_string script
Definition: font_config.cpp:67
static lg::log_domain log_font("font")
#define ERR_FT
Definition: font_config.cpp:37
t_string sans
Definition: font_config.cpp:65
Standard logging facilities (interface).
std::string get_cache_dir()
Definition: filesystem.cpp:866
utils::optional< std::string > get_wml_location(const std::string &path, const utils::optional< std::string > &current_dir)
Returns a translated path to the actual file or directory, if it exists.
std::string read_file(const std::string &fname)
Basic disk I/O - read file.
Graphical text output.
const t_string & get_font_families(family_class fclass)
Returns the currently defined fonts.
Definition: font_config.cpp:93
bool load_font_config()
Definition: font_config.cpp:78
std::string path
Definition: filesystem.cpp:93
config read(std::istream &in, abstract_validator *validator)
Definition: parser.cpp:627
filesystem::scoped_istream preprocess_file(const std::string &fname, preproc_map *defines)
Function to use the WML preprocessor on a file.
#define e