关于前端:使用-SVG-生成带标识的-favicon

39次阅读

共计 4449 个字符,预计需要花费 12 分钟才能阅读完成。

欢送关注我的公众号:前端侦探

之前做了一个 Chrome 插件,能够依据地址的不同生成不同的图标,这样能够很不便的辨别不同的开发环境,成果如下

次要实现过程其实不简单,首先获取网站 favicon,而后给 favicon 增加标识,从新绘制生成就行了

其中,这里的图标就是通过 SVG 生成的,上面看看具体实现吧。

一、favicon 的获取形式

想晓得获取形式,能够先理解设置形式。

个别有两种形式能够设置网站的 favicon

第一种,通过 link 标签设置(须要 rel="icon" 属性)

<link rel="icon" href="xxx.png">

第二种,间接在 网站根目录 放一张favicon.ico(必须是这个名称,浏览器默认的),html 中什么也不必设置

- 网站目录
    index.html
    favicon.ico

如果以上都没有设置,那么大概率会看到以下谬误

理解这些,获取就简略了,先通过 link 获取,只有 rel 蕴含 icon 就行了

const link = document.querySelector('link[rel~="icon"]');

如果找不到,能够申请/favicon.ico,这里间接增加一个link

function getLink(){const link = document.querySelector('link[rel~="icon"]');
    if (link) {return link} else {const link = document.createElement('link');
        link.rel = "icon";
        link.href = "/favicon.ico"
        document.head.append(link);
        return link
    }
}

这样获取的 link 就保障存在了,而后就是绘制

二、利用 canvas 进行绘制

因为须要合成图像,所以须要先绘制原有图像。提到图像绘制,能够想到 canvas 绘制,只须要一点点 canvas 基础知识就足够了。具体实现如下

const canvas = document.createElement('canvas'),
ctx = canvas.getContext('2d'),
img = new Image();
img.crossOrigin = 'anonymous';
img.onload = () => {
    canvas.height = img.height;
    canvas.width = img.width;
    ctx.drawImage(img, 0, 0, canvas.width, canvas.height);
    let dataURL = canvas.toDataURL("image/png");
    resolve(dataURL);
    canvas = null;
};
img.src = url;

因为存在 /favicon.ico 没有设置的状况,所以须要在 img 加载失败的时候给一张默认图

img.onerror = () => {resolve("默认图 base64");
}

这样就能获取到 favicon 的图像信息了

三、利用 SVG 进行图片合成

在下面的根底上,其实能够持续通过 canvas 进行绘制,再绘制一个标签也不是难事。不过这里能够采纳 SVG 的形式来进行绘制,有以下一些长处

  1. 老本更低,比 canvas 更易了解
  2. 灵活性高,能够通过 CSS 进行一些款式管制

首先,咱们能够在 HTML 中自在的、像失常网页开发一样,绘制这样一个布局,置信没有什么难度

<body>
  <strong>local</strong>
  <img src='xxx.png'>
</body>

因为宽度无限,所以须要强制文本换行,超出暗藏,要害款式如下

strong{
  position: absolute;
  bottom: 0;
  left: 50%;
  transform: translateX(-50%);
  text-transform: uppercase;
  background-color: orange;
  color: #fff;
  padding: 0px 2px;
  border-radius: 2px;
  font-size: 12px;
  box-sizing: border-box;
  max-width: 100%;
  width: max-content;
  height: 16px;
  line-height: 16px;
  word-break: break-all;
  overflow: hidden;
}

接着,将这一段 html 放入 foreignObject标签中,对于 foreignObject 的作用,有趣味的能够参考张鑫旭老师的这篇文章 SVG 简介与截图等利用,在这里,你能够简略了解为是能够蕴含 HTML 的标签,而 SVG 自身也是一种图片,这样就达到了合成图像的目标

具体实现如下

const link = getLink();
const icon = await img2Base64(link.href);
const favicon = `data:image/svg+xml;charset=utf-8,<svg xmlns="http://www.w3.org/2000/svg" width="32" height="32"><foreignObject x="0" y="0" width="100%" height="100%"><body xmlns="http://www.w3.org/1999/xhtml">
  <style>
    html,body{
      height: 100%;
      margin: 0;
      text-align: center;
    }
    img{
      display: block;
      width: 100%;
      height: 100%;
      object-fit: contain;
    }
    strong{
      position: absolute;
      bottom: 0;
      left: 50%;
      transform: translateX(-50%);
      text-transform: uppercase;
      background-color: ${color};
      color: #fff;
      padding: 0px 2px;
      border-radius: 2px;
      font-size: 12px;
      box-sizing: border-box;
      max-width: 100%;
      width: max-content;
      height: 16px;
      line-height: 16px;
      word-break: break-all;
      overflow: hidden;
    }
  </style>
  <strong>local</strong>
  <img src='${icon}'></img>
  </body></foreignObject></svg>`.replace(/\n/g, '').replace(/\t/g,'').replace(/#/g, '%23')

这里须要留神几点

  1. img 标签在 svg 中须要写成 <img></img> 闭合状态,不然会被认为构造谬误
  2. img 只能应用内联图片,比方 base64,这也是为何绘制原始 favicon 的起因
  3. 如果应用内联 svg,须要对 svg 进行本义
  4. 字符串中的单双引号问题也须要留神一下

而后,将生成的 SVG 间接设置为 favicon

link.href= favicon;

这样就能失常的渲染了~

残缺实现能够参考我的项目:https://github.com/XboxYan/auto-dev-favicon-chrome-plugin

四、一些局限性

首先是兼容性。

目前只有 Chrome 和 Firefox 是反对的,为了兼容其余浏览器,能够用一个 .ico来兜底

<link rel="icon" href="/favicon.ico" sizes="any">
<link rel="icon" href="/favicon.svg" type="image/svg+xml">

另外,在 Chrome 上还有一个限度(不晓得是不是 Chrome 更新后的限度),当 favicon 应用 svg 格局图片时,此时的 svg 会处于一种 secure-static-mode,在这种模式下,所有动画都不会执行,会处于第一帧,比方上面这个例子

<svg xmlns="http://www.w3.org/2000/svg" width="100%" height="100%">
  <foreignObject width="100%" height="100%">
      <body xmlns="http://www.w3.org/1999/xhtml">
        <style>
        html,body{
            margin: 0;
            height: 100%
        }
        div{
            height: 100%;
            background: pink;
            animation: hue 3s infinite;
        }
        @keyframes hue {
            to {filter: hue-rotate(360deg)
            }
        }
        </style>
        <div></div>
      </body>
    </foreignObject>
</svg>

很简略的一个背景色彩动画。在 Firefox 上是用作 favicon 是有动画的

然而,Chrome 上却不行,只有禁止的第一帧

所以之前想实现标识文本滚动的成果能够就此打住了😭

比拟相似的还有媒体查问,之前在网上看到有这样的实现,间接在 SVG 中实现光明模式

<svg width="128" height="128" viewBox="0 0 128 128" fill="none" xmlns="http://www.w3.org/2000/svg">
    <style>
        path {fill: #fff;}
        rect {fill: #B09AFE;}
        @media (prefers-color-scheme: dark) {
            path {fill: #B09AFE;}
            rect {fill: #fff;}
        }
    </style>
    <rect width="128" height="128" rx="64" fill="#B09AFE"/>
    <path d="M32.375 37.5714H22C22 58.004 38.2596 74.5714 58.3125 74.5714V98.3571C58.3125 99.8107 59.4797 101 60.9062 101H66.0937C67.5203 101 68.6875 99.8107 68.6875 98.3571V74.5714C68.6875 54.1388 52.4279 37.5714 32.375 37.5714ZM94.625 27C80.9754 27 69.109 34.6808 62.9002 46.0286C67.3906 51.017 70.7139 57.079 72.4646 63.8018C90.7344 61.8692 105 46.1442 105 27H94.625Z" fill="white"/>
</svg>

然而也是同样的问题,只有 Firefox 下可行,Chrome 是静止不动的

总的来说,SVG 在绘制方面提供了有限可能,不仅仅是本文中的案例,感觉 canvas 过于简单的都能够思考这一计划

最初,如果感觉还不错,对你有帮忙的话,欢送点赞、珍藏、转发❤❤❤

欢送关注我的公众号:前端侦探

正文完
 0