I wonder if Roblox will ever make their own way to upload custom fonts, great work btw
Its me again. I tried using this again, and it worked perfectly. Don’t know what i did wrong the first time though. This works great! but do you have any idea on how I could keep the text centered? im finding this difficult.
Glad you got it fixed.
With your UIListLayout, set the HorizontalAlignment and VerticalAlignment to centre. This should anchor the text to the middle of the frame it is contained in.
okay, thanks!! this works really well.
hello again! i need assistance with changing the font size as i had the resolution of the original image much higher than i want it to be. I want it to be in pixels but i can’t seem to figure out how to change the size. Can you help please? thanks
Hi, for some reason when I use my font, the size of everything becomes zero.
Here’s the source code:
local s = [[info face="Aqua" size=72 bold=0 italic=0 charset="" unicode=1 stretchH=100 smooth=1 aa=1 padding=1,1,1,1 spacing=1,1
common lineHeight=72 base=48 scaleW=171 scaleH=172 pages=1 packed=0
page id=0 file="Unnamed.png"
chars count=28
char id=32 x=0 y=0 width=0 height=0 xoffset=0 yoffset=0 xadvance=19 page=0 chnl=15
char id=39 x=154 y=32 width=9 height=9 xoffset=1 yoffset=20 xadvance=11 page=0 chnl=15
char id=65 x=0 y=105 width=34 height=36 xoffset=0 yoffset=9 xadvance=35 page=0 chnl=15
char id=66 x=74 y=61 width=26 height=36 xoffset=0 yoffset=8 xadvance=27 page=0 chnl=15
char id=67 x=74 y=98 width=26 height=35 xoffset=0 yoffset=7 xadvance=27 page=0 chnl=15
char id=68 x=42 y=102 width=28 height=36 xoffset=0 yoffset=7 xadvance=29 page=0 chnl=15
char id=69 x=130 y=101 width=22 height=35 xoffset=0 yoffset=8 xadvance=23 page=0 chnl=15
char id=70 x=42 y=139 width=19 height=33 xoffset=0 yoffset=9 xadvance=20 page=0 chnl=15
char id=71 x=42 y=0 width=31 height=35 xoffset=0 yoffset=9 xadvance=32 page=0 chnl=15
char id=72 x=42 y=36 width=31 height=34 xoffset=0 yoffset=9 xadvance=32 page=0 chnl=15
char id=73 x=130 y=137 width=19 height=30 xoffset=0 yoffset=11 xadvance=20 page=0 chnl=15
char id=74 x=42 y=71 width=30 height=30 xoffset=0 yoffset=10 xadvance=30 page=0 chnl=15
char id=75 x=103 y=133 width=24 height=34 xoffset=0 yoffset=9 xadvance=25 page=0 chnl=15
char id=76 x=130 y=35 width=23 height=33 xoffset=0 yoffset=10 xadvance=23 page=0 chnl=15
char id=77 x=0 y=142 width=34 height=30 xoffset=0 yoffset=10 xadvance=35 page=0 chnl=15
char id=78 x=74 y=30 width=27 height=30 xoffset=0 yoffset=10 xadvance=28 page=0 chnl=15
char id=79 x=0 y=70 width=35 height=34 xoffset=0 yoffset=10 xadvance=36 page=0 chnl=15
char id=80 x=103 y=95 width=24 height=37 xoffset=0 yoffset=6 xadvance=25 page=0 chnl=15
char id=81 x=0 y=32 width=39 height=37 xoffset=0 yoffset=5 xadvance=40 page=0 chnl=15
char id=82 x=130 y=0 width=23 height=34 xoffset=0 yoffset=6 xadvance=24 page=0 chnl=15
char id=83 x=154 y=0 width=17 height=31 xoffset=0 yoffset=8 xadvance=18 page=0 chnl=15
char id=84 x=74 y=134 width=26 height=32 xoffset=0 yoffset=8 xadvance=27 page=0 chnl=15
char id=85 x=74 y=0 width=28 height=29 xoffset=0 yoffset=13 xadvance=29 page=0 chnl=15
char id=86 x=130 y=69 width=23 height=31 xoffset=0 yoffset=13 xadvance=24 page=0 chnl=15
char id=87 x=0 y=0 width=41 height=31 xoffset=0 yoffset=13 xadvance=43 page=0 chnl=15
char id=88 x=103 y=0 width=26 height=32 xoffset=1 yoffset=12 xadvance=28 page=0 chnl=15
char id=89 x=103 y=33 width=26 height=31 xoffset=0 yoffset=11 xadvance=26 page=0 chnl=15
char id=90 x=103 y=65 width=25 height=29 xoffset=0 yoffset=13 xadvance=25 page=0 chnl=15
]]
local fontMap = 'rbxassetid://15540813455'
-----
local info = {}
info.fontInfo = {}
info.characterTable = {}
info.kernings = {}
local init, _ = s:find('kernings')
if init then
local kernings = s:sub(init, s:len()):split('\n')
local kerningsTable = info.kernings
for i,v in ipairs(kernings) do
local first, second, amount = v:match('kerning first=([%-?%.?%d?]+) second=([%-?%.?%d?]+) amount=([%-?%.?%d?]+)')
if first then
kerningsTable[utf8.char(first)] = kerningsTable[utf8.char(first)] or {}
kerningsTable[utf8.char(first)][utf8.char(second)] = amount
end
end
s = s:sub(1, init - 1)
end
local split = s:split('\n')
local characterTable = info.characterTable
for i = 3, 1, -1 do
local infoThisIteration = split[i]:split(' ')
for i,v in ipairs(infoThisIteration) do
local field, value = unpack(v:split('='))
if field and value then
field, value = field:gsub('"', ''), value:gsub('"', '')
info.fontInfo[field] = tonumber(value) or value
end
end
table.remove(split, i)
end
table.remove(split, 1)
for i = #split, 1, -1 do
local v = split[i]
local charId, x, y, width, height, xOffset, yOffset, xAdvance, page, chnl = v:match('char id=([%-?%.?%d?]+) x=([%-?%.?%d?]+) y=([%-?%.?%d?]+) width=([%-?%.?%d?]+) height=([%-?%.?%d?]+) xoffset=([%-?%.?%d?]+) yoffset=([%-?%.?%d?]+) xadvance=([%-?%.?%d?]+) page=([%-?%.?%d?]+) chnl=([%-?%.?%d?]+)')
if charId then
table.remove(split,i)
table.insert(characterTable, {
charId = charId;
x = x;
y = y;
width = width;
height = height;
xOffset = xOffset;
yOffset = yOffset;
xAdvance = xAdvance;
page = page;
chnl = chnl
})
end
end
local size = 72
local stringFolder = Instance.new('Folder')
stringFolder.Name = info.fontInfo.face
stringFolder.Parent = script
for i,v in ipairs(characterTable) do
local mainFrame = Instance.new('Frame')
mainFrame.Size = UDim2.fromOffset(v.xAdvance or v.width, size)
mainFrame.BackgroundTransparency = 1
mainFrame.Name = utf8.char(v.charId)
mainFrame.BackgroundTransparency = 1
local newLabel = Instance.new('ImageLabel')
newLabel.Image = fontMap
newLabel.Size = UDim2.fromOffset(v.width, v.height)
newLabel.Parent = mainFrame
newLabel.Name = utf8.char(v.charId)
newLabel.Position = UDim2.fromOffset(v.xOffset, v.yOffset)
newLabel.ImageRectSize = Vector2.new(v.width, v.height)
newLabel.ImageRectOffset = Vector2.new(v.x, v.y)
newLabel.Parent = mainFrame
newLabel.BackgroundTransparency = 1
newLabel.ScaleType = Enum.ScaleType.Fit
newLabel.BackgroundTransparency = 1
mainFrame.Parent = stringFolder
end
-----
local function feed(str: string)
-- example feed function:
-- first off, we have a function that will return a table of graphemes
local splitString = (function()
local graphemes = {}
for i,v in utf8.graphemes(str) do -- for each of the graphemes,
table.insert(graphemes, str:sub(i,v)) -- we have to use string.sub because utf8 returns a pair where i is the beginning of the grapheme and v is the end of the grapheme. We must use this because we're working with a non-ASCII character (№).
end
return graphemes -- returns the table of graphemes
end)() -- ensure you call this function so it actually assigns the graphemes table to the splitString variable
local newFrame = Instance.new('Frame') -- create a new frame that will store the glyphs
local newUIListLayout = Instance.new('UIListLayout')
newUIListLayout.Parent = newFrame
newUIListLayout.SortOrder = Enum.SortOrder.LayoutOrder -- we don't want them to be aligned vertically
newUIListLayout.FillDirection = Enum.FillDirection.Horizontal -- align them horizontally from left to right
newUIListLayout.Padding = UDim.new(0,0) -- we don't want any padding
for i,v in ipairs(splitString) do -- where i is the index and v is the glyph
local thisLetter = stringFolder:FindFirstChild(v) -- look inside the folder for a frame with the glyph's name
if thisLetter then -- if it found a frame with this name,
local newLetter = thisLetter:Clone() -- clone it and parent it to the frame
newLetter.LayoutOrder = i
newLetter.Parent = newFrame
local next = splitString[i + 1] -- look for the next character to handle kernings
if next then
local kerningThisLetter = info.kernings[v] -- inside the kernings table, look for the kerning for this letter and the next letter
if kerningThisLetter then -- if there is a kerning table for this letter then
local kerningNextLetter = kerningThisLetter[next] -- if inside that kerning table there is a kerning for the next letter
if kerningNextLetter then -- if there's a kerning then
--newFrame.Size -= UDim2.fromOffset(0,kerningNextLetter) -- subtract it from the frame's size, not the actual letter label's size because we don't want to clip the letter
end
end
end
end
end
return newFrame
end
wait(5)
local new = feed('wow custom fonts so cool right')
new.Parent = game.StarterGui.ScreenGui
print(new)
The frame comes out, it just has the size set to zero.
EDIT: I noticed something odd, this is how your example code looks:
And this is how my code looks:

Hey, sorry for the late response, I saw your message earlier but forgot to respond lol
For individual letters, you can resize the font size and line heights. It’ll be on the lefthand side. Use the font size one.
As for resizing the entire image, you can untick Auto-Pack and optionally tick Fixed Size. This will allow you to change the size of your image in the text boxes.
If your image is too small (ie all the characters can’t fit), you’ll get an error in red at the top of your screen.
Hey, this is because you’re passing lowercase letters to your feed function, your fontmap only has uppercase characters.
At the end, swap “wow custom fonts so cool right” with “WOW CUSTOM FONTS SO COOL RIGHT” and you should get your expected result.
Sorry to bother you again, still learning this, but I’m trying to figure out the “size” variable
local size = 1000
local stringFolder = Instance.new('Folder')
stringFolder.Name = info.fontInfo.face
stringFolder.Parent = script
No matter what I change it to, nothing changes. Any idea how to resize the text?
Edit:
Nevermind, I found out there was no built-in resizing method, so I just did this:
local textSize = 2
for i,v in ipairs(characterTable) do
local mainFrame = Instance.new('Frame')
mainFrame.Size = UDim2.fromOffset(v.xAdvance*textSize or v.width*textSize, size)
mainFrame.BackgroundTransparency = 1
mainFrame.Name = utf8.char(v.charId)
mainFrame.BackgroundTransparency = 1
local newLabel = Instance.new('ImageLabel')
newLabel.Image = fontMap
newLabel.Size = UDim2.fromOffset(v.width*textSize, v.height*textSize)
newLabel.Parent = mainFrame
newLabel.Name = utf8.char(v.charId)
newLabel.Position = UDim2.fromOffset(v.xOffset, v.yOffset)
newLabel.ImageRectSize = Vector2.new(v.width, v.height)
newLabel.ImageRectOffset = Vector2.new(v.x, v.y)
newLabel.Parent = mainFrame
newLabel.BackgroundTransparency = 1
newLabel.ScaleType = Enum.ScaleType.Fit
newLabel.BackgroundTransparency = 1
mainFrame.Parent = stringFolder
end
Sorry for the late response. I’ll add scaling when I get time which should allow people to resize the letters. Glad you got it figured out tho
i edited your scripts to fix this already, sizing works now for me.
May I know how? Thank you.
hold on, I will send how at some point
please note that the renderer section of the code is cut because I heavily modified it for something else…
local s = [[]]
local fontMap = ''
-----
local info = {}
info.fontInfo = {}
info.characterTable = {}
info.kernings = {}
local init, _ = s:find('kernings')
if init then
local kernings = s:sub(init, s:len()):split('\n')
local kerningsTable = info.kernings
for i,v in ipairs(kernings) do
local first, second, amount = v:match('kerning first=([%-?%.?%d?]+) second=([%-?%.?%d?]+) amount=([%-?%.?%d?]+)')
if first then
kerningsTable[utf8.char(first)] = kerningsTable[utf8.char(first)] or {}
kerningsTable[utf8.char(first)][utf8.char(second)] = amount
end
end
s = s:sub(1, init - 1)
end
local split = s:split('\n')
local characterTable = info.characterTable
for i = 3, 1, -1 do
local infoThisIteration = split[i]:split(' ')
for i,v in ipairs(infoThisIteration) do
local field, value = unpack(v:split('='))
if field and value then
field, value = field:gsub('"', ''), value:gsub('"', '')
info.fontInfo[field] = tonumber(value) or value
end
end
table.remove(split, i)
end
table.remove(split, 1)
for i = #split, 1, -1 do
local v = split[i]
local charId, x, y, width, height, xOffset, yOffset, xAdvance, page, chnl = v:match('char id=([%-?%.?%d?]+) x=([%-?%.?%d?]+) y=([%-?%.?%d?]+) width=([%-?%.?%d?]+) height=([%-?%.?%d?]+) xoffset=([%-?%.?%d?]+) yoffset=([%-?%.?%d?]+) xadvance=([%-?%.?%d?]+) page=([%-?%.?%d?]+) chnl=([%-?%.?%d?]+)')
if charId then
table.remove(split,i)
table.insert(characterTable, {
charId = charId;
x = x;
y = y;
width = width;
height = height;
xOffset = xOffset;
yOffset = yOffset;
xAdvance = xAdvance;
page = page;
chnl = chnl
})
end
end
local size = 0.16 ------------------SIZE IS RIGHT HERE!!!--------------------
local stringFolder = Instance.new('Folder')
stringFolder.Name = info.fontInfo.face
stringFolder.Parent = script
for i,v in ipairs(characterTable) do
local mainFrame = Instance.new('Frame')
mainFrame.Size = UDim2.fromOffset(v.xAdvance*size or v.width*size, size)
mainFrame.BackgroundTransparency = 1
mainFrame.Name = utf8.char(v.charId)
mainFrame.BackgroundTransparency = 1
local newLabel = Instance.new('ImageLabel')
newLabel.Image = fontMap
newLabel.Size = UDim2.fromOffset(v.width*size, v.height*size)
newLabel.Parent = mainFrame
newLabel.Name = utf8.char(v.charId)
newLabel.Position = UDim2.fromOffset(v.xOffset*size, v.yOffset*size)
newLabel.ImageRectSize = Vector2.new(v.width, v.height)
newLabel.ImageRectOffset = Vector2.new(v.x, v.y)
newLabel.Parent = mainFrame
newLabel.BackgroundTransparency = 1
newLabel.ScaleType = Enum.ScaleType.Fit
newLabel.BackgroundTransparency = 1
mainFrame.Parent = stringFolder
end
you can find the size
variable at a little below the middle of the code
This is very nice but, doesnt come useful in a mobile situation., it would be nice if you could at size support between all screens, (mobile), please.
can’t say i will use it, but it is super cool !! thanks for the resource
Anyone know how to drop a line using this resource? It’s really easy to use but the only issue i’m facing is the fact that I can’t go down a line. I’ve tried using a UiGridLayout but realized that the size for each of the letters aren’t the same so this method won’t work.
Edit:
I’ve tried using \n in my strings but it doesn’t seem to work. I’ve also tried using spaces to go to the next line but it either creates excessive or too little amount of space.
This whole script is helpful in my opinion. You know, I am sort of wondering,
would I be able to use it for game dialogues and all? I am certainly trying to work on it.
Got my font working and all that, I hope it will pull of well eventually.
If you’re looking for a better option, consider checking out my new custom font rendering module.
It’s performant, robust, and depsite custom font support, it is way more!
Hey there!
I know it was incredibly long since this message of yours, but I’d like to let you know about my custom font rendering module.
It’s performant, robust, and depsite custom font support, it is way more!