Allow custom font with adjusted size.
This commit is contained in:
parent
d944b4e4ef
commit
ae5a61f7ae
2 changed files with 69 additions and 39 deletions
|
|
@ -38,11 +38,16 @@ void style_InitFontsResource() {
|
||||||
namespace style {
|
namespace style {
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
CustomFont Custom;
|
QString Custom;
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
void SetCustomFont(const CustomFont &font) {
|
const QString &SystemFontTag() {
|
||||||
|
static const auto result = u"(system)"_q + QChar(0);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetCustomFont(const QString &font) {
|
||||||
Custom = font;
|
Custom = font;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -165,6 +170,8 @@ bool LoadCustomFont(const QString &filePath, const QString &familyName, int flag
|
||||||
}
|
}
|
||||||
|
|
||||||
[[nodiscard]] int ComputePixelSize(QFont font, uint32 flags, int size) {
|
[[nodiscard]] int ComputePixelSize(QFont font, uint32 flags, int size) {
|
||||||
|
constexpr auto kMaxSizeShift = 6;
|
||||||
|
|
||||||
const auto family = font.family();
|
const auto family = font.family();
|
||||||
const auto basic = GetFontOverride(flags);
|
const auto basic = GetFontOverride(flags);
|
||||||
if (family == basic) {
|
if (family == basic) {
|
||||||
|
|
@ -172,56 +179,68 @@ bool LoadCustomFont(const QString &filePath, const QString &familyName, int flag
|
||||||
}
|
}
|
||||||
auto copy = font;
|
auto copy = font;
|
||||||
copy.setFamily(basic);
|
copy.setFamily(basic);
|
||||||
const auto desired = QFontMetricsF(copy).capHeight();
|
const auto basicMetrics = QFontMetricsF(copy);
|
||||||
if (desired < 1.) {
|
const auto desiredHeight = basicMetrics.height();
|
||||||
|
const auto desiredCap = basicMetrics.capHeight();
|
||||||
|
if (desiredHeight < 1. || desiredCap < 1.) {
|
||||||
return size;
|
return size;
|
||||||
}
|
}
|
||||||
font.setPixelSize(size);
|
font.setPixelSize(size);
|
||||||
auto current = QFontMetricsF(font).capHeight();
|
const auto currentMetrics = QFontMetricsF(font);
|
||||||
constexpr auto kMaxSizeShift = 4;
|
auto currentHeight = currentMetrics.height();
|
||||||
if (current < 1. || std::abs(current - desired) < 0.2) {
|
auto currentCap = currentMetrics.capHeight();
|
||||||
|
const auto max = std::min(kMaxSizeShift, size - 1);
|
||||||
|
if (currentHeight < 1.
|
||||||
|
|| currentCap < 1.
|
||||||
|
|| std::abs(currentCap - desiredCap) < 0.2
|
||||||
|
|| std::abs(currentHeight - desiredHeight) < 0.2) {
|
||||||
return size;
|
return size;
|
||||||
} else if (current < desired) {
|
} else if (currentHeight < desiredHeight) {
|
||||||
for (auto i = 0; i != kMaxSizeShift; ++i) {
|
for (auto i = 0; i != max; ++i) {
|
||||||
const auto shift = i + 1;
|
const auto shift = i + 1;
|
||||||
font.setPixelSize(size + shift);
|
font.setPixelSize(size + shift);
|
||||||
const auto now = QFontMetricsF(font).capHeight();
|
const auto metrics = QFontMetricsF(font);
|
||||||
if (now > desired) {
|
const auto nowHeight = metrics.height();
|
||||||
return (now - desired * 2 < desired - current)
|
const auto nowCap = metrics.capHeight();
|
||||||
? (size + shift)
|
if (nowHeight > desiredHeight || nowCap > desiredCap) {
|
||||||
: (size + shift - 1);
|
return (size + shift - 1);
|
||||||
}
|
}
|
||||||
current = now;
|
currentHeight = nowHeight;
|
||||||
|
currentCap = nowCap;
|
||||||
}
|
}
|
||||||
return size + kMaxSizeShift;
|
return size + kMaxSizeShift;
|
||||||
} else {
|
} else {
|
||||||
for (auto i = 0; i != kMaxSizeShift; ++i) {
|
for (auto i = 0; i != max; ++i) {
|
||||||
const auto shift = i + 1;
|
const auto shift = i + 1;
|
||||||
font.setPixelSize(size - shift);
|
font.setPixelSize(size - shift);
|
||||||
const auto now = QFontMetricsF(font).capHeight();
|
const auto metrics = QFontMetricsF(font);
|
||||||
if (now < desired) {
|
const auto nowHeight = metrics.height();
|
||||||
return (desired - now * 2 < current - desired)
|
const auto nowCap = metrics.capHeight();
|
||||||
? (size - shift)
|
if (nowHeight < desiredHeight || nowCap < desiredCap) {
|
||||||
: (size - shift + 1);
|
return (size - shift + 1);
|
||||||
}
|
}
|
||||||
current = now;
|
currentHeight = nowHeight;
|
||||||
|
currentCap = nowCap;
|
||||||
}
|
}
|
||||||
return size - kMaxSizeShift;
|
return size - kMaxSizeShift;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
[[nodiscard]] QFont ResolveFont(
|
[[nodiscard]] QFont ResolveFont(
|
||||||
const QString &familyOverride,
|
const QString &family,
|
||||||
uint32 flags,
|
uint32 flags,
|
||||||
int size) {
|
int size,
|
||||||
|
bool skipSizeAdjustment) {
|
||||||
auto result = QFont();
|
auto result = QFont();
|
||||||
if (!familyOverride.isEmpty()) {
|
const auto monospace = (flags & FontMonospace) != 0;
|
||||||
result.setFamily(familyOverride);
|
const auto system = !monospace && (family == SystemFontTag());
|
||||||
} else if (flags & FontMonospace) {
|
const auto overriden = !monospace && !system && !family.isEmpty();
|
||||||
|
if (monospace) {
|
||||||
result.setFamily(MonospaceFont());
|
result.setFamily(MonospaceFont());
|
||||||
} else if (const auto name = std::get_if<QString>(&Custom)) {
|
} else if (system) {
|
||||||
result.setFamily(*name);
|
} else if (overriden) {
|
||||||
} else if (!v::is<SystemFont>(Custom)) {
|
result.setFamily(family);
|
||||||
|
} else {
|
||||||
result.setFamily(GetFontOverride(flags));
|
result.setFamily(GetFontOverride(flags));
|
||||||
#if QT_VERSION >= QT_VERSION_CHECK(6, 7, 0)
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 7, 0)
|
||||||
result.setFeature("ss03", true);
|
result.setFeature("ss03", true);
|
||||||
|
|
@ -233,9 +252,9 @@ bool LoadCustomFont(const QString &filePath, const QString &familyName, int flag
|
||||||
result.setItalic(flags & FontItalic);
|
result.setItalic(flags & FontItalic);
|
||||||
result.setUnderline(flags & FontUnderline);
|
result.setUnderline(flags & FontUnderline);
|
||||||
result.setStrikeOut(flags & FontStrikeOut);
|
result.setStrikeOut(flags & FontStrikeOut);
|
||||||
result.setPixelSize(familyOverride.isEmpty()
|
result.setPixelSize((monospace || !overriden || skipSizeAdjustment)
|
||||||
? ComputePixelSize(result, flags, size)
|
? size
|
||||||
: size);
|
: ComputePixelSize(result, flags, size));
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -320,7 +339,11 @@ int registerFontFamily(const QString &family) {
|
||||||
}
|
}
|
||||||
|
|
||||||
FontData::FontData(int size, uint32 flags, int family, Font *other)
|
FontData::FontData(int size, uint32 flags, int family, Font *other)
|
||||||
: f(ResolveFont(family ? fontFamilies[family] : QString(), flags, size))
|
: f(ResolveFont(
|
||||||
|
family ? fontFamilies[family] : Custom,
|
||||||
|
flags,
|
||||||
|
size,
|
||||||
|
family != 0))
|
||||||
, _m(f)
|
, _m(f)
|
||||||
, _size(size)
|
, _size(size)
|
||||||
, _flags(flags)
|
, _flags(flags)
|
||||||
|
|
@ -414,4 +437,9 @@ void Font::init(int size, uint32 flags, int family, Font *modified) {
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace internal
|
} // namespace internal
|
||||||
|
|
||||||
|
QFont ResolveFont(const QString &custom, uint32 flags, int size) {
|
||||||
|
return internal::ResolveFont(custom, flags, size, false);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace style
|
} // namespace style
|
||||||
|
|
|
||||||
|
|
@ -11,15 +11,17 @@
|
||||||
#include <QtGui/QFont>
|
#include <QtGui/QFont>
|
||||||
#include <QtGui/QFontMetrics>
|
#include <QtGui/QFontMetrics>
|
||||||
|
|
||||||
#include <variant>
|
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
|
|
||||||
namespace style {
|
namespace style {
|
||||||
|
|
||||||
struct SystemFont{
|
[[nodiscard]] const QString &SystemFontTag();
|
||||||
};
|
void SetCustomFont(const QString &font);
|
||||||
using CustomFont = std::variant<std::monostate, SystemFont, QString>;
|
|
||||||
void SetCustomFont(const CustomFont &font);
|
[[nodiscard]] QFont ResolveFont(
|
||||||
|
const QString &custom,
|
||||||
|
uint32 flags,
|
||||||
|
int size);
|
||||||
|
|
||||||
namespace internal {
|
namespace internal {
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Reference in a new issue