小tips: 使用JS检测用户是否安装某font-family字体

这篇文章发布于 2018年02月24日,星期六,00:08,归类于 JS实例。 阅读 51364 次, 今日 11 次 17 条评论

 

一、使用JS判断用户操作系统是否安装某字体

下午突发灵感,写了一段JavaScript小脚本,可以用来判断用户的操作系统是否安装了某字体,代码如下:

var isSupportFontFamily=function(f){if(typeof f!="string"){return false}var h="Arial";if(f.toLowerCase()==h.toLowerCase()){return true}var e="a";var d=100;var a=100,i=100;var c=document.createElement("canvas");var b=c.getContext("2d");c.width=a;c.height=i;b.textAlign="center";b.fillStyle="black";b.textBaseline="middle";var g=function(j){b.clearRect(0,0,a,i);b.font=d+"px "+j+", "+h;b.fillText(e,a/2,i/2);var k=b.getImageData(0,0,a,i).data;return[].slice.call(k).filter(function(l){return l!=0})};return g(h).join("")!==g(f).join("")};

语法

isSupportFontFamily(fontFamily);

其中fontFamily参数是必须的,为CSS中font-family设置的web可识别的字体名称,例如宋体'simsun',微软雅黑'Microsoft Yahei'等。

例如,我们要判断用户的操作系统是否安装了微软雅黑字体,可以这么处理:

// isSupportMicrosoftYahei为true或者false
var isSupportMicrosoftYahei = isSupportFontFamily('Microsoft Yahei');

如果用户的操作系统安装了微软雅黑,则isSupportMicrosoftYaheitrue,否则为false

由于本检测方法本质上是基于web检测,因此,fontFamily参数必须是web可识别的,因此,很多中文字体需要使用其英文名称如“思源黑体”, “兰亭黑体”等,具体名称是什么可以参见我之前的文章:“CSS font-family常见中文字体对应的英文名称”。

开源字体中英文对照

测试demo

为了演示上面JS脚本的准确性,我特意制作了一个实例页面,您可以狠狠的点击这里:一些中文字体系统是否安装检测demo

例如,在我的windows操作系统上,显示效果如下截图所示:

字体是否支持判断截图

另外,demo页面源代码有未压缩的JavaScript脚本。

实现原理

根据用户设置的字体将某一个字符绘制在canvas上(fillText()),并提取像素信息(getImageData()),然后和默认字体进行比对,如果像素不一致,说明字体生效,如果像素完全一致,说明字体不生效。

兼容性

IE9+,以及其他所有现代浏览器。

二、使用JS判断网页是否支持某字体渲染实例

什么场景下需要判断用户的浏览器是否支持某个字体的渲染呢?

例如一些阅读类的场景,希望用户可以自己选择自己喜欢的字体进行设置。对于photoshop,office,keynote等软件,都是直接系统搜刮字体,形成完整而准确的字体下拉。但是web页面并没有这样的权限,因此我们只能曲线救国。例如默认给定50个字体范围列表,遍历检测用户支持那些字体,支持哪个字体就增加哪个下拉选项,于是我们的字体配置选项就大大丰富了,体验也就更进一步了。

可以应用的场景示意

您可以狠狠地点击这里:选择合适的字体进行阅读demo

例如,在我自己windows台式机上,显示了如下图所示的字体列表,这些字体全部都是可以正常渲染的:

支持的中文字体列表

例如,我选择一个华文彩云,结果如下图:

设置华文彩云字体后的效果

三、结束语

经过自己的简单测试,在移动端上效果也是可以的,Android未测。由于此方法是自己突然想到的,目前尚未在实际项目中进行过实践,故并不能保证百分百准确,之后会进一步及时更新信息,也欢迎反馈各类使用问题。

好,就这些,希望本文的内容能够对您的工作学习有所帮助。

感谢阅读!

(本篇完)

分享到:


发表评论(目前17 条评论)

  1. momo说道:

    根据前辈的思路和代码,写了个自定义字体检查工具,感谢
    http://blog.luckly-mjw.cn/tool-show/local-font-checker/index.html

  2. 我叫悠哉说道:

    如何判断自己加入的字体包呢,比如在安卓加入了苹方字体

  3. 南京韶邵说道:

    喵啊,果然脑洞够大

  4. Mr.M说道:

    思路很巧妙,赞

  5. stanJ说道:

    android测试没效果

  6. Web前端之家说道:

    字体是一门艺术。。。

  7. lonhon说道:


    格式化后断点跑了一下,思路清奇。
    想到之前看到的用canvas获取GPU型号判断手机型号。

  8. 王念一说道:

    蛤。。。我还以为是用 RenderingContext2d 的原型方法 measureText() 判断字符宽= =

    • etherdream说道:

      measureText 是行不通的,有些字体和未知字体宽度一样,只有高度不同。而测量高度并不是所有浏览器都支持。之前研究 JS 指纹时就遇到这问题。(大部分 JS 指纹库都有枚举字体的功能,比如:fingerprintjs2 — https://github.com/Valve/fingerprintjs2)

      楼主这个办法虽然可行,但是效率不是最高的,每检测一个字体要一次图形渲染。目前试过最快的办法,是创建一个元素,内容设置几个字符(选择不同字体下宽度差距较大的字符。大家可以自己去试),然后设置各种 font-family,用 offsetWidth/offsetHeight 量出尺寸。这样可以共用一个元素,性能相对较高。

      当然不管什么浏览器、操作系统,加载没有缓存的字体是比较费时间的,一般枚举几百个字体都得费大几百 ms。有些浏览器比如 safari 甚至还会假死。。。

  9. ielgnaw说道:

    好点子!

  10. 心满说道:

    妙啊

  11. 依韵说道:

    把格式化出来学习了下,好奇妙想法啊!

    把确定支持的字体和待确认字体以相同的大小都画在canvas上,最后如果画出来的不一样 那就是系统认识这个字体!也就是支持!

    真有创造力!

  12. aaa说道:

    这个真牛啊。但是我还要多看几遍原理。

    • mfk说道:

      原理是:用一个字体显示字母a。如果这个字体不存在,系统会默认显示成Arial字体。
      如果用字体X显示出来的字母的字符点阵和字体Arial不一样,说明字体X存在。
      如果用字体X显示出来的字母的字符点阵和字体Arial一样,说明字体X不存在,系统用的默认字体Arial显示的。