我无法将图片显示为邮件消息的HTML正文的一部分,该邮件消息还包含日历附件(iCal)。
我想做的是: 发送Outlook可识别的日历邀请,还包含HTML正文(也显示在Outlook 2013中)。该HTML应该内嵌显示页眉和页脚图像。
我的代码在gmail中查看时有效,但它不会在Outlook中显示图像 - 而是显示" header.jpg"和" footer.jpg"在我想要实际显示图像的身体位置。
我已尝试对此代码库进行大量修改 - 忽略日历邀请部分(然后图像工作),或日历邀请部分工作,图像不会显示,但由&替换#34; header.jpg"和" screen.jpg"双击时打开的文件很好。我可以绕过这个并在Outlook中内嵌显示图像吗?
这是我正在使用的代码(剪掉不相关的东西):
var header = new LinkedResource("header.jpg", MediaTypeNames.Image.Jpeg);
header.ContentId = Guid.NewGuid().ToString();
header.TransferEncoding = TransferEncoding.Base64;
header.ContentType = new ContentType("image/jpg");
header.ContentType.Name = "header.jpg";
header.ContentLink = new Uri(string.Format("cid:{0}", header.ContentId));
var screen = new LinkedResource("screen.jpg", MediaTypeNames.Image.Jpeg);
screen.ContentId = Guid.NewGuid().ToString();
screen.TransferEncoding = TransferEncoding.Base64;
screen.ContentType = new ContentType("image/jpg");
screen.ContentType.Name = "screen.jpg";
screen.ContentLink = new Uri(string.Format("cid:{0}", screen.ContentId));
var sb = new System.Text.StringBuilder();
var dtStart = appointmentStart;
var dtEnd = appointmentEnd;
var htmlcontent = string.Format("<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0 Transitional//EN\">" +
"<HTML><HEAD><META http-equiv=Content-Type content=\"multipart/alternative; charset=iso-8859-1\"></HEAD>" +
"<BODY>" +
"<img src=\"cid:" + header.ContentId + "\"/>" +
"<div><strong>The bold works fine, but the header and footer don't work! /strong></div>" +
"<br />" +
"<div><img src=\"cid:" + screen.ContentId + "\"/></div>" +
"</BODY></HTML>"
);
var plaincontent = String.Format("snip snip");
var av1 = AlternateView.CreateAlternateViewFromString(htmlcontent, new ContentType("text/html"););
av1.LinkedResources.Add(header);
av1.LinkedResources.Add(screen);
sb.AppendLine("BEGIN:VCALENDAR");
sb.AppendLine("VERSION:2.0");
sb.AppendLine("METHOD:REQUEST");
sb.AppendLine("BEGIN:VEVENT");
sb.AppendLine(request.ContactEmail);
sb.AppendLine("CLASS:PUBLIC");
sb.AppendLine(string.Format("CREATED:{0:yyyyMMddTHHmmss}", DateTime.Now));
sb.AppendLine("DESCRIPTION:" + plaincontent);
sb.AppendLine("X-ALT-DESC;FMTTYPE=text/html:" + htmlcontent);
sb.AppendLine(string.Format("DTSTART:{0:yyyyMMddTHHmmssZ}", dtStart.ToUniversalTime()));
sb.AppendLine(string.Format("DTEND:{0:yyyyMMddTHHmmssZ}", dtEnd.ToUniversalTime()));
sb.AppendLine(string.Format("DTSTAMP:{0:yyyyMMddTHHmmssZ}", DateTime.Now.ToUniversalTime()));
sb.AppendLine("ORGANIZER;CN=\"TheOrganizer\":mailto:" + "TheOrganizer@foobar.com");
sb.AppendLine("SEQUENCE:0");
sb.AppendLine("UID:" + request.EmailNotificationUniqueId);
sb.AppendLine("LOCATION:" + request.Location + " : " + request.LocationInformation);
sb.AppendLine("SUMMARY;LANGUAGE=en-us:" + "removed");
sb.AppendLine("BEGIN:VALARM");
sb.AppendLine("TRIGGER:-PT1440M");
sb.AppendLine("ACTION:DISPLAY");
sb.AppendLine("DESCRIPTION:Reminder");
sb.AppendLine("END:VALARM");
sb.AppendLine("END:VEVENT");
sb.AppendLine("END:VCALENDAR");
var icsView = AlternateView.CreateAlternateViewFromString(sb.ToString(), new ContentType("text/calendar"));
var message = new MailMessage();
message.AlternateViews.Add(body);
message.AlternateViews.Add(icsView);
return message;
有什么根本错误吗?这看起来应该非常简单明了。我已经用普通的电子邮件做了很多次这样的事情,但似乎Outlook并不喜欢HTML备用视图正文中的图像附件.ics附件。 HTML正文工作正常( 显示为粗体)但是嵌入的图像只是不会显示。
我尝试使用base64手动编码图像,但根本没有帮助。我在htmlContent var中尝试使用和不使用HTML标头标记,但这也没有任何区别。
这不应该是这个困难 - 我错过了什么?
编辑:这是谷歌中的消息来源。这在谷歌浏览器窗口中工作正常(显示正常。)不幸的是在Outlook中消息不起作用...也许交换服务器正在抑制内部消息的图像?这甚至是一件事吗?
Delivered-To: snip@gmail.com
Received: by 10.182.167.74 with SMTP id zm10csp2847908obb;
Fri, 7 Oct 2016 15:42:14 -0700 (PDT)
X-Received: by 10.37.171.105 with SMTP id u96mr17483671ybi.63.1475880134797;
Fri, 07 Oct 2016 15:42:14 -0700 (PDT)
Return-Path: <snip@snip.com>
Received: from msg12.snip.com (msg12.snip.com. [192.195.66.28])
by mx.google.com with ESMTPS id q11si2985157ywc.340.2016.10.07.15.42.14
for <snip@gmail.com>
(version=TLS1_2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128);
Fri, 07 Oct 2016 15:42:14 -0700 (PDT)
Received-SPF: pass (google.com: domain of snip@snip.com designates 192.195.66.28 as permitted sender) client-ip=192.195.66.28;
Authentication-Results: mx.google.com;
spf=pass (google.com: domain of snip@snip.com designates 192.195.66.28 as permitted sender) smtp.mailfrom=snip@snip.com
Received: from int11.snip.pvt (int11.snip.pvt [153.6.62.222]) by msg12.snip.com (Sentrion-MTA-4.3.1/Sentrion-MTA-4.2.2) with ESMTP id u97MgC5a022486 (version=TLSv1.2 cipher=ECDHE-RSA-AES256-GCM-SHA384 bits=256 verify=FAIL) for <snip@gmail.com>; Fri, 7 Oct 2016 22:42:12 GMT
Received: from snip.com (snip.com [snip]) by int11.snip.pvt (Sentrion-MTA-4.3.1/Sentrion-MTA-4.2.2) with ESMTP id u97MgBAv016400 for <snip@gmail.com>; Fri, 7 Oct 2016 22:42:12 GMT
Received: from snip.com with Microsoft SMTPSVC(7.5.7601.17514);
Fri, 7 Oct 2016 18:42:12 -0400
MIME-Version: 1.0
From: snip <snip@snip.com>
To: snip@gmail.com
Date: 7 Oct 2016 18:42:12 -0400
Content-Type: multipart/alternative; boundary=--boundary_0_8278962b-71cf-4ca1-9a64-5fb9629f2042
Message-ID: <5U9T3T00000050@snip.com>
X-OriginalArrivalTime: 07 Oct 2016 22:42:12.0895 (UTC) FILETIME=[0BC44AF0:01D220EC]
X-Flow-Control: Sendmail Flow Controller v2.2.5 int11.snip.pvt u97MgBAv016400
X-Flow-Control-Info: class=Default rcpts=1 size=37636
----boundary_0_8278962b-71cf-4ca1-9a64-5fb9629f2042
Content-Type: multipart/related; boundary=--boundary_1_306bb8e7-fb92-4eab-bf5c-b7393cf499ac; type="text/html"
----boundary_1_306bb8e7-fb92-4eab-bf5c-b7393cf499ac
Content-Type: text/html
Content-Transfer-Encoding: quoted-printable
<img src=3D"cid:443ba735-6376-45a1-a5af-6c679321baa2"/><div><strong>The bol=
d works fine, but the header and footer don't work!</strong></div><br /><di=
v><img src=3D"cid:145c43b0-da9e-40c4-b149-3a149fbc4503"/></div>
----boundary_1_306bb8e7-fb92-4eab-bf5c-b7393cf499ac
Content-Type: image/jpeg
Content-Transfer-Encoding: base64
Content-ID: <443ba735-6376-45a1-a5af-6c679321baa2>
----boundary_1_306bb8e7-fb92-4eab-bf5c-b7393cf499ac
Content-Type: image/jpeg
Content-Transfer-Encoding: base64
Content-ID: <145c43b0-da9e-40c4-b149-3a149fbc4503>
----boundary_1_306bb8e7-fb92-4eab-bf5c-b7393cf499ac--
----boundary_0_8278962b-71cf-4ca1-9a64-5fb9629f2042
Content-Type: text/calendar
Content-Transfer-Encoding: base64
QkVHSU46VkNBTEVOREFSDQpWRVJTSU9OOjIuMA0KTUVUSE9EOlJFUVVFU1QNCkJFR0lOOlZFVkVO
VA0KSmRyYWhsQGdtYWlsLmNvbQ0KQ0xBU1M6UFVCTElDDQpDUkVBVEVEOjIwMTYxMDA3VDE4NDIx
MQ0KREVTQ1JJUFRJT046DQpYLUFMVC1ERVNDO0ZNVFRZUEU9dGV4dC9odG1sOjxpbWcgc3JjPSJj
aWQ6NDQzYmE3MzUtNjM3Ni00NWExLWE1YWYtNmM2NzkzMjFiYWEyIi8+PGRpdj48c3Ryb25nPlRo
ZSBib2xkIHdvcmtzIGZpbmUsIGJ1dCB0aGUgaGVhZGVyIGFuZCBmb290ZXIgZG9uJ3Qgd29yayE8
L3N0cm9uZz48L2Rpdj48YnIgLz48ZGl2PjxpbWcgc3JjPSJjaWQ6MTQ1YzQzYjAtZGE5ZS00MGM0
LWIxNDktM2ExNDlmYmM0NTAzIi8+PC9kaXY+DQpEVFNUQVJUOjIwMTYxMDI0VDE0MDAwMFoNCkRU
RU5EOjIwMTYxMDI0VDE4MDAwMFoNCkRUU1RBTVA6MjAxNjEwMDdUMjI0MjExWg0KT1JHQU5JWkVS
O0NOPSJEaXNuZXkgUmVmcmVzaCBUZWFtIjptYWlsdG86RGlzbmV5LlJlZnJlc2guVGVhbUBkaXNu
ZXkuY29tDQpTRVFVRU5DRTowDQpVSUQ6NGE5YjczNWQtOTQwYS00Mzk0LTkzODItMzMyMmQxNDg0
MTBiDQpMT0NBVElPTjpPcmxhbmRvIDogDQpTVU1NQVJZO0xBTkdVQUdFPWVuLXVzOkNvbXB1dGVy
IFJlZnJlc2ggQXBwb2ludG1lbnQNCkJFR0lOOlZBTEFSTQ0KVFJJR0dFUjotUFQxNDQwTQ0KQUNU
SU9OOkRJU1BMQVkNCkRFU0NSSVBUSU9OOlJlbWluZGVyDQpFTkQ6VkFMQVJNDQpFTkQ6VkVWRU5U
DQpFTkQ6VkNBTEVOREFSDQo=
----boundary_0_8278962b-71cf-4ca1-9a64-5fb9629f2042--
答案 0 :(得分:0)
尝试使用以下代码,其中包含附加内嵌图片的方法以及电子邮件邀请和普通附件,只需根据需要调整代码
private String senderAddress = "YOUR EMAIL ADDRESS";
private final String password = "YOUR EMAIL PASSWORD";
public static SimpleDateFormat dateFormat = new SimpleDateFormat("yyyyMMdd'T'HHmm'00'");
public static SimpleDateFormat dateParser = new SimpleDateFormat("dd-MM-yyyy HH:mm");
public static SimpleDateFormat dateFormater = new SimpleDateFormat("dd-MM-yyyy");
public String getSenderAddress() {
return senderAddress;
}
public String getPassword() {
return password;
}
public Session getMailSession() {
Properties props = new Properties();
props.put("mail.smtp.host", "smtp.gmail.com");
props.put("mail.smtp.port", "587");
props.put("mail.smtp.auth", "true");
props.put("mail.smtp.starttls.enable", "true");
return Session.getInstance(props, new javax.mail.Authenticator() {
protected PasswordAuthentication getPasswordAuthentication() {
return new PasswordAuthentication(getSenderAddress(), getPassword());
}
});
}
public String newSend(Invitation invitation, String methodType) {
String UID = null;
// register the text/calendar mime type
MimetypesFileTypeMap mimetypes = (MimetypesFileTypeMap) MimetypesFileTypeMap.getDefaultFileTypeMap();
mimetypes.addMimeTypes("text/calendar ics ICS");
// register the handling of text/calendar mime type
MailcapCommandMap mailcap = (MailcapCommandMap) MailcapCommandMap.getDefaultCommandMap();
mailcap.addMailcap("text/calendar;; x-java-content-handler=com.sun.mail.handlers.text_plain");
MimeMessage message = new MimeMessage(getMailSession());
try {
message.setFrom(new InternetAddress(senderAddress));
message.setSubject(invitation.getSubject());
message.addRecipient(Message.RecipientType.TO, new InternetAddress(invitation.getTo()));
Multipart multipart = new MimeMultipart("mixed");
invitation.setDescription(getHtmlWithEncodedImages(multipart,invitation));
Multipart mpMixedAlternative = newChild(multipart, "alternative");
// html part
BodyPart messageBodyPart = buildHtmlTextPart(invitation);
mpMixedAlternative.addBodyPart(messageBodyPart);
// Add part two, the calendar
BodyPart calendarPart = buildCalendarPart(invitation, methodType);
calendarPart.setDisposition(BodyPart.INLINE);
mpMixedAlternative.addBodyPart(calendarPart);
addAttachment(multipart);
// Put the multipart in message
message.setContent(multipart);
// send the message
Transport transport = getMailSession().getTransport("smtp");
transport.connect();
transport.sendMessage(message, message.getAllRecipients());
transport.close();
UID = invitation.getTo() + "" + invitation.getStartdate().toString() + "" + invitation.getStarttime();
} catch (Exception e) {
//TODO logger log exception
}
return UID;
}
private void addAttachment(Multipart multipart) {
MimeBodyPart mbpAttachment = new MimeBodyPart();
File file=new File("C:/users/ishan.juneja/Desktop/temp.txt");
DataSource datasource=new FileDataSource(file);
try {
mbpAttachment.setDataHandler(new DataHandler(datasource));
mbpAttachment.setDisposition(BodyPart.ATTACHMENT);
mbpAttachment.setFileName("temp.txt");
multipart.addBodyPart(mbpAttachment);
} catch (MessagingException e) {
// TODO logger log exception
}
}
public String uploadImage(MultipartFile file){
byte[] bytes;
try {
String path = System.getProperty("catalina.home");
File dir = new File(path+"/tmpFiles");
if(!dir.isDirectory()){
dir.mkdir();
}
bytes = file.getBytes();
File serverfile = new File(dir.getAbsolutePath()+"/"+file.getName()+Math.random()+".png");
BufferedOutputStream stream = new BufferedOutputStream(new FileOutputStream(serverfile));
stream.write(bytes);
stream.close();
System.out.println("file uploaded "+serverfile.getAbsolutePath());
String hostname="Unknown";
try {
hostname = InetAddress.getLocalHost().getHostName();
} catch (UnknownHostException e1) {
e1.printStackTrace();
}
return "{\"location\" : \"http://"+hostname+":8080/emailinvitation/image/path/"+serverfile.getName()+"\"}";
} catch (IOException e) {
}
return "{\"location\":\"http://localhost:8080/emailinvitation/image/path/unknown.png\"}";
}
private Multipart newChild(Multipart parent, String alternative) {
MimeMultipart child = new MimeMultipart(alternative);
MimeBodyPart mbp = new MimeBodyPart();
try {
parent.addBodyPart(mbp);
mbp.setContent(child);
} catch (MessagingException e) {
// TODO logger log exception
}
return child;
}
private BodyPart buildCalendarPart(Invitation invitation, String methodType) throws Exception {
Date date1 = dateParser.parse(dateFormater.format(invitation.getStartdate()) + " " + invitation.getStarttime());
String startDate = timeZoneToUTC(invitation, date1);
Date date2 = dateParser.parse(dateFormater.format(invitation.getEnddate()) + " " + invitation.getEndtime());
String endDate = timeZoneToUTC(invitation, date2);
BodyPart calendarPart = new MimeBodyPart();
String calendarContent = "BEGIN:VCALENDAR\n" + "METHOD:" + methodType + "\n" + "PRODID: BCP - Meeting\n"
+ "VERSION:2.0\n" + "X-WR-TIMEZONE:" + invitation.getTimezone() + "\n" + "BEGIN:VEVENT\n" + "DTSTAMP:"
+ startDate + "Z\n" + "DTSTART:" + startDate + "Z\n" + "DTEND:" + endDate + "Z\n"
+ "SUMMARY:test request\n" + "UID:" + invitation.getTo() + "" + startDate + "\n"
+ "ATTENDEE;ROLE=REQ-PARTICIPANT;PARTSTAT=NEEDS-ACTION;RSVP=FALSE:MAILTO:" + invitation.getTo() + "\n"
+ "ORGANIZER:MAILTO:" + senderAddress + "\n" + "LOCATION:Test\n" + "SEQUENCE:0\n"
+ "PRIORITY:5\n" + "CLASS:PUBLIC\n" + "STATUS:CONFIRMED\n" + "TRANSP:OPAQUE\n" + "BEGIN:VALARM\n"
+ "ACTION:DISPLAY\n" + "DESCRIPTION:REMINDER\n" + "TRIGGER;RELATED=START:-PT00H15M00S\n"
+ "END:VALARM\n" + "END:VEVENT\n" + "END:VCALENDAR";
calendarPart.addHeader("Content-Class", "urn:content-classes:calendarmessage");
calendarPart.setContent(calendarContent, "text/calendar;method=" + methodType);
return calendarPart;
}
private BodyPart buildHtmlTextPart(Invitation invitation) {
MimeBodyPart descriptionPart = new MimeBodyPart();
String content = invitation.getDescription();
try {
descriptionPart.setContent(content, "text/html; charset=utf-8");
} catch (MessagingException e) {
// TODO logger log exception
}
return descriptionPart;
}
private void addImages(Multipart parent,String filename) {
MimeBodyPart mbpAttachment = new MimeBodyPart();
try {
File file = new File("C:\\apache-tomcat-8.5.24\\tmpFiles\\"+filename);
FileDataSource ds = new FileDataSource(file);
mbpAttachment.setDataHandler(new DataHandler(ds));
mbpAttachment.setDisposition(BodyPart.INLINE);
mbpAttachment.setHeader("Content-ID", filename);
mbpAttachment.setFileName(filename);
parent.addBodyPart(mbpAttachment);
} catch (Exception e) {
//TODO logger log exception
}
}
private String timeZoneToUTC(Invitation invitation, Date date) {
ZoneId zone = ZoneId.of(invitation.getTimezone());
String date1 = dateParser.format(date);
DateTimeFormatter fmt = DateTimeFormatter.ofPattern("dd-MM-yyyy HH:mm").withZone(zone);
ZonedDateTime utc = ZonedDateTime.parse(date1, fmt).withZoneSameInstant(ZoneId.of("UTC"));
Timestamp sqlTs = Timestamp.valueOf(utc.toLocalDateTime());
return dateFormat.format(sqlTs);
}
public String getHtmlWithEncodedImages(Multipart parent,Invitation invitation) {
String temp = invitation.getDescription();
String html="<html><body>"+invitation.getDescription()+"</body></html>";
Document document = Jsoup.parse(html);
Elements allElements=document.body().getElementsByTag("img");
for (Element element : allElements) {
String elementstring=element.toString();
String tempfilename = elementstring.substring(elementstring.indexOf("file"));
String filename=tempfilename.substring(0,tempfilename.indexOf('"'));
temp=temp.replace(elementstring.substring(elementstring.indexOf('"'), elementstring.lastIndexOf('"')),"\"cid:"+filename);
addImages(parent, filename);
}
System.out.println("--++--"+temp);
return temp;
}