提问人:SqlRyan 提问时间:11/4/2008 最后编辑:Martijn PietersSqlRyan 更新时间:1/16/2023 访问量:300763
动态更改网站图标
Changing website favicon dynamically
问:
我有一个 Web 应用程序,它根据当前登录的用户进行标记。我想将页面的网站图标更改为自有品牌的徽标,但我无法找到任何代码或任何示例来执行此操作。以前有人成功做到这一点吗?
我想象一个文件夹中有十几个图标,并且对要使用的 favicon.ico 文件的引用只是与 HTML 页面一起动态生成的。思潮?
答:
网站图标在 head 标签中声明如下:
<link rel="shortcut icon" type="image/ico" href="favicon.ico">
您应该能够在视图数据中传递所需的图标名称并将其放入 head 标签中。
评论
为什么不呢?
var link = document.querySelector("link[rel~='icon']");
if (!link) {
link = document.createElement('link');
link.rel = 'icon';
document.head.appendChild(link);
}
link.href = 'https://stackoverflow.com/favicon.ico';
评论
如果您有以下 HTML 代码段:
<link id="favicon" rel="shortcut icon" type="image/png" href="favicon.png" />
例如,您可以通过更改此链接上的 HREF 元素来使用 Javascript 更改网站图标(假设您使用的是 JQuery):
$("#favicon").attr("href","favicon2.png");
您还可以创建一个 Canvas 元素并将 HREF 设置为画布的 ToDataURL(),就像 Favicon Defender 一样。
评论
favicon.png
href
#favicon
document.getElementById('favicon').setAttribute('href','favicon2.png')
根据 WikiPedia 的说法,您可以使用该部分中的标签指定要加载的网站图标文件,参数为 .link
head
rel="icon"
例如:
<link rel="icon" type="image/png" href="/path/image.png">
我想,如果您想为该调用编写一些动态内容,您将可以访问 cookie,以便您可以以这种方式检索会话信息并呈现适当的内容。
您可能会对文件格式感到不满(据报道,IE 仅支持它的 .ICO格式,而大多数其他人都支持PNG和GIF图像)以及可能的缓存问题,无论是在浏览器上还是通过代理。这是因为 favicon 的原始用途,特别是用于用网站的迷你徽标标记书签。
评论
以下是我用来为 Opera、Firefox 和 Chrome 添加动态图标支持的一些代码。不过,我无法让IE或Safari工作。基本上,Chrome允许动态网站图标,但据我所知,它只会在页面的位置(或其中的等)发生变化时更新它们:iframe
var IE = navigator.userAgent.indexOf("MSIE")!=-1
var favicon = {
change: function(iconURL) {
if (arguments.length == 2) {
document.title = optionalDocTitle}
this.addLink(iconURL, "icon")
this.addLink(iconURL, "shortcut icon")
// Google Chrome HACK - whenever an IFrame changes location
// (even to about:blank), it updates the favicon for some reason
// It doesn't work on Safari at all though :-(
if (!IE) { // Disable the IE "click" sound
if (!window.__IFrame) {
__IFrame = document.createElement('iframe')
var s = __IFrame.style
s.height = s.width = s.left = s.top = s.border = 0
s.position = 'absolute'
s.visibility = 'hidden'
document.body.appendChild(__IFrame)}
__IFrame.src = 'about:blank'}},
addLink: function(iconURL, relValue) {
var link = document.createElement("link")
link.type = "image/x-icon"
link.rel = relValue
link.href = iconURL
this.removeLinkIfExists(relValue)
this.docHead.appendChild(link)},
removeLinkIfExists: function(relValue) {
var links = this.docHead.getElementsByTagName("link");
for (var i=0; i<links.length; i++) {
var link = links[i]
if (link.type == "image/x-icon" && link.rel == relValue) {
this.docHead.removeChild(link)
return}}}, // Assuming only one match at most.
docHead: document.getElementsByTagName("head")[0]}
要更改网站图标,只需使用上述方法即可。favicon.change("ICON URL")
(感谢 http://softwareas.com/dynamic-favicons 提供我基于此的代码。
评论
以下是一些适用于 Firefox、Opera 和 Chrome 的代码(与此处发布的所有其他答案不同)。下面是一个在 IE11 中也适用的不同代码演示。以下示例在 Safari 或 Internet Explorer 中可能不起作用。
/*!
* Dynamically changing favicons with JavaScript
* Works in all A-grade browsers except Safari and Internet Explorer
* Demo: http://mathiasbynens.be/demo/dynamic-favicons
*/
// HTML5™, baby! http://mathiasbynens.be/notes/document-head
document.head = document.head || document.getElementsByTagName('head')[0];
function changeFavicon(src) {
var link = document.createElement('link'),
oldLink = document.getElementById('dynamic-favicon');
link.id = 'dynamic-favicon';
link.rel = 'shortcut icon';
link.href = src;
if (oldLink) {
document.head.removeChild(oldLink);
}
document.head.appendChild(link);
}
然后,您将按如下方式使用它:
var btn = document.getElementsByTagName('button')[0];
btn.onclick = function() {
changeFavicon('http://www.google.com/favicon.ico');
};
评论
document.head || document.head = document.getElementsByTagName('head')[0];
Uncaught ReferenceError: Invalid left-hand side in assignment
使此功能适用于 IE 的唯一方法是将 Web 服务器设置为处理对 *.ico 的请求,以调用服务器端脚本语言(PHP、.NET 等)。同时设置 *.ico 重定向到单个脚本,并让此脚本提供正确的网站图标文件。我敢肯定,如果您希望能够在同一个浏览器中在不同的网站图标之间来回反弹,缓存仍然会有一些有趣的问题。
我会使用 Greg 的方法并为 favicon.ico 制作一个自定义处理程序 下面是一个(简化的)处理程序,可以工作:
using System;
using System.IO;
using System.Web;
namespace FaviconOverrider
{
public class IcoHandler : IHttpHandler
{
public void ProcessRequest(HttpContext context)
{
context.Response.ContentType = "image/x-icon";
byte[] imageData = imageToByteArray(context.Server.MapPath("/ear.ico"));
context.Response.BinaryWrite(imageData);
}
public bool IsReusable
{
get { return true; }
}
public byte[] imageToByteArray(string imagePath)
{
byte[] imageByteArray;
using (FileStream fs = new FileStream(imagePath, FileMode.Open, FileAccess.Read))
{
imageByteArray = new byte[fs.Length];
fs.Read(imageByteArray, 0, imageByteArray.Length);
}
return imageByteArray;
}
}
}
然后,您可以在 IIS6 中 Web 配置的 httpHandlers 部分中使用该处理程序,或使用 IIS7 中的“处理程序映射”功能。
评论
jQuery版本:
$("link[rel='shortcut icon']").attr("href", "favicon.ico");
甚至更好:
$("link[rel*='icon']").attr("href", "favicon.ico");
Vanilla JS版本:
document.querySelector("link[rel='shortcut icon']").href = "favicon.ico";
document.querySelector("link[rel*='icon']").href = "favicon.ico";
评论
更现代的方法:
const changeFavicon = link => {
let $favicon = document.querySelector('link[rel="icon"]')
// If a <link rel="icon"> element already exists,
// change its href to the given link.
if ($favicon !== null) {
$favicon.href = link
// Otherwise, create a new element and append it to <head>.
} else {
$favicon = document.createElement("link")
$favicon.rel = "icon"
$favicon.href = link
document.head.appendChild($favicon)
}
}
然后,您可以像这样使用它:
changeFavicon("http://www.stackoverflow.com/favicon.ico")
是的,完全有可能
- 在 favicon.ico 之后使用查询字符串(和其他文件链接 - 请参阅下面的答案链接)
- 只需确保服务器响应“someUserId” 正确的映像文件(可以是静态路由规则或动态服务器端代码)。
例如:
<link rel="shortcut icon" href="/favicon.ico?userId=someUserId">
然后,无论您使用哪种服务器端语言/框架,都应该能够根据 userId 轻松找到文件并响应该请求提供它。
但是要正确地做网站图标(这实际上是一个非常复杂的主题),请看这里的答案 https://stackoverflow.com/a/45301651/661584
比自己解决所有细节要容易得多。
享受。
评论
<link>
apple-touch-icon
我在我的项目中使用 favico.js。
它允许将网站图标更改为一系列预定义的形状以及自定义形状。
在内部,它用于渲染和数据 URL 进行图标编码。canvas
base64
该库还具有不错的功能:图标徽章和动画;据称,您甚至可以将网络摄像头视频流式传输到图标:)
评论
对于使用jQuery的用户,有一个单行解决方案:
$("link[rel*='icon']").prop("href",'https://www.stackoverflow.com/favicon.ico');
或者,如果您想要一个表情符号:)
var canvas = document.createElement("canvas");
canvas.height = 64;
canvas.width = 64;
var ctx = canvas.getContext("2d");
ctx.font = "64px serif";
ctx.fillText("☠️", 0, 64);
$("link[rel*='icon']").prop("href", canvas.toDataURL());
道具 https://koddsson.com/posts/emoji-favicon/
评论
我在开发网站时一直使用这个功能......所以我可以一目了然地看到哪个选项卡中运行了本地、开发或生产。
现在 Chrome 支持 SVG 网站图标,它使它变得容易得多。
Tampermonkey 脚本
在 https://gist.github.com/elliz/bb7661d8ed1535c93d03afcd0609360f 上查看一个 tampermonkey 脚本,该脚本指向我在 https://elliz.github.io/svg-favicon/ 上搞砸的演示站点
基本代码
从另一个答案改编了这个......可以改进,但足以满足我的需求。
(function() {
'use strict';
// play with https://codepen.io/elliz/full/ygvgay for getting it right
// viewBox is required but does not need to be 16x16
const svg = `
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16">
<circle cx="8" cy="8" r="7.2" fill="gold" stroke="#000" stroke-width="1" />
<circle cx="8" cy="8" r="3.1" fill="#fff" stroke="#000" stroke-width="1" />
</svg>
`;
var favicon_link_html = document.createElement('link');
favicon_link_html.rel = 'icon';
favicon_link_html.href = svgToDataUri(svg);
favicon_link_html.type = 'image/svg+xml';
try {
let favicons = document.querySelectorAll('link[rel~="icon"]');
favicons.forEach(function(favicon) {
favicon.parentNode.removeChild(favicon);
});
const head = document.getElementsByTagName('head')[0];
head.insertBefore( favicon_link_html, head.firstChild );
}
catch(e) { }
// functions -------------------------------
function escapeRegExp(str) {
return str.replace(/([.*+?^=!:${}()|\[\]\/\\])/g, "\\$1");
}
function replaceAll(str, find, replace) {
return str.replace(new RegExp(escapeRegExp(find), 'g'), replace);
}
function svgToDataUri(svg) {
// these may not all be needed - used to be for uri-encoded svg in old browsers
var encoded = svg.replace(/\s+/g, " ")
encoded = replaceAll(encoded, "%", "%25");
encoded = replaceAll(encoded, "> <", "><"); // normalise spaces elements
encoded = replaceAll(encoded, "; }", ";}"); // normalise spaces css
encoded = replaceAll(encoded, "<", "%3c");
encoded = replaceAll(encoded, ">", "%3e");
encoded = replaceAll(encoded, "\"", "'"); // normalise quotes ... possible issues with quotes in <text>
encoded = replaceAll(encoded, "#", "%23"); // needed for ie and firefox
encoded = replaceAll(encoded, "{", "%7b");
encoded = replaceAll(encoded, "}", "%7d");
encoded = replaceAll(encoded, "|", "%7c");
encoded = replaceAll(encoded, "^", "%5e");
encoded = replaceAll(encoded, "`", "%60");
encoded = replaceAll(encoded, "@", "%40");
var dataUri = 'data:image/svg+xml;charset=UTF-8,' + encoded.trim();
return dataUri;
}
})();
只需将您自己的 SVG(如果您使用工具,可以用 Jake Archibald 的 SVGOMG 清理)放入顶部的 const 中即可。确保它是正方形的(使用 viewBox 属性),你就可以开始了。
评论
这里有一个片段,可以使网站图标成为表情符号或文本。当我在 stackoverflow 时,它可以在控制台中工作。
function changeFavicon(text) {
const canvas = document.createElement('canvas');
canvas.height = 64;
canvas.width = 64;
const ctx = canvas.getContext('2d');
ctx.font = '64px serif';
ctx.fillText(text, 0, 64);
const link = document.createElement('link');
const oldLinks = document.querySelectorAll('link[rel="shortcut icon"]');
oldLinks.forEach(e => e.parentNode.removeChild(e));
link.id = 'dynamic-favicon';
link.rel = 'shortcut icon';
link.href = canvas.toDataURL();
document.head.appendChild(link);
}
changeFavicon('❤️');
评论
在大多数情况下,Favicon 是这样声明的。
<link rel="icon" href"...." />
这样你就可以通过这个来获得对它的引用。
const linkElement = document.querySelector('link[rel=icon]');
你可以用这个改变图片
linkElement.href = 'url/to/any/picture/remote/or/relative';
在 Chrome 上测试 2021 年提出的解决方案时,我发现有时浏览器会缓存网站图标并且不显示更改,即使链接已更改
此代码有效(类似于之前的提议,但添加了一个随机参数以避免缓存)
let oldFavicon = document.getElementById('favicon')
var link = document.createElement('link')
link.id = 'favicon';
link.type = 'image/x-icon'
link.rel = 'icon';
link.href = new_favicon_url +'?=' + Math.random();
if (oldFavicon) {
document.head.removeChild(oldFavicon);
}
document.head.appendChild(link);
从 https://gist.github.com/mathiasbynens/428626#gistcomment-1809869 复制,以防其他人遇到同样的问题
评论