30 #include "wx/busyinfo.h"
39 wxDEFINE_EVENT(wxEVT_COMMAND_THUMBNAILTHREAD_UPDATE, wxCommandEvent);
76 : wxThread(wxTHREAD_DETACHED)
93 for (
int i = 0; i <
m_files.size(); ++i)
96 vigra::BRGBImage* preview=
new vigra::BRGBImage();
97 vigra::BImage* mask =
new vigra::BImage();
98 vigra::ImageImportInfo::ICCProfile iccProfile;
113 vigra::BRGBImage* scaledPreview =
new vigra::BRGBImage(vigra::Diff2D(
m_thumbnailSize.GetWidth(),
m_thumbnailSize.GetHeight()), vigra::RGBValue<vigra::UInt8>(255, 255, 255));
114 vigra::BImage* scaledMask =
new vigra::BImage(scaledPreview->size(), vigra::UInt8(0));
115 vigra::Point2D offset((scaledPreview->width() - preview->width()) / 2, (scaledPreview->height() - preview->height()) / 2);
120 preview = scaledPreview;
126 wxCommandEvent*
event =
new wxCommandEvent(wxEVT_COMMAND_THUMBNAILTHREAD_UPDATE);
128 event->SetEventObject(thumbnailImageData);
135 delete thumbnailImageData;
144 return (wxThread::ExitCode)0;
152 #define THUMBNAIL_SIZE 128
158 wxXmlResource::Get()->LoadDialog(
this, parent,
"browse_pto_dialog");
160 m_dirCtrl = XRCCTRL(*
this,
"browse_dirctrl", wxGenericDirCtrl);
162 m_listCtrl = XRCCTRL(*
this,
"browse_listctrl", wxListCtrl);
165 m_previewCtrl = XRCCTRL(*
this,
"browse_preview", wxStaticBitmap);
166 m_splitter1 = XRCCTRL(*
this,
"browse_splitter1", wxSplitterWindow);
167 m_splitter2 = XRCCTRL(*
this,
"browse_splitter2", wxSplitterWindow);
168 m_showMap = XRCCTRL(*
this,
"browse_show_map", wxButton);
169 m_labelControl = XRCCTRL(*
this,
"browse_statictext", wxStaticText);
172 m_listCtrl->SetWindowStyle(wxLC_REPORT | wxLC_AUTOARRANGE | wxLC_SINGLE_SEL | wxLC_HRULES | wxLC_VRULES);
175 m_listCtrl->InsertColumn(0, _(
"Filename"), wxLIST_FORMAT_LEFT, 300);
176 m_listCtrl->InsertColumn(1, _(
"# images"), wxLIST_FORMAT_RIGHT, 50);
177 m_listCtrl->InsertColumn(2, _(
"# active images"), wxLIST_FORMAT_RIGHT, 50);
178 m_listCtrl->InsertColumn(3, _(
"Projection"), wxLIST_FORMAT_LEFT, 200);
179 m_listCtrl->InsertColumn(4, _(
"Field of view"), wxLIST_FORMAT_LEFT, 100);
180 m_listCtrl->InsertColumn(5, _(
"Canvas size"), wxLIST_FORMAT_LEFT, 100);
181 m_listCtrl->InsertColumn(6, _(
"Model"), wxLIST_FORMAT_LEFT, 200);
182 m_listCtrl->InsertColumn(7, _(
"Lens"), wxLIST_FORMAT_LEFT, 250);
183 m_listCtrl->InsertColumn(8, _(
"Focal length"), wxLIST_FORMAT_LEFT, 150);
184 m_listCtrl->InsertColumn(9, _(
"Capture date"), wxLIST_FORMAT_LEFT, 150);
185 m_listCtrl->InsertColumn(10, _(
"Duration"), wxLIST_FORMAT_LEFT, 50);
186 m_listType = XRCCTRL(*
this,
"browse_list_type", wxChoice);
190 wxConfigBase* config = wxConfigBase::Get();
192 int splitter_pos = config->Read(
"/BrowsePTODialog/splitterPos1", -1l);
193 if (splitter_pos > 0)
197 splitter_pos = config->Read(
"/BrowsePTODialog/splitterPos2", -1l);
198 if (splitter_pos > 0)
203 for (
int j = 0; j <
m_listCtrl->GetColumnCount(); j++)
206 int width = config->Read(wxString::Format(
"/BrowsePTODialog/ColumnWidth%d", j), -1);
207 if (width != -1 && width > 5)
212 m_sortCol = config->Read(
"/BrowsePTODialog/SortColumn", -1);
213 m_sortAscending = config->Read(
"/BrowsePTODialog/SortAscending", 1) == 1 ?
true :
false;
220 if (!startDirectory.IsEmpty())
224 long listType = config->Read(
"/BrowsePTODialog/ListType", 0l);
226 wxCommandEvent event;
227 event.SetInt(listType);
240 wxConfigBase* config = wxConfigBase::Get();
241 config->Write(
"/BrowsePTODialog/splitterPos1",
m_splitter1->GetSashPosition());
242 config->Write(
"/BrowsePTODialog/splitterPos2",
m_splitter2->GetSashPosition());
244 for (
int j = 0; j <
m_listCtrl->GetColumnCount(); j++)
246 config->Write(wxString::Format(
"/BrowsePTODialog/ColumnWidth%d", j),
m_listCtrl->GetColumnWidth(j));
248 config->Write(
"/BrowsePTODialog/ListType",
m_listType->GetSelection());
249 config->Write(wxT(
"/BrowsePTODialog/SortColumn"),
m_sortCol);
250 config->Write(wxT(
"/BrowsePTODialog/SortAscending"),
m_sortAscending ? 1 : 0);
259 index =
m_listCtrl->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
261 if (index >= 0 && index <
m_ptoInfo.size())
267 return wxEmptyString;
305 pano_projection_features proj;
309 wxString str2(proj.name, wxConvLocal);
315 if (!activeImages.empty())
323 info.
camera = img0.getExifModel();
324 info.
lens = img0.getExifLens();
328 info.
focalLength35 = img0.getExifFocalLength() * img0.getCropFactor();
336 if (!img0.getExifDate().empty())
338 struct tm exifdatetime;
341 info.
start = wxDateTime(exifdatetime);
346 imgs.erase(imgs.begin());
347 for (
const auto& img : imgs)
355 memset(&exifdatetime, 0,
sizeof(exifdatetime));
356 if (!imgSrcImage.getExifDate().empty() && imgSrcImage.
getExifDateTime(&exifdatetime) == 0)
358 const wxDateTime dateTime = wxDateTime(exifdatetime);
359 if (info.
start.IsValid())
361 if (dateTime.IsEarlierThan(info.
start))
363 info.
start = dateTime;
365 if (dateTime.IsLaterThan(info.
end))
372 info.
start = dateTime;
377 if (info.
start.IsValid())
382 const auto& fileMetadata = img0.getFileMetadata();
383 const auto& latitude = fileMetadata.find(
"latitude");
384 const auto& longitude = fileMetadata.find(
"longitude");
385 if (latitude != fileMetadata.end())
389 if (longitude != fileMetadata.end())
412 if (info.
start.IsValid())
429 for (
size_t i = 0; i <
m_ptoInfo.size(); ++i)
443 wxWindowDisabler disableAll;
444 wxBusyInfo busy(wxBusyInfoFlags().Icon(
MainFrame::Get()->GetIcon()).Label(wxString::Format(_(
"Reading directory %s"),
m_dirCtrl->GetPath().c_str())));
450 SetTitle(wxString::Format(_(
"Browse project files in %s"),
m_dirCtrl->GetPath().c_str()));
452 wxDir::GetAllFiles(
m_dirCtrl->GetPath(), &files,
"*.pto", wxDIR_FILES | wxDIR_HIDDEN | wxDIR_NO_FOLLOW);
456 for (
size_t i = 0; i < files.size(); ++i)
458 const wxFileName filename(files[i]);
483 if (e.GetIndex() >= 0)
499 const int newCol = e.GetColumn();
517 const int index = e.GetInt();
520 for (
size_t i = 0; i <
m_listCtrl->GetItemCount(); ++i)
539 if (e.GetSelection() == 0)
550 UpdateItemTexts(wxLC_REPORT | wxLC_AUTOARRANGE | wxLC_SINGLE_SEL | wxLC_HRULES | wxLC_VRULES);
565 index =
m_listCtrl->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED);
567 if (index >= 0 && index <
m_ptoInfo.size())
571 const wxString openstreetMapLink =
"https://www.openstreetmap.org/?mlat=" + wxString::FromCDouble(
m_ptoInfo[index].GPSLatitude) +
"&mlon=" + wxString::FromCDouble(
m_ptoInfo[index].GPSLongitude);
572 wxLaunchDefaultBrowser(openstreetMapLink);
582 template <
class Type>
601 template <
class Type>
620 #define SORTASCENDING(functionName, var) \
621 int wxCALLBACK functionName(wxIntPtr item1, wxIntPtr item2, wxIntPtr sortData)\
623 std::vector<PanoInfo>* data = (std::vector<PanoInfo>*)(sortData);\
624 return GreaterComparisonOperator(data->at(item1).var, data->at(item2).var);\
626 #define SORTDESCENDING(functionName, var) \
627 int wxCALLBACK functionName(wxIntPtr item1, wxIntPtr item2, wxIntPtr sortData)\
629 std::vector<PanoInfo>* data = (std::vector<PanoInfo>*)(sortData);\
630 return SmallerComparisonOperator(data->at(item1).var, data->at(item2).var);\
633 SORTASCENDING(SortFilenameAscending, ptoFilename.GetFullName())
647 #undef SORTDESCENDING
650 #define SORTASCENDING(functionName, var) \
651 int wxCALLBACK functionName(wxIntPtr item1, wxIntPtr item2, wxIntPtr sortData)\
653 std::vector<PanoInfo>* data = (std::vector<PanoInfo>*)(sortData);\
654 return data->at(item1).var.CmpNoCase(data->at(item2).var);\
656 #define SORTDESCENDING(functionName, var) \
657 int wxCALLBACK functionName(wxIntPtr item1, wxIntPtr item2, wxIntPtr sortData)\
659 std::vector<PanoInfo>* data = (std::vector<PanoInfo>*)(sortData);\
660 return -(data->at(item1).var.CmpNoCase(data->at(item2).var));\
668 #undef SORTDESCENDING
671 int wxCALLBACK SortFieldOfViewAscending(wxIntPtr item1, wxIntPtr
item2, wxIntPtr sortData)
673 std::vector<PanoInfo>*
data = (std::vector<PanoInfo>*)(sortData);
674 const float fieldOfView1 = data->at(item1).options.getHFOV() * 1000 + data->at(item1).options.getVFOV();
675 const float fieldOfView2 = data->at(item2).options.getHFOV() * 1000 + data->at(item2).options.getVFOV();
681 std::vector<PanoInfo>* data = (std::vector<PanoInfo>*)(sortData);
682 const float fieldOfView1 = data->at(item1).options.getHFOV() * 1000 + data->at(item1).options.getVFOV();
683 const float fieldOfView2 = data->at(item2).options.getHFOV() * 1000 + data->at(item2).options.getVFOV();
690 std::vector<PanoInfo>* data = (std::vector<PanoInfo>*)(sortData);
691 if (data->at(item1).start.IsLaterThan(data->at(item2).start))
695 if (data->at(item1).start.IsEarlierThan(data->at(item2).start))
704 std::vector<PanoInfo>* data = (std::vector<PanoInfo>*)(sortData);
705 if (data->at(item1).start.IsEarlierThan(data->at(item2).start))
709 if (data->at(item1).start.IsLaterThan(data->at(item2).start))
719 std::vector<PanoInfo>* data = (std::vector<PanoInfo>*)(sortData);
720 if (data->at(item1).duration.IsLongerThan(data->at(item2).duration))
724 if (data->at(item1).duration.IsShorterThan(data->at(item2).duration))
733 std::vector<PanoInfo>* data = (std::vector<PanoInfo>*)(sortData);
734 if (data->at(item1).duration.IsShorterThan(data->at(item2).duration))
738 if (data->at(item1).duration.IsLongerThan(data->at(item2).duration))
871 bool deleteStoppedThread =
false;
882 deleteStoppedThread =
true;
886 if (deleteStoppedThread)
904 wxThread::This()->Sleep(1);
914 wxWindowDisabler disableAll;
915 wxBusyInfo busyInfo(wxBusyInfoFlags().Icon(
MainFrame::Get()->GetIcon()).Label(wxString::Format(_(
"Generating preview for %s"),
m_ptoInfo[index].ptoFilename.GetFullName().c_str())));
917 vigra::BRGBImage preview;
919 vigra::ImageImportInfo::ICCProfile iccProfile;
922 wxImage image(preview.width(), preview.height(), (
unsigned char*)preview.data(), (
unsigned char*)mask.data(),
true);
wxDEFINE_EVENT(EVT_QUEUE_PROGRESS, wxCommandEvent)
~BrowsePTOFilesDialog()
destructor, saves position
BrowsePTOFilesDialog(wxWindow *parent, const wxString startDirectory)
Constructor, read from xrc ressource; restore last uses settings and position.
WXIMPEX wxString GetFormattedTimeSpan(const wxTimeSpan &timeSpan)
PanoramaOptions::ProjectionFormat getProjection() const
generate thumbnail from given pto file
bool FileExists(const std::string &filename)
checks if file exists
int SmallerComparisonOperator(const Type &a, const Type &b)
SrcPanoImage getSrcImage(unsigned imgNr) const
get a description of a source image
unsigned int getHeight() const
get panorama height
int wxCALLBACK SortFieldOfViewDescending(wxIntPtr item1, wxIntPtr item2, wxIntPtr sortData)
ThumbnailThread(BrowsePTOFilesDialog *parentDialog, const wxArrayString &fileList, const wxSize size)
void FillPanoInfo(const PanoInfo &info, long index)
add a new item to wxListCtrl and populate all columns
#define SORTASCENDING(functionName, var)
void OnFileChanged(wxListEvent &e)
new file selected, generate preview for new file
friend class ThumbnailThread
void OnThumbnailUpdate(wxCommandEvent &e)
for notifing from ThumbnailThread about new generated thumbnail generated thumbnail is transfered in ...
void GetMonitorProfile(wxString &profileName, cmsHPROFILE &profile)
wxImage * GetwxImage() const
static huginApp * Get()
hack.. kind of a pseudo singleton...
void OnDblClickListCtrl(wxMouseEvent &e)
double click does open file
std::set< unsigned int > UIntSet
wxSplitterWindow * m_splitter1
PanoInfo ParsePTOFile(const wxFileName file)
read the given pto file and add all information to wxListCtrl
options getSize().area()) int wxCALLBACK SortFieldOfViewAscending(wxIntPtr item1
std::string getPathPrefix(const std::string &filename)
Get the path to a filename.
int wxCALLBACK SortDateAscending(wxIntPtr item1, wxIntPtr item2, wxIntPtr sortData)
void OnListColClick(wxListEvent &e)
click on header to sort by column
void CorrectImage(wxImage &image, const vigra::ImageImportInfo::ICCProfile &iccProfile, const cmsHPROFILE &monitorProfile)
apply color correction to given image using input iccProfile and monitor profile
wxString focalLengthString
std::size_t getNrOfImages() const
number of images.
static MainFrame * Get()
hack.. kind of a pseudo singleton...
wxStaticText * m_labelControl
int wxCALLBACK SortDurationAscending(wxIntPtr item1, wxIntPtr item2, wxIntPtr sortData)
#define SORTDESCENDING(functionName, var)
const int getExifDateTime(struct tm *datetime) const
try to convert Exif date time string to struct tm
Dialog for browsing pto files.
void StoreFramePosition(wxTopLevelWindow *frame, const wxString &basename)
Store window size and position in configfile/registry.
wxGenericDirCtrl * m_dirCtrl
vigra::BRGBImage * m_preview
bool ReadPTOFile(const std::string &filename, const std::string &prefix="")
read pto file from the given filename into Panorama object it does some checks on the file and issues...
void OnDirectoryChanged(wxTreeEvent &e)
directory changed, load files from new directory
wxImageList m_thumbnails
image list with all thumbnails
helper class to transfer thumbnail data from worker thread to GUI thread
bool HasMonitorProfile() const
return true if we found a suitable monitor profile and could loading it
wxDECLARE_EVENT(wxEVT_COMMAND_THUMBNAILTHREAD_UPDATE, wxCommandEvent)
HuginBase::PanoramaOptions options
void RestoreFramePosition(wxTopLevelWindow *frame, const wxString &basename)
Restore window size and position from configfile/registry.
void GeneratePreview(int index)
generate preview for pto file with index
wxSplitterWindow * m_splitter2
static double calcFocalLength(SrcPanoImage::Projection proj, double hfov, double crop, vigra::Size2D imageSize)
calcualte focal length, given crop factor and hfov
vigra::pair< typename ROIImage< Image, Alpha >::image_traverser, typename ROIImage< Image, Alpha >::ImageAccessor > destImage(ROIImage< Image, Alpha > &img)
bool stringToDouble(const STR &str_, double &dest)
convert a string to a double, ignore localisation.
std::vector< PanoInfo > m_ptoInfo
info about the pto file
void SortItems()
sort the items according to selected column
vigra::triple< typename ROIImage< Image, Mask >::image_const_traverser, typename ROIImage< Image, Mask >::image_const_traverser, typename ROIImage< Image, Mask >::ImageConstAccessor > srcImageRange(const ROIImage< Image, Mask > &img)
helper function for ROIImages
UIntSet getActiveImages() const
get active images
wxString GetSelectedProject()
return full path of selected project
unsigned int getWidth() const
const PanoramaOptions & getOptions() const
returns the options for this panorama
wxCriticalSection m_ThreadCS
critical section to synchronize with ThumbnailThread
options wxIntPtr wxIntPtr sortData std::vector< PanoInfo > * data
background thread to generate thumbnails of all pto files
wxStaticBitmap * m_previewCtrl
bool readEXIF()
try to fill out information about the image, by examining the exif data
bool GenerateThumbnail(const std::string pto_filename, const wxSize size, vigra::BRGBImage &panoImage, vigra::BImage &panoMask, vigra::ImageImportInfo::ICCProfile &iccProfile)
generate thumbnail image for given pto_filename
Definition of dialog to browse directory with pto files.
void OnListTypeChanged(wxCommandEvent &e)
Change display of wxListCtrl.
int wxCALLBACK SortDateDesending(wxIntPtr item1, wxIntPtr item2, wxIntPtr sortData)
void copyImage(SrcImageIterator src_upperleft, SrcImageIterator src_lowerright, SrcAccessor src_acc, DestImageIterator dest_upperleft, DestAccessor dest_acc)
BrowsePTOFilesDialog * m_parentDialog
static void info(const char *fmt,...)
class to store some information about a pto file on disc
void OnOk(wxCommandEvent &e)
Saves current expression when closing dialog with Ok.
ThreadImage(vigra::BRGBImage *thumbnail, vigra::BImage *mask)
int GreaterComparisonOperator(const Type &a, const Type &b)
All variables of a source image.
ThumbnailThread * m_thumbnailThread
background thumbnail creater thread
long TranslateIndex(const long index)
translate the index to the index of m_ptoInfo
int wxCALLBACK SortDurationDesending(wxIntPtr item1, wxIntPtr item2, wxIntPtr sortData)
void OnShowOnMap(wxCommandEvent &e)
show current pano on Openstreetmap
void EndThumbnailThread()
end background thumbnail creating thread
void UpdateItemTexts(long newStyle)
update all item texts
wxString GetSelectedPath()
return last selected path