summaryrefslogtreecommitdiffstats
path: root/print.c
diff options
context:
space:
mode:
Diffstat (limited to 'print.c')
-rw-r--r--print.c322
1 files changed, 312 insertions, 10 deletions
diff --git a/print.c b/print.c
index bdebcfe15..0e49bc9c3 100644
--- a/print.c
+++ b/print.c
@@ -11,20 +11,41 @@
#define FONT_SMALL (FONT_NORMAL / 1.2)
#define FONT_LARGE (FONT_NORMAL * 1.2)
-static void set_font(PangoLayout *layout, PangoFontDescription *font, double size, int align)
+static struct options print_options;
+
+/* Return the 'i'th dive for printing, taking our dive selection into account */
+static struct dive *get_dive_for_printing(int idx)
+{
+ if (print_options.print_selected) {
+ int i;
+ struct dive *dive;
+ for_each_dive(i, dive) {
+ if (!dive->selected)
+ continue;
+ if (!idx)
+ return dive;
+ idx--;
+ }
+ return NULL;
+ }
+ return get_dive(idx);
+}
+
+static void set_font(PangoLayout *layout, PangoFontDescription *font,
+ double size, int align)
{
pango_font_description_set_size(font, size * PANGO_SCALE);
pango_layout_set_font_description(layout, font);
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_END);
pango_layout_set_alignment(layout, align);
-
}
/*
* You know what? Maybe somebody can do a real Pango layout thing.
* This is hacky.
*/
-static void show_dive_text(struct dive *dive, cairo_t *cr, double w, double h, PangoFontDescription *font)
+static void show_dive_text(struct dive *dive, cairo_t *cr, double w,
+ double h, PangoFontDescription *font)
{
double depth;
const char *unit;
@@ -125,7 +146,143 @@ static void show_dive_text(struct dive *dive, cairo_t *cr, double w, double h, P
g_object_unref(layout);
}
-static void show_dive_profile(struct dive *dive, cairo_t *cr, double w, double h)
+static void show_table_header(cairo_t *cr, double w, double h,
+ PangoFontDescription *font)
+{
+ int maxwidth, maxheight, colwidth, i, curwidth;
+ PangoLayout *layout;
+ char headers[7][80]= { "Dive#", "Date", "Depth", "Time", "Master",
+ "Buddy", "Location" };
+
+ maxwidth = w * PANGO_SCALE;
+ maxheight = h * PANGO_SCALE * 0.9;
+ colwidth = maxwidth / 7;
+
+ layout = pango_cairo_create_layout(cr);
+
+ cairo_move_to(cr, 0, 0);
+ pango_layout_set_height(layout, maxheight);
+ set_font(layout, font, FONT_LARGE, PANGO_ALIGN_LEFT);
+
+
+ curwidth = 0;
+ for (i = 0; i < 7; i++) {
+ cairo_move_to(cr, curwidth / PANGO_SCALE, 0);
+ if (i == 0 || i == 2 || i == 3 ){
+ // Column 0, 2 and 3 (Dive #, Depth and Time) get 1/2 width
+ pango_layout_set_width(layout, colwidth/ (double) 2);
+ curwidth = curwidth + (colwidth / 2);
+ } else {
+ pango_layout_set_width(layout, colwidth);
+ curwidth = curwidth + colwidth;
+ }
+ pango_layout_set_text(layout, headers[i], -1);
+ pango_layout_set_justify(layout, 1);
+ pango_cairo_show_layout(cr, layout);
+ }
+
+ cairo_move_to(cr, 0, 0);
+ g_object_unref(layout);
+}
+
+static void show_dive_table(struct dive *dive, cairo_t *cr, double w,
+ double h, PangoFontDescription *font)
+{
+ double depth;
+ const char *unit;
+ int len, decimals, maxwidth, maxheight, colwidth, curwidth;
+ PangoLayout *layout;
+ struct tm *tm;
+ char buffer[160], divenr[20];
+
+ maxwidth = w * PANGO_SCALE;
+ maxheight = h * PANGO_SCALE * 0.9;
+
+ colwidth = maxwidth / 7;
+
+ layout = pango_cairo_create_layout(cr);
+
+ cairo_move_to(cr, 0, 0);
+ pango_layout_set_width(layout, colwidth);
+ pango_layout_set_height(layout, maxheight);
+ set_font(layout, font, FONT_NORMAL, PANGO_ALIGN_LEFT);
+ cairo_move_to(cr, 0, 0);
+ curwidth = 0;
+
+ // Col 1: Dive #
+ *divenr = 0;
+ if (dive->number)
+ snprintf(divenr, sizeof(divenr), "#%d", dive->number);
+ pango_layout_set_width(layout, colwidth/ (double) 2);
+ pango_layout_set_text(layout, divenr, -1);
+ pango_layout_set_justify(layout, 1);
+ pango_cairo_show_layout(cr, layout);
+ curwidth = curwidth + (colwidth / 2);
+
+ // Col 2: Date #
+ pango_layout_set_width(layout, colwidth);
+ tm = gmtime(&dive->when);
+ len = snprintf(buffer, sizeof(buffer),
+ "%s, %s %d, %d %dh%02d",
+ weekday(tm->tm_wday),
+ monthname(tm->tm_mon),
+ tm->tm_mday, tm->tm_year + 1900,
+ tm->tm_hour, tm->tm_min
+ );
+ cairo_move_to(cr, curwidth / PANGO_SCALE, 0);
+ pango_layout_set_text(layout, buffer, len);
+ pango_layout_set_justify(layout, 1);
+ pango_cairo_show_layout(cr, layout);
+ curwidth = curwidth + colwidth;
+
+ // Col 3: Depth
+ depth = get_depth_units(dive->maxdepth.mm, &decimals, &unit);
+ len = snprintf(buffer, sizeof(buffer),
+ "%.*f %s", decimals, depth, unit);
+ cairo_move_to(cr, curwidth / PANGO_SCALE, 0);
+ pango_layout_set_width(layout, colwidth/ (double) 2);
+ pango_layout_set_text(layout, buffer, len);
+ pango_layout_set_justify(layout, 1);
+ pango_cairo_show_layout(cr, layout);
+ curwidth = curwidth + (colwidth / 2);
+
+ // Col 4: Time
+ len = snprintf(buffer, sizeof(buffer),
+ "%d min",(dive->duration.seconds+59) / 60);
+ cairo_move_to(cr, curwidth / PANGO_SCALE, 0);
+ pango_layout_set_width(layout, colwidth/ (double) 2);
+ pango_layout_set_text(layout, buffer, len);
+ pango_layout_set_justify(layout, 1);
+ pango_cairo_show_layout(cr, layout);
+ curwidth = curwidth + (colwidth / 2);
+
+ // Col 5: Master
+ pango_layout_set_width(layout, colwidth);
+ cairo_move_to(cr, curwidth / PANGO_SCALE, 0);
+ pango_layout_set_text(layout, dive->divemaster ? : " ", -1);
+ pango_layout_set_justify(layout, 1);
+ pango_cairo_show_layout(cr, layout);
+ curwidth = curwidth + colwidth;
+
+ // Col 6: Buddy
+ cairo_move_to(cr, curwidth / PANGO_SCALE, 0);
+ pango_layout_set_text(layout, dive->buddy ? : " ", -1);
+ pango_layout_set_justify(layout, 1);
+ pango_cairo_show_layout(cr, layout);
+ curwidth = curwidth + colwidth;
+
+ // Col 7: Location
+ cairo_move_to(cr, curwidth / PANGO_SCALE, 0);
+ pango_layout_set_width(layout, maxwidth - curwidth);
+ pango_layout_set_text(layout, dive->location ? : " ", -1);
+ pango_layout_set_justify(layout, 1);
+ pango_cairo_show_layout(cr, layout);
+
+ g_object_unref(layout);
+}
+
+static void show_dive_profile(struct dive *dive, cairo_t *cr, double w,
+ double h)
{
cairo_rectangle_int_t drawing_area = { w/20.0, h/20.0, w, h};
struct graphics_context gc = {
@@ -137,11 +294,12 @@ static void show_dive_profile(struct dive *dive, cairo_t *cr, double w, double h
cairo_restore(cr);
}
-static void print(int divenr, cairo_t *cr, double x, double y, double w, double h, PangoFontDescription *font)
+static void print(int divenr, cairo_t *cr, double x, double y, double w,
+ double h, PangoFontDescription *font)
{
struct dive *dive;
- dive = get_dive(divenr);
+ dive = get_dive_for_printing(divenr);
if (!dive)
return;
cairo_save(cr);
@@ -165,6 +323,47 @@ static void print(int divenr, cairo_t *cr, double x, double y, double w, double
cairo_restore(cr);
}
+static void print_table_header(cairo_t *cr, double x, double y,
+ double w, double h, PangoFontDescription *font)
+{
+ cairo_save(cr);
+ cairo_translate(cr, x, y);
+
+ /* Plus 5% on all sides */
+ cairo_translate(cr, w/20, h/20);
+ w *= 0.9; h *= 0.9;
+
+ /* We actually want to scale the text and the lines now */
+ cairo_scale(cr, 0.5, 0.5);
+
+ show_table_header(cr, w*2, h*2, font);
+
+ cairo_restore(cr);
+}
+
+static void print_table(int divenr, cairo_t *cr, double x, double y,
+ double w, double h, PangoFontDescription *font)
+{
+ struct dive *dive;
+
+ dive = get_dive_for_printing(divenr);
+ if (!dive)
+ return;
+ cairo_save(cr);
+ cairo_translate(cr, x, y);
+
+ /* Plus 5% on all sides */
+ cairo_translate(cr, w/20, h/20);
+ w *= 0.9; h *= 0.9;
+
+ /* We actually want to scale the text and the lines now */
+ cairo_scale(cr, 0.5, 0.5);
+
+ show_dive_table(dive, cr, w*2, h*2, font);
+
+ cairo_restore(cr);
+}
+
static void draw_page(GtkPrintOperation *operation,
GtkPrintContext *context,
gint page_nr,
@@ -192,26 +391,129 @@ static void draw_page(GtkPrintOperation *operation,
pango_font_description_free(font);
}
+static void draw_table(GtkPrintOperation *operation,
+ GtkPrintContext *context,
+ gint page_nr,
+ gpointer user_data)
+{
+ int i, nr;
+ int n_dive_per_page = 25;
+ cairo_t *cr;
+ double w, h;
+ PangoFontDescription *font;
+
+ cr = gtk_print_context_get_cairo_context(context);
+ font = pango_font_description_from_string("Sans");
+
+ w = gtk_print_context_get_width(context);
+ h = gtk_print_context_get_height(context)/(n_dive_per_page+1);
+
+ nr = page_nr*n_dive_per_page;
+ print_table_header(cr, 0, 0+h, w, h, font);
+ for (i = 0; i < n_dive_per_page; i++) {
+ print_table(nr+i, cr, 0, h*1.5+h*i, w, h, font);
+ }
+
+ pango_font_description_free(font);
+}
+
+static int nr_selected_dives(void)
+{
+ int i, dives;
+ struct dive *dive;
+
+ dives = 0;
+ for_each_dive(i, dive)
+ dives += dive->selected;
+ return dives;
+}
+
static void begin_print(GtkPrintOperation *operation, gpointer user_data)
{
+ int pages, dives;
+ int dives_per_page;
+
+ dives = nr_selected_dives();
+ print_options.print_selected = dives > 1;
+ if (dives <= 1)
+ dives = dive_table.nr;
+
+ if (print_options.type == PRETTY) {
+ dives_per_page = 6;
+ } else {
+ dives_per_page = 25;
+ }
+ pages = (dives + dives_per_page - 1) / dives_per_page;
+ gtk_print_operation_set_n_pages(operation, pages);
+}
+
+
+#define OPTIONCALLBACK(name, type, value) \
+static void name(GtkWidget *w, gpointer data) \
+{\
+ if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(w))) \
+ print_options.type = value; \
+}
+
+OPTIONCALLBACK(set_pretty, type, PRETTY)
+OPTIONCALLBACK(set_table, type, TABLE)
+
+static GtkWidget *print_dialog(GtkPrintOperation *operation, gpointer user_data)
+{
+ GtkWidget *vbox, *radio1, *radio2, *frame, *box;
+ gtk_print_operation_set_custom_tab_label(operation, "Dive details");
+
+ vbox = gtk_vbox_new(TRUE, 5);
+
+ frame = gtk_frame_new("Print type");
+ gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 1);
+
+ box = gtk_hbox_new(FALSE, 2);
+ gtk_container_add(GTK_CONTAINER(frame), box);
+
+ radio1 = gtk_radio_button_new_with_label (NULL, "Pretty print");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(radio1),
+ print_options.type == PRETTY);
+ radio2 = gtk_radio_button_new_with_label_from_widget (
+ GTK_RADIO_BUTTON (radio1), "Table print");
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(radio2),
+ print_options.type == TABLE);
+ gtk_box_pack_start (GTK_BOX (box), radio1, TRUE, TRUE, 0);
+ gtk_box_pack_start (GTK_BOX (box), radio2, TRUE, TRUE, 0);
+
+ g_signal_connect(radio1, "toggled", G_CALLBACK(set_pretty), NULL);
+ g_signal_connect(radio2, "toggled", G_CALLBACK(set_table), NULL);
+
+ gtk_widget_show_all(vbox);
+ return vbox;
+}
+
+static void print_dialog_apply(GtkPrintOperation *operation, GtkWidget *widget, gpointer user_data)
+{
+ if (print_options.type == PRETTY) {
+ g_signal_connect(operation, "draw_page",
+ G_CALLBACK(draw_page), NULL);
+ } else {
+ g_signal_connect(operation, "draw_page",
+ G_CALLBACK(draw_table), NULL);
+ }
}
static GtkPrintSettings *settings = NULL;
void do_print(void)
{
- int pages;
GtkPrintOperation *print;
GtkPrintOperationResult res;
repaint_dive();
print = gtk_print_operation_new();
+ gtk_print_operation_set_unit(print, GTK_UNIT_POINTS);
if (settings != NULL)
gtk_print_operation_set_print_settings(print, settings);
- pages = (dive_table.nr + 5) / 6;
- gtk_print_operation_set_n_pages(print, pages);
+ g_signal_connect(print, "create-custom-widget", G_CALLBACK(print_dialog), NULL);
+ g_signal_connect(print, "custom-widget-apply", G_CALLBACK(print_dialog_apply), NULL);
g_signal_connect(print, "begin_print", G_CALLBACK(begin_print), NULL);
- g_signal_connect(print, "draw_page", G_CALLBACK(draw_page), NULL);
res = gtk_print_operation_run(print, GTK_PRINT_OPERATION_ACTION_PRINT_DIALOG,
GTK_WINDOW(main_window), NULL);
if (res == GTK_PRINT_OPERATION_RESULT_APPLY) {