4 // Main class for Aven.
6 // Copyright (C) 2001 Mark R. Shinwell.
7 // Copyright (C) 2002,2003,2004,2005,2006,2011,2013,2014,2015,2016,2017,2018 Olly Betts
9 // This program is free software; you can redistribute it and/or modify
10 // it under the terms of the GNU General Public License as published by
11 // the Free Software Foundation; either version 2 of the License, or
12 // (at your option) any later version.
14 // This program is distributed in the hope that it will be useful,
15 // but WITHOUT ANY WARRANTY; without even the implied warranty of
16 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 // GNU General Public License for more details.
19 // You should have received a copy of the GNU General Public License
20 // along with this program; if not, write to the Free Software
21 // Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
40 #include <wx/confbase.h>
43 // wxDisplay was added in wx 2.5; but it may not be built for mingw (because
44 // the header seems to be missing).
45 #include <wx/display.h>
52 static const struct option long_opts
[] = {
53 /* const char *name; int has_arg (0 no_argument, 1 required_*, 2 optional_*); int *flag; int val; */
54 {"survey", required_argument
, 0, 's'},
55 {"print", no_argument
, 0, 'p'},
56 {"help", no_argument
, 0, HLP_HELP
},
57 {"version", no_argument
, 0, HLP_VERSION
},
61 #define short_opts "s:p"
63 static struct help_msg help
[] = {
65 /* TRANSLATORS: --help output for --survey option.
67 * "this" has been added to English translation */
68 {HLP_ENCODELONG(0), /*only load the sub-survey with this prefix*/199, 0},
69 /* TRANSLATORS: --help output for aven --print option */
70 {HLP_ENCODELONG(1), /*print and exit (requires a 3d file)*/119, 0},
77 IMPLEMENT_APP_NO_MAIN(Aven
)
78 IMPLEMENT_WX_THEME_SUPPORT
82 m_Frame(NULL
), m_pageSetupData(NULL
)
84 wxFont::SetDefaultEncoding(wxFONTENCODING_UTF8
);
89 delete m_pageSetupData
;
92 static int getopt_first_response
= 0;
94 static char ** utf8_argv
;
97 bool Aven::Initialize(int& my_argc
, wxChar
**my_argv
)
99 const wxChar
* cmd_line
= GetCommandLineW();
101 // Horrible bodge to handle therion's assumptions about the "Process"
104 // None of these are valid aven command line options, so this is not
105 // going to be triggered accidentally.
106 const wxChar
* p
= wxStrstr(cmd_line
,
107 wxT("aven.exe\" --quiet --log --output="));
109 // Just change the command name in the command line string - that
110 // way the quoting should match what the C runtime expects.
111 wxString
cmd(cmd_line
, p
- cmd_line
);
114 exit(wxExecute(cmd
, wxEXEC_SYNC
));
120 // wxWidgets doesn't split up the command line in the standard way, so
121 // redo it ourselves using the standard API function.
123 // Warning: The returned array from this has no terminating NULL
125 wxChar
** new_argv
= NULL
;
127 new_argv
= CommandLineToArgvW(cmd_line
, &utf8_argc
);
128 bool failed
= (new_argv
== NULL
);
132 FORMAT_MESSAGE_ALLOCATE_BUFFER
|FORMAT_MESSAGE_FROM_SYSTEM
,
139 wxString m
= "CommandLineToArgvW failed: ";
141 wxMessageBox(m
, APP_NAME
, wxOK
| wxCENTRE
| wxICON_EXCLAMATION
);
147 // Convert wide characters to UTF-8.
148 utf8_argv
= new char * [utf8_argc
+ 1];
149 for (int i
= 0; i
< utf8_argc
; ++i
){
150 utf8_argv
[i
] = strdup(wxString(new_argv
[i
]).utf8_str());
152 utf8_argv
[utf8_argc
] = NULL
;
154 if (!failed
) LocalFree(new_argv
);
158 select_charset(CHARSET_UTF8
);
159 /* Want --version and decent --help output, which cmdline does for us.
160 * wxCmdLine is much less good.
162 /* TRANSLATORS: Here "survey" is a "cave map" rather than list of questions
163 * - it should be translated to the terminology that cavers using the
164 * language would use.
166 * Part of aven --help */
167 cmdline_set_syntax_message(/*[SURVEY_FILE]*/269, 0, NULL
);
168 cmdline_init(utf8_argc
, utf8_argv
, short_opts
, long_opts
, NULL
, help
, 0, 1);
169 getopt_first_response
= cmdline_getopt();
171 // The argc and argv arguments don't actually get used here.
173 return wxApp::Initialize(dummy_argc
, NULL
);
176 int main(int argc
, char **argv
)
179 // Currently wxGLCanvas doesn't work under Wayland, and the code segfaults.
180 // https://trac.wxwidgets.org/ticket/17702
181 // Setting GDK_BACKEND=x11 is the recommended workaround, and it seems to
182 // work to set it here. GTK2 doesn't support Wayland, so doesn't need
184 setenv("GDK_BACKEND", "x11", 1);
185 // FIXME: The OpenGL code needs work before scaling on hidpi displays will
186 // work usefully, so for now disable such scaling (which simulates how
187 // things are when using GTK2).
188 setenv("GDK_SCALE", "1", 1);
191 // MacOS passes a magic -psn_XXXX command line argument in argv[1] which
192 // wx ignores for us, but in wxApp::Initialize() which hasn't been
193 // called yet. So we need to remove it ourselves.
194 if (argc
> 1 && strncmp(argv
[1], "-psn_", 5) == 0) {
196 memmove(argv
+ 1, argv
+ 2, argc
* sizeof(char *));
199 // Call msg_init() and start processing the command line first so that
200 // we can respond to --help and --version even without an X display.
202 select_charset(CHARSET_UTF8
);
203 /* Want --version and decent --help output, which cmdline does for us.
204 * wxCmdLine is much less good.
206 cmdline_set_syntax_message(/*[SURVEY_FILE]*/269, 0, NULL
);
207 cmdline_init(argc
, argv
, short_opts
, long_opts
, NULL
, help
, 0, 1);
208 getopt_first_response
= cmdline_getopt();
213 wxWCharBuffer
buf(wxConvFileName
->cMB2WX(argv
[0]));
216 wargv
[0] = wxStrdup(buf
);
218 // Eep - couldn't convert the executable's name to wide characters!
219 wargv
[0] = wxStrdup(APP_NAME
);
223 return wxEntry(wargc
, wargv
);
225 char *dummy_argv
[2] = { argv
[0], NULL
};
227 return wxEntry(dummy_argc
, dummy_argv
);
234 wxLog::SetActiveTarget(new MyLogWindow());
237 // Suppress message box warnings about messages not found.
239 wxLocale
*loc
= new wxLocale();
240 loc
->AddCatalogLookupPathPrefix(wmsg_cfgpth());
241 wxString
msg_lang_str(msg_lang
, wxConvUTF8
);
242 const char *lang
= msg_lang2
? msg_lang2
: msg_lang
;
243 wxString
lang_str(lang
, wxConvUTF8
);
244 loc
->Init(msg_lang_str
, lang_str
, msg_lang_str
);
245 // The existence of the wxLocale object is enough - no need to keep a
249 const char* opt_survey
= NULL
;
250 bool print_and_exit
= false;
254 if (getopt_first_response
) {
255 opt
= getopt_first_response
;
256 getopt_first_response
= 0;
258 opt
= cmdline_getopt();
260 if (opt
== EOF
) break;
262 if (opt_survey
!= NULL
) {
263 // FIXME: Not a helpful error, but this is temporary until
264 // we actually hook up support for specifying multiple
265 // --survey options properly here.
272 print_and_exit
= true;
276 if (print_and_exit
&& !utf8_argv
[optind
]) {
277 cmdline_syntax(); // FIXME : not a helpful error...
282 if (utf8_argv
[optind
]) {
283 fnm
= wxString(utf8_argv
[optind
], wxConvUTF8
);
284 if (fnm
.empty() && *(utf8_argv
[optind
])) {
285 ReportError(wxT("File argument's filename has bad encoding"));
290 if (!GLACanvas::check_visual()) {
292 /* TRANSLATORS: %s will be replaced with "Aven" currently (and
293 * perhaps by "Survex" or other things in future). */
294 m
.Printf(wmsg(/*This version of %s requires OpenGL to work, but it isn’t available.*/405), APP_NAME
);
295 wxMessageBox(m
, APP_NAME
, wxOK
| wxCENTRE
| wxICON_EXCLAMATION
);
299 wxImage::AddHandler(new wxPNGHandler
);
301 // Obtain the screen geometry.
303 wxRect geom
= wxDisplay().GetGeometry();
306 wxClientDisplayRect(&geom
.x
, &geom
.y
, &geom
.width
, &geom
.height
);
309 wxPoint
pos(wxDefaultPosition
);
311 wxConfigBase::Get()->Read(wxT("width"), &width
, 0);
312 if (width
> 0) wxConfigBase::Get()->Read(wxT("height"), &height
, 0);
313 // We used to persist full screen mode (-1 was maximized,
314 // -2 full screen), but people would get stuck in full
315 // screen mode, unsure how to exit.
316 bool maximized
= (width
<= -1);
317 if (width
<= 0 || height
<= 0) {
321 height
= geom
.height
;
323 // Calculate a reasonable size for our window.
326 width
= width
* 3 / 4;
327 height
= height
* 3 / 4;
329 // Impose a minimum size for sanity, and make sure the window fits on
330 // the display (in case the current display is smaller than the one
331 // in use when the window size was saved). (480x320) is about the
332 // smallest usable size for aven's window.
333 const int min_width
= min(geom
.width
, 480);
334 const int min_height
= min(geom
.height
, 320);
335 if (width
< min_width
|| height
< min_height
) {
336 if (width
< min_width
) {
339 if (height
< min_height
) {
342 pos
.x
= geom
.x
+ (geom
.width
- width
) / 4;
343 pos
.y
= geom
.y
+ (geom
.height
- height
) / 4;
347 // Create the main window.
348 m_Frame
= new MainFrm(APP_NAME
, pos
, wxSize(width
, height
));
350 // Select maximised if that's the saved state.
355 if (utf8_argv
[optind
]) {
356 if (!opt_survey
) opt_survey
= "";
357 m_Frame
->OpenFile(fnm
, wxString(opt_survey
, wxConvUTF8
));
360 if (print_and_exit
) {
361 m_Frame
->PrintAndExit();
372 wxPageSetupDialogData
*
373 Aven::GetPageSetupDialogData()
375 if (!m_pageSetupData
) m_pageSetupData
= new wxPageSetupDialogData
;
377 // Fetch paper margins stored on disk.
378 int left
, right
, top
, bottom
;
379 wxConfigBase
* cfg
= wxConfigBase::Get();
380 // These default margins were chosen by looking at all the .ppd files
382 cfg
->Read(wxT("paper_margin_left"), &left
, 7);
383 cfg
->Read(wxT("paper_margin_right"), &right
, 7);
384 cfg
->Read(wxT("paper_margin_top"), &top
, 14);
385 cfg
->Read(wxT("paper_margin_bottom"), &bottom
, 14);
386 m_pageSetupData
->SetMarginTopLeft(wxPoint(left
, top
));
387 m_pageSetupData
->SetMarginBottomRight(wxPoint(right
, bottom
));
389 return m_pageSetupData
;
393 Aven::SetPageSetupDialogData(const wxPageSetupDialogData
& psdd
)
395 if (!m_pageSetupData
) m_pageSetupData
= new wxPageSetupDialogData
;
396 *m_pageSetupData
= psdd
;
398 wxPoint topleft
= psdd
.GetMarginTopLeft();
399 wxPoint bottomright
= psdd
.GetMarginBottomRight();
401 // Store user specified paper margins on disk/in registry.
402 wxConfigBase
* cfg
= wxConfigBase::Get();
403 cfg
->Write(wxT("paper_margin_left"), topleft
.x
);
404 cfg
->Write(wxT("paper_margin_right"), bottomright
.x
);
405 cfg
->Write(wxT("paper_margin_top"), topleft
.y
);
406 cfg
->Write(wxT("paper_margin_bottom"), bottomright
.y
);
413 Aven::MacOpenFiles(const wxArrayString
& filenames
)
415 if (filenames
.size() != 1) {
416 ReportError(wxT("Aven can only load one file at a time"));
419 m_Frame
->OpenFile(filenames
[0], wxString());
423 Aven::MacPrintFiles(const wxArrayString
& filenames
)
425 if (filenames
.size() != 1) {
426 ReportError(wxT("Aven can only print one file at a time"));
429 m_Frame
->OpenFile(filenames
[0], wxString());
430 m_Frame
->PrintAndExit();
434 void Aven::ReportError(const wxString
& msg
)
437 wxMessageBox(msg
, APP_NAME
, wxOK
| wxICON_ERROR
);
440 wxMessageDialog
dlg(m_Frame
, msg
, APP_NAME
, wxOK
| wxICON_ERROR
);
447 static wxString path
;
449 path
= wxString(msg_cfgpth(), wxConvUTF8
);
453 // called to report errors by message.c
455 aven_v_report(int severity
, const char *fnm
, int line
, int en
, va_list ap
)
459 m
= wxString(fnm
, wxConvUTF8
);
460 if (line
) m
+= wxString::Format(wxT(":%d"), line
);
465 m
+= wmsg(/*warning*/4);
470 vsnprintf(buf
, sizeof(buf
), msg(en
), ap
);
471 m
+= wxString(buf
, wxConvUTF8
);
472 if (wxTheApp
== NULL
) {
473 // We haven't initialised the Aven app object yet.
474 if (!wxInitialize()) {
479 wxMessageBox(m
, APP_NAME
, wxOK
| wxICON_ERROR
);
482 wxGetApp().ReportError(m
);