Improve codegen of Font roblox datatype enums

This commit is contained in:
Filip Tibell 2023-03-17 16:54:13 +01:00
parent d79895dc88
commit 9722f5b862
No known key found for this signature in database
2 changed files with 210 additions and 171 deletions

View file

@ -4,65 +4,90 @@ local contents = ""
contents ..= "\ntype FontData = (&'static str, FontWeight, FontStyle);\n"
local ENUM_IMPLEMENTATION = [[
#[derive(Debug, Clone, Copy, PartialEq)]
pub(crate) enum <<ENUM_NAME>> {
<<ENUM_NAMES>>
}
impl <<ENUM_NAME>> {
pub fn as_<<NUMBER_TYPE>>(&self) -> <<NUMBER_TYPE>> {
match self {
<<ENUM_TO_NUMBERS>>
}
}
pub fn from_<<NUMBER_TYPE>>(n: <<NUMBER_TYPE>>) -> Option<Self> {
match n {
<<NUMBERS_TO_ENUM>>
_ => None,
}
}
}
impl std::str::FromStr for <<ENUM_NAME>> {
type Err = &'static str;
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
<<STRINGS_TO_ENUM>>
_ => Err("Unknown <<ENUM_NAME>>"),
}
}
}
impl std::fmt::Display for <<ENUM_NAME>> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(
f,
"{}",
match self {
<<ENUM_TO_STRINGS>>
}
)
}
}
]]
-- FontWeight enum and implementation
local function makeRustEnum(enum, numType: string)
local name = string.gsub(tostring(enum), "^Enum.", "")
-- TODO: Use FromStr and AsStr traits for name functions
-- TODO: Match the naming convention that rbx-dom has for number/value functions
local enumNames = ""
local enumToNumbers = ""
local numbersToEnum = ""
local stringsToEnum = ""
local enumToStrings = ""
local result = ""
for _, enum in enum:GetEnumItems() do
enumNames ..= `\n{enum.Name},`
enumToNumbers ..= `\nSelf::{enum.Name} => {enum.Value},`
numbersToEnum ..= `\n{enum.Value} => Some(Self::{enum.Name}),`
stringsToEnum ..= `\n"{enum.Name}" => Ok(Self::{enum.Name}),`
enumToStrings ..= `\nSelf::{enum.Name} => "{enum.Name}",`
end
result ..= "\n#[derive(Debug, Clone, PartialEq)]"
result ..= "\npub(crate) enum " .. name .. " {"
for _, enum in enum:GetEnumItems() do
result ..= string.format("\n %s,", enum.Name)
end
result ..= "\n}\n"
local mappings: { [string]: string } = {
["<<ENUM_NAME>>"] = name,
["<<NUMBER_TYPE>>"] = numType,
["<<ENUM_NAMES>>"] = enumNames,
["<<ENUM_TO_NUMBERS>>"] = enumToNumbers,
["<<ENUM_TO_STRINGS>>"] = enumToStrings,
["<<NUMBERS_TO_ENUM>>"] = numbersToEnum,
["<<STRINGS_TO_ENUM>>"] = stringsToEnum,
}
result ..= "\nimpl " .. name .. " {"
result ..= "\n pub fn into_name(self) -> &'static str {"
result ..= "\n match self {"
for _, enum in enum:GetEnumItems() do
result ..= string.format('\n Self::%s => "%s",', enum.Name, enum.Name)
local result = ENUM_IMPLEMENTATION
for key, replacement in mappings do
result = string.gsub(result, "(\t*)" .. key, function(tabbing)
local spacing = string.gsub(tabbing, "\t", " ")
local inner = string.gsub(replacement, "\n", "\n" .. spacing)
inner = string.gsub(inner, "^\n+", "")
return inner
end)
end
result ..= "\n }"
result ..= "\n }"
result ..= "\n"
result ..= "\n pub fn from_name(name: impl AsRef<str>) -> Option<Self> {"
result ..= "\n match name.as_ref() {"
for _, enum in enum:GetEnumItems() do
result ..= string.format('\n "%s" => Some(Self::%s),', enum.Name, enum.Name)
end
result ..= "\n _ => None,"
result ..= "\n }"
result ..= "\n }"
result ..= "\n"
result ..= "\n pub fn into_num(self) -> " .. numType .. " {"
result ..= "\n match self {"
for _, enum in enum:GetEnumItems() do
result ..= string.format("\n Self::%s => %d,", enum.Name, enum.Value)
end
result ..= "\n }"
result ..= "\n }"
result ..= "\n"
result ..= "\n pub fn from_num(num: " .. numType .. ") -> Option<Self> {"
result ..= "\n match num {"
for _, enum in enum:GetEnumItems() do
result ..= string.format("\n %d => Some(Self::%s),", enum.Value, enum.Name)
end
result ..= "\n _ => None,"
result ..= "\n }"
result ..= "\n }"
result ..= "\n}\n"
result ..= "\nimpl fmt::Display for " .. name .. " {"
result ..= "\n fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {"
result ..= '\n write!(f, "{}", self.into_name())'
result ..= "\n }"
result ..= "\n}"
return result
end
@ -74,25 +99,35 @@ contents ..= "\n"
-- Font constant map from enum to font data
-- TODO: Fix formatting issues of output here
local longestNameLen = 0
local longestFamilyLen = 0
local longestWeightLen = 0
for _, enum in Enum.Font:GetEnumItems() do
longestNameLen = math.max(longestNameLen, #enum.Name)
if enum ~= Enum.Font.Unknown then
local font = Font.fromEnum(enum)
longestFamilyLen = math.max(longestFamilyLen, #font.Family)
longestWeightLen = math.max(longestWeightLen, #font.Weight.Name)
end
end
contents ..= "\n#[rustfmt::skip]\nconst FONT_ENUM_MAP: &[(&str, Option<FontData>)] = &[\n"
for _, enum in Enum.Font:GetEnumItems() do
if enum == Enum.Font.Unknown then
contents ..= ' ("Unknown", None),\n'
contents ..= string.format(
' ("Unknown",%s None),\n',
string.rep(" ", longestNameLen - #enum.Name)
)
else
local font = Font.fromEnum(enum)
contents ..= string.format(
' ("%s",%s Some(("%s", FontWeight::%s, FontStyle::%s))),\n',
' ("%s",%s Some(("%s",%s FontWeight::%s,%s FontStyle::%s))),\n',
enum.Name,
string.rep(" ", longestNameLen - #enum.Name),
font.Family,
string.rep(" ", longestFamilyLen - #font.Family),
font.Weight.Name,
string.rep(" ", longestWeightLen - #font.Weight.Name),
font.Style.Name
)
end

View file

@ -24,13 +24,11 @@ impl Font {
FONT_ENUM_MAP
.iter()
.find(|props| props.0 == material_enum_item.name && props.1.is_some())
.map(|props| {
let props = props.1.as_ref().unwrap().clone();
Font {
.map(|props| props.1.as_ref().unwrap())
.map(|props| Font {
family: props.0.to_string(),
weight: props.1,
style: props.2,
}
})
}
@ -64,7 +62,7 @@ impl LuaUserData for Font {
fn add_fields<'lua, F: LuaUserDataFields<'lua, Self>>(fields: &mut F) {
fields.add_field_method_get("Family", |_, this| Ok(this.family.clone()));
// TODO: Getters & setters for weight, style
fields.add_field_method_get("Bold", |_, this| Ok(this.weight.clone().into_num() >= 600));
fields.add_field_method_get("Bold", |_, this| Ok(this.weight.clone().as_u16() >= 600));
fields.add_field_method_set("Bold", |_, this, value: bool| {
if value {
this.weight = FontWeight::Bold;
@ -110,25 +108,25 @@ impl From<Font> for RbxFont {
impl From<RbxFontWeight> for FontWeight {
fn from(v: RbxFontWeight) -> Self {
FontWeight::from_num(v.as_u16()).expect("Missing font weight")
FontWeight::from_u16(v.as_u16()).expect("Missing font weight")
}
}
impl From<FontWeight> for RbxFontWeight {
fn from(v: FontWeight) -> Self {
RbxFontWeight::from_u16(v.into_num()).expect("Missing rbx font weight")
RbxFontWeight::from_u16(v.as_u16()).expect("Missing rbx font weight")
}
}
impl From<RbxFontStyle> for FontStyle {
fn from(v: RbxFontStyle) -> Self {
FontStyle::from_num(v.as_u8()).expect("Missing font weight")
FontStyle::from_u8(v.as_u8()).expect("Missing font weight")
}
}
impl From<FontStyle> for RbxFontStyle {
fn from(v: FontStyle) -> Self {
RbxFontStyle::from_u8(v.into_num()).expect("Missing rbx font weight")
RbxFontStyle::from_u8(v.as_u8()).expect("Missing rbx font weight")
}
}
@ -142,7 +140,7 @@ impl From<FontStyle> for RbxFontStyle {
type FontData = (&'static str, FontWeight, FontStyle);
#[derive(Debug, Clone, PartialEq)]
#[derive(Debug, Clone, Copy, PartialEq)]
pub(crate) enum FontWeight {
Thin,
ExtraLight,
@ -156,36 +154,7 @@ pub(crate) enum FontWeight {
}
impl FontWeight {
pub fn into_name(self) -> &'static str {
match self {
Self::Thin => "Thin",
Self::ExtraLight => "ExtraLight",
Self::Light => "Light",
Self::Regular => "Regular",
Self::Medium => "Medium",
Self::SemiBold => "SemiBold",
Self::Bold => "Bold",
Self::ExtraBold => "ExtraBold",
Self::Heavy => "Heavy",
}
}
pub fn from_name(name: impl AsRef<str>) -> Option<Self> {
match name.as_ref() {
"Thin" => Some(Self::Thin),
"ExtraLight" => Some(Self::ExtraLight),
"Light" => Some(Self::Light),
"Regular" => Some(Self::Regular),
"Medium" => Some(Self::Medium),
"SemiBold" => Some(Self::SemiBold),
"Bold" => Some(Self::Bold),
"ExtraBold" => Some(Self::ExtraBold),
"Heavy" => Some(Self::Heavy),
_ => None,
}
}
pub fn into_num(self) -> u16 {
pub fn as_u16(&self) -> u16 {
match self {
Self::Thin => 100,
Self::ExtraLight => 200,
@ -199,8 +168,8 @@ impl FontWeight {
}
}
pub fn from_num(num: u16) -> Option<Self> {
match num {
pub fn from_u16(n: u16) -> Option<Self> {
match n {
100 => Some(Self::Thin),
200 => Some(Self::ExtraLight),
300 => Some(Self::Light),
@ -215,43 +184,60 @@ impl FontWeight {
}
}
impl fmt::Display for FontWeight {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.clone().into_name())
impl std::str::FromStr for FontWeight {
type Err = &'static str;
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
"Thin" => Ok(Self::Thin),
"ExtraLight" => Ok(Self::ExtraLight),
"Light" => Ok(Self::Light),
"Regular" => Ok(Self::Regular),
"Medium" => Ok(Self::Medium),
"SemiBold" => Ok(Self::SemiBold),
"Bold" => Ok(Self::Bold),
"ExtraBold" => Ok(Self::ExtraBold),
"Heavy" => Ok(Self::Heavy),
_ => Err("Unknown FontWeight"),
}
}
}
#[derive(Debug, Clone, PartialEq)]
impl std::fmt::Display for FontWeight {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(
f,
"{}",
match self {
Self::Thin => "Thin",
Self::ExtraLight => "ExtraLight",
Self::Light => "Light",
Self::Regular => "Regular",
Self::Medium => "Medium",
Self::SemiBold => "SemiBold",
Self::Bold => "Bold",
Self::ExtraBold => "ExtraBold",
Self::Heavy => "Heavy",
}
)
}
}
#[derive(Debug, Clone, Copy, PartialEq)]
pub(crate) enum FontStyle {
Normal,
Italic,
}
impl FontStyle {
pub fn into_name(self) -> &'static str {
match self {
Self::Normal => "Normal",
Self::Italic => "Italic",
}
}
pub fn from_name(name: impl AsRef<str>) -> Option<Self> {
match name.as_ref() {
"Normal" => Some(Self::Normal),
"Italic" => Some(Self::Italic),
_ => None,
}
}
pub fn into_num(self) -> u8 {
pub fn as_u8(&self) -> u8 {
match self {
Self::Normal => 0,
Self::Italic => 1,
}
}
pub fn from_num(num: u8) -> Option<Self> {
match num {
pub fn from_u8(n: u8) -> Option<Self> {
match n {
0 => Some(Self::Normal),
1 => Some(Self::Italic),
_ => None,
@ -259,9 +245,27 @@ impl FontStyle {
}
}
impl fmt::Display for FontStyle {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.clone().into_name())
impl std::str::FromStr for FontStyle {
type Err = &'static str;
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
"Normal" => Ok(Self::Normal),
"Italic" => Ok(Self::Italic),
_ => Err("Unknown FontStyle"),
}
}
}
impl std::fmt::Display for FontStyle {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(
f,
"{}",
match self {
Self::Normal => "Normal",
Self::Italic => "Italic",
}
)
}
}