summaryrefslogtreecommitdiffstats
path: root/gfx/thebes/gfxFontFamilyList.h
diff options
context:
space:
mode:
Diffstat (limited to 'gfx/thebes/gfxFontFamilyList.h')
-rw-r--r--gfx/thebes/gfxFontFamilyList.h344
1 files changed, 344 insertions, 0 deletions
diff --git a/gfx/thebes/gfxFontFamilyList.h b/gfx/thebes/gfxFontFamilyList.h
new file mode 100644
index 0000000000..5ed650a646
--- /dev/null
+++ b/gfx/thebes/gfxFontFamilyList.h
@@ -0,0 +1,344 @@
+/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 2 -*-
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#ifndef GFX_FONT_FAMILY_LIST_H
+#define GFX_FONT_FAMILY_LIST_H
+
+#include "nsAtom.h"
+#include "nsDebug.h"
+#include "nsISupportsImpl.h"
+#include "nsReadableUtils.h"
+#include "nsString.h"
+#include "nsUnicharUtils.h"
+#include "nsTArray.h"
+#include "mozilla/MemoryReporting.h"
+#include "mozilla/NotNull.h"
+#include "mozilla/ServoStyleConsts.h"
+#include "mozilla/StaticPtr.h"
+
+namespace mozilla {
+
+/**
+ * font family name, an Atom for the name if not a generic and
+ * a font type indicated named family or which generic family
+ */
+
+struct FontFamilyName final {
+ using Syntax = StyleFontFamilyNameSyntax;
+
+ FontFamilyName() = delete;
+
+ // named font family - e.g. Helvetica
+ explicit FontFamilyName(nsAtom* aFamilyName, Syntax aSyntax)
+ : mName(aFamilyName), mSyntax(aSyntax) {}
+
+ explicit FontFamilyName(const nsACString& aFamilyName, Syntax aSyntax)
+ : mName(NS_Atomize(aFamilyName)), mSyntax(aSyntax) {}
+
+ // generic font family - e.g. sans-serif
+ explicit FontFamilyName(StyleGenericFontFamily aGeneric)
+ : mGeneric(aGeneric) {
+ MOZ_ASSERT(mGeneric != StyleGenericFontFamily::None);
+ }
+
+ FontFamilyName(const FontFamilyName&) = default;
+
+ bool IsNamed() const { return !!mName; }
+
+ bool IsGeneric() const { return !IsNamed(); }
+
+ bool IsQuoted() const { return mSyntax == StyleFontFamilyNameSyntax::Quoted; }
+
+ void AppendToString(nsACString& aFamilyList, bool aQuotes = true) const {
+ if (IsNamed()) {
+ if (mSyntax == Syntax::Identifiers) {
+ return aFamilyList.Append(nsAtomCString(mName));
+ }
+ if (aQuotes) {
+ aFamilyList.Append('"');
+ }
+ aFamilyList.Append(nsAtomCString(mName));
+ if (aQuotes) {
+ aFamilyList.Append('"');
+ }
+ return;
+ }
+ switch (mGeneric) {
+ case StyleGenericFontFamily::None:
+ case StyleGenericFontFamily::MozEmoji:
+ MOZ_FALLTHROUGH_ASSERT("Should never appear in a font-family name!");
+ case StyleGenericFontFamily::Serif:
+ return aFamilyList.AppendLiteral("serif");
+ case StyleGenericFontFamily::SansSerif:
+ return aFamilyList.AppendLiteral("sans-serif");
+ case StyleGenericFontFamily::Monospace:
+ return aFamilyList.AppendLiteral("monospace");
+ case StyleGenericFontFamily::Cursive:
+ return aFamilyList.AppendLiteral("cursive");
+ case StyleGenericFontFamily::Fantasy:
+ return aFamilyList.AppendLiteral("fantasy");
+ }
+ MOZ_ASSERT_UNREACHABLE("Unknown generic font-family!");
+ return aFamilyList.AppendLiteral("serif");
+ }
+
+ // helper method that converts generic names to the right enum value
+ static FontFamilyName Convert(const nsACString& aFamilyOrGenericName) {
+ // should only be passed a single font - not entirely correct, a family
+ // *could* have a comma in it but in practice never does so
+ // for debug purposes this is fine
+ NS_ASSERTION(aFamilyOrGenericName.FindChar(',') == -1,
+ "Convert method should only be passed a single family name");
+
+ auto genericType = StyleGenericFontFamily::None;
+ if (aFamilyOrGenericName.LowerCaseEqualsLiteral("serif")) {
+ genericType = StyleGenericFontFamily::Serif;
+ } else if (aFamilyOrGenericName.LowerCaseEqualsLiteral("sans-serif")) {
+ genericType = StyleGenericFontFamily::SansSerif;
+ } else if (aFamilyOrGenericName.LowerCaseEqualsLiteral("monospace") ||
+ aFamilyOrGenericName.LowerCaseEqualsLiteral("-moz-fixed")) {
+ genericType = StyleGenericFontFamily::Monospace;
+ } else if (aFamilyOrGenericName.LowerCaseEqualsLiteral("cursive")) {
+ genericType = StyleGenericFontFamily::Cursive;
+ } else if (aFamilyOrGenericName.LowerCaseEqualsLiteral("fantasy")) {
+ genericType = StyleGenericFontFamily::Fantasy;
+ } else {
+ return FontFamilyName(aFamilyOrGenericName, Syntax::Identifiers);
+ }
+
+ return FontFamilyName(genericType);
+ }
+
+ bool IsNamedFamily(const nsAString& aFamilyName) const {
+ if (!IsNamed()) {
+ return false;
+ }
+ nsDependentAtomString name{mName};
+ return name.Equals(aFamilyName, nsCaseInsensitiveStringComparator);
+ }
+
+ RefPtr<nsAtom> mName; // null if mGeneric != Default
+ StyleFontFamilyNameSyntax mSyntax = StyleFontFamilyNameSyntax::Quoted;
+ StyleGenericFontFamily mGeneric = StyleGenericFontFamily::None;
+};
+
+inline bool operator==(const FontFamilyName& a, const FontFamilyName& b) {
+ return a.mName == b.mName && a.mSyntax == b.mSyntax &&
+ a.mGeneric == b.mGeneric;
+}
+
+/**
+ * A refcounted array of FontFamilyNames. We use this to store the specified
+ * and computed value of the font-family property.
+ *
+ * TODO(heycam): It might better to define this type (and FontFamilyList and
+ * FontFamilyName) in Rust.
+ */
+class SharedFontList {
+ using Syntax = StyleFontFamilyNameSyntax;
+
+ public:
+ NS_INLINE_DECL_THREADSAFE_REFCOUNTING(SharedFontList);
+
+ SharedFontList() = default;
+
+ explicit SharedFontList(StyleGenericFontFamily aGenericType)
+ : mNames{FontFamilyName(aGenericType)} {}
+
+ SharedFontList(nsAtom* aFamilyName, Syntax aSyntax)
+ : mNames{FontFamilyName(aFamilyName, aSyntax)} {}
+
+ SharedFontList(const nsACString& aFamilyName, Syntax aSyntax)
+ : mNames{FontFamilyName(aFamilyName, aSyntax)} {}
+
+ explicit SharedFontList(nsTArray<FontFamilyName>&& aNames)
+ : mNames(std::move(aNames)) {}
+
+ size_t SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const {
+ size_t n = 0;
+ n += aMallocSizeOf(this);
+ n += mNames.ShallowSizeOfExcludingThis(aMallocSizeOf);
+ return n;
+ }
+
+ size_t SizeOfIncludingThisIfUnshared(MallocSizeOf aMallocSizeOf) const {
+ size_t n = 0;
+ if (mRefCnt.get() == 1) {
+ n += SizeOfIncludingThis(aMallocSizeOf);
+ }
+ return n;
+ }
+
+ const nsTArray<FontFamilyName> mNames{};
+
+ static void Initialize();
+ static void Shutdown();
+ static StaticRefPtr<SharedFontList> sEmpty;
+ static StaticRefPtr<SharedFontList>
+ sSingleGenerics[size_t(StyleGenericFontFamily::MozEmoji)];
+
+ private:
+ ~SharedFontList() = default;
+};
+
+/**
+ * font family list, array of font families and a default font type.
+ * font family names are either named strings or generics. the default
+ * font type is used to preserve the variable font fallback behavior
+ */
+class FontFamilyList {
+ using Syntax = StyleFontFamilyNameSyntax;
+
+ public:
+ FontFamilyList() = default;
+
+ explicit FontFamilyList(StyleGenericFontFamily aGenericType)
+ : mFontlist(MakeNotNull<SharedFontList*>(aGenericType)) {}
+
+ FontFamilyList(nsAtom* aFamilyName, Syntax aSyntax)
+ : mFontlist(MakeNotNull<SharedFontList*>(aFamilyName, aSyntax)) {}
+
+ FontFamilyList(const nsACString& aFamilyName, Syntax aSyntax)
+ : mFontlist(MakeNotNull<SharedFontList*>(aFamilyName, aSyntax)) {}
+
+ explicit FontFamilyList(nsTArray<FontFamilyName>&& aNames)
+ : mFontlist(MakeNotNull<SharedFontList*>(std::move(aNames))) {}
+
+ FontFamilyList(const FontFamilyList& aOther) = default;
+
+ explicit FontFamilyList(NotNull<SharedFontList*> aFontList)
+ : mFontlist(aFontList) {}
+
+ void SetFontlist(nsTArray<FontFamilyName>&& aNames) {
+ mFontlist = MakeNotNull<SharedFontList*>(std::move(aNames));
+ }
+
+ void SetFontlist(NotNull<SharedFontList*> aFontlist) {
+ mFontlist = aFontlist;
+ }
+
+ uint32_t Length() const { return mFontlist->mNames.Length(); }
+
+ bool IsEmpty() const { return mFontlist->mNames.IsEmpty(); }
+
+ NotNull<SharedFontList*> GetFontlist() const { return mFontlist; }
+
+ bool Equals(const FontFamilyList& aFontlist) const {
+ return (mFontlist == aFontlist.mFontlist ||
+ mFontlist->mNames == aFontlist.mFontlist->mNames) &&
+ mDefaultFontType == aFontlist.mDefaultFontType;
+ }
+
+ bool HasDefaultGeneric() const {
+ if (mDefaultFontType == StyleGenericFontFamily::None) {
+ return false;
+ }
+ for (const FontFamilyName& name : mFontlist->mNames) {
+ if (name.mGeneric == mDefaultFontType) {
+ return true;
+ }
+ }
+ return false;
+ }
+
+ // Find the first generic (but ignoring cursive and fantasy, as they are
+ // rarely configured in any useful way) in the list.
+ // If found, move it to the start and return true; else return false.
+ bool PrioritizeFirstGeneric() {
+ uint32_t len = mFontlist->mNames.Length();
+ for (uint32_t i = 0; i < len; i++) {
+ const FontFamilyName name = mFontlist->mNames[i];
+ if (name.IsGeneric()) {
+ if (name.mGeneric == StyleGenericFontFamily::Cursive ||
+ name.mGeneric == StyleGenericFontFamily::Fantasy) {
+ continue;
+ }
+ if (i > 0) {
+ nsTArray<FontFamilyName> names;
+ names.AppendElements(mFontlist->mNames);
+ names.RemoveElementAt(i);
+ names.InsertElementAt(0, name);
+ SetFontlist(std::move(names));
+ }
+ return true;
+ }
+ }
+ return false;
+ }
+
+ void PrependGeneric(StyleGenericFontFamily aGeneric) {
+ nsTArray<FontFamilyName> names;
+ names.AppendElements(mFontlist->mNames);
+ names.InsertElementAt(0, FontFamilyName(aGeneric));
+ SetFontlist(std::move(names));
+ }
+
+ void ToString(nsACString& aFamilyList, bool aQuotes = true,
+ bool aIncludeDefault = false) const {
+ aFamilyList =
+ StringJoin(","_ns, mFontlist->mNames,
+ [aQuotes](nsACString& dst, const FontFamilyName& name) {
+ name.AppendToString(dst, aQuotes);
+ });
+ if (aIncludeDefault && mDefaultFontType != StyleGenericFontFamily::None) {
+ if (!aFamilyList.IsEmpty()) {
+ aFamilyList.Append(',');
+ }
+ if (mDefaultFontType == StyleGenericFontFamily::Serif) {
+ aFamilyList.AppendLiteral("serif");
+ } else {
+ aFamilyList.AppendLiteral("sans-serif");
+ }
+ }
+ }
+
+ // searches for a specific non-generic name, case-insensitive comparison
+ bool Contains(const nsAString& aFamilyName) const {
+ for (const FontFamilyName& name : mFontlist->mNames) {
+ if (name.IsNamedFamily(aFamilyName)) {
+ return true;
+ }
+ }
+ return false;
+ }
+
+ StyleGenericFontFamily GetDefaultFontType() const { return mDefaultFontType; }
+ void SetDefaultFontType(StyleGenericFontFamily aType) {
+ NS_ASSERTION(aType == StyleGenericFontFamily::None ||
+ aType == StyleGenericFontFamily::Serif ||
+ aType == StyleGenericFontFamily::SansSerif,
+ "default font type must be either serif or sans-serif");
+ mDefaultFontType = aType;
+ }
+
+ // memory reporting
+ size_t SizeOfExcludingThis(mozilla::MallocSizeOf aMallocSizeOf) const {
+ size_t n = 0;
+ n += mFontlist->SizeOfIncludingThisIfUnshared(aMallocSizeOf);
+ return n;
+ }
+
+ size_t SizeOfIncludingThis(mozilla::MallocSizeOf aMallocSizeOf) const {
+ return aMallocSizeOf(this) + SizeOfExcludingThis(aMallocSizeOf);
+ }
+
+ protected:
+ NotNull<RefPtr<SharedFontList>> mFontlist{
+ WrapNotNull(SharedFontList::sEmpty.get())};
+ StyleGenericFontFamily mDefaultFontType =
+ StyleGenericFontFamily::None; // or serif, or sans-serif
+};
+
+inline bool operator==(const FontFamilyList& a, const FontFamilyList& b) {
+ return a.Equals(b);
+}
+
+inline bool operator!=(const FontFamilyList& a, const FontFamilyList& b) {
+ return !a.Equals(b);
+}
+
+} // namespace mozilla
+
+#endif /* GFX_FONT_FAMILY_LIST_H */