用JQuery选择Gmail中未读邮件的问题

发布于 2024-12-04 17:09:01 字数 1273 浏览 3 评论 0原文

在 Gmail 中,点击下面显示的复选框会选择所有邮件,我正在制作一个用户脚本(供个人使用,我需要它在 Chrome 中工作),该脚本将仅选择未读邮件(只有下面屏幕中的前 2 条消息未读),而不是该复选框的默认行为。

在此处输入图像描述

我的第一个想法是模拟点击事件,尽管我可以使用代码访问“未读”菜单项...

var unread_menuitem = document.getElementById('canvas_frame').contentWindow.document.getElementById(':s2');
$(unread_menuitem).css({'border':'thin red solid'});

在此处输入图像描述

并使用代码向其分派点击事件...

var clickEvent = document.createEvent('MouseEvents');
clickEvent.initEvent( 'click', true, true );
unread_menuitem.dispatchEvent(clickEvent); // Chrome's console returned 'true'

未读消息不会被选中。


我的第二个想法是通过检查复选框 $('#canvas_frame').contents().find('tr.zE input').prop('checked', true) 来强制选择并应用 Gmail 在手动点击事件上应用的 css 样式,但是虽然我能够在视觉上以及 DOM 方面匹配手动点击事件(据我所知)...

在此处输入图像描述

Gmail 在执行某些操作时显示“未选择任何对话”,在本例中我执行了“标记为已读”。我还想指出,使用我的强力方法手动单击处于此状态的复选框并没有像您期望的那样“取消选中”它们。他们需要额外手动点击一次才能取消选中。

在此处输入图像描述

我的两个想法都失败了,我想知道是否还有其他方法可以解决这个问题,或者是否有是改进我的上述想法以解决问题的方法。

In Gmail, clicking on the checkbox shown below selects all messages and I'm making a userscript (for personal use and I need it to work in Chrome) that'll select the unread messages only (only the first 2 messages in the screenie below are unread) instead of the default behavior of that checkbox.

enter image description here

My first idea is to simulate click events and although I could access the "unread" menuitem fine using the code...

var unread_menuitem = document.getElementById('canvas_frame').contentWindow.document.getElementById(':s2');
$(unread_menuitem).css({'border':'thin red solid'});

enter image description here

and dispatch the click event to it using the code...

var clickEvent = document.createEvent('MouseEvents');
clickEvent.initEvent( 'click', true, true );
unread_menuitem.dispatchEvent(clickEvent); // Chrome's console returned 'true'

the unread messages don't get selected.


My second idea was to brute force the selection by checking the checkbox $('#canvas_frame').contents().find('tr.zE input').prop('checked', true) and apply the css styles that Gmail applies on a manual click event, but while I was able to match the manual click event both visually as well as DOM-wise (afaik)...

enter image description here

Gmail says "No conversations selected" while performing some action, in this case I did a "Mark as Read". I also want to note that manually clicking on the checkboxes that were put in this state using my brute force method did not "uncheck" them as you'd expect. They needed one additional manual click to get unchecked.

enter image description here

Both my ideas have bombed and I want to know if there are others ways to tackle this, or if there are ways to improve upon my ideas above that can solve the problem.

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(2

凡间太子 2024-12-11 17:09:01

这里有一个脚本,看起来它可以完成您想要做的事情。这是您需要创建的整个脚本还是只是功能的一部分?

根据讨论,他们确实为 Firefox 创建了它,它可以在其中运行,有些人评论说它在 Chrome 中不起作用,所以你可能正在寻找一个需要针对不同浏览器的解决方案(你的问题没有指定是否它必须在 Chrome 中工作,只是你正在使用它)。

这就是他们用来选择未读消息的方法,看起来他们正在模拟每个项目上的 mousedown 和 mouseup 事件:

var handler = function(type,e) {
    e.preventDefault();
    e.stopPropagation();

    var e2 = document.createEvent('MouseEvents');
    e2.initEvent('mousedown',true,false); 
    var el = anchor.wrappedJSObject;
    el.dispatchEvent(e2);

    setTimeout(function() {
        var el = document.querySelectorAll('div[selector='+type+'] > div')[0].wrappedJSObject;
        var e2 = document.createEvent('MouseEvents');
        e2.initEvent('mouseup',true,false); 
        el.dispatchEvent(e2);
    },100);
}

他们通过进一步设置一个单击事件来调用此事件,该事件调用 handler('unread', e);

There's a script here that looks it does what you're trying to do. Is that the whole script you needed to create or was that just part of the functionality?

According to the discussions, they did create it for Firefox which it works in, some people have commented it doesn't work in Chrome, so you might be looking at a solution that needs to target different browsers (your question doesn't specify if it must work in Chrome, just that you were using it).

This is what they are using to select the unread messages, it looks like they are simulating the mousedown and mouseup events on each item:

var handler = function(type,e) {
    e.preventDefault();
    e.stopPropagation();

    var e2 = document.createEvent('MouseEvents');
    e2.initEvent('mousedown',true,false); 
    var el = anchor.wrappedJSObject;
    el.dispatchEvent(e2);

    setTimeout(function() {
        var el = document.querySelectorAll('div[selector='+type+'] > div')[0].wrappedJSObject;
        var e2 = document.createEvent('MouseEvents');
        e2.initEvent('mouseup',true,false); 
        el.dispatchEvent(e2);
    },100);
}

They are calling this by setting up a click event further down which calls handler('unread',e);

残月升风 2024-12-11 17:09:01

解决了这个问题,简单易行,现在我想起来,这是用户 PirateKitten 的答案中列出的解决方案 - 而不是像我的第二个想法一样“选中”未读消息旁边的复选框,模拟点击 这些复选框。工作起来就像一个魅力,这里是代码,您可以在使用 Gmail 时在 Chrome 控制台中运行该代码(不需要 jQuery 顺便说一句):

var unreadMessages = document.getElementById('canvas_frame').contentWindow.document.querySelectorAll('tr.zE input');
var numMessages = unreadMessages.length;
while ( numMessages-- ) {
    var clickEvent = document.createEvent('MouseEvents');
    clickEvent.initEvent( 'click', true, true );
    unreadMessages[numMessages].dispatchEvent(clickEvent); 
}

这是我的完整脚本,您可以在 Chrome 控制台中运行(或打开将其转换为扩展/用户脚本)以将复选框的默认行为从选择所有消息更改为仅选择未读消息:

var hasUILoaded = setInterval( function() {
    if( document.getElementById('canvas_frame').contentDocument.getElementsByClassName('J-Zh-I J-J5-Ji J-Pm-I L3') ) {
        clearInterval( hasUILoaded );
        setTimeout( function() {
            var content_frame = document.getElementById('canvas_frame').contentDocument;
            var chkbox = content_frame.getElementsByClassName( 'J-Zh-I J-J5-Ji J-Pm-I L3' );
            var chkbox = chkbox[0].childNodes[0];
            var unreadMessages = content_frame.getElementsByClassName('zE'); // DOM structure: <tr class=zE> <td> <img><input> </td> </tr> so we add ".childNodes[0].childNodes[1]" whenever we want to access the check boxes of each message.
            var allMessages = content_frame.getElementsByClassName('zA');            

            chkbox.onclick = function() {
                if( chkbox.checked ) {
                    var numUnread = unreadMessages.length;;
                    var numAll  = allMessages.length;
                    setTimeout(function () { 
                        if( allMessages[0].childNodes[0].childNodes[1].checked ) { 
                            while ( numAll-- ) {
                                var clickEvent = document.createEvent('MouseEvents');
                                clickEvent.initEvent( 'click', true, true ); 
                            allMessages[numAll].childNodes[0].childNodes[1].dispatchEvent(clickEvent);       
                            }
                        }
                    }, 10);

                    setTimeout(function (){
                        if(!unreadMessages[0].childNodes[0].childNodes[1].checked) {
                            while ( numUnread-- ) {
                                var clickEvent = document.createEvent('MouseEvents');
                                clickEvent.initEvent( 'click', true, true ); 
                unreadMessages[numUnread].childNodes[0].childNodes[1].dispatchEvent(clickEvent);  
                            }
                        }
                    }, 30);
                }
            }
        }, 100);
    }
}, 300);

Solved it, easy as pie and now that I think about it, it's the solution listed in user PirateKitten's answer - instead of "checking off" the checkboxes besides the unread messages like my second idea, simulate clicks on those checkboxes instead. Works like a charm and here's the code, which you can run in Chrome's console while using Gmail (doesn't need jQuery btw):

var unreadMessages = document.getElementById('canvas_frame').contentWindow.document.querySelectorAll('tr.zE input');
var numMessages = unreadMessages.length;
while ( numMessages-- ) {
    var clickEvent = document.createEvent('MouseEvents');
    clickEvent.initEvent( 'click', true, true );
    unreadMessages[numMessages].dispatchEvent(clickEvent); 
}

Here's my full script that you can run inside your Chrome console (or turn it into an extension/userscript) to change the default behavior of the checkbox from selecting ALL messages to just the unread messages only:

var hasUILoaded = setInterval( function() {
    if( document.getElementById('canvas_frame').contentDocument.getElementsByClassName('J-Zh-I J-J5-Ji J-Pm-I L3') ) {
        clearInterval( hasUILoaded );
        setTimeout( function() {
            var content_frame = document.getElementById('canvas_frame').contentDocument;
            var chkbox = content_frame.getElementsByClassName( 'J-Zh-I J-J5-Ji J-Pm-I L3' );
            var chkbox = chkbox[0].childNodes[0];
            var unreadMessages = content_frame.getElementsByClassName('zE'); // DOM structure: <tr class=zE> <td> <img><input> </td> </tr> so we add ".childNodes[0].childNodes[1]" whenever we want to access the check boxes of each message.
            var allMessages = content_frame.getElementsByClassName('zA');            

            chkbox.onclick = function() {
                if( chkbox.checked ) {
                    var numUnread = unreadMessages.length;;
                    var numAll  = allMessages.length;
                    setTimeout(function () { 
                        if( allMessages[0].childNodes[0].childNodes[1].checked ) { 
                            while ( numAll-- ) {
                                var clickEvent = document.createEvent('MouseEvents');
                                clickEvent.initEvent( 'click', true, true ); 
                            allMessages[numAll].childNodes[0].childNodes[1].dispatchEvent(clickEvent);       
                            }
                        }
                    }, 10);

                    setTimeout(function (){
                        if(!unreadMessages[0].childNodes[0].childNodes[1].checked) {
                            while ( numUnread-- ) {
                                var clickEvent = document.createEvent('MouseEvents');
                                clickEvent.initEvent( 'click', true, true ); 
                unreadMessages[numUnread].childNodes[0].childNodes[1].dispatchEvent(clickEvent);  
                            }
                        }
                    }, 30);
                }
            }
        }, 100);
    }
}, 300);
~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文