See ChangeLog
Monotone-Parent: da4549a652b63b86055b1d36eb59d91375871786 Monotone-Revision: 694484559aed659a4395cabbbbf946966b25ff4e Monotone-Author: crobert@inverse.ca Monotone-Date: 2008-09-30T13:42:48 Monotone-Branch: ca.inverse.sogomaint-2.0.2
parent
4b4a8d3446
commit
d228e96a78
16
ChangeLog
16
ChangeLog
|
@ -1,3 +1,19 @@
|
|||
2008-09-30 Cyril Robert <crobert@inverse.ca>
|
||||
|
||||
* SoObjects/Mailer/SOGoDraftsFolder.m
|
||||
Moved userSpoolFolderPath method to SOGoMailFolder.m
|
||||
* SoObjects/Mailer/SOGoMailFolder.h
|
||||
* SoObjects/Mailer/SOGoMailFolder.m
|
||||
Added userSpoolFolderPath from SOGoDraftsFolder
|
||||
Added archiveUIDs method to create an archive from email uids
|
||||
* UI/MailerUI/UIxMailFolderActions.h
|
||||
* UI/MailerUI/UIxMailFolderActions.m
|
||||
Added saveMessagesAction method to handle the /saveMessages call
|
||||
* UI/MailerUI/product.plist
|
||||
Added a definition for saveMessages
|
||||
|
||||
|
||||
|
||||
2008-09-28 Ludovic Marcotte <lmarcotte@inverse.ca>
|
||||
|
||||
* Modified the following files:
|
||||
|
|
|
@ -47,7 +47,7 @@ static unsigned int newCount;
|
|||
ud = [NSUserDefaults standardUserDefaults];
|
||||
spoolFolder = [ud stringForKey:@"SOGoMailSpoolPath"];
|
||||
if (![spoolFolder length])
|
||||
spoolFolder = @"/tmp/";
|
||||
spoolFolder = @"/tmp/";
|
||||
[spoolFolder retain];
|
||||
|
||||
NSLog(@"Note: using SOGo mail spool folder: %@", spoolFolder);
|
||||
|
@ -102,14 +102,5 @@ static unsigned int newCount;
|
|||
return YES;
|
||||
}
|
||||
|
||||
- (NSString *) userSpoolFolderPath
|
||||
{
|
||||
NSString *login;
|
||||
|
||||
login = [[context activeUser] login];
|
||||
|
||||
return [NSString stringWithFormat: @"%@/%@",
|
||||
spoolFolder, login];
|
||||
}
|
||||
|
||||
@end
|
||||
|
|
|
@ -24,6 +24,7 @@
|
|||
|
||||
#include <SoObjects/Mailer/SOGoMailBaseObject.h>
|
||||
#import <Foundation/NSRange.h>
|
||||
#import <NGObjWeb/WOResponse.h>
|
||||
|
||||
/*
|
||||
SOGoMailFolder
|
||||
|
@ -55,6 +56,7 @@ typedef enum {
|
|||
|
||||
/* messages */
|
||||
- (NSException *) deleteUIDs: (NSArray *) uids inContext: (id) context;
|
||||
- (WOResponse *) archiveUIDs: (NSArray *) uids inContext: (id) context;
|
||||
|
||||
- (NSArray *) fetchUIDsMatchingQualifier: (id)_q sortOrdering: (id) _so;
|
||||
- (NSArray *) fetchUIDs: (NSArray *) _uids parts: (NSArray *) _parts;
|
||||
|
@ -81,6 +83,9 @@ typedef enum {
|
|||
- (NSArray *) allFolderPaths;
|
||||
- (NSArray *) allFolderURLs;
|
||||
|
||||
- (NSString *) userSpoolFolderPath;
|
||||
- (BOOL) ensureSpoolFolderPath;
|
||||
|
||||
@end
|
||||
|
||||
@interface SOGoSpecialMailFolder : SOGoMailFolder
|
||||
|
|
|
@ -23,13 +23,16 @@
|
|||
#import <Foundation/NSEnumerator.h>
|
||||
#import <Foundation/NSURL.h>
|
||||
#import <Foundation/NSUserDefaults.h>
|
||||
#import <Foundation/NSTask.h>
|
||||
|
||||
#import <NGObjWeb/NSException+HTTP.h>
|
||||
#import <NGObjWeb/WOContext+SoObjects.h>
|
||||
#import <NGObjWeb/WOResponse.h>
|
||||
#import <NGExtensions/NSNull+misc.h>
|
||||
#import <NGExtensions/NSURL+misc.h>
|
||||
#import <NGExtensions/NSObject+Logs.h>
|
||||
#import <NGExtensions/NSString+misc.h>
|
||||
#import <NGExtensions/NSFileManager+Extensions.h>
|
||||
|
||||
#import <NGImap4/NGImap4Connection.h>
|
||||
#import <NGImap4/NGImap4Client.h>
|
||||
|
@ -51,6 +54,8 @@ static BOOL aclUsernamesAreQuoted = NO;
|
|||
/* http://www.tools.ietf.org/wg/imapext/draft-ietf-imapext-acl/ */
|
||||
static BOOL aclConformsToIMAPExt = NO;
|
||||
|
||||
static NSString *spoolFolder = nil;
|
||||
|
||||
@interface NGImap4Connection (PrivateMethods)
|
||||
|
||||
- (NSString *) imap4FolderNameForURL: (NSURL *) url;
|
||||
|
@ -65,19 +70,29 @@ static BOOL aclConformsToIMAPExt = NO;
|
|||
NSString *aclStyleStr;
|
||||
|
||||
if (aclStyle == undefined)
|
||||
{
|
||||
ud = [NSUserDefaults standardUserDefaults];
|
||||
aclStyleStr = [ud stringForKey: @"SOGoIMAPAclStyle"];
|
||||
if ([aclStyleStr isEqualToString: @"rfc2086"])
|
||||
aclStyle = rfc2086;
|
||||
else
|
||||
aclStyle = rfc4314;
|
||||
{
|
||||
ud = [NSUserDefaults standardUserDefaults];
|
||||
aclStyleStr = [ud stringForKey: @"SOGoIMAPAclStyle"];
|
||||
if ([aclStyleStr isEqualToString: @"rfc2086"])
|
||||
aclStyle = rfc2086;
|
||||
else
|
||||
aclStyle = rfc4314;
|
||||
|
||||
aclUsernamesAreQuoted
|
||||
= [ud boolForKey: @"SOGoIMAPAclUsernamesAreQuoted"];
|
||||
aclConformsToIMAPExt
|
||||
= [ud boolForKey: @"SOGoIMAPAclConformsToIMAPExt"];
|
||||
}
|
||||
aclUsernamesAreQuoted
|
||||
= [ud boolForKey: @"SOGoIMAPAclUsernamesAreQuoted"];
|
||||
aclConformsToIMAPExt
|
||||
= [ud boolForKey: @"SOGoIMAPAclConformsToIMAPExt"];
|
||||
}
|
||||
|
||||
if (!spoolFolder)
|
||||
{
|
||||
spoolFolder = [ud stringForKey:@"SOGoMailSpoolPath"];
|
||||
if (![spoolFolder length])
|
||||
spoolFolder = @"/tmp/";
|
||||
[spoolFolder retain];
|
||||
|
||||
NSLog(@"Note: using SOGo mail spool folder: %@", spoolFolder);
|
||||
}
|
||||
}
|
||||
|
||||
+ (SOGoIMAPAclStyle) imapAclStyle
|
||||
|
@ -302,6 +317,82 @@ static BOOL aclConformsToIMAPExt = NO;
|
|||
return error;
|
||||
}
|
||||
|
||||
- (WOResponse *) archiveUIDs: (NSArray *) uids
|
||||
inContext: (id) localContext
|
||||
{
|
||||
NSException *error;
|
||||
NSFileManager *fm;
|
||||
NSString *spoolPath, *fileName;
|
||||
NSDictionary *msgs;
|
||||
NSArray *messages;
|
||||
NSData *content, *zipContent;
|
||||
NSTask *zipTask;
|
||||
NSMutableArray *zipTaskArguments;
|
||||
WOResponse *response;
|
||||
int i;
|
||||
|
||||
spoolPath = [self userSpoolFolderPath];
|
||||
if ( ![self ensureSpoolFolderPath] ) {
|
||||
error = [NSException exceptionWithHTTPStatus: 500
|
||||
reason: @"spoolFolderPath doesn't exist"];
|
||||
return (WOResponse *)error;
|
||||
}
|
||||
|
||||
fm = [NSFileManager defaultManager];
|
||||
if ( ![fm fileExistsAtPath: @"/usr/bin/zip"] ) {
|
||||
error = [NSException exceptionWithHTTPStatus: 500
|
||||
reason: @"zip not available"];
|
||||
return (WOResponse *)error;
|
||||
}
|
||||
|
||||
zipTask = [[NSTask alloc] init];
|
||||
[zipTask setCurrentDirectoryPath: spoolPath];
|
||||
[zipTask setLaunchPath: @"/usr/bin/zip"];
|
||||
|
||||
zipTaskArguments = [NSMutableArray arrayWithObjects: nil];
|
||||
[zipTaskArguments addObject: @"SavedMessages.zip"];
|
||||
|
||||
msgs = (NSDictionary *)[self fetchUIDs: uids
|
||||
parts: [NSArray arrayWithObject: @"RFC822"]];
|
||||
messages = [[msgs objectForKey: @"fetch"] retain];
|
||||
|
||||
for (i = 0; i < [messages count]; i++) {
|
||||
content = [[messages objectAtIndex: i] objectForKey: @"message"];
|
||||
|
||||
[content writeToFile:
|
||||
[NSString stringWithFormat:@"%@/%d.eml", spoolPath, [uids objectAtIndex: i]]
|
||||
atomically: YES];
|
||||
|
||||
[zipTaskArguments addObject:
|
||||
[NSString stringWithFormat:@"%d.eml", [uids objectAtIndex: i]]];
|
||||
}
|
||||
|
||||
[zipTask setArguments: zipTaskArguments];
|
||||
[zipTask launch];
|
||||
[zipTask waitUntilExit];
|
||||
|
||||
[zipTask release];
|
||||
|
||||
zipContent = [[NSData alloc] initWithContentsOfFile:
|
||||
[NSString stringWithFormat: @"%@/SavedMessages.zip", spoolPath]];
|
||||
|
||||
for(i = 0; i < [zipTaskArguments count]; i++) {
|
||||
fileName = [zipTaskArguments objectAtIndex: i];
|
||||
[fm removeFileAtPath:
|
||||
[NSString stringWithFormat: @"%@/%@", spoolPath, fileName] handler: nil];
|
||||
}
|
||||
|
||||
response = [[WOResponse alloc] init];
|
||||
[response autorelease];
|
||||
[response setHeader: @"application/zip" forKey:@"content-type"];
|
||||
[response setHeader: @"attachment;filename=SavedMessages.zip" forKey: @"Content-Disposition"];
|
||||
[response setContent: zipContent];
|
||||
|
||||
[zipContent release];
|
||||
|
||||
return response;
|
||||
}
|
||||
|
||||
- (NSArray *) fetchUIDsMatchingQualifier: (id) _q
|
||||
sortOrdering: (id) _so
|
||||
{
|
||||
|
@ -891,6 +982,25 @@ static BOOL aclConformsToIMAPExt = NO;
|
|||
return [userURL absoluteString];
|
||||
}
|
||||
|
||||
- (NSString *) userSpoolFolderPath
|
||||
{
|
||||
NSString *login;
|
||||
|
||||
login = [[context activeUser] login];
|
||||
|
||||
return [NSString stringWithFormat: @"%@/%@",
|
||||
spoolFolder, login];
|
||||
}
|
||||
|
||||
- (BOOL) ensureSpoolFolderPath
|
||||
{
|
||||
NSFileManager *fm;
|
||||
|
||||
fm = [NSFileManager defaultManager];
|
||||
|
||||
return ([fm createDirectoriesAtPath: [self userSpoolFolderPath] attributes:nil]);
|
||||
}
|
||||
|
||||
@end /* SOGoMailFolder */
|
||||
|
||||
@implementation SOGoSpecialMailFolder
|
||||
|
|
|
@ -33,6 +33,7 @@
|
|||
- (WOResponse *) renameFolderAction;
|
||||
- (WOResponse *) deleteFolderAction;
|
||||
- (WOResponse *) deleteMessagesAction;
|
||||
- (WOResponse *) saveMessagesAction;
|
||||
- (WOResponse *) expungeAction;
|
||||
- (WOResponse *) emptyTrashAction;
|
||||
- (WOResponse *) subscribeAction;
|
||||
|
|
|
@ -209,6 +209,33 @@
|
|||
return response;
|
||||
}
|
||||
|
||||
- (WOResponse *) saveMessagesAction
|
||||
{
|
||||
SOGoMailFolder *co;
|
||||
WOResponse *response;
|
||||
NSArray *uids;
|
||||
NSString *value;
|
||||
|
||||
co = [self clientObject];
|
||||
value = [[context request] formValueForKey: @"uid"];
|
||||
response = nil;
|
||||
|
||||
if ([value length] > 0)
|
||||
{
|
||||
uids = [value componentsSeparatedByString: @","];
|
||||
response = [co archiveUIDs: uids inContext: context];
|
||||
if (!response)
|
||||
response = [self responseWith204];
|
||||
}
|
||||
else
|
||||
{
|
||||
response = [self responseWithStatus: 500];
|
||||
[response appendContentString: @"Missing 'uid' parameter."];
|
||||
}
|
||||
|
||||
return response;
|
||||
}
|
||||
|
||||
- (WOResponse *) _setFolderPurpose: (NSString *) purpose
|
||||
{
|
||||
SOGoMailFolder *co;
|
||||
|
|
|
@ -141,6 +141,11 @@
|
|||
actionClass = "UIxMailFolderActions";
|
||||
actionName = "deleteMessages";
|
||||
};
|
||||
saveMessages = {
|
||||
protectedBy = "View";
|
||||
actionClass = "UIxMailFolderActions";
|
||||
actionName = "saveMessages";
|
||||
};
|
||||
setAsDraftsFolder = {
|
||||
protectedBy = "View";
|
||||
actionClass = "UIxMailFolderActions";
|
||||
|
|
|
@ -124,6 +124,7 @@
|
|||
<li><var:string label:value="Label"/></li>
|
||||
<li><var:string label:value="Mark"/></li>
|
||||
<li><!-- separator --></li>
|
||||
<li><var:string label:value="Save As..."/></li>
|
||||
<li><var:string label:value="Print Preview"/></li>
|
||||
<li><var:string label:value="Print..."/></li>
|
||||
<li><var:string label:value="Delete Selected Messages"/></li>
|
||||
|
|
|
@ -539,7 +539,7 @@ function initMailEditor() {
|
|||
var sigLimit = textContent.lastIndexOf("--");
|
||||
if (sigLimit > -1)
|
||||
signatureLength = (textContent.length - sigLimit);
|
||||
if ( userDefaults["SignaturePlacement"] != "above" ) {
|
||||
if ( userDefaults["ReplyPlacement"] != "above" ) {
|
||||
textarea.scrollTop = textarea.scrollHeight;
|
||||
}
|
||||
textarea.observe("focus", onTextFirstFocus);
|
||||
|
|
Loading…
Reference in New Issue