Compare commits

...

69 Commits

Author SHA1 Message Date
nou d288810d5d Fix saving image 2024-08-16 15:16:37 +02:00
nou fb66e82428 Workaround for huge PCL keywords 2024-08-16 15:04:25 +02:00
nou 71486efeef Remove unused method 2024-06-17 09:47:56 +02:00
nou 8213f6213f Fix convert scripts 2024-06-17 09:44:48 +02:00
nou f8c9fec77e Add scripts 2024-06-16 16:31:53 +02:00
nou af4be850cb Embeded scripts 2024-06-16 00:14:51 +02:00
nou ca1a13ed9d Solve deprectation warnings 2024-06-15 17:45:30 +02:00
nou 1873da6c49 Fix issue in thumbnails 2024-06-11 17:09:35 +02:00
nou 92345f82ca Update metainfo 2024-06-10 22:39:52 +02:00
nou 3c8f49e932 Fix some bugs in scripting 2024-06-10 18:55:13 +02:00
nou 37dd97e361 Fix compilation error 2024-06-10 16:46:21 +02:00
nou da31187aa3 Fix small typo in help 2024-06-09 17:15:53 +02:00
nou 4801338160 Fix compilation error 2024-06-09 15:47:30 +02:00
nou fb9d026ff5 Open script folder on MacOS 2024-06-09 15:42:31 +02:00
nou c2810faf8f Update french translation 2024-06-09 13:56:04 +02:00
nou d3d302fd38 Fix some typos in help 2024-06-08 23:02:13 +02:00
nou a0497c7d19 Update translations 2024-06-08 21:19:14 +02:00
nou 8818e25eda Help documentation for batch processing 2024-06-08 20:21:46 +02:00
nou c3baa18087 Fix text color in log 2024-06-08 20:19:48 +02:00
nou 66f0c05a48 Fix calling GUI methods from script thread 2024-06-08 20:11:25 +02:00
nou 461ffea874 Default params for convert() 2024-06-08 19:42:21 +02:00
nou 7535ad87e7 Batchprocessing improvments 2024-06-06 12:00:00 +02:00
nou 273aef1594 Add getInt getString getFloat methods to scripting 2024-06-05 22:27:35 +02:00
nou 9519c9830c Improve text coloring in log 2024-06-04 16:41:50 +02:00
nou 342e5cc5db Add FITSRecordModify for XISF files 2024-06-04 16:41:25 +02:00
nou ae84cbdfe0 Add modifing FITS records 2024-04-12 09:58:21 +02:00
nou 933fd4a2a0 Additional work on batch processing 2024-03-29 18:08:57 +01:00
nou c3588e1c36 Rate limit conversion from script 2024-03-26 14:54:19 +01:00
nou 174134a9ee Skip dummy HDU in compressed FITS 2024-03-25 22:53:51 +01:00
nou bbc13ec8a5 Add compression parameters 2024-03-25 22:53:13 +01:00
nou 9f7e2ab6b4 Add convert function to script 2024-03-25 20:25:47 +01:00
nou 4fe56acbd9 Fix bug when saving color FITS/XISF files 2024-03-24 23:55:50 +01:00
nou f35db9d1af Small fixes 2024-03-24 18:39:46 +01:00
nou 81d138f799 Add bayer mask icons 2024-02-12 17:57:35 +01:00
nou ae07d4793b Draw only visible filenames in thumbnails 2024-02-11 13:52:52 +01:00
nou dc2a781d3b Add calculating stats with script 2024-02-04 00:11:31 +01:00
nou 90035f44ed Refractor LoadRunable 2024-02-04 00:09:46 +01:00
nou 53c9a58125 Prevent symlink loop when indexing 2024-02-03 15:32:34 +01:00
nou 3f7e3689e8 Prevent symlink loops 2024-02-02 22:41:44 +01:00
nou af9187737f Add recursive directory 2024-02-02 20:55:58 +01:00
nou 4e952873e3 Fix metainfo 2024-02-02 00:09:42 +01:00
nou fb24800050 Add DBus for MacOS to fix build issue 2024-02-01 23:27:40 +01:00
nou ea0dcc226a Update metainfo 2024-02-01 23:26:08 +01:00
nou 6a7b677b95 Translatiotions 2024-02-01 23:22:35 +01:00
nou 0cee4c9c53 Add thumbnail quality to settings 2024-02-01 23:03:21 +01:00
nou d5f2351905 Only degress should show negative sign 2024-01-22 21:32:15 +01:00
nou 18732a8cbf Limit image info to 1024 characters 2024-01-22 21:31:28 +01:00
nou 8c9c1d8d06 Flip image according to ROWORDER 2024-01-22 21:30:39 +01:00
nou e5f425ff8d Fix some edge cases when stretch 2024-01-18 16:10:11 +01:00
nou 428f9c360a Small optimization in resample 2024-01-15 08:58:14 +01:00
nou a8a1509db7 Show error message in main window when image fail to load 2024-01-14 14:32:01 +01:00
nou 6539c78c57 Add box resize algorithm 2024-01-14 14:28:28 +01:00
nou 0e0d29320e Set unlimited image size so it doesn't fail to load big images 2024-01-14 14:04:54 +01:00
nou 1efe8e6974 Fix date in meta info 2024-01-11 13:30:30 +01:00
nou dae10182d1 Fix SSE instricts ifdef 2024-01-09 15:40:27 +01:00
nou ed5fc9c1c2 Increase number max preload images to 32 2024-01-08 16:52:46 +01:00
nou cd6a64a98b Additional work on batch processing 2024-01-08 15:44:05 +01:00
nou 67355a82b7 Add bayer mask selection 2024-01-08 15:43:21 +01:00
nou 8fc2078a3a Don't skip images before they load 2024-01-05 13:37:13 +01:00
nou da9b389409 Add slideshow 2024-01-05 13:36:06 +01:00
nou 7818b8d3e9 Fix icon instalation 2023-12-31 16:04:32 +01:00
nou 11294bfcb0 Scripting module 2023-12-31 16:04:16 +01:00
nou faecb385aa Reorganize resources 2023-12-22 11:20:03 +01:00
nou e5be04926b Fix warnings 2023-12-20 11:31:55 +01:00
nou eaf2c7094b Migrate to Qt6 2023-12-20 11:31:51 +01:00
nou aef41f5f6b Fix Qt deprectation warnings 2023-12-18 16:03:04 +01:00
nou 2134f13b06 Add nearest and bicubic filtering 2023-12-18 15:54:15 +01:00
nou 0e9c980325 Add support for CR3 files 2023-11-25 18:06:38 +01:00
nou b9bf6bf183 Fixed scaling for int32 2023-11-21 18:31:07 +01:00
66 changed files with 3847 additions and 299 deletions
+18 -15
View File
@@ -17,7 +17,7 @@ if(SANITIZE_ADDRESS_LEAK)
set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -fsanitize=address -fsanitize=leak")
endif(SANITIZE_ADDRESS_LEAK)
find_package(Qt5 COMPONENTS Widgets Sql OpenGL REQUIRED)
find_package(Qt6 COMPONENTS Widgets Sql OpenGLWidgets Qml REQUIRED)
find_library(GSL_LIB gsl REQUIRED)
find_library(GSLCBLAS_LIB gslcblas REQUIRED)
find_library(EXIF_LIB exif REQUIRED)
@@ -29,6 +29,7 @@ add_subdirectory(libXISF)
set(TENMON_SRC
about.cpp about.h
batchprocessing.cpp batchprocessing.h batchprocessing.ui
database.cpp database.h
databaseview.cpp databaseview.h
delete.cpp
@@ -44,6 +45,7 @@ set(TENMON_SRC
markedfiles.cpp markedfiles.h
rawimage.cpp rawimage.h
rawimage_sse.cpp
scriptengine.cpp scriptengine.h
settingsdialog.cpp settingsdialog.h
starfit.cpp starfit.h
statusbar.cpp statusbar.h
@@ -52,28 +54,29 @@ set(TENMON_SRC
)
option(COLOR_MANAGMENT "Enable sRGB framebuffer support for gamma correct images and color profiles support" ON)
if(${Qt5Core_VERSION_STRING} VERSION_LESS "5.14")
set(COLOR_MANAGMENT OFF)
endif(${Qt5Core_VERSION_STRING} VERSION_LESS "5.14")
if(COLOR_MANAGMENT)
add_compile_definitions("COLOR_MANAGMENT")
endif(COLOR_MANAGMENT)
qt5_add_resources(TENMON_SRC resources.qrc)
qt_add_resources(TENMON_SRC resources/resources.qrc)
qt_add_resources(TENMON_SRC shaders/shaders.qrc)
qt_add_resources(TENMON_SRC scripts/scripts.qrc)
if(WIN32)
list(APPEND TENMON_SRC icon.rc)
list(APPEND TENMON_SRC resources/icon.rc)
set(tenmon_ICON "")
elseif(APPLE)
set(tenmon_ICON ${CMAKE_CURRENT_SOURCE_DIR}/tenmon.icns)
set(tenmon_ICON ${CMAKE_CURRENT_SOURCE_DIR}/resources/tenmon.icns)
find_package(Qt6 COMPONENTS DBus REQUIRED)
set_source_files_properties(${tenmon_ICON} PROPERTIES MACOSX_PACKAGE_LOCATION "Resources")
else()
set(tenmon_ICON "")
find_package(Qt6 COMPONENTS DBus REQUIRED)
find_package(PkgConfig REQUIRED)
pkg_search_module(GIO REQUIRED gio-2.0)
endif()
add_executable(tenmon WIN32 MACOSX_BUNDLE ${tenmon_ICON} ${TENMON_SRC})
qt_add_executable(tenmon WIN32 MACOSX_BUNDLE ${tenmon_ICON} ${TENMON_SRC})
find_path(FITS_INCLUDE fitsio2.h PATH_SUFFIXES cfitsio REQUIRED)
target_include_directories(tenmon PRIVATE ${FITS_INCLUDE} ${CMAKE_BINARY_DIR} ${libXISF_SOURCE_DIR})
@@ -82,16 +85,16 @@ if(UNIX AND NOT APPLE)
target_include_directories(tenmon PRIVATE ${GIO_INCLUDE_DIRS})
endif()
target_link_libraries(tenmon Qt5::Widgets Qt5::Sql ${GSL_LIB} ${GSLCBLAS_LIB} ${EXIF_LIB} ${FITS_LIB} ${RAW_LIB} ${WCS_LIB} XISF)
target_link_libraries(tenmon PRIVATE Qt6::Widgets Qt6::Sql Qt6::OpenGLWidgets Qt6::Qml ${GSL_LIB} ${GSLCBLAS_LIB} ${EXIF_LIB} ${FITS_LIB} ${RAW_LIB} ${WCS_LIB} XISF)
if(APPLE)
target_link_libraries(tenmon "-framework CoreFoundation")
else()
target_link_libraries(tenmon ${GIO_LDFLAGS})
target_link_libraries(tenmon PRIVATE Qt6::DBus "-framework CoreFoundation")
elseif(UNIX)
target_link_libraries(tenmon PRIVATE Qt6::DBus ${GIO_LDFLAGS})
endif(APPLE)
if(LIBRAW_STATIC)
add_compile_definitions("LIBRAW_NODLL")
target_link_libraries(tenmon jasper)
target_link_libraries(tenmon PRIVATE jasper)
endif()
install(TARGETS tenmon BUNDLE DESTINATION .)
@@ -102,8 +105,8 @@ if(UNIX AND NOT APPLE)
install(SCRIPT install.cmake)
else()
install(FILES space.nouspiro.tenmon.desktop DESTINATION "${CMAKE_INSTALL_DATADIR}/applications")
install(FILES space.nouspiro.tenmon.png DESTINATION "${CMAKE_INSTALL_DATADIR}/icons/hicolor/64x64/apps")
install(FILES space.nouspiro.tenmon_128.png DESTINATION "${CMAKE_INSTALL_DATADIR}/icons/hicolor/128x128/apps" RENAME space.nouspiro.tenmon.png)
install(FILES resources/space.nouspiro.tenmon.png DESTINATION "${CMAKE_INSTALL_DATADIR}/icons/hicolor/64x64/apps")
install(FILES resources/space.nouspiro.tenmon_128.png DESTINATION "${CMAKE_INSTALL_DATADIR}/icons/hicolor/128x128/apps" RENAME space.nouspiro.tenmon.png)
endif()
install(FILES space.nouspiro.tenmon.metainfo.xml DESTINATION "${CMAKE_INSTALL_DATADIR}/metainfo")
endif(UNIX AND NOT APPLE)
+4 -4
View File
@@ -2,20 +2,20 @@ FITS/XISF image viewer with multithreaded image loading
To get all dependencies install these packages
sudo apt install qtbase5-dev libraw-dev libexif-dev libcfitsio-dev libgsl-dev wcslib-dev libopencv-dev cmake
sudo apt install qt6-base-dev qt6-declarative-dev libqt6opengl6-dev libraw-dev libexif-dev libcfitsio-dev libgsl-dev wcslib-dev cmake
on OpenSUSE
sudo zypper install opencv-devel gsl-devel exif-devel libraw-devel wcslib-devel libqt5-qtbase-devel
sudo zypper install gsl-devel exif-devel libraw-devel wcslib-devel libqt6-qtbase-devel
MacOS X
To compile on MacOS install XCode first. Then install homebrew in x86_64 mode
with "arch -i x86_64". Building on native ARM is not supported.
homebrew install qt5 libraw cfitsio libexif libgsl wcslib opencv
homebrew install qt6 libraw cfitsio libexif libgsl wcslib
You may need to set CMAKE_PREFIX_PATH for Qt5 and OpenCV so CMake can find them.
You may need to set CMAKE_PREFIX_PATH for Qt6 so CMake can find them.
Then to build run standard cmake
+118 -1
View File
@@ -14,7 +14,7 @@ img { margin: 5px; }
<li>FITS 8, 16, 32 bit integer and 32, 64 bit float</li>
<li>XISF 8, 16, 32 bit integer and 32, 64 bit float</li>
<li>JPEG, PNG, BMP, GIF, PBM, PGM, PPM and SVG images</li>
<li>CR2, NEF, DNG raw images</li>
<li>CR2, CR3, NEF, DNG raw images</li>
</ul>
</p>
@@ -115,6 +115,123 @@ Pressing Enter or clicking on <i>Filter</i> button will filter out database reco
<br><img src=":/about/filter.png"><br>
This example filters for files where: "Bias" is in the file name, the OBJECT property is "M_42" (where the underscore can be any single character), and the DATE property begins with "2022".
</p>
<h3>Batch processing</h3>
This module allow to write scripts in JavaScript that process image files. Batch Processing window consist from three main parts. On top is list of input files and directories.
You can add directories or individual files to this list. Directories are scanned recursively to find all files even non image files. This list of files is then passed to script in array named <b>files</b>.
In script you can then iterate through files like this.
<pre>for(file of files)
{
if(file.suffix() == "fits")
{
core.log(file.fileName());
file.convert(file.relativeFilePath(), "XISF");
}
}
</pre>
<p>Bellow this list is output directory. All relative paths passed as arguments into methods like copy(), move() or convert() will be relative to this output directory. If you pass absolute path to methods then
this output directory is ignored.</p>
<p>Bellow that is list of scripts. These scripts are located in application data directory. Select script which you want to run by clicking on it. Clicking on <i>Open scripts</i> will open directory with these scripts where you create new and edit them.
Location of this directory is on Windows: "C:/Users/<USER>/AppData/Roaming/nou/Tenmon" Linux: "~/.local/share/nou/Tenmon/scripts" MacOS: "~/Library/Application Support/nou/Tenmon/scripts"</p>
<p>Next is Log windows that contain any messages that come from scripts. Mainly calls to <code>core.log()</code> At bottom there buttons that can start or stop execution of selected scripts.</p>
<h4>core</h4>
There is global object called <b>core</b> that have these methods.
<ul>
<li><b>log(message)</b> print message to log window.</li>
<li><b>mark(file)</b> mark file same way as in GUI. Takes object of type <i>File</i> as argument.</li>
<li><b>unmark(file)</b> unmark file same was as in GUI. Takes object of type <i>File</i> as argument.</li>
<li><b>isMarked(file)</b> check if file was marked. Takes object of type <i>File</i> as argument.</li>
<li><b>setMaxThread(maxthread)</b> set maximum number of concurrent thread when doing asynchronous task.</li>
<li><b>sync()</b> wait until all asynchronous tasks are done.</li>
<li><b>getString(label = "", text = "")</b> show dialog box to get string value from user. String value passed in first argument is used as description label. Second argument text is default value in text box.
Both parameters are optional so calling just <i>getString()</i> is valid. When cancel is pressed it return Undefined.</li>
<li><b>getInt(label = "", value = 0)</b> show dialog box with input box to retrieve integer value. String value passed in first argument is used as description label.
Second parameter is default value in input box. Both parameters are optional. When cancel is pressed it return Undefined.</li>
<li><b>getFloat(label = "", value = 0, decimals = 3)</b> show dialog box with input box to retrieve decimal value. String value passed in first argument is used as description label.
Second parameter is default value in input box. All three parameters are optional. When cancel is pressed it return Undefined.</li>
<li><b>getItem(items)</b> show selection dialog which allow to select one item from array of items. It return selected item as string. When cancel is pressed it return Undefined.</li>
</ul>
<h4>File</h4>
In <b>files</b> array there are instances of type <b>File</b> objects that have these methods.
<ul>
<li><b>fileName()</b> returns the name of the file, excluding the path.</li>
<li><b>absoluteFilePath()</b> returns an absolute path including the file name.</li>
<li><b>absolutePath()</b> returns an absolute path without the file name</li>
<li><b>relativeFilePath()</b> return relative path including file name relative to directory that was in list of directories to be scanned. For example you add C:/images as input directory. In this directory there
is file <i>C:/images/lights/red/M42_001.fits</i> then this method will return <i>lights/red/M42_001.fits</i></li>
<li><b>relativePath()</b> return same path as previous method just without file name. <i>lights/red</i></li>
<li><b>baseName()</b> return file name up to the first dot. For example for <i>some.file.name.fits</i> it will return <i>some</i></li>
<li><b>completeBaseName()</b> return file name up to the last dot. For example for <i>some.file.name.fits</i> it will return <i>some.file.name</i></li>
<li><b>suffix()</b> return string after last dot in file name. For example <i>fits</i></li>
<li><b>size()</b> return size of file in bytes.</li>
<li><b>fitsKeywords()</b> return array of strings with every keyword that is in header. <i>SIMPLE,BITPIX,NAXIS,NAXIS1,NAXIS2,EXTEND,COMMENT</i></li>
<li><b>fitsValue(key)</b> return value for keyword. In case that there is multiple occurrences it return last one.</li>
<li><b>fitsValues(key)</b> return array of values for keyword.</li>
<li><b>fitsRecords()</b> return array of objects with properties <b>key, value</b> and <b>comment</b> </li>
<li><b>modifyFITSRecords(FITSRecordModify)</b> modify FITS header by adding, removing or updating FITS record. Return true on success. Refer to <i>FITSRecordModify</i></li>
<li><b>isMarked()</b> return true if file is marked.</li>
<li><b>copy(newpath)</b> copy file to new location. It return instance of new <i>File<i> object that represent this copied file. This path can be relative or absolute. In case that <i>newpath</i> parameter is relative
path then it "Output directory" from GUI windows is used as base directory. Parameter <i>newpath</i> can absolute path. File is then copied to this path. In case that copy fail it return null.</li>
<li><b>move(newpath)</b> move file to new location. It return false if move failed. This can happend if destination is not writable but also if destination file already exist. This functions does not overwrite existing file.
This path can be relative or absolute. In case that <i>newpath</i> parameter is relative path then it "Output directory" from GUI windows is used as base directory. Parameter <i>newpath</i> can be absolute path.
File is then moved to this path.</li>
<li><b>convert(outpath, format, params)</b> convert image file from any format that program is able to open into FITS, XISF, JPEG, PNG, BMP.
Parameters are: <i>outputpath</i> path where converted image will be saved. It automatically replace suffix according to format. <i>format</i> one of "FITS" "XISF", "JPG", "PNG" or "BMP". <i>params</i> object with attributes "compressionType" and "compressionLevel".
Valid values for compressionType are be "gzip" or "rice" when converting to FITS. When converting to XISF compressionType can be "zlib", "lz4", "lz4hc", "zstd", "zlib+sh", "lz4+sh", "lz4hc+sh", "zstd+sh".
It is recommended to use "+sh" variants of compression.
XISF format also accept "compressionLevel" in range 0-100 where zero is fastest compression and 100 slowest. If you omit this attribute or set it to -1 then default compression level will be used.
It return new instance of <i>File</i> that point to converted file.
<pre>file.convert("converted_file.xisf", "xisf", {"compressionType": "zstd+sh", "compressionLevel": 70});
file.convert("converted_file.fits", "fits", {"compressionType": "rice"});
file.convert("converted_file.jpg", "png");</pre>
</li>
<li><b>convertAsync(outpath, format, params)</b> same as previous method but it does conversion in separated thread asynchronously and in parallel. Before calling any method on object returned by this method you must call
<code>core.sync();</code> to ensure that conversion is done and destination file exists.
<pre>let compression = {"compressionType": "zstd+sh"};
let convertedFiles = [];
for(file in files)
{
if(file.suffix() == "fits")
convertedFiles.push(file.convertAsync("xisf/" + file.fileName(), "XISF", compression));
}
core.sync(); // ensure that files exist
for(file of convertedFiles)// now we can iterate over the files
{
core.log(file.fileName() + " " + file.size()); // let print compressed file sizes
}</pre></li>
<li><b>stats()</b> calculate basic images statistics and return them as object with attributes "mean", "stddev", "median", "min", "max" and "mad".
<pre>let s = file.stats();
core.log("Median value is " + s.median);</pre></li>
</ul>
<h4>FITSRecordModify</h4>
This class is used to define modify operation FITS header in FITS and XISF files. It can remove update and add records. Order of operation is also remove then update and last add.
The keyword names may be up to 8 characters long and can only contain uppercase letters, the digits 0-9, the hyphen, and the underscore character.
<pre>let modify = new FITSRecordModify();
modify.updateKeyword("OBJECT", "M42");
modify.updateKeyword("MYTILE", "PART1", "adding custom keyword so WBPP can group it");
modify.removeKeyword("OBJECT");
// doesn't matter that it is specified as last. This will first remove
// existing OBJECT record and then add again OBJECT=M42
for(file in files)
{
file.modifyFITSRecords(modify);
}</pre>
<ul>
<li><b>new FITSRecordModify()</b> create new instance of object.</li>
<li><b>removeKeyword(key);</b> specify removing of record with <i>key</i> as keyword.</li>
<li><b>updateKeyword(key, value, comment = "")</b> specify updating existing keyword with value and comment. Comment is optional parameter. If record with keyword doesn't exist then it will add new one.
Unless you want to have multiple records with same keyword (for example HISTORY) always use this method and not addKeyword.</li>
<li><b>addKeyword(key, value, comment = "")</b> specify adding new keyword</li>
</ul>
<p><small>PS: Kanji in icon means astronomy in Japanese</small></p>
</body>
</html>
+109 -1
View File
@@ -14,7 +14,7 @@ img { margin: 5px; }
<li>FITS 8, 16 bit entier et 32 bit point flottant</li>
<li>XISF 8, 16 bit entier et 32 bit point flottant</li>
<li>images JPEG, PNG, BMP, GIF, PBM, PGM, PPM et SVG</li>
<li>images RAW CR2, NEF, DNG</li>
<li>images RAW CR2, CR3, NEF, DNG</li>
</ul>
</p>
@@ -101,6 +101,114 @@ En appuyant sur la touche Enter ou en cliquant sur le bouton <i>Filtre</i>, les
<br><img src=":/about/filter.png"><br>
Cet exemple filtre les fichiers où : "Bias" figure dans le nom de fichier, la propriété OBJECT est "M_42" (où le trait de soulignement peut être n'importe quel caractère) et la propriété DATE commence par "2022".
</p>
<h3>Traitement par lot</h3>
Ce module permet d'écrire des scripts en JavaScript qui traitent des fichiers images. La fenêtre de traitement par lots se compose de trois parties principales. En haut se trouve la liste des fichiers et répertoires d'entrée.
Vous pouvez ajouter des répertoires ou des fichiers individuels à cette liste. Les répertoires sont analysés de manière récursive pour trouver tous les fichiers, même les fichiers non image. Cette liste de fichiers est ensuite transmise au script dans un tableau nommé <b>files</b>.
Dans le script, vous pouvez ensuite parcourir les fichiers comme ici.
<pre>for(file of files)
{
if(file.suffix() == "fits")
{
core.log(file.fileName());
file.convert(file.relativeFilePath(), "XISF");
}
}
</pre>
<h4>core</h4>
Il existe un objet global appelé <b>core</b> qui possède ces méthodes.
<ul>
<li><b>log(message)</b> afficher le message dans la fenêtre du journal.</li>
<li><b>mark(file)</b> marquer le fichier de la même manière que dans l'interface graphique. Prend un objet de type <i>File</i> comme argument.</li>
<li><b>unmark(file)</b> décoche le fichier de la même manière que dans l'interface graphique. Prend un objet de type <i>File</i> comme argument.</li>
<li><b>isMarked(file)</b> vérifie si le fichier a été marqué. Prend un objet de type <i>File</i> comme argument.</li>
<li><b>setMaxThread(maxthread)</b> définir le nombre maximal de threads simultanés lors de l'exécution d'une tâche asynchrone.</li>
<li><b>sync()</b> attendre que toutes les tâches asynchrones soient terminées.</li>
<li><b>getString(label = "", text = "")</b> affiche la boîte de dialogue pour obtenir un text de l'utilisateur. La valeur text passée dans le premier argument est utilisée comme label de description. Le texte du deuxième argument est la valeur par défaut dans la zone de texte.
Les deux paramètres sont facultatifs, donc l'appel à <i>getString()</i> est valide. Lorsque vous appuyez sur Annuler, il renvoie Undefined</li>
<li><b>getInt(label = "", value = 0)</b> affiche une boîte de dialogue avec une zone de saisie pour récupérer une valeur entière. Le texte passé dans le premier argument est utilisé comme label de description.
Le deuxième paramètre est la valeur par défaut dans la zone de saisie. Les deux paramètres sont facultatifs. Lorsque vous appuyez sur Annuler, il renvoie Undefined.</li>
<li><b>getFloat(label = "", value = 0, decimals = 3)</b> affiche une boîte de dialogue avec une zone de saisie pour récupérer une valeur décimale. Le texte passé dans le premier argument est utilisé comme label de description.
Le deuxième paramètre est la valeur par défaut dans la zone de saisie. Les trois paramètres sont facultatifs. Lorsque vous appuyez sur Annuler, il renvoie Undefined.</li>
<li><b>getItem(items)</b> affiche une boîte de dialogue de sélection qui permet de sélectionner un élément dans un tableau d'éléments. Lorsque vous appuyez sur Annuler, il renvoie Undefined.</li>
</ul>
<h4>File</h4>
Dans le tableau <b>files</b>, il y a des instances d'objets de type <b>File</b> qui ont ces méthodes.
<ul>
<li><b>fileName()</b> renvoie le nom du fichier, à l'exclusion du chemin.</li>
<li><b>absoluteFilePath()</b> renvoie un chemin absolu incluant le nom du fichier.</li>
<li><b>absolutePath()</b> renvoie un chemin absolu sans le nom du fichier</li>
<li><b>relativeFilePath()</b> renvoie le chemin relatif incluant le nom du fichier par rapport au répertoire qui était dans la liste des répertoires à analyser. Par exemple, vous ajoutez C:/images comme répertoire d'entrée. Dans ce répertoire, il y a
le fichier <i>C:/images/lights/red/M42_001.fits</i>, alors cette méthode renverra <i>lights/red/M42_001.fits</i></li>
<li><b>relativePath()</b> renvoie le même chemin que la méthode précédente, mais sans le nom de fichier. <i>lights/red</i></li>
<li><b>baseName()</b> renvoie le nom du fichier jusqu'au premier point. Par exemple, pour <i>some.file.name.fits</i>, il renverra <i>some</i></li>
<li><b>completeBaseName()</b> renvoie le nom du fichier jusqu'au dernier point. Par exemple, pour <i>some.file.name.fits</i>, il renverra <i>some.file.name</i></li>
<li><b>suffix()</b> renvoie la chaîne après le dernier point du nom de fichier. Par exemple <i>fits</i></li>
<li><b>size()</b> renvoie la taille du fichier en octets.</li>
<li><b>fitsKeywords()</b> renvoie un tableau de chaînes avec chaque mot-clé présent dans l'en-tête. <i>SIMPLE,BITPIX,NAXIS,NAXIS1,NAXIS2,EXTEND,COMMENT</i></li>
<li><b>fitsValue(key)</b> renvoie la valeur pour le mot-clé. En cas d'occurrences multiples, la dernière est renvoyée.</li>
<li><b>fitsValues(key)</b> renvoie un tableau de valeurs pour le mot clé.</li>
<li><b>fitsRecords()</b> renvoie un tableau d'objets avec des propriétés <b>key, value</b> et <b>comment</b> </li>
<li><b>modifyFITSRecords(FITSRecordModify)</b> modifier l'en-tête FITS en ajoutant, supprimant ou mettant à jour l'enregistrement FITS. Renvoie true en cas de succès. Reportez-vous à <i>FITSRecordModify</i></li>
<li><b>isMarked()</b> renvoie true si le fichier est marqué.</li>
<li><b>copy(newpath)</b> Copie le fichier vers un nouvel emplacement. Il renvoie une instance du nouvel objet <i>File<i> qui représente ce fichier copié. Ce chemin peut être relatif ou absolu. Dans le cas où le paramètre <i>newpath</i> est un chemin relatif, le "Répertoire de sortie" des fenêtres de l'interface graphique est utilisé comme répertoire de base. Le paramètre <i>newpath</i> peut être un chemin absolu. Le fichier est ensuite copié vers ce chemin. En cas d'échec de la copie, il renvoie null.</li>
<li><b>move(newpath)</b> déplacer le fichier vers un nouvel emplacement. Il renvoie false si le déplacement a échoué. Cela peut se produire si la destination n'est pas accessible en écriture mais aussi si le fichier de destination existe déjà. Cette fonction n'écrase pas le fichier existant.
Ce chemin peut être relatif ou absolu. Dans le cas où le paramètre <i>newpath</i> est un chemin relatif, le "répertoire de sortie" des fenêtres de l'interface graphique est utilisé comme répertoire de base. Le paramètre <i>newpath</i> peut être un chemin absolu.
Le fichier est ensuite déplacé vers ce chemin.</li>
<li><b>convert(outpath, format, params)</b> Convertir un fichier image à partir de n'importe quel format que le programme peut ouvrir en FITS, XISF, JPEG, PNG, BMP.
Les paramètres sont : <i>outputpath</i> chemin où l'image convertie sera enregistrée. Il remplace automatiquement le suffixe en fonction du format. <i>format</i> l'un des éléments suivants : "FITS", "XISF", "JPG", "PNG" ou "BMP". <i>params</i> objet avec les attributs "compressionType" et "compressionLevel".
Les valeurs valides pour compressionType sont "gzip" ou "rice" lors de la conversion en FITS. Lors de la conversion en XISF, compressionType peut être "zlib", "lz4", "lz4hc", "zstd", "zlib+sh", "lz4+sh", "lz4hc+sh", "zstd+sh".
Il est recommandé d'utiliser les variantes de compression "+sh".
Le format XISF accepte également les "compressionLevel" dans la plage 0-100, où zéro est la compression la plus rapide et 100 la plus lente. Si vous omettez cet attribut ou le définissez sur -1, le niveau de compression par défaut sera utilisé.
Il renvoie une nouvelle instance de <i>File</i> qui pointe vers le fichier converti.
<pre>file.convert("converted_file.xisf", "xisf", {"compressionType": "zstd+sh", "compressionLevel": 70});
file.convert("converted_file.fits", "fits", {"compressionType": "rice"});
file.convert("converted_file.jpg", "png");</pre>
</li>
<li><b>convertAsync(outpath, format, params)</b> même méthode que la précédente, mais effectue la conversion dans un thread séparé de manière asynchrone et en parallèle. Avant d'appeler une méthode sur un objet renvoyé par cette méthode, vous devez appeler
<code>core.sync();</code> pour s'assurer que la conversion est effectuée et que le fichier de destination existe.
<pre>let compression = {"compressionType": "zstd+sh"};
let convertedFiles = [];
for(file of files)
{
if(file.suffix() == "fits")
convertedFiles.push(file.convertAsync("xisf/" + file.fileName(), "XISF", compression));
}
core.sync(); // ensure that files exist
for(file of convertedFiles)// now we can iterate over the files
{
core.log(file.fileName() + " " + file.size()); // let print compressed file sizes
}</pre></li>
<li><b>stats()</b> calculer les statistiques d'images de base et les renvoyer sous forme d'objet avec des attributs "mean", "stddev", "median", "min", "max" et "mad".
<pre>let s = file.stats();
core.log("Median value is " + s.median);</pre></li>
</ul>
<h4>FITSRecordModify</h4>
Cette classe est utilisée pour définir l'en-tête FITS des opérations de modification dans les fichiers FITS et XISF. Elle peut supprimer, mettre à jour et ajouter des enregistrements. L'ordre des opérations est également le suivant : suppression, puis mise à jour et enfin ajout.
Les noms des mots-clés peuvent comporter jusqu'à 8 caractères et ne peuvent contenir que des lettres majuscules, les chiffres de 0 à 9, le trait d'union et le caractère de soulignement.
<pre>let modify = new FITSRecordModify();
modify.updateKeyword("OBJECT", "M42");
modify.updateKeyword("MYTILE", "PART1", "adding custom keyword so WBPP can group it");
modify.removeKeyword("OBJECT");
// Peu importe qu'il soit spécifié comme dernier. Cela supprimera d'abord
// l'enregistrement OBJECT existant, puis ajoutera à nouveau OBJECT=M42
for(file in files)
{
file.modifyFITSRecords(modify);
}</pre>
<ul>
<li><b>new FITSRecordModify()</b> créer une nouvelle instance de l'objet.</li>
<li><b>removeKeyword(key);</b> spécifier la suppression de l'enregistrement avec <i>key</i> comme mot-clé.</li>
<li><b>updateKeyword(key, value, comment = "")</b> spécifiez la mise à jour du mot-clé existant avec la valeur et le commentaire. Le commentaire est un paramètre facultatif. Si l'enregistrement avec le mot-clé n'existe pas, il en ajoutera un nouveau.
À moins que vous ne souhaitiez avoir plusieurs enregistrements avec le même mot-clé (par exemple HISTORY), utilisez toujours cette méthode et non addKeyword.</li>
<li><b>addKeyword(key, value, comment = "")</b> spécifier l'ajout d'un nouveau mot-clé</li>
</ul>
<p><small>PS: Le Kanji de icône (tenmon) signifie astronomie en japonais</small></p>
</body>
</html>
+120 -1
View File
@@ -13,7 +13,7 @@ p { padding:0px; margin:5px 5px 10px 5px; }
<li>FITS 8, 16, 32 bitové celočíselné 32 a 64 bitové s plávajúcou čiarkou</li>
<li>XISF 8, 16, 32 bitové celočíselné 32 a 64 bitové s plávajúcou čiarkou</li>
<li>JPEG, PNG, BMP, GIF, PBM, PGM, PPM a SVG obrázky</li>
<li>CR2, NEF, DNG raw obrázky</li>
<li>CR2, CR3, NEF, DNG raw obrázky</li>
</ul>
</p>
@@ -78,9 +78,128 @@ Pre indexovanie nových súborov je treba znova pustiť indexáciu.</p>
kde sú jednotlivé stĺpcoch zobrazené vybrané vlastnosti. V spodnej časti panelu je tlačidlo ktoré zobrazí dialóg na výber zobrazovaných
sĺpcov. Nasledujú tri výberové a textové polia. Tieto slúžia na vyhľadávanie v databáze. Výberové pole určuje stĺpec v ktorom sa
má vyhľadávať a do textového poľa sa zadáva hodnota na vyhľadanie.
<p>Zastupné znaky:
<ul>
<li><b>%</b> (percent) je zastupný znak reprezentujúci žiadný alebo hocikoľko znakov.</li>
<li><b>_</b> (underscore) je zastupný znak nahrádzajúci presne jeden znak.</i>
<li>Bez zástupných znako sa hľadá presná zhoda.</li>
</ul>
</p>
<br><img src=":/about/filter.png"><br>
V nasledovnom príklade sa vyhľadajú súbory ktoré majú v mene súboru "Bias", OBJECT je M_42 a DATE začína reťazcom 2022. Znak % sa berie ako
zástupný znak za hocijaký reťazec znakov aj žiadny. Znak _ je tiež zástupný znak zastupujúci práve jeden znak.
Bez použitia zástupných znakov sa vyhľadá iba presný výskyt.</p>
<h3>Hromadné spracovanie</h3>
Tento modul umožnuje písanie skriptov v JavaScripte ktoré spracujú súbory obrázkov. Okno Hromadného spracovanie pozostáva z troch častí. Navrchu je zoznam vstupných súborov a adresárov.
Do zoznamu môžete pridať adresáre alebo jednotlivé súbory. Adresáre sú rekurzívne prehľadané na všetky súbory. Zoznam súborov je potom predaný do skriptu ako pole nazvané <b>files</b>.
V skripte potom cez toto pole iteruje nasledovne.
<pre>for(file in files)
{
if(file.suffix() == "fits")
{
core.log(file.fileName());
file.convert(file.relativeFilePath(), "XISF");
}
}
</pre>
<h4>core</h4>
V skripte je dostupný globálny objekt nazvaný <b>core</b> ktorý má nasledovné metódy.
<ul>
<li><b>log(message)</b> vypíše message do okna záznamu.</li>
<li><b>mark(file)</b> označí súbor rovnako ako cez GUI. Parameter je objekt typu <i>File</i>.</li>
<li><b>unmark(file)</b> odznačí súbor rovnako ako cez GUI. Parameter je objekt typu <i>File</i>.</li>
<li><b>isMarked(file)</b> overenie či je súbor označený. Parameter je objekt typu <i>File</i>.</li>
<li><b>setMaxThread(maxthread)</b> nastavý maximálny počet paralelných vlákien pri vykonávaní asynchrónnych úloh.</li>
<li><b>sync()</b> počká kým sa dokončia všetky asynchrónne úlohy.</li>
<li><b>getString(label = "", text = "")</b> ukáže dialóg pre získanie textovej hodnoty od používateľa. Prvý parameter je textový popis. Druhý parameter je východzia hodnota. Obydva parametre sú voliteľné takže aj volanie <i>getString()</i> je valdiné.
Metoda vracia textový retazec alebo Undefined ak bolo stlačené tlačidlo zrušiť.</li>
<li><b>getInt(label = "", value = 0)</b> ukáže diálog pre získanie celočíselnej hodnoty. Prvý parameter je textový popis. Druhý parameter je východzia hodnota. Obydva parametre sú voliteľné. Vracia zadané číslo alebo ak je stlačené tlačidlo zrušiť vráti Undefined.</li>
<li><b>getFloat(label = "", value = 0, decimals = 3)</b> ukáže diálog pre získanie reálneho čísla. Prvý parameter je textový popis. Druhý parameter je východzia hodnota. Tretí parameter je počet desatinných miest.
Obydva parametre sú voliteľné. Vracia zadané číslo alebo ak je stlačené tlačidlo zrušiť vráti Undefined.</li>
<li><b>getItem(items)</b> ukáže dialog pre výber jednej hodnoty z poľa hodnôt. Vracia vybranú hodnotu ako String alebo ak je stlačené tlačidlo zrušiť vráti Undefined.</li>
</ul>
<h4>File</h4>
V poli <b>files</b> sú inštancie objektu typu <b>File</b> ktorý ma nasledovné metódy.
<ul>
<li><b>fileName()</b> returns the name of the file, excluding the path.</li>
<li><b>absoluteFilePath()</b> returns an absolute path including the file name.</li>
<li><b>absolutePath()</b> returns an absolute path without the file name</li>
<li><b>relativeFilePath()</b> return relative path including file name relative to directory that was in list of directories to be scanned. For example you add C:/images as input directory. In this directory there
is file <i>C:/images/lights/red/M42_001.fits</i> then this method will return <i>lights/red/M42_001.fits</i></li>
<li><b>relativePath()</b> return same path as previous method just without file name. <i>lights/red</i></li>
<li><b>baseName()</b> return file name up to the first dot. For example for <i>some.file.name.fits</i> it will return <i>some</i></li>
<li><b>completeBaseName()</b> return file name up to the last dot. For example for <i>some.file.name.fits</i> it will return <i>some.file.name</i></li>
<li><b>suffix()</b> return string after last dot in file name. For example <i>fits</i></li>
<li><b>size()</b> return size of file in bytes.</li>
<li><b>fitsKeywords()</b> return array of strings with every keyword that is in header. <i>SIMPLE,BITPIX,NAXIS,NAXIS1,NAXIS2,EXTEND,COMMENT</i></li>
<li><b>fitsValue(key)</b> return value for keyword. In case that there is multiple occurrences it return last one.</li>
<li><b>fitsValues(key)</b> return array of values for keyword.</li>
<li><b>fitsRecords()</b> return array of objects with properties <b>key, value</b> and <b>comment</b> </li>
<li><b>modifyFITSRecords(FITSRecordModify)</b> modify FITS header by adding, removing or updating FITS record. Return true on success. Refer to <i>FITSRecordModify</i></li>
<li><b>isMarked()</b> return true if file is marked.</li>
<li><b>copy(newpath)</b> copy file to new location. It return instance of new <i>File<i> object that represent this copied file. This path can be relative or absolute. In case that <i>newpath</i> parameter is relative
path then it "Output directory" from GUI windows is used as base directory. Parameter <i>newpath</i> can absolute path. File is then copied to this path. In case that copy fail it return null.</li>
<li><b>move(newpath)</b> move file to new location. It return false if move failed. This can happend if destination is not writable but also if destination file already exist. This functions does not overwrite existing file.
This path can be relative or absolute. In case that <i>newpath</i> parameter is relative path then it "Output directory" from GUI windows is used as base directory. Parameter <i>newpath</i> can be absolute path.
File is then moved to this path.</li>
<li><b>convert(outpath, format, params)</b> convert image file from any format that program is able to open into FITS, XISF, JPEG, PNG, BMP.
Parameters are: <i>outputpath</i> path where converted image will be saved. It automatically replace suffix according to format. <i>format</i> one of "FITS" "XISF", "JPG", "PNG" or "BMP". <i>params</i> object with attributes "compressionType" and "compressionLevel".
Valid values for compressionType are be "gzip" or "rice" when converting to FITS. When converting to XISF compressionType can be "zlib", "lz4", "lz4hc", "zstd", "zlib+sh", "lz4+sh", "lz4hc+sh", "zstd+sh".
It is recommended to use "+sh" variants of compression.
XISF format also accept "compressionLevel" in range 0-100 where zero is fastest compression and 100 slowest. If you omit this attribute or set it to -1 then default compression level will be used.
It return new instance of <i>File</i> that point to converted file.
<pre>file.convert("converted_file.xisf", "xisf", {"compressionType": "zstd+sh", "compressionLevel": 70});
file.convert("converted_file.fits", "fits", {"compressionType": "rice"});
file.convert("converted_file.jpg", "png");</pre>
</li>
<li><b>convertAsync(outpath, format, params)</b> same as previous method but it does conversion in separated thread asynchronously and in parallel. Before calling any method on object returned by this method you must call
<code>core.sync();</code> to ensure that conversion is done and destination file exists.
<pre>let compression = {"compressionType": "zstd+sh"};
let convertedFiles = [];
for(file of files)
{
if(file.suffix() == "fits")
convertedFiles.push(file.convertAsync("xisf/" + file.fileName(), "XISF", compression));
}
core.sync(); // ensure that files exist
for(file of convertedFiles)// now we can iterate over the files
{
core.log(file.fileName() + " " + file.size()); // let print compressed file sizes
}</pre></li>
<li><b>stats()</b> calculate basic images statistics and return them as object with attributes "mean", "stddev", "median", "min", "max" and "mad".
<pre>let s = file.stats();
core.log("Median value is " + s.median);</pre></li>
</ul>
<h4>FITSRecordModify</h4>
This class is used to define modify operation FITS header in FITS and XISF files. It can remove update and add records. Order of operation is also remove then update and last add.
The keyword names may be up to 8 characters long and can only contain uppercase letters, the digits 0-9, the hyphen, and the underscore character.
<pre>let modify = new FITSRecordModify();
modify.updateKeyword("OBJECT", "M42");
modify.updateKeyword("MYTILE", "PART1", "adding custom keyword so WBPP can group it");
modify.removeKeyword("OBJECT");
// doesn't matter that it is specified as last. This will first remove
// existing OBJECT record and then add again OBJECT=M42
for(file in files)
{
file.modifyFITSRecords(modify);
}</pre>
<ul>
<li><b>new FITSRecordModify()</b> create new instance of object.</li>
<li><b>removeKeyword(key);</b> specify removing of record with <i>key</i> as keyword.</li>
<li><b>updateKeyword(key, value, comment = "")</b> specify updating existing keyword with value and comment. Comment is optional parameter. If record with keyword doesn't exist then it will add new one.
Unless you want to have multiple records with same keyword (for example HISTORY) always use this method and not addKeyword.</li>
<li><b>addKeyword(key, value, comment = "")</b> specify adding new keyword</li>
</ul>
<p><small>PS: Kanji v ikone programu znamená "astronomia" v Japončine</small></p>
</body>
</html>
+280
View File
@@ -0,0 +1,280 @@
#include "batchprocessing.h"
#include "ui_batchprocessing.h"
#include <functional>
#include <QDir>
#include <QFileDialog>
#include <QStandardPaths>
#include <QProcess>
#include <QSettings>
#include <QCloseEvent>
#include <QMessageBox>
#include <QDesktopServices>
#include <QInputDialog>
#include "scriptengine.h"
#ifdef Q_OS_LINUX
#include <QCloseEvent>
#include <QDBusConnection>
#include <QDBusMessage>
#endif
QList<QPair<QString, QString>> scanDirectories(const QStringList &paths)
{
QList<QPair<QString, QString>> files;
QStringList scannedDirs;
std::function<void(const QString &root, const QString &path)> scanDirectory = [&](const QString &root, const QString &path)
{
QFileInfo info(path);
if(info.isDir() && !scannedDirs.contains(info.canonicalFilePath()))
{
scannedDirs.append(info.canonicalFilePath());
QDir dir(path);
QStringList entries = dir.entryList(QDir::Dirs | QDir::Files | QDir::NoDotAndDotDot);
for(QString &entry : entries)
scanDirectory(root, dir.absoluteFilePath(entry));
}
else if(info.isFile())
{
if(path == root)
files.append({path, info.absolutePath()});
else
files.append({path, root});
}
};
for(const QString &path : paths)
scanDirectory(path, path);
return files;
}
void BatchProcessing::scanScriptDir()
{
QString current;
if(_ui->scriptsList->currentItem())
current = _ui->scriptsList->currentItem()->text();
_ui->scriptsList->clear();
QDir dir(_scriptBasePath);
QDir embededDir(":/scripts");
QStringList scripts = dir.entryList(QDir::Files | QDir::Readable);
scripts.append(embededDir.entryList(QDir::Files));
scripts.removeDuplicates();
_ui->scriptsList->addItems(scripts);
int idx = scripts.indexOf(current);
if(idx>=0)_ui->scriptsList->setCurrentRow(idx);
}
BatchProcessing::BatchProcessing(QWidget *parent) : QDialog(parent)
{
_ui = new Ui::BatchProcessing;
_ui->setupUi(this);
QStringList scriptsPath = QStandardPaths::standardLocations(QStandardPaths::AppDataLocation);
if(scriptsPath.size())
{
QDir dir(scriptsPath.first());
if(!dir.exists("scripts"))
{
if(!dir.mkpath("scripts"))
qWarning() << "Failed to create scripts directory";
}
dir.cd("scripts");
_scriptBasePath = dir.absolutePath() + "/";
scanScriptDir();
_fileWatcher.addPath(_scriptBasePath);
connect(&_fileWatcher, &QFileSystemWatcher::directoryChanged, this, &BatchProcessing::scanScriptDir);
}
else
{
qWarning() << "Failed to get app data location";
}
connect(_ui->addFilesButton, &QPushButton::released, this, &BatchProcessing::addFiles);
connect(_ui->addDirButton, &QPushButton::released, this, &BatchProcessing::addDir);
connect(_ui->removeButton, &QPushButton::released, this, &BatchProcessing::removePath);
connect(_ui->removeAllButton, &QPushButton::released, this, &BatchProcessing::removeAllPaths);
connect(_ui->startButton, &QPushButton::released, this, &BatchProcessing::runScript);
connect(_ui->stopButton, &QPushButton::released, this, &BatchProcessing::stopScript);
connect(_ui->browseButton, &QPushButton::released, this, &BatchProcessing::browse);
connect(_ui->openScriptsButton, &QPushButton::released, this, &BatchProcessing::openScriptDir);
_textColor = _ui->log->palette().text().color();
QSettings settings;
_ui->outputPath->setText(settings.value("batchprocessing/outputpath", QStandardPaths::standardLocations(QStandardPaths::PicturesLocation).first()).toString());
}
BatchProcessing::~BatchProcessing()
{
delete _engineThread;
QSettings settings;
settings.setValue("batchprocessing/outputpath", _ui->outputPath->text());
delete _ui;
}
void BatchProcessing::closeEvent(QCloseEvent *event)
{
if(_engineThread)
{
QMessageBox::StandardButton ret = QMessageBox::question(this, tr("Interrupt running script?"), tr("Interrupt running script?"));
if(ret == QMessageBox::StandardButton::Yes)
{
_engineThread->interrupt();
event->accept();
}
else
{
event->ignore();
}
}
else
{
event->accept();
}
}
void BatchProcessing::addFiles()
{
QSettings settings;
QStringList files = QFileDialog::getOpenFileNames(this, tr("Select files"), settings.value("batchprocessing/inputpath", QDir::homePath()).toString());
if(!files.isEmpty())
{
_ui->pathsList->addItems(files);
settings.setValue("batchprocessing/inputpath", QFileInfo(files.first()).absolutePath());
}
}
void BatchProcessing::addDir()
{
QSettings settings;
QString dir = QFileDialog::getExistingDirectory(this, tr("Select directory"), settings.value("batchprocessing/inputpath", QDir::homePath()).toString());
if(!dir.isEmpty())
{
_ui->pathsList->addItem(dir);
settings.setValue("batchprocessing/inputpath", dir);
}
}
void BatchProcessing::removePath()
{
for(auto &item : _ui->pathsList->selectedItems())
delete item;
}
void BatchProcessing::removeAllPaths()
{
_ui->pathsList->clear();
}
void BatchProcessing::browse()
{
QString output = QFileDialog::getExistingDirectory(this, tr("Select output directory"), "/home/nou/Obrázky");
if(!output.isEmpty())
_ui->outputPath->setText(output);
}
void BatchProcessing::openScriptDir()
{
#ifdef Q_OS_LINUX
QDBusConnection con = QDBusConnection::sessionBus();
QDBusMessage message = QDBusMessage::createMethodCall("org.freedesktop.FileManager1", "/org/freedesktop/FileManager1", "org.freedesktop.FileManager1", "ShowFolders");
QList<QVariant> args = {QStringList(QUrl::fromLocalFile(_scriptBasePath).toString()), QString()};
message.setArguments(args);
con.call(message);
#endif
#ifdef Q_OS_WINDOWS
QProcess::startDetached("explorer.exe", {QDir::toNativeSeparators(_scriptBasePath)});
#endif
#ifdef Q_OS_MACOS
QDesktopServices::openUrl(QUrl::fromLocalFile(_scriptBasePath));
#endif
}
void BatchProcessing::runScript()
{
_ui->log->clear();
auto selectedItems = _ui->scriptsList->selectedItems();
if(selectedItems.size())
{
_engineThread = new Script::ScriptEngineThread(this);
connect(_engineThread, &Script::ScriptEngineThread::newMessage, this, &BatchProcessing::newMessage);
connect(_engineThread, &Script::ScriptEngineThread::finished, this, &BatchProcessing::scriptFinished);
QStringList paths;
for(int i=0; i<_ui->pathsList->count(); i++)
paths.append(_ui->pathsList->item(i)->text());
QFileInfo outDir(_ui->outputPath->text());
if(outDir.exists() && outDir.isWritable())
{
QString script = selectedItems.first()->text();
if(QDir(_scriptBasePath).exists(script))
script = _scriptBasePath + script;
else
script = ":/scripts/" + script;
_engineThread->setParams(script, scanDirectories(paths), _ui->outputPath->text());
_engineThread->start();
_ui->startButton->setEnabled(false);
_ui->stopButton->setEnabled(true);
}
else
{
QMessageBox::warning(this, tr("Invalid output directory"), tr("Output directory path doesn't exist or is not writable"));
}
}
}
void BatchProcessing::stopScript()
{
qDebug() << "Stop script";
if(_engineThread)
_engineThread->interrupt();
}
void BatchProcessing::scriptFinished()
{
_ui->startButton->setEnabled(true);
_ui->stopButton->setEnabled(false);
qDebug() << "script finished";
_engineThread->deleteLater();
_engineThread = nullptr;
}
void BatchProcessing::newMessage(const QString &message, bool error)
{
if(error)_ui->log->setTextColor(Qt::red);
else _ui->log->setTextColor(_textColor);
_ui->log->append(message);
}
QJSValue BatchProcessing::getString(const QString &label, const QString &text)
{
bool ok = false;
QString ret = QInputDialog::getText(this, tr("Enter text"), label, QLineEdit::Normal, text, &ok);
return ok ? ret : QJSValue();
}
QJSValue BatchProcessing::getInt(const QString &label, int value)
{
bool ok = false;
int ret = QInputDialog::getInt(this, tr("Enter integer number"), label, value, INT_MIN, INT_MAX, 1, &ok);
return ok ? ret : QJSValue();
}
QJSValue BatchProcessing::getFloat(const QString &label, double value, int decimals)
{
bool ok = false;
double ret = QInputDialog::getDouble(this, tr("Enter float number"), label, value, -INFINITY, INFINITY, decimals, &ok);
return ok ? ret : QJSValue();
}
QJSValue BatchProcessing::getItem(const QStringList &items, const QString &label, int current)
{
bool ok = false;
QString ret = QInputDialog::getItem(this, tr("Select item"), label, items, current, false, &ok);
return ok ? ret : QJSValue();
}
+43
View File
@@ -0,0 +1,43 @@
#ifndef BATCHPROCESSING_H
#define BATCHPROCESSING_H
#include <QDialog>
#include <QFileSystemWatcher>
#include "scriptengine.h"
namespace Ui { class BatchProcessing; }
class BatchProcessing : public QDialog
{
Q_OBJECT
Ui::BatchProcessing *_ui;
QString _scriptBasePath;
QFileSystemWatcher _fileWatcher;
Script::ScriptEngineThread *_engineThread = nullptr;
QColor _textColor;
private slots:
void scanScriptDir();
public:
explicit BatchProcessing(QWidget *parent = nullptr);
~BatchProcessing();
protected:
void closeEvent(QCloseEvent *event);
public slots:
void addFiles();
void addDir();
void removePath();
void removeAllPaths();
void browse();
void openScriptDir();
void runScript();
void stopScript();
void scriptFinished();
void newMessage(const QString &message, bool error);
QJSValue getString(const QString &label, const QString &text);
QJSValue getInt(const QString &label, int value);
QJSValue getFloat(const QString &label, double value, int decimals);
QJSValue getItem(const QStringList &items, const QString &label, int current);
};
#endif // BATCHPROCESSING_H
+226
View File
@@ -0,0 +1,226 @@
<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
<class>BatchProcessing</class>
<widget class="QDialog" name="BatchProcessing">
<property name="geometry">
<rect>
<x>0</x>
<y>0</y>
<width>1024</width>
<height>768</height>
</rect>
</property>
<property name="windowTitle">
<string>Batch Processing</string>
</property>
<layout class="QVBoxLayout" name="verticalLayout_2">
<item>
<layout class="QVBoxLayout" name="verticalLayout">
<item>
<widget class="QLabel" name="label_2">
<property name="text">
<string>Input files and directories</string>
</property>
</widget>
</item>
<item>
<widget class="QListWidget" name="pathsList">
<property name="sizePolicy">
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
<horstretch>0</horstretch>
<verstretch>1</verstretch>
</sizepolicy>
</property>
<property name="selectionMode">
<enum>QAbstractItemView::MultiSelection</enum>
</property>
</widget>
</item>
<item>
<layout class="QHBoxLayout" name="horizontalLayout">
<item>
<widget class="QPushButton" name="addFilesButton">
<property name="text">
<string>Add files</string>
</property>
</widget>
</item>
<item>
<widget class="QPushButton" name="addDirButton">
<property name="text">
<string>Add directories</string>
</property>
</widget>
</item>
<item>
<widget class="QPushButton" name="removeButton">
<property name="text">
<string>Remove</string>
</property>
</widget>
</item>
<item>
<widget class="QPushButton" name="removeAllButton">
<property name="text">
<string>Remove all</string>
</property>
</widget>
</item>
</layout>
</item>
</layout>
</item>
<item>
<layout class="QHBoxLayout" name="horizontalLayout_2">
<item>
<widget class="QLabel" name="label">
<property name="text">
<string>Output directory</string>
</property>
</widget>
</item>
<item>
<widget class="QLineEdit" name="outputPath">
<property name="readOnly">
<bool>true</bool>
</property>
</widget>
</item>
<item>
<widget class="QPushButton" name="browseButton">
<property name="text">
<string>Browse</string>
</property>
</widget>
</item>
</layout>
</item>
<item>
<layout class="QHBoxLayout" name="horizontalLayout_3">
<item>
<widget class="QLabel" name="label_4">
<property name="text">
<string>Scripts</string>
</property>
</widget>
</item>
<item>
<spacer name="horizontalSpacer">
<property name="orientation">
<enum>Qt::Horizontal</enum>
</property>
<property name="sizeHint" stdset="0">
<size>
<width>40</width>
<height>20</height>
</size>
</property>
</spacer>
</item>
<item>
<widget class="QPushButton" name="openScriptsButton">
<property name="text">
<string>Open scripts</string>
</property>
</widget>
</item>
</layout>
</item>
<item>
<widget class="QListWidget" name="scriptsList">
<property name="sizePolicy">
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
<horstretch>0</horstretch>
<verstretch>1</verstretch>
</sizepolicy>
</property>
</widget>
</item>
<item>
<widget class="QLabel" name="label_3">
<property name="text">
<string>Log</string>
</property>
</widget>
</item>
<item>
<widget class="QTextEdit" name="log">
<property name="sizePolicy">
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
<horstretch>0</horstretch>
<verstretch>4</verstretch>
</sizepolicy>
</property>
<property name="font">
<font>
<family>FreeMono</family>
</font>
</property>
<property name="readOnly">
<bool>true</bool>
</property>
</widget>
</item>
<item>
<layout class="QHBoxLayout" name="horizontalLayout_4">
<item>
<spacer name="horizontalSpacer_2">
<property name="orientation">
<enum>Qt::Horizontal</enum>
</property>
<property name="sizeHint" stdset="0">
<size>
<width>40</width>
<height>20</height>
</size>
</property>
</spacer>
</item>
<item>
<widget class="QPushButton" name="startButton">
<property name="text">
<string>Start script</string>
</property>
</widget>
</item>
<item>
<widget class="QPushButton" name="stopButton">
<property name="enabled">
<bool>false</bool>
</property>
<property name="text">
<string>Stop script</string>
</property>
</widget>
</item>
<item>
<widget class="QPushButton" name="closeButton">
<property name="text">
<string>Close</string>
</property>
</widget>
</item>
</layout>
</item>
</layout>
</widget>
<resources/>
<connections>
<connection>
<sender>closeButton</sender>
<signal>released()</signal>
<receiver>BatchProcessing</receiver>
<slot>close()</slot>
<hints>
<hint type="sourcelabel">
<x>973</x>
<y>745</y>
</hint>
<hint type="destinationlabel">
<x>511</x>
<y>383</y>
</hint>
</hints>
</connection>
</connections>
</ui>
BIN
View File
Binary file not shown.

Before

Width:  |  Height:  |  Size: 380 B

+18 -9
View File
@@ -10,12 +10,12 @@ Database::Database(QObject *parent) : QObject(parent)
{
}
bool Database::init()
bool Database::init(const QLatin1String &connectionName)
{
QString path = QStandardPaths::writableLocation(QStandardPaths::AppDataLocation);
QDir dir(path);
QSqlDatabase m_database = QSqlDatabase::addDatabase("QSQLITE");
QSqlDatabase m_database = QSqlDatabase::addDatabase("QSQLITE", connectionName);
if(!dir.mkpath("."))
return false;
@@ -156,23 +156,29 @@ int Database::checkVersion()
static QStringList nameFilters = {"*.fit", "*.fits", "*.xisf"};
static int countFiles(const QDir &dir, int count = 0)
static int countFiles(const QDir &dir, QStringList &scannedDirs)
{
count += dir.entryList(nameFilters, QDir::Files).size();
if(scannedDirs.contains(dir.canonicalPath()))return 0;
scannedDirs.append(dir.canonicalPath());
int count = dir.entryList(nameFilters, QDir::Files).size();
QStringList dirs = dir.entryList(QDir::Dirs | QDir::NoDotAndDotDot);
for(const QString &d : dirs)
count += countFiles(dir.filePath(d));
count += countFiles(dir.filePath(d), scannedDirs);
return count;
}
void Database::indexDir(const QDir &dir, QProgressDialog *progress)
{
m_progress = 0;
int count = countFiles(dir);
QStringList scannedDirs;
int count = countFiles(dir, scannedDirs);
progress->setMaximum(count);
QSqlDatabase database = QSqlDatabase::database();
database.transaction();
if(indexDir2(dir, progress))
scannedDirs.clear();
if(indexDir2(dir, progress, scannedDirs))
{
database.commit();
emit databaseChanged();
@@ -225,14 +231,17 @@ QStringList Database::getFitsKeywords()
return keywords;
}
bool Database::indexDir2(const QDir &dir, QProgressDialog *progress)
bool Database::indexDir2(const QDir &dir, QProgressDialog *progress, QStringList &scannedDirs)
{
if(scannedDirs.contains(dir.canonicalPath()))return true;
scannedDirs.append(dir.canonicalPath());
QFileInfoList files = dir.entryInfoList(nameFilters, QDir::Files);
QStringList dirs = dir.entryList(QDir::Dirs | QDir::NoDotAndDotDot);
for(const QString &d : dirs)
{
if(!indexDir2(dir.filePath(d), progress))
if(!indexDir2(dir.filePath(d), progress, scannedDirs))
return false;
}
for(const QFileInfo &file : files)
+2 -2
View File
@@ -24,7 +24,7 @@ class Database : public QObject
int m_progress;
public:
explicit Database(QObject *parent = 0);
bool init();
bool init(const QLatin1String &connectionName = QLatin1String(QSqlDatabase::defaultConnection));
bool mark(const QString &filename);
bool unmark(const QString &filename);
bool mark(const QStringList &filenames);
@@ -37,7 +37,7 @@ public:
void reindex(QProgressDialog *progress);
QStringList getFitsKeywords();
protected:
bool indexDir2(const QDir &dir, QProgressDialog *progress);
bool indexDir2(const QDir &dir, QProgressDialog *progress, QStringList &scannedDirs);
bool indexFile(const QFileInfo &file);
bool checkError(QSqlQuery &query);
int checkVersion();
+3 -1
View File
@@ -201,7 +201,8 @@ void FITSFileModel::prepareQuery()
if(lastError().type() != QSqlError::NoError)
qDebug() << "Database error" << lastError();
m_markedFiles = m_database->getMarkedFiles().toSet();
QStringList list = m_database->getMarkedFiles();
m_markedFiles = QSet<QString>(list.begin(), list.end());
}
DatabaseTableView::DatabaseTableView(QWidget *parent) : QTableView(parent)
@@ -285,6 +286,7 @@ DataBaseView::DataBaseView(Database *database, QWidget *parent) : QWidget(parent
for(int i=0; i<3; i++)
{
m_filterKeyword[i] = new QComboBox(this);
m_filterKeyword[i]->setMaximumWidth(300);
addFilterItems(m_filterKeyword[i], fitsKeywords);
+3 -2
View File
@@ -48,6 +48,7 @@ FITSRecord::FITSRecord(const LibXISF::Property &property)
key = property.id.c_str();
value = QString::fromStdString(property.value.toString());
comment = property.comment.c_str();
xisf = true;
}
QByteArray FITSRecord::valueToByteArray() const
@@ -81,7 +82,7 @@ void ImageInfo::setInfo(const ImageInfoData &info)
QTreeWidgetItem *fitsHeader = new QTreeWidgetItem({tr("FITS Header")});
for(const FITSRecord &record : info.fitsHeader)
{
new QTreeWidgetItem(fitsHeader, {record.key, record.value.toString(), record.comment});
new QTreeWidgetItem(fitsHeader, {record.key, record.value.toString().left(1024), record.comment});
}
addTopLevelItem(fitsHeader);
}
@@ -279,7 +280,7 @@ QString SkyPoint::toString() const
t = t.addSecs(ra * 240);
double deg, min, sec;
min = std::modf(dec, &deg) * 60;
min = std::abs(std::modf(dec, &deg) * 60);
sec = std::modf(min, &min) * 60;
return QString("RA: %1 DEC: %2° %3' %4\"").arg(t.toString("HH'h' mm'm' ss's'")).arg(deg, 2, 'f', 0, '0').arg(min, 2, 'f', 0, '0').arg(sec, 2, 'f', 0, '0');
}
+1
View File
@@ -13,6 +13,7 @@ struct FITSRecord
QByteArray key;
QVariant value;
QByteArray comment;
bool xisf = false;
bool editable() const;
FITSRecord(){}
FITSRecord(const QByteArray &key, const QVariant &value, const QByteArray &comment);
+50 -5
View File
@@ -1,6 +1,9 @@
#include "imageringlist.h"
#include <functional>
#include <QThreadPool>
#include <QDir>
#include <QSettings>
#include <QTimer>
#include "loadrunable.h"
#include "rawimage.h"
#include "database.h"
@@ -108,6 +111,9 @@ ImageRingList::ImageRingList(Database *database, const QStringList &nameFilter,
connect(&m_fileSystemWatcher, SIGNAL(directoryChanged(QString)), this, SLOT(dirChanged(QString)));
m_nameFilter.replaceInStrings(QRegularExpression("^"), "*.");
m_thumbPool = new QThreadPool(this);
m_slideShowTimer = new QTimer(this);
connect(m_slideShowTimer, &QTimer::timeout, this, static_cast<void (ImageRingList::*)()>(&ImageRingList::increment));
}
ImageRingList::~ImageRingList()
@@ -119,21 +125,39 @@ ImageRingList::~ImageRingList()
m_thumbPool->waitForDone();
}
bool ImageRingList::setDir(const QString path, const QString &currentFile)
bool ImageRingList::setDir(const QString path, const QString &currentFile, bool recursive)
{
QDir dir(path);
if(dir.exists())
{
QStringList scannedDirs;
QStringList absolutePaths;
std::function<void(const QString&)> scanDir = [&](const QString &path)
{
QDir dir(path);
if(scannedDirs.contains(dir.canonicalPath()))return;
scannedDirs.append(dir.canonicalPath());
QDir::SortFlags sortFlags = m_liveMode ? QDir::Time : m_sort | QDir::IgnoreCase;
if(m_reversed)sortFlags |= QDir::Reversed;
if(recursive)
{
QStringList dirs = dir.entryList(QDir::Readable | QDir::Dirs | QDir::NoDotAndDotDot, sortFlags);
for(const QString &subdir : dirs)
scanDir(dir.absoluteFilePath(subdir));
}
QStringList list = dir.entryList(m_nameFilter, QDir::Files | QDir::Readable, sortFlags);
QStringList absolutePaths;
foreach(const QString &file, list)
for(const QString &file : list)
{
absolutePaths.append(dir.absoluteFilePath(file));
}
setFiles(absolutePaths, m_liveMode ? list.first() : currentFile);
};
scanDir(path);
qDebug() << absolutePaths.size();
setFiles(absolutePaths, m_liveMode ? absolutePaths.first() : currentFile);
m_fileSystemWatcher.removePaths(m_fileSystemWatcher.directories());
m_fileSystemWatcher.addPath(path);
@@ -146,7 +170,7 @@ void ImageRingList::setFile(const QString &file)
{
QFileInfo info(file);
if(info.isDir())
setDir(file);
setDir(file, QString(), true);
else
setDir(info.absolutePath(), file);
}
@@ -163,6 +187,10 @@ void ImageRingList::increment()
{
if(m_images.size())
{
//don't increment if current image was not loaded yet
if(!(*m_currImage)->rawImage())
return;
(*m_firstImage)->release();
m_firstImage = increment(m_firstImage);
m_currImage = increment(m_currImage);
@@ -295,11 +323,13 @@ void ImageRingList::clearThumbnails()
QModelIndex ImageRingList::index(int row, int column, const QModelIndex &parent) const
{
Q_UNUSED(parent);
return createIndex(row, column, m_images.at(row).get());
}
QModelIndex ImageRingList::parent(const QModelIndex &child) const
{
Q_UNUSED(child);
return QModelIndex();
}
@@ -313,6 +343,7 @@ int ImageRingList::rowCount(const QModelIndex &parent) const
int ImageRingList::columnCount(const QModelIndex &parent) const
{
Q_UNUSED(parent);
return 1;
}
@@ -398,6 +429,20 @@ void ImageRingList::reverseSort()
}
}
void ImageRingList::toggleSlideshow(bool start)
{
if(start)
{
QSettings settings;
int time = settings.value("settings/slideshowtime", 1.0).toDouble() * 1000;
m_slideShowTimer->start(time);
}
else
{
m_slideShowTimer->stop();
}
}
void ImageRingList::setFiles(const QStringList files, const QString &currentFile)
{
QThreadPool::globalInstance()->clear();
+5 -3
View File
@@ -65,14 +65,13 @@ class ImageRingList : public QAbstractItemModel
QThreadPool *m_thumbPool;
Database *m_database;
QStringList m_nameFilter;
QTimer *m_slideShowTimer;
public:
explicit ImageRingList(Database *database, const QStringList &nameFilter, QObject *parent = 0);
~ImageRingList() override;
bool setDir(const QString path, const QString &currentFile = QString());
bool setDir(const QString path, const QString &currentFile = QString(), bool recursive = false);
void setFile(const QString &file);
ImagePtr currentImage();
void increment();
void decrement();
void setLiveMode(bool live);
void setCalculateStats(bool stats);
void setFindPeaks(bool findPeaks);
@@ -96,6 +95,9 @@ public slots:
void setPreload(int width);
void setSort(QDir::SortFlag sort);
void reverseSort();
void toggleSlideshow(bool start);
void increment();
void decrement();
protected:
void setFiles(const QStringList files, const QString &currentFile = QString());
QList<ImagePtr>::iterator increment(QList<ImagePtr>::iterator iter);
+2 -2
View File
@@ -106,7 +106,7 @@ void ImageScrollArea::wheelEvent(QWheelEvent *event)
m_scale = (float)size().width()/m_pixmap.size().width();
QPointF top(horizontalScrollBar()->value(), verticalScrollBar()->value());
QPointF mousePos = (top + event->posF()) / m_scale;
QPointF mousePos = (top + event->position()) / m_scale;
QPoint delta = event->angleDelta();
if(delta.y() > 0)
@@ -115,7 +115,7 @@ void ImageScrollArea::wheelEvent(QWheelEvent *event)
setScale(m_scale - 0.1);
mousePos *= m_scale;
top = mousePos - event->posF();
top = mousePos - event->position();
horizontalScrollBar()->setValue(top.x());
verticalScrollBar()->setValue(top.y());
}
+68 -34
View File
@@ -1,5 +1,6 @@
#include "imagescrollareagl.h"
#include <QOpenGLFunctions>
#include <QOpenGLVersionFunctionsFactory>
#include <QDebug>
#include <QKeyEvent>
#include <QOpenGLDebugLogger>
@@ -14,6 +15,8 @@
#include <cmath>
#include <QElapsedTimer>
int FILTERING = 1;
struct RawImageType
{
QOpenGLTexture::PixelFormat pixelFormat;
@@ -46,6 +49,10 @@ RawImageType getRawImageType(const RawImage *img)
else
type.textureFormat = QOpenGLTexture::R32F;
type.dataType = QOpenGLTexture::Float32;
break;
default:
qWarning() << "Invalid format" << img->type();
break;
}
if(img->channels() >= 3)
@@ -83,14 +90,27 @@ ImageWidget::~ImageWidget()
void ImageWidget::setImage(std::shared_ptr<RawImage> image, int index)
{
if(image == nullptr)return;
m_currentImg = index;
if(!image || !image->valid())
{
m_imgWidth = 0;
m_imgHeight = 0;
m_error = tr("Failed to load image");
m_rawImage.reset();
update();
return;
}
m_error.clear();
makeCurrent();
m_rawImage = image;
m_rawImage->downscaleTo(m_maxTextureSize);
if((int)image->width() > m_maxTextureSize || (int)image->height() > m_maxTextureSize)
m_rawImage->resize(std::min(m_maxTextureSize, (int)image->width()), std::min(m_maxTextureSize, (int)image->height()));
m_imgWidth = image->width();
m_imgHeight = image->height();
m_currentImg = index;
if(!m_image)return;
@@ -115,16 +135,11 @@ void ImageWidget::setImage(std::shared_ptr<RawImage> image, int index)
m_unit_scale[0] = 1.0f;
m_unit_scale[1] = 0.0f;
auto &stats = image->imageStats();
if(image->type() == RawImage::FLOAT32 || image->type() == RawImage::FLOAT64)
if(image->type() == RawImage::FLOAT32)
{
float min = *std::min_element(stats.m_min, stats.m_min + 4);
float max = *std::max_element(stats.m_max, stats.m_max + 4);
if(min < 0.0f || max > 1.0f)
{
m_unit_scale[0] = 1.0f / (max - min);
m_unit_scale[1] = min * m_unit_scale[0];
}
auto unitScaling = image->unitScale();
m_unit_scale[0] = unitScaling.first;
m_unit_scale[1] = unitScaling.second;
}
if(m_debayerTex)
@@ -173,12 +188,6 @@ void ImageWidget::bestFit()
setOffset(0, 0);
}
void ImageWidget::blockRepaint(bool block)
{
m_blockRepaint = block;
if(!block)update();
}
void ImageWidget::allocateThumbnails(const QStringList &paths)
{
makeCurrent();
@@ -196,7 +205,7 @@ void ImageWidget::allocateThumbnails(const QStringList &paths)
m_thumbnailTexture->create();
m_thumbnailTexture->setFormat(QOpenGLTexture::RGB16_UNorm);
m_thumbnailTexture->setSize(THUMB_SIZE, THUMB_SIZE);
m_thumbnailTexture->setLayers(paths.size());
m_thumbnailTexture->setLayers(std::min((int)paths.size(), m_maxArrayLayers));
m_thumbnailTexture->allocateStorage();
}
@@ -213,6 +222,18 @@ QVector2D ImageWidget::getImagePixelCoord(const QVector2D &pos)
return (pos + offset) / m_scale;
}
void ImageWidget::setBayerMask(int mask)
{
m_firstRed[0] = mask & 0x1;
m_firstRed[1] = (mask & 0x2) >> 1;
if(m_debayerTex)
{
f->glDeleteTextures(1, &m_debayerTex);
m_debayerTex = 0;
}
update();
}
void ImageWidget::setMTFParams(const MTFParam &params)
{
m_mtfParams = params;
@@ -274,9 +295,12 @@ QImage ImageWidget::renderToImage()
void ImageWidget::thumbnailLoaded(const Image *image)
{
if(image->number() >= m_maxArrayLayers)
return;
makeCurrent();
const RawImage *raw = image->thumbnail();
if(!raw)return;
if(!raw || !raw->valid())return;
m_thumbnailTexture->setData(0, image->number(), QOpenGLTexture::RGBA, QOpenGLTexture::UInt16, raw->data(), m_transferOptions.get());
float a = raw->thumbAspect();
int sizes[3] = { std::max(1, a > 1.0f ? THUMB_SIZE : (int)(THUMB_SIZE * a)), std::max(1, a < 1.0f ? THUMB_SIZE : (int)(THUMB_SIZE / a)), image->number() };
@@ -293,8 +317,6 @@ void ImageWidget::showThumbnail(bool enable)
void ImageWidget::paintGL()
{
if(m_blockRepaint)return;
float dx = m_dx;
float dy = m_dy;
if(m_width > m_image->width() * m_scale)
@@ -303,6 +325,7 @@ void ImageWidget::paintGL()
dy = -height() * 0.5f + m_image->height() * m_scale * 0.5f;
QBrush highlight = style()->standardPalette().highlight();
f->glClear(GL_COLOR_BUFFER_BIT);
if(m_showThumbnails)
{
m_vaoThumb->bind();
@@ -336,7 +359,9 @@ void ImageWidget::paintGL()
QPainter painter(this);
const int w = width()/THUMB_SIZE_BORDER;
const int off = (THUMB_SIZE_BORDER - THUMB_SIZE) / 2;
for(int i=0; i < m_thumbnailCount; i++)
int start = std::max((int)(m_dy / THUMB_SIZE_BORDER_Y * w - w), 0);
int end = std::min((int)(m_dy + m_height) / THUMB_SIZE_BORDER_Y * w + w, m_thumbnailCount);
for(int i=start; i < end; i++)
{
float x = (i % w) * THUMB_SIZE_BORDER;
float y = i / w * THUMB_SIZE_BORDER_Y + THUMB_SIZE - m_dy + off;
@@ -361,6 +386,13 @@ void ImageWidget::paintGL()
}
}
}
else if(!m_error.isEmpty())
{
QPainter painter(this);
painter.setPen(Qt::red);
painter.setFont(QFont("Sans", 24, QFont::Bold));
painter.drawText(0, 0, width(), height(), Qt::AlignCenter | Qt::AlignHCenter, m_error);
}
else
{
debayer();
@@ -380,6 +412,7 @@ void ImageWidget::paintGL()
m_program->setUniformValue("bw", m_bwImg && !m_superpixel);
m_program->setUniformValue("false_color", m_falseColor && m_bwImg);
m_program->setUniformValue("invert", m_invert);
m_program->setUniformValue("filtering", m_scale > 1.0f ? FILTERING : 1);
#ifdef COLOR_MANAGMENT
m_program->setUniformValue("srgb", m_srgb);
#endif
@@ -401,7 +434,7 @@ void ImageWidget::initializeGL()
{
f = context()->functions();
f->glClearColor(0.5f, 0.5f, 0.5f, 1.0f);
f3 = context()->versionFunctions<QOpenGLFunctions_3_3_Core>();
f3 = QOpenGLVersionFunctionsFactory::get<QOpenGLFunctions_3_3_Core>(context());
if(f3 == nullptr)
QMessageBox::critical(this, tr("OpenGL error"), tr("Could not initialize OpenGL 3.3 context. Ensure that proper GPU driver is installed."));
@@ -444,8 +477,8 @@ void ImageWidget::initializeGL()
// f->glVertexAttribPointer(0, 2, GL_FLOAT, false, sizeof(float)*4, 0);
m_program = std::unique_ptr<QOpenGLShaderProgram>(new QOpenGLShaderProgram);
m_program->addShaderFromSourceFile(QOpenGLShader::Vertex, ":/shaders/image.vert");
m_program->addShaderFromSourceFile(QOpenGLShader::Fragment, ":/shaders/image.frag");
m_program->addShaderFromSourceFile(QOpenGLShader::Vertex, ":/image.vert");
m_program->addShaderFromSourceFile(QOpenGLShader::Fragment, ":/image.frag");
if(!m_program->link())
{
@@ -461,8 +494,8 @@ void ImageWidget::initializeGL()
m_program->setUniformValue("scale", 1.0f, 0.0f);
m_debayerProgram = std::unique_ptr<QOpenGLShaderProgram>(new QOpenGLShaderProgram);
m_debayerProgram->addShaderFromSourceFile(QOpenGLShader::Vertex, ":/shaders/debayer.vert");
m_debayerProgram->addShaderFromSourceFile(QOpenGLShader::Fragment, ":/shaders/debayer.frag");
m_debayerProgram->addShaderFromSourceFile(QOpenGLShader::Vertex, ":/debayer.vert");
m_debayerProgram->addShaderFromSourceFile(QOpenGLShader::Fragment, ":/debayer.frag");
m_debayerProgram->bind();
m_debayerProgram->enableAttributeArray("qt_Vertex");
@@ -478,8 +511,8 @@ void ImageWidget::initializeGL()
m_vaoThumb->bind();
m_thumbnailProgram = std::unique_ptr<QOpenGLShaderProgram>(new QOpenGLShaderProgram);
m_thumbnailProgram->addShaderFromSourceFile(QOpenGLShader::Vertex, ":/shaders/thumb.vert");
m_thumbnailProgram->addShaderFromSourceFile(QOpenGLShader::Fragment, ":/shaders/thumb.frag");
m_thumbnailProgram->addShaderFromSourceFile(QOpenGLShader::Vertex, ":/thumb.vert");
m_thumbnailProgram->addShaderFromSourceFile(QOpenGLShader::Fragment, ":/thumb.frag");
m_thumbnailProgram->bind();
m_thumbnailProgram->enableAttributeArray("qt_Vertex");
@@ -559,7 +592,7 @@ void ImageWidget::mousePressEvent(QMouseEvent *event)
else
{
if(event->button() == Qt::LeftButton)
m_lastPos = event->localPos();
m_lastPos = event->position();
}
}
@@ -571,8 +604,8 @@ void ImageWidget::mouseMoveEvent(QMouseEvent *event)
}
else if(!m_lastPos.isNull())
{
QPointF off = event->localPos() - m_lastPos;
m_lastPos = event->localPos();
QPointF off = event->position() - m_lastPos;
m_lastPos = event->position();
setOffset(m_dx - off.x(), m_dy - off.y());
return;
}
@@ -638,7 +671,7 @@ void ImageWidget::wheelEvent(QWheelEvent *event)
else
{
if(std::abs(event->angleDelta().y()) > 15)
zoom(event->angleDelta().y(), event->modifiers() & Qt::ShiftModifier ? QPointF() : event->posF());
zoom(event->angleDelta().y(), event->modifiers() & Qt::ShiftModifier ? QPointF() : event->position());
}
}
@@ -681,6 +714,7 @@ void ImageWidget::debayer()
f->glViewport(0, 0, m_imgWidth, m_imgHeight);
m_debayerProgram->bind();
f->glUniform2i(m_debayerProgram->uniformLocation("firstRed"), m_firstRed[0], m_firstRed[1]);
m_image->bind(0);
f->glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
+3 -2
View File
@@ -53,7 +53,6 @@ class ImageWidget : public QOpenGLWidget
float m_scale = 1.0f;
int m_scaleStop = 0;
bool m_bestFit = false;
bool m_blockRepaint = false;
bool m_bwImg = false;
bool m_falseColor = false;
bool m_invert = false;
@@ -65,9 +64,11 @@ class ImageWidget : public QOpenGLWidget
int m_thumbnailCount = 0;
int m_maxTextureSize = 0;
int m_maxArrayLayers = 0;
int m_firstRed[2] = {0, 0};
QVector<ImageThumb> m_thumnails;
Database *m_database = nullptr;
QPointF m_lastPos;
QString m_error;
public:
explicit ImageWidget(Database *database, QWidget *parent = nullptr);
~ImageWidget() override;
@@ -76,9 +77,9 @@ public:
void setWCS(std::shared_ptr<WCSData> wcs);
void zoom(int zoom, const QPointF &mousePos = QPointF());
void bestFit();
void blockRepaint(bool block);
void allocateThumbnails(const QStringList &paths);
QVector2D getImagePixelCoord(const QVector2D &pos);
void setBayerMask(int mask);
public slots:
void setMTFParams(const MTFParam &params);
void setOffset(float dx, float dy);
+2 -2
View File
@@ -1,5 +1,5 @@
find_program(XDG-DESKTOP-MENU_EXECUTABLE xdg-desktop-menu)
find_program(XDG-ICON-RESOURCE_EXECUTABLE xdg-icon-resource)
execute_process(COMMAND ${XDG-DESKTOP-MENU_EXECUTABLE} install --novendor space.nouspiro.tenmon.desktop WORKING_DIRECTORY ${CMAKE_CURRENT_LIST_DIR})
execute_process(COMMAND ${XDG-ICON-RESOURCE_EXECUTABLE} install --novendor --size 64 space.nouspiro.tenmon.png space.nouspiro.tenmon WORKING_DIRECTORY ${CMAKE_CURRENT_LIST_DIR})
execute_process(COMMAND ${XDG-ICON-RESOURCE_EXECUTABLE} install --novendor --size 128 space.nouspiro.tenmon_128.png space.nouspiro.tenmon WORKING_DIRECTORY ${CMAKE_CURRENT_LIST_DIR})
execute_process(COMMAND ${XDG-ICON-RESOURCE_EXECUTABLE} install --novendor --size 64 resources/space.nouspiro.tenmon.png space.nouspiro.tenmon WORKING_DIRECTORY ${CMAKE_CURRENT_LIST_DIR})
execute_process(COMMAND ${XDG-ICON-RESOURCE_EXECUTABLE} install --novendor --size 128 resources/space.nouspiro.tenmon_128.png space.nouspiro.tenmon WORKING_DIRECTORY ${CMAKE_CURRENT_LIST_DIR})
+1 -1
Submodule libXISF updated: 8a1f305cc7...d00de2041d
+169 -103
View File
@@ -7,6 +7,7 @@
#include <QElapsedTimer>
#include <QDebug>
#include <iostream>
#include <algorithm>
#include <libexif/exif-data.h>
#include <fitsio2.h>
#include <libxisf.h>
@@ -190,19 +191,22 @@ bool loadFITS(const QString path, ImageInfoData &info, std::shared_ptr<RawImage>
{
fitsfile *file;
int status = 0;
int type;
fits_open_image(&file, path.toLocal8Bit().data(), READONLY, &status);
fits_get_hdu_type(file, &type, &status);
int type = -1;
fits_open_diskfile(&file, path.toLocal8Bit().data(), READONLY, &status);
int num = 0;
fits_get_num_hdus(file, &num, &status);
if(type == IMAGE_HDU)
{
int imgtype;
int naxis;
long naxes[3] = {0};
for(int i=1; i <= num; i++)
{
fits_movabs_hdu(file, i, IMAGE_HDU, &status);
fits_get_hdu_type(file, &type, &status);
fits_get_img_param(file, 3, &imgtype, &naxis, naxes, &status);
fits_get_img_equivtype(file, &imgtype, &status);
if(naxis >= 2 && naxis <= 3 && status == 0)
if(type == IMAGE_HDU && naxis >= 2 && naxis <= 3 && status == 0)
{
RawImage::DataType type;
int fitstype;
@@ -268,12 +272,23 @@ bool loadFITS(const QString path, ImageInfoData &info, std::shared_ptr<RawImage>
if(image)
image->convertToGLFormat();
break;
}
}
noload:
if(file)
loadFITSHeader(file, info);
if(image)
{
for(auto fits : info.fitsHeader)
{
if(fits.key == "ROWORDER" && fits.value == "BOTTOM-UP")
image->flip();
}
}
fits_close_file(file, &status);
if(status)
{
@@ -363,44 +378,10 @@ void LoadRunable::run()
info.info.append({QObject::tr("Filename"), finfo.fileName()});
std::shared_ptr<RawImage> rawImage;
timer.start();
if(m_file.endsWith(".CR2", Qt::CaseInsensitive) || m_file.endsWith(".NEF", Qt::CaseInsensitive) || m_file.endsWith(".DNG", Qt::CaseInsensitive))
{
loadRAW(m_file, info, rawImage);
qDebug() << "LoadRAW" << timer.elapsed();
}
else if(m_file.endsWith(".FIT", Qt::CaseInsensitive) || m_file.endsWith(".FITS", Qt::CaseInsensitive))
{
loadFITS(m_file, info, rawImage);
qDebug() << "LoadFITS" << timer.elapsed();
}
else if(m_file.endsWith(".XISF", Qt::CaseInsensitive))
{
loadXISF(m_file, info, rawImage);
qDebug() << "LoadXISF" << timer.elapsed();
}
else
{
QImage img(m_file);
#ifdef COLOR_MANAGMENT
if(img.colorSpace().isValid() && img.colorSpace() != QColorSpace::SRgb)
img.convertToColorSpace(QColorSpace::SRgb);
#endif
if(!loadImage(m_file, info, rawImage))
info.info.append({QObject::tr("Error"), QObject::tr("Failed to load image")});
ExifData *exif = exif_data_new_from_file(m_file.toLocal8Bit().constData());
info.info.append({QObject::tr("Width"), QString::number(img.width())});
info.info.append({QObject::tr("Height"), QString::number(img.height())});
if(exif)
{
loadExifEntry(info, exif->ifd[EXIF_IFD_EXIF], EXIF_TAG_ISO_SPEED_RATINGS);
loadExifEntry(info, exif->ifd[EXIF_IFD_EXIF], EXIF_TAG_SHUTTER_SPEED_VALUE);
exif_data_free(exif);
}
rawImage = std::make_shared<RawImage>(img);
qDebug() << "LoadQImage" << timer.elapsed();
}
if(rawImage /*&& m_analyzeLevel >= Statistics*/ && !m_thumbnail)
if(rawImage && !m_thumbnail)
{
timer.start();
rawImage->calcStats();
@@ -428,68 +409,27 @@ void LoadRunable::run()
.arg(100.0 * stats.m_saturated[1] / rawImage->size())
.arg(100.0 * stats.m_saturated[2] / rawImage->size())});
}
if(m_analyzeLevel >= Peaks)
{
std::vector<Peak> peaks;
/*RawImage *medianImage = rawImage->medianFilter();
qDebug() << "median" << timer.restart();
int numPeaks = medianImage->findPeaks(median+stdDev*2, 20, peaks);
delete medianImage;
qDebug() << "peaks" << timer.restart();
//if(m_analyzeLevel == Peaks)
// drawPeaks(img, peaks);
qDebug() << "draw peaks" << timer.restart();
info.info.append({QObject::tr("Peaks"), QString::number(numPeaks)});
//info.info.append({QObject::tr("Peaks draw"), QString::number(peaks.size())});
if(m_analyzeLevel>= Stars)
{
double fwhmX = 0;
double fwhmY = 0;
const int radius = 13;
StarFit starFit(radius);
std::vector<Star> stars;
for(uint i=0; i<peaks.size(); i++)
{
Peak p = peaks[i];
std::vector<double> r;
int x = p.x();
int y = p.y();
rawImage->rect(x, y, radius, radius, r);
Star star = starFit.fitStar(r, false);
if(star.valid())
{
//printStarModel(radius, r, star);
star.m_x += x;
star.m_y += y;
fwhmX += star.fwhmX();
fwhmY += star.fwhmY();
stars.push_back(star);
}
}
//drawStars(img, stars);
info.info.append({QObject::tr("FWHM X"), QString::number(fwhmX/stars.size())});
info.info.append({QObject::tr("FWHM Y"), QString::number(fwhmY/stars.size())});
}
qDebug() << "Star fit" << timer.restart();*/
}
}
if(m_thumbnail)
{
if(rawImage)
if(rawImage && rawImage->valid())
{
if(QUALITY_RESIZE)
rawImage->resize(THUMB_SIZE, THUMB_SIZE);
rawImage->convertToThumbnail();
}
QMetaObject::invokeMethod(m_receiver, "thumbnailLoadFinish", Qt::QueuedConnection, Q_ARG(std::shared_ptr<RawImage>, rawImage));
}
}
else
{
QMetaObject::invokeMethod(m_receiver, "imageLoaded", Qt::QueuedConnection, Q_ARG(std::shared_ptr<RawImage>, rawImage), Q_ARG(ImageInfoData, info));
}
}
catch(std::exception e)
{
qDebug() << e.what();
qDebug() << m_file << e.what();
}
}
@@ -537,10 +477,56 @@ bool readXISFHeader(const QString &path, ImageInfoData &info)
return true;
}
ConvertRunable::ConvertRunable(const QString &in, const QString &out, const QString &format) :
bool loadImage(const QString &path, ImageInfoData &info, std::shared_ptr<RawImage> &rawImage)
{
bool ret = false;
QElapsedTimer timer;
timer.start();
if(path.endsWith(".CR2", Qt::CaseInsensitive) || path.endsWith(".CR3", Qt::CaseInsensitive) || path.endsWith(".NEF", Qt::CaseInsensitive) || path.endsWith(".DNG", Qt::CaseInsensitive))
{
ret = loadRAW(path, info, rawImage);
qDebug() << "LoadRAW" << timer.elapsed();
}
else if(path.endsWith(".FIT", Qt::CaseInsensitive) || path.endsWith(".FITS", Qt::CaseInsensitive))
{
ret = loadFITS(path, info, rawImage);
qDebug() << "LoadFITS" << timer.elapsed();
}
else if(path.endsWith(".XISF", Qt::CaseInsensitive))
{
ret = loadXISF(path, info, rawImage);
qDebug() << "LoadXISF" << timer.elapsed();
}
else
{
QImage img(path);
#ifdef COLOR_MANAGMENT
if(img.colorSpace().isValid() && img.colorSpace() != QColorSpace::SRgb)
img.convertToColorSpace(QColorSpace::SRgb);
#endif
ExifData *exif = exif_data_new_from_file(path.toLocal8Bit().constData());
info.info.append({QObject::tr("Width"), QString::number(img.width())});
info.info.append({QObject::tr("Height"), QString::number(img.height())});
if(exif)
{
loadExifEntry(info, exif->ifd[EXIF_IFD_EXIF], EXIF_TAG_ISO_SPEED_RATINGS);
loadExifEntry(info, exif->ifd[EXIF_IFD_EXIF], EXIF_TAG_SHUTTER_SPEED_VALUE);
exif_data_free(exif);
}
rawImage = std::make_shared<RawImage>(img);
qDebug() << "LoadQImage" << timer.elapsed();
ret = !img.isNull();
}
return ret;
}
ConvertRunable::ConvertRunable(const QString &in, const QString &out, const QString &format, const ConvertParams &params, QSemaphore *semaphore) :
m_infile(in),
m_outfile(out),
m_format(format)
m_format(format),
m_params(params),
m_semaphore(semaphore)
{
}
@@ -587,10 +573,12 @@ void writeFITSImage(fitsfile *fw, std::shared_ptr<RawImage> rawimage, ImageInfoD
fits_write_pix(fw, TFLOAT, firstpix, rawimage->size(), planes[i].data(), &status);
}
break;
default:
return;
}
for(const FITSRecord &record : imageinfo.fitsHeader)
{
if(skipKeys.contains(record.key))continue;
if(skipKeys.contains(record.key) || record.xisf)continue;
bool isdouble;
bool isint;
@@ -616,16 +604,18 @@ void writeFITSImage(fitsfile *fw, std::shared_ptr<RawImage> rawimage, ImageInfoD
void ConvertRunable::run()
{
QSemaphoreReleaser release;
if(m_semaphore)release = QSemaphoreReleaser(m_semaphore);
ImageInfoData imageinfo;
std::shared_ptr<RawImage> rawimage;
if(m_infile.endsWith(".FITS", Qt::CaseInsensitive) || m_infile.endsWith(".FIT", Qt::CaseInsensitive))
loadFITS(m_infile, imageinfo, rawimage);
if(m_infile.endsWith(".XISF", Qt::CaseInsensitive))
loadXISF(m_infile, imageinfo, rawimage);
loadImage(m_infile, imageinfo, rawimage);
QFileInfo info(m_outfile);
info.dir().mkpath(".");
if(rawimage)
{
if(m_format == "XISF")
if(m_format == "xisf")
{
try
{
@@ -640,16 +630,43 @@ void ConvertRunable::run()
default: return;
}
LibXISF::Image image(rawimage->width(), rawimage->height(), channelCount, sampleFormat, channelCount == 1 ? LibXISF::Image::Gray : LibXISF::Image::RGB, LibXISF::Image::Normal);
LibXISF::Image image(rawimage->width(), rawimage->height(), channelCount, sampleFormat, channelCount == 1 ? LibXISF::Image::Gray : LibXISF::Image::RGB, LibXISF::Image::Planar);
if(channelCount == 1)
{
std::memcpy(image.imageData(), rawimage->data(), image.imageDataSize());
}
else
{
size_t off = 0;
std::vector<RawImage> planes = rawimage->split();
for(const auto &plane : planes)
{
std::memcpy(image.imageData<uint8_t>() + off, plane.data(), plane.size() * RawImage::typeSize(plane.type()));
off += plane.size() * RawImage::typeSize(plane.type());
}
}
for(auto &record : imageinfo.fitsHeader)
{
if(record.value.type() == QVariant::Bool)
if(record.xisf)continue;
if(record.value.typeId() == QMetaType::Bool)
image.addFITSKeyword({record.key.toStdString(), record.value.toBool() ? "T" : "F", record.comment.toStdString()});
else
image.addFITSKeyword({record.key.toStdString(), record.value.toString().toStdString(), record.comment.toStdString()});
}
if(m_params.compressionType.startsWith("zstd") && LibXISF::DataBlock::CompressionCodecSupported(LibXISF::DataBlock::ZSTD))
image.setCompression(LibXISF::DataBlock::ZSTD, m_params.compressionLevel);
else if(m_params.compressionType.startsWith("lz4hc"))
image.setCompression(LibXISF::DataBlock::LZ4HC, m_params.compressionLevel);
else if(m_params.compressionType.startsWith("lz4"))
image.setCompression(LibXISF::DataBlock::LZ4, m_params.compressionLevel);
else if(m_params.compressionType.startsWith("zlib"))
image.setCompression(LibXISF::DataBlock::Zlib, m_params.compressionLevel);
if(m_params.compressionType.endsWith("+sh"))
image.setByteshuffling(true);
xisf.writeImage(image);
xisf.save(m_outfile.toLocal8Bit().data());
}
@@ -657,16 +674,65 @@ void ConvertRunable::run()
{
qDebug() << "Failed to save XISF image" << err.what();
}
return;
}
if(m_format == "FITS")
if(m_format == "fits")
{
int status = 0;
fitsfile *fw;
if(QFileInfo(m_outfile).exists())QFile::remove(m_outfile);
fits_create_diskfile(&fw, m_outfile.toLocal8Bit().data(), &status);
if(!m_params.compressionType.isEmpty())
{
if(m_params.compressionType == "gzip")
fits_set_compression_type(fw, GZIP_1, &status);
else if(m_params.compressionType == "rice")
fits_set_compression_type(fw, RICE_1, &status);
}
writeFITSImage(fw, rawimage, imageinfo);
fits_close_file(fw, &status);
return;
}
// if nothing else try QImage
{
QImage::Format format = QImage::Format_Invalid;
int width = rawimage->widthBytes();
switch(rawimage->type())
{
case RawImage::UINT8:
if(rawimage->channels() == 1)format = QImage::Format_Grayscale8;
else if(rawimage->channels() == 3)format = QImage::Format_RGBX8888;
else if(rawimage->channels() == 4)format = QImage::Format_RGBA8888;
break;
case RawImage::UINT16:
if(rawimage->channels() == 1)format = QImage::Format_Grayscale16;
else if(rawimage->channels() == 3)format = QImage::Format_RGBX64;
else if(rawimage->channels() == 4)format = QImage::Format_RGBA64;
width *= 2;
break;
default:
return;
}
if(format == QImage::Format_Invalid)return;
QImage qimage(rawimage->width(), rawimage->height(), format);
for(uint32_t i=0; i < rawimage->height(); i++)
std::memcpy(qimage.scanLine(i), rawimage->data(i), width);
qimage.save(m_outfile);
}
}
}
ConvertRunable::ConvertParams::ConvertParams(const QVariantMap &map)
{
bool ok = false;
if(map.contains("compressionLevel"))
compressionLevel = std::clamp(map["compressionLevel"].toInt(&ok), -1, 100);
if(!ok)compressionLevel = -1;
if(map.contains("compressionType"))
compressionType = map["compressionType"].toString();
}
+18 -3
View File
@@ -3,10 +3,14 @@
#include <QRunnable>
#include <QString>
#include <QSemaphore>
#include "imageinfo.h"
class RawImage;
bool readFITSHeader(const QString &path, ImageInfoData &info);
bool readXISFHeader(const QString &path, ImageInfoData &info);
bool loadImage(const QString &path, ImageInfoData &info, std::shared_ptr<RawImage> &rawImage);
class Image;
@@ -21,14 +25,25 @@ public:
void run() override;
};
class ConvertRunable : public QRunnable
{
public:
struct ConvertParams
{
int compressionLevel = -1;
QString compressionType;
ConvertParams(){}
ConvertParams(const QVariantMap &map);
};
ConvertRunable(const QString &in, const QString &out, const QString &format, const ConvertParams &params = ConvertParams(), QSemaphore *semaphore = nullptr);
void run() override;
private:
QString m_infile;
QString m_outfile;
QString m_format;
public:
ConvertRunable(const QString &in, const QString &out, const QString &format);
void run() override;
ConvertParams m_params;
QSemaphore *m_semaphore;
};
#endif // LOADRUNABLE_H
+86 -28
View File
@@ -10,6 +10,7 @@
#include <QProgressDialog>
#include <QDebug>
#include <QDockWidget>
#include <QActionGroup>
#include <signal.h>
#include <unistd.h>
#include <QSettings>
@@ -24,6 +25,7 @@
#include "statusbar.h"
#include "settingsdialog.h"
#include "histogram.h"
#include "batchprocessing.h"
#ifdef __linux__
#include <sys/ioctl.h>
@@ -57,9 +59,10 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent)
_openFilter.append(" ");
nameFilter.append(mimeType.suffixes());
}
_openFilter.append("*.fit *.fits *.xisf *.cr2 *.nef *.dng)");
_openFilter.append("*.fit *.fits *.xisf *.cr2 *.cr3 *.nef *.dng)");
_openFilter.append(tr(";;All files (*)"));
nameFilter.append({"fit", "fits", "xisf", "cr2", "nef", "dng"});
nameFilter.append({"fit", "fits", "xisf", "cr2", "cr3", "nef", "dng"});
QImageReader::setAllocationLimit(0);
m_info = new ImageInfo(this);
QDockWidget *infoDock = new QDockWidget(tr("Image info"), this);
@@ -140,16 +143,22 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent)
connect(m_imageGL->imageWidget(), &ImageWidget::fileDropped, this, static_cast<void (MainWindow::*)(const QString &)>(&MainWindow::loadFile));
QMenu *fileMenu = new QMenu(tr("File"), this);
fileMenu->addAction(tr("Open"), this, SLOT(loadFile()), QKeySequence::Open);
fileMenu->addAction(tr("Save as"), this, SLOT(saveAs()), QKeySequence::Save);
fileMenu->addAction(tr("Open"), QKeySequence::Open, this, SLOT(loadFile()));
fileMenu->addAction(tr("Open directory recursively"), this, &MainWindow::loadDir);
fileMenu->addAction(tr("Save as"), QKeySequence::Save, this, SLOT(saveAs()));
fileMenu->addSeparator();
fileMenu->addAction(tr("Copy marked files"), this, SLOT(copyMarked()), Qt::Key_F5);
fileMenu->addAction(tr("Move marked files"), this, SLOT(moveMarked()), Qt::Key_F6);
fileMenu->addAction(tr("Move marked files to trash"), this, &MainWindow::deleteMarked, QKeySequence::Delete);
fileMenu->addAction(tr("Copy marked files"), Qt::Key_F5, this, SLOT(copyMarked()));
fileMenu->addAction(tr("Move marked files"), Qt::Key_F6, this, SLOT(moveMarked()));
fileMenu->addAction(tr("Move marked files to trash"), QKeySequence::Delete, this, &MainWindow::deleteMarked);
fileMenu->addSeparator();
fileMenu->addAction(tr("Index directory"), this, SLOT(indexDir()));
fileMenu->addAction(tr("Reindex files"), this, SLOT(reindex()));
fileMenu->addAction(tr("Export database to CSV"), this, &MainWindow::exportCSV);
fileMenu->addAction(tr("Batch processing"), Qt::Key_B | Qt::CTRL, [this](){
BatchProcessing *batchProcessing = new BatchProcessing(this);
batchProcessing->exec();
delete batchProcessing;
});
fileMenu->addSeparator();
QAction *liveModeAction = fileMenu->addAction(tr("Live mode"), this, SLOT(liveMode(bool)));
liveModeAction->setCheckable(true);
@@ -162,30 +171,52 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent)
menuBar()->addMenu(editMenu);
QMenu *viewMenu = new QMenu(tr("View"), this);
viewMenu->addAction(tr("Zoom In"), m_imageGL, SLOT(zoomIn()), QKeySequence::ZoomIn);
viewMenu->addAction(tr("Zoom Out"), m_imageGL, SLOT(zoomOut()), QKeySequence::ZoomOut);
viewMenu->addAction(tr("Best Fit"), m_imageGL, SLOT(bestFit()), QKeySequence("Ctrl+1"));
viewMenu->addAction(tr("Zoom In"), QKeySequence::ZoomIn, m_imageGL, SLOT(zoomIn()));
viewMenu->addAction(tr("Zoom Out"), QKeySequence::ZoomOut, m_imageGL, SLOT(zoomOut()));
viewMenu->addAction(tr("Best Fit"), QKeySequence("Ctrl+1"), m_imageGL, SLOT(bestFit()));
viewMenu->addAction(tr("100%"), m_imageGL, SLOT(oneToOne()));
QAction *thumbnailsAction = viewMenu->addAction(tr("Thumbnails"), [this](bool checked){
viewMenu->addSeparator();
QMenu *bayerMenu = viewMenu->addMenu(tr("Bayer mask"));
QActionGroup *bayerActionGroup = new QActionGroup(this);
QAction *rggbAction = bayerActionGroup->addAction(tr("RGGB"));//0 0
QAction *grbgAction = bayerActionGroup->addAction(tr("GRBG"));//1 0
QAction *gbrgAction = bayerActionGroup->addAction(tr("GBRG"));//0 1
QAction *bggrAction = bayerActionGroup->addAction(tr("BGGR"));//1 1
rggbAction->setCheckable(true); rggbAction->setData(0); rggbAction->setIcon(QIcon(":/bayer.png"));
grbgAction->setCheckable(true); grbgAction->setData(1); grbgAction->setIcon(QIcon(":/grbg.png"));
gbrgAction->setCheckable(true); gbrgAction->setData(2); gbrgAction->setIcon(QIcon(":/gbrg.png"));
bggrAction->setCheckable(true); bggrAction->setData(3); bggrAction->setIcon(QIcon(":/bggr.png"));
bayerMenu->addActions({rggbAction, grbgAction, gbrgAction, bggrAction});
viewMenu->addMenu(bayerMenu);
connect(bayerActionGroup, &QActionGroup::triggered, [this](QAction *action){
int data = action->data().toInt();
m_imageGL->imageWidget()->setBayerMask(data);
QSettings settings;
settings.setValue("mainwindow/bayermask", data);
});
QAction *thumbnailsAction = viewMenu->addAction(tr("Thumbnails"), Qt::Key_F2, [this](bool checked){
if(SettingsDialog::loadThumbsizes())m_ringList->clearThumbnails();
m_imageGL->imageWidget()->allocateThumbnails(m_ringList->imageNames());
m_imageGL->imageWidget()->showThumbnail(checked);
if(checked)m_ringList->loadThumbnails();
else m_ringList->stopLoading();
}, Qt::Key_F2);
});
thumbnailsAction->setCheckable(true);
QAction *slideshowAction = viewMenu->addAction(tr("Slideshow"), Qt::Key_F3, m_ringList, &ImageRingList::toggleSlideshow);
slideshowAction->setCheckable(true);
menuBar()->addMenu(viewMenu);
QMenu *selectMenu = new QMenu(tr("Select"), this);
selectMenu->addAction(tr("Mark"), this, SLOT(markImage()), Qt::Key_F7);
selectMenu->addAction(tr("Unmark"), this, SLOT(unmarkImage()), Qt::Key_F8);
selectMenu->addAction(tr("Mark"), Qt::Key_F7, this, SLOT(markImage()));
selectMenu->addAction(tr("Unmark"), Qt::Key_F8, this, SLOT(unmarkImage()));
selectMenu->addSeparator();
selectMenu->addAction(tr("Mark and next"), this, SLOT(markAndNext()), Qt::Key_M);
selectMenu->addAction(tr("Unmark and next"), this, SLOT(unmarkAndNext()), Qt::Key_X);
selectMenu->addAction(tr("Mark and next"), Qt::Key_M, this, SLOT(markAndNext()));
selectMenu->addAction(tr("Unmark and next"), Qt::Key_X, this, SLOT(unmarkAndNext()));
selectMenu->addAction(tr("Show marked"), this, &MainWindow::showMarkFilesDialog);
menuBar()->addMenu(selectMenu);
QMenu *analyzeMenu = new QMenu(tr("Analyze"), this);
/*QMenu *analyzeMenu = new QMenu(tr("Analyze"), this);
QActionGroup *analyzeGroup = new QActionGroup(this);
connect(analyzeGroup, &QActionGroup::triggered, [](QAction* action) {
static QAction* lastAction = nullptr;
@@ -208,7 +239,7 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent)
connect(peakAction, SIGNAL(toggled(bool)), this, SLOT(peakFinder(bool)));
connect(starAction, SIGNAL(toggled(bool)), this, SLOT(starFinder(bool)));
analyzeMenu->addActions({statsAction, peakAction, starAction});
//menuBar()->addMenu(analyzeMenu);
menuBar()->addMenu(analyzeMenu);*/
QMenu *dockMenu = new QMenu(tr("Docks"), this);
dockMenu->addAction(infoDock->toggleViewAction());
@@ -220,7 +251,7 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent)
menuBar()->addMenu(dockMenu);
QMenu *helpMenu = menuBar()->addMenu(tr("Help"));
helpMenu->addAction(tr("Help"), [this]{ HelpDialog help(this); help.exec(); }, QKeySequence::HelpContents);
helpMenu->addAction(tr("Help"), QKeySequence::HelpContents, [this]{ HelpDialog help(this); help.exec(); });
helpMenu->addAction(tr("About Tenmon"), [this]{ About about(this); about.exec(); });
helpMenu->addAction(tr("About Qt"), [this](){ QMessageBox::aboutQt(this); });
@@ -228,6 +259,19 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent)
QSettings settings;
restoreGeometry(settings.value("mainwindow/geometry").toByteArray());
restoreState(settings.value("mainwindow/state").toByteArray());
int bayermask = settings.value("mainwindow/bayermask", 0).toInt();
switch(bayermask)
{
default:
case 0:
rggbAction->setChecked(true); break;
case 1:
grbgAction->setChecked(true); break;
case 2:
gbrgAction->setChecked(true); break;
case 3:
bggrAction->setChecked(true); break;
}
QStringList standardLocations = QStandardPaths::standardLocations(QStandardPaths::PicturesLocation);
if(standardLocations.size())
@@ -444,6 +488,20 @@ void MainWindow::loadFile(int row)
m_ringList->loadFile(row);
}
void MainWindow::loadDir()
{
QString dir = QFileDialog::getExistingDirectory(this,
tr("Open directory recursively"),
_lastDir);
if(!dir.isEmpty())
{
_lastDir = dir;
m_ringList->setDir(dir, QString(), true);
QSettings settings;
settings.setValue("mainwindow/lastdir", _lastDir);
}
}
void MainWindow::indexDir()
{
QString dir = QFileDialog::getExistingDirectory(this, tr("Index directory"), _lastDir, QFileDialog::ShowDirsOnly);
@@ -478,21 +536,21 @@ void MainWindow::saveAs()
auto filterToFormat = [](const QString &file, const QString &filter) -> const char*
{
QString suffix = QFileInfo(file).suffix();
if(!suffix.compare("jpg", Qt::CaseInsensitive) || !suffix.compare("jpeg", Qt::CaseInsensitive))return "JPEG";
if(!suffix.compare("png", Qt::CaseInsensitive))return "PNG";
if(!suffix.compare("fits", Qt::CaseInsensitive) || !suffix.compare("fit", Qt::CaseInsensitive))return "FITS";
if(!suffix.compare("xisf", Qt::CaseInsensitive))return "XISF";
if(filter.contains("png"))return "PNG";
if(filter.contains("fits"))return "FITS";
if(filter.contains("xisf"))return "XISF";
return "JPEG";
if(!suffix.compare("jpg", Qt::CaseInsensitive) || !suffix.compare("jpeg", Qt::CaseInsensitive))return "jpeg";
if(!suffix.compare("png", Qt::CaseInsensitive))return "png";
if(!suffix.compare("fits", Qt::CaseInsensitive) || !suffix.compare("fit", Qt::CaseInsensitive))return "fits";
if(!suffix.compare("xisf", Qt::CaseInsensitive))return "xisf";
if(filter.contains("png"))return "png";
if(filter.contains("fits"))return "fits";
if(filter.contains("xisf"))return "xisf";
return "jpeg";
};
if(!file.isEmpty())
{
QString format = filterToFormat(file, selectedFilter);
if(format == "FITS" || format == "XISF")
if(format == "fits" || format == "xisf")
{
convert(file, format);
}
+1
View File
@@ -48,6 +48,7 @@ protected slots:
void loadFile();
void loadFile(const QString &path);
void loadFile(int row);
void loadDir();
void indexDir();
void indexDir(const QString &dir);
void reindex();
+126 -9
View File
@@ -7,9 +7,12 @@ int THUMB_SIZE = 128;
int THUMB_SIZE_BORDER = 138;
int THUMB_SIZE_BORDER_Y = 158;
double SATURATION = 0.95;
bool QUALITY_RESIZE = true;
#ifdef __SSE2__
template<typename T, int ch>
void fromPlanarSSE(const void *in, void *out, size_t count);
#endif
size_t RawImage::typeSize(RawImage::DataType type)
{
@@ -117,7 +120,7 @@ RawImage::RawImage(const QImage &img)
m_stats.m_stats = false;
}
const RawImage::Stats& RawImage::imageStats()
const RawImage::Stats& RawImage::imageStats() const
{
return m_stats;
}
@@ -348,6 +351,11 @@ uint32_t RawImage::norm() const
}
}
uint32_t RawImage::widthBytes() const
{
return m_ch * m_width;
}
void* RawImage::data()
{
return m_pixels.get();
@@ -384,8 +392,17 @@ const void *RawImage::origData(uint32_t row, uint32_t col) const
return m_pixels.get() + (m_width * row * m_ch + col * m_ch) * typeSize(m_type);
}
bool RawImage::planar() const
{
return m_planar;
}
void RawImage::convertToThumbnail()
{
if(!valid())
return;
if(m_thumbAspect == 0.0f)
m_thumbAspect = (float)width() / height();
std::unique_ptr<PixelType[]> outptr = std::make_unique<PixelType[]>(THUMB_SIZE * THUMB_SIZE * 4 * sizeof(uint16_t));
uint16_t *out = reinterpret_cast<uint16_t*>(outptr.get());
@@ -398,6 +415,7 @@ void RawImage::convertToThumbnail()
{
int idx = (i*THUMB_SIZE + o)*4;
int idx2 = ((i * m_height / THUMB_SIZE * m_width) + (o * m_width / THUMB_SIZE)) * m_ch;
if(m_channels == 1)
{
out[idx] = out[idx + 1] = out[idx + 2] = in[idx2] * scale;
@@ -557,14 +575,107 @@ bool RawImage::pixel(int x, int y, double &r, double &g, double &b) const
return true;
}
void RawImage::downscaleTo(uint32_t size)
template<typename T>
void boxResample(uint32_t w, uint32_t h, uint32_t ch, uint32_t oldw, uint32_t oldh, const uint8_t *in_, uint8_t *out_)
{
/*if(size < width() || size < height())
if(oldw == 0 || oldh == 0)return;
const T *in = reinterpret_cast<const T*>(in_);
T *out = reinterpret_cast<T*>(out_);
float max = 255.0f;
if constexpr(std::is_same<T, uint16_t>::value)
max = UINT16_MAX;
float sx = (float)w / oldw;
float sy = (float)h / oldh;
for(uint32_t y = 0; y < h; y++)//iterate over destination Y
{
double s = (double)size / std::max(width(), height());
cv::Size dsize(std::floor(width() * s), std::floor(height() * s));
cv::resize(m_img, m_img, dsize, 0, 0, cv::INTER_AREA);
}*/
for(uint32_t x = 0; x < w; x++)//iterate over destination X
{
float p[4] = {0.0f};
uint32_t xx = x * oldw / w;//calculate source rect
uint32_t yy = y * oldh / h;
uint32_t xe = std::min((x + 1) * oldw / w, oldw - 1);
uint32_t ye = std::min((y + 1) * oldh / h, oldh - 1);
for(uint32_t o = yy; o <= ye; o++)//iterate over source Y
{
float cy = o * sy - y;
if(cy < 0.0f)cy = sy + cy;
else if(sy + cy > 1.0f)cy = 1.0f - cy;
else cy = sy;
if(yy==ye)cy = 1.0f;
for(uint32_t i = xx; i <= xe; i++)//iterate over source X
{
float cx = i * sx - x;
if(cx < 0.0f)cx = sx + cx;
else if(sx + cx > 1.0f)cx = 1.0f - cx;
else cx = sx;
if(xx==xe)cx = 1.0f;
for(uint32_t z = 0; z < ch; z++)
p[z] += in[(o * oldw + i) * ch + z] * cy * cx;
}
}
for(uint32_t z = 0; z < ch; z++)
if constexpr(std::is_floating_point<T>::value)
out[(y * w + x) * ch + z] = p[z];
else
out[(y * w + x) * ch + z] = std::clamp(std::round(p[z]), 0.0f, max);
}
}
}
void RawImage::resize(uint32_t w, uint32_t h)
{
if(!valid())return;
std::unique_ptr<PixelType[]> old_pixels = std::move(m_pixels);
uint32_t oldw = m_width;
uint32_t oldh = m_height;
m_thumbAspect = (float)m_width / m_height;
allocate(w, h, m_channels, m_type);
switch(m_type)
{
case RawImage::UINT8:
boxResample<uint8_t>(w, h, m_ch, oldw, oldh, old_pixels.get(), m_pixels.get());
break;
case RawImage::UINT16:
boxResample<uint16_t>(w, h, m_ch, oldw, oldh, old_pixels.get(), m_pixels.get());
break;
case RawImage::FLOAT32:
boxResample<float>(w, h, m_ch, oldw, oldh, old_pixels.get(), m_pixels.get());
break;
default:
qWarning() << "Resizing format not supported";
break;
}
}
std::pair<float, float> RawImage::unitScale() const
{
float min = *std::min_element(m_stats.m_min, m_stats.m_min + 4);
float max = *std::max_element(m_stats.m_max, m_stats.m_max + 4);
if(m_origType == UINT32)
{
min /= (float)UINT32_MAX;
max /= (float)UINT32_MAX;
}
if(min < 0.0f || max > 1.0f)
return {1.0f / (max - min), min / (max - min)};
else
return {1.0f, 0.0f};
}
void RawImage::flip()
{
std::unique_ptr<PixelType[]> tmp = std::move(m_pixels);
allocate(m_width, m_height, m_channels, m_type);
uint32_t rowSize = m_width * m_ch * typeSize(m_type);
for(uint32_t i=0; i<m_height; i++)
std::memcpy(m_pixels.get() + rowSize * (m_height - i - 1), tmp.get() + rowSize * i, rowSize);
}
std::shared_ptr<RawImage> RawImage::fromPlanar(const RawImage &img)
@@ -634,6 +745,7 @@ std::shared_ptr<RawImage> RawImage::fromPlanar(const void *pixels, uint32_t w, u
convert(static_cast<const double*>(pixels), static_cast<double*>(image->data()), 1);
break;
}
image->m_planar = false;
return image;
}
@@ -647,11 +759,11 @@ std::vector<RawImage> RawImage::split() const
size_t s = size();
auto extract = [&](auto *in, auto *out, size_t off)
{
for(size_t i=0; i < s; i+=m_ch)
for(size_t i=0; i < s; i++)
out[i] = in[i*m_ch + off];
};
for(uint32_t i=0; i<m_ch; i++)
for(uint32_t i=0; i<m_channels; i++)
{
switch(m_type)
{
@@ -673,3 +785,8 @@ std::vector<RawImage> RawImage::split() const
return planes;
}
bool RawImage::valid() const
{
return m_width > 0 && m_height > 0;
}
+10 -2
View File
@@ -12,6 +12,7 @@
extern int THUMB_SIZE;
extern int THUMB_SIZE_BORDER;
extern int THUMB_SIZE_BORDER_Y;
extern bool QUALITY_RESIZE;
class Peak
{
@@ -70,6 +71,7 @@ protected:
DataType m_origType = UINT8;
float m_thumbAspect = 0.0;
Stats m_stats;
bool m_planar = false;
void allocate(uint32_t w, uint32_t h, uint32_t ch, DataType type);
public:
RawImage();
@@ -77,7 +79,7 @@ public:
RawImage(const RawImage &d);
RawImage(RawImage &&d);
RawImage(const QImage &img);
const RawImage::Stats& imageStats();
const RawImage::Stats& imageStats() const;
void calcStats();
void rect(int &x, int &y, int w, int h, std::vector<double> &r) const;
int findPeaks(double background, double distance, std::vector<Peak> &peaks) const;
@@ -87,22 +89,28 @@ public:
uint32_t size() const;
DataType type() const;
uint32_t norm() const;
uint32_t widthBytes() const;
void* data();
const void* data() const;
void* data(uint32_t row, uint32_t col = 0);
const void* data(uint32_t row, uint32_t col = 0) const;
const void *origData() const;
const void *origData(uint32_t row, uint32_t col = 0) const;
bool planar() const;
void setPlanar();
void convertToThumbnail();
void convertToGLFormat();
float thumbAspect() const;
bool pixel(int x, int y, double &r, double &g, double &b) const;
void downscaleTo(uint32_t size);
void resize(uint32_t w, uint32_t h);
std::pair<float, float> unitScale() const;
void flip();
static std::shared_ptr<RawImage> fromPlanar(const RawImage &img);
static std::shared_ptr<RawImage> fromPlanar(const void *pixels, uint32_t w, uint32_t h, uint32_t ch, DataType type);
static size_t typeSize(DataType type);
std::vector<RawImage> split() const;
bool valid() const;
};
//Q_DECLARE_SMART_POINTER_METATYPE(std::shared_ptr);
+4
View File
@@ -1,4 +1,6 @@
#include "rawimage.h"
#ifdef __SSE2__
#include <x86intrin.h>
template<typename T, int ch>
@@ -109,3 +111,5 @@ template void fromPlanarSSE<uint32_t, 3>(const void *in, void *out, size_t count
template void fromPlanarSSE<uint32_t, 4>(const void *in, void *out, size_t count);
template void fromPlanarSSE<float, 3>(const void *in, void *out, size_t count);
template void fromPlanarSSE<float, 4>(const void *in, void *out, size_t count);
#endif
-32
View File
@@ -1,32 +0,0 @@
<RCC>
<qresource prefix="/">
<file>invert.png</file>
<file>nuke.png</file>
<file>bayer.png</file>
<file>space.nouspiro.tenmon.png</file>
<file>nuke_a.png</file>
<file>about/tenmon</file>
<file>translations/tenmon_en.qm</file>
<file>translations/tenmon_sk.qm</file>
<file>about/filter.png</file>
<file>about/stretch-panel.png</file>
<file>translations/tenmon_fr.qm</file>
<file>shaders/image.frag</file>
<file>shaders/image.vert</file>
<file>shaders/thumb.frag</file>
<file>shaders/thumb.vert</file>
<file>shaders/debayer.frag</file>
<file>shaders/debayer.vert</file>
<file>falsecolor.png</file>
<file>link.png</file>
</qresource>
<qresource prefix="/" lang="en">
<file alias="help">about/help_en</file>
</qresource>
<qresource prefix="/" lang="sk">
<file alias="help">about/help_sk</file>
</qresource>
<qresource prefix="/" lang="fr">
<file alias="help">about/help_fr</file>
</qresource>
</RCC>
Binary file not shown.

After

Width:  |  Height:  |  Size: 122 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 316 B

Before

Width:  |  Height:  |  Size: 947 B

After

Width:  |  Height:  |  Size: 947 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 316 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 316 B

View File

Before

Width:  |  Height:  |  Size: 22 KiB

After

Width:  |  Height:  |  Size: 22 KiB

View File
View File

Before

Width:  |  Height:  |  Size: 5.1 KiB

After

Width:  |  Height:  |  Size: 5.1 KiB

View File

Before

Width:  |  Height:  |  Size: 2.2 KiB

After

Width:  |  Height:  |  Size: 2.2 KiB

View File

Before

Width:  |  Height:  |  Size: 1.2 KiB

After

Width:  |  Height:  |  Size: 1.2 KiB

View File

Before

Width:  |  Height:  |  Size: 454 B

After

Width:  |  Height:  |  Size: 454 B

+29
View File
@@ -0,0 +1,29 @@
<RCC>
<qresource prefix="/">
<file>invert.png</file>
<file>nuke.png</file>
<file>bayer.png</file>
<file>space.nouspiro.tenmon.png</file>
<file>nuke_a.png</file>
<file>../about/tenmon</file>
<file>../translations/tenmon_en.qm</file>
<file>../translations/tenmon_sk.qm</file>
<file>../about/filter.png</file>
<file>../about/stretch-panel.png</file>
<file>../translations/tenmon_fr.qm</file>
<file>falsecolor.png</file>
<file>link.png</file>
<file>bggr.png</file>
<file>grbg.png</file>
<file>gbrg.png</file>
</qresource>
<qresource lang="en" prefix="/">
<file alias="help">../about/help_en</file>
</qresource>
<qresource lang="sk" prefix="/">
<file alias="help">../about/help_sk</file>
</qresource>
<qresource lang="fr" prefix="/">
<file alias="help">../about/help_fr</file>
</qresource>
</RCC>

Before

Width:  |  Height:  |  Size: 1.8 KiB

After

Width:  |  Height:  |  Size: 1.8 KiB

Before

Width:  |  Height:  |  Size: 2.1 KiB

After

Width:  |  Height:  |  Size: 2.1 KiB

Before

Width:  |  Height:  |  Size: 3.1 KiB

After

Width:  |  Height:  |  Size: 3.1 KiB

+628
View File
@@ -0,0 +1,628 @@
#include "scriptengine.h"
#include <QDir>
#include <QFileInfo>
#include <QDebug>
#include <QInputDialog>
#include "loadrunable.h"
#include "rawimage.h"
#include "loadrunable.h"
#include "batchprocessing.h"
#include <fitsio2.h>
#include "libXISF/libxisf.h"
namespace Script
{
ScriptEngine::ScriptEngine(BatchProcessing *parent)
: _jsEngine(new QJSEngine(this))
, _database(new Database(this))
, _parent(parent)
, _pool(new QThreadPool(this))
{
QJSValue core = _jsEngine->newQObject(this);
_jsEngine->globalObject().setProperty("core", core);
QJSValue fitsRecordObject = _jsEngine->newQMetaObject(&FITSRecordModify::staticMetaObject);
_jsEngine->globalObject().setProperty("FITSRecordModify", fitsRecordObject);
_database->init(QLatin1String("scriptengine"));
_semaphore.release(_pool->maxThreadCount());
}
void ScriptEngine::setParams(const QString &scriptPath, const QList<QPair<QString, QString>> &paths, const QString &outputDir)
{
_scriptPath = scriptPath;
_paths = paths;
_outputDir = outputDir + "/";
}
void ScriptEngine::reportError(const QString &message)
{
_jsEngine->throwError(message);
}
const QString &ScriptEngine::outputDir() const
{
return _outputDir;
}
void ScriptEngine::interrupt()
{
_jsEngine->setInterrupted(true);
}
void ScriptEngine::logError(const QString &message)
{
emit newMessage(message, true);
}
void ScriptEngine::log(const QString &message)
{
emit newMessage(message, false);
}
void ScriptEngine::mark(File *file)
{
_database->mark(file->absoluteFilePath());
}
void ScriptEngine::unmark(File *file)
{
_database->unmark(file->absoluteFilePath());
}
bool ScriptEngine::isMarked(const File *file) const
{
return _database->isMarked(file->absoluteFilePath());
}
void ScriptEngine::setMaxThread(int maxthread)
{
int newval = std::max(std::min(QThread::idealThreadCount(), maxthread), 1);
int oldval = _pool->maxThreadCount();
if(newval > oldval)
_semaphore.release(newval - oldval);
else if(newval < oldval)
_semaphore.acquire(oldval - newval);
_pool->setMaxThreadCount(newval);
}
void ScriptEngine::sync()
{
_pool->waitForDone();
}
QJSValue ScriptEngine::getString(const QString &label, const QString &text) const
{
QJSValue ret;
QMetaObject::invokeMethod(_parent, "getString", Qt::BlockingQueuedConnection, Q_RETURN_ARG(QJSValue, ret), Q_ARG(QString, label), Q_ARG(QString, text));
return ret;
}
QJSValue ScriptEngine::getInt(const QString &label, int value)
{
QJSValue ret;
QMetaObject::invokeMethod(_parent, "getInt", Qt::BlockingQueuedConnection, Q_RETURN_ARG(QJSValue, ret), Q_ARG(QString, label), Q_ARG(int, value));
return ret;
}
QJSValue ScriptEngine::getFloat(const QString &label, double value, int decimals) const
{
QJSValue ret;
QMetaObject::invokeMethod(_parent, "getFloat", Qt::BlockingQueuedConnection, Q_RETURN_ARG(QJSValue, ret), Q_ARG(QString, label), Q_ARG(double, value), Q_ARG(int, decimals));
return ret;
}
QJSValue ScriptEngine::getItem(const QStringList &items, const QString &label, int current) const
{
QJSValue ret;
QMetaObject::invokeMethod(_parent, "getItem", Qt::BlockingQueuedConnection, Q_RETURN_ARG(QJSValue, ret), Q_ARG(QStringList, items), Q_ARG(QString, label), Q_ARG(int, current));
return ret;
}
bool ScriptEngine::convert(File *file, QString &outpath, const QString &format, const QVariantMap &params, bool async)
{
QString path;
QDir dir(_outputDir);
QFileInfo info(outpath);
QString suffix = info.suffix();
if(info.isAbsolute())
path = info.absolutePath();
else
path = dir.absoluteFilePath(outpath);
QString f = format.toLower();
if(f != "xisf" && f != "fits" && f != "png" && f != "bmp" && f != "jpg")
{
logError("Output format must be one of xisf fits jpg png bmp");
return false;
}
info.setFile(path);
outpath = info.absolutePath() + "/" + info.completeBaseName() + "." + f;
if(async)
{
_semaphore.acquire();
_pool->start(new ConvertRunable(file->absoluteFilePath(), outpath, f, params, &_semaphore));
}
else
{
ConvertRunable crun(file->absoluteFilePath(), outpath, f, params, nullptr);
crun.run();
}
return true;
}
QJSValue ScriptEngine::newObject()
{
return _jsEngine->newObject();
}
QJSValue ScriptEngine::newArray(uint size)
{
return _jsEngine->newArray(size);
}
void ScriptEngine::run()
{
QJSValue jsPaths = _jsEngine->newArray(_paths.size());
for(qsizetype i=0; i<_paths.size(); i++)
jsPaths.setProperty(i, _jsEngine->newQObject(new File(_paths[i].first, _paths[i].second, this)));
_jsEngine->globalObject().setProperty("files", jsPaths);
QFile scriptFile(_scriptPath);
if(!scriptFile.open(QIODevice::ReadOnly))
{
emit newMessage("Failed to open " + _scriptPath, true);
emit finished();
return;
}
QTextStream stream(&scriptFile);
QString contents = stream.readAll();
scriptFile.close();
QJSValue result = _jsEngine->evaluate(contents, _scriptPath);
qDebug() << result.isError() << result.toString();
_pool->waitForDone();
if(result.isError())
{
QString error = result.property("name").toString() + " on line " + result.property("lineNumber").toString() + " : " + result.toString();
error += "\n" + result.property("stack").toString();
emit newMessage(error, true);
}
emit finished();
}
void File::loadFitsKeywords()
{
if(!_fitsKeywordsLoaded)
{
_fitsKeywordsLoaded = true;
ImageInfoData info;
if(suffix().toLower() == "xisf")
{
readXISFHeader(_path, info);
}
else if(suffix().toLower() == "fits" || suffix().toLower() == "fit")
{
readFITSHeader(_path, info);
}
else return;
for(auto &record : info.fitsHeader)
{
_fitsKeywords.append(record.key);
_fitsRecords.insert(record.key, record);
}
}
}
bool File::mkpath(const QString &path) const
{
QFileInfo info(path);
if(!info.isRelative())
{
_engine->logError("Destination path is not relative");
return false;
}
QDir dir(_engine->outputDir());
if(dir.mkpath(info.path()))
{
return true;
}
else
{
_engine->logError("Failed to create dir " + info.path());
return false;
}
}
File::File(const QString &path, Script::ScriptEngine *engine) : File(path, QString(), engine)
{
}
File::File(const QString &path, const QString &root, ScriptEngine *engine) :
_engine(engine),
_path(path),
_root(root),
_info(path)
{
}
QString File::fileName() const
{
return _info.fileName();
}
QString File::absoluteFilePath() const
{
return _info.absoluteFilePath();
}
QString File::absolutePath() const
{
return _info.absolutePath();
}
QString File::relativeFilePath() const
{
QDir dir(_root);
return dir.relativeFilePath(_info.absoluteFilePath());
}
QString File::relativePath() const
{
QDir dir(_root);
return dir.relativeFilePath(_info.absolutePath());
}
QString File::baseName() const
{
return _info.baseName();
}
QString File::completeBaseName() const
{
return _info.completeBaseName();
}
QString File::suffix() const
{
return _info.suffix();
}
qint64 File::size() const
{
return _info.size();
}
QStringList File::fitsKeywords()
{
loadFitsKeywords();
return _fitsKeywords;
}
QString File::fitsValue(const QString &key)
{
loadFitsKeywords();
if(_fitsRecords.contains(key))
return _fitsRecords[key].value.toString();
else
return QString();
}
QJSValue File::fitsValues(const QString &key)
{
loadFitsKeywords();
if(_fitsRecords.contains(key))
{
QList<FITSRecord> values = _fitsRecords.values(key);
QJSValue array = _engine->newArray(values.size());
for(qsizetype i=0; i<values.size(); i++)
array.setProperty(i, values[i].value.toString());
return array;
}
else
return QString();
}
QJSValue File::fitsRecords()
{
loadFitsKeywords();
QJSValue array = _engine->newArray(_fitsRecords.size());
uint i = 0;
for(auto &record : _fitsRecords)
{
QJSValue item = _engine->newObject();
item.setProperty("key", QString::fromUtf8(record.key));
item.setProperty("value", record.value.toString());
item.setProperty("comment", QString::fromUtf8(record.comment));
item.setProperty("xisf", record.xisf);
array.setProperty(i++, item);
}
return array;
}
bool File::modifyFITSRecords(const FITSRecordModify *modify)
{
_fitsKeywordsLoaded = false;
_fitsKeywords.clear();
if(QRegularExpression("fits?", QRegularExpression::CaseInsensitiveOption).match(suffix()).hasMatch())
{
fitsfile *file;
int status = 0;
fits_open_diskfile(&file, _path.toLocal8Bit().data(), READWRITE, &status);
int num = 0;
fits_get_num_hdus(file, &num, &status);
if(status)
{
_engine->newMessage("Failed to open FITS file", true);
return false;
}
int imgtype;
int naxis;
long naxes[3] = {0};
int type = -1;
for(int i=1; i <= num; i++)
{
fits_movabs_hdu(file, i, IMAGE_HDU, &status);
fits_get_hdu_type(file, &type, &status);
fits_get_img_param(file, 3, &imgtype, &naxis, naxes, &status);
if(type == IMAGE_HDU && naxis >= 2 && naxis <= 3 && status == 0)
break;
if(i == num)return false;
}
for(auto &remove : modify->_remove)
{
int status = 0;//we ignore errors from here
fits_delete_key(file, remove.toLatin1().data(), &status);
}
for(auto &record : modify->_update)
{
switch(record.value.typeId())
{
case QMetaType::Bool:
{
int val = record.value.toBool();
fits_update_key(file, TLOGICAL, record.key.data(), &val, record.comment.isEmpty() ? nullptr : record.comment.data(), &status);
break;
}
case QMetaType::Int:
case QMetaType::UInt:
{
long long val = record.value.toLongLong();
fits_update_key(file, TLONGLONG, record.key.data(), &val, record.comment.isEmpty() ? nullptr : record.comment.data(), &status);
break;
}
case QMetaType::QString:
{
QByteArray val = record.value.toString().toLatin1();
fits_update_key(file, TSTRING, record.key.data(), val.isEmpty() ? nullptr : val.data(), record.comment.isEmpty() ? nullptr : record.comment.data(), &status);
break;
}
case QMetaType::Float:
case QMetaType::Double:
{
double val = record.value.toDouble();
fits_update_key(file, TDOUBLE, record.key.data(), &val, record.comment.isEmpty() ? nullptr : record.comment.data(), &status);
break;
}
default:
_engine->newMessage("Unknown type for KEY " + record.key, true);
return false;
break;
}
if(status)
{
char error[100];
fits_get_errstatus(status, error);
_engine->newMessage(QString("Error when updating KEY %1 %2").arg(record.key).arg(error), true);
return false;
}
}
for(auto &record : modify->_add)
{
switch(record.value.typeId())
{
case QMetaType::Bool:
{
int val = record.value.toBool();
fits_write_key(file, TLOGICAL, record.key.data(), &val, record.comment.isEmpty() ? nullptr : record.comment.data(), &status);
break;
}
case QMetaType::Int:
case QMetaType::UInt:
{
long long val = record.value.toLongLong();
fits_write_key(file, TLONGLONG, record.key.data(), &val, record.comment.isEmpty() ? nullptr : record.comment.data(), &status);
break;
}
case QMetaType::QString:
{
QByteArray val = record.value.toString().toLatin1();
fits_write_key(file, TSTRING, record.key.data(), val.isEmpty() ? nullptr : val.data(), record.comment.isEmpty() ? nullptr : record.comment.data(), &status);
break;
}
case QMetaType::Float:
case QMetaType::Double:
{
double val = record.value.toDouble();
fits_write_key(file, TDOUBLE, record.key.data(), &val, record.comment.isEmpty() ? nullptr : record.comment.data(), &status);
break;
}
default:
_engine->newMessage("Unknown type for KEY " + record.key, true);
return false;
break;
}
if(status)
{
char error[100];
fits_get_errstatus(status, error);
_engine->newMessage(QString("Error when adding KEY {} {}").arg(record.key).arg(error), true);
return false;
}
}
fits_close_file(file, &status);
return status == 0;
}
else if(suffix() == "xisf")
{
try
{
LibXISF::XISFModify modifyXISF;
modifyXISF.open(_path.toLocal8Bit().data());
QFileInfo in(_path);
QFileInfo out(_path + "~");
for(auto &remove : modify->_remove)
modifyXISF.removeFITSKeyword(0, remove.toStdString());
for(auto &record : modify->_update)
modifyXISF.updateFITSKeyword(0, {record.key.toStdString(), record.value.toString().toStdString(), record.value.toString().toStdString()}, true);
for(auto &record : modify->_add)
modifyXISF.addFITSKeyword(0, {record.key.toStdString(), record.value.toString().toStdString(), record.value.toString().toStdString()});
modifyXISF.save(out.absoluteFilePath().toLocal8Bit().toStdString());
modifyXISF.close();
std::filesystem::rename(out.filesystemAbsoluteFilePath(), in.filesystemAbsoluteFilePath());
}
catch(LibXISF::Error &err)
{
_engine->newMessage("Failed to modify file " + _path + " " + err.what(), true);
return false;
}
}
return false;
}
bool File::isMarked() const
{
return _engine->isMarked(this);
}
File* File::copy(const QString &newpath) const
{
if(mkpath(newpath))
{
if(QFile::copy(_path, _engine->outputDir() + newpath))
return new File(_engine->outputDir() + newpath, _engine);
_engine->logError("Failed copy to " + newpath);
return nullptr;
}
return nullptr;
}
bool File::move(const QString &newpath)
{
if(mkpath(newpath))
{
if(QFile::rename(_path, _engine->outputDir() + newpath))
{
_path = _engine->outputDir() + newpath;
return true;
}
_engine->logError("Failed move to " + newpath);
return false;
}
return false;
}
File* File::convert(const QString &outpath, const QString &format, const QVariantMap &params)
{
QString path = outpath;
if(_engine->convert(this, path, format, params, false))
return new File(path, _engine);
else
return nullptr;
}
File* File::convertAsync(const QString &outpath, const QString &format, const QVariantMap &params)
{
QString path = outpath;
if(_engine->convert(this, path, format, params, true))
return new File(path, _engine);
else
return nullptr;
}
QJSValue File::stats()
{
if(_stats.isUndefined())
{
ImageInfoData info;
std::shared_ptr<RawImage> rawImage;
loadImage(_path, info, rawImage);
rawImage->calcStats();
RawImage::Stats stats = rawImage->imageStats();
_stats = _engine->newObject();
_stats.setProperty("mean", stats.m_mean[0]);
_stats.setProperty("stddev", stats.m_stdDev[0]);
_stats.setProperty("median", stats.m_median[0]);
_stats.setProperty("min", stats.m_min[0]);
_stats.setProperty("max", stats.m_max[0]);
_stats.setProperty("mad", stats.m_mean[0]);
}
return _stats;
}
ScriptEngineThread::ScriptEngineThread(BatchProcessing *parent) : QObject(parent)
{
_thread = new QThread();
_thread->setObjectName("ScriptEngine");
_engine = new ScriptEngine(parent);
_engine->moveToThread(_thread);
connect(_engine, &ScriptEngine::finished, _thread, &QThread::quit);
connect(_engine, &ScriptEngine::newMessage, this, &ScriptEngineThread::newMessage);
connect(_thread, &QThread::started, _engine, &ScriptEngine::run);
connect(_thread, &QThread::finished, _engine, &ScriptEngine::deleteLater);
connect(_engine, &ScriptEngine::destroyed, [this](){ _engine = nullptr; });
connect(_thread, &QThread::finished, _thread, &QThread::deleteLater);
connect(_thread, &QThread::finished, this, &ScriptEngineThread::finished);
}
ScriptEngineThread::~ScriptEngineThread()
{
if(_engine)_engine->interrupt();
}
void ScriptEngineThread::setParams(const QString &scriptPath, const QList<QPair<QString, QString>> &paths, const QString &outputDir)
{
_engine->setParams(scriptPath, paths, outputDir);
}
void ScriptEngineThread::start()
{
_thread->start();
}
void ScriptEngineThread::interrupt()
{
if(_engine)_engine->interrupt();
}
void FITSRecordModify::removeKeyword(const QString &key)
{
if(!_remove.contains(key))
_remove.append(key);
}
void FITSRecordModify::updateKeyword(const QString &key, const QVariant &value, const QString &comment)
{
_update.append({key.toLatin1(), value, comment.toLatin1()});
}
void FITSRecordModify::addKeyword(const QString &key, const QVariant &value, const QString &comment)
{
_update.append({key.toLatin1(), value, comment.toLatin1()});
}
}
+131
View File
@@ -0,0 +1,131 @@
#ifndef SCRIPTENGINE_H
#define SCRIPTENGINE_H
#include <QObject>
#include <QJSEngine>
#include <QFileInfo>
#include <QThread>
#include <QThreadPool>
#include <QSemaphore>
#include "database.h"
#include "imageinfo.h"
class BatchProcessing;
namespace Script
{
class File;
class ScriptEngine : public QObject
{
Q_OBJECT
QJSEngine *_jsEngine;
Database *_database;
BatchProcessing *_parent;
QThreadPool *_pool;
QSemaphore _semaphore;
QString _scriptPath;
QString _outputDir;
QList<QPair<QString, QString>> _paths;
public:
explicit ScriptEngine(BatchProcessing *parent = nullptr);
void setParams(const QString &scriptPath, const QList<QPair<QString, QString>> &paths, const QString &outputDir);
void reportError(const QString &message);
const QString& outputDir() const;
void interrupt();
void logError(const QString &message);
Q_INVOKABLE void log(const QString &message);
Q_INVOKABLE void mark(File *file);
Q_INVOKABLE void unmark(File *file);
Q_INVOKABLE bool isMarked(const File *file) const;
Q_INVOKABLE void setMaxThread(int maxthread);
Q_INVOKABLE void sync();
Q_INVOKABLE QJSValue getString(const QString &label = QString(), const QString &text = QString()) const;
Q_INVOKABLE QJSValue getInt(const QString &label = QString(), int value = 0);
Q_INVOKABLE QJSValue getFloat(const QString &label = QString(), double value = 0, int decimals = 3) const;
Q_INVOKABLE QJSValue getItem(const QStringList &items, const QString &label = "", int current = 0) const;
bool convert(File *file, QString &outpath, const QString &format, const QVariantMap &params, bool async);
QJSValue newObject();
QJSValue newArray(uint size);
public slots:
void run();
signals:
void newMessage(const QString &message, bool error);
void finished();
};
class ScriptEngineThread : public QObject
{
Q_OBJECT
QThread *_thread;
ScriptEngine *_engine;
public:
ScriptEngineThread(BatchProcessing *parent = nullptr);
~ScriptEngineThread();
void setParams(const QString &scriptPath, const QList<QPair<QString, QString>> &paths, const QString &outputDir);
void start();
void interrupt();
signals:
void newMessage(const QString &message, bool error);
void finished();
};
class FITSRecordModify;
class File : public QObject
{
Q_OBJECT
ScriptEngine *_engine;
QString _path;
QString _root;
QFileInfo _info;
bool _fitsKeywordsLoaded = false;
QStringList _fitsKeywords;
QMultiHash<QString, FITSRecord> _fitsRecords;
void loadFitsKeywords();
bool mkpath(const QString &path) const;
QJSValue _stats;
public:
explicit File(const QString &path, ScriptEngine *engine);
explicit File(const QString &path, const QString &root, ScriptEngine *engine);
Q_INVOKABLE QString fileName() const;
Q_INVOKABLE QString absoluteFilePath() const;
Q_INVOKABLE QString absolutePath() const;
Q_INVOKABLE QString relativeFilePath() const;
Q_INVOKABLE QString relativePath() const;
Q_INVOKABLE QString baseName() const;
Q_INVOKABLE QString completeBaseName() const;
Q_INVOKABLE QString suffix() const;
Q_INVOKABLE qint64 size() const;
Q_INVOKABLE QStringList fitsKeywords();
Q_INVOKABLE QString fitsValue(const QString &key);
Q_INVOKABLE QJSValue fitsValues(const QString &key);
Q_INVOKABLE QJSValue fitsRecords();
Q_INVOKABLE bool modifyFITSRecords(const FITSRecordModify *modify);
Q_INVOKABLE bool isMarked() const;
Q_INVOKABLE File* copy(const QString &newpath) const;
Q_INVOKABLE bool move(const QString &newpath);
Q_INVOKABLE File* convert(const QString &outpath, const QString &format, const QVariantMap &params = QVariantMap());
Q_INVOKABLE File* convertAsync(const QString &outpath, const QString &format, const QVariantMap &params = QVariantMap());
Q_INVOKABLE QJSValue stats();
};
class FITSRecordModify : public QObject
{
Q_OBJECT
QStringList _remove;
QVector<FITSRecord> _update;
QVector<FITSRecord> _add;
friend class File;
public:
Q_INVOKABLE FITSRecordModify(){};
Q_INVOKABLE void removeKeyword(const QString &key);
Q_INVOKABLE void updateKeyword(const QString &key, const QVariant &value, const QString &comment = QString());
Q_INVOKABLE void addKeyword(const QString &key, const QVariant &value, const QString &comment = QString());
};
}
#endif // SCRIPTENGINE_H
+18
View File
@@ -0,0 +1,18 @@
core.log("This script convert any FITS file into XISF with ZSTD compression");
if(files.length == 0)
{
core.log("No input files");
throw "";
}
let compression = {"compressionType": "zstd+sh"};
for(file of files)
{
if(file.suffix() == "fits" || file.suffix() == "fit")
{
core.log("Converting " + file.fileName());
file.convertAsync(file.relativeFilePath(), "XISF", compression);
}
}
+36
View File
@@ -0,0 +1,36 @@
// how to get input from user
let d = core.getFloat("Getting float value");
let i = core.getInt("Getting integer value");
let s = core.getString("Getting string value");
// print user input
core.log("Your input " + d + " " + i + " " + s);
for(file of files)
{
if(file.suffix() == "fits" || file.suffix() == "fit" || file.suffix() == "xisf")
{
let keywords = file.fitsKeywords();
let item = core.getItem(keywords, "Select keyword");
core.log("You selected keyword " + item); core.log(file.fitsKeywords());
core.log("fileName() " + file.fileName());
core.log("absoluteFilePath() " + file.absoluteFilePath());
core.log("absolutePath() " + file.absolutePath());
core.log("relativeFilePath() " + file.relativeFilePath());
core.log("relativePath() " + file.relativePath());
core.log("baseName() " + file.baseName());
core.log("completeBase() " + file.completeBaseName());
core.log("suffix() " + file.suffix());
core.log("size() " + file.size());
let stats = file.stats();
core.log("Image statistics");
core.log("\tMinimum: " + stats.min);
core.log("\tMaximum: " + stats.max);
core.log("\tMedian:" + stats.median);
core.log("\tStandard deviation:" + stats.stddev);
core.log("\tMedian Absolute Deviation:" + stats.mad);
break;
}
}
+8
View File
@@ -0,0 +1,8 @@
for(file of files)
{
if(file.suffix() == "fits" || file.suffix() == "fit" || file.suffix() == "xisf")
{
let stats = file.stats();
core.log("File: \"" + file.fileName() + "\" - median: " + stats.median);
}
}
+50
View File
@@ -0,0 +1,50 @@
function checkFITS(key)
{
const noEditableKey = ["SIMPLE", "BITPIX", "NAXIS", "NAXIS1", "NAXIS2", "NAXIS3", "EXTEND", "BZERO", "BSCALE"];
return noEditableKey.indexOf(key) < 0;
}
if(files.length == 0)
{
core.log("No input files");
throw "";
}
let action = core.getItem(["UPDATE", "ADD", "REMOVE"], "Do you want update, add or remove record?");
let modify = new FITSRecordModify();
if(action == "UPDATE")
{
let keywords = files[0].fitsKeywords().filter(checkFITS);
let keyword = core.getItem(keywords, "Select keyword to update");
let value = files[0].fitsValue(keyword);
if(isNaN(value))
value = core.getString("Enter new value", value);
else
value = core.getFloat("Enter new value", value);
modify.updateKeyword(keyword, value);
}
else if(action == "ADD")
{
let keyword = core.getString("Enter keyword to add");
let value = core.getString("Enter new value");
keyword = keyword.toUpperCase();
modify.addKeyword(keyword, value);
}
else if(action == "REMOVE")
{
let keywords = files[0].fitsKeywords().filter(checkFITS);
let keyword = core.getItem(keywords, "Select keyword to remove");
modify.removeKeyword(keyword);
}
for(file of files)
{
if(file.suffix() == "fits" || file.suffix() == "fit" || file.suffix() == "xisf")
{
core.log("Modifing " + file.fileName());
file.modifyFITSRecords(modify);
}
}
+8
View File
@@ -0,0 +1,8 @@
<RCC>
<qresource prefix="/scripts">
<file>example script</file>
<file>convert to XISF</file>
<file>median</file>
<file>modify FITS header</file>
</qresource>
</RCC>
+27 -1
View File
@@ -8,6 +8,7 @@
extern int DEFAULT_WIDTH;
extern double SATURATION;
extern int FILTERING;
class EvenNumber : public QSpinBox
{
@@ -39,7 +40,7 @@ SettingsDialog::SettingsDialog(QWidget *parent) : QDialog(parent)
QSettings settings;
m_preloadImages = new QSpinBox(this);
m_preloadImages->setRange(0, 8);
m_preloadImages->setRange(0, 32);
m_preloadImages->setValue(settings.value("settings/preloadimagecount", DEFAULT_WIDTH).toInt());
m_preloadImages->setToolTip(tr("How many images are preloaded before and after current image."));
@@ -56,12 +57,30 @@ SettingsDialog::SettingsDialog(QWidget *parent) : QDialog(parent)
m_saturation->setValue(settings.value("settings/saturation", SATURATION * 100.0).toDouble());
m_saturation->setToolTip(tr("Set threshold value that is considered saturated when showing statistics.\nFor RAW files you may set 22%"));
m_slideShowTime = new QDoubleSpinBox(this);
m_slideShowTime->setMinimum(0.01);
m_slideShowTime->setMaximum(10);
m_slideShowTime->setSuffix(" s");
m_slideShowTime->setValue(settings.value("settings/slideshowtime", 1.0).toDouble());
m_slideShowTime->setSingleStep(0.1);
m_useNativeDialog = new QCheckBox(tr("Don't use native file dialog"), this);
m_useNativeDialog->setChecked(QApplication::testAttribute(Qt::AA_DontUseNativeDialogs));
m_filtering = new QComboBox(this);
m_filtering->addItems({tr("Nearest"), tr("Linear"), tr("Cubic")});
m_filtering->setCurrentIndex(FILTERING);
m_qualityThumbnail = new QCheckBox(tr("Smooth thumbnails"), this);
m_qualityThumbnail->setChecked(QUALITY_RESIZE);
m_qualityThumbnail->setToolTip(tr("Use box filter when downsampling thumbnails instead of nearest. Slightly slower."));
layout->addRow(tr("Image preload count"), m_preloadImages);
layout->addRow(tr("Thumbnails size"), m_thumSize);
layout->addRow(tr("Saturation"), m_saturation);
layout->addRow(tr("Slideshow interval"), m_slideShowTime);
layout->addRow(tr("Image interpolation"), m_filtering);
layout->addRow(m_qualityThumbnail);
layout->addRow(m_useNativeDialog);
//layout->addRow(new QLabel(tr("Changes in settings will take effect after program restart.")));
@@ -82,6 +101,8 @@ void SettingsDialog::loadSettings()
THUMB_SIZE_BORDER_Y = THUMB_SIZE + 30;
DEFAULT_WIDTH = settings.value("settings/preloadimagecount", DEFAULT_WIDTH).toInt();
SATURATION = settings.value("settings/saturation", 95.0).toDouble() / 100.0;
FILTERING = settings.value("settings/filtering", FILTERING).toInt();
QUALITY_RESIZE = settings.value("settings/qualitythumbnail", QUALITY_RESIZE).toBool();
QApplication::setAttribute(Qt::AA_DontUseNativeDialogs, settings.value("settings/dontusenativedialogs", false).toBool());
}
@@ -102,6 +123,11 @@ void SettingsDialog::saveSettings()
settings.setValue("settings/preloadimagecount", m_preloadImages->value());
settings.setValue("settings/dontusenativedialogs", m_useNativeDialog->isChecked());
settings.setValue("settings/saturation", m_saturation->value());
settings.setValue("settings/slideshowtime", m_slideShowTime->value());
settings.setValue("settings/qualitythumbnail", m_qualityThumbnail->isChecked());
QUALITY_RESIZE = m_qualityThumbnail->isChecked();
FILTERING = m_filtering->currentIndex();
settings.setValue("settings/filtering", FILTERING);
SATURATION = m_saturation->value() / 100.0;
QApplication::setAttribute(Qt::AA_DontUseNativeDialogs, m_useNativeDialog->isChecked());
if(DEFAULT_WIDTH != m_preloadImages->value())
+4
View File
@@ -4,6 +4,7 @@
#include <QDialog>
#include <QSpinBox>
#include <QCheckBox>
#include <QComboBox>
class SettingsDialog : public QDialog
{
@@ -19,8 +20,11 @@ private:
QSpinBox *m_preloadImages;
QSpinBox *m_thumSize;
QDoubleSpinBox *m_slideShowTime;
QDoubleSpinBox *m_saturation;
QCheckBox *m_useNativeDialog;
QCheckBox *m_qualityThumbnail;
QComboBox *m_filtering;
};
#endif // SETTINGSDIALOG_H
+2 -1
View File
@@ -1,6 +1,7 @@
#version 330
uniform sampler2D qt_Texture0;
uniform ivec2 firstRed;
in vec2 qt_TexCoord0;
in vec2 center;
layout(location = 0) out vec4 color;
@@ -11,7 +12,7 @@ void main(void)
{
ivec2 texSize = textureSize(qt_Texture0, 0);
ivec2 icenter = ivec2(center);
ivec2 alternate = icenter % 2;
ivec2 alternate = (icenter + firstRed) % 2;
// cross, checker, theta, phi
const vec4 kA = vec4(-1.0, -1.5, 0.5, -1.0) / 8.0;
+98 -1
View File
@@ -7,6 +7,7 @@ uniform bool bw;
uniform bool invert;
uniform bool srgb;
uniform bool false_color;
uniform int filtering;
in vec2 qt_TexCoord0;
layout(location = 0) out vec4 color;
@@ -35,7 +36,8 @@ vec3 falsecolor(float color)
vec3(1.0, 0.0, 0.0), vec3(1.0, 0.0, 0.0));//red
color *= 5.0;
int i = int(color);
return mix(pallete[i], pallete[i+1], fract(color));
float f = fract(color);
return mix(pallete[i], pallete[i+1], f);// * (f * 0.5 + 0.5);
}
vec3 checker()
@@ -44,9 +46,104 @@ vec3 checker()
return vec3(step(pattern.x * pattern.y, 0.0) * 0.25 + 0.25);
}
vec4 cubic(float v)
{
vec4 n = vec4(1.0, 2.0, 3.0, 4.0) - v;
vec4 s = n * n * n;
float x = s.x;
float y = s.y - 4.0 * s.x;
float z = s.z - 4.0 * s.y + 6.0 * s.x;
float w = 6.0 - x - y - z;
return vec4(x, y, z, w) * (1.0/6.0);
}
vec4 textureBicubic(sampler2D sampler, vec2 texCoords)
{
vec2 texSize = textureSize(sampler, 0);
vec2 invTexSize = 1.0 / texSize;
texCoords = texCoords * texSize - 0.5;
vec2 fxy = fract(texCoords);
texCoords -= fxy;
vec4 xcubic = cubic(fxy.x);
vec4 ycubic = cubic(fxy.y);
vec4 c = texCoords.xxyy + vec2 (-0.5, +1.5).xyxy;
vec4 s = vec4(xcubic.xz + xcubic.yw, ycubic.xz + ycubic.yw);
vec4 offset = c + vec4 (xcubic.yw, ycubic.yw) / s;
offset *= invTexSize.xxyy;
vec4 sample0 = texture(sampler, offset.xz);
vec4 sample1 = texture(sampler, offset.yz);
vec4 sample2 = texture(sampler, offset.xw);
vec4 sample3 = texture(sampler, offset.yw);
float sx = s.x / (s.x + s.y);
float sy = s.z / (s.z + s.w);
return mix(mix(sample3, sample2, sx), mix(sample1, sample0, sx), sy);
}
vec4 textureCatmul(sampler2D sampler, vec2 texCoords)
{
ivec2 texSize = textureSize(sampler, 0);
texCoords = texCoords * vec2(texSize) - 0.5;
ivec2 texel = ivec2(floor(texCoords));
vec2 fra = fract(texCoords);
texSize -= 1;
const mat4 CatMul = mat4(0, 1, 0, 0, -0.5, 0, 0.5, 0, 1, -2.5, 2, -0.5, -0.5, 1.5, -1.5, 0.5);
vec4 xx = CatMul * vec4(1.0, fra.x, fra.x*fra.x, fra.x*fra.x*fra.x);
vec4 yy = CatMul * vec4(1.0, fra.y, fra.y*fra.y, fra.y*fra.y*fra.y);
vec4 a00 = texelFetch(sampler, clamp(texel + ivec2(-1, -1), ivec2(0, 0), texSize), 0) * xx.x;
vec4 a01 = texelFetch(sampler, clamp(texel + ivec2( 0, -1), ivec2(0, 0), texSize), 0) * xx.y;
vec4 a02 = texelFetch(sampler, clamp(texel + ivec2( 1, -1), ivec2(0, 0), texSize), 0) * xx.z;
vec4 a03 = texelFetch(sampler, clamp(texel + ivec2( 2, -1), ivec2(0, 0), texSize), 0) * xx.w;
vec4 a10 = texelFetch(sampler, clamp(texel + ivec2(-1, 0), ivec2(0, 0), texSize), 0) * xx.x;
vec4 a11 = texelFetch(sampler, clamp(texel + ivec2( 0, 0), ivec2(0, 0), texSize), 0) * xx.y;
vec4 a12 = texelFetch(sampler, clamp(texel + ivec2( 1, 0), ivec2(0, 0), texSize), 0) * xx.z;
vec4 a13 = texelFetch(sampler, clamp(texel + ivec2( 2, 0), ivec2(0, 0), texSize), 0) * xx.w;
vec4 a20 = texelFetch(sampler, clamp(texel + ivec2(-1, 1), ivec2(0, 0), texSize), 0) * xx.x;
vec4 a21 = texelFetch(sampler, clamp(texel + ivec2( 0, 1), ivec2(0, 0), texSize), 0) * xx.y;
vec4 a22 = texelFetch(sampler, clamp(texel + ivec2( 1, 1), ivec2(0, 0), texSize), 0) * xx.z;
vec4 a23 = texelFetch(sampler, clamp(texel + ivec2( 2, 1), ivec2(0, 0), texSize), 0) * xx.w;
vec4 a30 = texelFetch(sampler, clamp(texel + ivec2(-1, 2), ivec2(0, 0), texSize), 0) * xx.x;
vec4 a31 = texelFetch(sampler, clamp(texel + ivec2( 0, 2), ivec2(0, 0), texSize), 0) * xx.y;
vec4 a32 = texelFetch(sampler, clamp(texel + ivec2( 1, 2), ivec2(0, 0), texSize), 0) * xx.z;
vec4 a33 = texelFetch(sampler, clamp(texel + ivec2( 2, 2), ivec2(0, 0), texSize), 0) * xx.w;
vec4 c = vec4(0.0);
c += (a00 + a01 + a02 + a03) * yy.x;
c += (a10 + a11 + a12 + a13) * yy.y;
c += (a20 + a21 + a22 + a23) * yy.z;
c += (a30 + a31 + a32 + a33) * yy.w;
return c;
}
void main(void)
{
switch(filtering)
{
case 0://nearest
color = texelFetch(qt_Texture0, ivec2(qt_TexCoord0 * textureSize(qt_Texture0, 0)), 0);
break;
default:
case 1://bilinear
color = texture(qt_Texture0, qt_TexCoord0);
break;
case 2://catmul bicubic
color = textureCatmul(qt_Texture0, qt_TexCoord0);
break;
}
color.rgb = color.rgb * unit_scale.x + unit_scale.y;
if(bw)color = color.rrra;
color = MTF(color, vec4(mtf_param[0], 0.0), vec4(mtf_param[1], 0.5), vec4(mtf_param[2], 1.0));
+10
View File
@@ -0,0 +1,10 @@
<RCC>
<qresource prefix="/">
<file>debayer.frag</file>
<file>debayer.vert</file>
<file>image.frag</file>
<file>image.vert</file>
<file>thumb.frag</file>
<file>thumb.vert</file>
</qresource>
</RCC>
+41 -1
View File
@@ -4,6 +4,10 @@
<launchable type="desktop-id">space.nouspiro.tenmon.desktop</launchable>
<name>Tenmon</name>
<summary>FITS/XISF image viewer, converter, index and search</summary>
<developer id="nouspiro.space">
<name>Dušan Poizl</name>
</developer>
<developer_name>Dušan Poizl</developer_name>
<metadata_license>CC0-1.0</metadata_license>
<project_license>GPL-3.0</project_license>
<description>
@@ -11,7 +15,7 @@
<ul>
<li>FITS 8, 16, 32 bit integer and 32, 64 bit float</li>
<li>XISF 8, 16, 32 bit integer and 32, 64 bit float</li>
<li>RAW CR2, DNG, NEF</li>
<li>RAW CR2/CR3, DNG, NEF</li>
<li>JPEG, PNG, BMP, GIF, PBM, PGM, PPM and SVG images</li>
</ul>
<p>Features of application:</p>
@@ -43,12 +47,48 @@
<url type="bugtracker">https://github.com/flathub/space.nouspiro.tenmon/issues</url>
<screenshots>
<screenshot type="default">
<caption>Main window with image</caption>
<image>https://nouspiro.space/wp-content/uploads/2022/04/tenmon-1024x579.png</image>
</screenshot>
<screenshot type="default">
<caption>Thumnail view</caption>
<image>https://nouspiro.space/wp-content/uploads/2022/12/tenmon2-1024x645.png</image>
</screenshot>
</screenshots>
<content_rating type="oars-1.1"/>
<releases>
<release version="20240816" date="2024-08-16">
<description>
Fix saving image
</description>
</release>
<release version="20240616" date="2024-06-16">
<description>
<ul>
<li>Batch processing with JavaScript</li>
<li>Opening directory recursively</li>
</ul>
</description>
</release>
<release version="20240201" date="2024-02-01">
<description>
<ul>
<li>Smooth thumbnails</li>
<li>Respect ROWORDER</li>
<li>Bugfixes</li>
</ul>
</description>
</release>
<release version="20240108" date="2024-01-08">
<description>
<ul>
<li>Update to Qt6</li>
<li>Add support for CR3 RAW files</li>
<li>Slideshow</li>
<li>Improved rapid image view</li>
</ul>
</description>
</release>
<release version="20231116" date="2023-11-16">
<description>
<ul>
+18 -16
View File
@@ -105,43 +105,44 @@ void STFSlider::paintEvent(QPaintEvent *event)
void STFSlider::mouseMoveEvent(QMouseEvent *event)
{
if(std::abs(m_blackPoint*width() - event->x()) < 5 ||
std::abs((m_blackPoint + (m_whitePoint - m_blackPoint) * m_midPoint)*width() - event->x()) < 5 ||
std::abs(m_whitePoint*width() - event->x()) < 5)
const qreal x = event->position().x();
if(std::abs(m_blackPoint*width() - x) < 5 ||
std::abs((m_blackPoint + (m_whitePoint - m_blackPoint) * m_midPoint)*width() - x) < 5 ||
std::abs(m_whitePoint*width() - x) < 5)
setCursor(Qt::SplitHCursor);
else
unsetCursor();
qreal x = (qreal)event->x()/width();
qreal xw = x/width();
if(event->modifiers() & Qt::ShiftModifier && !m_fineTune)
{
m_fineTune = true;
m_fineTuneX = x;
m_fineTuneX = xw;
}
if(!(event->modifiers() & Qt::ShiftModifier) && m_fineTune)
m_fineTune = false;
if(m_fineTune)
{
x = m_fineTuneX + (x - m_fineTuneX) * 0.2;
xw = m_fineTuneX + (xw - m_fineTuneX) * 0.2;
}
switch(m_grabbed)
{
case 0:
m_blackPoint = clamp(x);
m_blackPoint = clamp(xw);
m_whitePoint = std::max(m_whitePoint, m_blackPoint);
QToolTip::showText(event->globalPos(), QString::number(m_blackPoint), this);
QToolTip::showText(event->globalPosition().toPoint(), QString::number(m_blackPoint), this);
break;
case 1:
m_midPoint = (x - m_blackPoint) / (m_whitePoint - m_blackPoint);
m_midPoint = (xw - m_blackPoint) / (m_whitePoint - m_blackPoint);
m_midPoint = clamp(m_midPoint);
QToolTip::showText(event->globalPos(), QString::number(m_midPoint), this);
QToolTip::showText(event->globalPosition().toPoint(), QString::number(m_midPoint), this);
break;
case 2:
m_whitePoint = clamp(x);
m_whitePoint = clamp(xw);
m_blackPoint = std::min(m_blackPoint, m_whitePoint);
QToolTip::showText(event->globalPos(), QString::number(m_whitePoint), this);
QToolTip::showText(event->globalPosition().toPoint(), QString::number(m_whitePoint), this);
break;
}
if(m_grabbed >= 0)
@@ -153,17 +154,18 @@ void STFSlider::mouseMoveEvent(QMouseEvent *event)
void STFSlider::mousePressEvent(QMouseEvent *event)
{
const qreal x = event->position().x();
if(event->modifiers() & Qt::ShiftModifier)
{
m_fineTune = true;
m_fineTuneX = (qreal)event->x()/width();
m_fineTuneX = x/width();
}
if(std::abs((m_blackPoint + (m_whitePoint - m_blackPoint) * m_midPoint)*width() - event->x()) < 5)
if(std::abs((m_blackPoint + (m_whitePoint - m_blackPoint) * m_midPoint)*width() - x) < 5)
m_grabbed = 1;
else if(std::abs(m_blackPoint*width() - event->x()) < 5)
else if(std::abs(m_blackPoint*width() - x) < 5)
m_grabbed = 0;
else if(std::abs(m_whitePoint*width() - event->x()) < 5)
else if(std::abs(m_whitePoint*width() - x) < 5)
m_grabbed = 2;
else
m_grabbed = -1;
+7 -4
View File
@@ -120,12 +120,15 @@ void StretchToolbar::stretchImage(Image *img)
mad = stats.m_mad[i];
max = stats.m_max[i];
median /= img->rawImage()->norm();
bool a = median > 0.5 ? true : false;
mad /= img->rawImage()->norm();
max /= img->rawImage()->norm();
float bp = median + mad * BLACK_POINT_SIGMA * MAD_TO_SIGMA;
float mid = MTF(median - bp, TARGET_BACKGROUND);
max = 1.0f;// /= img->rawImage()->norm();
float bp = a || mad == 0.0f ? 0.0f : std::clamp(median + mad * BLACK_POINT_SIGMA * MAD_TO_SIGMA, 0.0, 1.0);
if(a && mad != 0.0f)
max = std::clamp(median - mad * BLACK_POINT_SIGMA * MAD_TO_SIGMA, 0.0, 1.0);
float mid = !a ? MTF(median - bp, TARGET_BACKGROUND) : MTF(TARGET_BACKGROUND, max - median);
m_mtfParam.blackPoint[i-o] = bp;
m_mtfParam.midPoint[i-o] = mid / max;
m_mtfParam.midPoint[i-o] = mid;// / max;
m_mtfParam.whitePoint[i-o] = max;
bp2 += bp;
mid2 += mid;
Binary file not shown.
+412 -1
View File
@@ -1,24 +1,191 @@
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE TS>
<TS version="2.1" language="en_US" sourcelanguage="en">
<TS version="2.1" language="en_US" sourcelanguage="en_US">
<context>
<name>About</name>
<message>
<location filename="../about.cpp" line="12"/>
<source>About Tenmon</source>
<translation>About Tenmon</translation>
</message>
</context>
<context>
<name>BatchProcessing</name>
<message>
<location filename="../batchprocessing.ui" line="14"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="208"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="208"/>
<source>Batch Processing</source>
<translation>Batch Processing</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="22"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="209"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="209"/>
<source>Input files and directories</source>
<translation>Input files and directories</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="44"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="210"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="210"/>
<source>Add files</source>
<translation>Add files</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="51"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="211"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="211"/>
<source>Add directories</source>
<translation>Add directory</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="58"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="212"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="212"/>
<source>Remove</source>
<translation>Remove</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="65"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="213"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="213"/>
<source>Remove all</source>
<translation>Remove all</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="78"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="214"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="214"/>
<source>Output directory</source>
<translation>Output directory</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="92"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="215"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="215"/>
<source>Browse</source>
<translation>Browse</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="103"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="216"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="216"/>
<source>Scripts</source>
<translation>Scripts</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="123"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="217"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="217"/>
<source>Open scripts</source>
<translation>Open scripts</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="142"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="218"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="218"/>
<source>Log</source>
<translation>Log</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="182"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="227"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="219"/>
<source>Start script</source>
<translation>Start script</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="192"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="228"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="220"/>
<source>Stop script</source>
<translation>Stop script</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="199"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="229"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="221"/>
<source>Close</source>
<translation>Close</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="118"/>
<source>Interrupt running script?</source>
<translation>Interrupt running script?</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="138"/>
<source>Select files</source>
<translation>Select files</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="149"/>
<source>Select directory</source>
<translation>Select directory</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="170"/>
<source>Select output directory</source>
<translation>Select output directory</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="212"/>
<source>Invalid output directory</source>
<translation>Invalid output directory</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="212"/>
<source>Output directory path doesn&apos;t exist or is not writable</source>
<translation>Output directory path doesn&apos;t exist or is not writable</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="243"/>
<source>Enter text</source>
<translation>Enter text</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="250"/>
<source>Enter integer number</source>
<translation>Enter integer number</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="257"/>
<source>Enter float number</source>
<translation>Enter real number</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="264"/>
<source>Select item</source>
<translation>Select item</translation>
</message>
<message>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="219"/>
<source>&lt;!DOCTYPE HTML PUBLIC &quot;-//W3C//DTD HTML 4.0//EN&quot; &quot;http://www.w3.org/TR/REC-html40/strict.dtd&quot;&gt;
&lt;html&gt;&lt;head&gt;&lt;meta name=&quot;qrichtext&quot; content=&quot;1&quot; /&gt;&lt;meta charset=&quot;utf-8&quot; /&gt;&lt;style type=&quot;text/css&quot;&gt;
p, li { white-space: pre-wrap; }
hr { height: 1px; border-width: 0; }
li.unchecked::marker { content: &quot;\2610&quot;; }
li.checked::marker { content: &quot;\2612&quot;; }
&lt;/style&gt;&lt;/head&gt;&lt;body style=&quot; font-family:&apos;FreeMono&apos;; font-size:11pt; font-weight:400; font-style:normal;&quot;&gt;
&lt;p style=&quot;-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;&quot;&gt;&lt;br /&gt;&lt;/p&gt;&lt;/body&gt;&lt;/html&gt;</source>
<translation type="unfinished"></translation>
</message>
</context>
<context>
<name>DataBaseView</name>
<message>
<location filename="../databaseview.cpp" line="255"/>
<source>Select columns</source>
<translation>Select columns</translation>
</message>
<message>
<location filename="../databaseview.cpp" line="293"/>
<source>Text to search, you can % as wildcard</source>
<translation>Text to search, you can % as wildcard</translation>
</message>
<message>
<location filename="../databaseview.cpp" line="309"/>
<source>Filter</source>
<translation>Filter</translation>
</message>
@@ -26,10 +193,12 @@
<context>
<name>DatabaseTableView</name>
<message>
<location filename="../databaseview.cpp" line="215"/>
<source>Mark</source>
<translation>Mark</translation>
</message>
<message>
<location filename="../databaseview.cpp" line="216"/>
<source>Unmark</source>
<translation>Unmark</translation>
</message>
@@ -37,6 +206,7 @@
<context>
<name>FITSFileModel</name>
<message>
<location filename="../databaseview.cpp" line="194"/>
<source>File name</source>
<translation>File name</translation>
</message>
@@ -44,22 +214,27 @@
<context>
<name>FilesystemWidget</name>
<message>
<location filename="../filesystemwidget.cpp" line="26"/>
<source>Sort by filename</source>
<translation>Sort by file name</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="27"/>
<source>Sort by time</source>
<translation>Sort by time</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="28"/>
<source>Sort by size</source>
<translation>Sort by size</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="29"/>
<source>Sort by type</source>
<translation>Sort by type</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="30"/>
<source>Reverse</source>
<translation>Reverse order</translation>
</message>
@@ -67,34 +242,43 @@
<context>
<name>Filetree</name>
<message>
<location filename="../filesystemwidget.cpp" line="90"/>
<location filename="../filesystemwidget.cpp" line="94"/>
<source>Open</source>
<translation>Open</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="96"/>
<source>Copy marked files</source>
<translation>Copy marked files</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="97"/>
<source>Move marked files</source>
<translation>Move marked files</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="98"/>
<source>Index directory</source>
<translation>Index directory</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="95"/>
<source>Set as root</source>
<translation>Set as root directory</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="102"/>
<source>Reset root</source>
<translation>Reset root directory</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="103"/>
<source>Go up</source>
<translation>Go up</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="104"/>
<source>Show hidden files</source>
<translation>Show hidden files</translation>
</message>
@@ -102,6 +286,7 @@
<context>
<name>HelpDialog</name>
<message>
<location filename="../about.cpp" line="33"/>
<source>Help</source>
<translation>Help</translation>
</message>
@@ -109,6 +294,7 @@
<context>
<name>Histogram</name>
<message>
<location filename="../histogram.cpp" line="72"/>
<source>Logarithmic scale</source>
<translation>Logarithmic scale</translation>
</message>
@@ -116,22 +302,27 @@
<context>
<name>ImageInfo</name>
<message>
<location filename="../imageinfo.cpp" line="65"/>
<source>Property</source>
<translation>Property</translation>
</message>
<message>
<location filename="../imageinfo.cpp" line="65"/>
<source>Value</source>
<translation>Value</translation>
</message>
<message>
<location filename="../imageinfo.cpp" line="65"/>
<source>Comment</source>
<translation>Comment</translation>
</message>
<message>
<location filename="../imageinfo.cpp" line="82"/>
<source>FITS Header</source>
<translation>FITS Header</translation>
</message>
<message>
<location filename="../imageinfo.cpp" line="91"/>
<source>Image info</source>
<translation>Image info</translation>
</message>
@@ -139,6 +330,7 @@
<context>
<name>ImageRingList</name>
<message>
<location filename="../imageringlist.cpp" line="375"/>
<source>Name</source>
<translation>Name</translation>
</message>
@@ -146,284 +338,410 @@
<context>
<name>ImageWidget</name>
<message>
<location filename="../imagescrollareagl.cpp" line="78"/>
<location filename="../imagescrollareagl.cpp" line="447"/>
<source>OpenGL error</source>
<translation>OpenGL error</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="78"/>
<location filename="../imagescrollareagl.cpp" line="447"/>
<source>Could not initialize OpenGL 3.3 context. Ensure that proper GPU driver is installed.</source>
<translation>Could not initialize OpenGL 3.3 context. Ensure that proper GPU driver is installed.</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="674"/>
<source>L:%1</source>
<translation>L:%1</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="674"/>
<location filename="../imagescrollareagl.cpp" line="676"/>
<source>X:%3 Y:%4</source>
<translation>X:%3 Y:%4</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="676"/>
<source>R:%1 G:%2 B:%3</source>
<translation>R:%1 G:%2 B:%3</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="99"/>
<source>Failed to load image</source>
<translation>Failed to load image</translation>
</message>
</context>
<context>
<name>MainWindow</name>
<message>
<location filename="../mainwindow.cpp" line="68"/>
<source>Image info</source>
<translation>Image info</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="77"/>
<source>Can&apos;t open DB</source>
<translation>Can&apos;t open DB</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="77"/>
<source>Can&apos;t open SQLITE database</source>
<translation>Can&apos;t open SQLITE database</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="110"/>
<source>Filesystem</source>
<translation>File system</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="134"/>
<source>Tenmon</source>
<translation>Tenmon</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="145"/>
<source>File</source>
<translation>File</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="146"/>
<source>Open</source>
<translation>Open</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="150"/>
<source>Copy marked files</source>
<translation>Copy marked files</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="148"/>
<location filename="../mainwindow.cpp" line="532"/>
<location filename="../mainwindow.cpp" line="694"/>
<source>Save as</source>
<translation>Save as</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="147"/>
<location filename="../mainwindow.cpp" line="494"/>
<source>Open directory recursively</source>
<translation>Open directory recursively</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="157"/>
<source>Batch processing</source>
<translation>Batch processing</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="163"/>
<source>Live mode</source>
<translation>Live mode</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="165"/>
<source>Exit</source>
<translation>Exit</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="173"/>
<source>View</source>
<translation>View</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="174"/>
<source>Zoom In</source>
<translation>Zoom In</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="175"/>
<source>Zoom Out</source>
<translation>Zoom Out</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="176"/>
<source>Best Fit</source>
<translation>Best Fit</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="177"/>
<source>100%</source>
<translation>100%</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="210"/>
<source>Select</source>
<translation>Select</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="211"/>
<source>Mark</source>
<translation>Mark</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="212"/>
<source>Unmark</source>
<translation>Unmark</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="214"/>
<source>Mark and next</source>
<translation>Mark and next</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="215"/>
<source>Unmark and next</source>
<translation>Unmark and next</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="219"/>
<source>Analyze</source>
<translation>Analyze</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="232"/>
<source>Image statistics</source>
<translation>Image statistics</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="233"/>
<source>Peak finder</source>
<translation>Peak finder</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="244"/>
<source>Docks</source>
<translation>Docks</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="464"/>
<source>Open file</source>
<translation>Open file</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="378"/>
<source>Select destination</source>
<translation>Select destination</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="391"/>
<source>Copying</source>
<translation>Copying</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="391"/>
<location filename="../mainwindow.cpp" line="515"/>
<location filename="../mainwindow.cpp" line="523"/>
<location filename="../mainwindow.cpp" line="634"/>
<source>Cancel</source>
<translation>Cancel</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="151"/>
<source>Move marked files</source>
<translation>Move marked files</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="154"/>
<location filename="../mainwindow.cpp" line="507"/>
<source>Index directory</source>
<translation>Index directory</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="198"/>
<source>Thumbnails</source>
<translation>Thumbnails</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="216"/>
<source>Show marked</source>
<translation>Show marked</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="253"/>
<location filename="../mainwindow.cpp" line="254"/>
<source>Help</source>
<translation>Help</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="255"/>
<source>About Tenmon</source>
<translation>About Tenmon</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="256"/>
<source>About Qt</source>
<translation>About Qt</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="391"/>
<source>Moving</source>
<translation>Moving</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="515"/>
<location filename="../mainwindow.cpp" line="523"/>
<source>Indexing FITS files</source>
<translation>Indexing FITS files</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="155"/>
<source>Reindex files</source>
<translation>Reindex files</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="115"/>
<source>FITS/XISF files database</source>
<translation>FITS/XISF files database</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="121"/>
<source>File tree</source>
<translation>File tree</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="234"/>
<source>Star finder</source>
<translation>Star finder</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="169"/>
<source>Edit</source>
<translation>Edit</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="170"/>
<source>Settings</source>
<translation>Settings</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="49"/>
<source>Images (</source>
<translation>Images (</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="48"/>
<source>FITS (*.fits *.fit);;XISF (*.xisf);;</source>
<translation>FITS image (*.fits *.fit);;XISF image (*.xisf);;</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="431"/>
<source>Failed to copy</source>
<translation>Failed to copy</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="431"/>
<source>Failed to move</source>
<translation>Failed to move</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="432"/>
<source>Failed to move from %1 to %2</source>
<translation>Failed to move from %1 to %2</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="432"/>
<source>Failed to copy from %1 to %2</source>
<translation>Failed to copy from %1 to %2</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="63"/>
<source>;;All files (*)</source>
<translation>;;All files (*)</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="631"/>
<source>Move files to trash?</source>
<translation>Move files to trash?</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="631"/>
<source>Do you want to move %1 files to trash?</source>
<translation>Do you want to move %1 files to trash?</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="648"/>
<source>Failed to move file to trash</source>
<translation>Failed to move file to trash</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="648"/>
<source>Failed to move file to trash %1</source>
<translation>Failed to move file to trash %1</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="152"/>
<source>Move marked files to trash</source>
<translation>Move marked files to trash</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="634"/>
<source>Moving marked files to trash</source>
<translation>Moving marked files to trash</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="156"/>
<source>Export database to CSV</source>
<translation>Export database to CSV file</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="696"/>
<source>CSV file (*.csv)</source>
<translation>CSV files (*.csv)</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="128"/>
<source>Histogram</source>
<translation>Histogram</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="179"/>
<source>Bayer mask</source>
<translation>Bayer mask</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="181"/>
<source>RGGB</source>
<translation>RGGB</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="182"/>
<source>GRBG</source>
<translation>GRBG</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="183"/>
<source>GBRG</source>
<translation>GBRG</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="184"/>
<source>BGGR</source>
<translation>BGGR</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="206"/>
<source>Slideshow</source>
<translation>Slideshow</translation>
</message>
</context>
<context>
<name>MarkedFiles</name>
<message>
<location filename="../markedfiles.cpp" line="11"/>
<source>Marked files</source>
<translation>Marked files</translation>
</message>
<message>
<location filename="../markedfiles.cpp" line="22"/>
<source>Filename</source>
<translation>File name</translation>
</message>
<message>
<location filename="../markedfiles.cpp" line="30"/>
<source>Clear selected</source>
<translation>Clear selected</translation>
</message>
<message>
<location filename="../markedfiles.cpp" line="31"/>
<source>Clear all</source>
<translation>Clear all</translation>
</message>
@@ -431,58 +749,94 @@
<context>
<name>QObject</name>
<message>
<location filename="../loadrunable.cpp" line="125"/>
<source>ISO</source>
<translation>ISO</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="126"/>
<source>Shutter speed</source>
<translation>Shutter speed</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="123"/>
<location filename="../loadrunable.cpp" line="250"/>
<location filename="../loadrunable.cpp" line="325"/>
<location filename="../loadrunable.cpp" line="509"/>
<source>Width</source>
<translation>Width</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="124"/>
<location filename="../loadrunable.cpp" line="251"/>
<location filename="../loadrunable.cpp" line="326"/>
<location filename="../loadrunable.cpp" line="510"/>
<source>Height</source>
<translation>Height</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="241"/>
<location filename="../loadrunable.cpp" line="297"/>
<location filename="../loadrunable.cpp" line="363"/>
<location filename="../loadrunable.cpp" line="382"/>
<source>Error</source>
<translation>Error</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="378"/>
<source>Filename</source>
<translation>File name</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="382"/>
<source>Failed to load image</source>
<translation>Failed to load image</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="392"/>
<location filename="../loadrunable.cpp" line="402"/>
<source>Mean</source>
<translation>Mean</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="393"/>
<location filename="../loadrunable.cpp" line="403"/>
<source>Standart deviation</source>
<translation>Standart deviation</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="394"/>
<location filename="../loadrunable.cpp" line="404"/>
<source>Median</source>
<translation>Median</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="395"/>
<location filename="../loadrunable.cpp" line="405"/>
<source>Minimum</source>
<translation>Minimum</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="396"/>
<location filename="../loadrunable.cpp" line="406"/>
<source>Maximum</source>
<translation>Maximum</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="397"/>
<location filename="../loadrunable.cpp" line="407"/>
<source>MAD</source>
<translation>MAD</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="241"/>
<location filename="../loadrunable.cpp" line="363"/>
<source>Unsupported sample format</source>
<translation>Unsupported sample format</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="398"/>
<location filename="../loadrunable.cpp" line="408"/>
<source>Saturated</source>
<translation>Saturated</translation>
</message>
@@ -490,6 +844,7 @@
<context>
<name>STFSlider</name>
<message>
<location filename="../stfslider.cpp" line="41"/>
<source>Press Shift for fine tuning</source>
<translation>Press Shift for fine tuning</translation>
</message>
@@ -497,6 +852,7 @@
<context>
<name>SelectColumnsDialog</name>
<message>
<location filename="../databaseview.cpp" line="52"/>
<source>Select columns</source>
<translation>Select columns</translation>
</message>
@@ -504,71 +860,126 @@
<context>
<name>SettingsDialog</name>
<message>
<location filename="../settingsdialog.cpp" line="38"/>
<source>Settings</source>
<translation>Settings</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="45"/>
<source>How many images are preloaded before and after current image.</source>
<translation>How many images are preloaded before and after current image.</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="51"/>
<source>Thumbnail size in pixels</source>
<translation>Thumbnail size in pixels</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="78"/>
<source>Image preload count</source>
<translation>Image preload count</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="79"/>
<source>Thumbnails size</source>
<translation>Thumbnails size</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="67"/>
<source>Don&apos;t use native file dialog</source>
<translation>Don&apos;t use native file dialog</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="58"/>
<source>Set threshold value that is considered saturated when showing statistics.
For RAW files you may set 22%</source>
<translation>Set threshold value that is considered saturated when showing statistics.
For RAW files you may set 22%</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="80"/>
<source>Saturation</source>
<translation>Saturated</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="71"/>
<source>Nearest</source>
<translation>Nearest</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="71"/>
<source>Linear</source>
<translation>Linear</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="71"/>
<source>Cubic</source>
<translation>Cubic</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="74"/>
<source>Smooth thumbnails</source>
<translation>Smooth thumbnails</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="76"/>
<source>Use box filter when downsampling thumbnails instead of nearest. Slightly slower.</source>
<translation>Use box filter when downsampling thumbnails instead of nearest. Slightly slower.</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="81"/>
<source>Slideshow interval</source>
<translation>Slideshow interval</translation>
</message>
<message>
<source>Image filtering</source>
<translation type="obsolete">Image sampling</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="82"/>
<source>Image interpolation</source>
<translation>Image interpolation</translation>
</message>
</context>
<context>
<name>StretchToolbar</name>
<message>
<location filename="../stretchtoolbar.cpp" line="18"/>
<source>Stretch toolbar</source>
<translation>Stretch toolbar</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="72"/>
<source>Auto Stretch F12</source>
<translation>Auto Stretch F12</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="76"/>
<source>Reset Screen Transfer Function F11</source>
<translation>Reset Screen Transfer Function F11</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="80"/>
<source>Invert colors</source>
<translation>Invert colors</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="92"/>
<source>Apply auto stretch on load</source>
<translation>Apply auto stretch on load</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="88"/>
<source>Debayer CFA</source>
<translation>Debayer CFA</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="84"/>
<source>False colors</source>
<translation>False colors</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="67"/>
<source>Linked channels</source>
<translation>Linked channels</translation>
</message>
Binary file not shown.
+413 -2
View File
@@ -1,24 +1,191 @@
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE TS>
<TS version="2.1" language="fr_FR" sourcelanguage="en">
<TS version="2.1" language="fr_FR" sourcelanguage="en_US">
<context>
<name>About</name>
<message>
<location filename="../about.cpp" line="12"/>
<source>About Tenmon</source>
<translation>A propos de Tenmon</translation>
</message>
</context>
<context>
<name>BatchProcessing</name>
<message>
<location filename="../batchprocessing.ui" line="14"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="208"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="208"/>
<source>Batch Processing</source>
<translation>Traitement par lot</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="22"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="209"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="209"/>
<source>Input files and directories</source>
<translation>Fichiers et répertoires d&apos;entrée</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="44"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="210"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="210"/>
<source>Add files</source>
<translation>Ajout de fichiers</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="51"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="211"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="211"/>
<source>Add directories</source>
<translation>Ajout de répertoires</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="58"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="212"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="212"/>
<source>Remove</source>
<translation>Supprimer</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="65"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="213"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="213"/>
<source>Remove all</source>
<translation>Supprimer tout</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="78"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="214"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="214"/>
<source>Output directory</source>
<translation>Répertoire de sortie</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="92"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="215"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="215"/>
<source>Browse</source>
<translation>Naviger</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="103"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="216"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="216"/>
<source>Scripts</source>
<translation>Scripts</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="123"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="217"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="217"/>
<source>Open scripts</source>
<translation>Ouvrir les scripts</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="142"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="218"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="218"/>
<source>Log</source>
<translation>Journal</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="182"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="227"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="219"/>
<source>Start script</source>
<translation>Lancer le script</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="192"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="228"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="220"/>
<source>Stop script</source>
<translation>Arrêter le script</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="199"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="229"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="221"/>
<source>Close</source>
<translation>Fermer</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="118"/>
<source>Interrupt running script?</source>
<translation>Interrompre l&apos;execution du script?</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="138"/>
<source>Select files</source>
<translation>Choisir les fichiers</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="149"/>
<source>Select directory</source>
<translation>Choisir le répertoire</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="170"/>
<source>Select output directory</source>
<translation>Choisir le répertoire de sortie</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="212"/>
<source>Invalid output directory</source>
<translation>Répertoire invalide</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="212"/>
<source>Output directory path doesn&apos;t exist or is not writable</source>
<translation>Le répertoire de sortie n&apos;existe pas ou ne peut pas être écrit</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="243"/>
<source>Enter text</source>
<translation>Entrer le texte</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="250"/>
<source>Enter integer number</source>
<translation>Entrer un nombre entier</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="257"/>
<source>Enter float number</source>
<translation>Entrer un nombre flottant</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="264"/>
<source>Select item</source>
<translation>Choisir l&apos;élément</translation>
</message>
<message>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="219"/>
<source>&lt;!DOCTYPE HTML PUBLIC &quot;-//W3C//DTD HTML 4.0//EN&quot; &quot;http://www.w3.org/TR/REC-html40/strict.dtd&quot;&gt;
&lt;html&gt;&lt;head&gt;&lt;meta name=&quot;qrichtext&quot; content=&quot;1&quot; /&gt;&lt;meta charset=&quot;utf-8&quot; /&gt;&lt;style type=&quot;text/css&quot;&gt;
p, li { white-space: pre-wrap; }
hr { height: 1px; border-width: 0; }
li.unchecked::marker { content: &quot;\2610&quot;; }
li.checked::marker { content: &quot;\2612&quot;; }
&lt;/style&gt;&lt;/head&gt;&lt;body style=&quot; font-family:&apos;FreeMono&apos;; font-size:11pt; font-weight:400; font-style:normal;&quot;&gt;
&lt;p style=&quot;-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;&quot;&gt;&lt;br /&gt;&lt;/p&gt;&lt;/body&gt;&lt;/html&gt;</source>
<translation type="unfinished"></translation>
</message>
</context>
<context>
<name>DataBaseView</name>
<message>
<location filename="../databaseview.cpp" line="255"/>
<source>Select columns</source>
<translation>Choix des colonnes</translation>
</message>
<message>
<location filename="../databaseview.cpp" line="293"/>
<source>Text to search, you can % as wildcard</source>
<translation>Texte à chercher, utilisez % comme caractère générique</translation>
</message>
<message>
<location filename="../databaseview.cpp" line="309"/>
<source>Filter</source>
<translation>Filtre</translation>
</message>
@@ -26,10 +193,12 @@
<context>
<name>DatabaseTableView</name>
<message>
<location filename="../databaseview.cpp" line="215"/>
<source>Mark</source>
<translation>Marquer</translation>
</message>
<message>
<location filename="../databaseview.cpp" line="216"/>
<source>Unmark</source>
<translation>Décocher</translation>
</message>
@@ -37,6 +206,7 @@
<context>
<name>FITSFileModel</name>
<message>
<location filename="../databaseview.cpp" line="194"/>
<source>File name</source>
<translation>Nom de fichier</translation>
</message>
@@ -44,22 +214,27 @@
<context>
<name>FilesystemWidget</name>
<message>
<location filename="../filesystemwidget.cpp" line="26"/>
<source>Sort by filename</source>
<translation>Trier par nom de fichier</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="27"/>
<source>Sort by time</source>
<translation>Trier par heure</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="28"/>
<source>Sort by size</source>
<translation>Trier par taille</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="29"/>
<source>Sort by type</source>
<translation>Trier par type</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="30"/>
<source>Reverse</source>
<translation>Ordre inverse</translation>
</message>
@@ -67,34 +242,43 @@
<context>
<name>Filetree</name>
<message>
<location filename="../filesystemwidget.cpp" line="90"/>
<location filename="../filesystemwidget.cpp" line="94"/>
<source>Open</source>
<translation>Ouvrir</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="96"/>
<source>Copy marked files</source>
<translation>Copier les fichiers marqués</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="97"/>
<source>Move marked files</source>
<translation>Déplacer les fichiers marqués</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="98"/>
<source>Index directory</source>
<translation>Indexer le répertoire</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="95"/>
<source>Set as root</source>
<translation>Définir comme répertoire racine</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="102"/>
<source>Reset root</source>
<translation>Réinitialiser la racine</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="103"/>
<source>Go up</source>
<translation>Monter</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="104"/>
<source>Show hidden files</source>
<translation>Afficher les fichiers cachés</translation>
</message>
@@ -102,6 +286,7 @@
<context>
<name>HelpDialog</name>
<message>
<location filename="../about.cpp" line="33"/>
<source>Help</source>
<translation>Aide</translation>
</message>
@@ -109,6 +294,7 @@
<context>
<name>Histogram</name>
<message>
<location filename="../histogram.cpp" line="72"/>
<source>Logarithmic scale</source>
<translation>Échelle logarithmique</translation>
</message>
@@ -116,22 +302,27 @@
<context>
<name>ImageInfo</name>
<message>
<location filename="../imageinfo.cpp" line="65"/>
<source>Property</source>
<translation>Propriété</translation>
</message>
<message>
<location filename="../imageinfo.cpp" line="65"/>
<source>Value</source>
<translation>Valeur</translation>
</message>
<message>
<location filename="../imageinfo.cpp" line="65"/>
<source>Comment</source>
<translation>Commentaire</translation>
</message>
<message>
<location filename="../imageinfo.cpp" line="82"/>
<source>FITS Header</source>
<translation>En-tête FITS</translation>
</message>
<message>
<location filename="../imageinfo.cpp" line="91"/>
<source>Image info</source>
<translation>Informations sur l&apos;image</translation>
</message>
@@ -139,6 +330,7 @@
<context>
<name>ImageRingList</name>
<message>
<location filename="../imageringlist.cpp" line="375"/>
<source>Name</source>
<translation>Nom</translation>
</message>
@@ -146,284 +338,410 @@
<context>
<name>ImageWidget</name>
<message>
<location filename="../imagescrollareagl.cpp" line="78"/>
<location filename="../imagescrollareagl.cpp" line="447"/>
<source>OpenGL error</source>
<translation>Erreur OpenGL</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="78"/>
<location filename="../imagescrollareagl.cpp" line="447"/>
<source>Could not initialize OpenGL 3.3 context. Ensure that proper GPU driver is installed.</source>
<translation>Impossible d&apos;initialiser le contexte OpenGL 3.3. Assurez-vous que le pilote GPU approprié est installé.</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="674"/>
<source>L:%1</source>
<translation>L:%1</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="674"/>
<location filename="../imagescrollareagl.cpp" line="676"/>
<source>X:%3 Y:%4</source>
<translation>X:%3 Y:%4</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="676"/>
<source>R:%1 G:%2 B:%3</source>
<translation>R:%1 G:%2 B:%3</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="99"/>
<source>Failed to load image</source>
<translation>Échec du chargement de l&apos;image</translation>
</message>
</context>
<context>
<name>MainWindow</name>
<message>
<location filename="../mainwindow.cpp" line="68"/>
<source>Image info</source>
<translation>Information sur l&apos;image</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="77"/>
<source>Can&apos;t open DB</source>
<translation>Ne peut ouvrir la base de donnée</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="77"/>
<source>Can&apos;t open SQLITE database</source>
<translation>Ne peut ouvrir la base de donnée SQLITE</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="110"/>
<source>Filesystem</source>
<translation>Système de fichier</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="134"/>
<source>Tenmon</source>
<translation>Tenmon</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="145"/>
<source>File</source>
<translation>Fichier</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="146"/>
<source>Open</source>
<translation>Ouvrir</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="150"/>
<source>Copy marked files</source>
<translation>Copier les fichiers marqués</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="148"/>
<location filename="../mainwindow.cpp" line="532"/>
<location filename="../mainwindow.cpp" line="694"/>
<source>Save as</source>
<translation>Enregistrer sous</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="147"/>
<location filename="../mainwindow.cpp" line="494"/>
<source>Open directory recursively</source>
<translation>Ouvrir le répertoire de manière récursive</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="157"/>
<source>Batch processing</source>
<translation>Traitement par lot</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="163"/>
<source>Live mode</source>
<translation>Mode temps réel</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="165"/>
<source>Exit</source>
<translation>Sortir</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="173"/>
<source>View</source>
<translation>Voir</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="174"/>
<source>Zoom In</source>
<translation>Zoom avant</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="175"/>
<source>Zoom Out</source>
<translation>Zoom arrière</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="176"/>
<source>Best Fit</source>
<translation>Meilleur ajustement</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="177"/>
<source>100%</source>
<translation>100%</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="210"/>
<source>Select</source>
<translation>Sélectionner</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="211"/>
<source>Mark</source>
<translation>Marquer</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="212"/>
<source>Unmark</source>
<translation>Décocher</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="214"/>
<source>Mark and next</source>
<translation>Marquer et suivant</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="215"/>
<source>Unmark and next</source>
<translation>Décocher et suivant</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="219"/>
<source>Analyze</source>
<translation>Analyse</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="232"/>
<source>Image statistics</source>
<translation>Statistiques de l&apos;image</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="233"/>
<source>Peak finder</source>
<translation>Détecteur de pic</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="244"/>
<source>Docks</source>
<translation>Fenêtres encrables</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="464"/>
<source>Open file</source>
<translation>Ouvrir le ficher</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="378"/>
<source>Select destination</source>
<translation>Choisir la destination</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="391"/>
<source>Copying</source>
<translation>Copier</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="391"/>
<location filename="../mainwindow.cpp" line="515"/>
<location filename="../mainwindow.cpp" line="523"/>
<location filename="../mainwindow.cpp" line="634"/>
<source>Cancel</source>
<translation>Abandon</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="151"/>
<source>Move marked files</source>
<translation>Déplacer les fichiers marqués</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="154"/>
<location filename="../mainwindow.cpp" line="507"/>
<source>Index directory</source>
<translation>Indexer le répertoire</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="198"/>
<source>Thumbnails</source>
<translation>Vignettes</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="216"/>
<source>Show marked</source>
<translation>Afficher marqué</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="253"/>
<location filename="../mainwindow.cpp" line="254"/>
<source>Help</source>
<translation>Aide</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="255"/>
<source>About Tenmon</source>
<translation>A propos de Tenmon</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="256"/>
<source>About Qt</source>
<translation>A propos de Qt</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="391"/>
<source>Moving</source>
<translation>Déplacement</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="515"/>
<location filename="../mainwindow.cpp" line="523"/>
<source>Indexing FITS files</source>
<translation>Indexation des fichiers FITS</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="155"/>
<source>Reindex files</source>
<translation>-indexer les fichiers</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="115"/>
<source>FITS/XISF files database</source>
<translation>Base de donnée FITS/XISF</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="121"/>
<source>File tree</source>
<translation>Arborescence de fichiers</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="234"/>
<source>Star finder</source>
<translation>Détecteur d&apos;étoiles</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="169"/>
<source>Edit</source>
<translation>Éditer</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="170"/>
<source>Settings</source>
<translation>Réglages</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="49"/>
<source>Images (</source>
<translation>Images (</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="48"/>
<source>FITS (*.fits *.fit);;XISF (*.xisf);;</source>
<translation>FITS image (*.fits *.fit);;XISF image (*.xisf);;</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="431"/>
<source>Failed to copy</source>
<translation>Échec de la copie</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="431"/>
<source>Failed to move</source>
<translation>Échec du déplacement</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="432"/>
<source>Failed to move from %1 to %2</source>
<translation>Échec du déplacement de %1 vers %2</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="432"/>
<source>Failed to copy from %1 to %2</source>
<translation>Échec de la copie de %1 vers %2</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="63"/>
<source>;;All files (*)</source>
<translation>;;Tout les fichiers (*)</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="631"/>
<source>Move files to trash?</source>
<translation>Déplacer les fichiers dans la corbeille?</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="631"/>
<source>Do you want to move %1 files to trash?</source>
<translation>Voulez-vous déplacer le fichier %1 dans la corbeille?</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="648"/>
<source>Failed to move file to trash</source>
<translation>Echec du déplacement dans la corbeille</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="648"/>
<source>Failed to move file to trash %1</source>
<translation>Echec du déplacement de %1 dans la corbeille</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="152"/>
<source>Move marked files to trash</source>
<translation>Déplacer les fichiers marqués dans la corbeille</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="634"/>
<source>Moving marked files to trash</source>
<translation>Déplacement des fichiers marqués dans la corbeille</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="156"/>
<source>Export database to CSV</source>
<translation>Exporter la base de données vers un fichier CSV</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="696"/>
<source>CSV file (*.csv)</source>
<translation>Fichiers CSV (*.csv)</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="128"/>
<source>Histogram</source>
<translation>Histogramme</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="179"/>
<source>Bayer mask</source>
<translation>Masque Bayer</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="181"/>
<source>RGGB</source>
<translation>RGGB</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="182"/>
<source>GRBG</source>
<translation>GRBG</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="183"/>
<source>GBRG</source>
<translation>GBRG</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="184"/>
<source>BGGR</source>
<translation>BGGR</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="206"/>
<source>Slideshow</source>
<translation>Diaporama</translation>
</message>
</context>
<context>
<name>MarkedFiles</name>
<message>
<location filename="../markedfiles.cpp" line="11"/>
<source>Marked files</source>
<translation>Fichiers marqués</translation>
</message>
<message>
<location filename="../markedfiles.cpp" line="22"/>
<source>Filename</source>
<translation>Nom de fichier</translation>
</message>
<message>
<location filename="../markedfiles.cpp" line="30"/>
<source>Clear selected</source>
<translation>Effacer la sélection</translation>
</message>
<message>
<location filename="../markedfiles.cpp" line="31"/>
<source>Clear all</source>
<translation>Effacer tout</translation>
</message>
@@ -431,58 +749,94 @@
<context>
<name>QObject</name>
<message>
<location filename="../loadrunable.cpp" line="125"/>
<source>ISO</source>
<translation>ISO</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="126"/>
<source>Shutter speed</source>
<translation>Vitesse d&apos;obturation</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="123"/>
<location filename="../loadrunable.cpp" line="250"/>
<location filename="../loadrunable.cpp" line="325"/>
<location filename="../loadrunable.cpp" line="509"/>
<source>Width</source>
<translation>Largeur</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="124"/>
<location filename="../loadrunable.cpp" line="251"/>
<location filename="../loadrunable.cpp" line="326"/>
<location filename="../loadrunable.cpp" line="510"/>
<source>Height</source>
<translation>Hauteur</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="241"/>
<location filename="../loadrunable.cpp" line="297"/>
<location filename="../loadrunable.cpp" line="363"/>
<location filename="../loadrunable.cpp" line="382"/>
<source>Error</source>
<translation>Erreur</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="378"/>
<source>Filename</source>
<translation>Nom de fichier</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="382"/>
<source>Failed to load image</source>
<translation>Échec du chargement de l&apos;image</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="392"/>
<location filename="../loadrunable.cpp" line="402"/>
<source>Mean</source>
<translation>Moyenne</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="393"/>
<location filename="../loadrunable.cpp" line="403"/>
<source>Standart deviation</source>
<translation>Écart-type</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="394"/>
<location filename="../loadrunable.cpp" line="404"/>
<source>Median</source>
<translation>Médiane</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="395"/>
<location filename="../loadrunable.cpp" line="405"/>
<source>Minimum</source>
<translation>Minimum</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="396"/>
<location filename="../loadrunable.cpp" line="406"/>
<source>Maximum</source>
<translation>Maximum</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="397"/>
<location filename="../loadrunable.cpp" line="407"/>
<source>MAD</source>
<translation>MAD</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="241"/>
<location filename="../loadrunable.cpp" line="363"/>
<source>Unsupported sample format</source>
<translation>Format non pris en charge</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="398"/>
<location filename="../loadrunable.cpp" line="408"/>
<source>Saturated</source>
<translation>Saturé</translation>
</message>
@@ -490,6 +844,7 @@
<context>
<name>STFSlider</name>
<message>
<location filename="../stfslider.cpp" line="41"/>
<source>Press Shift for fine tuning</source>
<translation>Appuyez sur Shift pour un réglage fin</translation>
</message>
@@ -497,6 +852,7 @@
<context>
<name>SelectColumnsDialog</name>
<message>
<location filename="../databaseview.cpp" line="52"/>
<source>Select columns</source>
<translation>Choix des colonnes</translation>
</message>
@@ -504,73 +860,128 @@
<context>
<name>SettingsDialog</name>
<message>
<location filename="../settingsdialog.cpp" line="38"/>
<source>Settings</source>
<translation>Réglages</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="45"/>
<source>How many images are preloaded before and after current image.</source>
<translation>Combien d&apos;images sont préchargées avant et après l&apos;image courante.</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="51"/>
<source>Thumbnail size in pixels</source>
<translation>Taille des vignettes en pixels</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="78"/>
<source>Image preload count</source>
<translation>Nombre d&apos;images préchargées</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="79"/>
<source>Thumbnails size</source>
<translation>Taille des vignette</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="67"/>
<source>Don&apos;t use native file dialog</source>
<translation>N&apos;utilisez pas la boîte de dialogue de fichier natif</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="58"/>
<source>Set threshold value that is considered saturated when showing statistics.
For RAW files you may set 22%</source>
<translation>Définissez la valeur seuil qui est considérée comme saturée lors de l&apos;affichage des statistiques.
Pour les fichiers RAW, vous pouvez définir 22&#xa0;%</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="80"/>
<source>Saturation</source>
<translation>Saturé</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="71"/>
<source>Nearest</source>
<translation>Voisin le plus proche</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="71"/>
<source>Linear</source>
<translation>Linéaire</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="71"/>
<source>Cubic</source>
<translation>Cubique</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="74"/>
<source>Smooth thumbnails</source>
<translation>Miniatures fluides</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="76"/>
<source>Use box filter when downsampling thumbnails instead of nearest. Slightly slower.</source>
<translation>Utilisez un filtre boîte lors du sous-échantillonnage des vignettes au lieu des plus proches. Un peu plus lent.</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="81"/>
<source>Slideshow interval</source>
<translation>Intervalle du diaporama</translation>
</message>
<message>
<source>Image filtering</source>
<translation type="obsolete">Échantillonnage d&apos;images</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="82"/>
<source>Image interpolation</source>
<translation>Interpolation d&apos;images</translation>
</message>
</context>
<context>
<name>StretchToolbar</name>
<message>
<location filename="../stretchtoolbar.cpp" line="18"/>
<source>Stretch toolbar</source>
<translation>Réglage de la luminosité</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="72"/>
<source>Auto Stretch F12</source>
<translation>Luminosité automatique F12</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="76"/>
<source>Reset Screen Transfer Function F11</source>
<translation>Réinitialiser la fonction de transfert d&apos;écran F11</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="80"/>
<source>Invert colors</source>
<translation>Inverser les couleurs</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="92"/>
<source>Apply auto stretch on load</source>
<translation>Appliquer la luminosité automatiquement au chargement</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="88"/>
<source>Debayer CFA</source>
<translation>Débayeriser CFA</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="84"/>
<source>False colors</source>
<translation>Fausses couleurs</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="67"/>
<source>Linked channels</source>
<translation>Chaînes liées</translation>
<translation>Canaux liés</translation>
</message>
</context>
</TS>
Binary file not shown.
+408 -1
View File
@@ -1,24 +1,191 @@
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE TS>
<TS version="2.1" language="sk_SK" sourcelanguage="en">
<TS version="2.1" language="sk_SK" sourcelanguage="en_US">
<context>
<name>About</name>
<message>
<location filename="../about.cpp" line="12"/>
<source>About Tenmon</source>
<translation>O Tenmon</translation>
</message>
</context>
<context>
<name>BatchProcessing</name>
<message>
<location filename="../batchprocessing.ui" line="14"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="208"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="208"/>
<source>Batch Processing</source>
<translation>Hromadné spracovanie</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="22"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="209"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="209"/>
<source>Input files and directories</source>
<translation>Vstupné súbory a adresáry</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="44"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="210"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="210"/>
<source>Add files</source>
<translation>Pridať súbory</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="51"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="211"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="211"/>
<source>Add directories</source>
<translation>Pridať adresár</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="58"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="212"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="212"/>
<source>Remove</source>
<translation>Odstrániť</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="65"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="213"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="213"/>
<source>Remove all</source>
<translation>Odstrániť všetko</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="78"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="214"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="214"/>
<source>Output directory</source>
<translation>Výstupný adresár</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="92"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="215"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="215"/>
<source>Browse</source>
<translation>Vybrať adresár</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="103"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="216"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="216"/>
<source>Scripts</source>
<translation>Skripty</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="123"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="217"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="217"/>
<source>Open scripts</source>
<translation>Otvoriť scripty</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="142"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="218"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="218"/>
<source>Log</source>
<translation>Záznam</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="182"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="227"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="219"/>
<source>Start script</source>
<translation>Spustiť script</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="192"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="228"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="220"/>
<source>Stop script</source>
<translation>Zastaviť script</translation>
</message>
<message>
<location filename="../batchprocessing.ui" line="199"/>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="229"/>
<location filename="../build-release/tenmon_autogen/include/ui_batchprocessing.h" line="221"/>
<source>Close</source>
<translation>Zatvoriť</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="118"/>
<source>Interrupt running script?</source>
<translation>Prerušiť bežiaci script?</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="138"/>
<source>Select files</source>
<translation>Vybrať súbory</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="149"/>
<source>Select directory</source>
<translation>Vybrať adresár</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="170"/>
<source>Select output directory</source>
<translation>Vybrať výstupný adresár</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="212"/>
<source>Invalid output directory</source>
<translation>Neplatný výstupný adresár</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="212"/>
<source>Output directory path doesn&apos;t exist or is not writable</source>
<translation>Vystupný adresár neexistuje alebo sa doňho nedá zapisovať</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="243"/>
<source>Enter text</source>
<translation>Zadajte text</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="250"/>
<source>Enter integer number</source>
<translation>Zadajte celé číslo</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="257"/>
<source>Enter float number</source>
<translation>Zadajte reálne číslo</translation>
</message>
<message>
<location filename="../batchprocessing.cpp" line="264"/>
<source>Select item</source>
<translation>Vyberte položku</translation>
</message>
<message>
<location filename="../build-debug/tenmon_autogen/include/ui_batchprocessing.h" line="219"/>
<source>&lt;!DOCTYPE HTML PUBLIC &quot;-//W3C//DTD HTML 4.0//EN&quot; &quot;http://www.w3.org/TR/REC-html40/strict.dtd&quot;&gt;
&lt;html&gt;&lt;head&gt;&lt;meta name=&quot;qrichtext&quot; content=&quot;1&quot; /&gt;&lt;meta charset=&quot;utf-8&quot; /&gt;&lt;style type=&quot;text/css&quot;&gt;
p, li { white-space: pre-wrap; }
hr { height: 1px; border-width: 0; }
li.unchecked::marker { content: &quot;\2610&quot;; }
li.checked::marker { content: &quot;\2612&quot;; }
&lt;/style&gt;&lt;/head&gt;&lt;body style=&quot; font-family:&apos;FreeMono&apos;; font-size:11pt; font-weight:400; font-style:normal;&quot;&gt;
&lt;p style=&quot;-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;&quot;&gt;&lt;br /&gt;&lt;/p&gt;&lt;/body&gt;&lt;/html&gt;</source>
<translation type="unfinished"></translation>
</message>
</context>
<context>
<name>DataBaseView</name>
<message>
<location filename="../databaseview.cpp" line="255"/>
<source>Select columns</source>
<translation>Vyber stĺpce</translation>
</message>
<message>
<location filename="../databaseview.cpp" line="293"/>
<source>Text to search, you can % as wildcard</source>
<translation>Text na vyhľadanie, môžete použit % ako zástupný znak</translation>
</message>
<message>
<location filename="../databaseview.cpp" line="309"/>
<source>Filter</source>
<translatorcomment>Meno súboru</translatorcomment>
<translation>Filter</translation>
@@ -27,10 +194,12 @@
<context>
<name>DatabaseTableView</name>
<message>
<location filename="../databaseview.cpp" line="215"/>
<source>Mark</source>
<translation>Označiť</translation>
</message>
<message>
<location filename="../databaseview.cpp" line="216"/>
<source>Unmark</source>
<translation>Odznačiť</translation>
</message>
@@ -38,6 +207,7 @@
<context>
<name>FITSFileModel</name>
<message>
<location filename="../databaseview.cpp" line="194"/>
<source>File name</source>
<translation>Meno súboru</translation>
</message>
@@ -45,22 +215,27 @@
<context>
<name>FilesystemWidget</name>
<message>
<location filename="../filesystemwidget.cpp" line="26"/>
<source>Sort by filename</source>
<translation>Zoradiť podľa názvu súboru</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="27"/>
<source>Sort by time</source>
<translation>Zoradiť podľa času</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="28"/>
<source>Sort by size</source>
<translation>Zoradiť podľa veľkosti</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="29"/>
<source>Sort by type</source>
<translation>Zoradiť podľa typu</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="30"/>
<source>Reverse</source>
<translation>Otočit poradie</translation>
</message>
@@ -68,34 +243,43 @@
<context>
<name>Filetree</name>
<message>
<location filename="../filesystemwidget.cpp" line="90"/>
<location filename="../filesystemwidget.cpp" line="94"/>
<source>Open</source>
<translation>Otvoriť</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="96"/>
<source>Copy marked files</source>
<translation>Skopírovať označené súbory</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="97"/>
<source>Move marked files</source>
<translation>Presunúť označené súbory</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="98"/>
<source>Index directory</source>
<translation>Indexovať adresár</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="95"/>
<source>Set as root</source>
<translation>Nastav koreňový adresár</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="102"/>
<source>Reset root</source>
<translation>Resetuj koreňový adresár</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="103"/>
<source>Go up</source>
<translation>O úroveň vyššie</translation>
</message>
<message>
<location filename="../filesystemwidget.cpp" line="104"/>
<source>Show hidden files</source>
<translation>Zobraz skryté súbory</translation>
</message>
@@ -103,6 +287,7 @@
<context>
<name>HelpDialog</name>
<message>
<location filename="../about.cpp" line="33"/>
<source>Help</source>
<translation>Pomoc</translation>
</message>
@@ -110,6 +295,7 @@
<context>
<name>Histogram</name>
<message>
<location filename="../histogram.cpp" line="72"/>
<source>Logarithmic scale</source>
<translation>Logaritmická stupnica</translation>
</message>
@@ -117,22 +303,27 @@
<context>
<name>ImageInfo</name>
<message>
<location filename="../imageinfo.cpp" line="65"/>
<source>Property</source>
<translation>Vlastnosť</translation>
</message>
<message>
<location filename="../imageinfo.cpp" line="65"/>
<source>Value</source>
<translation>Hodnota</translation>
</message>
<message>
<location filename="../imageinfo.cpp" line="65"/>
<source>Comment</source>
<translation>Komentár</translation>
</message>
<message>
<location filename="../imageinfo.cpp" line="82"/>
<source>FITS Header</source>
<translation>FITS hlavička</translation>
</message>
<message>
<location filename="../imageinfo.cpp" line="91"/>
<source>Image info</source>
<translation>Informácie o obrázku</translation>
</message>
@@ -140,6 +331,7 @@
<context>
<name>ImageRingList</name>
<message>
<location filename="../imageringlist.cpp" line="375"/>
<source>Name</source>
<translation>Meno</translation>
</message>
@@ -147,284 +339,410 @@
<context>
<name>ImageWidget</name>
<message>
<location filename="../imagescrollareagl.cpp" line="78"/>
<location filename="../imagescrollareagl.cpp" line="447"/>
<source>OpenGL error</source>
<translation>OpenGL chyba</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="78"/>
<location filename="../imagescrollareagl.cpp" line="447"/>
<source>Could not initialize OpenGL 3.3 context. Ensure that proper GPU driver is installed.</source>
<translation>Nepodarilo sa incializovať OpenGL 3.3 context. Ubezpečte sa že nainštalované ovládače pre GPU.</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="674"/>
<source>L:%1</source>
<translation>L:%1</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="674"/>
<location filename="../imagescrollareagl.cpp" line="676"/>
<source>X:%3 Y:%4</source>
<translation>X:%3 Y:%4</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="676"/>
<source>R:%1 G:%2 B:%3</source>
<translation>R:%1 G:%2 B:%3</translation>
</message>
<message>
<location filename="../imagescrollareagl.cpp" line="99"/>
<source>Failed to load image</source>
<translation>Zlyhalo načítanie obrázka</translation>
</message>
</context>
<context>
<name>MainWindow</name>
<message>
<location filename="../mainwindow.cpp" line="68"/>
<source>Image info</source>
<translation>Informácie o obrázku</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="77"/>
<source>Can&apos;t open DB</source>
<translation>Nie je možné otvoriť DB</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="77"/>
<source>Can&apos;t open SQLITE database</source>
<translation>Nie je možné otvoriť SQLITE databázu</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="110"/>
<source>Filesystem</source>
<translation>Zoznam súborov</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="134"/>
<source>Tenmon</source>
<translation>Tenmon</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="145"/>
<source>File</source>
<translation>Súbor</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="146"/>
<source>Open</source>
<translation>Otvoriť</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="150"/>
<source>Copy marked files</source>
<translation>Skopírovať označené súbory</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="148"/>
<location filename="../mainwindow.cpp" line="532"/>
<location filename="../mainwindow.cpp" line="694"/>
<source>Save as</source>
<translation>Uložiť ako</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="147"/>
<location filename="../mainwindow.cpp" line="494"/>
<source>Open directory recursively</source>
<translation>Otvoriť adresár rekurzívne</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="157"/>
<source>Batch processing</source>
<translation>Hromadné spracovanie</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="165"/>
<source>Exit</source>
<translation>Ukončiť</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="173"/>
<source>View</source>
<translation>Zobrazenie</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="174"/>
<source>Zoom In</source>
<translation>Priblížiť</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="175"/>
<source>Zoom Out</source>
<translation>Oddialiť</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="176"/>
<source>Best Fit</source>
<translation>Najlepšia veľkosť</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="177"/>
<source>100%</source>
<translation>100%</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="210"/>
<source>Select</source>
<translation>Výber</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="211"/>
<source>Mark</source>
<translation>Označiť</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="212"/>
<source>Unmark</source>
<translation>Odznačiť</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="214"/>
<source>Mark and next</source>
<translation>Označiť a ďaľší</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="215"/>
<source>Unmark and next</source>
<translation>Odznačiť a ďaľší</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="219"/>
<source>Analyze</source>
<translation>Analýza</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="232"/>
<source>Image statistics</source>
<translation>Štatistiky obrázka</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="244"/>
<source>Docks</source>
<translation>Dokovacie panely</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="464"/>
<source>Open file</source>
<translation>Otvoriť súbor</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="378"/>
<source>Select destination</source>
<translation>Vybrať cieľ</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="391"/>
<source>Copying</source>
<translation>Kopírovanie</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="391"/>
<location filename="../mainwindow.cpp" line="515"/>
<location filename="../mainwindow.cpp" line="523"/>
<location filename="../mainwindow.cpp" line="634"/>
<source>Cancel</source>
<translation>Zrušiť</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="151"/>
<source>Move marked files</source>
<translation>Presunúť označené súbory</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="154"/>
<location filename="../mainwindow.cpp" line="507"/>
<source>Index directory</source>
<translation>Indexovať adresár</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="163"/>
<source>Live mode</source>
<translation>Živý mód</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="198"/>
<source>Thumbnails</source>
<translation>Náhľady</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="216"/>
<source>Show marked</source>
<translation>Ukázať označené</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="233"/>
<source>Peak finder</source>
<translation>Vyhľadávač vrcholov</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="253"/>
<location filename="../mainwindow.cpp" line="254"/>
<source>Help</source>
<translation>Pomoc</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="255"/>
<source>About Tenmon</source>
<translation>O Tenmon</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="256"/>
<source>About Qt</source>
<translation>O Qt</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="391"/>
<source>Moving</source>
<translation>Presúvanie</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="515"/>
<location filename="../mainwindow.cpp" line="523"/>
<source>Indexing FITS files</source>
<translation>Indexovanie FITS/XISF súborov</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="155"/>
<source>Reindex files</source>
<translation>Reindexuj súbory</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="115"/>
<source>FITS/XISF files database</source>
<translation>Databáza FITS/XISF súborov</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="121"/>
<source>File tree</source>
<translation>Strom súborov</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="234"/>
<source>Star finder</source>
<translation>Vyhľadávač hviezd</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="169"/>
<source>Edit</source>
<translation>Upraviť</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="170"/>
<source>Settings</source>
<translation>Nastavenia</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="49"/>
<source>Images (</source>
<translation>Obrázky (</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="48"/>
<source>FITS (*.fits *.fit);;XISF (*.xisf);;</source>
<translation>Obrázok FITS (*.fits *.fit);;Obrázok XISF (*.xisf);;</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="431"/>
<source>Failed to copy</source>
<translation>Zlyhalo kopírovanie</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="431"/>
<source>Failed to move</source>
<translation>Zlyhalo presúvanie</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="432"/>
<source>Failed to move from %1 to %2</source>
<translation>Zlyhalo presúvanie z %1 do %2</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="432"/>
<source>Failed to copy from %1 to %2</source>
<translation>Zlyhalo kopírovanie z %1 do %2</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="63"/>
<source>;;All files (*)</source>
<translation>;;Všetky súbory (*)</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="631"/>
<source>Move files to trash?</source>
<translation>Presunúť súbory do koša?</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="631"/>
<source>Do you want to move %1 files to trash?</source>
<translation>Presunúť %1 súborov do koša?</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="648"/>
<source>Failed to move file to trash</source>
<translation>Zlyhalo presunutie súbora do koša</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="648"/>
<source>Failed to move file to trash %1</source>
<translation>Zlyhalo presunutie súbora do koša %1</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="152"/>
<source>Move marked files to trash</source>
<translation>Presunúť označené súbory do koša</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="634"/>
<source>Moving marked files to trash</source>
<translation>Presúvanie do koša</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="156"/>
<source>Export database to CSV</source>
<translation>Exportovať databázu do CSV súboru</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="696"/>
<source>CSV file (*.csv)</source>
<translation>Súbory CSV (*.csv)</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="128"/>
<source>Histogram</source>
<translation>Histogram</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="179"/>
<source>Bayer mask</source>
<translation>Bayerova maska</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="181"/>
<source>RGGB</source>
<translation>RGGB</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="182"/>
<source>GRBG</source>
<translation>GRBG</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="183"/>
<source>GBRG</source>
<translation>GBRG</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="184"/>
<source>BGGR</source>
<translation>BGGR</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="206"/>
<source>Slideshow</source>
<translation>Prezentácia</translation>
</message>
</context>
<context>
<name>MarkedFiles</name>
<message>
<location filename="../markedfiles.cpp" line="11"/>
<source>Marked files</source>
<translation>Označené súbory</translation>
</message>
<message>
<location filename="../markedfiles.cpp" line="22"/>
<source>Filename</source>
<translation>Meno súboru</translation>
</message>
<message>
<location filename="../markedfiles.cpp" line="30"/>
<source>Clear selected</source>
<translation>Zrušiť vybrané</translation>
</message>
<message>
<location filename="../markedfiles.cpp" line="31"/>
<source>Clear all</source>
<translation>Zrušiť všetky</translation>
</message>
@@ -432,58 +750,94 @@
<context>
<name>QObject</name>
<message>
<location filename="../loadrunable.cpp" line="125"/>
<source>ISO</source>
<translation>ISO</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="126"/>
<source>Shutter speed</source>
<translation>Rýchlosť uzávierky</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="241"/>
<location filename="../loadrunable.cpp" line="297"/>
<location filename="../loadrunable.cpp" line="363"/>
<location filename="../loadrunable.cpp" line="382"/>
<source>Error</source>
<translation>Chyba</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="241"/>
<location filename="../loadrunable.cpp" line="363"/>
<source>Unsupported sample format</source>
<translation>Nepodporovaný formát</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="123"/>
<location filename="../loadrunable.cpp" line="250"/>
<location filename="../loadrunable.cpp" line="325"/>
<location filename="../loadrunable.cpp" line="509"/>
<source>Width</source>
<translation>Šírka</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="124"/>
<location filename="../loadrunable.cpp" line="251"/>
<location filename="../loadrunable.cpp" line="326"/>
<location filename="../loadrunable.cpp" line="510"/>
<source>Height</source>
<translation>Výška</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="378"/>
<source>Filename</source>
<translation>Meno súboru</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="382"/>
<source>Failed to load image</source>
<translation>Zlyhalo načítanie obrázka</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="392"/>
<location filename="../loadrunable.cpp" line="402"/>
<source>Mean</source>
<translation>Priemer</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="393"/>
<location filename="../loadrunable.cpp" line="403"/>
<source>Standart deviation</source>
<translation>Štandardná odchýlka</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="394"/>
<location filename="../loadrunable.cpp" line="404"/>
<source>Median</source>
<translation>Medián</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="395"/>
<location filename="../loadrunable.cpp" line="405"/>
<source>Minimum</source>
<translation>Minimum</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="396"/>
<location filename="../loadrunable.cpp" line="406"/>
<source>Maximum</source>
<translation>Maximum</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="397"/>
<location filename="../loadrunable.cpp" line="407"/>
<source>MAD</source>
<translation>MAD</translation>
</message>
<message>
<location filename="../loadrunable.cpp" line="398"/>
<location filename="../loadrunable.cpp" line="408"/>
<source>Saturated</source>
<translation>Saturované</translation>
</message>
@@ -491,6 +845,7 @@
<context>
<name>STFSlider</name>
<message>
<location filename="../stfslider.cpp" line="41"/>
<source>Press Shift for fine tuning</source>
<translation>Stlačte Shift pre jemné ladenie</translation>
</message>
@@ -498,6 +853,7 @@
<context>
<name>SelectColumnsDialog</name>
<message>
<location filename="../databaseview.cpp" line="52"/>
<source>Select columns</source>
<translation>Výber stĺpcov</translation>
</message>
@@ -505,71 +861,122 @@
<context>
<name>SettingsDialog</name>
<message>
<location filename="../settingsdialog.cpp" line="38"/>
<source>Settings</source>
<translation>Nastavenia</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="45"/>
<source>How many images are preloaded before and after current image.</source>
<translation>Koľko obrázkov sa prednačíta pred a za aktuálnym obrázkom.</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="51"/>
<source>Thumbnail size in pixels</source>
<translation>Veľkosť náhľadu v pixeloch</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="78"/>
<source>Image preload count</source>
<translation>Počet prednačítaných obrázkov</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="79"/>
<source>Thumbnails size</source>
<translation>Veľkosť náhľadu</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="67"/>
<source>Don&apos;t use native file dialog</source>
<translation>Nepoužívať natívny súborový dialóg</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="58"/>
<source>Set threshold value that is considered saturated when showing statistics.
For RAW files you may set 22%</source>
<translation>Nastavuje prahovú hodnotu ktorá sa považuje za saturovanú.
Pre RAW súbory možno treba nastaviť 22%</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="80"/>
<source>Saturation</source>
<translation>Saturované</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="71"/>
<source>Nearest</source>
<translation>Žiadna</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="71"/>
<source>Linear</source>
<translation>Lineárna</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="71"/>
<source>Cubic</source>
<translation>Kubická</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="74"/>
<source>Smooth thumbnails</source>
<translation>Hladké náhľady</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="76"/>
<source>Use box filter when downsampling thumbnails instead of nearest. Slightly slower.</source>
<translation>Pri zemnšovaní obrázkov na náhľady sa pixely spriemerujú namiesto najblžšej hodnoty. Trocha pomalšie.</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="81"/>
<source>Slideshow interval</source>
<translation>Interval medzi obrázkami</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="82"/>
<source>Image interpolation</source>
<translation>Interpolácia obrázku</translation>
</message>
</context>
<context>
<name>StretchToolbar</name>
<message>
<location filename="../stretchtoolbar.cpp" line="18"/>
<source>Stretch toolbar</source>
<translation>Panel úrovní</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="72"/>
<source>Auto Stretch F12</source>
<translation>Automatické natiahnutie F12</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="76"/>
<source>Reset Screen Transfer Function F11</source>
<translation>Resetuj funkciu prevodu na obrazovku F11</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="80"/>
<source>Invert colors</source>
<translation>Invertuj farby</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="92"/>
<source>Apply auto stretch on load</source>
<translation>Aplikuj automatické natiahnutie pri načítaní</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="88"/>
<source>Debayer CFA</source>
<translation>Preveď CFA na farbu</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="84"/>
<source>False colors</source>
<translation>Falošné farby</translation>
</message>
<message>
<location filename="../stretchtoolbar.cpp" line="67"/>
<source>Linked channels</source>
<translation>Prepojené kanály</translation>
</message>