matrix-react-sdk
Version:
SDK for matrix.org using React
131 lines (127 loc) • 22.7 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _logger = require("matrix-js-sdk/src/logger");
var _Exporter = _interopRequireDefault(require("./Exporter"));
var _languageHandler = require("../../languageHandler");
var _TextForEvent = require("../../TextForEvent");
var _EventTileFactory = require("../../events/EventTileFactory");
var _SettingsStore = _interopRequireDefault(require("../../settings/SettingsStore"));
var _DateUtils = require("../../DateUtils");
/*
Copyright 2024 New Vector Ltd.
Copyright 2021, 2022 The Matrix.org Foundation C.I.C.
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
class PlainTextExporter extends _Exporter.default {
constructor(room, exportType, exportOptions, setProgressText) {
super(room, exportType, exportOptions, setProgressText);
(0, _defineProperty2.default)(this, "totalSize", void 0);
(0, _defineProperty2.default)(this, "mediaOmitText", void 0);
(0, _defineProperty2.default)(this, "textForReplyEvent", content => {
const REPLY_REGEX = /> <(.*?)>(.*?)\n\n(.*)/s;
const REPLY_SOURCE_MAX_LENGTH = 32;
const match = REPLY_REGEX.exec(content.body);
// if the reply format is invalid, then return the body
if (!match) return content.body;
let rplSource;
const rplName = match[1];
const rplText = match[3];
rplSource = match[2].substring(1);
// Get the first non-blank line from the source.
const lines = rplSource.split("\n").filter(line => !/^\s*$/.test(line));
if (lines.length > 0) {
// Cut to a maximum length.
rplSource = lines[0].substring(0, REPLY_SOURCE_MAX_LENGTH);
// Ellipsis if needed.
if (lines[0].length > REPLY_SOURCE_MAX_LENGTH) {
rplSource = rplSource + "...";
}
// Wrap in formatting
rplSource = ` "${rplSource}"`;
} else {
// Don't show a source because we couldn't format one.
rplSource = "";
}
return `<${rplName}${rplSource}> ${rplText}`;
});
(0, _defineProperty2.default)(this, "plainTextForEvent", async mxEv => {
const senderDisplayName = mxEv.sender && mxEv.sender.name ? mxEv.sender.name : mxEv.getSender();
let mediaText = "";
if (this.isAttachment(mxEv)) {
if (this.exportOptions.attachmentsIncluded) {
try {
const blob = await this.getMediaBlob(mxEv);
if (this.totalSize + blob.size > this.exportOptions.maxSize) {
mediaText = ` (${this.mediaOmitText})`;
} else {
this.totalSize += blob.size;
const filePath = this.getFilePath(mxEv);
mediaText = " (" + (0, _languageHandler._t)("export_chat|file_attached") + ")";
this.addFile(filePath, blob);
if (this.totalSize == this.exportOptions.maxSize) {
this.exportOptions.attachmentsIncluded = false;
}
}
} catch (error) {
mediaText = " (" + (0, _languageHandler._t)("export_chat|error_fetching_file") + ")";
_logger.logger.log("Error fetching file " + error);
}
} else mediaText = ` (${this.mediaOmitText})`;
}
if (this.isReply(mxEv)) return senderDisplayName + ": " + this.textForReplyEvent(mxEv.getContent()) + mediaText;else return (0, _TextForEvent.textForEvent)(mxEv, this.room.client) + mediaText;
});
this.totalSize = 0;
this.mediaOmitText = !this.exportOptions.attachmentsIncluded ? (0, _languageHandler._t)("export_chat|media_omitted") : (0, _languageHandler._t)("export_chat|media_omitted_file_size");
}
get destinationFileName() {
return this.makeFileNameNoExtension() + ".txt";
}
async createOutput(events) {
let content = "";
for (let i = 0; i < events.length; i++) {
const event = events[i];
this.updateProgress((0, _languageHandler._t)("export_chat|processing_event_n", {
number: i + 1,
total: events.length
}), false, true);
if (this.cancelled) return this.cleanUp();
if (!(0, _EventTileFactory.haveRendererForEvent)(event, this.room.client, false)) continue;
const textForEvent = await this.plainTextForEvent(event);
content += textForEvent && `${(0, _DateUtils.formatFullDate)(new Date(event.getTs()), _SettingsStore.default.getValue("showTwelveHourTimestamps"))} - ${textForEvent}\n`;
}
return content;
}
async export() {
this.updateProgress((0, _languageHandler._t)("export_chat|starting_export"));
this.updateProgress((0, _languageHandler._t)("export_chat|fetching_events"));
const fetchStart = performance.now();
const res = await this.getRequiredEvents();
const fetchEnd = performance.now();
_logger.logger.log(`Fetched ${res.length} events in ${(fetchEnd - fetchStart) / 1000}s`);
this.updateProgress((0, _languageHandler._t)("export_chat|creating_output"));
const text = await this.createOutput(res);
if (this.files.length) {
this.addFile("export.txt", new Blob([text]));
await this.downloadZIP();
} else {
const fileName = this.destinationFileName;
this.downloadPlainText(fileName, text);
}
const exportEnd = performance.now();
if (this.cancelled) {
_logger.logger.info("Export cancelled successfully");
} else {
_logger.logger.info("Export successful!");
_logger.logger.log(`Exported ${res.length} events in ${(exportEnd - fetchStart) / 1000} seconds`);
}
this.cleanUp();
}
}
exports.default = PlainTextExporter;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_logger","require","_Exporter","_interopRequireDefault","_languageHandler","_TextForEvent","_EventTileFactory","_SettingsStore","_DateUtils","PlainTextExporter","Exporter","constructor","room","exportType","exportOptions","setProgressText","_defineProperty2","default","content","REPLY_REGEX","REPLY_SOURCE_MAX_LENGTH","match","exec","body","rplSource","rplName","rplText","substring","lines","split","filter","line","test","length","mxEv","senderDisplayName","sender","name","getSender","mediaText","isAttachment","attachmentsIncluded","blob","getMediaBlob","totalSize","size","maxSize","mediaOmitText","filePath","getFilePath","_t","addFile","error","logger","log","isReply","textForReplyEvent","getContent","textForEvent","client","destinationFileName","makeFileNameNoExtension","createOutput","events","i","event","updateProgress","number","total","cancelled","cleanUp","haveRendererForEvent","plainTextForEvent","formatFullDate","Date","getTs","SettingsStore","getValue","export","fetchStart","performance","now","res","getRequiredEvents","fetchEnd","text","files","Blob","downloadZIP","fileName","downloadPlainText","exportEnd","info","exports"],"sources":["../../../src/utils/exportUtils/PlainTextExport.ts"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2021, 2022 The Matrix.org Foundation C.I.C.\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport { Room, IContent, MatrixEvent } from \"matrix-js-sdk/src/matrix\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\nimport React from \"react\";\n\nimport Exporter from \"./Exporter\";\nimport { _t } from \"../../languageHandler\";\nimport { ExportType, IExportOptions } from \"./exportUtils\";\nimport { textForEvent } from \"../../TextForEvent\";\nimport { haveRendererForEvent } from \"../../events/EventTileFactory\";\nimport SettingsStore from \"../../settings/SettingsStore\";\nimport { formatFullDate } from \"../../DateUtils\";\n\nexport default class PlainTextExporter extends Exporter {\n    protected totalSize: number;\n    protected mediaOmitText: string;\n\n    public constructor(\n        room: Room,\n        exportType: ExportType,\n        exportOptions: IExportOptions,\n        setProgressText: React.Dispatch<React.SetStateAction<string>>,\n    ) {\n        super(room, exportType, exportOptions, setProgressText);\n        this.totalSize = 0;\n        this.mediaOmitText = !this.exportOptions.attachmentsIncluded\n            ? _t(\"export_chat|media_omitted\")\n            : _t(\"export_chat|media_omitted_file_size\");\n    }\n\n    public get destinationFileName(): string {\n        return this.makeFileNameNoExtension() + \".txt\";\n    }\n\n    public textForReplyEvent = (content: IContent): string => {\n        const REPLY_REGEX = /> <(.*?)>(.*?)\\n\\n(.*)/s;\n        const REPLY_SOURCE_MAX_LENGTH = 32;\n\n        const match = REPLY_REGEX.exec(content.body);\n\n        // if the reply format is invalid, then return the body\n        if (!match) return content.body;\n\n        let rplSource: string;\n        const rplName = match[1];\n        const rplText = match[3];\n\n        rplSource = match[2].substring(1);\n        // Get the first non-blank line from the source.\n        const lines = rplSource.split(\"\\n\").filter((line) => !/^\\s*$/.test(line));\n        if (lines.length > 0) {\n            // Cut to a maximum length.\n            rplSource = lines[0].substring(0, REPLY_SOURCE_MAX_LENGTH);\n            // Ellipsis if needed.\n            if (lines[0].length > REPLY_SOURCE_MAX_LENGTH) {\n                rplSource = rplSource + \"...\";\n            }\n            // Wrap in formatting\n            rplSource = ` \"${rplSource}\"`;\n        } else {\n            // Don't show a source because we couldn't format one.\n            rplSource = \"\";\n        }\n\n        return `<${rplName}${rplSource}> ${rplText}`;\n    };\n\n    protected plainTextForEvent = async (mxEv: MatrixEvent): Promise<string> => {\n        const senderDisplayName = mxEv.sender && mxEv.sender.name ? mxEv.sender.name : mxEv.getSender();\n        let mediaText = \"\";\n        if (this.isAttachment(mxEv)) {\n            if (this.exportOptions.attachmentsIncluded) {\n                try {\n                    const blob = await this.getMediaBlob(mxEv);\n                    if (this.totalSize + blob.size > this.exportOptions.maxSize) {\n                        mediaText = ` (${this.mediaOmitText})`;\n                    } else {\n                        this.totalSize += blob.size;\n                        const filePath = this.getFilePath(mxEv);\n                        mediaText = \" (\" + _t(\"export_chat|file_attached\") + \")\";\n                        this.addFile(filePath, blob);\n                        if (this.totalSize == this.exportOptions.maxSize) {\n                            this.exportOptions.attachmentsIncluded = false;\n                        }\n                    }\n                } catch (error) {\n                    mediaText = \" (\" + _t(\"export_chat|error_fetching_file\") + \")\";\n                    logger.log(\"Error fetching file \" + error);\n                }\n            } else mediaText = ` (${this.mediaOmitText})`;\n        }\n        if (this.isReply(mxEv)) return senderDisplayName + \": \" + this.textForReplyEvent(mxEv.getContent()) + mediaText;\n        else return textForEvent(mxEv, this.room.client) + mediaText;\n    };\n\n    protected async createOutput(events: MatrixEvent[]): Promise<string> {\n        let content = \"\";\n        for (let i = 0; i < events.length; i++) {\n            const event = events[i];\n            this.updateProgress(\n                _t(\"export_chat|processing_event_n\", {\n                    number: i + 1,\n                    total: events.length,\n                }),\n                false,\n                true,\n            );\n            if (this.cancelled) return this.cleanUp();\n            if (!haveRendererForEvent(event, this.room.client, false)) continue;\n            const textForEvent = await this.plainTextForEvent(event);\n            content +=\n                textForEvent &&\n                `${formatFullDate(\n                    new Date(event.getTs()),\n                    SettingsStore.getValue(\"showTwelveHourTimestamps\"),\n                )} - ${textForEvent}\\n`;\n        }\n        return content;\n    }\n\n    public async export(): Promise<void> {\n        this.updateProgress(_t(\"export_chat|starting_export\"));\n        this.updateProgress(_t(\"export_chat|fetching_events\"));\n\n        const fetchStart = performance.now();\n        const res = await this.getRequiredEvents();\n        const fetchEnd = performance.now();\n\n        logger.log(`Fetched ${res.length} events in ${(fetchEnd - fetchStart) / 1000}s`);\n\n        this.updateProgress(_t(\"export_chat|creating_output\"));\n        const text = await this.createOutput(res);\n\n        if (this.files.length) {\n            this.addFile(\"export.txt\", new Blob([text]));\n            await this.downloadZIP();\n        } else {\n            const fileName = this.destinationFileName;\n            this.downloadPlainText(fileName, text);\n        }\n\n        const exportEnd = performance.now();\n\n        if (this.cancelled) {\n            logger.info(\"Export cancelled successfully\");\n        } else {\n            logger.info(\"Export successful!\");\n            logger.log(`Exported ${res.length} events in ${(exportEnd - fetchStart) / 1000} seconds`);\n        }\n\n        this.cleanUp();\n    }\n}\n"],"mappings":";;;;;;;;AASA,IAAAA,OAAA,GAAAC,OAAA;AAGA,IAAAC,SAAA,GAAAC,sBAAA,CAAAF,OAAA;AACA,IAAAG,gBAAA,GAAAH,OAAA;AAEA,IAAAI,aAAA,GAAAJ,OAAA;AACA,IAAAK,iBAAA,GAAAL,OAAA;AACA,IAAAM,cAAA,GAAAJ,sBAAA,CAAAF,OAAA;AACA,IAAAO,UAAA,GAAAP,OAAA;AAlBA;AACA;AACA;AACA;AACA;AACA;AACA;;AAce,MAAMQ,iBAAiB,SAASC,iBAAQ,CAAC;EAI7CC,WAAWA,CACdC,IAAU,EACVC,UAAsB,EACtBC,aAA6B,EAC7BC,eAA6D,EAC/D;IACE,KAAK,CAACH,IAAI,EAAEC,UAAU,EAAEC,aAAa,EAAEC,eAAe,CAAC;IAAC,IAAAC,gBAAA,CAAAC,OAAA;IAAA,IAAAD,gBAAA,CAAAC,OAAA;IAAA,IAAAD,gBAAA,CAAAC,OAAA,6BAWhCC,OAAiB,IAAa;MACtD,MAAMC,WAAW,GAAG,yBAAyB;MAC7C,MAAMC,uBAAuB,GAAG,EAAE;MAElC,MAAMC,KAAK,GAAGF,WAAW,CAACG,IAAI,CAACJ,OAAO,CAACK,IAAI,CAAC;;MAE5C;MACA,IAAI,CAACF,KAAK,EAAE,OAAOH,OAAO,CAACK,IAAI;MAE/B,IAAIC,SAAiB;MACrB,MAAMC,OAAO,GAAGJ,KAAK,CAAC,CAAC,CAAC;MACxB,MAAMK,OAAO,GAAGL,KAAK,CAAC,CAAC,CAAC;MAExBG,SAAS,GAAGH,KAAK,CAAC,CAAC,CAAC,CAACM,SAAS,CAAC,CAAC,CAAC;MACjC;MACA,MAAMC,KAAK,GAAGJ,SAAS,CAACK,KAAK,CAAC,IAAI,CAAC,CAACC,MAAM,CAAEC,IAAI,IAAK,CAAC,OAAO,CAACC,IAAI,CAACD,IAAI,CAAC,CAAC;MACzE,IAAIH,KAAK,CAACK,MAAM,GAAG,CAAC,EAAE;QAClB;QACAT,SAAS,GAAGI,KAAK,CAAC,CAAC,CAAC,CAACD,SAAS,CAAC,CAAC,EAAEP,uBAAuB,CAAC;QAC1D;QACA,IAAIQ,KAAK,CAAC,CAAC,CAAC,CAACK,MAAM,GAAGb,uBAAuB,EAAE;UAC3CI,SAAS,GAAGA,SAAS,GAAG,KAAK;QACjC;QACA;QACAA,SAAS,GAAG,KAAKA,SAAS,GAAG;MACjC,CAAC,MAAM;QACH;QACAA,SAAS,GAAG,EAAE;MAClB;MAEA,OAAO,IAAIC,OAAO,GAAGD,SAAS,KAAKE,OAAO,EAAE;IAChD,CAAC;IAAA,IAAAV,gBAAA,CAAAC,OAAA,6BAE6B,MAAOiB,IAAiB,IAAsB;MACxE,MAAMC,iBAAiB,GAAGD,IAAI,CAACE,MAAM,IAAIF,IAAI,CAACE,MAAM,CAACC,IAAI,GAAGH,IAAI,CAACE,MAAM,CAACC,IAAI,GAAGH,IAAI,CAACI,SAAS,CAAC,CAAC;MAC/F,IAAIC,SAAS,GAAG,EAAE;MAClB,IAAI,IAAI,CAACC,YAAY,CAACN,IAAI,CAAC,EAAE;QACzB,IAAI,IAAI,CAACpB,aAAa,CAAC2B,mBAAmB,EAAE;UACxC,IAAI;YACA,MAAMC,IAAI,GAAG,MAAM,IAAI,CAACC,YAAY,CAACT,IAAI,CAAC;YAC1C,IAAI,IAAI,CAACU,SAAS,GAAGF,IAAI,CAACG,IAAI,GAAG,IAAI,CAAC/B,aAAa,CAACgC,OAAO,EAAE;cACzDP,SAAS,GAAG,KAAK,IAAI,CAACQ,aAAa,GAAG;YAC1C,CAAC,MAAM;cACH,IAAI,CAACH,SAAS,IAAIF,IAAI,CAACG,IAAI;cAC3B,MAAMG,QAAQ,GAAG,IAAI,CAACC,WAAW,CAACf,IAAI,CAAC;cACvCK,SAAS,GAAG,IAAI,GAAG,IAAAW,mBAAE,EAAC,2BAA2B,CAAC,GAAG,GAAG;cACxD,IAAI,CAACC,OAAO,CAACH,QAAQ,EAAEN,IAAI,CAAC;cAC5B,IAAI,IAAI,CAACE,SAAS,IAAI,IAAI,CAAC9B,aAAa,CAACgC,OAAO,EAAE;gBAC9C,IAAI,CAAChC,aAAa,CAAC2B,mBAAmB,GAAG,KAAK;cAClD;YACJ;UACJ,CAAC,CAAC,OAAOW,KAAK,EAAE;YACZb,SAAS,GAAG,IAAI,GAAG,IAAAW,mBAAE,EAAC,iCAAiC,CAAC,GAAG,GAAG;YAC9DG,cAAM,CAACC,GAAG,CAAC,sBAAsB,GAAGF,KAAK,CAAC;UAC9C;QACJ,CAAC,MAAMb,SAAS,GAAG,KAAK,IAAI,CAACQ,aAAa,GAAG;MACjD;MACA,IAAI,IAAI,CAACQ,OAAO,CAACrB,IAAI,CAAC,EAAE,OAAOC,iBAAiB,GAAG,IAAI,GAAG,IAAI,CAACqB,iBAAiB,CAACtB,IAAI,CAACuB,UAAU,CAAC,CAAC,CAAC,GAAGlB,SAAS,CAAC,KAC3G,OAAO,IAAAmB,0BAAY,EAACxB,IAAI,EAAE,IAAI,CAACtB,IAAI,CAAC+C,MAAM,CAAC,GAAGpB,SAAS;IAChE,CAAC;IArEG,IAAI,CAACK,SAAS,GAAG,CAAC;IAClB,IAAI,CAACG,aAAa,GAAG,CAAC,IAAI,CAACjC,aAAa,CAAC2B,mBAAmB,GACtD,IAAAS,mBAAE,EAAC,2BAA2B,CAAC,GAC/B,IAAAA,mBAAE,EAAC,qCAAqC,CAAC;EACnD;EAEA,IAAWU,mBAAmBA,CAAA,EAAW;IACrC,OAAO,IAAI,CAACC,uBAAuB,CAAC,CAAC,GAAG,MAAM;EAClD;EA+DA,MAAgBC,YAAYA,CAACC,MAAqB,EAAmB;IACjE,IAAI7C,OAAO,GAAG,EAAE;IAChB,KAAK,IAAI8C,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAGD,MAAM,CAAC9B,MAAM,EAAE+B,CAAC,EAAE,EAAE;MACpC,MAAMC,KAAK,GAAGF,MAAM,CAACC,CAAC,CAAC;MACvB,IAAI,CAACE,cAAc,CACf,IAAAhB,mBAAE,EAAC,gCAAgC,EAAE;QACjCiB,MAAM,EAAEH,CAAC,GAAG,CAAC;QACbI,KAAK,EAAEL,MAAM,CAAC9B;MAClB,CAAC,CAAC,EACF,KAAK,EACL,IACJ,CAAC;MACD,IAAI,IAAI,CAACoC,SAAS,EAAE,OAAO,IAAI,CAACC,OAAO,CAAC,CAAC;MACzC,IAAI,CAAC,IAAAC,sCAAoB,EAACN,KAAK,EAAE,IAAI,CAACrD,IAAI,CAAC+C,MAAM,EAAE,KAAK,CAAC,EAAE;MAC3D,MAAMD,YAAY,GAAG,MAAM,IAAI,CAACc,iBAAiB,CAACP,KAAK,CAAC;MACxD/C,OAAO,IACHwC,YAAY,IACZ,GAAG,IAAAe,yBAAc,EACb,IAAIC,IAAI,CAACT,KAAK,CAACU,KAAK,CAAC,CAAC,CAAC,EACvBC,sBAAa,CAACC,QAAQ,CAAC,0BAA0B,CACrD,CAAC,MAAMnB,YAAY,IAAI;IAC/B;IACA,OAAOxC,OAAO;EAClB;EAEA,MAAa4D,MAAMA,CAAA,EAAkB;IACjC,IAAI,CAACZ,cAAc,CAAC,IAAAhB,mBAAE,EAAC,6BAA6B,CAAC,CAAC;IACtD,IAAI,CAACgB,cAAc,CAAC,IAAAhB,mBAAE,EAAC,6BAA6B,CAAC,CAAC;IAEtD,MAAM6B,UAAU,GAAGC,WAAW,CAACC,GAAG,CAAC,CAAC;IACpC,MAAMC,GAAG,GAAG,MAAM,IAAI,CAACC,iBAAiB,CAAC,CAAC;IAC1C,MAAMC,QAAQ,GAAGJ,WAAW,CAACC,GAAG,CAAC,CAAC;IAElC5B,cAAM,CAACC,GAAG,CAAC,WAAW4B,GAAG,CAACjD,MAAM,cAAc,CAACmD,QAAQ,GAAGL,UAAU,IAAI,IAAI,GAAG,CAAC;IAEhF,IAAI,CAACb,cAAc,CAAC,IAAAhB,mBAAE,EAAC,6BAA6B,CAAC,CAAC;IACtD,MAAMmC,IAAI,GAAG,MAAM,IAAI,CAACvB,YAAY,CAACoB,GAAG,CAAC;IAEzC,IAAI,IAAI,CAACI,KAAK,CAACrD,MAAM,EAAE;MACnB,IAAI,CAACkB,OAAO,CAAC,YAAY,EAAE,IAAIoC,IAAI,CAAC,CAACF,IAAI,CAAC,CAAC,CAAC;MAC5C,MAAM,IAAI,CAACG,WAAW,CAAC,CAAC;IAC5B,CAAC,MAAM;MACH,MAAMC,QAAQ,GAAG,IAAI,CAAC7B,mBAAmB;MACzC,IAAI,CAAC8B,iBAAiB,CAACD,QAAQ,EAAEJ,IAAI,CAAC;IAC1C;IAEA,MAAMM,SAAS,GAAGX,WAAW,CAACC,GAAG,CAAC,CAAC;IAEnC,IAAI,IAAI,CAACZ,SAAS,EAAE;MAChBhB,cAAM,CAACuC,IAAI,CAAC,+BAA+B,CAAC;IAChD,CAAC,MAAM;MACHvC,cAAM,CAACuC,IAAI,CAAC,oBAAoB,CAAC;MACjCvC,cAAM,CAACC,GAAG,CAAC,YAAY4B,GAAG,CAACjD,MAAM,cAAc,CAAC0D,SAAS,GAAGZ,UAAU,IAAI,IAAI,UAAU,CAAC;IAC7F;IAEA,IAAI,CAACT,OAAO,CAAC,CAAC;EAClB;AACJ;AAACuB,OAAA,CAAA5E,OAAA,GAAAR,iBAAA","ignoreList":[]}