Skip to content
Snippets Groups Projects
Unverified Commit 317e19ea authored by Danial's avatar Danial Committed by GitHub
Browse files

Add support for simple-icons, colored icons with ?logoColor (#1810)

* add simple-icons

* handle undefined

* support logoColor param

* our icon > simple-icon

* dont crash :white_check_mark:

* return false → undefined

* update test

* add test

* support logoColor on our logos

* cache as base64, pre-load-simple-icons, logo-helper

* add ?logoColor information

* and simple-icons reference, link to github master branch for our logos

* update simple-icons

update to 1.7.1

* Revert "and simple-icons reference, link to github master branch for our logos"

This reverts commit 5e99d5f8db360ed49ce08851d3ba84e35682b7c5.

* add link to simple-icons

* Add snapshot test

* support dash in place of space for logo name
parent 8020ff00
Branches
Tags
No related merge requests found
......@@ -8,3 +8,35 @@ exports['The badge generator JSON should always produce the same JSON (unless we
"value": "grown"
}
`
exports['shields GitHub logo default color (#333333) 1'] = `
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="113" height="20"><linearGradient id="b" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/><stop offset="1" stop-opacity=".1"/></linearGradient><clipPath id="a"><rect width="113" height="20" rx="3" fill="#fff"/></clipPath><g clip-path="url(#a)"><path fill="#555" d="M0 0h54v20H0z"/><path fill="#007ec6" d="M54 0h59v20H54z"/><path fill="url(#b)" d="M0 0h113v20H0z"/></g><g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="110"><image x="5" y="3" width="14" height="14" xlink:href=""/> <text x="365" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="270">label</text><text x="365" y="140" transform="scale(.1)" textLength="270">label</text><text x="825" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="490">message</text><text x="825" y="140" transform="scale(.1)" textLength="490">message</text></g> </svg>
`
exports['shields GitHub logo custom color (whitesmoke) 1'] = `
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="113" height="20"><linearGradient id="b" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/><stop offset="1" stop-opacity=".1"/></linearGradient><clipPath id="a"><rect width="113" height="20" rx="3" fill="#fff"/></clipPath><g clip-path="url(#a)"><path fill="#555" d="M0 0h54v20H0z"/><path fill="#007ec6" d="M54 0h59v20H54z"/><path fill="url(#b)" d="M0 0h113v20H0z"/></g><g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="110"><image x="5" y="3" width="14" height="14" xlink:href=""/> <text x="365" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="270">label</text><text x="365" y="140" transform="scale(.1)" textLength="270">label</text><text x="825" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="490">message</text><text x="825" y="140" transform="scale(.1)" textLength="490">message</text></g> </svg>
`
exports['simple-icons javascript logo default color (#F7DF1E) 1'] = `
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="113" height="20"><linearGradient id="b" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/><stop offset="1" stop-opacity=".1"/></linearGradient><clipPath id="a"><rect width="113" height="20" rx="3" fill="#fff"/></clipPath><g clip-path="url(#a)"><path fill="#555" d="M0 0h54v20H0z"/><path fill="#007ec6" d="M54 0h59v20H54z"/><path fill="url(#b)" d="M0 0h113v20H0z"/></g><g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="110"><image x="5" y="3" width="14" height="14" xlink:href=""/> <text x="365" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="270">label</text><text x="365" y="140" transform="scale(.1)" textLength="270">label</text><text x="825" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="490">message</text><text x="825" y="140" transform="scale(.1)" textLength="490">message</text></g> </svg>
`
exports['simple-icons javascript logo custom color (rgba(46,204,113,0.8)) 1'] = `
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="113" height="20"><linearGradient id="b" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/><stop offset="1" stop-opacity=".1"/></linearGradient><clipPath id="a"><rect width="113" height="20" rx="3" fill="#fff"/></clipPath><g clip-path="url(#a)"><path fill="#555" d="M0 0h54v20H0z"/><path fill="#007ec6" d="M54 0h59v20H54z"/><path fill="url(#b)" d="M0 0h113v20H0z"/></g><g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="110"><image x="5" y="3" width="14" height="14" xlink:href=""/> <text x="365" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="270">label</text><text x="365" y="140" transform="scale(.1)" textLength="270">label</text><text x="825" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="490">message</text><text x="825" y="140" transform="scale(.1)" textLength="490">message</text></g> </svg>
`
exports['The badge generator badges with logos should always produce the same badge shields GitHub logo default color (#333333) 1'] = `
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="113" height="20"><linearGradient id="b" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/><stop offset="1" stop-opacity=".1"/></linearGradient><clipPath id="a"><rect width="113" height="20" rx="3" fill="#fff"/></clipPath><g clip-path="url(#a)"><path fill="#555" d="M0 0h54v20H0z"/><path fill="#4c1" d="M54 0h59v20H54z"/><path fill="url(#b)" d="M0 0h113v20H0z"/></g><g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="110"><image x="5" y="3" width="14" height="14" xlink:href="github"/> <text x="365" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="270">label</text><text x="365" y="140" transform="scale(.1)" textLength="270">label</text><text x="825" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="490">message</text><text x="825" y="140" transform="scale(.1)" textLength="490">message</text></g> </svg>
`
exports['The badge generator badges with logos should always produce the same badge shields GitHub logo custom color (whitesmoke) 1'] = `
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="113" height="20"><linearGradient id="b" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/><stop offset="1" stop-opacity=".1"/></linearGradient><clipPath id="a"><rect width="113" height="20" rx="3" fill="#fff"/></clipPath><g clip-path="url(#a)"><path fill="#555" d="M0 0h54v20H0z"/><path fill="#4c1" d="M54 0h59v20H54z"/><path fill="url(#b)" d="M0 0h113v20H0z"/></g><g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="110"><image x="5" y="3" width="14" height="14" xlink:href="github"/> <text x="365" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="270">label</text><text x="365" y="140" transform="scale(.1)" textLength="270">label</text><text x="825" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="490">message</text><text x="825" y="140" transform="scale(.1)" textLength="490">message</text></g> </svg>
`
exports['The badge generator badges with logos should always produce the same badge simple-icons javascript logo default color (#F7DF1E) 1'] = `
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="113" height="20"><linearGradient id="b" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/><stop offset="1" stop-opacity=".1"/></linearGradient><clipPath id="a"><rect width="113" height="20" rx="3" fill="#fff"/></clipPath><g clip-path="url(#a)"><path fill="#555" d="M0 0h54v20H0z"/><path fill="#4c1" d="M54 0h59v20H54z"/><path fill="url(#b)" d="M0 0h113v20H0z"/></g><g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="110"><image x="5" y="3" width="14" height="14" xlink:href="javascript"/> <text x="365" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="270">label</text><text x="365" y="140" transform="scale(.1)" textLength="270">label</text><text x="825" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="490">message</text><text x="825" y="140" transform="scale(.1)" textLength="490">message</text></g> </svg>
`
exports['The badge generator badges with logos should always produce the same badge simple-icons javascript logo custom color (rgba(46,204,113,0.8)) 1'] = `
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="113" height="20"><linearGradient id="b" x2="0" y2="100%"><stop offset="0" stop-color="#bbb" stop-opacity=".1"/><stop offset="1" stop-opacity=".1"/></linearGradient><clipPath id="a"><rect width="113" height="20" rx="3" fill="#fff"/></clipPath><g clip-path="url(#a)"><path fill="#555" d="M0 0h54v20H0z"/><path fill="#4c1" d="M54 0h59v20H54z"/><path fill="url(#b)" d="M0 0h113v20H0z"/></g><g fill="#fff" text-anchor="middle" font-family="DejaVu Sans,Verdana,Geneva,sans-serif" font-size="110"><image x="5" y="3" width="14" height="14" xlink:href="javascript"/> <text x="365" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="270">label</text><text x="365" y="140" transform="scale(.1)" textLength="270">label</text><text x="825" y="150" fill="#010101" fill-opacity=".3" transform="scale(.1)" textLength="490">message</text><text x="825" y="140" transform="scale(.1)" textLength="490">message</text></g> </svg>
`
......@@ -169,7 +169,7 @@ export default class Usage extends React.PureComponent {
<code>?logo=appveyor</code>
</td>
<td>
Insert one of the named logos ({this.constructor.renderNamedLogos()})
Insert one of the named logos from ({this.constructor.renderNamedLogos()}) or <a href="https://simpleicons.org/" target="_BLANK">simple-icons</a>
</td>
</tr>
<tr>
......@@ -178,6 +178,12 @@ export default class Usage extends React.PureComponent {
</td>
<td>Insert custom logo image ( 14px high)</td>
</tr>
<tr>
<td>
<code>?logoColor=violet</code>
</td>
<td>Set the color of the logo (hex, rgb, rgba, hsl, hsla and css named colors supported)</td>
</tr>
<tr>
<td>
<code>?logoWidth=40</code>
......
......@@ -2,6 +2,11 @@
const isCSSColor = require('is-css-color');
const logos = require('./load-logos')();
const simpleIcons = require('./load-simple-icons')();
const {
svg2base64,
isDataUri,
} = require('./logo-helper');
const colorschemes = require('./colorscheme.json');
function toArray(val) {
......@@ -14,10 +19,6 @@ function toArray(val) {
}
}
function isDataUri(s) {
return s !== undefined && /^(data:)([^;]+);([^,]+),(.+)$/.test(s);
}
function prependPrefix(s, prefix) {
if (s === undefined) {
return undefined;
......@@ -39,8 +40,12 @@ function isHexColor (s = ''){
function makeColor(color) {
if (isHexColor(color)) {
return '#' + color;
} else {
} else if (colorschemes[color] !== undefined){
return colorschemes[color].colorB;
} else if (isCSSColor(color)){
return color;
} else {
return undefined;
}
}
......@@ -69,9 +74,27 @@ function makeLabel(defaultLabel, overrides) {
return '' + (overrides.label === undefined ? defaultLabel || '' : overrides.label);
}
function getShieldsIcon(icon = '', color = ''){
icon = typeof icon === 'string' ? icon.toLowerCase() : '';
if (!logos[icon]){
return undefined;
}
color = makeColor(color);
return color ? logos[icon].svg.replace(/fill="(.+?)"/g, `fill="${color}"`) : logos[icon].base64;
}
function getSimpleIcon(icon = '', color = null){
icon = typeof icon === 'string' ? icon.toLowerCase().replace(/ /g, '-') : '';
if (!simpleIcons[icon]){
return undefined;
}
color = makeColor(color);
return color ? simpleIcons[icon].svg.replace('<svg', `<svg fill="${color}"`) : simpleIcons[icon].base64;
}
function makeLogo(defaultNamedLogo, overrides) {
if (overrides.logo === undefined){
return logos[defaultNamedLogo];
return svg2base64(getShieldsIcon(defaultNamedLogo, overrides.logoColor) || getSimpleIcon(defaultNamedLogo, overrides.logoColor));
}
// +'s are replaced with spaces when used in query params, this returns them to +'s, then removes remaining whitespace - #1546
......@@ -80,7 +103,7 @@ function makeLogo(defaultNamedLogo, overrides) {
if (isDataUri(maybeDataUri)) {
return maybeDataUri;
} else {
return logos[overrides.logo];
return svg2base64(getShieldsIcon(overrides.logo, overrides.logoColor) || getSimpleIcon(overrides.logo, overrides.logoColor));
}
}
......@@ -116,7 +139,6 @@ function makeBadgeData(defaultLabel, overrides) {
module.exports = {
toArray,
prependPrefix,
isDataUri,
isHexColor,
makeLabel,
makeLogo,
......
......@@ -3,12 +3,12 @@
const { expect } = require('chai');
const { test, given, forCases } = require('sazerac');
const {
isDataUri,
prependPrefix,
isHexColor,
makeLabel,
makeLogo,
makeBadgeData,
makeColor,
setBadgeColor
} = require('./badge-data');
......@@ -19,14 +19,6 @@ describe('Badge data helpers', function() {
given(undefined, 'data:').expect(undefined);
});
test(isDataUri, () => {
given('').expect(true);
forCases([
given('data:foobar'),
given('foobar'),
]).expect(false);
});
test(isHexColor, () => {
forCases([
given('f00bae'),
......@@ -80,11 +72,20 @@ describe('Badge data helpers', function() {
logoPosition: 10,
logoWidth: 25,
links: ['https://example.com/'],
colorA: 'blue',
colorA: '#007ec6',
colorB: '#f00bae',
});
});
test(makeColor, () => {
given('red').expect('#e05d44');
given('blue').expect('#007ec6');
given('4c1').expect('#4c1');
given('f00f00').expect('#f00f00');
given('papayawhip').expect('papayawhip');
given('purple').expect('purple');
});
test(setBadgeColor, () => {
given({}, 'red').expect({ colorscheme: 'red' });
given({}, 'f00f00').expect({ colorB: '#f00f00' });
......
......@@ -2,6 +2,7 @@
const fs = require('fs');
const path = require('path');
const { svg2base64 } = require('./logo-helper');
function loadLogos () {
// Cache svg logos from disk in base64 string
......@@ -12,11 +13,14 @@ function loadLogos () {
if (filename[0] === '.') { return; }
// filename is eg, github.svg
const svg = fs.readFileSync(logoDir + '/' + filename).toString();
const base64 = svg2base64(svg);
// eg, github
const name = filename.slice(0, -('.svg'.length));
logos[name] = 'data:image/svg+xml;base64,' +
Buffer.from(svg).toString('base64');
const name = filename.slice(0, -('.svg'.length)).toLowerCase();
logos[name] = {
svg,
base64
};
});
return logos;
}
......
'use strict';
const simpleIcons = require('simple-icons');
const { svg2base64 } = require('./logo-helper');
function loadSimpleIcons(){
Object.keys(simpleIcons).forEach(function (key) {
const k = key.toLowerCase().replace(/ /g, '-');
if (k !== key) {
simpleIcons[k] = simpleIcons[key];
delete simpleIcons[key];
}
simpleIcons[k].base64 = svg2base64(simpleIcons[k].svg.replace('<svg', `<svg fill="#${simpleIcons[k].hex}"`));
});
return (simpleIcons);
}
module.exports = loadSimpleIcons;
'use strict';
function isDataUri(s) {
return s !== undefined && /^(data:)([^;]+);([^,]+),(.+)$/.test(s);
}
function svg2base64(svg){
if (typeof svg !== 'string'){
return undefined;
}
// Check if logo is already base64
return isDataUri(svg) ? svg : 'data:image/svg+xml;base64,' + Buffer.from(svg).toString('base64');
}
module.exports = {
svg2base64,
isDataUri,
}
'use strict';
const {
test,
given,
forCases,
} = require('sazerac');
const {
svg2base64,
isDataUri,
} = require('./logo-helper');
describe('Logo helpers', function() {
test(svg2base64, () => {
given('').expect('');
given('<svg xmlns="http://www.w3.org/2000/svg"/>').expect('');
given(undefined).expect(undefined);
});
test(isDataUri, () => {
given('').expect(true);
forCases([
given('data:foobar'),
given('foobar'),
]).expect(false);
});
});
......@@ -127,4 +127,25 @@ describe('The badge generator', () => {
expect(svg).to.include('""').and.to.include('some-value');
});
});
describe('badges with logos should always produce the same badge', () => {
it('shields GitHub logo default color (#333333)', () => {
const svg = makeBadge({ text: ['label', 'message'], format: 'svg', logo: 'github' });
snapshot(svg);
});
it('shields GitHub logo custom color (whitesmoke)', () => {
const svg = makeBadge({ text: ['label', 'message'], format: 'svg', logo: 'github', logoColor: 'whitesmoke' });
snapshot(svg);
});
it('simple-icons javascript logo default color (#F7DF1E)', () => {
const svg = makeBadge({ text: ['label', 'message'], format: 'svg', logo: 'javascript' });
snapshot(svg);
});
it('simple-icons javascript logo custom color (rgba(46,204,113,0.8))', () => {
const svg = makeBadge({ text: ['label', 'message'], format: 'svg', logo: 'javascript', logoColor: 'rgba(46,204,113,0.8)' });
snapshot(svg);
});
});
});
......@@ -40,6 +40,7 @@ const globalQueryParams = new Set([
'style',
'link',
'logo',
'logoColor',
'logoPosition',
'logoWidth',
'link',
......
......@@ -5369,6 +5369,7 @@
"resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.5.tgz",
"integrity": "sha1-k4NwpXtKUd6ix3wV1cX9+JUWQAk=",
"dev": true,
"optional": true,
"requires": {
"delayed-stream": "~1.0.0"
}
......@@ -5440,7 +5441,8 @@
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
"integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=",
"dev": true
"dev": true,
"optional": true
},
"delegates": {
"version": "1.0.0",
......@@ -5765,13 +5767,15 @@
"version": "1.27.0",
"resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.27.0.tgz",
"integrity": "sha1-gg9XIpa70g7CXtVeW13oaeVDbrE=",
"dev": true
"dev": true,
"optional": true
},
"mime-types": {
"version": "2.1.15",
"resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.15.tgz",
"integrity": "sha1-pOv1BkCUVpI3uM9wBGd20J/JKu0=",
"dev": true,
"optional": true,
"requires": {
"mime-db": "~1.27.0"
}
......@@ -5855,7 +5859,8 @@
"version": "1.0.1",
"resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz",
"integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=",
"dev": true
"dev": true,
"optional": true
},
"oauth-sign": {
"version": "0.8.2",
......@@ -13130,6 +13135,11 @@
"integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=",
"dev": true
},
"simple-icons": {
"version": "1.7.1",
"resolved": "https://registry.npmjs.org/simple-icons/-/simple-icons-1.7.1.tgz",
"integrity": "sha1-xoVlvjKsRsq4N7IZrnGuu6T9JVM="
},
"sinon": {
"version": "6.0.0",
"resolved": "https://registry.npmjs.org/sinon/-/sinon-6.0.0.tgz",
......
......@@ -48,6 +48,7 @@
"redis": "~2.6.2",
"request": "~2.87.0",
"semver": "~5.5.0",
"simple-icons": "^1.7.1",
"svgo": "~1.0.5",
"xml2js": "~0.4.16",
"xmldom": "~0.1.27",
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Please register or to comment