r5417 - trunk/eda/fped
werner at docs.openmoko.org
werner at docs.openmoko.org
Tue Aug 11 03:20:15 CEST 2009
Author: werner
Date: 2009-08-11 03:20:15 +0200 (Tue, 11 Aug 2009)
New Revision: 5417
Added:
trunk/eda/fped/postscript.c
trunk/eda/fped/postscript.h
Modified:
trunk/eda/fped/Makefile
trunk/eda/fped/gui.c
Log:
- change file save logic to write to a temporary file first, so that we don't
leave an empty/corrupt main file if we crash during saving
- started adding Postscript output
- renamed "Save as KiCad" to "Write KiCad"
- switched from "manual" menu bar creation to GtkItemFactory
Modified: trunk/eda/fped/Makefile
===================================================================
--- trunk/eda/fped/Makefile 2009-08-10 21:02:26 UTC (rev 5416)
+++ trunk/eda/fped/Makefile 2009-08-11 01:20:15 UTC (rev 5417)
@@ -11,7 +11,7 @@
#
OBJS = fped.o expr.o coord.o obj.o delete.o inst.o util.o error.o \
- unparse.o dump.o kicad.o meas.o \
+ unparse.o dump.o kicad.o postscript.o meas.o \
cpp.o lex.yy.o y.tab.o \
gui.o gui_util.o gui_style.o gui_inst.o gui_status.o gui_canvas.o \
gui_tool.o gui_over.o gui_meas.o gui_frame.o
Modified: trunk/eda/fped/gui.c
===================================================================
--- trunk/eda/fped/gui.c 2009-08-10 21:02:26 UTC (rev 5416)
+++ trunk/eda/fped/gui.c 2009-08-11 01:20:15 UTC (rev 5417)
@@ -22,6 +22,7 @@
#include "obj.h"
#include "dump.h"
#include "kicad.h"
+#include "postscript.h"
#include "gui_util.h"
#include "gui_style.h"
#include "gui_status.h"
@@ -48,7 +49,7 @@
static GtkWidget *stuff_image[2], *meas_image[2];
-/* ----- menu bar ---------------------------------------------------------- */
+/* ----- save/write operations --------------------------------------------- */
static char *set_extension(const char *name, const char *ext)
@@ -67,57 +68,93 @@
}
+static int save_to(const char *name, int (*fn)(FILE *file))
+{
+ FILE *file;
+
+ file = fopen(name, "w");
+ if (!file) {
+ perror(name);
+ return 0;
+ }
+ if (!fn(file)) {
+ perror(name);
+ return 0;
+ }
+ if (fclose(file) == EOF) {
+ perror(name);
+ return 0;
+ }
+ return 1;
+}
+
+
static void save_with_backup(const char *name, int (*fn)(FILE *file))
{
- FILE *file;
char *s = stralloc(name);
- char *slash, *dot, *tmp;
+ char *back, *tmp;
+ char *slash, *dot;
int n;
struct stat st;
+ /* save to temporary file */
+
slash = strrchr(s, '/');
+ if (!slash)
+ tmp = stralloc_printf("~%s", s);
+ else {
+ *slash = 0;
+ tmp = stralloc_printf("%s/~%s", s, slash+1);
+ *slash = '/';
+ }
+
+ if (!save_to(tmp, fn))
+ return;
+
+ /* move existing file out of harm's way */
+
dot = strrchr(slash ? slash : s, '.');
if (dot)
*dot = 0;
n = 0;
while (1) {
- tmp = stralloc_printf("%s~%d%s%s",
+ back = stralloc_printf("%s~%d%s%s",
s, n, dot ? "." : "", dot ? dot+1 : "");
- if (stat(tmp, &st) < 0) {
+ if (stat(back, &st) < 0) {
if (errno == ENOENT)
break;
- perror(tmp);
- free(tmp);
+ perror(back);
+ free(back);
return;
}
- free(tmp);
+ free(back);
n++;
}
- if (rename(name, tmp) < 0) {
+ if (rename(name, back) < 0) {
if (errno != ENOENT) {
perror(name);
- free(tmp);
+ free(back);
return;
}
} else {
- fprintf(stderr, "renamed %s to %s\n", name, tmp);
+ fprintf(stderr, "renamed %s to %s\n", name, back);
}
- free(tmp);
+ free(back);
- file = fopen(name, "w");
- if (!file) {
+ /* rename to final name */
+
+ if (rename(tmp, name) < 0) {
perror(name);
+ free(tmp);
return;
}
- if (!fn(file))
- perror(name);
- if (fclose(file) == EOF)
- perror(name);
+ free(tmp);
+
fprintf(stderr, "saved to %s\n", name);
}
-static void menu_save(GtkWidget *widget, gpointer user)
+static void menu_save(void)
{
if (save_file)
save_with_backup(save_file, dump);
@@ -128,13 +165,13 @@
}
-static void menu_save_kicad(GtkWidget *widget, gpointer user)
+static void menu_write_kicad(void)
{
char *name;
if (save_file) {
name = set_extension(save_file, "mod");
- save_with_backup(name, kicad);
+ save_to(name, kicad);
free(name);
} else {
if (!kicad(stdout))
@@ -143,34 +180,47 @@
}
-static void make_menu_bar(GtkWidget *hbox)
+static void menu_write_ps(void)
{
- GtkWidget *bar;
- GtkWidget *file_menu, *file, *quit, *save;
+ char *name;
- bar = gtk_menu_bar_new();
- gtk_box_pack_start(GTK_BOX(hbox), bar, TRUE, TRUE, 0);
+ if (save_file) {
+ name = set_extension(save_file, "ps");
+ save_to(name, postscript);
+ free(name);
+ } else {
+ if (!postscript(stdout))
+ perror("stdout");
+ }
+}
- file_menu = gtk_menu_new();
- file = gtk_menu_item_new_with_label("File");
- gtk_menu_item_set_submenu(GTK_MENU_ITEM(file), file_menu);
- gtk_menu_shell_append(GTK_MENU_SHELL(bar), file);
+/* ----- menu bar ---------------------------------------------------------- */
- save = gtk_menu_item_new_with_label("Save");
- gtk_menu_shell_append(GTK_MENU_SHELL(file_menu), save);
- g_signal_connect(G_OBJECT(save), "activate",
- G_CALLBACK(menu_save), NULL);
- save = gtk_menu_item_new_with_label("Save as KiCad");
- gtk_menu_shell_append(GTK_MENU_SHELL(file_menu), save);
- g_signal_connect(G_OBJECT(save), "activate",
- G_CALLBACK(menu_save_kicad), NULL);
+static GtkItemFactoryEntry menu_entries[] = {
+ { "/File", NULL, NULL, 0, "<Branch>" },
+ { "/File/Save", NULL, menu_save, 0, "<Item>" },
+ { "/File/sep0", NULL, NULL, 0, "<Separator>" },
+ { "/File/Write KiCad", NULL, menu_write_kicad, 0, "<Item>" },
+ { "/File/Write Postscript",
+ NULL, menu_write_ps, 0, "<Item>" },
+ { "/File/sep2", NULL, NULL, 0, "<Separator>" },
+ { "/File/Quit", NULL, gtk_main_quit, 0, "<Item>" },
+};
- quit = gtk_menu_item_new_with_label("Quit");
- gtk_menu_shell_append(GTK_MENU_SHELL(file_menu), quit);
- g_signal_connect(G_OBJECT(quit), "activate",
- G_CALLBACK(gtk_main_quit), NULL);
+
+static void make_menu_bar(GtkWidget *hbox)
+{
+ GtkItemFactory *factory;
+ GtkWidget *bar;
+
+ factory = gtk_item_factory_new(GTK_TYPE_MENU_BAR, "<FpedMenu>", NULL);
+ gtk_item_factory_create_items(factory,
+ sizeof(menu_entries)/sizeof(*menu_entries), menu_entries, NULL);
+
+ bar = gtk_item_factory_get_widget(factory, "<FpedMenu>");
+ gtk_box_pack_start(GTK_BOX(hbox), bar, TRUE, TRUE, 0);
}
Added: trunk/eda/fped/postscript.c
===================================================================
--- trunk/eda/fped/postscript.c (rev 0)
+++ trunk/eda/fped/postscript.c 2009-08-11 01:20:15 UTC (rev 5417)
@@ -0,0 +1,201 @@
+/*
+ * postscript.c - Dump objects in Postscript
+ *
+ * Written 2009 by Werner Almesberger
+ * Copyright 2009 by Werner Almesberger
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ */
+
+
+#include <stdlib.h>
+#include <stdio.h>
+
+#include "coord.h"
+#include "inst.h"
+#include "postscript.h"
+
+
+#define DOT_DIST mm_to_units(0.03)
+#define DOT_DIAM mm_to_units(0.01)
+#define HATCH mm_to_units(0.1)
+#define HATCH_LINE mm_to_units(0.02)
+
+
+struct postscript_params postscript_params = {
+ .zoom = 10.0,
+ .show_pad_names = 0,
+ .show_stuff = 0,
+ .label_vecs = 0,
+ .show_meas = 0,
+};
+
+
+static void ps_pad(FILE *file, const struct inst *inst)
+{
+ struct coord a = inst->base;
+ struct coord b = inst->u.pad.other;
+
+ fprintf(file, "0 setgray %d setlinewidth\n", HATCH_LINE);
+ fprintf(file, " %d %d moveto\n", a.x, a.y);
+ fprintf(file, " %d %d lineto\n", b.x, a.y);
+ fprintf(file, " %d %d lineto\n", b.x, b.y);
+ fprintf(file, " %d %d lineto\n", a.x, b.y);
+ fprintf(file, " closepath gsave hatchpath grestore stroke\n");
+}
+
+
+static void ps_line(FILE *file, const struct inst *inst)
+{
+ struct coord a = inst->base;
+ struct coord b = inst->u.pad.other;
+
+ fprintf(file, "1 setlinecap 0.5 setgray %d setlinewidth\n",
+ inst->u.rect.width);
+ fprintf(file, " %d %d moveto %d %d lineto stroke\n",
+ a.x, a.y, b.x, b.y);
+}
+
+
+static void ps_rect(FILE *file, const struct inst *inst)
+{
+ struct coord a = inst->base;
+ struct coord b = inst->u.rect.end;
+
+ fprintf(file, "1 setlinecap 0.5 setgray %d setlinewidth\n",
+ inst->u.rect.width);
+ fprintf(file, " %d %d moveto\n", a.x, a.y);
+ fprintf(file, " %d %d lineto\n", b.x, a.y);
+ fprintf(file, " %d %d lineto\n", b.x, b.y);
+ fprintf(file, " %d %d lineto\n", a.x, b.y);
+ fprintf(file, " closepath stroke\n");
+}
+
+
+static void ps_arc(FILE *file, const struct inst *inst)
+{
+ double a1, a2;
+
+ a1 = inst->u.arc.a1;
+ a2 = inst->u.arc.a2;
+ if (a2 <= a1)
+ a2 += 360;
+
+ fprintf(file, "1 setlinecap 0.5 setgray %d setlinewidth\n",
+ inst->u.arc.width);
+ fprintf(file, " newpath %d %d %d %f %f arc stroke\n",
+ inst->base.x, inst->base.y, inst->u.arc.r, a1, a2);
+}
+
+
+static void ps_frame(FILE *file, const struct inst *inst)
+{
+}
+
+
+static void ps_vec(FILE *file, const struct inst *inst)
+{
+}
+
+
+static void ps_meas(FILE *file, const struct inst *inst)
+{
+}
+
+
+static void ps_background(FILE *file, enum inst_prio prio,
+ const struct inst *inst)
+{
+ switch (prio) {
+ case ip_line:
+ ps_line(file, inst);
+ break;
+ case ip_rect:
+ ps_rect(file, inst);
+ break;
+ case ip_circ:
+ case ip_arc:
+ ps_arc(file, inst);
+ break;
+ default:
+ break;
+ }
+}
+
+
+static void ps_foreground(FILE *file, enum inst_prio prio,
+ const struct inst *inst)
+{
+ switch (prio) {
+ case ip_pad:
+ ps_pad(file, inst);
+ break;
+ case ip_vec:
+ if (postscript_params.show_stuff)
+ ps_vec(file, inst);
+ break;
+ case ip_frame:
+ if (postscript_params.show_stuff)
+ ps_frame(file, inst);
+ break;
+ case ip_meas:
+ if (postscript_params.show_meas)
+ ps_meas(file, inst);
+ break;
+ default:
+ break;
+ }
+}
+
+
+int postscript(FILE *file)
+{
+ enum inst_prio prio;
+ const struct inst *inst;
+
+ fprintf(file, "%%!PS\n");
+
+ fprintf(file,
+"currentpagedevice /PageSize get\n"
+" aload pop\n"
+" 2 div exch 2 div exch\n"
+" translate\n"
+" %f 72 mul %d div 1000 div dup scale\n",
+ (double) postscript_params.zoom , (int) MIL_UNITS);
+
+ fprintf(file,
+"/dotpath {\n"
+" gsave pathbbox clip newpath\n"
+" 1 setlinecap %d setlinewidth\n"
+" /ury exch def /urx exch def /lly exch def /llx exch def\n"
+" llx %d urx {\n"
+" lly %d ury {\n"
+" 1 index exch moveto 0 0 rlineto stroke\n"
+" } for\n"
+" } for\n"
+" grestore newpath } def\n", DOT_DIAM, DOT_DIST, DOT_DIST);
+
+ fprintf(file,
+"/hatchpath {\n"
+" gsave pathbbox clip newpath\n"
+" /ury exch def /urx exch def /lly exch def /llx exch def\n"
+" lly ury sub %d urx llx sub {\n" /* for -(ury-lly) to urx-llx */
+" llx add dup lly moveto\n"
+" ury lly sub add ury lineto stroke\n"
+" } for\n"
+" grestore newpath } def\n", HATCH);
+
+ FOR_INSTS_UP(prio, inst)
+ ps_background(file, prio, inst);
+ FOR_INSTS_UP(prio, inst)
+ ps_foreground(file, prio, inst);
+
+ fprintf(file, "showpage\n");
+ fprintf(file, "%%%%EOF\n");
+
+ fflush(file);
+ return !ferror(file);
+}
Added: trunk/eda/fped/postscript.h
===================================================================
--- trunk/eda/fped/postscript.h (rev 0)
+++ trunk/eda/fped/postscript.h 2009-08-11 01:20:15 UTC (rev 5417)
@@ -0,0 +1,31 @@
+/*
+ * ps.h - Dump objects in Postscript
+ *
+ * Written 2009 by Werner Almesberger
+ * Copyright 2009 by Werner Almesberger
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ */
+
+
+#ifndef POSTSCRIPT_H
+#define POSTSCRIPT_H
+
+#include <stdio.h>
+
+
+struct postscript_params {
+ double zoom;
+ int show_pad_names;
+ int show_stuff; /* vecs and frames */
+ int label_vecs;
+ int show_meas;
+};
+
+
+int postscript(FILE *file);
+
+#endif /* !POSTSCRIPT_H */
More information about the commitlog
mailing list