Intercept HTML5 Web Notifications in a browser environment

后端 未结 1 1192
走了就别回头了
走了就别回头了 2021-01-18 05:39

I would like to intercept HTML5 Web Notifications. I have read the following answer where a user suggests that it is possible to override the window.Notification

相关标签:
1条回答
  • 2021-01-18 05:58

    The problem is that an arrow function can't be used as a constructor (Source).

    The project that uses this code still has an arrow function: https://github.com/jiahaog/nativefier/blob/master/app/src/static/preload.js but it runs in Electron which might explain why it behaves differently.

    If targeting recent browsers, rather use a named function like this:

    (function () {
    
        function notifyCallback(title, opt) {
            console.log("title", title);
        }
    
        const OldNotify = window.Notification;
    
        function newNotify(title, opt) {
            notifyCallback(title, opt);
            return new OldNotify(title, opt);
        }
    
        newNotify.requestPermission = OldNotify.requestPermission.bind(OldNotify);
        Object.defineProperty(newNotify, 'permission', {
            get: function() {
                return OldNotify.permission;
            }
        });
    
        window.Notification = newNotify;
    })();
    
    Notification.requestPermission(function (permission) {
        if (permission === "granted") {
            const notif = new Notification('My title');
        }
    });
    

    The proxy thus created will then be effective when other code/libraries call new Notification() like in my example. I have moved the proxifying logic to the top-level to ensure that the other code/libraries won't keep a reference on the native Notification before the user accepts to receive notifications. You must also put the code at the very first place to guarantee that.

    And if your target browsers support ECMAScript 6, there is a much more elegant way to do it:

    (function () {
    
        function notifyCallback(title, opt) {
            console.log("title", title);
        }
    
        const handler = {
            construct(target, args) {
                notifyCallback(...args);
                return new target(...args);
            }
        };
    
        const ProxifiedNotification= new Proxy(Notification, handler);
    
        window.Notification = ProxifiedNotification;
    })();
    
    Notification.requestPermission(function (permission) {
        if (permission === "granted") {
            const notif = new Notification('My title');
        }
    });
    

    It's much more scalable (no impact when the Notification API changes in future ECMAScript versions since it allows to manipulate the native Notification rather than a handmade one).

    0 讨论(0)
提交回复
热议问题