Mailspring/app/src/canvas-utils.es6
Ben Gotow 1a3cca8d0a
Totally overhauled composer based on Slate (#524)
* Remove the composer contenteditable, replace with basic <textarea>

* Beginning broader cleanup of draft session

* DraftJS composer with color, style support

* Serialization/unserialization of basic styles, toolbar working

* WIP

* Switch to draft-js-plugins approach, need to revisit HTML

* Move HTML conversion functionality into plugins

* Add spellcheck context menu to editor

* Initial work on quoted text

* Further work on quoted text

* BLOCK approach

* Entity approach - better, does not bump out to top level

* Hiding and showing quoted text via CSS

* Get rid of ability to inject another subject line component

* Clean up specs, DraftFactory to ES6

* Remove old initial focus hack

* Fix focusing, initial text selection

* Remove participant “collapsing” support, it can be confusing

* Correctly terminate links on carriage returns

* Initial signature support, allow removal of uneditable blocks

* Sync body string with body editorstate

* Simplify draft editor session, finish signatures

* Templates

* Minor fixes

* Simplify link/open tracking, ensure it works

* Reorg composer, rework template editor

* Omg the slowness is all the stupid emoji button

* Polish and small fixes

* Performance improvements, new templates UI

* Don’t assume nodes are elements

* Fix for sending drafts twice due to back-to-back saves

* Fix order of operations on app quit to save drafts reliably

* Improve DraftJS-Convert whitespace handling

* Use contentID throughout attachment lifecycle

* Try to fix images

* Switch to Slate instead of DraftJS… much better

* Fix newline handling

* Bug fixes

* Cleanup

* Finish templates plugin

* Clean up text editing / support for Gmail email styles

* Support for color + size on the same node, clean trailing whitespace

* Restore emoji typeahead / emoji picker

* Fix scrolling in template editor

* Fix specs

* Fix newlines

* Re-implement spellcheck to be faster

* Make spellcheck decorator changes invisible to the undo/redo stack

* Remove comment

* Polish themplates panel

* Fix #521
2018-01-11 15:55:56 -08:00

148 lines
7.6 KiB
JavaScript

const ThreadDragImage = document.createElement('img');
ThreadDragImage.src = ``;
const DragCanvas = document.createElement('canvas');
DragCanvas.style.position = 'absolute';
DragCanvas.style.zIndex = 0;
DragCanvas.style.top = 0;
document.body.appendChild(DragCanvas);
const PercentLoadedCache = {};
const PercentLoadedCanvas = document.createElement('canvas');
PercentLoadedCanvas.style.position = 'absolute';
PercentLoadedCanvas.style.zIndex = 0;
PercentLoadedCanvas.style.top = 0;
document.body.appendChild(PercentLoadedCanvas);
const SystemTrayCanvas = document.createElement('canvas');
export function roundRect(ctx, x, y, width, height, radius = 5, fill, stroke = true) {
ctx.beginPath();
ctx.moveTo(x + radius, y);
ctx.lineTo(x + width - radius, y);
ctx.quadraticCurveTo(x + width, y, x + width, y + radius);
ctx.lineTo(x + width, y + height - radius);
ctx.quadraticCurveTo(x + width, y + height, x + width - radius, y + height);
ctx.lineTo(x + radius, y + height);
ctx.quadraticCurveTo(x, y + height, x, y + height - radius);
ctx.lineTo(x, y + radius);
ctx.quadraticCurveTo(x, y, x + radius, y);
ctx.closePath();
if (stroke) {
ctx.stroke();
}
if (fill) {
ctx.fill();
}
}
export function dataURIForLoadedPercent(_percent) {
const percent = Math.floor(_percent / 5.0) * 5.0;
const cacheKey = `${percent}%`;
if (!PercentLoadedCache[cacheKey]) {
const canvas = PercentLoadedCanvas;
const scale = window.devicePixelRatio;
canvas.width = 20 * scale;
canvas.height = 20 * scale;
canvas.style.width = '30px';
canvas.style.height = '30px';
const half = 10 * scale;
const ctx = canvas.getContext('2d');
ctx.strokeStyle = '#AAA';
ctx.lineWidth = 3 * scale;
ctx.clearRect(0, 0, 20 * scale, 20 * scale);
ctx.beginPath();
ctx.arc(
half,
half,
half - ctx.lineWidth,
-0.5 * Math.PI,
-0.5 * Math.PI + 2 * Math.PI * percent / 100.0
);
ctx.stroke();
PercentLoadedCache[cacheKey] = canvas.toDataURL();
}
return PercentLoadedCache[cacheKey];
}
export function canvasWithThreadDragImage(count) {
const canvas = DragCanvas;
// Make sure the canvas has a 2x pixel density on retina displays
const scale = window.devicePixelRatio;
canvas.width = 58 * scale;
canvas.height = 55 * scale;
canvas.style.width = '58px';
canvas.style.height = '55px';
const ctx = canvas.getContext('2d');
// mail background image
if (count > 1) {
ctx.rotate(-20 * Math.PI / 180);
ctx.drawImage(ThreadDragImage, -10 * scale, 2 * scale, 48 * scale, 48 * scale);
ctx.rotate(20 * Math.PI / 180);
}
ctx.drawImage(ThreadDragImage, 0, 0, 48 * scale, 48 * scale);
// count bubble
const dotGradient = ctx.createLinearGradient(0, 0, 0, 15 * scale);
dotGradient.addColorStop(0, 'rgb(116, 124, 143)');
dotGradient.addColorStop(1, 'rgb(67, 77, 104)');
ctx.strokeStyle = 'rgba(39, 48, 68, 0.6)';
ctx.lineWidth = 1;
ctx.fillStyle = dotGradient;
let textX = 49;
let text = `${count}`;
if (count < 10) {
roundRect(ctx, 41 * scale, 1 * scale, 16 * scale, 14 * scale, 7 * scale, true, true);
} else if (count < 100) {
roundRect(ctx, 37 * scale, 1 * scale, 20 * scale, 14 * scale, 7 * scale, true, true);
textX = 46;
} else {
roundRect(ctx, 33 * scale, 1 * scale, 25 * scale, 14 * scale, 7 * scale, true, true);
text = '99+';
textX = 46;
}
// count text
ctx.fillStyle = 'rgba(255,255,255,0.9)';
ctx.font = `${11 * scale}px sans-serif`;
ctx.textAlign = 'center';
ctx.fillText(text, textX * scale, 12 * scale, 30 * scale);
return DragCanvas;
}
export function measureTextInCanvas(text, font) {
const canvas = document.createElement('canvas');
const context = canvas.getContext('2d');
context.font = font;
return Math.ceil(context.measureText(text).width);
}
export function canvasWithSystemTrayIconAndText(img, text) {
const canvas = SystemTrayCanvas;
const w = img.width;
const h = img.height;
const font = '26px Nylas-Pro, sans-serif';
const textWidth = text.length > 0 ? measureTextInCanvas(text, font) + 2 : 0;
canvas.width = w + textWidth;
canvas.height = h;
const context = canvas.getContext('2d');
context.font = font;
context.fillStyle = 'black';
context.textAlign = 'start';
context.textBaseline = 'middle';
context.drawImage(img, 0, 0);
// Place after img, vertically aligned
context.fillText(text, w, h / 2);
return canvas;
}