diff --git a/src/utils/exportUtils/Exporter.ts b/src/utils/exportUtils/Exporter.ts index c782cc5a63..5e2de1faf1 100644 --- a/src/utils/exportUtils/Exporter.ts +++ b/src/utils/exportUtils/Exporter.ts @@ -1,7 +1,36 @@ import { MatrixEvent } from "matrix-js-sdk/src/models/event"; import { Room } from "matrix-js-sdk/src/models/room"; +import { MatrixClientPeg } from "../../MatrixClientPeg"; +import { TimelineWindow } from "matrix-js-sdk/src/timeline-window"; +import { arrayFastClone } from "../arrays"; export abstract class Exporter { - constructor(protected res: MatrixEvent[], protected room: Room) {} + constructor(protected room: Room) {} + + protected getTimelineConversation = () : MatrixEvent => { + if (!this.room) return; + + const cli = MatrixClientPeg.get(); + + const timelineSet = this.room.getUnfilteredTimelineSet(); + + const timelineWindow = new TimelineWindow( + cli, timelineSet, + {windowLimit: Number.MAX_VALUE}); + + timelineWindow.load(null, 30); + + const events = timelineWindow.getEvents(); + + // Clone and reverse the events so that we preserve the order + arrayFastClone(events) + .reverse() + .forEach(event => { + cli.decryptEventIfNeeded(event); + }); + + return events; + }; + abstract export(): Promise; } diff --git a/src/utils/exportUtils/HtmlExport.tsx b/src/utils/exportUtils/HtmlExport.tsx index 1432b61e85..c3174285e5 100644 --- a/src/utils/exportUtils/HtmlExport.tsx +++ b/src/utils/exportUtils/HtmlExport.tsx @@ -11,25 +11,25 @@ import { Layout } from "../../settings/Layout"; import { shouldFormContinuation } from "../../components/structures/MessagePanel"; import { formatFullDateNoDay, formatFullDateNoDayNoTime, wantsDateSeparator } from "../../DateUtils"; import { RoomPermalinkCreator } from "../permalinks/Permalinks"; +import { _t } from "../../languageHandler"; +import { MatrixClientPeg } from "../../MatrixClientPeg"; +import { EventType } from "matrix-js-sdk/src/@types/event"; import * as ponyfill from "web-streams-polyfill/ponyfill" import * as Avatar from "../../Avatar"; import EventTile, { haveTileForEvent } from "../../components/views/rooms/EventTile"; import DateSeparator from "../../components/views/messages/DateSeparator"; +import BaseAvatar from "../../components/views/avatars/BaseAvatar"; import exportCSS from "./exportCSS"; import exportJS from "./exportJS"; -import BaseAvatar from "../../components/views/avatars/BaseAvatar"; import exportIcons from "./exportIcons"; -import { _t } from "../../languageHandler"; -import { MatrixClientPeg } from "../../MatrixClientPeg"; -import { EventType } from "matrix-js-sdk/src/@types/event"; export default class HTMLExporter extends Exporter { protected zip: JSZip; protected avatars: Map; protected permalinkCreator: RoomPermalinkCreator; - constructor(res: MatrixEvent[], room: Room) { - super(res, room); + constructor(room: Room) { + super(room); this.zip = new JSZip(); this.avatars = new Map(); this.permalinkCreator = new RoomPermalinkCreator(this.room); @@ -57,16 +57,16 @@ export default class HTMLExporter extends Exporter { return renderToStaticMarkup(avatar); } - protected async wrapHTML(content: string, room: Room) { + protected async wrapHTML(content: string) { const roomAvatar32 = await this.getRoomAvatar(32); const exportDate = formatFullDateNoDayNoTime(new Date()); const cli = MatrixClientPeg.get(); - const creator = room.currentState.getStateEvents(EventType.RoomCreate, "")?.getSender(); - const creatorName = room?.getMember(creator)?.rawDisplayName || creator; + const creator = this.room.currentState.getStateEvents(EventType.RoomCreate, "")?.getSender(); + const creatorName = this.room?.getMember(creator)?.rawDisplayName || creator; const exporter = cli.getUserId(); - const exporterName = room?.getMember(exporter)?.rawDisplayName; - const topic = room.currentState.getStateEvents(EventType.RoomTopic, "")?.getContent()?.topic - || room.topic || ""; + const exporterName = this.room?.getMember(exporter)?.rawDisplayName; + const topic = this.room.currentState.getStateEvents(EventType.RoomTopic, "")?.getContent()?.topic + || this.room.topic || ""; const createdText = _t("%(creatorName)s created this room.", { creatorName, }); @@ -74,7 +74,7 @@ export default class HTMLExporter extends Exporter { const exportedText = _t(`This is the start of export of %(roomName)s. Exported by %(exporterDetails)s at %(exportDate)s. `, { exportDate, - roomName: room.name, + roomName: this.room.name, exporterDetails: ` ${exporterName ? `${ exporterName }(${ exporter })` : `${ exporter }`} `, @@ -115,9 +115,9 @@ export default class HTMLExporter extends Exporter {
- ${room.name} + ${this.room.name}
${topic}
@@ -144,7 +144,7 @@ export default class HTMLExporter extends Exporter { >
${roomAvatar52} -

${room.name}

+

${this.room.name}

${createdText}

${exportedText}

${topicText}

@@ -294,7 +294,7 @@ export default class HTMLExporter extends Exporter { return renderToStaticMarkup(eventTile); } - protected async createHTML(events: MatrixEvent[], room: Room) { + protected async createHTML(events: MatrixEvent[]) { let content = ""; let prevEvent = null; for (const event of events) { @@ -308,17 +308,21 @@ export default class HTMLExporter extends Exporter { content += body; prevEvent = event; } - return await this.wrapHTML(content, room); + return await this.wrapHTML(content); } public async export() { - const html = await this.createHTML(this.res, this.room); + const res = this.getTimelineConversation(); + const html = await this.createHTML(res); + this.zip.file("index.html", html); this.zip.file("css/style.css", exportCSS); this.zip.file("js/script.js", exportJS); + for (const iconName in exportIcons) { this.zip.file(`icons/${iconName}`, exportIcons[iconName]); } + const filename = `matrix-export-${formatFullDateNoDay(new Date())}.zip`; //Generate the zip file asynchronously diff --git a/src/utils/exportUtils/exportUtils.ts b/src/utils/exportUtils/exportUtils.ts index 5e1c4fffbb..34edee0ac6 100644 --- a/src/utils/exportUtils/exportUtils.ts +++ b/src/utils/exportUtils/exportUtils.ts @@ -1,7 +1,4 @@ -import { MatrixClientPeg } from "../../MatrixClientPeg"; -import { arrayFastClone } from "../arrays"; -import { TimelineWindow } from "matrix-js-sdk/src/timeline-window"; -import Room from 'matrix-js-sdk/src/models/room'; +import { Room } from 'matrix-js-sdk/src/models/room'; import HTMLExporter from "./HtmlExport"; export enum exportFormats { @@ -14,37 +11,10 @@ export enum exportOptions { TIMELINE = "TIMELINE", } -const getTimelineConversation = (room: Room) => { - if (!room) return; - - const cli = MatrixClientPeg.get(); - - const timelineSet = room.getUnfilteredTimelineSet(); - - const timelineWindow = new TimelineWindow( - cli, timelineSet, - {windowLimit: Number.MAX_VALUE}); - - timelineWindow.load(null, 30); - - const events = timelineWindow.getEvents(); - - // Clone and reverse the events so that we preserve the order - arrayFastClone(events) - .reverse() - .forEach(event => { - cli.decryptEventIfNeeded(event); - }); - - return events; -}; - - const exportConversationalHistory = async (room: Room, format: string, options) => { - const res = getTimelineConversation(room); switch (format) { case exportFormats.HTML: - await new HTMLExporter(res, room).export(); + await new HTMLExporter(room).export(); break; case exportFormats.JSON: break;