Skip to content
This repository has been archived by the owner on Jun 25, 2020. It is now read-only.

embedded font issues #24

Closed
tedwardmah opened this issue Feb 7, 2015 · 10 comments
Closed

embedded font issues #24

tedwardmah opened this issue Feb 7, 2015 · 10 comments

Comments

@tedwardmah
Copy link

The script is nicely pulling in all of my svg styles when generating the png, except for fonts embedded using @font-face { ... } . It looks like the saveSvgAsPng script should handle embedded fonts within the styles() function on lines 73-74, so I'm wondering if there's something I'm missing here. I'm using a truetype font in my @font-face declaration. Are there any known issues when using embedded fonts?

@exupero
Copy link
Owner

exupero commented Feb 8, 2015

Font face issues have been reported in the past, and the code you pointed out was intended to fix the problem, but after some investigation, there's more to it than simply including @font-face directives in the embedded CSS.

It turns out custom font faces needs to be inlined similar to the way images are being inlined before being loaded into a canvas. I've added a failing test for custom fonts to the tests page to make sure the problem is fixed.

Until saveSvgAsPng automatically inlines custom fonts for you, you can work around the problem by manually inlining them. For instance, the tests page uses the custom font Stalemate:

@font-face {
  font-family: 'Stalemate';
  font-style: normal;
  font-weight: 400;
  src: url(stalemate.ttf) format('truetype');
}

To inline the font yourself, change the value of the src property to

src: url("data:font/ttf;charset=utf-8;base64,<data>");

where <data> is the base64-encoded contents of your fonts file.

Hopefully I'll figure out a fix for for this sometime this week.

@tedwardmah
Copy link
Author

Eric, thanks for this workaround, I feel very fortunate to have come across your script and blog entry. I had been inlining the @font-face declarations but had not been directly encoding in base64, which I believe was the issue. Everything is working perfectly now.

@arunkjn
Copy link
Contributor

arunkjn commented Mar 10, 2015

FYI-
This format worked for me for embedding fonts-
src: url(data:application/x-font-woff;charset=utf-8;base64,) format('woff');

@nsonnad
Copy link
Contributor

nsonnad commented May 7, 2015

Hey @exupero do you have any guidance on how best to implement the font embedding? I've started trying some things for a project I'm working on but not sure my approach is very good.

I've been able to convert the fonts to base64 strings by doing XHR GET requests for them, then encoding the xhr.responseText. My plan is to then write some hairy regex to replace the src: url(...) entries with the encoded string.

Here's an initial attempt that doesn't appear to be working: https://github.com/nsonnad/saveSvgAsPng/blob/inline-fonts/saveSvgAsPng.js

Thanks!

@exupero
Copy link
Owner

exupero commented May 11, 2015

So far I haven't had any better ideas on how to tackle inlining fonts than doing a regex replace, so if you can make it work, go for it.

I haven't looked very closely at your code yet, but when I took a stab at the problem a few months ago, I ran into trouble wrangling the requests to get font data with the requests to inline image data.

@nsonnad
Copy link
Contributor

nsonnad commented May 11, 2015

Thanks. The latest version of my code almost works. It succeeds in getting the fonts, processing them as base 64, and replacing the url() CSS. The in-browser test you added passes.

The outstanding issue appears to be that on PNG download, the fonts don't work the first time, but work thereafter. I've moved to just inlining the fonts with a preprocessor, but think I'm pretty close. Will try to get back to it in the near future but would be great if you could have a look as well.

@exupero
Copy link
Owner

exupero commented May 11, 2015

Excellent! Go ahead and make a pull request. I'll see if I can sort out the download problem.

Thanks for your work!

@nsonnad
Copy link
Contributor

nsonnad commented May 11, 2015

PR is at #29 ^

vancetran added a commit to vancetran/memekit that referenced this issue Aug 26, 2015
work around for fonts not showing up in PNG output
exupero/saveSvgAsPng#24
@exupero exupero mentioned this issue Jan 25, 2016
@exupero exupero mentioned this issue Nov 15, 2016
anvaka added a commit to anvaka/saveSvgAsPng that referenced this issue Dec 3, 2016
The support is done by parsing all font-face urls, fetching them as
binary files, and inlining them as data uri (https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/Data_URIs)

Note: it's very rudimentary and fragile. But it works when applied
correctly, and I thought it could be useful for more people.

exupero#24
exupero pushed a commit that referenced this issue Dec 4, 2016
* Added support for custom fonts

The support is done by parsing all font-face urls, fetching them as
binary files, and inlining them as data uri (https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/Data_URIs)

Note: it's very rudimentary and fragile. But it works when applied
correctly, and I thought it could be useful for more people.

#24

* use https version of jquery

* added notes

* Added back wrongfully removed code
@austinclemens
Copy link

austinclemens commented May 8, 2017

I am encountering the problem described by nsonnad in his 5/11/2015 comment - fonts do not render on the first save out, but do on every consecutive save. After a lot of testing I have no idea what is causing this but it does seem to be related to size of the SVG element - data uris and existing embedded fonts seem to cause problems.

My SVG is pretty complex and has an embedded font in defs and a medium sized image embedded as a URI. I thought maybe it was that my SVG element had some stuff already defined in defs (bc savesvg creates a 2nd defs element in the clone) but in simple examples this only causes the problem intermittently. Here's a minimal working example of the problem: http://www.austinclemens.com/test/test.html

At least in Chrome, the text won't render the first time the button is pressed after refreshing or loading the page. I think this is clearly related to overall size of the SVG - I have a huge data URI in this example for the chrome logo - but it's confusing to me that it then works on the 2nd try. Nothing changes about the clone object or the defs objects created by savesvg.

@exupero
Copy link
Owner

exupero commented May 15, 2018

@austinclemens Thanks for the bug report. I suspect this is related to font loading. Since it's tangential to the main problem, please open a new issue if it's still a problem.

@exupero exupero closed this as completed May 15, 2018
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
None yet
Projects
None yet
Development

No branches or pull requests

5 participants