mirror of
https://github.com/discourse/discourse.git
synced 2025-01-07 16:35:58 +08:00
307 lines
8.6 KiB
JavaScript
307 lines
8.6 KiB
JavaScript
/*eslint no-bitwise:0 */
|
|
import getUrl from 'discourse-common/lib/get-url';
|
|
|
|
export const LOREM = `
|
|
Lorem ipsum dolor sit amet, consectetur adipiscing elit.
|
|
Nullam eget sem non elit tincidunt rhoncus. Fusce velit nisl,
|
|
porttitor sed nisl ac, consectetur interdum metus. Fusce in
|
|
consequat augue, vel facilisis felis. Nunc tellus elit, and
|
|
semper vitae orci nec, blandit pharetra enim. Aenean a ebus
|
|
posuere nunc. Maecenas ultrices viverra enim ac commodo
|
|
Vestibulum nec quam sit amet libero ultricies sollicitudin.
|
|
Nulla quis scelerisque sem, eget volutpat velit. Fusce eget
|
|
accumsan sapien, nec feugiat quam. Quisque non risus.
|
|
placerat lacus vitae, lacinia nisi. Sed metus arcu, iaculis
|
|
sit amet cursus nec, sodales at eros.`;
|
|
|
|
const scaled = {};
|
|
|
|
function canvasFor(image, w, h) {
|
|
w = Math.ceil(w);
|
|
h = Math.ceil(h);
|
|
|
|
const can = document.createElement('canvas');
|
|
can.width = w;
|
|
can.height = h;
|
|
|
|
const ctx = can.getContext('2d');
|
|
ctx.drawImage(image, 0, 0, w, h);
|
|
return can;
|
|
}
|
|
|
|
export function createPreviewComponent(width, height, obj) {
|
|
return Ember.Component.extend({
|
|
layoutName: 'components/theme-preview',
|
|
width,
|
|
height,
|
|
ctx: null,
|
|
loaded: false,
|
|
|
|
didInsertElement() {
|
|
this._super();
|
|
const c = this.$('canvas')[0];
|
|
this.ctx = c.getContext("2d");
|
|
this.reload();
|
|
},
|
|
|
|
images() { },
|
|
|
|
loadImages() {
|
|
const images = this.images();
|
|
if (images) {
|
|
return Ember.RSVP.Promise.all(Object.keys(images).map(id => {
|
|
return loadImage(images[id]).then(img => this[id] = img);
|
|
}));
|
|
}
|
|
return Ember.RSVP.Promise.resolve();
|
|
},
|
|
|
|
reload() {
|
|
this.loadImages().then(() => {
|
|
this.loaded = true;
|
|
this.triggerRepaint();
|
|
});
|
|
},
|
|
|
|
triggerRepaint() {
|
|
Ember.run.scheduleOnce('afterRender', this, 'repaint');
|
|
},
|
|
|
|
repaint() {
|
|
if (!this.loaded) { return false; }
|
|
|
|
const colors = this.get('wizard').getCurrentColors();
|
|
if (!colors) { return; }
|
|
|
|
const { ctx } = this;
|
|
|
|
ctx.fillStyle = colors.secondary;
|
|
ctx.fillRect(0, 0, width, height);
|
|
|
|
this.paint(ctx, colors, this.width, this.height);
|
|
|
|
// draw border
|
|
ctx.beginPath();
|
|
ctx.strokeStyle='rgba(0, 0, 0, 0.2)';
|
|
ctx.rect(0, 0, width, height);
|
|
ctx.stroke();
|
|
},
|
|
|
|
categories() {
|
|
return [{name: 'consecteteur', color: '#652D90'}, {name: 'ultrices', color: '#3AB54A'}];
|
|
},
|
|
|
|
scaleImage(image, x, y, w, h) {
|
|
w = Math.floor(w);
|
|
h = Math.floor(h);
|
|
|
|
const { ctx } = this;
|
|
|
|
const key = `${image.src}-${w}-${h}`;
|
|
|
|
if (!scaled[key]) {
|
|
|
|
let copy = image;
|
|
let ratio = copy.width / copy.height;
|
|
let newH = (copy.height * 0.5);
|
|
while (newH > h) {
|
|
copy = canvasFor(copy, ratio * newH, newH);
|
|
newH = newH * 0.5;
|
|
}
|
|
|
|
scaled[key] = copy;
|
|
}
|
|
|
|
ctx.drawImage(scaled[key], x, y, w, h);
|
|
},
|
|
|
|
drawFullHeader(colors) {
|
|
const { ctx } = this;
|
|
|
|
const headerHeight = height * 0.15;
|
|
drawHeader(ctx, colors, width, headerHeight);
|
|
|
|
const avatarSize = height * 0.1;
|
|
|
|
// Logo
|
|
const headerMargin = headerHeight * 0.2;
|
|
const logoHeight = headerHeight - (headerMargin * 2);
|
|
|
|
if (this.logo) {
|
|
const logoWidth = (logoHeight / this.logo.height) * this.logo.width;
|
|
this.scaleImage(this.logo, headerMargin, headerMargin, logoWidth, logoHeight);
|
|
}
|
|
|
|
// Top right menu
|
|
this.scaleImage(this.avatar, width - avatarSize - headerMargin, headerMargin, avatarSize, avatarSize);
|
|
ctx.fillStyle = darkLightDiff(colors.primary, colors.secondary, 45, 55);
|
|
|
|
const headerFontSize = headerHeight / 44;
|
|
|
|
ctx.font = `${headerFontSize}em FontAwesome`;
|
|
ctx.fillText("\uf0c9", width - (avatarSize * 2) - (headerMargin * 0.5), avatarSize);
|
|
ctx.fillText("\uf002", width - (avatarSize * 3) - (headerMargin * 0.5), avatarSize);
|
|
},
|
|
|
|
drawPills(colors, headerHeight, opts) {
|
|
opts = opts || {};
|
|
|
|
const { ctx } = this;
|
|
|
|
const categoriesSize = headerHeight * 2;
|
|
const badgeHeight = categoriesSize * 0.25;
|
|
const headerMargin = headerHeight * 0.2;
|
|
|
|
ctx.beginPath();
|
|
ctx.fillStyle = darkLightDiff(colors.primary, colors.secondary, 90, -65);
|
|
ctx.rect(headerMargin, headerHeight + headerMargin, categoriesSize, badgeHeight);
|
|
ctx.fill();
|
|
|
|
const fontSize = Math.round(badgeHeight * 0.5);
|
|
ctx.font = `${fontSize}px 'Arial'`;
|
|
ctx.fillStyle = colors.primary;
|
|
ctx.fillText("all categories", headerMargin * 1.5, headerHeight + (headerMargin * 1.42) + fontSize);
|
|
|
|
ctx.font = "0.9em 'FontAwesome'";
|
|
ctx.fillStyle = colors.primary;
|
|
ctx.fillText("\uf0da", categoriesSize - (headerMargin / 4), headerHeight + (headerMargin * 1.6) + fontSize);
|
|
|
|
const text = opts.categories ? "Categories" : "Latest";
|
|
|
|
const activeWidth = categoriesSize * (opts.categories ? 0.80 : 0.55);
|
|
ctx.beginPath();
|
|
ctx.fillStyle = colors.quaternary;
|
|
ctx.rect((headerMargin * 2)+ categoriesSize, headerHeight + headerMargin, activeWidth, badgeHeight);
|
|
ctx.fill();
|
|
|
|
ctx.font = `${fontSize}px 'Arial'`;
|
|
ctx.fillStyle = colors.secondary;
|
|
let x = (headerMargin * 3.0) + categoriesSize;
|
|
ctx.fillText(text, x - (headerMargin * 0.1), headerHeight + (headerMargin * 1.5) + fontSize);
|
|
|
|
ctx.fillStyle = colors.primary;
|
|
x += categoriesSize * (opts.categories ? 0.8 : 0.6);
|
|
ctx.fillText("New", x, headerHeight + (headerMargin * 1.5) + fontSize);
|
|
|
|
x += categoriesSize * 0.4;
|
|
ctx.fillText("Unread", x, headerHeight + (headerMargin * 1.5) + fontSize);
|
|
|
|
x += categoriesSize * 0.6;
|
|
ctx.fillText("Top", x, headerHeight + (headerMargin * 1.5) + fontSize);
|
|
}
|
|
|
|
}, obj);
|
|
}
|
|
|
|
function loadImage(src) {
|
|
if (!src) { return Ember.RSVP.Promise.resolve(); }
|
|
|
|
const img = new Image();
|
|
img.src = getUrl(src);
|
|
return new Ember.RSVP.Promise(resolve => img.onload = () => resolve(img));
|
|
};
|
|
|
|
export function parseColor(color) {
|
|
const m = color.match(/^#([0-9a-f]{6})$/i);
|
|
if (m) {
|
|
const c = m[1];
|
|
return [ parseInt(c.substr(0,2),16), parseInt(c.substr(2,2),16), parseInt(c.substr(4,2),16) ];
|
|
}
|
|
|
|
return [0, 0, 0];
|
|
}
|
|
|
|
export function brightness(color) {
|
|
return (color[0] * 0.299) + (color[1] * 0.587) + (color[2] * 0.114);
|
|
}
|
|
|
|
function rgbToHsl(r, g, b) {
|
|
r /= 255;
|
|
g /= 255;
|
|
b /= 255;
|
|
let max = Math.max(r, g, b), min = Math.min(r, g, b);
|
|
let h, s, l = (max + min) / 2;
|
|
|
|
if (max === min) {
|
|
h = s = 0;
|
|
} else {
|
|
const d = max - min;
|
|
s = l > 0.5 ? d / (2 - max - min) : d / (max + min);
|
|
switch(max){
|
|
case r: h = (g - b) / d + (g < b ? 6 : 0); break;
|
|
case g: h = (b - r) / d + 2; break;
|
|
case b: h = (r - g) / d + 4; break;
|
|
}
|
|
h /= 6;
|
|
}
|
|
|
|
return [h, s, l];
|
|
}
|
|
|
|
function hue2rgb(p, q, t) {
|
|
if (t < 0) { t += 1; }
|
|
if (t > 1) { t -= 1; }
|
|
if (t < 1/6) { return p + (q - p) * 6 * t; }
|
|
if (t < 1/2) { return q; }
|
|
if (t < 2/3) { return p + (q - p) * (2/3 - t) * 6; }
|
|
return p;
|
|
}
|
|
|
|
function hslToRgb(h, s, l) {
|
|
let r, g, b;
|
|
|
|
if (s === 0) {
|
|
r = g = b = l; // achromatic
|
|
} else {
|
|
const q = l < 0.5 ? l * (1 + s) : l + s - l * s;
|
|
const p = 2 * l - q;
|
|
r = hue2rgb(p, q, h + 1/3);
|
|
g = hue2rgb(p, q, h);
|
|
b = hue2rgb(p, q, h - 1/3);
|
|
}
|
|
|
|
return [r * 255, g * 255, b * 255];
|
|
}
|
|
|
|
export function lighten(color, percent) {
|
|
const hsl = rgbToHsl(color[0], color[1], color[2]);
|
|
const scale = percent / 100.0;
|
|
const diff = scale > 0 ? 1.0 - hsl[2] : hsl[2];
|
|
|
|
hsl[2] = hsl[2] + diff * scale;
|
|
color = hslToRgb(hsl[0], hsl[1], hsl[2]);
|
|
|
|
return '#' +
|
|
((0|(1<<8) + color[0]).toString(16)).substr(1) +
|
|
((0|(1<<8) + color[1]).toString(16)).substr(1) +
|
|
((0|(1<<8) + color[2]).toString(16)).substr(1);
|
|
}
|
|
|
|
export function chooseBrighter(primary, secondary) {
|
|
const primaryCol = parseColor(primary);
|
|
const secondaryCol = parseColor(secondary);
|
|
return brightness(primaryCol) < brightness(secondaryCol) ? secondary : primary;
|
|
}
|
|
|
|
export function darkLightDiff(adjusted, comparison, lightness, darkness) {
|
|
const adjustedCol = parseColor(adjusted);
|
|
const comparisonCol = parseColor(comparison);
|
|
return lighten(adjustedCol, (brightness(adjustedCol) < brightness(comparisonCol)) ?
|
|
lightness : darkness);
|
|
}
|
|
|
|
|
|
export function drawHeader(ctx, colors, width, headerHeight) {
|
|
ctx.save();
|
|
ctx.beginPath();
|
|
ctx.rect(0, 0, width, headerHeight);
|
|
ctx.fillStyle = colors.header_background;
|
|
ctx.shadowColor = "rgba(0, 0, 0, 0.25)";
|
|
ctx.shadowBlur = 2;
|
|
ctx.shadowOffsetX = 0;
|
|
ctx.shadowOffsetY = 2;
|
|
ctx.fill();
|
|
ctx.restore();
|
|
}
|
|
|