1
0
mirror of https://github.com/gryf/pentadactyl-pm.git synced 2025-12-22 22:08:00 +01:00
Files
pentadactyl-pm/content/mail.js
2008-04-30 11:08:32 +00:00

635 lines
23 KiB
JavaScript

/***** BEGIN LICENSE BLOCK ***** {{{
Version: MPL 1.1/GPL 2.0/LGPL 2.1
The contents of this file are subject to the Mozilla Public License Version
1.1 (the "License"); you may not use this file except in compliance with
the License. You may obtain a copy of the License at
http://www.mozilla.org/MPL/
Software distributed under the License is distributed on an "AS IS" basis,
WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
for the specific language governing rights and limitations under the
License.
(c) 2006-2008: Martin Stubenschrott <stubenschrott@gmx.net>
Alternatively, the contents of this file may be used under the terms of
either the GNU General Public License Version 2 or later (the "GPL"), or
the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
in which case the provisions of the GPL or the LGPL are applicable instead
of those above. If you wish to allow use of your version of this file only
under the terms of either the GPL or the LGPL, and not to allow others to
use your version of this file under the terms of the MPL, indicate your
decision by deleting the provisions above and replace them with the notice
and other provisions required by the GPL or the LGPL. If you do not delete
the provisions above, a recipient may use your version of this file under
the terms of any one of the MPL, the GPL or the LGPL.
}}} ***** END LICENSE BLOCK *****/
liberator.Mail = function ()
{
////////////////////////////////////////////////////////////////////////////////
////////////////////// PRIVATE SECTION /////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////{{{
// used for asynchronously selecting messages after wrapping folders
var selectMessageKeys = [];
var selectMessageCount = 1;
var selectMessageReverse = false;
var folderListener = {
OnItemAdded: function(parentItem, item) { },
OnItemRemoved: function(parentItem, item) { },
OnItemPropertyChanged: function(item, property, oldValue, newValue) { },
OnItemIntPropertyChanged: function(item, property, oldValue, newValue) { },
OnItemBoolPropertyChanged: function(item, property, oldValue, newValue) { },
OnItemUnicharPropertyChanged: function(item, property, oldValue, newValue) { },
OnItemPropertyFlagChanged: function(item, property, oldFlag, newFlag) { },
OnItemEvent: function(folder, event)
{
var eventType = event.toString();
if (eventType == "FolderLoaded")
{
if (folder)
{
var msgFolder = folder.QueryInterface(Components.interfaces.nsIMsgFolder);
dump (msgFolder.prettiestName + " loaded\n");
// Jump to a message when requested
var indices = [];
if (selectMessageKeys.length > 0)
{
for (var j = 0; j < selectMessageKeys.length; j++)
indices.push([gDBView.findIndexFromKey(selectMessageKeys[j], true), selectMessageKeys[j]]);
indices.sort();
let index = selectMessageCount - 1;
if (selectMessageReverse)
index = selectMessageKeys.length - 1 - index;
gDBView.selectMsgByKey(indices[index][1]);
selectMessageKeys = [];
}
}
}
/*else if (eventType == "ImapHdrDownloaded") { }
else if (eventType == "DeleteOrMoveMsgCompleted") { }
else if (eventType == "DeleteOrMoveMsgFailed") { }
else if (eventType == "AboutToCompact") { }
else if (eventType == "CompactCompleted") { }
else if (eventType == "RenameCompleted") { }
else if (eventType == "JunkStatusChanged") { }*/
}
}
var mailSession = Components.classes[mailSessionContractID]
.getService(Components.interfaces.nsIMsgMailSession);
var nsIFolderListener = Components.interfaces.nsIFolderListener;
var notifyFlags = nsIFolderListener.intPropertyChanged | nsIFolderListener.event;
mailSession.AddFolderListener(folderListener, notifyFlags);
function moveOrCopy(copy, destinationFolder, operateOnThread)
{
if (!destinationFolder)
{
liberator.echoerr("E471: Argument required");
return false;
}
var folders = liberator.mail.getFolders(destinationFolder);
if (folders.length == 0)
{
liberator.echoerr("E94: No matching folder for " + destinationFolder);
return false;
}
else if (folders.length > 1)
{
liberator.echoerr("E93: More than one match for " + destinationFolder);
return false;
}
var count = gDBView.selection.count;
if (!count)
{
liberator.beep();
return false;
}
if (copy)
{
MsgCopyMessage(folders[0].URI);
setTimeout(function() { liberator.echo(count + " message(s) copied to " + folders[0].prettyName); }, 100);
}
else
{
MsgMoveMessage(folders[0].URI);
setTimeout(function() { liberator.echo(count + " message(s) moved to " + folders[0].prettyName); }, 100);
}
return true;
}
/////////////////////////////////////////////////////////////////////////////}}}
////////////////////// OPTIONS /////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////{{{
// liberator.options.add(["editor"],
// "Set the external text editor",
// "string", "gvim -f");
liberator.options.add(["layout"],
"Set the layout of the mail window",
"string", "inherit",
{
validator: function (value) { return /^(classic|wide|vertical|inherit)$/.test(value); },
setter: function (value)
{
switch (value)
{
case "classic": ChangeMailLayout(0); break;
case "wide": ChangeMailLayout(1); break;
case "vertical": ChangeMailLayout(2); break;
// case "inherit" just does nothing
}
}
});
/////////////////////////////////////////////////////////////////////////////}}}
////////////////////// MAPPINGS ////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////{{{
var modes = liberator.config.mailModes || [liberator.modes.NORMAL];
liberator.mappings.add(modes, ["<Return>", "i"],
"Inspect (focus) message",
function () { content.focus(); });
liberator.mappings.add(modes, ["d", "<Del>"],
"Move mail to Trash folder",
function () { goDoCommand("cmd_delete"); });
liberator.mappings.add(modes, ["j", "<Right>"],
"Select next message",
function (count) { liberator.mail.selectMessage(function (msg) { return true; }, false, false, count); },
{ flags: liberator.Mappings.flags.COUNT });
liberator.mappings.add(modes, ["J", "<Tab>"],
"Select next unread message",
function (count) { liberator.mail.selectMessage(function (msg) { return !msg.isRead; }, true, false, count); },
{ flags: liberator.Mappings.flags.COUNT });
liberator.mappings.add(modes, ["k", "<Left>"],
"Select previous message",
function (count) { liberator.mail.selectMessage(function (msg) { return true; }, false, true, count); },
{ flags: liberator.Mappings.flags.COUNT });
liberator.mappings.add(modes, ["K"],
"Select previous unread message",
function (count) { liberator.mail.selectMessage(function (msg) { return !msg.isRead; }, true, true, count); },
{ flags: liberator.Mappings.flags.COUNT });
liberator.mappings.add(modes, ["r"],
"Reply to sender",
function () { goDoCommand("cmd_reply"); });
liberator.mappings.add(modes, ["f"],
"Forward message",
function () { goDoCommand("cmd_forward"); });
liberator.mappings.add(modes, ["F"],
"Forward message inline",
function () { goDoCommand("cmd_forwardInline"); });
liberator.mappings.add(modes, ["r"],
"Reply to sender",
function () { goDoCommand("cmd_reply"); });
// GETTING MAIL
liberator.mappings.add(modes, ["gm"],
"Get new messages",
function () { liberator.mail.getNewMessages(); });
liberator.mappings.add(modes, ["gM"],
"Get new messages for current account only",
function () { liberator.mail.getNewMessages(true); });
// MOVING MAIL
liberator.mappings.add(modes, ["c"],
"Change folders",
function () { liberator.commandline.open(":", "goto ", liberator.modes.EX); });
liberator.mappings.add(modes, ["s"],
"Move selected messages",
function () { liberator.commandline.open(":", "moveto ", liberator.modes.EX); });
liberator.mappings.add(modes, ["S"],
"Copy selected messages",
function () { liberator.commandline.open(":", "copyto ", liberator.modes.EX); });
liberator.mappings.add(modes, ["<C-s>"],
"Archive message",
function () { moveOrCopy(false, "Archive"); });
liberator.mappings.add(modes, ["]s"],
"Select next starred message",
function (count) { liberator.mail.selectMessage(function(msg) { return msg.isFlagged; }, true, false, count); },
{ flags: liberator.Mappings.flags.COUNT });
liberator.mappings.add(modes, ["[s"],
"Select previous starred message",
function (count) { liberator.mail.selectMessage(function(msg) { return msg.isFlagged; }, true, true, count); },
{ flags: liberator.Mappings.flags.COUNT });
liberator.mappings.add(modes, ["]a"],
"Select next message with an attachment",
function (count) { liberator.mail.selectMessage(function(msg) { return gDBView.db.HasAttachments(msg.messageKey); }, true, false, count); },
{ flags: liberator.Mappings.flags.COUNT });
liberator.mappings.add(modes, ["[a"],
"Select previous message with an attachment",
function (count) { liberator.mail.selectMessage(function(msg) { return gDBView.db.HasAttachments(msg.messageKey); }, true, true, count); },
{ flags: liberator.Mappings.flags.COUNT });
// FOLDER SWITCHING
liberator.mappings.add(modes, ["gi"],
"Go to inbox",
function (count)
{
var folder = liberator.mail.getFolders("Inbox", false, true)[(count > 0) ? (count - 1) : 0];
if (folder)
SelectFolder(folder.URI);
else
liberator.beep();
},
{ flags: liberator.Mappings.flags.COUNT });
liberator.mappings.add(modes, ["<C-n>"],
"Select next folder",
function (count)
{
count = (count > 0 ) ? count : 1;
var tree = GetFolderTree();
var c = tree.currentIndex;
if (c + count >= tree.view.rowCount)
{
liberator.beep();
return;
}
tree.view.selection.timedSelect(c + count, tree._selectDelay );
},
{ flags: liberator.Mappings.flags.COUNT });
liberator.mappings.add(modes, ["<C-p>"],
"Select previous folder",
function (count)
{
count = (count > 0 ) ? count : 1;
var tree = GetFolderTree();
var c = tree.currentIndex;
if (c - count < 0)
{
liberator.beep();
return;
}
tree.view.selection.timedSelect(c - count, tree._selectDelay );
},
{ flags: liberator.Mappings.flags.COUNT });
// THREADING
liberator.mappings.add(modes, ["za"],
"Toggle thread collapsed/expanded",
function () { if (!liberator.mail.expandThread()) liberator.mail.collapseThread(); });
liberator.mappings.add(modes, ["zc"],
"Collapse thread",
function () { liberator.mail.collapseThread(); });
liberator.mappings.add(modes, ["zo"],
"Open thread",
function () { liberator.mail.expandThread(); });
liberator.mappings.add(modes, ["zr", "zR"],
"Expand all threads",
function () { goDoCommand("cmd_expandAllThreads"); });
liberator.mappings.add(modes, ["zm", "zM"],
"Collapse all threads",
function () { goDoCommand("cmd_collapseAllThreads"); });
liberator.mappings.add(modes, ["<C-i>"],
"Go forward",
function (count) { if (count < 1) count = 1; while (count--) GoNextMessage(nsMsgNavigationType.forward, true); },
{ flags: liberator.Mappings.flags.COUNT });
liberator.mappings.add(modes, ["<C-o>"],
"Go back",
function (count) { if (count < 1) count = 1; while (count--) GoNextMessage(nsMsgNavigationType.back, true); },
{ flags: liberator.Mappings.flags.COUNT });
liberator.mappings.add(modes, ["gg"],
"Select first message",
function (count) { if (count < 1) count = 1; while (count--) GoNextMessage(nsMsgNavigationType.firstMessage, true); },
{ flags: liberator.Mappings.flags.COUNT });
liberator.mappings.add(modes, ["G"],
"Select last message",
function (count) { if (count < 1) count = 1; while (count--) GoNextMessage(nsMsgNavigationType.lastMessage, false); },
{ flags: liberator.Mappings.flags.COUNT });
// tagging messages
liberator.mappings.add(modes, ["tr"],
"Toggle selected messages read",
function ()
{
if (!GetSelectedMessages())
{
liberator.beep();
return;
}
MsgMarkMsgAsRead();
});
liberator.mappings.add(modes, ["tR"],
"Tag thread as read",
function ()
{
// TODO: ensure thread or beep
MsgMarkThreadAsRead();
});
liberator.mappings.add(modes, ["ts"],
"Toggle selected messages starred",
function ()
{
if (!GetSelectedMessages())
{
liberator.beep();
return;
}
MsgMarkMsgAsFlagged();
});
liberator.mappings.add(modes, ["T"],
"Mark current folder as read",
function ()
{
if (liberator.mail.currentFolder.isServer)
{
liberator.beep();
return;
}
liberator.mail.currentFolder.markAllMessagesRead();
});
liberator.mappings.add(modes, ["<C-t>"],
"Mark all messages as read",
function ()
{
liberator.mail.getFolders("", false).forEach(function(folder) {
folder.markAllMessagesRead();
});
});
/////////////////////////////////////////////////////////////////////////////}}}
////////////////////// COMMANDS ////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////{{{
liberator.commands.add(["go[to]"],
"Select a folder",
function (args, special, count)
{
args = args || "Inbox";
count = count > 0 ? (count - 1) : 0;
var folder = liberator.mail.getFolders(args, true, true)[count];
if (!folder)
liberator.echoerr("Folder \"" + args + "\" does not exist");
else
SelectFolder(folder.URI);
});
liberator.commands.add(["copy[to]"],
"Copy selected messages",
function (args, special) { moveOrCopy(true, args); });
liberator.commands.add(["move[to]"],
"Move selected messages",
function (args, special) { moveOrCopy(false, args); });
liberator.commands.add(["get[messages]"],
"Check for new messages",
function (args, special)
{
if (args)
{
liberator.echoerr("E488: Trailing characters");
return;
}
liberator.mail.getNewMessages(!special);
});
/////////////////////////////////////////////////////////////////////////////}}}
////////////////////// PUBLIC SECTION //////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////{{{
return {
get currentAccount() { return this.currentFolder.rootFolder; },
get currentFolder() {
var tree = GetFolderTree();
return GetFolderResource(tree, tree.currentIndex).
QueryInterface(Components.interfaces.nsIMsgFolder);
},
getFolders: function(filter, includeServers, includeMsgFolders)
{
var folders = [];
if (!filter)
filter = "";
if (typeof includeServers == "undefined")
includeServers = false;
if (typeof includeMsgFolders == "undefined")
includeMsgFolders = true;
var tree = GetFolderTree();
for (let i = 0; i < tree.view.rowCount; i++)
{
var resource = GetFolderResource(tree, i).QueryInterface(Components.interfaces.nsIMsgFolder);
if ((resource.isServer && !includeServers) || (!resource.isServer && !includeMsgFolders))
continue;
if (resource.prettiestName.toLowerCase().indexOf(filter.toLowerCase()) >= 0)
folders.push(resource);
}
return folders;
},
getNewMessages: function(currentAccountOnly)
{
var accounts = currentAccountOnly ? [this.currentAccount]
: this.getFolders("", true, false);
accounts.forEach( function(account) { account.getNewMessages(msgWindow, null); });
},
getStatistics: function(currentAccountOnly)
{
var accounts = currentAccountOnly ? [this.currentAccount]
: this.getFolders("", true, false);
var unreadCount = 0, totalCount = 0, newCount = 0;;
for (var i = 0; i < accounts.length; i++)
{
var account = accounts[i];
unreadCount += account.getNumUnread(true); // true == deep (includes subfolders)
totalCount += account.getTotalMessages(true);
newCount += account.getNumUnread(true);
}
return { numUnread: unreadCount, numTotal: totalCount, numNew: newCount }
},
collapseThread: function()
{
var tree = GetThreadTree();
if (tree)
{
var parent = tree.currentIndex;
while (true)
{
var tmp = tree.view.getParentIndex(parent);
if (tmp >= 0)
parent = tmp;
else
break;
}
if (tree.changeOpenState(parent, false))
{
tree.view.selection.select(parent);
tree.treeBoxObject.ensureRowIsVisible(parent);
return true;
}
}
return false;
},
expandThread: function()
{
var tree = GetThreadTree();
if (tree)
{
var row = tree.currentIndex;
if (row >= 0 && tree.changeOpenState(row, true))
return true;
}
return false;
},
selectMessage: function(validatorFunc, canWrap, reverse, count)
{
if (typeof validatorFunc != "function")
return;
if (typeof count != "number" || count < 1)
count = 1;
// first try to find in current folder
if (gDBView)
{
//var curIndex =
// FIXME: doesn't work with collapsed threads
for (var i = gDBView.selection.currentIndex + (reverse ? -1 : 1);
(reverse ? ( i >= 0) : (i < gDBView.rowCount)); reverse ? i-- : i++)
{
var key = gDBView.getKeyAt(i);
var msg = gDBView.db.GetMsgHdrForKey(key);
if (validatorFunc(msg))
count--;
if (count == 0)
{
// gDBView.selectMsgByKey(key);
gDBView.selection.timedSelect(i, GetThreadTree()._selectDelay || 500);
GetThreadTree().treeBoxObject.ensureRowIsVisible(i);
return;
}
}
}
// then in other folders
if (canWrap)
{
selectMessageReverse = reverse;
var folders = this.getFolders("", true, true);
var ci = GetFolderTree().currentIndex;
for (var i = 1; i <= folders.length; i++)
{
let index = (i + ci) % folders.length;
if (reverse)
index = folders.length - 1 - index;
var folder = folders[index];
if (folder.isServer)
continue;
selectMessageCount = count;
selectMessageKeys = [];
// sometimes folder.getMessages can fail with an exception
// TODO: find out why, and solve the problem
try
{
var msgs = folder.getMessages(msgWindow);
}
catch (e)
{
dump("ERROR: " + folder.prettyName + " failed to getMessages\n");
continue;
}
while (msgs.hasMoreElements())
{
var msg = msgs.getNext().QueryInterface(Components.interfaces.nsIMsgDBHdr);
if (validatorFunc(msg))
{
count--;
selectMessageKeys.push(msg.messageKey);
}
}
if (count <= 0)
{
// SelectFolder is asynchronous, message is selected in folderListener
SelectFolder(folder.URI);
return;
}
}
}
// TODO: finally for the "rest" of the current folder
liberator.beep();
}
};
//}}}
};
// vim: set fdm=marker sw=4 ts=4 et: