Jak mogę wysłać pocztę z aplikacji na iPhone'a?

242

Chcę wysłać wiadomość e-mail z mojej aplikacji na iPhone'a. Słyszałem, że iOS SDK nie ma interfejsu API poczty e-mail. Nie chcę używać następującego kodu, ponieważ spowoduje on zamknięcie mojej aplikacji:

NSString *url = [NSString stringWithString: @"mailto:[email protected][email protected]&subject=Greetings%20from%20Cupertino!&body=Wish%20you%20were%20here!"];
[[UIApplication sharedApplication] openURL: [NSURL URLWithString: url]];

Jak mogę wysłać wiadomość e-mail z mojej aplikacji?

Khushi
źródło

Odpowiedzi:

430

W iOS 3.0 i nowszych wersjach powinieneś użyć MFMailComposeViewControllerklasy i MFMailComposeViewControllerDelegateprotokołu, które są schowane w frameworku MessageUI.

Najpierw dodaj ramę i zaimportuj:

#import <MessageUI/MFMailComposeViewController.h>

Następnie, aby wysłać wiadomość:

MFMailComposeViewController* controller = [[MFMailComposeViewController alloc] init];
controller.mailComposeDelegate = self;
[controller setSubject:@"My Subject"];
[controller setMessageBody:@"Hello there." isHTML:NO]; 
if (controller) [self presentModalViewController:controller animated:YES];
[controller release];

Następnie użytkownik wykonuje pracę i otrzymujesz oddzwonienie na czas:

- (void)mailComposeController:(MFMailComposeViewController*)controller  
          didFinishWithResult:(MFMailComposeResult)result 
                        error:(NSError*)error;
{
  if (result == MFMailComposeResultSent) {
    NSLog(@"It's away!");
  }
  [self dismissModalViewControllerAnimated:YES];
}

Pamiętaj, aby sprawdzić, czy urządzenie jest skonfigurowane do wysyłania wiadomości e-mail:

if ([MFMailComposeViewController canSendMail]) {
  // Show the composer
} else {
  // Handle the error
}
PeyloW
źródło
5
+1. Struktury, które wymagają importu, wymieniono tutaj ( mobileorchard.com/... ).
Dan Rosenstark
71
Aby zaoszczędzić ci skoku, musisz #import <MessageUI / MFMailComposeViewController.h>
TomH
22
Wystarczy zauważyć, że ponieważ ta odpowiedź została napisana metod UIViewController za presentModalViewController:animated:i dismissModalViewControllerAnimated:zostały oznaczone jako przestarzałe - zamiast metod zastępczych blokowe oparte presentViewController:animated:completion:i dismissViewControllerAnimated:completion:powinny być stosowane.
2
I nie zapomnij ustawić delegata w .h @interface viewController : UIViewController <MFMailComposeViewControllerDelegate>
Nazir
18
A w IOS 6 [self presentModalViewController:controller animated:YES]; zamień na [self presentViewController:controller animated:YES completion:nil]; i [self dismissModalViewControllerAnimated:YES]; zamień na [self dismissViewControllerAnimated:YES completion:nil];
Nazir
61

MFMailComposeViewController to sposób, w który należy przejść po wydaniu oprogramowania iPhone OS 3.0. Możesz spojrzeć na przykładowy kod lub samouczek, który napisałem .

Mugunth
źródło
2
Niesamowity post autorstwa Mugunth. Dobra droga, kolego!
Jordan,
1
To naprawdę niesamowite. Dzięki. Zaprojektowałem widok specjalnie do przyjmowania wiadomości e-mail i tematu od użytkownika. implementując ten sam kod, ponownie pokazuje nieco podobny widok. czy mogę wywołać metodę delegowania z mojego zdarzenia naciśnięcia przycisku w klasie kontrolera widoku Dziękujemy za pomoc,
Shibin
Pobrałem ten sam przykładowy kod, ale nie wysyłam żadnej wiadomości. Wyświetla jedynie monit o pomyślne wysłanie poczty, ale nie jest odbierana. Próbowałem dodać framework MessageUI, który domyślnie był wyświetlany w kolorze czerwonym, ale nadal aplikacja nie wysyła wiadomości. Każda pomoc w tym zakresie będzie bardzo mile widziana. Testuję aplikację w symulatorze.
Ravi shankar,
Nie można wysłać wiadomości e-mail z symulatora.
malaki1974,
20

Kilka rzeczy, które chciałbym tutaj dodać:

  1. Użycie adresu URL mailto nie będzie działać w symulatorze, ponieważ mail.app nie jest zainstalowany w symulatorze. Działa jednak na urządzeniu.

  2. Długość adresu URL mailto jest ograniczona. Jeśli adres URL jest dłuższy niż 4096 znaków, mail.app nie uruchomi się.

  3. W OS 3.0 wprowadzono nową klasę, która umożliwia wysyłanie wiadomości e-mail bez opuszczania aplikacji. Zobacz klasę MFMailComposeViewController.

Jeff Atwood
źródło
13

Jeśli chcesz wysłać wiadomość e-mail z aplikacji, powyższy kod jest jedynym sposobem, aby to zrobić, chyba że kodujesz własnego klienta poczty (SMTP) w aplikacji lub nie masz serwera, który wyśle ​​Ci pocztę.

Na przykład możesz zakodować aplikację, aby wywoływała adres URL na serwerze, który wysyłałby pocztę za Ciebie. Następnie wystarczy wywołać adres URL ze swojego kodu.

Zauważ, że z powyższym kodem nie możesz dołączyć niczego do wiadomości e-mail, co pozwoli ci na to metoda klienta SMTP, a także metoda po stronie serwera.

Genericrich
źródło
12

Poniższy kod jest używany w mojej aplikacji do wysyłania wiadomości e-mail z załącznikiem. Załączniki są obrazem. Możesz wysłać dowolny typ pliku, należy tylko pamiętać, że musisz określić poprawny typ „mimeType”

dodaj to do pliku .h

#import <MessageUI/MFMailComposeViewController.h>

Dodaj MessageUI.framework do pliku projektu

NSArray *paths = SSearchPathForDirectoriesInDomains(NSDocumentDirectory,NSUserDomainMask, YES);

NSString *documentsDirectory = [paths objectAtIndex:0];

NSString *getImagePath = [documentsDirectory stringByAppendingPathComponent:@"myGreenCard.png"];



MFMailComposeViewController* controller = [[MFMailComposeViewController alloc] init];
controller.mailComposeDelegate = self;
[controller setSubject:@"Green card application"];
[controller setMessageBody:@"Hi , <br/>  This is my new latest designed green card." isHTML:YES]; 
[controller addAttachmentData:[NSData dataWithContentsOfFile:getImagePath] mimeType:@"png" fileName:@"My Green Card"];
if (controller)
    [self presentModalViewController:controller animated:YES];
[controller release];

Metoda delegowania jest pokazana poniżej

  -(void)mailComposeController:(MFMailComposeViewController*)controller didFinishWithResult:(MFMailComposeResult)result error:(NSError*)error;
{
    if (result == MFMailComposeResultSent) {
        NSLog(@"It's away!");
    }
    [self dismissModalViewControllerAnimated:YES];
}
Kannan Prasad
źródło
11

Jest to kod, który może pomóc, ale nie zapomnij dołączyć framewark ui wiadomości i dołączyć metodę delegatów MFMailComposeViewControllerDelegate

-(void)EmailButtonACtion{

        if ([MFMailComposeViewController canSendMail])
        {
            MFMailComposeViewController *controller = [[MFMailComposeViewController alloc] init];
            controller.mailComposeDelegate = self;
            [controller.navigationBar setBackgroundImage:[UIImage imageNamed:@"navigation_bg_iPhone.png"] forBarMetrics:UIBarMetricsDefault];
            controller.navigationBar.tintColor = [UIColor colorWithRed:51.0/255.0 green:51.0/255.0 blue:51.0/255.0 alpha:1.0];
            [controller setSubject:@""];
            [controller setMessageBody:@" " isHTML:YES];
            [controller setToRecipients:[NSArray arrayWithObjects:@"",nil]];
            UIPasteboard *pasteboard = [UIPasteboard generalPasteboard];
            UIImage *ui = resultimg.image;
            pasteboard.image = ui;
            NSData *imageData = [NSData dataWithData:UIImagePNGRepresentation(ui)];
            [controller addAttachmentData:imageData mimeType:@"image/png" fileName:@" "];
            [self presentViewController:controller animated:YES completion:NULL];
        }
        else{
            UIAlertView *alert=[[UIAlertView alloc] initWithTitle:@"alrt" message:nil delegate:self cancelButtonTitle:@"ok" otherButtonTitles: nil] ;
            [alert show];
        }

    }
    -(void)mailComposeController:(MFMailComposeViewController*)controller didFinishWithResult:(MFMailComposeResult)result error:(NSError*)error
    {

        [MailAlert show];
        switch (result)
        {
            case MFMailComposeResultCancelled:
                MailAlert.message = @"Email Cancelled";
                break;
            case MFMailComposeResultSaved:
                MailAlert.message = @"Email Saved";
                break;
            case MFMailComposeResultSent:
                MailAlert.message = @"Email Sent";
                break;
            case MFMailComposeResultFailed:
                MailAlert.message = @"Email Failed";
                break;
            default:
                MailAlert.message = @"Email Not Sent";
                break;
        }
        [self dismissViewControllerAnimated:YES completion:NULL];
        [MailAlert show];
    }
mandeep
źródło
Dziękuję bardzo! Bardzo przydatny przykład z treścią HTML.
Resty,
4

Swift 2.2. Na podstawie odpowiedzi Esq

import Foundation
import MessageUI

class MailSender: NSObject, MFMailComposeViewControllerDelegate {

    let parentVC: UIViewController

    init(parentVC: UIViewController) {
        self.parentVC = parentVC
        super.init()
    }

    func send(title: String, messageBody: String, toRecipients: [String]) {
        if MFMailComposeViewController.canSendMail() {
            let mc: MFMailComposeViewController = MFMailComposeViewController()
            mc.mailComposeDelegate = self
            mc.setSubject(title)
            mc.setMessageBody(messageBody, isHTML: false)
            mc.setToRecipients(toRecipients)
            parentVC.presentViewController(mc, animated: true, completion: nil)
        } else {
            print("No email account found.")
        }
    }

    func mailComposeController(controller: MFMailComposeViewController,
        didFinishWithResult result: MFMailComposeResult, error: NSError?) {

            switch result.rawValue {
            case MFMailComposeResultCancelled.rawValue: print("Mail Cancelled")
            case MFMailComposeResultSaved.rawValue: print("Mail Saved")
            case MFMailComposeResultSent.rawValue: print("Mail Sent")
            case MFMailComposeResultFailed.rawValue: print("Mail Failed")
            default: break
            }

            parentVC.dismissViewControllerAnimated(false, completion: nil)
    }
}

Kod klienta :

var ms: MailSender?

@IBAction func onSendPressed(sender: AnyObject) {
    ms = MailSender(parentVC: self)
    let title = "Title"
    let messageBody = "/programming/310946/how-can-i-send-mail-from-an-iphone-application this question."
    let toRecipents = ["[email protected]"]
    ms?.send(title, messageBody: messageBody, toRecipents: toRecipents)
}
Evdzhan Mustafa
źródło
4

Aby wysłać wiadomość e-mail z aplikacji iPhone'a, musisz wykonać poniższą listę zadań.

Krok 1: Zaimportuj do#import <MessageUI/MessageUI.h> klasy kontrolera, do której chcesz wysłać wiadomość e-mail.

Krok 2: Dodaj delegata do kontrolera, jak pokazano poniżej

 @interface <yourControllerName> : UIViewController <MFMessageComposeViewControllerDelegate, MFMailComposeViewControllerDelegate>

Krok 3: Dodaj poniższą metodę wysyłania wiadomości e-mail.

 - (void) sendEmail {
 // Check if your app support the email.
 if ([MFMailComposeViewController canSendMail]) {
    // Create an object of mail composer.
    MFMailComposeViewController *mailComposer =      [[MFMailComposeViewController alloc] init];
    // Add delegate to your self.
    mailComposer.mailComposeDelegate = self;
    // Add recipients to mail if you do not want to add default recipient then remove below line.
    [mailComposer setToRecipients:@[<add here your recipient objects>]];
    // Write email subject.
    [mailComposer setSubject:@“<Your Subject Here>”];
    // Set your email body and if body contains HTML then Pass “YES” in isHTML.
    [mailComposer setMessageBody:@“<Your Message Body>” isHTML:NO];
    // Show your mail composer.
    [self presentViewController:mailComposer animated:YES completion:NULL];
 }
 else {
 // Here you can show toast to user about not support to sending email.
}
}

Krok 4: Wdrożenie MFMailComposeViewController Delegat

- (void)mailComposeController:(MFMailComposeViewController *)controller didFinishWithResult:(MFMailComposeResult)result error:(nullable NSError *)error {
[controller dismissViewControllerAnimated:TRUE completion:nil];


switch (result) {
   case MFMailComposeResultSaved: {
    // Add code on save mail to draft.
    break;
}
case MFMailComposeResultSent: {
    // Add code on sent a mail.
    break;
}
case MFMailComposeResultCancelled: {
    // Add code on cancel a mail.
    break;
}
case MFMailComposeResultFailed: {
    // Add code on failed to send a mail.
    break;
}
default:
    break;
}
}
Patrick R.
źródło
czy ta odpowiedź zawiera jakieś nowe informacje, które nie są jeszcze zawarte w jednej z istniejących odpowiedzi?
Florian Koch,
2

Swift 2.0

func mailComposeController(controller: MFMailComposeViewController, didFinishWithResult result: MFMailComposeResult, error: NSError?){
    if let error = error{
        print("Error: \(error)")
    }else{
        //NO Error
        //------------------------------------------------
        var feedbackMsg = ""

        switch result.rawValue {
        case MFMailComposeResultCancelled.rawValue:
            feedbackMsg = "Mail Cancelled"
        case MFMailComposeResultSaved.rawValue:
            feedbackMsg = "Mail Saved"
        case MFMailComposeResultSent.rawValue:
            feedbackMsg = "Mail Sent"
        case MFMailComposeResultFailed.rawValue:
            feedbackMsg = "Mail Failed"
        default:
            feedbackMsg = ""
        }

        print("Mail: \(feedbackMsg)")

        //------------------------------------------------
    }
}
brian.clear
źródło
1

Oto wersja Swift:

import MessageUI

class YourVC: UIViewController {
    override func viewDidLoad() {
        super.viewDidLoad()
        if MFMailComposeViewController.canSendMail() {
            var emailTitle = "Vea Software Feedback"
            var messageBody = "Vea Software! :) "
            var toRecipents = ["[email protected]"]
            var mc:MFMailComposeViewController = MFMailComposeViewController()
            mc.mailComposeDelegate = self
            mc.setSubject(emailTitle)
            mc.setMessageBody(messageBody, isHTML: false)
            mc.setToRecipients(toRecipents)
            self.presentViewController(mc, animated: true, completion: nil)
        } else {
            println("No email account found")
        }
    }
}

extension YourVC: MFMailComposeViewControllerDelegate {
    func mailComposeController(controller: MFMailComposeViewController!, didFinishWithResult result: MFMailComposeResult, error: NSError!) {
        switch result.value {
        case MFMailComposeResultCancelled.value:
            println("Mail Cancelled")
        case MFMailComposeResultSaved.value:
            println("Mail Saved")
        case MFMailComposeResultSent.value:
            println("Mail Sent")
        case MFMailComposeResultFailed.value:
            println("Mail Failed")
        default:
            break
        }
        self.dismissViewControllerAnimated(false, completion: nil)
    }
}

Źródło

Esqarrouth
źródło
0

Napisałem proste opakowanie o nazwie KRNSendEmail, które upraszcza wysyłanie wiadomości e-mail do jednego wywołania metody.

KRNSendEmail jest dobrze udokumentowany i dodany do CocoaPods.

https://github.com/ulian-onua/KRNSendEmail

Julian D.
źródło