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 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 : '>=0.53.0',
48 default_options : ['warning_level=3', 'buildtype=debugoptimized']
51 # To inhibit warnings from the generated files icons_inline.h and ui_icons.h
52 add_global_arguments('-Wno-overlength-strings', language : 'c')
54 # To compile originally-C files as C++
55 add_global_arguments('-fpermissive', language : 'cpp')
56 add_global_arguments('-Wno-error=deprecated-declarations', language : 'cpp')
57 add_global_arguments('-Wno-error=sign-compare', language : 'cpp')
58 add_global_arguments('-Wno-error=return-type', language : 'cpp')
59 add_global_arguments('-Wno-error=literal-suffix', language : 'cpp')
60 add_global_arguments('-Wno-error=write-strings', language : 'cpp')
62 # Project requirements
64 gnome = import('gnome')
65 thread_dep = dependency('threads')
66 cc = meson.get_compiler('c')
69 configuration_inc = include_directories('.')
72 gdk_pixbuf_csource = find_program('gdk-pixbuf-csource', required : true)
73 glib_genmarshal = find_program('glib-genmarshal', required : true)
75 option = get_option('git')
76 if not option.disabled()
77 running_from_git = find_program('git', required: false).found() and fs.is_dir('.git')
79 running_from_git = false
80 summary({'git' : ['disabled - ChangeLog, ChangeLog.html, lua-api help file created:', false]}, section : 'Documentation', bool_yn : true)
83 debug = get_option('debug')
85 # Note that main.cc sets prefix to the directory above where the executable is run from.
86 # This is to allow AppImages to be used
88 # These gq_* variables are paths relative to /prefix/,
89 # and are also used in defines in the source as GQ_*
90 if get_option('gq_appdir') == ''
91 gq_appdir = join_paths(get_option('datadir'), 'geeqie')
93 gq_appdir = get_option('gq_appdir')
96 # This is not the same as Meson bindir
97 if get_option('gq_bindir') == ''
98 gq_bindir = 'lib/geeqie'
100 gq_bindir = get_option('gq_bindir')
103 if get_option('gq_helpdir') == ''
104 gq_helpdir = join_paths(get_option('datadir'), 'doc/geeqie')
106 gq_helpdir = get_option('gq_helpdir')
109 if get_option('gq_htmldir') == ''
110 gq_htmldir = join_paths(get_option('datadir'), 'doc/geeqie/html')
112 gq_htmldir = get_option('gq_htmldir')
115 if get_option('gq_localedir') == ''
116 gq_localedir = join_paths(get_option('datadir'), 'locale')
118 gq_localedir = get_option('gq_localedir')
122 # Set up the absolute directory paths used
123 prefix = get_option('prefix')
124 datadir = join_paths(prefix, get_option('datadir'))
126 # Installation paths are absolute
127 appdir = join_paths(prefix, gq_appdir)
128 appdatadir = join_paths(datadir, 'metainfo')
129 desktopdir = join_paths(datadir, meson.project_name(), 'applications')
130 helpdir = join_paths(prefix, gq_helpdir)
131 htmldir = join_paths(prefix, gq_htmldir)
132 icondir = join_paths(datadir, 'pixmaps')
133 mandir1 = join_paths(datadir, 'man', 'man1')
135 podir = join_paths(meson.source_root(), 'po')
137 summary({'gq_appdir': gq_appdir,
138 'gq_bindir': gq_helpdir,
139 'gq_helpdir': gq_helpdir,
140 'gq_htmldir': gq_htmldir,
141 'gq_localedir': gq_localedir,
142 }, section: 'Directories')
144 # Create the define constants used in the sources. Set via config.h.in
145 conf_data = configuration_data()
146 conf_data.set_quoted('VERSION', meson.project_version())
147 conf_data.set('DEBUG', debug)
149 gtk_dep = dependency('gtk+-3.0', version : '>=3.22', required: true)
150 glib_dep = dependency('glib-2.0', version : '>=2.52', required: true)
153 req_version = '>=3.4.0'
154 option = get_option('archive')
155 if not option.disabled()
156 libarchive_dep = dependency('libarchive', version : req_version, required : get_option('archive'))
157 if libarchive_dep.found()
158 conf_data.set('HAVE_ARCHIVE', 1)
159 summary({'archive' : ['archive files e.g. .zip supported:', true]}, section : 'Configuration', bool_yn : true)
161 summary({'archive' : ['libarchive ' + req_version + ' not found - archive files e.g. .zip supported::', false]}, section : 'Configuration', bool_yn : true)
164 summary({'archive' : ['disabled - archive files e.g. .zip supported:', false]}, section : 'Configuration', bool_yn : true)
168 req_version = '>=2.0'
169 option = get_option('cms')
170 if not option.disabled()
171 xxd = find_program('xxd', 'xxdi.pl', required : false)
173 lcms_dep = dependency('lcms2', version : req_version, required : get_option('cms'))
175 conf_data.set('HAVE_LCMS', 1)
176 conf_data.set('HAVE_LCMS2', 1)
177 summary({'cms' : ['color management supported:', true]}, section : 'Configuration', bool_yn : true)
179 summary({'cms' : ['lcms2' + req_version + ' not found - color management supported:', false]}, section : 'Configuration', bool_yn : true)
182 summary({'cms' : ['xxd or xxdi.pl not found - color management supported:', false]}, section : 'Configuration', bool_yn : true)
185 summary({'cms' : ['disabled - color management supported:', false]}, section : 'Configuration', bool_yn : true)
189 req_version = '>=2.5.27'
190 option = get_option('djvu')
191 if not option.disabled()
192 ddjvuapi_dep = dependency('ddjvuapi', version : req_version, required : get_option('djvu'))
193 if ddjvuapi_dep.found()
194 conf_data.set('HAVE_DJVU', 1)
195 summary({'djvu' : ['djvu files supported:', true]}, section : 'Configuration', bool_yn : true)
197 summary({'djvu' : ['ddjvuapi ' + req_version + ' not found - djvu files supported:', false]}, section : 'Configuration', bool_yn : true)
200 summary({'djvu' : ['disabled - djvu files supported:', false]}, section : 'Configuration', bool_yn : true)
203 option = get_option('evince')
204 if not option.disabled()
205 evince = find_program('evince', required : false)
207 summary({'print preview' : ['print preview supported:', true]}, section : 'Configuration', bool_yn : true)
209 summary({'print preview' : ['evince not found - print preview supported:', false]}, section : 'Configuration', bool_yn : true)
212 summary({'print preview' : ['disabled - print preview supported:', false]}, section : 'Configuration', bool_yn : true)
216 req_version = '>=0.11'
217 option = get_option('exiv2')
218 if not option.disabled()
219 exiv2_dep = dependency('exiv2', version : req_version, required : get_option('exiv2'))
221 conf_data.set('HAVE_EXIV2', 1)
222 summary({'exiv2' : ['image metadata processed by exiv2:', true]}, section : 'Configuration', bool_yn : true)
224 summary({'exiv2' : ['exiv2 ' + req_version + ' not found - image data not processed by exiv2:', false]}, section : 'Configuration', bool_yn : true)
227 summary({'exiv2' : ['disabled - image data processed by exiv2:', false]}, section : 'Configuration', bool_yn : true)
231 champlain_gtk_dep = []
234 req_version_champlain = '>=0.12'
235 req_version_champlain_gtk = '>=0.12'
236 req_version_clutter = '>=1.0'
237 req_version_clutter_gtk = '>=1.0'
238 option = get_option('gps-map')
239 if not option.disabled()
240 champlain_dep = dependency('champlain-0.12', version : req_version_champlain, required : get_option('gps-map'))
241 champlain_gtk_dep = dependency('champlain-gtk-0.12', version : req_version_champlain_gtk, required : get_option('gps-map'))
242 if champlain_dep.found() and champlain_gtk_dep.found()
243 clutter_dep = dependency('clutter-1.0', version : req_version_clutter, required : get_option('gps-map'))
244 clutter_gtk_dep = dependency('clutter-gtk-1.0', version : req_version_clutter_gtk, required : get_option('gps-map'))
245 if clutter_dep.found() and clutter_gtk_dep.found()
246 conf_data.set('HAVE_CLUTTER', 1)
247 conf_data.set('HAVE_LIBCHAMPLAIN', 1)
248 conf_data.set('HAVE_LIBCHAMPLAIN_GTK', 1)
249 summary({'gps-map' : ['GPS map displayed', true]}, section : 'Configuration', bool_yn : true)
251 if not clutter_dep.found()
252 summary({'gps-map-clutter' : ['clutter-1.0 ' + req_version_clutter + ' not found - GPS map displayed:', false]}, section : 'Configuration', bool_yn : true)
254 if not clutter_gtk_dep.found()
255 summary({'gps-map-clutter-gtk' : ['clutter-gtk-1.0 ' + req_version_clutter_gtk + ' not found - GPS map displayed:', false]}, section : 'Configuration', bool_yn : true)
259 if not champlain_dep.found()
260 summary({'gps-map-champlain' : ['champlain-0.12 ' + req_version_champlain + ' not found - GPS map displayed:', false]}, section : 'Configuration', bool_yn : true)
262 if not champlain_gtk_dep.found()
263 summary({'gps-map-champlain-gtk' : ['champlain-gtk-0.12 ' + req_version_champlain_gtk + ' not found - GPS map displayed:', false]}, section : 'Configuration', bool_yn : true)
267 summary({'gps-map' : ['disabled - GPS map displayed:', false]}, section : 'Configuration', bool_yn : true)
271 req_version = '>=1.3.2'
272 option = get_option('heif')
273 if not option.disabled()
274 libheif_dep = dependency('libheif', version : req_version, required : get_option('heif'))
275 if libheif_dep.found()
276 conf_data.set('HAVE_HEIF', 1)
277 summary({'heif' : ['heif files supported:', true]}, section : 'Configuration', bool_yn : true)
279 summary({'heif' : ['libheif ' + req_version + ' not found - heif files supported:', false]}, section : 'Configuration', bool_yn : true)
282 summary({'heif' : ['disabled - heif files supported:', false]}, section : 'Configuration', bool_yn : true)
286 req_version = '>=2.3.0'
287 option = get_option('j2k')
288 if not option.disabled()
289 libopenjp2_dep = dependency('libopenjp2', version : req_version, required : get_option('j2k'))
290 if libopenjp2_dep.found()
291 conf_data.set('HAVE_J2K', 1)
292 summary({'j2k' : ['j2k files supported:', true]}, section : 'Configuration', bool_yn : true)
294 summary({'j2k' : ['libopenjp2 ' + req_version + ' not found - j2k files supported:', false]}, section : 'Configuration', bool_yn : true)
297 summary({'j2k' : ['disabled - j2k files supported:', false]}, section : 'Configuration', bool_yn : true)
301 option = get_option('jpeg')
302 if not option.disabled()
303 libjpeg_dep = dependency('libjpeg', required : get_option('jpeg'))
304 if libjpeg_dep.found()
305 if cc.has_function('jpeg_destroy_decompress', dependencies : libjpeg_dep)
306 conf_data.set('HAVE_JPEG', 1)
307 summary({'jpeg' : ['jpeg files supported:', true]}, section : 'Configuration', bool_yn : true)
309 summary({'jpeg' : ['jpeg_destroy_decompress not found - jpeg files supported:', false]}, section : 'Configuration', bool_yn : true)
312 summary({'jpeg' : ['libjpeg: not found', false]}, section : 'Configuration', bool_yn : true)
315 summary({'jpeg' : ['disabled - jpeg files supported:', false]}, section : 'Configuration', bool_yn : true)
319 req_version = '>=0.3.7'
320 option = get_option('jpegxl')
321 if not option.disabled()
322 libjxl_dep = dependency('libjxl', version : req_version, required : get_option('jpegxl'))
323 if libjxl_dep.found()
324 conf_data.set('HAVE_JPEGXL', 1)
325 summary({'jpegxl' : ['jpegxl files supported:', true]}, section : 'Configuration', bool_yn : true)
327 summary({'jpegxl' : ['libjxl ' + req_version + ' not found - jpegxl files supported:', false]}, section : 'Configuration', bool_yn : true)
330 summary({'jpegxl' : ['disabled - jpegxl files supported:', false]}, section : 'Configuration', bool_yn : true)
334 req_version = '>=0.20'
335 option = get_option('libraw')
336 if not option.disabled()
337 libraw_dep = dependency('libraw', version : req_version, required : get_option('libraw'))
338 if libraw_dep.found()
339 conf_data.set('HAVE_RAW', 1)
340 summary({'libraw' : ['.cr3 files supported:', true]}, section : 'Configuration', bool_yn : true)
342 summary({'libraw' : ['libraw ' + req_version + ' not found - .cr3 files supported:', false]}, section : 'Configuration', bool_yn : true)
345 summary({'libraw' : ['disabled - .cr3 files supported:', false]}, section : 'Configuration', bool_yn : true)
349 req_version = '>=5.3'
350 option = get_option('lua')
351 if not option.disabled()
352 foreach name : ['lua', 'lua5.3', 'lua-5.3', 'lua53']
353 lua_dep = dependency(name, version: req_version, required: get_option('lua'))
359 conf_data.set('HAVE_LUA', 1)
360 summary({'lua' : ['lua supported:', true]}, section : 'Configuration', bool_yn : true)
362 summary({'lua' : ['lua ' + req_version + ' not found - lua supported:', false]}, section : 'Configuration', bool_yn : true)
365 summary({'lua' : ['disabled - lua supported:', false]}, section : 'Configuration', bool_yn : true)
368 option = get_option('pandoc')
369 if not option.disabled()
370 pandoc = find_program('pandoc', required : false)
372 readme_html = custom_target(
375 output: 'README.html',
376 command: [pandoc, '@INPUT@', '-o', '@OUTPUT@'],
378 install_dir: helpdir)
380 summary({'README' : ['README.html created:', true]}, section : 'Documentation', bool_yn : true)
382 summary({'README' : ['pandoc not found - README.html created:', false]}, section : 'Documentation', bool_yn : true)
384 install_data('README.md', 'COPYING', 'TODO', 'AUTHORS', install_dir : helpdir)
386 summary({'pandoc' : ['disabled - README.html created:', false]}, section : 'Documentation', bool_yn : true)
389 poppler_glib_dep = []
390 req_version = '>=0.62'
391 option = get_option('pdf')
392 if not option.disabled()
393 poppler_glib_dep = dependency('poppler-glib', version : req_version, required : get_option('pdf'))
394 if poppler_glib_dep.found()
395 conf_data.set('HAVE_PDF', 1)
396 summary({'pdf' : ['pdf files supported:', true]}, section : 'Configuration', bool_yn : true)
398 summary({'pdf' : ['poppler-glib ' + req_version + ' not found - pdf files supported:', false]}, section : 'Configuration', bool_yn : true)
401 summary({'pdf' : ['disabled - pdf files supported:', false]}, section : 'Configuration', bool_yn : true)
405 req_version = '>=1.6'
406 option = get_option('spell')
407 if not option.disabled()
408 gspell_dep = dependency('gspell-1', version : req_version, required: get_option('spell'))
409 if gspell_dep.found()
410 conf_data.set('HAVE_SPELL', 1)
411 summary({'spell' : ['spelling checks enabled', true]}, section : 'Configuration', bool_yn : true)
413 summary({'spell' : ['gspell-1 ' + req_version + ' not found - spelling checks enabled', false]}, section : 'Configuration', bool_yn : true)
416 summary({'spell' : ['disabled - spelling checks enabled', false]}, section : 'Configuration', bool_yn : true)
420 option = get_option('tiff')
421 if not option.disabled()
422 tiff_dep = dependency('libtiff-4', required: get_option('tiff'))
424 if cc.has_function('TIFFClientOpen', dependencies : tiff_dep)
425 conf_data.set('HAVE_TIFF', 1)
426 summary({'tiff' : ['tiff files supported:', true]}, section : 'Configuration', bool_yn : true)
428 summary({'tiff' : ['TIFFClientOpen not found - tiff files supported:', false]}, section : 'Configuration', bool_yn : true)
431 summary({'tiff' : ['libtiff not found - tiff files supported:', false]}, section : 'Configuration', bool_yn : true)
434 summary({'tiff' : ['disabled - tiff files supported:', false]}, section : 'Configuration', bool_yn : true)
437 libffmpegthumbnailer_dep = []
438 req_version = '>=2.1.0'
439 option = get_option('videothumbnailer')
440 if not option.disabled()
441 libffmpegthumbnailer_dep = dependency('libffmpegthumbnailer',
442 version : req_version,
443 required : get_option('videothumbnailer'))
445 if libffmpegthumbnailer_dep.found()
446 conf_data.set('HAVE_FFMPEGTHUMBNAILER', 1)
447 summary({'videothumbnailer' : ['thumbnails of video files supported:', true]}, section : 'Configuration', bool_yn : true)
449 result = cc.has_member('struct video_thumbnailer_struct', 'prefer_embedded_metadata', prefix : '#include <libffmpegthumbnailer/videothumbnailerc.h>')
451 conf_data.set('HAVE_FFMPEGTHUMBNAILER_METADATA', 1)
453 summary({'fmpegthumbnailer_metadata' : ['fmpegthumbnailer_metadata found:', result]}, section : 'Thumbnailer', bool_yn : true)
455 result = cc.has_member('struct image_data_struct', 'image_data_width', prefix : '#include <libffmpegthumbnailer/videothumbnailerc.h>' )
457 conf_data.set('HAVE_FFMPEGTHUMBNAILER_RGB', 1)
459 summary({'fmpegthumbnailer_rgb' : ['fmpegthumbnailer_rgb found:', result]}, section : 'Thumbnailer', bool_yn : true)
461 result = cc.has_function('video_thumbnailer_set_size', dependencies : libffmpegthumbnailer_dep)
463 conf_data.set('HAVE_FFMPEGTHUMBNAILER_WH', 1)
465 summary({'fmpegthumbnailer_set_size' : ['fmpegthumbnailer_set_size found:', result]}, section : 'Thumbnailer', bool_yn : true)
467 summary({'videothumbnailer' : ['libvideothumbnailer ' + req_version + ' not found - thumbnails of video files supported', false]}, section : 'Configuration', bool_yn : true)
470 summary({'videothumbnailer' : ['disabled -thumbnails of video files supported', false]}, section : 'Configuration', bool_yn : true)
474 req_version = '>=0.6.1'
475 option = get_option('webp')
476 if not option.disabled()
477 libwebp_dep = dependency('libwebp', version : req_version, required : get_option('webp'))
478 if libwebp_dep.found()
479 conf_data.set('HAVE_WEBP', 1)
480 summary({'webp' : ['webp files supported:', true]}, section : 'Configuration', bool_yn : true)
482 summary({'webp' : ['libwebp ' + req_version + ' not found - webp files supported:', false]}, section : 'Configuration', bool_yn : true)
485 summary({'webp' : ['disabled - webp files supported:', false]}, section : 'Configuration', bool_yn : true)
488 # Check for nl_langinfo and _NL_TIME_FIRST_WEEKDAY
489 code = '''#include <langinfo.h>
491 int main (int argc, char ** argv) {
493 c = nl_langinfo(_NL_TIME_FIRST_WEEKDAY);
496 if cc.links(code, name : 'nl_langinfo and _NL_TIME_FIRST_WEEKDAY')
497 conf_data.set('HAVE__NL_TIME_FIRST_WEEKDAY', 1)
498 summary({'nl_langinfo' : ['first weekday depends on locale:', true]}, section : 'Documentation', bool_yn : true)
500 summary({'nl_langinfo' : ['nl_langinfo not found - first weekday depends on locale:', false, 'first weekday defaults to Monday']}, section : 'Documentation', bool_yn : true)
503 # Required only for backtrace debugging
504 result = cc.check_header('execinfo.h')
506 conf_data.set('HAVE_EXECINFO_H', 1)
509 conf_data.set_quoted('GETTEXT_PACKAGE', meson.project_name())
510 conf_data.set_quoted('GQ_APPDIR', gq_appdir)
511 conf_data.set_quoted('GQ_BINDIR', gq_bindir)
512 conf_data.set_quoted('GQ_HELPDIR', gq_helpdir)
513 conf_data.set_quoted('GQ_HTMLDIR', gq_htmldir)
514 conf_data.set_quoted('GQ_LOCALEDIR', gq_localedir)
516 conf_data.set_quoted('PACKAGE', meson.project_name())
517 conf_data.set_quoted('PACKAGE_NAME', meson.project_name())
518 conf_data.set_quoted('PACKAGE_STRING', meson.project_version())
519 conf_data.set_quoted('PACKAGE_TARNAME', meson.project_name())
520 conf_data.set_quoted('PACKAGE_VERSION', meson.project_version())
521 conf_data.set_quoted('VERSION', meson.project_version())
523 configure_file(input : 'config.h.in',
526 configuration : conf_data)
528 # Process subdirs before the sources
532 # Generate the executable
535 # Generate the help files
538 # Install other project files
540 cmd = [find_program('gen_changelog.sh'), meson.current_source_dir(), meson.current_build_dir()]
543 input: 'ChangeLog.gqview',
544 output: ['ChangeLog', 'ChangeLog.html'],
547 install_dir: helpdir)
548 meson.add_dist_script(cmd)
549 summary({'ChangeLog' : ['ChangeLog, ChangeLog.html created:', true]}, section : 'Documentation', bool_yn : true)
550 elif fs.exists('ChangeLog.html')
551 install_data('ChangeLog', 'ChangeLog.html', install_dir: helpdir)
552 summary({'ChangeLog' : ['ChangeLog, ChangeLog.html installed from dist:', true]}, section : 'Documentation', bool_yn : true)
555 install_data('geeqie.png', install_dir : icondir)
556 install_data('geeqie.1', install_dir : mandir1)
559 input : 'geeqie.desktop.in',
560 output : 'geeqie.desktop',
564 install_dir : join_paths(datadir, 'applications'))
567 input : 'org.geeqie.Geeqie.appdata.xml.in',
568 output : 'org.geeqie.Geeqie.appdata.xml',
572 install_dir : appdatadir)
574 configure_file(input: 'geeqie.spec.in', output: 'geeqie.spec', configuration: conf_data)