subsurface/print.c
Linus Torvalds 74988c86de Fix up printing some more
This makes things slightly prettier and adds back the depth and duration
details to the printout.

Still a few known problems: font choice, and the depth/duration thing
can end up overlapping with a long location name.  But it looks pretty
good on the whole.

Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2011-09-21 08:24:12 -07:00

233 lines
5.3 KiB
C

#include <stdio.h>
#include <string.h>
#include <stdarg.h>
#include <gtk/gtk.h>
#include "dive.h"
#include "display.h"
#include "display-gtk.h"
/* Why doesn't pango/gtk have these quoting functions? */
static inline int add_char(char *buffer, size_t size, int len, char c)
{
if (len < size)
buffer[len++] = c;
return len;
}
/* Add an escape string "atomically" - all or nothing */
static int add_str(char *buffer, size_t size, int len, const char *s)
{
int oldlen = len;
char c;
while ((c = *s++) != 0) {
if (len >= size)
return oldlen;
buffer[len++] = c;
}
return len;
}
static int add_quoted_string(char *buffer, size_t size, int len, const char *s)
{
if (!s)
return len;
/* Room for '\0' */
size--;
for (;;) {
const char *escape;
unsigned char c = *s++;
switch(c) {
default:
len = add_char(buffer, size, len, c);
continue;
case 0:
escape = "\n";
break;
case '&':
escape = "&amp;";
break;
case '>':
escape = "&gt;";
break;
case '<':
escape = "&lt;";
break;
}
len = add_str(buffer, size, len, escape);
if (c)
continue;
buffer[len] = 0;
return len;
}
}
/*
* 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)
{
int len;
PangoLayout *layout;
struct tm *tm;
char buffer[1024], divenr[20];
layout = pango_cairo_create_layout(cr);
pango_layout_set_width(layout, w * PANGO_SCALE);
pango_layout_set_height(layout, h * PANGO_SCALE * 0.9);
pango_layout_set_wrap(layout, PANGO_WRAP_WORD_CHAR);
*divenr = 0;
if (dive->number)
snprintf(divenr, sizeof(divenr), "Dive #%d - ", dive->number);
tm = gmtime(&dive->when);
len = snprintf(buffer, sizeof(buffer),
"<span size=\"large\">"
"%s%s, %s %d, %d %d:%02d"
"</span>\n",
divenr,
weekday(tm->tm_wday),
monthname(tm->tm_mon),
tm->tm_mday, tm->tm_year + 1900,
tm->tm_hour, tm->tm_min);
/*
* Leave an empty line even if no location: otherwise the notes can
* overrun the depth/duration information.
*/
if (dive->location)
len = add_quoted_string(buffer, sizeof(buffer), len, dive->location);
else
len = add_char(buffer, sizeof(buffer), len, '\n');
if (dive->notes) {
len = add_char(buffer, sizeof(buffer), len, '\n');
len = add_quoted_string(buffer, sizeof(buffer), len, dive->notes);
}
pango_layout_set_justify(layout, 1);
pango_layout_set_ellipsize(layout, PANGO_ELLIPSIZE_END);
pango_layout_set_alignment(layout, PANGO_ALIGN_LEFT);
pango_layout_set_markup(layout, buffer, len);
cairo_move_to(cr, 0, 0);
pango_cairo_show_layout(cr, layout);
/*
* This is still problematic: a long dive location will clash
* with the depth/duration information. Need to mask that or
* create a box or something.
*/
snprintf(buffer, sizeof(buffer),
"<span size=\"small\">"
"Max depth: %d ft\n"
"Duration: %d:%02d"
"</span>",
to_feet(dive->maxdepth),
dive->duration.seconds / 60,
dive->duration.seconds % 60);
pango_layout_set_alignment(layout, PANGO_ALIGN_RIGHT);
pango_layout_set_markup(layout, buffer, -1);
cairo_move_to(cr, 0, 0);
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)
{
struct graphics_context gc = {
.printer = 1,
.cr = cr
};
cairo_save(cr);
plot(&gc, w, h, dive);
cairo_restore(cr);
}
static void print(int divenr, cairo_t *cr, double x, double y, double w, double h)
{
struct dive *dive;
dive = get_dive(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);
/* Dive plot in the upper two thirds - note the scaling */
show_dive_profile(dive, cr, w*2, h*1.33);
/* Dive information in the lower third */
cairo_translate(cr, 0, h*1.33);
show_dive_text(dive, cr, w*2, h*0.67);
cairo_restore(cr);
}
static void draw_page(GtkPrintOperation *operation,
GtkPrintContext *context,
gint page_nr,
gpointer user_data)
{
int nr;
cairo_t *cr;
double w, h;
cr = gtk_print_context_get_cairo_context(context);
w = gtk_print_context_get_width(context)/2;
h = gtk_print_context_get_height(context)/3;
nr = page_nr*6;
print(nr+0, cr, 0, 0, w, h);
print(nr+1, cr, w, 0, w, h);
print(nr+2, cr, 0, h, w, h);
print(nr+3, cr, w, h, w, h);
print(nr+4, cr, 0, 2*h, w, h);
print(nr+5, cr, w, 2*h, w, h);
}
static void begin_print(GtkPrintOperation *operation, gpointer user_data)
{
}
static GtkPrintSettings *settings = NULL;
void do_print(void)
{
int pages;
GtkPrintOperation *print;
GtkPrintOperationResult res;
repaint_dive();
print = gtk_print_operation_new();
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, "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) {
if (settings != NULL)
g_object_unref(settings);
settings = g_object_ref(gtk_print_operation_get_print_settings(print));
}
g_object_unref(print);
}