On iOS Unread has a Mail Content article action. This lets customers quickly compose an email with the HTML content of an article. It uses MFMailComposeViewController. MFMailComposeViewController has the unfortunate limitation of only working with Apple Mail, but it is otherwise great. The Mail Content article action is popular with my customers.
macOS has a Compose Email Sharing Service, making it easy for an app to ask the system to create an email message with a recipient, subject, and message body. I was excited to find that this service works with the user’s default email client. It is not limited to Apple Mail.
The Compose Email Sharing Service can accept either a plain text string, or an attributed string with formatting and embedded images.
But after spending more time working with the Compose Email Sharing Service, I was disappointed by some of its weaknesses.
Bugs Around Embedded Images
When sharing articles as rich text, I found that embedded images often get moved to the end of the article. I was able to generate a simple test case – an attributed string with an embedded image at the beginning of it. If I share that attributed string with the sharing service, Apple Mail will generate a message with the image moved to the end. I filed FB13668576 documenting this.
Plain Text Messages
If Apple Mail’s Message format setting (under the Composing pane of the Apple Mail settings window) is set to Plain Text, outgoing messages created using the sharing service are converted to plain text. All formatting and embedded images are lost. Arguably that is the correct behavior because it honors the user’s message format setting.
However a bigger problem is that there is no spacing between paragraphs. The resulting message for a multi-paragraph article is a wall of text. If Unread had a way to determine whether Apple Mail’s message format setting was plain text, Unread could add its own blank lines between paragraphs. But since I cannot find a way to get that information from a sandboxed Mac app, I can either share an attributed string with formatting and embedded images or I can share a plain text string with my own Markdown-like formatting. I cannot do both.
Third Party Email Apps
When the user’s default email client is something other than Apple Mail and an app sends an attributed string, the email client just gets the plain text. The plain text version of the string will also lack basic plain text formatting such as blank lines between paragraphs.
Scripting Apple Mail Directly
I considered trying to send an Apple Event to Apple Mail, hoping that I could simply tell Apple Mail to create an outbound email message with specific HTML content. However the sdef for Apple Mail states that setting the html content attribute of an outgoing message has no effect. The setting is deprecated. My testing confirms that setting that attribute has no effect.
Feedback Reports
I filed these feedback reports
FB13668659: .composeEmail NSSharingService - Multiple paragraphs, Plain Text Message Format
FB13669961: Formatting lost when sharing multi-paragraph attributed strings to third party email clients via the .composeEmail NSSharingService
FB13668576: When creating an email using the .composeEmail NSSharingService using an attributedString, images are moved to the bottom of the message
Related Performance Tip
I find using NSWorkspace.shared.open(URL) with a mailto: URL instead of the Compose Email Sharing Service to be about twice as fast for sending plain text email.
The Supreme Court must really be backlogged if they cannot hear the presidential immunity case until the end of April. President Biden should appoint a few additional Supreme Court justices to give them the capacity they need.
At this evening’s CocoaHeads Boston meeting on Zoom, Ed Arenberg will discuss his experiences developing on the Vision Pro and show his game “Connect3D - Spatial” running on-device. New folks and folks from outside the Boston area are always welcome.
I have a bunch of loose markdown files in a version-controlled folder that I consider my “company wiki”. Making it navigable with a web browser and letting documents link to each other would be a significant improvement.
CocoaHeads Boston will meet Thursday evening via Zoom at 7:00 PM Boston time. New participants and participants from outside the Boston area are always welcome.
I am now posting to my blog, and automatically cross-posting to Mastodon. The best way to follow or reach me via Mastodon (or Mastodon-compatible) service is @johnbrayton@iosdev.space. You can also follow my posts via my blog and its RSS feed.
I just had a new furnace and water heater installed. The old furnace was 29 years old, the old water heater was 15 years old. I am ready for winter. (Or at least my house is.)
I recently implemented a search suggestions menu on macOS for Unread.
Even though Safari, Mail, and Finder implement search suggestion menus, there is no standard AppKit mechanism for providing them in AppKit. Popping up an NSMenu does not work because the search field cannot receive keystrokes while the menu is displayed.
Interestingly there is a good mechanism for providing search suggestion menus in SwiftUI, but incorporating that mechanism into an AppKit-based application looks much more difficult than just using an NSHostingView.
I ended up starting with a CustomMenus sample project. This 2018 project by Doug Stein is a Swift port of Apple Objective-C sample code from 2012.
With that as a base, I updated it to support things like dark mode. I also made it a little more focused on search suggestion menus, as opposed to custom menus in general.
We have a great agenda for this evening’s CocoaHeads Boston meeting via Zoom. Conall McCabe will give a presentation pertaining to Core Motion, axes of device orientation, and associated data structures.
My glass shower door (which has probably been in place for 29 years) spontaneously shattered earlier today. No one was anywhere near it when this happened.
Tonight’s online CocoaHeads meeting will include a presentation on Skip, an Xcode plugin that transpiles native SwiftUI iOS apps into Android apps using Jetpack Compose and Kotlin. New participants and participants from outside the Boston area are always welcome.
I was also a precinct clerk, but we don’t get stickers for that. 🤣
We had a preliminary election for mayor, city council, and school committee in my city today. This is a big one. Our mayor of 20 years is retiring, and it’s the first city election since deciding to have ward-specific city councilors and school committee members.
Hilariously we (probably) have 3 more elections between now and the end of March.
I filed an FB a few days ago regarding the need for an API that does not exist on macOS. Apple replied with “We believe the reported problem has been fixed. Please verify that the issue does not reproduce on macOS 14 beta 7…” 🤦♂️