-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathv2.js
113 lines (103 loc) · 3.73 KB
/
v2.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
/*
Copyright (c) 2016, Yahoo Inc.
Copyrights licensed under the New BSD License. See the accompanying LICENSE file for terms.
*/
var colors = require('css-color-names'),
dot = require('dot'),
fs = require('fs'),
path = require('path'),
utils = require('./utils'),
SVGO = require('svgo'),
svgo = new SVGO({
plugins: [{
sortDefsChildren: false
}]
}),
TEMPLATE = dot.template(fs.readFileSync(path.join(__dirname, 'templates', 'v2.svg'), 'utf-8')),
COLOR_REGEX = /^[0-9a-f]{6}$/i,
STROKE_REGEX = /^s\{(.+?)\}$/i,
DEFAULT_COLOR_FIRST = '#696969', // dimgrey
DEFAULT_COLOR_REST = '#d3d3d3', // lightgrey
PAD_X = 5,
PAD_Y = 4,
LINE_HEIGHT = 12,
DECENDER_HEIGHT = 2,
DEFAULT_LETTER_WIDTH = 8; // probably unicode, hard to guess width
/**
* Validate and return appropriate color code from input string
* @method getColorCode
* @param {String} input String to check for valid color code
*/
function getColorCode(input) {
if (COLOR_REGEX.test(input)) {
return '#' + input.toLowerCase();
}
if (colors[input]) {
return colors[input];
}
return false;
}
function sectionsToData(sections) {
var badgeData = {
width: 0,
height: 0,
sections: [],
};
sections.forEach(function(section, s) {
var sectionData = {
x: 0,
width: 0,
lines: [],
},
sectionHeight,
text,
lines;
if (! Array.isArray(section)) {
section = [ section ];
}
text = section.shift();
sectionData.x = badgeData.width;
sectionData.color = (s === 0 ? DEFAULT_COLOR_FIRST : DEFAULT_COLOR_REST);
section.forEach(function(attribute) {
// stroke attribute `s{color}` as CSS color or color code in hex
var strokeAttribute = STROKE_REGEX.exec(attribute);
if (strokeAttribute) {
sectionData.stroke = getColorCode(strokeAttribute[1]) || null;
}
// fill color attribute (without attribute qualifier) as CSS color or color code in hex
if (getColorCode(attribute)) {
sectionData.color = getColorCode(attribute);
}
// FUTURE -- text alignment `a{align}` lmr (only matters when multiline)
// FUTURE -- font `f{font}` mainly for monospace (`fm`)
});
lines = text.split('\n');
lines.forEach(function(line, l) {
var lineData = {
x: 0,
y: 0,
text: line,
},
lineWidth;
lineWidth = (2 * PAD_X) + utils.textWidth(lineData.text, DEFAULT_LETTER_WIDTH);
lineData.x = badgeData.width + PAD_X;
lineData.y = (LINE_HEIGHT * l) + PAD_Y + LINE_HEIGHT - DECENDER_HEIGHT;
sectionData.lines.push(lineData);
sectionData.width = Math.max(sectionData.width, lineWidth);
});
badgeData.sections.push(sectionData);
sectionHeight = (2 * PAD_Y) + (lines.length * LINE_HEIGHT);
badgeData.height = Math.max(badgeData.height, sectionHeight);
badgeData.width += sectionData.width;
});
return badgeData;
}
module.exports = function badge_v2(sections, callback) {
var raw = TEMPLATE(sectionsToData(sections));
return svgo.optimize(raw).then(function(optimized) {
if (callback) callback(undefined, optimized.data);
return optimized.data;
});
};
// mainly for unit testing
module.exports.sectionsToData = sectionsToData;