Sto*_*dio 1 class-hierarchy messageui ios swift mfmailcomposeviewcontroller
非常感谢 Paulw11 帮助我解决了这个问题。我在这里添加了完整的代码以便于重用:
班级:
import UIKit
import MessageUI
struct Feedback {
let recipients: [String]
let subject: String
let body: String
let footer: String
}
class FeedbackManager: NSObject, MFMailComposeViewControllerDelegate {
private var feedback: Feedback
private var completion: ((Result<MFMailComposeResult,Error>)->Void)?
override init() {
fatalError("Use FeedbackManager(feedback:)")
}
init?(feedback: Feedback) {
guard MFMailComposeViewController.canSendMail() else {
return nil
}
self.feedback = feedback
}
func send(on viewController: UIViewController, completion:(@escaping(Result<MFMailComposeResult,Error>)->Void)) {
let mailVC = MFMailComposeViewController()
self.completion = completion
mailVC.mailComposeDelegate = self
mailVC.setToRecipients(feedback.recipients)
mailVC.setSubject(feedback.subject)
mailVC.setMessageBody("<p>\(feedback.body)<br><br><br><br><br>\(feedback.footer)</p>", isHTML: true)
viewController.present(mailVC, animated:true)
}
func mailComposeController(_ controller: MFMailComposeViewController, didFinishWith result: MFMailComposeResult, error: Error?) {
if let error = error {
completion?(.failure(error))
controller.dismiss(animated: true)
} else {
completion?(.success(result))
controller.dismiss(animated: true)
}
}
}
Run Code Online (Sandbox Code Playgroud)
在视图控制器中:
添加变量:
var feedbackManager: FeedbackManager?
Run Code Online (Sandbox Code Playgroud)
使用:
let feedback = Feedback(recipients: "String", subject: "String", body: "Body", footer: "String")
if let feedManager = FeedbackManager(feedback: feedback) {
self.feedbackManager = feedManager
self.feedbackManager?.send(on: self) { [weak self] result in
switch result {
case .failure(let error):
print("error: ", error.localizedDescription)
// Do something with the error
case .success(let mailResult):
print("Success")
// Do something with the result
}
self?.feedbackManager = nil
}
} else { // Cant Send Email: // Added UI Alert:
let failedMenu = UIAlertController(title: "String", message: nil, preferredStyle: .alert)
let okAlert = UIAlertAction(title: "String", style: .default)
failedMenu.addAction(okAlert)
present(failedMenu, animated: true)
}
Run Code Online (Sandbox Code Playgroud)
我正在尝试创建一个类来处理初始化 MFMailComposeViewController 以在应用程序内发送电子邮件。
我在使其正常工作时遇到问题。好吧,而不是让它在不起作用时不会崩溃。
班级:
import UIKit
import MessageUI
struct Feedback {
let recipients = "String"
let subject: String
let body: String
}
class FeedbackManager: MFMailComposeViewController, MFMailComposeViewControllerDelegate {
func sendEmail(feedback: Feedback) {
if MFMailComposeViewController.canSendMail() {
self.mailComposeDelegate = self
self.setToRecipients([feedback.recipients])
self.setSubject("Feedback: \(feedback.subject)")
self.setMessageBody("<p>\(feedback.body)</p>", isHTML: true)
} else {
print("else:")
mailFailed()
}
}
func mailFailed() {
print("mailFailed():")
let failedMenu = UIAlertController(title: "Please Email Me!", message: nil, preferredStyle: .alert)
let okAlert = UIAlertAction(title: "Ok!", style: .default)
failedMenu.addAction(okAlert)
self.present(failedMenu, animated: true, completion: nil)
}
func mailComposeController(_ controller: MFMailComposeViewController, didFinishWith result: MFMailComposeResult, error: Error?) {
controller.dismiss(animated: true)
}
}
Run Code Online (Sandbox Code Playgroud)
然后从不同的视图控制器调用它:
let feedbackManager = FeedbackManager()
feedbackManager.sendEmail(feedback: Feedback(subject: "String", body: "String"))
self.present(feedbackManager, animated: true, completion: nil)
tableView.deselectRow(at: indexPath, animated: true)
Run Code Online (Sandbox Code Playgroud)
如果 MFMailComposeViewController.canSendMail() == true,上面的方法就可以正常工作。我面临的问题是,如果 canSendMail() 不正确,那么该类显然无法初始化并崩溃。这是有道理的。
错误:
Unable to initialize due to + [MFMailComposeViewController canSendMail] returns NO.
Run Code Online (Sandbox Code Playgroud)
我不知道从这里到哪里去如何让它工作。我尝试将 FeedbackManager 从 MFMailComposeViewController 更改为 UIViewController。这似乎有效,但因为它在堆栈上添加了一个视图,所以导致了奇怪的图形显示。
我可以做的另一件事是导入 MessageUI,并为我希望能够从中发送电子邮件的每个控制器遵守 MFMailComposeViewController 。这样我就可以在尝试初始化 FeedbackManager() 之前检查 canSendMail() 。但这似乎也不是最好的答案。
我还能怎样让它工作?
编辑:我已经得到了处理这个问题的代码,但是,在呈现 MFMailComposeViewController 之前将视图添加到堆栈上会出现一个丑陋的过渡。
class FeedbackManager: UIViewController, MFMailComposeViewControllerDelegate {
func sendEmail(feedback: Feedback, presentingViewController: UIViewController) -> UIViewController {
if MFMailComposeViewController.canSendMail() {
let mail = MFMailComposeViewController()
mail.mailComposeDelegate = self
mail.setToRecipients([feedback.recipients])
mail.setSubject("Feedback: \(feedback.subject)")
mail.setMessageBody("<p>\(feedback.body)</p>", isHTML: true)
present(mail, animated: true)
return self
} else {
print("else:")
return mailFailed(presentingViewController: presentingViewController)
}
}
func mailFailed(presentingViewController: UIViewController) -> UIViewController {
print("mailFailed():")
let failedMenu = UIAlertController(title: "Please Email Me!", message: nil, preferredStyle: .alert)
let okAlert = UIAlertAction(title: "Ok!", style: .default)
failedMenu.addAction(okAlert)
return failedMenu
}
func mailComposeController(_ controller: MFMailComposeViewController, didFinishWith result: MFMailComposeResult, error: Error?) {
controller.dismiss(animated: true)
self.dismiss(animated: false)
}
}
Run Code Online (Sandbox Code Playgroud)
子类化MFMailComposeViewController是错误的方法。此类旨在“按原样”使用。如果您愿意,您可以构建一个包装类:
struct Feedback {
let recipients = "String"
let subject: String
let body: String
}
class FeedbackManager: NSObject, MFMailComposeViewControllerDelegate {
private var feedback: Feedback
private var completion: ((Result<MFMailComposeResult,Error>)->Void)?
override init() {
fatalError("Use FeedbackManager(feedback:)")
}
init?(feedback: Feedback) {
guard MFMailComposeViewController.canSendMail() else {
return nil
}
self.feedback = feedback
}
func send(on viewController: UIViewController, completion:(@escaping(Result<MFMailComposeResult,Error>)->Void)) {
let mailVC = MFMailComposeViewController()
self.completion = completion
mailVC.mailComposeDelegate = self
mailVC.setToRecipients([feedback.recipients])
mailVC.setSubject("Feedback: \(feedback.subject)")
mailVC.setMessageBody("<p>\(feedback.body)</p>", isHTML: true)
viewController.present(mailVC, animated:true)
}
func mailComposeController(_ controller: MFMailComposeViewController, didFinishWith result: MFMailComposeResult, error: Error?) {
if let error = error {
completion?(.failure(error))
} else {
completion?(.success(result))
}
}
}
Run Code Online (Sandbox Code Playgroud)
然后从视图控制器使用它:
let feedback = Feedback(subject: "String", body: "Body")
if let feedbackMgr = FeedbackManager(feedback: feedback) {
self.feedbackManager = feedbackMgr
feedback.send(on: self) { [weak self], result in
switch result {
case .failure(let error):
// Do something with the error
case .success(let mailResult):
// Do something with the result
}
self.feedbackManager = nil
}
} else {
// Can't send email
}
Run Code Online (Sandbox Code Playgroud)
您需要在FeedbackManager属性中保留对 的强引用,否则一旦包含的函数退出,它将被释放。我上面的代码引用了一个属性
var feedbackManager: FeedbackManager?
Run Code Online (Sandbox Code Playgroud)
虽然这可行,但更好的用户体验是canSendMail直接检查并禁用/隐藏允许他们发送反馈的 UI 组件
| 归档时间: |
|
| 查看次数: |
1319 次 |
| 最近记录: |