253
257
help=Evolution.__doc__)
256
class Mutt(ExternalMailClient):
257
"""Mutt mail client."""
260
class Mutt(BodyExternalMailClient):
261
__doc__ = """Mutt mail client."""
259
263
_client_commands = ['mutt']
261
def _get_compose_commandline(self, to, subject, attach_path):
265
def _get_compose_commandline(self, to, subject, attach_path, body=None):
262
266
"""See ExternalMailClient._get_compose_commandline"""
263
267
message_options = []
264
268
if subject is not None:
266
270
if attach_path is not None:
267
271
message_options.extend(['-a',
268
272
self._encode_path(attach_path, 'attachment')])
274
# Store the temp file object in self, so that it does not get
275
# garbage collected and delete the file before mutt can read it.
276
self._temp_file = tempfile.NamedTemporaryFile(
277
prefix="mutt-body-", suffix=".txt")
278
self._temp_file.write(body)
279
self._temp_file.flush()
280
message_options.extend(['-i', self._temp_file.name])
269
281
if to is not None:
270
message_options.append(self._encode_safe(to))
282
message_options.extend(['--', self._encode_safe(to)])
271
283
return message_options
272
284
mail_client_registry.register('mutt', Mutt,
273
285
help=Mutt.__doc__)
276
288
class Thunderbird(BodyExternalMailClient):
277
"""Mozilla Thunderbird (or Icedove)
289
__doc__ = """Mozilla Thunderbird (or Icedove)
279
291
Note that Thunderbird 1.5 is buggy and does not support setting
280
292
"to" simultaneously with including a attachment.
331
343
class Claws(ExternalMailClient):
332
"""Claws mail client."""
344
__doc__ = """Claws mail client."""
334
348
_client_commands = ['claws-mail']
336
def _get_compose_commandline(self, to, subject, attach_path):
350
def _get_compose_commandline(self, to, subject, attach_path, body=None,
337
352
"""See ExternalMailClient._get_compose_commandline"""
338
compose_url = ['mailto:']
340
compose_url.append(self._encode_safe(to))
341
compose_url.append('?')
354
if from_ is not None:
355
compose_url.append('from=' + urllib.quote(from_))
342
356
if subject is not None:
343
357
# Don't use urllib.quote_plus because Claws doesn't seem
344
358
# to recognise spaces encoded as "+".
345
359
compose_url.append(
346
'subject=%s' % urllib.quote(self._encode_safe(subject)))
360
'subject=' + urllib.quote(self._encode_safe(subject)))
363
'body=' + urllib.quote(self._encode_safe(body)))
364
# to must be supplied for the claws-mail --compose syntax to work.
366
raise errors.NoMailAddressSpecified()
367
compose_url = 'mailto:%s?%s' % (
368
self._encode_safe(to), '&'.join(compose_url))
347
369
# Collect command-line options.
348
message_options = ['--compose', ''.join(compose_url)]
370
message_options = ['--compose', compose_url]
349
371
if attach_path is not None:
350
372
message_options.extend(
351
373
['--attach', self._encode_path(attach_path, 'attachment')])
352
374
return message_options
376
def _compose(self, prompt, to, subject, attach_path, mime_subtype,
377
extension, body=None, from_=None):
378
"""See ExternalMailClient._compose"""
380
from_ = self.config.get_user_option('email')
381
super(Claws, self)._compose(prompt, to, subject, attach_path,
382
mime_subtype, extension, body, from_)
353
385
mail_client_registry.register('claws', Claws,
354
386
help=Claws.__doc__)
357
389
class XDGEmail(BodyExternalMailClient):
358
"""xdg-email attempts to invoke the user's preferred mail client"""
390
__doc__ = """xdg-email attempts to invoke the user's preferred mail client"""
360
392
_client_commands = ['xdg-email']
484
521
class MAPIClient(BodyExternalMailClient):
485
"""Default Windows mail client launched using MAPI."""
522
__doc__ = """Default Windows mail client launched using MAPI."""
487
524
def _compose(self, prompt, to, subject, attach_path, mime_subtype,
525
extension, body=None):
489
526
"""See ExternalMailClient._compose.
491
528
This implementation uses MAPI via the simplemapi ctypes wrapper
502
539
help=MAPIClient.__doc__)
542
class MailApp(BodyExternalMailClient):
543
__doc__ = """Use MacOS X's Mail.app for sending email messages.
545
Although it would be nice to use appscript, it's not installed
546
with the shipped Python installations. We instead build an
547
AppleScript and invoke the script using osascript(1). We don't
548
use the _encode_safe() routines as it's not clear what encoding
549
osascript expects the script to be in.
552
_client_commands = ['osascript']
554
def _get_compose_commandline(self, to, subject, attach_path, body=None,
556
"""See ExternalMailClient._get_compose_commandline"""
558
fd, self.temp_file = tempfile.mkstemp(prefix="bzr-send-",
561
os.write(fd, 'tell application "Mail"\n')
562
os.write(fd, 'set newMessage to make new outgoing message\n')
563
os.write(fd, 'tell newMessage\n')
565
os.write(fd, 'make new to recipient with properties'
566
' {address:"%s"}\n' % to)
567
if from_ is not None:
568
# though from_ doesn't actually seem to be used
569
os.write(fd, 'set sender to "%s"\n'
570
% sender.replace('"', '\\"'))
571
if subject is not None:
572
os.write(fd, 'set subject to "%s"\n'
573
% subject.replace('"', '\\"'))
575
# FIXME: would be nice to prepend the body to the
576
# existing content (e.g., preserve signature), but
577
# can't seem to figure out the right applescript
579
os.write(fd, 'set content to "%s\\n\n"\n' %
580
body.replace('"', '\\"').replace('\n', '\\n'))
582
if attach_path is not None:
583
# FIXME: would be nice to first append a newline to
584
# ensure the attachment is on a new paragraph, but
585
# can't seem to figure out the right applescript
587
os.write(fd, 'tell content to make new attachment'
588
' with properties {file name:"%s"}'
589
' at after the last paragraph\n'
590
% self._encode_path(attach_path, 'attachment'))
591
os.write(fd, 'set visible to true\n')
592
os.write(fd, 'end tell\n')
593
os.write(fd, 'end tell\n')
595
os.close(fd) # Just close the handle but do not remove the file.
596
return [self.temp_file]
597
mail_client_registry.register('mail.app', MailApp,
598
help=MailApp.__doc__)
505
601
class DefaultMail(MailClient):
506
"""Default mail handling. Tries XDGEmail (or MAPIClient on Windows),
602
__doc__ = """Default mail handling. Tries XDGEmail (or MAPIClient on Windows),
507
603
falls back to Editor"""
509
607
def _mail_client(self):
510
608
"""Determine the preferred mail client for this platform"""
511
609
if osutils.supports_mapi():