aboutsummaryrefslogtreecommitdiff
path: root/src/hyper/event.pamphlet
diff options
context:
space:
mode:
Diffstat (limited to 'src/hyper/event.pamphlet')
-rw-r--r--src/hyper/event.pamphlet1099
1 files changed, 1099 insertions, 0 deletions
diff --git a/src/hyper/event.pamphlet b/src/hyper/event.pamphlet
new file mode 100644
index 00000000..3dcb69ff
--- /dev/null
+++ b/src/hyper/event.pamphlet
@@ -0,0 +1,1099 @@
+\documentclass{article}
+\usepackage{axiom}
+\begin{document}
+\title{\$SPAD/src/event}
+\author{The Axiom Team}
+\maketitle
+\begin{abstract}
+\end{abstract}
+\eject
+\tableofcontents
+\eject
+\section{event.h}
+<<event.h>>=
+<<license>>
+#ifndef _EVENT_H_
+#define _EVENT_H_ 1
+
+#include "hyper.h"
+
+extern Window gActiveWindow;
+extern int gNeedIconName;
+
+#endif
+@
+\section{event.c}
+<<event.c>>=
+#define _EVENT_C
+#include "axiom-c-macros.h"
+#include "useproto.h"
+#include "debug.h"
+
+
+#include "hyper.h"
+
+#include <X11/X.h>
+#include <X11/Xatom.h>
+#include <X11/cursorfont.h>
+#include <sys/signal.h>
+#include <setjmp.h>
+#include <sys/types.h>
+#include <sys/time.h>
+
+#ifdef SGIplatform
+#include <bstring.h>
+#endif
+
+#include "event.h"
+#include "keyin.h"
+#include "mem.h"
+#include "display.h"
+#include "parse.h"
+#include "parse-paste.h"
+#include "initx.h"
+#include "scrollbar.h"
+#include "group.h"
+
+#include "all_hyper_proto.H1"
+#include "sockio-c.H1"
+
+jmp_buf env;
+Window gActiveWindow;
+int motion = 0;
+int gNeedIconName = 0;
+unsigned long bigmask= 0xffffffff;
+static HyperLink *gSavedInputAreaLink = NULL;
+
+
+
+/*
+ * This is the main X loop. It keeps grabbing events. Since the only way the
+ * window can die is through an event, it never actually end. One of the
+ * subroutines it calls is responsible for killing everything
+ */
+
+void
+mainEventLoop(void)
+{
+ XEvent event;
+ int Xcon;
+ fd_set rd, dum1, dum2;
+ motion = 0;
+ gActiveWindow = -1;
+ set_error_handlers();
+ Xcon = ConnectionNumber(gXDisplay);
+
+ while (1) {
+/* fprintf(stderr,"event:mainEventLoop: loop top\n");*/
+ while (gSessionHashTable.num_entries == 0)
+ pause();
+
+ /* XFlush(gXDisplay); */
+
+ if (!motion)
+ init_cursor_states();
+ motion = 0;
+
+ if (!spad_socket == 0) {
+ FD_ZERO(&rd);
+ FD_ZERO(&dum1);
+ FD_ZERO(&dum2);
+ FD_CLR(0, &dum1);
+ FD_CLR(0, &dum2);
+ FD_CLR(0, &rd);
+ FD_SET(spad_socket->socket, &rd);
+ FD_SET(Xcon, &rd);
+ if (!session_server == 0) {
+ FD_SET(session_server->socket, &rd);
+ }
+ if (XEventsQueued(gXDisplay, QueuedAlready)) {
+ XNextEvent(gXDisplay, &event);
+ handle_event(&event);
+ }
+ else {
+ select(FD_SETSIZE,(void *)&rd,(void *)&dum1,(void *)&dum2,NULL);
+ if (FD_ISSET(Xcon, &rd) ||
+ XEventsQueued(gXDisplay, QueuedAfterFlush)) {
+ XNextEvent(gXDisplay, &event);
+ handle_event(&event);
+ }
+ else if FD_ISSET
+ (spad_socket->socket, &rd)
+ /*
+ * Axiom Socket do what handle_event does The 100 is
+ * $SpadStuff in hypertex.boot
+ */
+ {
+ if (100 == get_int(spad_socket)) {
+ set_window(gParentWindow->fMainWindow);
+ make_busy_cursors();
+ get_new_window();
+ }
+ }
+ /*
+ * Session Socket Telling us about the death of a spadbuf
+ * (plus maybe more later) service_session_socket in
+ * spadint.c
+ */
+ else
+ if (session_server && FD_ISSET(session_server->socket, &rd)) {
+ service_session_socket();
+ }
+ }
+ }
+ else {
+ XNextEvent(gXDisplay, &event);
+ handle_event(&event);
+ }
+ }
+}
+
+static void
+handle_event(XEvent * event)
+{
+ XWindowAttributes wa;
+/* fprintf(stderr,"event:handle_event entered\n");*/
+ set_window(event->xany.window);
+ if (event->type == MotionNotify) {
+/* fprintf(stderr,"event:handle_event type=MotionNotify\n");*/
+ handle_motion_event((XMotionEvent *)event);
+ motion = 1;
+ return;
+ }
+ make_busy_cursors();
+ switch (event->type) {
+ case DestroyNotify:
+/* fprintf(stderr,"event:handle_event type=DestroyNotify\n");*/
+ break;
+ case Expose:
+/* fprintf(stderr,"event:handle_event type=Expose\n");*/
+ XGetWindowAttributes(gXDisplay, gWindow->fMainWindow, &wa);
+ if ((gWindow->width == 0 && gWindow->height == 0) ||
+ (wa.width != gWindow->width || wa.height != gWindow->height)) {
+ gWindow->width = wa.width;
+ gWindow->height = wa.height;
+ display_page(gWindow->page);
+ gWindow->fWindowHashTable = gWindow->page->fLinkHashTable;
+ }
+ else /** just redraw the thing **/
+ expose_page(gWindow->page);
+ XFlush(gXDisplay);
+ clear_exposures(gWindow->fMainWindow);
+ clear_exposures(gWindow->fScrollWindow);
+ break;
+ case ButtonPress:
+/* fprintf(stderr,"event:handle_event type=ButtonPress\n");*/
+ handle_button(event->xbutton.button, (XButtonEvent *)event);
+ XFlush(gXDisplay);
+ if (gWindow) {
+ while (XCheckTypedWindowEvent(gXDisplay, gWindow->fMainWindow,
+ Expose, event));
+ while (XCheckTypedWindowEvent(gXDisplay, gWindow->fScrollWindow,
+ Expose, event));
+ }
+ break;
+ case KeyPress:
+/* fprintf(stderr,"event:handle_event type=KeyPress\n");*/
+ handle_key(event);
+ if (gWindow) {
+ while (XCheckTypedWindowEvent(gXDisplay, gWindow->fMainWindow,
+ Expose, event));
+ while (XCheckTypedWindowEvent(gXDisplay, gWindow->fScrollWindow,
+ Expose, event));
+ }
+ break;
+ case MapNotify:
+/* fprintf(stderr,"event:handle_event type=MapNotify\n");*/
+ create_window();
+ break;
+
+ case SelectionNotify:
+/* fprintf(stderr,"event:handle_event type=SelectionNotify\n");*/
+ /* this is in response to a previous request in an input area */
+ if ( gSavedInputAreaLink ) {
+ XSelectionEvent *pSelEvent;
+ Atom dataProperty;
+ pSelEvent = (XSelectionEvent *) event;
+ dataProperty = XInternAtom(gXDisplay, "PASTE_SELECTION", False);
+ /* change the input focus */
+
+ /* change_input_focus(gSavedInputAreaLink); */
+
+ /* try to get the selection as a window property */
+
+ if ( pSelEvent->requestor == gWindow->fMainWindow &&
+ pSelEvent->selection == XA_PRIMARY &&
+ /* pSelEvent->time == CurrentTime && */
+ pSelEvent->target == XA_STRING &&
+ pSelEvent->property == dataProperty )
+ {
+ Atom actual_type;
+ int actual_format;
+ unsigned long nitems, leftover;
+ char *pSelection = NULL;
+
+ if (Success == XGetWindowProperty(gXDisplay,
+ gWindow->fMainWindow,
+ pSelEvent->property, 0L, 100000000L, True,
+ AnyPropertyType, &actual_type, &actual_format,
+ &nitems, &leftover, (unsigned char **) &pSelection) )
+ {
+ char *pBuffer;
+ InputItem *item = gSavedInputAreaLink->reference.string;
+
+ for (pBuffer = pSelection; *pBuffer; ++pBuffer)
+ add_buffer_to_sym(pBuffer, item);
+
+ XFree(pSelection);
+ }
+ }
+
+ /* clear the link info */
+
+ gSavedInputAreaLink = NULL;
+ }
+ break;
+
+ default:
+/* fprintf(stderr,"event:handle_event type=default\n");*/
+ break;
+ }
+
+}
+
+static void
+create_window(void)
+{
+ XWindowAttributes wa;
+
+ XGetWindowAttributes(gXDisplay, gWindow->fMainWindow, &wa);
+
+ gWindow->width = wa.width;
+ gWindow->height = wa.height;
+ display_page(gWindow->page);
+ gWindow->fWindowHashTable = gWindow->page->fLinkHashTable;
+
+ /* then select for the events I normally would like to catch */
+ XSelectInput(gXDisplay, gWindow->fMainWindow, ButtonPress | KeyPressMask |
+ PointerMotionMask |
+ ExposureMask /* | EnterWindowMask | LeaveWindowMask */ );
+ XSelectInput(gXDisplay, gWindow->fScrollWindow, ExposureMask);
+
+}
+
+/*
+ * This routine is called when the quitbutton is hit. For the moment I am
+ * just going to leave it all behind
+ */
+
+void
+quitHyperDoc(void)
+{
+ HyperDocPage *page;
+
+ if (gSessionHashTable.num_entries == 1 || gParentWindow == gWindow) {
+ if (!strcmp(gWindow->page->name, "ProtectedQuitPage")){
+ exitHyperDoc();
+ }
+ page = (HyperDocPage *) hash_find(gWindow->fPageHashTable, "ProtectedQuitPage");
+ if (page == NULL) {
+ fprintf(stderr, "Unknown page name %s\n", "ProtectedQuitPage");
+ exitHyperDoc();
+ return;
+ }
+ if (gWindow->fDownLinkStackIndex == MaxDownlinkDepth)
+ fprintf(stderr, "exceeded maximum link nesting level\n");
+ else
+ gWindow->fDownLinkStack[gWindow->fDownLinkStackIndex++] = gWindow->page;
+ gWindow->page = page;
+ display_page(gWindow->page);
+ gWindow->fWindowHashTable = gWindow->page->fLinkHashTable;
+ }
+ else
+ exitHyperDoc();
+}
+
+
+/*
+ * find_page takes as an argument the HyperDoc for a page name and returns
+ * the associated page
+ */
+
+static HyperDocPage *
+find_page(TextNode * node)
+{
+ char *page_name;
+ HyperDocPage *page;
+
+ /* try and find the page name */
+ page_name = print_to_string(node);
+ page = (HyperDocPage *) hash_find(gWindow->fPageHashTable, page_name);
+
+ if (page == NULL) {
+ /* try to find the unknown page */
+ page = (HyperDocPage *) hash_find(gWindow->fPageHashTable, "UnknownPage");
+ if (page == NULL) {
+ /* Yikes, Even that could not be found */
+ fprintf(stderr, "Unknown page name %s\n", page_name);
+ }
+ else {
+ if (page->type == UnloadedPageType)
+ page->type = UlUnknownPage;
+ else
+ page->type = UnknownPage;
+ }
+ }
+ return page;
+}
+
+/*
+ * These are macros for taking care of the downlink stack, and the memolink
+ * stack.
+ */
+
+#define NotSpecial(t) ((t == Quitbutton || t == Returnbutton || \
+ t == Upbutton || t == UnknownPage || \
+ t == UlUnknownPage || t == ErrorPage) ?(0):(1))
+
+/* pushes a page onto the down link stack */
+
+static void
+downlink(void)
+{
+ if (gWindow->fDownLinkStackIndex == MaxDownlinkDepth)
+ fprintf(stderr, "exceeded maximum link nesting level\n");
+ else
+ gWindow->fDownLinkStack[gWindow->fDownLinkStackIndex++] = gWindow->page;
+}
+
+static void
+memolink(void)
+{
+ if (gWindow->fMemoStackIndex == MaxMemoDepth)
+ fprintf(stderr, "exceeded maximum link nesting level\n");
+ else {
+ gWindow->fMemoStack[gWindow->fMemoStackIndex] = gWindow->page;
+ gWindow->fDownLinkStackTop[gWindow->fMemoStackIndex++] = gWindow->fDownLinkStackIndex;
+ }
+}
+
+static void
+killAxiomPage(HyperDocPage * page)
+{
+ char command[512];
+
+ sprintf(command, "(|htpDestroyPage| '%s)", page->name);
+ send_lisp_command(command);
+}
+
+static void
+kill_page(HyperDocPage * page)
+{
+ page->scroll_off = 0;
+ if (page->type == SpadGen) {
+ hash_delete(gWindow->fPageHashTable, page->name);
+ killAxiomPage(page);
+ free_page(page);
+ }
+}
+
+/* pops the memo stack */
+
+static HyperDocPage *
+returnlink(void)
+{
+ int i;
+
+ if (gWindow->fMemoStackIndex == 0) {
+ BeepAtTheUser();
+ return NULL;
+ }
+ else {
+ kill_page(gWindow->page);
+ for (i = gWindow->fDownLinkStackIndex - 1;
+ i >= gWindow->fDownLinkStackTop[gWindow->fMemoStackIndex - 1];
+ i--)
+ {
+ kill_page(gWindow->fDownLinkStack[i]);
+ }
+ gWindow->fDownLinkStackIndex =
+ gWindow->fDownLinkStackTop[--gWindow->fMemoStackIndex];
+ return (gWindow->fMemoStack[gWindow->fMemoStackIndex]);
+ }
+}
+
+/* pops a page if it can from the downlink stack */
+
+static HyperDocPage *
+uplink(void)
+{
+ if (gWindow->fDownLinkStackIndex == 0)
+ return returnlink();
+ else {
+ kill_page(gWindow->page);
+ return (gWindow->fDownLinkStack[--gWindow->fDownLinkStackIndex]);
+ }
+}
+
+static void
+windowlink_handler(TextNode * node)
+{
+ char *page_name;
+
+ /* first try and find the page */
+ page_name = print_to_string(node);
+
+ if (init_top_window(page_name) == -1) {
+ return;
+ }
+/* gWindow->fWindowHashTable = gWindow->page->fLinkHashTable;*/
+}
+
+void
+make_window_link(char *name)
+{
+ if (init_top_window(name) != -1)
+{}/* gWindow->fWindowHashTable = gWindow->page->fLinkHashTable; */
+}
+
+
+static void
+lispwindowlink_handler(HyperLink * link)
+{
+
+ /*
+ * Since we are popping up a new window, then we had better change all
+ * the cursors right away. We won't get another chance at it.
+ */
+
+ if (init_top_window(NULL) != -1) {
+ HyperDocPage *page = NULL;
+ int frame = gWindow->fAxiomFrame;
+
+ page = issue_server_command(link);
+ gWindow->fAxiomFrame = frame;
+ gWindow->page = page;
+/* gWindow->fWindowHashTable = gWindow->page->fLinkHashTable;*/
+ }
+}
+
+static HyperDocPage *
+paste_button(PasteNode * paste)
+{
+ HyperDocPage *page = NULL;
+ int pastewhere=paste->where;
+
+
+ if ( paste->end_node ==NULL || paste->begin_node==NULL || paste->arg_node==NULL ){
+ BeepAtTheUser();
+ return NULL;
+ }
+
+ page=parse_patch(paste);
+/* paste has changed after this call so use pastewhere*/
+
+ if (pastewhere && page ) {
+ if (0 == strcmp(page->name, "ErrorPage"))
+ page = NULL;
+ }
+ else
+ BeepAtTheUser();
+
+ return page;
+}
+
+void
+helpForHyperDoc(void)
+{
+ HyperDocPage *page = NULL;
+
+ /* do not do anything if we are already at the "no more help" page */
+
+ if (0 == strcmp(gWindow->page->name, NoMoreHelpPage))
+ return;
+
+ /* if no help page recorded, use the standard "no more help" page */
+
+ if (!gWindow->page->helppage)
+ gWindow->page->helppage = alloc_string(NoMoreHelpPage);
+
+ /* if we are on the main help page, use "no more help" page */
+
+ if (0 == strcmp(gWindow->page->name, TopLevelHelpPage))
+ gWindow->page->helppage = alloc_string(NoMoreHelpPage);
+
+ page = (HyperDocPage *) hash_find(gWindow->fPageHashTable, gWindow->page->helppage);
+
+ if (page)
+ make_window_link(gWindow->page->helppage);
+ else
+ BeepAtTheUser();
+}
+
+static HyperLink *
+findButtonInList(HDWindow * window, int x, int y)
+{
+ ButtonList *bl;
+
+ if (!window || window->page->type == UnloadedPageType)
+ return NULL;
+ for (bl = window->page->s_button_list; bl != NULL; bl = bl->next)
+ if (x >= bl->x0 && x <= bl->x1 && y >= bl->y0 && y <= bl->y1)
+ return bl->link;
+ for (bl = window->page->button_list; bl != NULL; bl = bl->next)
+ if (x >= bl->x0 && x <= bl->x1 && y >= bl->y0 && y <= bl->y1)
+ return bl->link;
+ return NULL;
+}
+
+static HyperLink *
+get_hyper_link(XButtonEvent * event)
+{
+ HyperLink *l1, *l2;
+
+ l1 = (HyperLink *) hash_find(gWindow->fWindowHashTable, (char *)&(event->window));
+ if (l1)
+ return l1;
+ l2 = findButtonInList(gWindow, event->x, event->y);
+ return l2;
+}
+
+/*
+ * Handle a button pressed event. window is the subwindow in which the event
+ * occured, and button is the button which was pressed
+ */
+
+static void
+handle_button(int button, XButtonEvent * event)
+{
+ HyperLink *link;
+ HyperDocPage *page = NULL;
+ char *page_name;
+
+ /* find page name from sub-window handle */
+
+ link = get_hyper_link(event);
+
+ if (link == NULL) { /* user clicked on an inactive area */
+/* BeepAtTheUser(); */ /* I always thought this was annoying. RSS */
+ return;
+ }
+
+ switch (link->type) {
+ case Pastebutton:
+ page = paste_button(link->reference.paste);
+ break;
+ case Link:
+ page_name = print_to_string(link->reference.node);
+ page = (HyperDocPage *) hash_find(gWindow->fPageHashTable, page_name);
+ break;
+ case Helpbutton:
+ helpForHyperDoc();
+ page = NULL;
+ break;
+ case Scrollbar:
+ scrollScroller(event);
+ break;
+ case Scrollupbutton:
+ scrollUp();
+ break;
+ case Scrolldownbutton:
+ scrollDown();
+ break;
+
+ case Inputstring:
+ /* We must be changing input focus or getting a selection */
+
+ change_input_focus(link);
+ if ( button == Button2 ) {
+ XConvertSelection(gXDisplay, XA_PRIMARY, XA_STRING,
+ XInternAtom(gXDisplay, "PASTE_SELECTION", False),
+ gWindow->fMainWindow, CurrentTime);
+ gSavedInputAreaLink = link;
+ }
+ break;
+
+ case SimpleBox:
+ page = NULL;
+ toggle_input_box(link);
+ break;
+ case Radiobox:
+ page = NULL;
+ toggle_radio_box(link);
+ break;
+ case Quitbutton:
+ quitHyperDoc();
+ break;
+ case Returnbutton: /* pop memo information */
+ page = returnlink();
+ break;
+ case Upbutton: /* pop downlink information */
+ page = uplink();
+ break;
+ case Downlink:
+ page = find_page(link->reference.node);
+ if (page && NotSpecial(page->type))
+ downlink();
+ break;
+ case Memolink:
+ page = find_page(link->reference.node);
+ if (page && NotSpecial(page->type))
+ memolink();
+ break;
+ case Windowlink:
+ page = find_page(link->reference.node);
+ if (page && NotSpecial(page->type)) {
+ windowlink_handler(link->reference.node);
+ gNeedIconName = 1;
+ page = NULL;
+ }
+ break;
+ case Lispwindowlink:
+ lispwindowlink_handler(link);
+ gNeedIconName = 1;
+ page = NULL;
+ break;
+ case LispMemoLink:
+ case Spadmemolink:
+ page = issue_server_command(link);
+ if (page && NotSpecial(page->type))
+ memolink();
+ break;
+ case LispDownLink:
+ case Spaddownlink:
+ page = issue_server_command(link);
+ if (page && NotSpecial(page->type))
+ downlink();
+ break;
+ case Spadlink:
+ case Lisplink:
+ page = issue_server_command(link);
+ break;
+ case Lispcommand:
+ case Qspadcall:
+ case Spadcall:
+ page = issue_server_command(link);
+ break;
+ case Lispcommandquit:
+ case Spadcallquit:
+ case Qspadcallquit:
+ page = issue_server_command(link);
+ exitHyperDoc();
+ break;
+ case Spadcommand:
+ case Spadgraph:
+ case Spadsrc:
+ issue_spadcommand(gWindow->page, link->reference.node,
+ button == Button1, link->type);
+ break;
+ case Unixlink:
+ page = issue_unixlink(link->reference.node);
+ if (page && NotSpecial(page->type)) {
+ downlink();
+ }
+ break;
+ case Unixcommand:
+ issue_unixcommand(link->reference.node);
+ break;
+ default:
+ break;
+ }
+
+ if (page) {
+ switch (page->type) { /* check for special button types */
+ case Quitbutton:
+ exitHyperDoc();
+ return;
+ case Returnbutton:
+ gWindow->page = returnlink();
+ break;
+ case Upbutton:
+ gWindow->page = uplink();
+ break;
+ case ErrorPage:
+ case UnknownPage:
+ case UlUnknownPage:
+ if (page->type == UlUnknownPage)
+ page->type = UnloadedPageType;
+ downlink();
+ gWindow->page = page;
+ break;
+ default: /* a normal link */
+ gWindow->page = page;
+ break;
+ }
+ if (link->type != Pastebutton)
+ display_page(gWindow->page);
+ gWindow->fWindowHashTable = gWindow->page->fLinkHashTable; /* reset the window hash */
+ }
+}
+
+
+void
+exitHyperDoc(void)
+{
+ XEvent event;
+
+
+ if (gSessionHashTable.num_entries == 1 || gParentWindow == gWindow) {
+ free_hd_window(gWindow);
+ exit(0);
+ }
+ hash_delete(&gSessionHashTable, (char *)&gWindow->fMainWindow);
+
+ /*
+ * Now we quickly want to flush all the events associated with this
+ * window from existence
+ */
+
+ XFlush(gXDisplay);
+ while (XCheckWindowEvent(gXDisplay, gWindow->fMainWindow, bigmask, &event)) {
+ }
+ while (XCheckWindowEvent(gXDisplay, gWindow->fScrollWindow,bigmask, &event)) {
+ }
+ while (XCheckWindowEvent(gXDisplay, gWindow->fDisplayedWindow, bigmask, &event)) {
+ }
+ while (XCheckWindowEvent(gXDisplay, gWindow->fScrollUpWindow, bigmask, &event)) {
+ }
+ while (XCheckWindowEvent(gXDisplay, gWindow->fScrollDownWindow, bigmask, &event)) {
+ }
+ while (XCheckWindowEvent(gXDisplay, gWindow->scrollbar, bigmask, &event)) {
+ }
+ while (XCheckWindowEvent(gXDisplay, gWindow->scroller, bigmask, &event)) {
+ }
+
+ XDestroyWindow(gXDisplay, gWindow->fMainWindow);
+ free_hd_window(gWindow);
+ gWindow = NULL;
+ gActiveWindow = -1;
+ XFlush(gXDisplay);
+}
+
+static int
+set_window(Window window)
+{
+ Window root, parent, *children, grandparent,myarg;
+ HDWindow *htw;
+ unsigned int nchildren;
+ int st;
+
+ myarg=window;
+ nchildren = 0;
+ htw = (HDWindow *) hash_find(&gSessionHashTable, (char *)&myarg);
+ if (htw != NULL) {
+ gWindow = htw;
+ return 1;
+ }
+ st = XQueryTree(gXDisplay, myarg, &root, &parent, &children, &nchildren);
+ if (st==0) goto ERROR;
+ if (nchildren > 0)
+ XFree(children);
+ htw = (HDWindow *) hash_find(&gSessionHashTable, (char *)&parent);
+ if (htw != NULL) {
+ gWindow = htw;
+ return 1;
+
+ }
+ else {
+ /* check for a grandparent */
+ st = XQueryTree(gXDisplay, parent, &root, &grandparent, &children, &nchildren);
+ if (st==0) goto ERROR;
+ if (nchildren > 0)
+ XFree(children);
+ htw = (HDWindow *) hash_find(&gSessionHashTable, (char *)&grandparent);
+ if (htw != NULL) {
+ gWindow = htw;
+ return 1;
+ }
+ }
+
+ /*
+ * fprintf(stderr, "window(%d) and it's parent(%d) aren't in
+ * gSessionHashTable\n", window, parent);
+
+ we never found that window. this happens if (not iff) we exit from
+ an unfocused non-main window under certain wm's and click-to-type. the program returns here with
+ the window handle that was just destroyed. So let's set the global gWindow
+ to the main window.
+ */
+
+ERROR:
+ gWindow=gParentWindow;
+ return 0;
+}
+
+/*
+ * This procedure whips thru the stack and clears all expose events for the
+ * given routine
+ */
+static void
+clear_exposures(Window w)
+{
+ XEvent report;
+
+ XFlush(gXDisplay);
+ while (XCheckTypedWindowEvent(gXDisplay, w, Expose, &report));
+}
+void
+get_new_window(void)
+{
+
+ int val;
+ char buf[128];
+ int frame;
+ Window wid;
+ HDWindow *htw;
+ HyperDocPage *hpage;
+
+
+ /*
+ * If I am going to try and start a new window, then I should make sure I
+ * have a coonection to listen on
+ *
+ * BUT This code is entered when a socket selects
+ *
+ * if (spad_socket == NULL) { spad_socket =
+ * connect_to_local_server(SpadServer, MenuServer, 10); if (spad_socket
+ * == NULL) { fprintf(stderr, "Get_new_window: Couldn't Connect to
+ * SpadServer\n"); return -1; } }
+ *
+ */
+
+
+ frame = get_int(spad_socket);
+ val = get_int(spad_socket);
+ switch (val) {
+ case StartPage:
+ init_top_window(NULL);
+ val = get_int(spad_socket);
+ init_scanner();
+ input_type = FromSpadSocket;
+ input_string = "";
+ gWindow->page = parse_page_from_socket();
+ gWindow->fAxiomFrame = frame;
+ XFlush(gXDisplay);
+ break;
+ case LinkToPage:
+ get_string_buf(spad_socket, buf, 128);
+ if (init_top_window(buf) == -1) {
+ fprintf(stderr, "get_new_window: Did not find page %s\n", buf);
+ /* return -1; */
+ }
+ gWindow->fAxiomFrame = frame;
+ break;
+ case PopUpPage:
+ val = get_int(spad_socket);
+ init_form_window(NULL, val);
+ send_int(spad_socket, gWindow->fMainWindow);
+ init_scanner();
+ input_type = FromSpadSocket;
+ input_string = "";
+ gWindow->page = parse_page_from_socket();
+ compute_form_page(gWindow->page);
+
+ XMapWindow(gXDisplay, gWindow->fMainWindow);
+
+ gWindow->fWindowHashTable = gWindow->page->fLinkHashTable;
+ gWindow->fAxiomFrame = frame;
+ XFlush(gXDisplay);
+ break;
+ case PopUpNamedPage:
+ val = get_int(spad_socket);
+ get_string_buf(spad_socket, buf, 128);
+
+ if (init_form_window(buf, val) == -1) {
+ send_int(spad_socket, -1);
+ break;
+ }
+ load_page(gWindow->page);
+ compute_form_page(gWindow->page);
+
+ XMapWindow(gXDisplay, gWindow->fMainWindow);
+
+ gWindow->fWindowHashTable = gWindow->page->fLinkHashTable;
+ gWindow->fAxiomFrame = frame;
+ XFlush(gXDisplay);
+ send_int(spad_socket, gWindow->fMainWindow);
+ /* fprintf(stderr, "Window Id was %d\n", gWindow->fMainWindow); */
+ break;
+ case ReplaceNamedPage:
+ wid = (Window) get_int(spad_socket);
+ get_string_buf(spad_socket, buf, 128);
+
+ htw = (HDWindow *) hash_find(&gSessionHashTable,(char *)&wid);
+ if (htw == NULL) break;
+ hpage = (HyperDocPage *) hash_find(gWindow->fPageHashTable, buf);
+ if (hpage == NULL) break;
+ gWindow = htw;
+ gWindow->page = hpage;
+ display_page(gWindow->page);
+ gWindow->fWindowHashTable = gWindow->page->fLinkHashTable;
+ clear_exposures(gWindow->fMainWindow);
+ clear_exposures(gWindow->fScrollWindow);
+ XFlush(gXDisplay);
+ break;
+ case ReplacePage:
+ wid = (Window) get_int(spad_socket);
+ set_window(wid);
+ init_scanner();
+ input_type = FromSpadSocket;
+ input_string = "";
+ gWindow->page = parse_page_from_socket();
+ display_page(gWindow->page);
+ gWindow->fWindowHashTable = gWindow->page->fLinkHashTable;
+ clear_exposures(gWindow->fMainWindow);
+ clear_exposures(gWindow->fScrollWindow);
+ XFlush(gXDisplay);
+ break;
+ case KillPage:
+ /* Here the user wishes to kill the page */
+ wid = (Window) get_int(spad_socket);
+ htw = (HDWindow *) hash_find(&gSessionHashTable,(char *)&wid);
+ if (htw !=NULL) {
+ gWindow = htw;
+ exitHyperDoc();
+ break;
+ }
+ break;
+ }
+ }
+static void
+set_cursor(HDWindow *window,Cursor state)
+{
+ if (state == gBusyCursor)
+ XDefineCursor(gXDisplay, window->fMainWindow, gBusyCursor);
+ else if (state == gActiveCursor)
+ XDefineCursor(gXDisplay, window->fMainWindow, gActiveCursor);
+ else
+ XDefineCursor(gXDisplay, window->fMainWindow, gNormalCursor);
+ XFlush(gXDisplay);
+}
+
+static void
+change_cursor(Cursor state, HDWindow *window)
+{
+ if (window->fDisplayedCursor == state)
+ return;
+ window->fDisplayedCursor = state;
+ set_cursor(window, state);
+}
+
+static void
+handle_motion_event(XMotionEvent *event)
+{
+ if (!gWindow)
+ return;
+ if (findButtonInList(gWindow, event->x, event->y) != NULL)
+ change_cursor(gActiveCursor, gWindow);
+ else
+ change_cursor(gNormalCursor, gWindow);
+}
+
+static void
+init_cursor_state(HDWindow *window)
+{
+ if (window) {
+ int x, y, rx, ry, but;
+ Window r, c;
+
+ XQueryPointer(gXDisplay, window->fMainWindow,
+ &r, &c, &rx, &ry, &x, &y,(unsigned int *) &but);
+ if (findButtonInList(window, x, y) != NULL)
+ change_cursor(gActiveCursor, window);
+ else
+ change_cursor(gNormalCursor, window);
+ }
+}
+
+static void
+init_cursor_states(void)
+{
+ hash_map(&gSessionHashTable,(MappableFunction) init_cursor_state);
+}
+
+
+static void
+make_busy_cursor(HDWindow *window)
+{
+ change_cursor(gBusyCursor, window);
+}
+
+static void
+make_busy_cursors(void)
+{
+ hash_map(&gSessionHashTable, (MappableFunction)make_busy_cursor);
+}
+
+static int
+HyperDocErrorHandler(Display *display, XErrorEvent *xe)
+{
+ if (xe->request_code != 15) {
+ char buf[1024];
+
+ XGetErrorText(display, xe->error_code, buf, sizeof(buf));
+
+ fprintf(stderr, "error code = %d\n", xe->error_code);
+ fprintf(stderr, "major op code = %d\n", xe->request_code);
+ fprintf(stderr, "minor op code = %d\n", xe->minor_code);
+ fprintf(stderr, "XID = %ld\n", xe->resourceid);
+ fprintf(stderr, "%s\n", buf);
+
+ if (xe->request_code != 15)
+ exit(-1);
+ }
+ return(0);
+}
+
+
+
+static void
+set_error_handlers(void)
+{
+ XSetErrorHandler(HyperDocErrorHandler);
+}
+@
+\section{License}
+<<license>>=
+/*
+Copyright (c) 1991-2002, The Numerical ALgorithms Group Ltd.
+All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are
+met:
+
+ - Redistributions of source code must retain the above copyright
+ notice, this list of conditions and the following disclaimer.
+
+ - Redistributions in binary form must reproduce the above copyright
+ notice, this list of conditions and the following disclaimer in
+ the documentation and/or other materials provided with the
+ distribution.
+
+ - Neither the name of The Numerical ALgorithms Group Ltd. nor the
+ names of its contributors may be used to endorse or promote products
+ derived from this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
+IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
+TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
+PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
+OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
+SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+*/
+
+@
+<<*>>=
+<<license>>
+<<event.c>>
+@
+\eject
+\begin{thebibliography}{99}
+\bibitem{1} nothing
+\end{thebibliography}
+\end{document}