gds-render/main.c

349 lines
10 KiB
C
Raw Normal View History

/*
2018-05-15 22:54:10 +02:00
* GDSII-Converter
* Copyright (C) 2018 Mario Hüttel <mario.huettel@gmx.net>
*
* This file is part of GDSII-Converter.
*
* GDSII-Converter is free software: you can redistribute it and/or modify
2018-05-16 16:29:34 +02:00
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*
2018-05-15 22:54:10 +02:00
* GDSII-Converter is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with GDSII-Converter. If not, see <http://www.gnu.org/licenses/>.
*/
2019-03-30 19:51:56 +01:00
/**
* @file main.c
* @brief main.c
* @author Mario Hüttel <mario.huettel@gmx.net>
*/
2018-05-07 13:27:50 +02:00
#include <stdio.h>
#include <gtk/gtk.h>
2018-07-23 17:00:37 +02:00
#include <glib.h>
#include <gds-render/gds-render-gui.h>
#include <gds-render/command-line.h>
#include <gds-render/output-renderers/external-renderer.h>
#include <gds-render/version.h>
2018-05-08 16:50:30 +02:00
2019-03-30 19:51:56 +01:00
/**
* @brief Structure containing The GtkApplication and a list containing the GdsRenderGui objects.
*/
2018-07-23 15:10:40 +02:00
struct application_data {
GtkApplication *app;
GList *gui_list;
2018-07-23 15:10:40 +02:00
};
2019-03-30 19:51:56 +01:00
/**
* @brief Callback for the menu entry 'Quit'
*
* Destroys all GUIs contained in the application_data structure
* provided by \p user_data.
*
* The complete suspension of all main windows leads to the termination of the
* GApplication.
*
* @param action unused
* @param parameter unused
* @param user_data application_data structure
*/
2018-07-23 15:10:40 +02:00
static void app_quit(GSimpleAction *action, GVariant *parameter, gpointer user_data)
{
2019-03-15 20:36:23 +01:00
struct application_data * const appdata = (struct application_data *)user_data;
2018-12-15 00:03:40 +01:00
(void)action;
(void)parameter;
GList *list_iter;
GdsRenderGui *gui;
2018-12-11 00:06:27 +01:00
2019-03-15 23:16:39 +01:00
/* Dispose all GUIs */
for (list_iter = appdata->gui_list; list_iter != NULL; list_iter = g_list_next(list_iter)) {
gui = RENDERER_GUI(list_iter->data);
g_object_unref(gui);
}
g_list_free(appdata->gui_list);
2019-03-15 20:36:23 +01:00
appdata->gui_list = NULL;
2018-07-23 15:10:40 +02:00
}
2019-03-30 19:51:56 +01:00
/**
* @brief Callback for the 'About' menu entry
*
* This function shows the about dialog.
*
* @param action GSimpleAction, unused
* @param parameter Unused.
* @param user_data Unused
*/
2018-07-23 15:10:40 +02:00
static void app_about(GSimpleAction *action, GVariant *parameter, gpointer user_data)
{
GtkBuilder *builder;
GtkDialog *dialog;
GdkPixbuf *logo_buf;
GError *error = NULL;
2019-03-16 16:24:39 +01:00
(void)user_data;
2018-12-15 00:03:40 +01:00
(void)action;
(void)parameter;
2018-07-23 15:10:40 +02:00
builder = gtk_builder_new_from_resource("/gui/about.glade");
2018-07-23 15:10:40 +02:00
dialog = GTK_DIALOG(gtk_builder_get_object(builder, "about-dialog"));
gtk_window_set_transient_for(GTK_WINDOW(dialog), NULL);
2019-03-14 21:01:16 +01:00
gtk_about_dialog_set_version(GTK_ABOUT_DIALOG(dialog), _app_version_string);
/* Load icon from resource */
logo_buf = gdk_pixbuf_new_from_resource_at_scale("/images/logo.svg", 100, 100, TRUE, &error);
if (logo_buf) {
/* Set logo */
gtk_about_dialog_set_logo(GTK_ABOUT_DIALOG(dialog), logo_buf);
/* Pixbuf is now owned by about dialog. Unref */
g_object_unref(logo_buf);
} else if (error) {
fprintf(stderr, "Logo could not be displayed: %s\n", error->message);
g_error_free(error);
}
2018-07-23 15:10:40 +02:00
gtk_dialog_run(dialog);
2018-07-23 17:00:37 +02:00
gtk_widget_destroy(GTK_WIDGET(dialog));
2018-07-23 15:10:40 +02:00
g_object_unref(builder);
}
2019-03-30 19:51:56 +01:00
/**
* @brief Contains the application menu entries
*/
2018-09-22 23:32:22 +02:00
const static GActionEntry app_actions[] = {
{"quit", app_quit, NULL, NULL, NULL, {0}},
{"about", app_about, NULL, NULL, NULL, {0}}
2018-07-23 15:10:40 +02:00
};
2019-03-30 19:51:56 +01:00
/**
* @brief Called when a GUI main window is closed
*
* The GdsRenderGui object associated with the closed main window
2019-03-30 19:55:47 +01:00
* is removed from the list of open GUIs (\p user_data) and unreferenced.
2019-03-30 19:51:56 +01:00
*
* @param gui The GUI instance the closed main window belongs to
* @param user_data List of GUIs
*/
static void gui_window_closed_callback(GdsRenderGui *gui, gpointer user_data)
{
GList **gui_list = (GList **)user_data;
/* Dispose of Gui element */
*gui_list = g_list_remove(*gui_list, gui);
g_object_unref(gui);
}
2019-03-30 19:51:56 +01:00
/**
* @brief Activation of the GUI
* @param app The GApplication reference
* @param user_data Used to store the individual GUI instances.
*/
static void gapp_activate(GApplication *app, gpointer user_data)
{
GtkWindow *main_window;
GdsRenderGui *gui;
2019-03-07 20:14:44 +01:00
struct application_data * const appdata = (struct application_data *)user_data;
gui = gds_render_gui_new();
appdata->gui_list = g_list_append(appdata->gui_list, gui);
g_signal_connect(gui, "window-closed", G_CALLBACK(gui_window_closed_callback), &appdata->gui_list);
main_window = gds_render_gui_get_main_window(gui);
gtk_application_add_window(GTK_APPLICATION(app), main_window);
gtk_widget_show(GTK_WIDGET(main_window));
}
2018-05-08 16:02:44 +02:00
2019-03-30 19:51:56 +01:00
/**
* @brief Start the graphical interface.
*
* This function starts the GUI. If there's already a
* running instance of this program, a second window will be
* created in that instance and the second one is terminated.
*
* @param argc
* @param argv
* @return
*/
2018-07-23 17:00:37 +02:00
static int start_gui(int argc, char **argv)
{
2018-07-23 17:00:37 +02:00
GtkApplication *gapp;
int app_status;
static struct application_data appdata = {
.gui_list = NULL
};
2018-07-23 15:10:40 +02:00
GMenu *menu;
GMenu *m_quit;
GMenu *m_about;
gapp = gtk_application_new("de.shimatta.gds-render", G_APPLICATION_FLAGS_NONE);
2018-07-23 15:10:40 +02:00
g_application_register(G_APPLICATION(gapp), NULL, NULL);
2018-12-11 00:06:27 +01:00
g_signal_connect(gapp, "activate", G_CALLBACK(gapp_activate), &appdata);
2018-07-23 15:10:40 +02:00
2019-03-15 20:36:23 +01:00
if (g_application_get_is_remote(G_APPLICATION(gapp)) == TRUE) {
g_application_activate(G_APPLICATION(gapp));
printf("There is already an open instance. Will open second window in said instance.\n");
return 0;
}
2018-07-23 15:10:40 +02:00
menu = g_menu_new();
m_quit = g_menu_new();
m_about = g_menu_new();
g_menu_append(m_quit, "Quit", "app.quit");
g_menu_append(m_about, "About", "app.about");
g_menu_append_section(menu, NULL, G_MENU_MODEL(m_about));
g_menu_append_section(menu, NULL, G_MENU_MODEL(m_quit));
2018-12-11 00:06:27 +01:00
g_action_map_add_action_entries(G_ACTION_MAP(gapp), app_actions,
G_N_ELEMENTS(app_actions), &appdata);
2018-07-23 15:10:40 +02:00
gtk_application_set_app_menu(GTK_APPLICATION(gapp), G_MENU_MODEL(menu));
g_object_unref(m_quit);
g_object_unref(m_about);
g_object_unref(menu);
2018-12-11 00:06:27 +01:00
app_status = g_application_run(G_APPLICATION(gapp), argc, argv);
g_object_unref(gapp);
2018-05-07 13:27:50 +02:00
g_list_free(appdata.gui_list);
2018-07-23 17:00:37 +02:00
return app_status;
}
2019-03-30 19:51:56 +01:00
/**
* @brief Print the application version string to stdout
*/
static void print_version(void)
{
printf("This is gds-render, version: %s\n\nFor a list of supported commands execute with --help option.\n",
_app_version_string);
}
2019-03-30 19:51:56 +01:00
/**
* @brief The "entry point" of the application
* @param argc Number of command line parameters
* @param argv Command line parameters
* @return Execution status of the application
*/
2018-07-23 17:00:37 +02:00
int main(int argc, char **argv)
{
int i;
2018-07-23 17:00:37 +02:00
GError *error = NULL;
GOptionContext *context;
2018-07-23 21:12:25 +02:00
gchar *gds_name;
gchar *basename;
gchar *output_path = NULL, *mappingname = NULL, *cellname = NULL;
gchar *renderer_arg = NULL;
gboolean version = FALSE, pdf_standalone = FALSE, pdf_layers = FALSE;
gchar *custom_library_path = NULL;
2018-07-23 17:00:37 +02:00
int scale = 1000;
int app_status = 0;
enum command_line_renderer renderer = CMD_NONE;
enum cmd_options opt = CMD_OPT_NONE;
2018-07-23 17:00:37 +02:00
2018-12-11 00:06:27 +01:00
GOptionEntry entries[] = {
{"version", 'v', 0, G_OPTION_ARG_NONE, &version, "Print version", NULL},
{"renderer", 'r', 0, G_OPTION_ARG_STRING, &renderer_arg, "Renderer to use", "pdf|svg|tikz|ext"},
{"scale", 's', 0, G_OPTION_ARG_INT, &scale, "Divide output coordinates by <SCALE>", "<SCALE>" },
{"output-file", 'o', 0, G_OPTION_ARG_FILENAME, &output_path, "Output file path", "PATH" },
{"mapping", 'm', 0, G_OPTION_ARG_FILENAME, &mappingname, "Path for Layer Mapping File", "PATH" },
{"cell", 'c', 0, G_OPTION_ARG_STRING, &cellname, "Cell to render", "NAME" },
{"tex-standalone", 'a', 0, G_OPTION_ARG_NONE, &pdf_standalone, "Create standalone PDF", NULL },
{"tex-layers", 'l', 0, G_OPTION_ARG_NONE, &pdf_layers, "Create PDF Layers (OCG)", NULL },
{"custom-render-lib", 'P', 0, G_OPTION_ARG_FILENAME, &custom_library_path, "Path to a custom shared object, that implements the " EXTERNAL_LIBRARY_FUNCTION " function", "PATH"},
{NULL}
2018-07-23 17:00:37 +02:00
};
context = g_option_context_new(" FILE - Convert GDS file <FILE> to graphic");
g_option_context_add_main_entries(context, entries, NULL);
g_option_context_add_group(context, gtk_get_option_group(TRUE));
2018-12-11 00:06:27 +01:00
if (!g_option_context_parse(context, &argc, &argv, &error)) {
g_print("Option parsing failed: %s\n", error->message);
exit(1);
}
2018-07-23 17:00:37 +02:00
if (version) {
print_version();
goto ret_status;
}
2018-07-23 17:00:37 +02:00
if (argc >= 2) {
2018-07-23 21:12:25 +02:00
if (scale < 1) {
printf("Scale < 1 not allowed. Setting to 1\n");
scale = 1;
}
/* Get gds name */
gds_name = argv[1];
/* Print out additional arguments as ignored */
for (i = 2; i < argc; i++) {
printf("Ignored argument: %s", argv[i]);
}
2018-07-23 21:12:25 +02:00
/* Check if PDF/TeX names are supplied. if not generate */
basename = g_path_get_basename(gds_name);
if (!strcmp(renderer_arg, "pdf")) {
renderer = CMD_CAIRO_PDF;
if (!output_path)
output_path = g_strdup_printf("./%s.pdf", basename);
}
else if (!strcmp(renderer_arg, "svg")) {
renderer = CMD_NONE; // To buggy atm CMD_CAIRO_SVG;
if (!output_path)
output_path = g_strdup_printf("./%s.svg", basename);
} else if (!strcmp(renderer_arg, "tikz")) {
renderer = CMD_LATEX;
if (pdf_layers)
opt |= CMD_OPT_LATEX_LAYERS;
if (pdf_standalone)
opt |= CMD_OPT_LATEX_STANDALONE;
if (!output_path)
output_path = g_strdup_printf("./%s.tex", basename);
} else if (!strcmp(renderer_arg, "ext")) {
renderer = CMD_EXTERNAL;
} else {
fprintf(stderr, "No valid renderer specified\n");
}
2018-07-23 21:12:25 +02:00
if (basename)
g_free(basename);
2018-07-23 21:12:25 +02:00
if (!output_path || strlen(output_path) == 0) {
app_status = -2;
goto ret_free_renderer;
}
command_line_convert_gds(gds_name, cellname, output_path, mappingname, custom_library_path, renderer, opt, scale);
2018-07-23 21:12:25 +02:00
/* Clean up */
app_status = 0;
ret_free_renderer:
if (output_path)
g_free(output_path);
if (renderer_arg)
g_free(renderer_arg);
2018-07-23 21:12:25 +02:00
if (mappingname)
g_free(mappingname);
if (cellname)
g_free(cellname);
2018-07-23 17:00:37 +02:00
} else {
app_status = start_gui(argc, argv);
}
ret_status:
return app_status;
2018-05-07 13:27:50 +02:00
}