Skip to content

Commit

Permalink
Merge pull request #4993 from camptocamp/svg-url
Browse files Browse the repository at this point in the history
Allows to have SVG URL
  • Loading branch information
sbrunner authored Jul 10, 2019
2 parents 3dffbca + 4896337 commit 8dc116a
Show file tree
Hide file tree
Showing 19 changed files with 724 additions and 31 deletions.
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -15,3 +15,4 @@
/contribs/gmf/build/
/contribs/gmf/examples/https.js
/openlayers_src/
/dist/
2 changes: 1 addition & 1 deletion Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ NGEO_EXAMPLES_JS_FILES := $(NGEO_EXAMPLES_HTML_FILES:.html=.js)

GMF_PARTIALS_FILES := $(shell find contribs/gmf/src/ -name *.html)
GMF_JS_FILES := $(shell find contribs/gmf/src/ -type f -name '*.js')
GMF_ALL_SRC_FILES := $(shell find contribs/gmf/src/ -type f) $(shell find contribs/gmf/src/cursors/ -type f) $(NGEO_ALL_SRC_FILES)
GMF_ALL_SRC_FILES := $(shell find contribs/gmf/src/ -type f) $(NGEO_ALL_SRC_FILES)
GMF_TEST_JS_FILES := $(shell find contribs/gmf/test/ -type f -name '*.js')
GMF_EXAMPLES_HTML_FILES := $(shell ls -1 contribs/gmf/examples/*.html)
GMF_EXAMPLES_JS_FILES := $(GMF_EXAMPLES_HTML_FILES:.html=.js)
Expand Down
98 changes: 98 additions & 0 deletions buildtools/generate-xml-from-tokens.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,98 @@
// Initially get from https://github.com/tildeio/simple-html-tokenizer/blob/v0.1.1/lib/simple-html-tokenizer/generator.js

const escape = (function() {
const test = /[&<>"'`]/;
const replace = /[&<>"'`]/g;
const map = {
'&': '&amp;',
'<': '&lt;',
'>': '&gt;',
'"': '&quot;',
'\'': '&#x27;',
'`': '&#x60;'
};
function escapeChar(char) {
return map[char];
}
return function escape(string) {
if (!test.test(string)) {
return string;
}
return string.replace(replace, escapeChar);
};
}());

function Generator() {
this.escape = escape;
}

Generator.prototype = {
generate: function(tokens) {
let buffer = '';
for (let i = 0; i < tokens.length; i++) {
const token = tokens[i];
buffer += this[token.type](token);
}
return buffer;
},

escape: function(text) {
const unsafeCharsMap = this.unsafeCharsMap;
return text.replace(this.unsafeChars, function(char) {
return unsafeCharsMap[char] || char;
});
},

StartTag: function(token) {
let out = '<';
out += token.tagName;

if (token.attributes.length) {
out += ' ' + this.Attributes(token.attributes);
}

out += token.selfClosing ? '/>' : '>';

return out;
},

EndTag: function(token) {
return '</' + token.tagName + '>';
},

Chars: function(token) {
return this.escape(token.chars);
},

Comment: function(token) {
return '<!--' + token.chars + '-->';
},

Attributes: function(attributes) {
const out = [];

for (let i = 0, l = attributes.length; i < l; i++) {
const attribute = attributes[i];

out.push(this.Attribute(attribute[0], attribute[1]));
}

return out.join(' ');
},

Attribute: function(name, value) {
let attrString = name;

if (value) {
value = this.escape(value);
attrString += '="' + value + '"';
}

return attrString;
}
};

module.exports = function(tokens) {
const generator = new Generator();
return generator.generate(tokens);
};
69 changes: 58 additions & 11 deletions buildtools/svg-viewbox-loader.js
Original file line number Diff line number Diff line change
@@ -1,32 +1,79 @@
const simpleHTMLTokenizer = require('simple-html-tokenizer');
const generate = require('./generate-xml-from-tokens.js');

module.exports = function(source) {
this.cacheable(true);
let tokens = simpleHTMLTokenizer.tokenize(source);

tokens = tokens.map((tag) => {
let width = undefined;
let height = undefined;
if (tag.type === 'StartTag' && tag.tagName === 'svg') {
let width = undefined;
let height = undefined;
let x = 0;
let y = 0;
for (const attribute of tag.attributes) {
if (attribute[0] === 'width') {
width = parseFloat(attribute[1]);
try {
width = parseFloat(attribute[1]);
} catch (e) {
console.warn('Unable to read width: ' + attribute[1]);
}
}
if (attribute[0] === 'height') {
height = parseFloat(attribute[1]);
try {
height = parseFloat(attribute[1]);
} catch (e) {
console.warn('Unable to read height: ' + attribute[1]);
}
}
if (attribute[0] === 'x') {
try {
x = parseFloat(attribute[1]);
} catch (e) {
console.warn('Unable to read x: ' + attribute[1]);
}
}
if (attribute[0] === 'y') {
try {
y = parseFloat(attribute[1]);
} catch (e) {
console.warn('Unable to read y: ' + attribute[1]);
}
}
if (attribute[0] === 'viewBox') {
try {
const attrs = attribute[1].split(' ');
x = parseFloat(attrs[0]);
y = parseFloat(attrs[1]);
width = parseFloat(attrs[2]);
height = parseFloat(attrs[3]);
} catch (e) {
console.warn('Unable to read viewbox: ' + attribute[1]);
}
}
}
if (width !== undefined && height != undefined) {
tag.attributes = tag.attributes.filter((attribute) => {
return attribute[0] !== 'width' && attribute[0] != 'height'
})
tag.attributes.push(['viewBox', `0 0 ${width} ${height}`, true]);
tag.attributes.push(['height', '1em', true]);
tag.attributes.push(['width', `${width / height}em`, true]);
return attribute[0] !== 'width' && attribute[0] != 'height' && attribute[0] != 'viewBox';
});
if (x) {
tag.attributes.push(['x', x, true]);
}
if (y) {
tag.attributes.push(['y', y, true]);
}
if (this.resourceQuery.search(/inline/) >= 0) {
tag.attributes.push(['width', width, true]);
tag.attributes.push(['height', height, true]);
} else {
tag.attributes.push(['viewBox', `0 0 ${width} ${height}`, true]);
tag.attributes.push(['height', '1em', true]);
tag.attributes.push(['width', `${width / height}em`, true]);
}
}
}
return tag;
})
});

return simpleHTMLTokenizer.generate(tokens)
return generate(tokens);
};
14 changes: 0 additions & 14 deletions buildtools/webpack.commons.js
Original file line number Diff line number Diff line change
Expand Up @@ -61,19 +61,6 @@ const htmlRule = {
use: 'ejs-loader',
};

const svgRule = {
test: /\.svg$/,
use: [
{
loader: 'svg-inline-loader',
options: {
removeSVGTagAttrs: false,
},
},
'./buildtools/svg-viewbox-loader',
'svgo-loader',
]
};

function get_comp(firsts, lasts) {
return (f1, f2) => {
Expand Down Expand Up @@ -142,7 +129,6 @@ const config = function(hardSourceConfig, babelLoaderCacheDirectory) {
cssRule,
sassRule,
htmlRule,
svgRule,
ngeoRule,
otherRule,
]
Expand Down
29 changes: 29 additions & 0 deletions buildtools/webpack.dev.js
Original file line number Diff line number Diff line change
Expand Up @@ -12,19 +12,48 @@ const resourcesRule = {
}
};

const svgRule = {
test: /\.svg$/,
oneOf: [{
resourceQuery: /url/,
use: [
{
loader: 'file-loader',
options: {
name: '[name].[ext]'
},
},
'svgo-loader',
]
}, {
use: [
{
loader: 'svg-inline-loader',
options: {
removeSVGTagAttrs: false,
},
},
'./buildtools/svg-viewbox-loader',
'svgo-loader',
]
}]
};

new webpack.LoaderOptionsPlugin({
debug: false
});


module.exports = {
mode: 'development',
// devtool: 'eval',
output: {
filename: '[name].js'
},
module: {
rules: [
resourcesRule,
svgRule,
]
},
};
39 changes: 39 additions & 0 deletions buildtools/webpack.prod.js
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,44 @@ const resourcesRule = {
}
};

const svgRule = {
test: /\.svg$/,
oneOf: [{
resourceQuery: /url/,
use: [
{
loader: 'file-loader',
options: {
name: '[name].[hash:6].[ext]'
},
},
'svgo-loader',
]
}, {
resourceQuery: /inline/,
use: [
{
loader: 'svg-inline-loader',
options: {
removeSVGTagAttrs: false,
},
},
'svgo-loader',
]
}, {
use: [
{
loader: 'svg-inline-loader',
options: {
removeSVGTagAttrs: false,
},
},
'./buildtools/svg-viewbox-loader',
'svgo-loader',
]
}]
};

module.exports = function(TerserPluginCache) {
return {
mode: 'production',
Expand All @@ -24,6 +62,7 @@ module.exports = function(TerserPluginCache) {
module: {
rules: [
resourcesRule,
svgRule,
]
},
optimization: {
Expand Down
20 changes: 19 additions & 1 deletion contribs/gmf/apps/desktop_alt/Controller.js
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,11 @@ import ngeoRoutingModule from 'ngeo/routing/module.js';
import EPSG2056 from '@geoblocks/proj/src/EPSG_2056.js';
import EPSG21781 from '@geoblocks/proj/src/EPSG_21781.js';
import ngeoStatemanagerWfsPermalink from 'ngeo/statemanager/WfsPermalink.js';
import {Circle, Fill, Stroke, Style} from 'ol/style';
import Style from 'ol/style/Style.js';
import Circle from 'ol/style/Circle.js';
import Fill from 'ol/style/Fill.js';
import Stroke from 'ol/style/Stroke.js';
import Icon from 'ol/style/Icon.js';
import Raven from 'raven-js/src/raven.js';
import RavenPluginsAngular from 'raven-js/plugins/angular.js';

Expand Down Expand Up @@ -194,4 +198,18 @@ const module = angular.module('Appdesktop_alt', [

module.controller('AlternativeDesktopController', Controller);


module.value('gmfPermalinkOptions', /** @type {import('gmf/permalink/Permalink.js').PermalinkOptions} */ ({
crosshairStyle: [
new Style({
image: new Icon({
src: 'data:image/svg+xml;base64,' + btoa(require('./image/crosshair.svg?inline')),
// Also working
// src: require('./image/crosshair.svg?url'),
imgSize: [22, 22],
})
})
]
}));

export default module;
17 changes: 17 additions & 0 deletions contribs/gmf/apps/desktop_alt/image/crosshair.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file removed contribs/gmf/apps/desktop_alt/image/logo.png
Binary file not shown.
Loading

0 comments on commit 8dc116a

Please sign in to comment.