mirror of
https://github.com/searxng/searxng
synced 2024-01-01 18:24:07 +00:00
eff0884498
Show default image `img/img_load_error.svg` when image can't be loaded.
----
Some words about class ImageLayout:
The https://github.com/searxng/searxng/blob/master/searx/static/themes/simple/js/searxng.js is build by a grunt task ..
d0e21a01b4/searx/static/themes/simple/gruntfile.js (L91-L93)
The `/__common__/js/*.js` concats also https://github.com/searxng/searxng/blob/master/searx/static/themes/__common__/js/image_layout.js where a modified copy of the of the "Google-image-layout" (`ImageLayout`) is implemented [1][2][3].
[1] https://ptgamr.github.io/2014-09-12-google-image-layout/
[2] https://ptgamr.github.io/google-image-layout/src/google-image-layout.js
[3] https://github.com/ptgamr/google-image-layout/tree/master
----
Signed-off-by: Markus Heiser <markus.heiser@darmarit.de>
312 lines
8.8 KiB
JavaScript
312 lines
8.8 KiB
JavaScript
/*jshint esversion: 6 */
|
|
|
|
module.exports = function(grunt) {
|
|
|
|
const eachAsync = require('each-async');
|
|
|
|
grunt.initConfig({
|
|
|
|
_brand: '../../../../src/brand',
|
|
_templates: '../../../templates',
|
|
|
|
pkg: grunt.file.readJSON('package.json'),
|
|
watch: {
|
|
scripts: {
|
|
files: ['gruntfile.js', 'src/**'],
|
|
tasks: [
|
|
'eslint',
|
|
'copy',
|
|
'concat',
|
|
'uglify',
|
|
'less:development',
|
|
'less:production',
|
|
'image',
|
|
'svg2png',
|
|
'svg2jinja'
|
|
]
|
|
}
|
|
},
|
|
eslint: {
|
|
options: {
|
|
overrideConfigFile: '.eslintrc.json',
|
|
failOnError: false
|
|
},
|
|
target: [
|
|
'svg4web.svgo.js',
|
|
'src/js/main/*.js',
|
|
'src/js/head/*.js',
|
|
'../__common__/js/*.js'
|
|
],
|
|
},
|
|
stylelint: {
|
|
options: {
|
|
formatter: 'unix',
|
|
},
|
|
src: [
|
|
'src/less/**/*.less',
|
|
]
|
|
},
|
|
copy: {
|
|
js: {
|
|
expand: true,
|
|
cwd: './node_modules',
|
|
dest: './js/',
|
|
flatten: true,
|
|
filter: 'isFile',
|
|
timestamp: true,
|
|
src: [
|
|
'./leaflet/dist/leaflet.js',
|
|
]
|
|
},
|
|
css: {
|
|
expand: true,
|
|
cwd: './node_modules',
|
|
dest: './css/',
|
|
flatten: true,
|
|
filter: 'isFile',
|
|
timestamp: true,
|
|
src: [
|
|
'./leaflet/dist/leaflet.css',
|
|
]
|
|
},
|
|
leaflet_images: {
|
|
expand: true,
|
|
cwd: './node_modules',
|
|
dest: './css/images/',
|
|
flatten: true,
|
|
filter: 'isFile',
|
|
timestamp: true,
|
|
src: [
|
|
'./leaflet/dist/images/*.png',
|
|
]
|
|
},
|
|
},
|
|
concat: {
|
|
head_and_body: {
|
|
options: {
|
|
separator: ';'
|
|
},
|
|
files: {
|
|
'js/searxng.head.js': ['src/js/head/*.js'],
|
|
'js/searxng.js': [
|
|
'src/js/main/*.js',
|
|
'../__common__/js/*.js',
|
|
'./node_modules/autocomplete-js/dist/autocomplete.js'
|
|
]
|
|
}
|
|
}
|
|
},
|
|
uglify: {
|
|
options: {
|
|
output: {
|
|
comments: 'some'
|
|
},
|
|
ie8: false,
|
|
warnings: true,
|
|
compress: false,
|
|
mangle: true,
|
|
sourceMap: true
|
|
},
|
|
dist: {
|
|
files: {
|
|
'js/searxng.head.min.js': ['js/searxng.head.js'],
|
|
'js/searxng.min.js': ['js/searxng.js']
|
|
}
|
|
}
|
|
},
|
|
less: {
|
|
development: {
|
|
options: {
|
|
paths: ["less"],
|
|
},
|
|
files: {
|
|
"css/searxng.css": "src/less/style.less",
|
|
"css/searxng-rtl.css": "src/less/style-rtl.less"
|
|
}
|
|
},
|
|
production: {
|
|
options: {
|
|
paths: ["less"],
|
|
plugins: [
|
|
new (require('less-plugin-clean-css'))()
|
|
],
|
|
sourceMap: true,
|
|
sourceMapURL: (name) => { const s = name.split('/'); return s[s.length - 1] + '.map';},
|
|
outputSourceFiles: false,
|
|
sourceMapRootpath: '../',
|
|
},
|
|
files: {
|
|
"css/searxng.min.css": "src/less/style.less",
|
|
"css/searxng-rtl.min.css": "src/less/style-rtl.less"
|
|
}
|
|
},
|
|
},
|
|
image: {
|
|
svg4web: {
|
|
options: {
|
|
svgo: ['--config', 'svg4web.svgo.js']
|
|
},
|
|
files: {
|
|
'<%= _templates %>/__common__/searxng-wordmark.min.svg': '<%= _brand %>/searxng-wordmark.svg',
|
|
'img/searxng.svg': '<%= _brand %>/searxng.svg'
|
|
}
|
|
},
|
|
favicon: {
|
|
options: {
|
|
svgo: ['--config', 'svg4favicon.svgo.js']
|
|
},
|
|
files: {
|
|
'img/favicon.svg': '<%= _brand %>/searxng-wordmark.svg'
|
|
}
|
|
},
|
|
},
|
|
svg2png: {
|
|
favicon: {
|
|
files: {
|
|
'img/favicon.png': '<%= _brand %>/searxng-wordmark.svg',
|
|
'img/searxng.png': '<%= _brand %>/searxng.svg',
|
|
'img/img_load_error.svg': '<%= _brand %>/img_load_error.svg'
|
|
}
|
|
}
|
|
},
|
|
svg2jinja: {
|
|
all: {
|
|
src: {
|
|
'warning': 'node_modules/ionicons/dist/svg/alert-outline.svg',
|
|
'close': 'node_modules/ionicons/dist/svg/close-outline.svg',
|
|
'chevron-up-outline': 'node_modules/ionicons/dist/svg/chevron-up-outline.svg',
|
|
'chevron-right': 'node_modules/ionicons/dist/svg/chevron-forward-outline.svg',
|
|
'chevron-left': 'node_modules/ionicons/dist/svg/chevron-back-outline.svg',
|
|
'menu-outline': 'node_modules/ionicons/dist/svg/menu-outline.svg',
|
|
'ellipsis-vertical-outline': 'node_modules/ionicons/dist/svg/ellipsis-vertical-outline.svg',
|
|
'magnet-outline': 'node_modules/ionicons/dist/svg/magnet-outline.svg',
|
|
'globe-outline': 'node_modules/ionicons/dist/svg/globe-outline.svg',
|
|
'search-outline': 'node_modules/ionicons/dist/svg/search-outline.svg',
|
|
'image-outline': 'node_modules/ionicons/dist/svg/image-outline.svg',
|
|
'play-outline': 'node_modules/ionicons/dist/svg/play-outline.svg',
|
|
'newspaper-outline': 'node_modules/ionicons/dist/svg/newspaper-outline.svg',
|
|
'location-outline': 'node_modules/ionicons/dist/svg/location-outline.svg',
|
|
'musical-notes-outline': 'node_modules/ionicons/dist/svg/musical-notes-outline.svg',
|
|
'layers-outline': 'node_modules/ionicons/dist/svg/layers-outline.svg',
|
|
'school-outline': 'node_modules/ionicons/dist/svg/school-outline.svg',
|
|
'file-tray-full-outline': 'node_modules/ionicons/dist/svg/file-tray-full-outline.svg',
|
|
'people-outline': 'node_modules/ionicons/dist/svg/people-outline.svg',
|
|
},
|
|
dest: '../../../templates/simple/icons.html',
|
|
},
|
|
},
|
|
});
|
|
|
|
grunt.registerMultiTask('svg2jinja', 'Create Jinja2 macro', function() {
|
|
const ejs = require('ejs'), svgo = require('svgo');
|
|
const icons = {}
|
|
for(const iconName in this.data.src) {
|
|
const svgFileName = this.data.src[iconName];
|
|
try {
|
|
const svgContent = grunt.file.read(svgFileName, { encoding: 'utf8' })
|
|
const svgoResult = svgo.optimize(svgContent, {
|
|
path: svgFileName,
|
|
multipass: true,
|
|
plugins: [
|
|
{
|
|
name: "removeTitle",
|
|
},
|
|
{
|
|
name: "removeXMLNS",
|
|
},
|
|
{
|
|
name: "addAttributesToSVGElement",
|
|
params: {
|
|
attributes: [
|
|
{ "aria-hidden": "true" }
|
|
]
|
|
}
|
|
}
|
|
]
|
|
});
|
|
icons[iconName] = svgoResult.data.replace("'", "\\'");
|
|
} catch (err) {
|
|
grunt.log.error(err);
|
|
}
|
|
}
|
|
const template = `{# this file was generated by searx/static/themes/simple/gruntfile.js #}
|
|
{%- set icons = {
|
|
<% for (const iconName in icons) { %> '<%- iconName %>':'<%- icons[iconName] %>',
|
|
<% } %>
|
|
}
|
|
-%}
|
|
|
|
{% macro icon(action, alt) -%}
|
|
{{ icons[action] | replace("ionicon", "ion-icon") | safe }}
|
|
{%- endmacro %}
|
|
|
|
{% macro icon_small(action) -%}
|
|
{{ icons[action] | replace("ionicon", "ion-icon-small") | safe }}
|
|
{%- endmacro %}
|
|
|
|
{% macro icon_big(action, alt) -%}
|
|
{{ icons[action] | replace("ionicon", "ion-icon-big") | safe }}
|
|
{%- endmacro %}
|
|
`;
|
|
const result = ejs.render(template, { icons });
|
|
grunt.file.write(this.data.dest, result, { encoding: 'utf8' });
|
|
grunt.log.ok(this.data.dest + " created");
|
|
});
|
|
|
|
grunt.registerMultiTask('svg2png', 'Convert SVG to PNG', function () {
|
|
const sharp = require('sharp'), done = this.async();
|
|
eachAsync(this.files, async (file, _index, next) => {
|
|
try {
|
|
if (file.src.length != 1) {
|
|
next("this task supports only one source per destination");
|
|
}
|
|
const info = await sharp(file.src[0])
|
|
.png({
|
|
force: true,
|
|
compressionLevel: 9,
|
|
palette: true,
|
|
})
|
|
.toFile(file.dest);
|
|
grunt.log.ok(file.dest + ' created (' + info.size + ' bytes, ' + info.width + 'px * ' + info.height + 'px)');
|
|
next();
|
|
} catch (error) {
|
|
grunt.fatal(error);
|
|
next(error);
|
|
}
|
|
}, error => {
|
|
if (error) {
|
|
grunt.fatal(error);
|
|
done(error);
|
|
} else {
|
|
done();
|
|
}
|
|
});
|
|
});
|
|
|
|
grunt.loadNpmTasks('grunt-contrib-watch');
|
|
grunt.loadNpmTasks('grunt-contrib-copy');
|
|
grunt.loadNpmTasks('grunt-contrib-uglify');
|
|
grunt.loadNpmTasks('grunt-image');
|
|
grunt.loadNpmTasks('grunt-contrib-jshint');
|
|
grunt.loadNpmTasks('grunt-contrib-concat');
|
|
grunt.loadNpmTasks('grunt-contrib-less');
|
|
grunt.loadNpmTasks('grunt-contrib-cssmin');
|
|
grunt.loadNpmTasks('grunt-stylelint');
|
|
grunt.loadNpmTasks('grunt-eslint');
|
|
|
|
grunt.registerTask('test', ['jshint']);
|
|
|
|
grunt.registerTask('default', [
|
|
'eslint',
|
|
'stylelint',
|
|
'copy',
|
|
'concat',
|
|
'uglify',
|
|
'less:development',
|
|
'less:production',
|
|
'image',
|
|
'svg2png',
|
|
'svg2jinja',
|
|
]);
|
|
};
|