1 # This file is a part of Geeqie project (https://www.geeqie.org/).
2 # Copyright (C) 2008 - 2022 The Geeqie Team
4 # This program is free software; you can redistribute it and/or modify
5 # it under the terms of the GNU General Public License as published by
6 # the Free Software Foundation; either version 2 of the License, or
7 # (at your option) any later version.
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
14 # Meson default directories used in this project:
23 # Project expanded default directories:
25 # bindir /usr/local/bin geeqie executable
26 # gq_bindir /usr/local/lib/geeqie * plugins scripts
27 # datadir /usr/local/share/
28 # /usr/local/share/applications org.geeqie.Geeqie.desktop
29 # [gq_]appdir /usr/local/share/geeqie/ * template.desktop
30 # desktopdir /usr/local/share/geeqie/applications plugin desktop files
31 # appdatadir /usr/local/share/metainfo org.geeqie.Geeqie.appdata.xml
32 # icondir /usr/local/share/pixmaps geeqie.png icon
33 # [gq_]helpdir /usr/local/share/doc/geeqie * readme files etc.
34 # [gq_]htmldir /usr/local/share/doc/geeqie/html * help files
35 # gq_localedir /usr/locale/share/locale
36 # mandir1 /usr/local/share/man/man1 man page
37 # podir project_root/po
39 # * See meson_options.txt file
45 version : run_command('./version.sh', check : true).stdout().strip(),
46 license : ['GPL-2.0-or-later'],
47 meson_version : '>=1.0.0',
48 default_options : ['cpp_std=c++14', 'warning_level=3', 'buildtype=debugoptimized', 'cpp_link_args=-rdynamic']
51 # To inhibit warnings from the generated file icons.h
52 add_global_arguments('-Wno-overlength-strings', language : 'c')
54 # To compile originally-C files as C++
55 add_global_arguments('-Wno-error=deprecated-declarations', language : 'cpp')
57 # Project requirements
59 gnome = import('gnome')
60 thread_dep = dependency('threads')
61 cc = meson.get_compiler('c')
64 configuration_inc = include_directories('.')
66 # Extended stack trace using backward-app
67 option = get_option('devel')
69 if cc.has_link_argument('-ldwarf')
70 add_project_link_arguments('-ldwarf', language: 'cpp')
75 glib_compile_resources = find_program('glib-compile-resources', required : true)
76 glib_genmarshal = find_program('glib-genmarshal', required : true)
78 option = get_option('git')
79 if not option.disabled()
80 running_from_git = find_program('git', required: false).found() and fs.is_dir('.git')
82 running_from_git = false
83 summary({'git' : ['disabled - ChangeLog, ChangeLog.html, lua-api help file created:', false]}, section : 'Documentation', bool_yn : true)
86 debug = get_option('debug')
88 # Note that main.cc sets prefix to the directory above where the executable is run from.
89 # This is to allow AppImages to be used
91 # These gq_* variables are paths relative to /prefix/,
92 # and are also used in defines in the source as GQ_*
93 if get_option('gq_appdir') == ''
94 gq_appdir = join_paths(get_option('datadir'), 'geeqie')
96 gq_appdir = get_option('gq_appdir')
99 # This is not the same as Meson bindir
100 if get_option('gq_bindir') == ''
101 gq_bindir = 'lib/geeqie'
103 gq_bindir = get_option('gq_bindir')
106 if get_option('gq_helpdir') == ''
107 gq_helpdir = join_paths(get_option('datadir'), 'doc/geeqie')
109 gq_helpdir = get_option('gq_helpdir')
112 if get_option('gq_htmldir') == ''
113 gq_htmldir = join_paths(get_option('datadir'), 'doc/geeqie/html')
115 gq_htmldir = get_option('gq_htmldir')
118 if get_option('gq_localedir') == ''
119 gq_localedir = join_paths(get_option('datadir'), 'locale')
121 gq_localedir = get_option('gq_localedir')
125 # Set up the absolute directory paths used
126 prefix = get_option('prefix')
127 datadir = join_paths(prefix, get_option('datadir'))
129 # Installation paths are absolute
130 appdir = join_paths(prefix, gq_appdir)
131 appdatadir = join_paths(datadir, 'metainfo')
132 desktopdir = join_paths(datadir, meson.project_name(), 'applications')
133 helpdir = join_paths(prefix, gq_helpdir)
134 htmldir = join_paths(prefix, gq_htmldir)
135 icondir = join_paths(datadir, 'pixmaps')
136 mandir1 = join_paths(datadir, 'man', 'man1')
138 podir = join_paths(meson.project_source_root(), 'po')
139 scriptsdir = join_paths(meson.project_source_root(), 'scripts')
141 summary({'gq_appdir': gq_appdir,
142 'gq_bindir': gq_helpdir,
143 'gq_helpdir': gq_helpdir,
144 'gq_htmldir': gq_htmldir,
145 'gq_localedir': gq_localedir,
146 }, section: 'Directories')
148 # Create the define constants used in the sources. Set via config.h.in
149 conf_data = configuration_data()
150 conf_data.set_quoted('VERSION', meson.project_version())
151 conf_data.set('DEBUG', debug)
153 option = get_option('gtk4')
155 gtk_dep = dependency('gtk4', required: true)
156 conf_data.set('HAVE_GTK4', 1)
158 gtk_dep = dependency('gtk+-3.0', version : '>=3.24', required: true)
160 glib_dep = dependency('glib-2.0', version : '>=2.52', required: true)
162 # Required only when backward-cpp is used
165 option = get_option('devel')
167 libdw_dep = dependency('libdw', required : true)
169 libunwind_dep = dependency('libunwind', required : true)
170 if libunwind_dep.found()
171 conf_data.set('HAVE_DEVELOPER', 1)
172 summary({'developer mode' : ['extended stacktrace:', true]}, section : 'Debugging', bool_yn : true)
174 summary({'developer mode' : ['libunwind not found. extended stacktrace:', false]}, section : 'Debugging', bool_yn : true)
177 summary({'developer mode' : ['libdw not found. extended stacktrace:', false]}, section : 'Debugging', bool_yn : true)
180 summary({'developer mode' : ['extended stacktrace:', false]}, section : 'Debugging', bool_yn : true)
183 # Required only for seg. fault stacktrace and backtrace debugging
184 option = get_option('execinfo')
186 if not option.disabled()
187 result = cc.check_header('execinfo.h')
189 # Include unconditionally dependency for NetBSD.
190 libexecinfo_dep = cc.find_library('execinfo', required : false)
191 conf_data.set('HAVE_EXECINFO_H', 1)
192 summary({'execinfo' : ['stacktrace supported:', true]}, section : 'Debugging', bool_yn : true)
194 summary({'execinfo' : ['stacktrace supported:', false]}, section : 'Debugging', bool_yn : true)
197 summary({'execinfo' : ['stacktrace supported:', false]}, section : 'Debugging', bool_yn : true)
201 req_version = '>=3.4.0'
202 option = get_option('archive')
203 if not option.disabled()
204 libarchive_dep = dependency('libarchive', version : req_version, required : get_option('archive'))
205 if libarchive_dep.found()
206 conf_data.set('HAVE_ARCHIVE', 1)
207 summary({'archive' : ['archive files e.g. .zip supported:', true]}, section : 'Configuration', bool_yn : true)
209 summary({'archive' : ['libarchive ' + req_version + ' not found - archive files e.g. .zip supported::', false]}, section : 'Configuration', bool_yn : true)
212 summary({'archive' : ['disabled - archive files e.g. .zip supported:', false]}, section : 'Configuration', bool_yn : true)
216 req_version = '>=2.0'
217 option = get_option('cms')
218 if not option.disabled()
219 xxd = find_program('xxd', 'xxdi.pl', required : false)
221 lcms_dep = dependency('lcms2', version : req_version, required : get_option('cms'))
223 conf_data.set('HAVE_LCMS', 1)
224 conf_data.set('HAVE_LCMS2', 1)
225 summary({'cms' : ['color management supported:', true]}, section : 'Configuration', bool_yn : true)
227 summary({'cms' : ['lcms2' + req_version + ' not found - color management supported:', false]}, section : 'Configuration', bool_yn : true)
230 summary({'cms' : ['xxd or xxdi.pl not found - color management supported:', false]}, section : 'Configuration', bool_yn : true)
233 summary({'cms' : ['disabled - color management supported:', false]}, section : 'Configuration', bool_yn : true)
237 req_version = '>=2.5.27'
238 option = get_option('djvu')
239 if not option.disabled()
240 ddjvuapi_dep = dependency('ddjvuapi', version : req_version, required : get_option('djvu'))
241 if ddjvuapi_dep.found()
242 conf_data.set('HAVE_DJVU', 1)
243 summary({'djvu' : ['djvu files supported:', true]}, section : 'Configuration', bool_yn : true)
245 summary({'djvu' : ['ddjvuapi ' + req_version + ' not found - djvu files supported:', false]}, section : 'Configuration', bool_yn : true)
248 summary({'djvu' : ['disabled - djvu files supported:', false]}, section : 'Configuration', bool_yn : true)
251 option = get_option('evince')
252 if not option.disabled()
253 evince = find_program('evince', required : false)
255 summary({'print preview' : ['print preview supported:', true]}, section : 'Configuration', bool_yn : true)
257 summary({'print preview' : ['evince not found - print preview supported:', false]}, section : 'Configuration', bool_yn : true)
260 summary({'print preview' : ['disabled - print preview supported:', false]}, section : 'Configuration', bool_yn : true)
263 # Required only for seg. fault stacktrace and backtrace debugging
264 option = get_option('execinfo')
265 if not option.disabled()
266 result = cc.check_header('execinfo.h')
268 conf_data.set('HAVE_EXECINFO_H', 1)
269 summary({'execinfo' : ['stacktrace supported:', true]}, section : 'Configuration', bool_yn : true)
271 summary({'execinfo' : ['stacktrace supported:', false]}, section : 'Configuration', bool_yn : true)
274 summary({'execinfo' : ['stacktrace supported:', false]}, section : 'Configuration', bool_yn : true)
278 req_version = '>=0.11'
279 option = get_option('exiv2')
280 if not option.disabled()
281 exiv2_dep = dependency('exiv2', version : req_version, required : get_option('exiv2'))
283 conf_data.set('HAVE_EXIV2', 1)
284 summary({'exiv2' : ['image metadata processed by exiv2:', true]}, section : 'Configuration', bool_yn : true)
286 summary({'exiv2' : ['exiv2 ' + req_version + ' not found - image data not processed by exiv2:', false]}, section : 'Configuration', bool_yn : true)
289 summary({'exiv2' : ['disabled - image data processed by exiv2:', false]}, section : 'Configuration', bool_yn : true)
293 champlain_gtk_dep = []
296 req_version_champlain = '>=0.12'
297 req_version_champlain_gtk = '>=0.12'
298 req_version_clutter = '>=1.0'
299 req_version_clutter_gtk = '>=1.0'
300 option = get_option('gps-map')
301 if not option.disabled()
302 champlain_dep = dependency('champlain-0.12', version : req_version_champlain, required : get_option('gps-map'))
303 champlain_gtk_dep = dependency('champlain-gtk-0.12', version : req_version_champlain_gtk, required : get_option('gps-map'))
304 if champlain_dep.found() and champlain_gtk_dep.found()
305 clutter_dep = dependency('clutter-1.0', version : req_version_clutter, required : get_option('gps-map'))
306 clutter_gtk_dep = dependency('clutter-gtk-1.0', version : req_version_clutter_gtk, required : get_option('gps-map'))
307 if clutter_dep.found() and clutter_gtk_dep.found()
308 conf_data.set('HAVE_CLUTTER', 1)
309 conf_data.set('HAVE_LIBCHAMPLAIN', 1)
310 conf_data.set('HAVE_LIBCHAMPLAIN_GTK', 1)
311 summary({'gps-map' : ['GPS map displayed', true]}, section : 'Configuration', bool_yn : true)
313 if not clutter_dep.found()
314 summary({'gps-map-clutter' : ['clutter-1.0 ' + req_version_clutter + ' not found - GPS map displayed:', false]}, section : 'Configuration', bool_yn : true)
316 if not clutter_gtk_dep.found()
317 summary({'gps-map-clutter-gtk' : ['clutter-gtk-1.0 ' + req_version_clutter_gtk + ' not found - GPS map displayed:', false]}, section : 'Configuration', bool_yn : true)
321 if not champlain_dep.found()
322 summary({'gps-map-champlain' : ['champlain-0.12 ' + req_version_champlain + ' not found - GPS map displayed:', false]}, section : 'Configuration', bool_yn : true)
324 if not champlain_gtk_dep.found()
325 summary({'gps-map-champlain-gtk' : ['champlain-gtk-0.12 ' + req_version_champlain_gtk + ' not found - GPS map displayed:', false]}, section : 'Configuration', bool_yn : true)
329 summary({'gps-map' : ['disabled - GPS map displayed:', false]}, section : 'Configuration', bool_yn : true)
333 req_version = '>=1.3.2'
334 option = get_option('heif')
335 if not option.disabled()
336 libheif_dep = dependency('libheif', version : req_version, required : get_option('heif'))
337 if libheif_dep.found()
338 conf_data.set('HAVE_HEIF', 1)
339 summary({'heif' : ['heif files supported:', true]}, section : 'Configuration', bool_yn : true)
341 summary({'heif' : ['libheif ' + req_version + ' not found - heif files supported:', false]}, section : 'Configuration', bool_yn : true)
344 summary({'heif' : ['disabled - heif files supported:', false]}, section : 'Configuration', bool_yn : true)
348 req_version = '>=2.3.0'
349 option = get_option('j2k')
350 if not option.disabled()
351 libopenjp2_dep = dependency('libopenjp2', version : req_version, required : get_option('j2k'))
352 if libopenjp2_dep.found()
353 conf_data.set('HAVE_J2K', 1)
354 summary({'j2k' : ['j2k files supported:', true]}, section : 'Configuration', bool_yn : true)
356 summary({'j2k' : ['libopenjp2 ' + req_version + ' not found - j2k files supported:', false]}, section : 'Configuration', bool_yn : true)
359 summary({'j2k' : ['disabled - j2k files supported:', false]}, section : 'Configuration', bool_yn : true)
363 option = get_option('jpeg')
364 if not option.disabled()
365 libjpeg_dep = dependency('libjpeg', required : get_option('jpeg'))
366 if libjpeg_dep.found()
367 if cc.has_function('jpeg_destroy_decompress', dependencies : libjpeg_dep)
368 conf_data.set('HAVE_JPEG', 1)
369 summary({'jpeg' : ['jpeg files supported:', true]}, section : 'Configuration', bool_yn : true)
371 summary({'jpeg' : ['jpeg_destroy_decompress not found - jpeg files supported:', false]}, section : 'Configuration', bool_yn : true)
374 summary({'jpeg' : ['libjpeg: not found', false]}, section : 'Configuration', bool_yn : true)
377 summary({'jpeg' : ['disabled - jpeg files supported:', false]}, section : 'Configuration', bool_yn : true)
381 req_version = '>=0.3.7'
382 option = get_option('jpegxl')
383 if not option.disabled()
384 libjxl_dep = dependency('libjxl', version : req_version, required : get_option('jpegxl'))
385 if libjxl_dep.found()
386 conf_data.set('HAVE_JPEGXL', 1)
387 summary({'jpegxl' : ['jpegxl files supported:', true]}, section : 'Configuration', bool_yn : true)
389 summary({'jpegxl' : ['libjxl ' + req_version + ' not found - jpegxl files supported:', false]}, section : 'Configuration', bool_yn : true)
392 summary({'jpegxl' : ['disabled - jpegxl files supported:', false]}, section : 'Configuration', bool_yn : true)
396 req_version = '>=0.20'
397 option = get_option('libraw')
398 if not option.disabled()
399 libraw_dep = dependency('libraw', version : req_version, required : get_option('libraw'))
400 if libraw_dep.found()
401 conf_data.set('HAVE_RAW', 1)
402 summary({'libraw' : ['.cr3 files supported:', true]}, section : 'Configuration', bool_yn : true)
404 summary({'libraw' : ['libraw ' + req_version + ' not found - .cr3 files supported:', false]}, section : 'Configuration', bool_yn : true)
407 summary({'libraw' : ['disabled - .cr3 files supported:', false]}, section : 'Configuration', bool_yn : true)
411 req_version = '>=5.3'
412 option = get_option('lua')
413 if not option.disabled()
414 lua_dep = dependency('lua', 'lua5.3', 'lua-5.3', 'lua53', version: req_version, required: get_option('lua'))
416 conf_data.set('HAVE_LUA', 1)
417 summary({'lua' : ['lua supported:', true]}, section : 'Configuration', bool_yn : true)
419 summary({'lua' : ['lua ' + req_version + ' not found - lua supported:', false]}, section : 'Configuration', bool_yn : true)
422 summary({'lua' : ['disabled - lua supported:', false]}, section : 'Configuration', bool_yn : true)
425 # Install standard documents
426 option = get_option('pandoc')
427 if not option.disabled()
428 pandoc = find_program('pandoc', required : false)
430 readme_html = custom_target(
433 output: 'README.html',
434 command: [pandoc, '@INPUT@', '-o', '@OUTPUT@'],
436 install_dir: helpdir)
438 summary({'README' : ['README.html created:', true]}, section : 'Documentation', bool_yn : true)
440 summary({'README' : ['pandoc not found - README.html created:', false]}, section : 'Documentation', bool_yn : true)
443 summary({'pandoc' : ['disabled - README.html created:', false]}, section : 'Documentation', bool_yn : true)
445 install_data('README.md', 'COPYING', 'TODO', install_dir : helpdir)
447 poppler_glib_dep = []
448 req_version = '>=0.62'
449 option = get_option('pdf')
450 if not option.disabled()
451 poppler_glib_dep = dependency('poppler-glib', version : req_version, required : get_option('pdf'))
452 if poppler_glib_dep.found()
453 conf_data.set('HAVE_PDF', 1)
454 summary({'pdf' : ['pdf files supported:', true]}, section : 'Configuration', bool_yn : true)
456 summary({'pdf' : ['poppler-glib ' + req_version + ' not found - pdf files supported:', false]}, section : 'Configuration', bool_yn : true)
459 summary({'pdf' : ['disabled - pdf files supported:', false]}, section : 'Configuration', bool_yn : true)
463 req_version = '>=1.6'
464 option = get_option('spell')
465 if not option.disabled()
466 gspell_dep = dependency('gspell-1', version : req_version, required: get_option('spell'))
467 if gspell_dep.found()
468 conf_data.set('HAVE_SPELL', 1)
469 summary({'spell' : ['spelling checks enabled', true]}, section : 'Configuration', bool_yn : true)
471 summary({'spell' : ['gspell-1 ' + req_version + ' not found - spelling checks enabled', false]}, section : 'Configuration', bool_yn : true)
474 summary({'spell' : ['disabled - spelling checks enabled', false]}, section : 'Configuration', bool_yn : true)
478 option = get_option('tiff')
479 if not option.disabled()
480 tiff_dep = dependency('libtiff-4', required: get_option('tiff'))
482 if cc.has_function('TIFFClientOpen', dependencies : tiff_dep)
483 conf_data.set('HAVE_TIFF', 1)
484 summary({'tiff' : ['tiff files supported:', true]}, section : 'Configuration', bool_yn : true)
486 summary({'tiff' : ['TIFFClientOpen not found - tiff files supported:', false]}, section : 'Configuration', bool_yn : true)
489 summary({'tiff' : ['libtiff not found - tiff files supported:', false]}, section : 'Configuration', bool_yn : true)
492 summary({'tiff' : ['disabled - tiff files supported:', false]}, section : 'Configuration', bool_yn : true)
495 libffmpegthumbnailer_dep = []
496 req_version = '>=2.1.0'
497 option = get_option('videothumbnailer')
498 if not option.disabled()
499 libffmpegthumbnailer_dep = dependency('libffmpegthumbnailer',
500 version : req_version,
501 required : get_option('videothumbnailer'))
503 if libffmpegthumbnailer_dep.found()
504 conf_data.set('HAVE_FFMPEGTHUMBNAILER', 1)
505 summary({'videothumbnailer' : ['thumbnails of video files supported:', true]}, section : 'Configuration', bool_yn : true)
507 result = cc.has_member('struct video_thumbnailer_struct', 'prefer_embedded_metadata', prefix : '#include <libffmpegthumbnailer/videothumbnailerc.h>')
509 conf_data.set('HAVE_FFMPEGTHUMBNAILER_METADATA', 1)
511 summary({'fmpegthumbnailer_metadata' : ['fmpegthumbnailer_metadata found:', result]}, section : 'Thumbnailer', bool_yn : true)
513 result = cc.has_member('struct image_data_struct', 'image_data_width', prefix : '#include <libffmpegthumbnailer/videothumbnailerc.h>' )
515 conf_data.set('HAVE_FFMPEGTHUMBNAILER_RGB', 1)
517 summary({'fmpegthumbnailer_rgb' : ['fmpegthumbnailer_rgb found:', result]}, section : 'Thumbnailer', bool_yn : true)
519 result = cc.has_function('video_thumbnailer_set_size', dependencies : libffmpegthumbnailer_dep)
521 conf_data.set('HAVE_FFMPEGTHUMBNAILER_WH', 1)
523 summary({'fmpegthumbnailer_set_size' : ['fmpegthumbnailer_set_size found:', result]}, section : 'Thumbnailer', bool_yn : true)
525 summary({'videothumbnailer' : ['libvideothumbnailer ' + req_version + ' not found - thumbnails of video files supported', false]}, section : 'Configuration', bool_yn : true)
528 summary({'videothumbnailer' : ['disabled -thumbnails of video files supported', false]}, section : 'Configuration', bool_yn : true)
532 req_version = '>=0.6.1'
533 option = get_option('webp')
534 if not option.disabled()
535 libwebp_dep = dependency('libwebp', version : req_version, required : get_option('webp'))
536 if libwebp_dep.found()
537 conf_data.set('HAVE_WEBP', 1)
538 summary({'webp' : ['webp files supported:', true]}, section : 'Configuration', bool_yn : true)
540 summary({'webp' : ['libwebp ' + req_version + ' not found - webp files supported:', false]}, section : 'Configuration', bool_yn : true)
543 summary({'webp' : ['disabled - webp files supported:', false]}, section : 'Configuration', bool_yn : true)
546 # Check for nl_langinfo and _NL_TIME_FIRST_WEEKDAY
547 code = '''#include <langinfo.h>
549 int main (int argc, char ** argv) {
551 c = nl_langinfo(_NL_TIME_FIRST_WEEKDAY);
554 if cc.links(code, name : 'nl_langinfo and _NL_TIME_FIRST_WEEKDAY')
555 conf_data.set('HAVE__NL_TIME_FIRST_WEEKDAY', 1)
556 summary({'nl_langinfo' : ['first weekday depends on locale:', true]}, section : 'Documentation', bool_yn : true)
558 summary({'nl_langinfo' : ['nl_langinfo not found - first weekday depends on locale:', false, 'first weekday defaults to Monday']}, section : 'Documentation', bool_yn : true)
561 conf_data.set_quoted('GETTEXT_PACKAGE', meson.project_name())
562 conf_data.set_quoted('GQ_APPDIR', gq_appdir)
563 conf_data.set_quoted('GQ_BINDIR', gq_bindir)
564 conf_data.set_quoted('GQ_HELPDIR', gq_helpdir)
565 conf_data.set_quoted('GQ_HTMLDIR', gq_htmldir)
566 conf_data.set_quoted('GQ_LOCALEDIR', gq_localedir)
568 conf_data.set_quoted('PACKAGE', meson.project_name())
569 conf_data.set_quoted('PACKAGE_NAME', meson.project_name())
570 conf_data.set_quoted('PACKAGE_STRING', meson.project_version())
571 conf_data.set_quoted('PACKAGE_TARNAME', meson.project_name())
572 conf_data.set_quoted('PACKAGE_VERSION', meson.project_version())
573 conf_data.set_quoted('VERSION', meson.project_version())
575 configure_file(input : 'config.h.in',
578 configuration : conf_data)
580 # For shellcheck on scripts
584 # For gtk builder checks on .ui files
587 # Process subdirs before the sources
591 # Generate the executable
594 # Generate the help files
597 # Install other project files
599 cmd = [find_program('gen_changelog.sh'), meson.current_source_dir(), meson.current_build_dir()]
602 input: 'ChangeLog.gqview',
603 output: ['ChangeLog', 'ChangeLog.html'],
606 install_dir: helpdir)
607 meson.add_dist_script(cmd)
608 summary({'ChangeLog' : ['ChangeLog, ChangeLog.html created:', true]}, section : 'Documentation', bool_yn : true)
609 elif fs.exists('ChangeLog.html')
610 install_data('ChangeLog', 'ChangeLog.html', install_dir: helpdir)
611 summary({'ChangeLog' : ['ChangeLog, ChangeLog.html installed from dist:', true]}, section : 'Documentation', bool_yn : true)
614 install_data('geeqie.png', install_dir : icondir)
615 install_data('geeqie.1', install_dir : mandir1)
618 input : 'org.geeqie.Geeqie.desktop.in',
619 output : 'org.geeqie.Geeqie.desktop',
623 install_dir : join_paths(datadir, 'applications'))
626 input : 'org.geeqie.Geeqie.appdata.xml.in',
627 output : 'org.geeqie.Geeqie.appdata.xml',
631 install_dir : appdatadir)
633 configure_file(input: 'geeqie.spec.in', output: 'geeqie.spec', configuration: conf_data)
635 # Basic test of the executable
636 # is_parallel false is to avoid problems with images tests
637 xvfb = find_program('xvfb-run', required : false)
639 test('Basic test', xvfb, args: ['--auto-servernum', geeqie_exe, '--version'], is_parallel : false, timeout: 100)
640 summary({'xvfb' : ['Test runs:', true]}, section : 'Testing', bool_yn : true)
642 summary({'xvfb' : ['Test runs:', false]}, section : 'Testing', bool_yn : true)
645 # The tests are run on GitHub with all options disabled, and then
646 # with all or most options enabled. Shellcheck and GtkBuilder need only
647 # be run once when options are disabled. Use option archive as a flag.
648 # Image tests use option devel as a flag so that normal users do not
649 # download the test image database.
652 option = get_option('archive')
654 shellcheck_exe = find_program('shellcheck', required : false)
655 script_sources += files('gen_changelog.sh',
656 'geeqie-install-debian.sh',
659 if shellcheck_exe.found()
660 foreach script : script_sources
661 script_path = '@0@'.format(script)
662 test('Shellcheck_ ' + script_path, shellcheck_exe, args: ['--norc', '--shell=sh', '--enable=add-default-case,avoid-nullary-conditions,check-unassigned-uppercase,deprecate-which,quote-safe-variables', script], timeout: 100)
664 summary({'shellcheck' : ['Test runs:', true]}, section : 'Testing', bool_yn : true)
666 summary({'shellcheck' : ['Test runs:', false]}, section : 'Testing', bool_yn : true)
669 summary({'shellcheck' : ['Test runs:', false]}, section : 'Testing', bool_yn : true)
672 # GtkBuilder .ui check
673 option = get_option('archive')
676 gtk_builder_tool = find_program('gtk-builder-tool', required : false)
677 if gtk_builder_tool.found()
678 foreach ui_file : ui_sources
679 ui_path = '@0@'.format(ui_file)
680 test('UI Build_ ' + ui_path, xvfb, args: ['--auto-servernum', gtk_builder_tool.full_path(), 'validate', ui_file], timeout: 100)
682 summary({'gtk-builder-tool' : ['Test runs:', true]}, section : 'Testing', bool_yn : true)
684 summary({'gtk-builder-tool' : ['Test runs:', false]}, section : 'Testing', bool_yn : true)
687 summary({'gtk-builder-tool xvfb' : ['Test runs:', false]}, section : 'Testing', bool_yn : true)
690 summary({'gtk-builder-tool xvfb' : ['Test runs:', false]}, section : 'Testing', bool_yn : true)
694 option = get_option('devel')
697 get_test_images_sh = find_program('get-test-images.sh', dirs : scriptsdir, required : true)
698 image_test_sh = find_program('image-test.sh', dirs : scriptsdir, required : true)
700 images_dir = join_paths(meson.current_build_dir(), 'test-images.p')
702 message('Downloading test images')
703 sources_list = run_command(get_test_images_sh, images_dir, 'https://github.com/caclark/geeqie-test.git', check: true)
705 sources = sources_list.stdout().strip().split('\n')
707 foreach image : sources
708 image_path = '@0@'.format(image)
709 path_array = image_path.split('/')
710 image_name = path_array[path_array.length() - 1]
712 if image_name.startswith('fail')
713 test('Image_ ' + image_name, image_test_sh, args: [geeqie_exe, image], is_parallel : false, should_fail : true, timeout: 100)
715 test('Image_ ' + image_name, image_test_sh, args: [geeqie_exe, image], is_parallel : false, timeout: 100)
718 summary({'Image tests' : ['Test runs:', true]}, section : 'Testing', bool_yn : true)
720 summary({'Image tests' : ['Test runs:', false]}, section : 'Testing', bool_yn : true)
723 summary({'Image tests' : ['Test runs:', false]}, section : 'Testing', bool_yn : true)
726 # Code correctness checks
728 clang_tidy_exe = find_program('clang-tidy', required : false)
729 if clang_tidy_exe.found()
730 git_exe = find_program('git', required : true)
731 modified_file_list = run_command(git_exe, 'diff', '--name-only', check: true)
732 modified_files = modified_file_list.stdout().strip().split('\n')
734 foreach modified_file : modified_files
735 if modified_file.endswith('.cc')
736 modified_file_path = '@0@'.format(modified_file)
737 path_array = modified_file_path.split('/')
738 modified_file_name = path_array[path_array.length() - 1]
739 modified_file_full_path = join_paths(meson.project_source_root(), modified_file)
741 test('Code Correctness_ ' + modified_file_name, clang_tidy_exe, args : ['-p', './build', '-quiet', modified_file_full_path], timeout : 100)
745 summary({'Code Correctness' : ['Test runs:', true]}, section : 'Testing', bool_yn : true)
747 summary({'Code Correctness' : ['Test runs:', false]}, section : 'Testing', bool_yn : true)
750 summary({'Code Correctness' : ['Test runs:', false]}, section : 'Testing', bool_yn : true)
754 option = get_option('lua')
755 if not option.disabled()
758 lua_test_sh = find_program('lua-test.sh', dirs : scriptsdir, required : true)
759 test('Lua test', lua_test_sh, args: [geeqie_exe], is_parallel : false, timeout: 100)
761 summary({'lua' : ['Test runs:', true]}, section : 'Testing', bool_yn : true)
763 summary({'lua' : ['Test runs:', false]}, section : 'Testing', bool_yn : true)
766 summary({'lua' : ['Test runs:', false]}, section : 'Testing', bool_yn : true)
769 summary({'lua' : ['Test runs:', false]}, section : 'Testing', bool_yn : true)