|
5 | 5 | var { ExtensionCommon } = ChromeUtils.import("resource://gre/modules/ExtensionCommon.jsm"); |
6 | 6 | var { Services } = ChromeUtils.import("resource://gre/modules/Services.jsm"); |
7 | 7 | var { cal } = ChromeUtils.import("resource:///modules/calendar/calUtils.jsm"); |
| 8 | +var { jsmime } = ChromeUtils.import("resource:///modules/jsmime.jsm"); |
8 | 9 |
|
9 | 10 | var { ExtensionAPI, EventManager } = ExtensionCommon; |
10 | 11 |
|
11 | 12 | class ExtCalendarProvider extends cal.provider.BaseClass { |
12 | | - QueryInterface = ChromeUtils.generateQI(["calICalendar", "calIChangeLog"]); |
| 13 | + QueryInterface = ChromeUtils.generateQI(["calICalendar", "calIChangeLog", "calISchedulingSupport", "calIItipTransport"]); |
13 | 14 |
|
14 | 15 | static register(extension) { |
15 | 16 | let calmgr = cal.getCalendarManager(); |
@@ -103,11 +104,49 @@ class ExtCalendarProvider extends cal.provider.BaseClass { |
103 | 104 | : ["unsubscribe"]; |
104 | 105 | case "requiresNetwork": |
105 | 106 | return !(this.capabilities.requires_network === false); |
| 107 | + case "itip.transport": |
| 108 | + if (this.extension.emitter.has("calendar.provider.onSend")) { |
| 109 | + return this; |
| 110 | + } |
| 111 | + break; |
106 | 112 | } |
107 | 113 |
|
108 | 114 | return super.getProperty(name); |
109 | 115 | } |
110 | 116 |
|
| 117 | + scheme = "mailto"; |
| 118 | + |
| 119 | + async sendItems(aRecipients, aItipItem) { |
| 120 | + let method = aItipItem.responseMethod; |
| 121 | + let transport = super.getProperty("itip.transport").wrappedJSObject; |
| 122 | + let { subject, body } = transport._prepareItems(aItipItem); |
| 123 | + let serializer = Cc["@mozilla.org/calendar/ics-serializer;1"].createInstance(Ci.calIIcsSerializer); |
| 124 | + let itemList = aItipItem.getItemList(); |
| 125 | + serializer.addItems(itemList, itemList.length); |
| 126 | + let methodProp = cal.getIcsService().createIcalProperty("METHOD"); |
| 127 | + methodProp.value = method; |
| 128 | + serializer.addProperty(methodProp); |
| 129 | + let icsText = serializer.serializeToString(); |
| 130 | + let boundary = Array.from({ length: 24 }, () => Math.floor(Math.random() * 16).toString(16)).join(""); |
| 131 | + let headers = new Map(); |
| 132 | + headers.set("Date", [new Date()]); |
| 133 | + headers.set("Subject", [subject]); |
| 134 | + headers.set("To", aRecipients.map(attendee => ({ name: attendee.commonName, email: attendee.id.replace(/^mailto:/, "")}))); |
| 135 | + headers.set("Content-Type", ["multipart/mixed; boundary=\"" + boundary + "\""]); |
| 136 | + let mimeContent = jsmime.headeremitter.emitStructuredHeaders(headers, { hardMargin: 800 }) + "\r\n"; |
| 137 | + mimeContent += "--" + boundary + "\r\n"; |
| 138 | + mimeContent += "Content-Type: text/plain; charset=UTF-8\r\n\r\n"; |
| 139 | + mimeContent += body + "\r\n"; |
| 140 | + mimeContent += "--" + boundary + "\r\n"; |
| 141 | + mimeContent += "Content-Type: text/calendar; method=" + aItipItem.responseMethod + "; charset=UTF-8\r\nContent-Transfer-Encoding: 8BIT\r\n\r\n"; |
| 142 | + mimeContent += icsText; |
| 143 | + mimeContent += "--" + boundary + "\r\n"; |
| 144 | + mimeContent += "Content-Type: application/ics; name=\"invite.ics\"\r\nContent-Disposition: attachment; filename=\"invite.ics\"\r\nContent-Transfer-Encoding: 8BIT\r\n\r\n"; |
| 145 | + mimeContent += icsText; |
| 146 | + mimeContent += "--" + boundary + "--\r\n"; |
| 147 | + await this.extension.emit("calendar.provider.onSend", this, mimeContent, aRecipients.map(attendee => attendee.id.replace(/^mailto:/, ""))); |
| 148 | + } |
| 149 | + |
111 | 150 | addItem(aItem, aListener) { |
112 | 151 | return this.adoptItem(aItem.clone(), aListener); |
113 | 152 | } |
@@ -408,6 +447,21 @@ this.calendar_provider = class extends ExtensionAPI { |
408 | 447 | }; |
409 | 448 | }, |
410 | 449 | }).api(), |
| 450 | + |
| 451 | + onSend: new EventManager({ |
| 452 | + context, |
| 453 | + name: "calendar.provider.onSend", |
| 454 | + register: fire => { |
| 455 | + let listener = (event, calendar, content) => { |
| 456 | + return fire.async(convertCalendar(context.extension, calendar), content); |
| 457 | + }; |
| 458 | + |
| 459 | + context.extension.on("calendar.provider.onSend", listener); |
| 460 | + return () => { |
| 461 | + context.extension.off("calendar.provider.onSend", listener); |
| 462 | + }; |
| 463 | + }, |
| 464 | + }).api(), |
411 | 465 | }, |
412 | 466 | }, |
413 | 467 | }; |
|
0 commit comments