Mailspring/app/internal_packages/composer-templates/lib/template-store.ts
Ben Gotow 21bc2ef398
Upgrade to Slate 0.45.1 for better composer perf and "setStart on Range" error fix (#1518)
* Move to Slate 44, start using types more extensively in Composer

* More types and cleanup

* Expose the editor object to the draft session, try exposing editor to session

* Bump to Slate 45 for https://github.com/ianstormtaylor/slate/pull/2225/files

* How did the unsubscribe plugin get in here

* Bump Slate types, fix TS errors, start testing

* Polish

* Fix issue with emails not shrinking when you close quoted text

* Fix the “remove quoted text” button

* More polish

* Fix issues with PR, improve typings

* Remove spurious log
2019-06-10 19:15:07 -05:00

305 lines
8.6 KiB
TypeScript

/* eslint global-require: 0*/
import {
localized,
DraftStore,
Actions,
QuotedHTMLTransformer,
RegExpUtils,
} from 'mailspring-exports';
import { remote } from 'electron';
import MailspringStore from 'mailspring-store';
import path from 'path';
import fs from 'fs';
// Support accented characters in template names
// https://regex101.com/r/nD3eY8/1
const INVALID_TEMPLATE_NAME_REGEX = /[^a-zA-Z\u00C0-\u017F0-9_\- ]+/g;
class TemplateStore extends MailspringStore {
private _items = [];
private _templatesDir = path.join(AppEnv.getConfigDirPath(), 'templates');
private _watcher = null;
constructor() {
super();
this.listenTo(Actions.insertTemplateId, this._onInsertTemplateId);
this.listenTo(Actions.createTemplate, this._onCreateTemplate);
this.listenTo(Actions.showTemplates, this._onShowTemplates);
this.listenTo(Actions.deleteTemplate, this._onDeleteTemplate);
this.listenTo(Actions.renameTemplate, this._onRenameTemplate);
const welcomeName = 'Welcome to Templates.html';
const welcomePath = path.join(__dirname, '..', 'assets', welcomeName);
// I know this is a bit of pain but don't do anything that
// could possibly slow down app launch
fs.exists(this._templatesDir, exists => {
if (exists) {
this._populate();
this.watch();
} else {
fs.mkdir(this._templatesDir, () => {
fs.readFile(welcomePath, (err, welcome) => {
fs.writeFile(path.join(this._templatesDir, welcomeName), welcome, () => {
this.watch();
});
});
});
}
});
}
directory() {
return this._templatesDir;
}
watch() {
if (!this._watcher) {
try {
this._watcher = fs.watch(this._templatesDir, () => this._populate());
} catch (err) {
// usually an ENOSPC error
console.warn(err);
}
}
}
unwatch() {
if (this._watcher) {
this._watcher.close();
}
this._watcher = null;
}
items() {
return this._items;
}
_populate() {
fs.readdir(this._templatesDir, (err, filenames) => {
if (err) {
AppEnv.showErrorDialog({
title: localized('Cannot scan templates directory'),
message: localized(
'Mailspring was unable to read the contents of your templates directory (%@). You may want to delete this folder or ensure filesystem permissions are set correctly.',
this._templatesDir
),
});
return;
}
this._items = [];
for (let i = 0, filename; i < filenames.length; i++) {
filename = filenames[i];
if (filename[0] === '.') {
continue;
}
const displayname = path.basename(filename, path.extname(filename));
this._items.push({
id: filename,
name: displayname,
path: path.join(this._templatesDir, filename),
});
}
this.trigger(this);
});
}
_onCreateTemplate({
headerMessageId,
name,
contents,
}: { headerMessageId?: string; name?: string; contents?: string } = {}) {
if (headerMessageId) {
this._onCreateTemplateFromDraft(headerMessageId);
return;
}
if (!name || name.length === 0) {
this._displayError(localized('You must provide a name for your template.'));
}
if (!contents || contents.length === 0) {
this._displayError(localized('You must provide contents for your template.'));
}
this.saveNewTemplate(name, contents, this._onShowTemplates);
}
async _onCreateTemplateFromDraft(headerMessageId) {
const session = await DraftStore.sessionForClientId(headerMessageId);
const draft = session.draft();
const draftName = draft.subject.replace(INVALID_TEMPLATE_NAME_REGEX, '');
let draftContents = QuotedHTMLTransformer.removeQuotedHTML(draft.body);
const sigIndex = draftContents.search(RegExpUtils.mailspringSignatureRegex());
draftContents = sigIndex > -1 ? draftContents.substr(0, sigIndex) : draftContents;
if (!draftName || draftName.length === 0) {
this._displayError(localized('Give your draft a subject to name your template.'));
}
if (!draftContents || draftContents.length === 0) {
this._displayError(
localized('To create a template you need to fill the body of the current draft.')
);
}
this.saveNewTemplate(draftName, draftContents, this._onShowTemplates);
}
_onShowTemplates() {
Actions.switchPreferencesTab('Templates');
Actions.openPreferences();
}
_displayError(message) {
remote.dialog.showErrorBox(localized('Template Creation Error'), message);
}
_displayDialog(title, message, buttons) {
return (
remote.dialog.showMessageBox({
title: title,
message: title,
detail: message,
buttons: buttons,
type: 'info',
}) === 0
);
}
saveNewTemplate(name, contents, callback) {
if (!name || name.length === 0) {
this._displayError(localized('You must provide a template name.'));
return;
}
if (name.match(INVALID_TEMPLATE_NAME_REGEX)) {
this._displayError(
localized(
'Invalid template name! Names can only contain letters, numbers, spaces, dashes, and underscores.'
)
);
return;
}
let number = 1;
let resolvedName = name;
const sameName = t => t.name === resolvedName;
while (this._items.find(sameName)) {
resolvedName = `${name} ${number}`;
number += 1;
}
this.saveTemplate(resolvedName, contents, callback);
this.trigger(this);
}
saveTemplate(name, contents, callback) {
const filename = `${name}.html`;
const templatePath = path.join(this._templatesDir, filename);
let template = this._items.find(t => t.name === name);
this.unwatch();
fs.writeFile(templatePath, contents, err => {
this.watch();
if (err) {
this._displayError(err);
}
if (!template) {
template = {
id: filename,
name: name,
path: templatePath,
};
this._items.unshift(template);
}
if (callback) {
callback(template);
}
});
}
_onDeleteTemplate(name) {
const template = this._items.find(t => t.name === name);
if (!template) {
return;
}
if (
this._displayDialog(
localized('Delete Template?'),
localized('The template and its file will be permanently deleted.'),
[localized('Delete'), localized('Cancel')]
)
) {
fs.unlink(template.path, () => {
this._populate();
});
}
}
_onRenameTemplate(name, newName) {
const template = this._items.find(t => t.name === name);
if (!template) {
return;
}
if (newName.match(INVALID_TEMPLATE_NAME_REGEX)) {
this._displayError(
localized(
'Invalid template name! Names can only contain letters, numbers, spaces, dashes, and underscores.'
)
);
return;
}
if (newName.length === 0) {
this._displayError(localized('You must provide a template name.'));
return;
}
const newFilename = `${newName}.html`;
const oldPath = path.join(this._templatesDir, `${name}.html`);
const newPath = path.join(this._templatesDir, newFilename);
fs.rename(oldPath, newPath, () => {
template.name = newName;
template.id = newFilename;
template.path = newPath;
this.trigger(this);
});
}
async _onInsertTemplateId({
templateId,
headerMessageId,
}: { templateId?: string; headerMessageId?: string } = {}) {
const template = this._items.find(t => t.id === templateId);
const templateBody = fs.readFileSync(template.path).toString();
const session = await DraftStore.sessionForClientId(headerMessageId);
let proceed = true;
if (!session.draft().pristine && !session.draft().hasEmptyBody()) {
proceed = this._displayDialog(
localized('Replace draft contents?'),
localized(
'It looks like your draft already has some content. Loading this template will overwrite all draft contents.'
),
[localized('Replace contents'), localized('Cancel')]
);
}
if (proceed) {
const current = session.draft().body;
let insertion = current.length;
for (const s of [
'<signature',
'<div class="gmail_quote_attribution"',
'<blockquote class="gmail_quote"',
]) {
const i = current.indexOf(s);
if (i !== -1) {
insertion = Math.min(insertion, i);
}
}
session.changes.add({ body: `${templateBody}${current.substr(insertion)}` });
}
}
}
export default new TemplateStore();