Recently I was implementing the ability to send QR codes in the RSVP confirmation emails. While doing that I ran into a stumbling block on how to send attachments inline. Since there didn’t seem to be any good documentation on how to do this minus some StackOverflow posts, I thought I would share how I did it and the lessons I learned in regards to sending images as emails in WordPress.
The First Attempt
Initially, when implementing this feature, I tried to do a base64-encoded image inline with the email. However, that wouldn’t show up when the email was viewed in Gmail’s web interface. So, I kept that there and also attached the QR code at the bottom of the email as well. That worked but was pretty ugly as that meant every email with QR shortcodes had at least two QR codes in it. That would be pretty confusing for both attendees and people working events.
Inline Attachments to the Rescue
To fix this I went with inline attachments. The only problem with this and the reason I didn’t do it this way in the first place is that there was no good documentation on how to do this. Especially when you were sending out many different emails. Below is, in general, the solution I came up with which worked for my use-case. Hopefully, it can be useful to some other person as well.
In my case, the code, that sends the email is separated from the part of the code that prepares the email body for sending. To get around this, I used a global array in my plugin to temporarily store the images I wanted to inline attach. If there was one object that was just being passed around, I would have stored the information in there, but that is currently not the case for my code base. I secondly created a wrapper around the wp_mail function which we will use later to implement the inline attachments. Below is the starting of the wrapper:
function example_send_mail( $email, $subject, $body, $headers = '', $attachments = array() ) {
    wp_mail( $email, $subject, $body, $headers, $attachments );
}
I then went through my code and replaced all of the calls to wp_mail with this new wrapper function. One thing I made sure to do with the wrapper is to use the exact same parameter ordering and defaults as wp_mail, so it was simple to swap.
Once I had that in place, it was now time for me to get inline attachments working. This was broken up into two separate pieces of work:
- 
Email body generation
- 
Attaching the files to the email right before it sends
Generating Content
While generating content for the email I would check to see if an inline image was needed. If needed I would create the image and then add it to the global array with the following attributes:
- 
uid: A unique identifier that is used for inline attachments this needs to be referenced to inline attach the image
- 
name: The name of the attachment that the user will see
- 
file: The path to the file that needs to be attached
The code looked something like.
$uid                     = uniqid();
$name                 = 'Inline Image.png';
$file_path            = generate_and_save_image();
$inline_attachments[] = array(
    'uid'  => $uid,
    'name' => $name,
    'file' => $file_path,
);
We then use the $uid variable for the inline image in the body of the email.
$body .= '<img src="cid:' . $uid . '" />'
Basically, the email clients see the “cid” and try to find an attachment with that ID and then shows it inline.
Attaching the Images to the Email
Now that the content has been generated we need to go back to the wrapper function we wrote earlier and attach the images to the email.
To do this, we will want to add an action with a function to embed the image. It looks like.
add_action( 'phpmailer_init', function( &$phpmailer ) use( $inline_attachments ) {
    $phpmailer->SMTPKeepAlive=true;
    foreach ( $inline_attachments as $a ) {
        $phpmailer->AddEmbeddedImage( $a['file'], $a['uid'], $a['name'] );
    }
});
What this function does is adds an embedded image with the attributes we specified when we were creating the email content. We want to do this before the wp_mail function is called. After the wp_mail function is called, we want to clean-up the images that we generated as well. Here is roughly what I wrote.
foreach ( $inline_attachments as $a ) {
    if ( file_exists( $a['name'] ) ) {
        unlink( $a['name'] );
    }
}
This is what the finished wrapper function looks like.
/**
 * Example wrapper for sending email. This is used to allow for
 * inline attachments, etc...
 *
 * @param string $email The email address we want to send the email to.
 * @param string $subject The subject for the email.
 * @param string $body The body of the email.
 * @param array  $headers The headers for the email.
 * @param array  $attachments The attachments for the email.
 */
function example_send_mail( $email, $subject, $body, $headers = '', $attachments = array() ) {
    global $inline_attachments;
    add_action( 'phpmailer_init', function( &$phpmailer ) use( $inline_attachments ) {
        $phpmailer->SMTPKeepAlive=true;
        foreach ( $inline_attachments as $a ) {
            $phpmailer->AddEmbeddedImage( $a['file'], $a['uid'], $a['name'] );
        }
    });
    wp_mail( $email, $subject, $body, $headers, $attachments );
    foreach ( $inline_attachments as $a ) {
        if ( file_exists( $a['name'] ) ) {
            unlink( $a['name'] );
        }
    }
    $inline_attachments = array();
}
That is it. I hope this is helpful to others and provides an example of how you can embed an inline attachment with WordPress.
