Мне удалось наконец получить кросс-браузерное решение (Chrome 32, Firefox 27, IE 11, Safari 6), работающее с комбинацией this и супер-простым расширением Safari. Большая часть этого решения так или иначе упоминается в этом и этом другом вопросе .
Вот сценарий:
function launchCustomProtocol(elem, url, callback) {
var iframe, myWindow, success = false;
if (Browser.name === "Internet Explorer") {
myWindow = window.open('', '', 'width=0,height=0');
myWindow.document.write("<iframe src='" + url + "'></iframe>");
setTimeout(function () {
try {
myWindow.location.href;
success = true;
} catch (ex) {
console.log(ex);
}
if (success) {
myWindow.setTimeout('window.close()', 100);
} else {
myWindow.close();
}
callback(success);
}, 100);
} else if (Browser.name === "Firefox") {
try {
iframe = $("<iframe />");
iframe.css({"display": "none"});
iframe.appendTo("body");
iframe[0].contentWindow.location.href = url;
success = true;
} catch (ex) {
success = false;
}
iframe.remove();
callback(success);
} else if (Browser.name === "Chrome") {
elem.css({"outline": 0});
elem.attr("tabindex", "1");
elem.focus();
elem.blur(function () {
success = true;
callback(true); // true
});
location.href = url;
setTimeout(function () {
elem.off('blur');
elem.removeAttr("tabindex");
if (!success) {
callback(false); // false
}
}, 1000);
} else if (Browser.name === "Safari") {
if (myappinstalledflag) {
location.href = url;
success = true;
} else {
success = false;
}
callback(success);
}
}
Расширение Safari было легко реализовать. Он состоял из одной строки сценария внедрения:
myinject.js:
window.postMessage("myappinstalled", window.location.origin);
Затем на веб-странице JavaScript необходимо сначала зарегистрировать событие сообщения и установить флаг, если сообщение получено:
window.addEventListener('message', function (msg) {
if (msg.data === "myappinstalled") {
myappinstalledflag = true;
}
}, false);
Предполагается, что приложение, связанное с пользовательским протоколом, будет управлять установкой расширения Safari.
Во всех случаях, если обратный вызов возвращает false, вы знаете, чтобы сообщить пользователю, что приложение (т. Е. Его собственный протокол) не установлено.