#ifndef lint static char rcsId[]="$Header: /usr/local/rcs/Newt/XmHTML/RCS/example_2.c,v 1.12 1997/08/31 17:45:46 newt Exp newt $"; #endif /***** * example_2.c : a more enhanced XmHTML example * * This file Version $Revision: 1.12 $ * * Creation date: Wed Jan 29 01:43:38 GMT+0100 1997 * Last modification: $Date: 1997/08/31 17:45:46 $ * By: $Author: newt $ * Current State: $State: Exp $ * * Author: newt * * Copyright (C) 1994-1997 by Ripley Software Development * All Rights Reserved * * This file is part of the XmHTML Widget Library. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the Free * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * *****/ /***** * ChangeLog * $Log: example_2.c,v $ * Revision 1.12 1997/08/31 17:45:46 newt * Several HTML-form related bugfixes. Added the debug menu stuff. * * Revision 1.11 1997/08/30 01:55:25 newt * Added an Options menu to test XmHTML's SetValues method. Proved to * be *very* usefull... * Added support for a font and charset tags. * XmImage & XmImageInfo changes. * A few bugfixes in image loading. * * Revision 1.10 1997/05/28 02:06:34 newt * Replaced image caching to use caching offered by cache.c. Made resolveFile a * lot smarter. Added a comment header on all functions. * * Revision 1.9 1997/04/29 14:35:03 newt * XmHTMLXYToInfo + XmHTMLGetHeadAttributes added. * Changes in image and document cache. * * Revision 1.8 1997/04/03 05:43:30 newt * frame and flushImages bugfixes. * * Revision 1.7 1997/03/28 07:29:49 newt * Frame support added. * * Revision 1.6 1997/03/20 08:21:27 newt * quite a few changes: imageCache and docCache related. Not working well yet. * * Revision 1.5 1997/03/11 20:09:50 newt * -root option (Ricky Ralston); renamed convenience functions; * ImageCache changes * * Revision 1.4 1997/03/04 18:50:57 newt * Animation stuff and command line args added * * Revision 1.3 1997/03/04 01:03:13 newt * Delayed Image Loading. This example is becoming a real testbed for XmHTML. * * Revision 1.2 1997/03/02 23:25:42 newt * Added image callback. Added path-related functions * * Revision 1.1 1997/02/11 01:58:29 newt * Initial Revision * *****/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include /* isspace */ #include #include #include /* also for getwd on non-POSIX systems */ /***** * Change this to change the application class of the examples *****/ #define APP_CLASS "HTMLDemos" /***** * We want to have the full XmHTML instance definition available * when debugging this stuff, so include XmHTMLP.h * Note: * Including XmHTMLP.h normally doesn't pull in any of the internal XmHTML * functions. If you want to do this anyway, you need to include *both* * XmHTMLP.h and XmHTMLI.h (in that order). *****/ #include #include "debug.h" /* we want to be able to enable debugging */ #include "debug_menu.h" /* and do it on the fly as well */ /* imagecache stuff */ #include "cache.h" /* catch NULL strdup's for debug builds */ #if defined(DEBUG) && !defined(DMALLOC) extern char *__rsd_strdup(const char *s1, char *file, int line); #define strdup(PTR) __rsd_strdup(PTR, __FILE__, __LINE__) #endif #ifdef NEED_STRCASECMP # include extern int my_strcasecmp(const char *s1, const char *s2); extern int my_strncasecmp(const char *s1, const char *s2, size_t n); #define strcasecmp(S1,S2) my_strcasecmp(S1,S2) #define strncasecmp(S1,S2,N) my_strncasecmp(S1,S2,N) #endif /*** External Function Prototype Declarations ***/ /* from visual.c */ extern int getStartupVisual(Widget shell, Visual **visual, int *depth, Colormap *colormap); /* can be found in XmHTML */ extern char *my_strcasestr(const char *s1, const char *s2); /* from misc.c */ extern int parseFilename(char *fullname, char *filename, char *pathname); extern void XMessage(Widget widget, String msg); #ifdef DEBUG extern void _XmHTMLUnloadFonts(XmHTMLWidget); #endif /*** Public Variable Declarations ***/ /*** Private Datatype Declarations ****/ #define MAX_HISTORY_ITEMS 100 /* save up to this many links */ #define MAX_PATHS 25 /* size of visited path cache */ #define MAX_IMAGE_ITEMS 512 /* max. no of images per document */ #define MAX_HTML_WIDGETS 10 /* max. no of HTML widgets allowed */ #define FILE_OPEN 1 #define FILE_RELOAD 2 #define FILE_SAVEAS 3 #define FILE_RAISE 5 #define FILE_LOWER 6 #define FILE_INFO 7 #define FILE_VIEW 8 #define FILE_QUIT 9 /* Link defines */ #define LINK_MADE 0 #define LINK_HOMEPAGE 1 #define LINK_TOC 2 #define LINK_INDEX 3 #define LINK_GLOSSARY 4 #define LINK_COPYRIGHT 5 #define LINK_PREVIOUS 6 #define LINK_UP 7 #define LINK_DOWN 8 #define LINK_NEXT 9 #define LINK_HELP 10 #define LINK_LAST 11 /* options menu toggle buttons defines. These may *not* be changed */ #define OPTIONS_ANCHOR_BUTTONS 0 #define OPTIONS_HIGHLIGHT_ON_ENTER 1 #define OPTIONS_ENABLE_STRICT_HTML32 2 #define OPTIONS_ENABLE_BODYCOLORS 3 #define OPTIONS_ENABLE_BODYIMAGES 4 #define OPTIONS_ENABLE_DOCUMENT_COLORS 5 #define OPTIONS_ENABLE_DOCUMENT_FONTS 6 #define OPTIONS_ENABLE_OUTLINING 7 #define OPTIONS_DISABLE_WARNINGS 8 #define OPTIONS_FREEZE_ANIMATIONS 9 #define OPTIONS_AUTO_IMAGE_LOAD 10 #define OPTIONS_FANCY_TRACKING 11 #define OPTIONS_ENABLE_IMAGES 12 #define OPTIONS_LAST 13 /* options menu pushbutton defines */ #define OPTIONS_ANCHOR 20 #define OPTIONS_FONTS 21 #define OPTIONS_BODY 22 #define OPTIONS_IMAGE 23 /* document cache */ typedef struct{ String path; /* path to this document */ String file; /* full filename of this document */ int current_ref; /* last activated hyperlink */ int nrefs; /* total no of activated hyperlinks in this document */ String refs[MAX_HISTORY_ITEMS]; /* list of activated hyperlinks */ String visited[MAX_HISTORY_ITEMS]; /* list of visited hyperlins */ int nvisited; /* total no of visited hyperlinks */ int nimages; /* no of images in this document */ String images[MAX_IMAGE_ITEMS]; /* image urls for this document */ }DocumentCache; /* List of all HTML widgets (especially for frames) */ typedef struct{ Boolean active; /* is this an active frame? */ Boolean used; /* is this frame currently being used? */ String name; /* name of this frame */ String src; /* source file for this frame */ Widget html; /* XmHTMLWidget id for this frame */ }HTMLWidgetList; #define MIME_HTML 0 /* text/html */ #define MIME_HTML_PERFECT 1 /* text/html-perfect */ #define MIME_IMAGE 2 /* image/whatever */ #define MIME_PLAIN 3 /* text/plain/unknown */ #define MIME_IMG_UNSUP 4 /* unsupported image type */ #define MIME_ERR 5 /* some error occured */ typedef struct{ int link_type; /* 0 = rev, 1 = rel, which means fetch it */ Boolean have_data; String href; String title; }documentLinks; typedef struct{ Widget w; String name; Boolean value; }optionsStruct; /*** Private Function Prototype Declarations ****/ static DocumentCache *getDocFromCache(String file); static void storeDocInCache(String file); static void storeInHistory(String file, String loc); static void storeAnchor(String href); static void removeDocFromCache(int doc); static void flushImages(Widget w); static void killImages(void); /* XmHTMLWidget callbacks */ static void anchorCB(Widget w, XtPointer arg1, XmHTMLAnchorPtr href_data); static void trackCB(Widget w, XtPointer arg1, XmHTMLAnchorPtr href_data); static void docCB(Widget w, XtPointer arg1, XmHTMLDocumentPtr cbs); static void linkCB(Widget w, XtPointer arg1, XmHTMLLinkPtr cbs); static void frameCB(Widget w, XtPointer arg1, XmHTMLFramePtr cbs); static String collapseURL(String url); static void infoCB(Widget parent, Widget popup, XButtonPressedEvent *event); /* XmHTMLWidget functions */ static XmImageInfo *loadImage(Widget w, String url); static int testAnchor(Widget w, String href); static void jumpToFrame(String filename, String loc, String target, Boolean store); static int getImageData(XmHTMLPLCStream *stream, XtPointer buffer); static void endImageData(XmHTMLPLCStream *stream, XtPointer data, int type, Boolean ok); /* Menu and button bar callbacks */ static void linkButtonCB(Widget w, XtPointer arg1, XtPointer arg2); static void docInfoCB(Boolean font_only); static void infoPopupCB(Widget w, int item); static void fileCB(Widget widget, int item); static void historyCB(Widget w, int button); static void progressiveButtonCB(Widget w, int reset); static void optionsCB(Widget w, int item); static void showImageInfo(XmImageInfo *info); static void viewSource(Widget w); /*** Private Variable Declarations ***/ static XtAppContext context; static Widget back, forward, load_images, label, toplevel = NULL, reload; static Widget link_button, link_dialog, html32, verified, info_dialog; static Widget prg_button, image_dialog; static documentLinks document_links[LINK_LAST]; static XmImage *preview_image; static Widget link_buttons[LINK_LAST]; static char default_font[128], current_font[128]; static char default_charset[128], current_charset[128]; static String link_labels[LINK_LAST] = {"Mail Author", "Home", "TOC", "Index", "Glossary", "Copyright", "Prev", "Up", "Down", "Next", "Help"}; static String image_types[] = {"(error occured)", "Unknown Image type", "X11 Pixmap", "X11 Bitmap", "CompuServe(C) Gif87a or Gif89a", "Animated Gif89a", "Animated Gif89a with NETSCAPE2.0 loop extension", "CompuServe(C) Compatible Gzf87a or Gzf89a", "Gif89a Compatible animation", "Gif89a compatible animation with NETSCAPE2.0 loop extension", "JPEG", "PNG", "Fast Loadable Graphic"}; /***** * global XmHTML configuration. Elements must be in the same order as the * OPTIONS_ defines above. *****/ static optionsStruct html_config[OPTIONS_LAST] = { { NULL, XmNanchorButtons, True }, { NULL, XmNhighlightOnEnter, True }, { NULL, XmNstrictHTMLChecking, False }, { NULL, XmNenableBodyColors, True }, { NULL, XmNenableBodyImages, True }, { NULL, XmNenableDocumentColors, True }, { NULL, XmNenableDocumentFonts, True }, { NULL, XmNenableOutlining, True }, { NULL, XmNenableBadHTMLWarnings, True }, { NULL, XmNfreezeAnimations, False }, { NULL, "autoImageLoad", True }, { NULL, "fancyMouseTracking", False }, { NULL, XmNimageEnable, True }, }; /* Command line options */ static Boolean root_window, noframe, external_client; static Boolean progressive_images, allow_exec; static int animation_timeout = 175; #define MAX_PROGRESSIVE_DATA_SKIP 2048 static int progressive_data_skip = MAX_PROGRESSIVE_DATA_SKIP; static int progressive_data_inc = 0; #ifdef DEBUG static Boolean debug = False; #define Debug(MSG) do { \ if(debug) { printf MSG ; fflush(stdout); } }while(0) #else #define Debug(MSG) /* emtpy */ #endif static String usage = {"Options:\n" "\t-allow_exec : honor href=\"exec:\" or href=\"xexec:\"\n" "\t-animation_timeout : animation timeout in milliseconds. Default is 175\n" #ifdef DEBUG "\t-debug : enable application debug output\n" #endif "\t-images_delayed : delay image loading\n" "\t-netscape : fire netscape for unsupported URL's\n" "\t-noframe : don't put a frame around the HTML display area\n" "\t-root : act as root window\n" "\t-progressive : load images progressively (only GZF for now)\n" "\t-prg_skip [num] : progressive data skip. Default is 2048\n" "\t-prg_inc [num] : progressive data increment. Resets prg_skip to 256.\n" "\t Using this option overrides any prg_skip value.\n" "\t-h, --help : print this help\n"}; /* document cache */ static DocumentCache doc_cache[MAX_HISTORY_ITEMS]; static int current_doc, last_doc; /***** * List of all html widgets. The first slot is the toplevel HTML widget and * is never freed. All other slots are used by frames. *****/ static HTMLWidgetList html_widgets[MAX_HTML_WIDGETS]; /* visited paths */ static String paths[MAX_PATHS][1024]; static int max_paths; /* default settings */ static String appFallbackResources[] = { "*fontList: *-adobe-helvetica-bold-r-*-*-*-120-*-*-p-*-*-*", "*sourceView.width: 550", "*sourceView.height: 500", "*XmHTML.width: 575", "*XmHTML.height: 600", NULL}; /***** * Name: setBusy * Return Type: void * Description: changes the cursor from or to a stopwatch to indicate we are * busy doing something lengthy processing which can't be * interrupted. * In: * state: True to display the cursor as busy, False to display the * normal cursor. * Returns: * nothing. *****/ static void setBusy(Boolean state) { static Boolean busy; static Cursor cursor; Display *display = XtDisplay(toplevel); if(!cursor) { cursor = XCreateFontCursor(display, XC_watch); busy = False; } if(busy != state) { busy = state; if(busy) XDefineCursor(display, XtWindow(toplevel), cursor); else XUndefineCursor(display, XtWindow(toplevel)); } XFlush(display); } /***** * Name: addPath * Return Type: void * Description: adds a path to the list of visited paths if it hasn't been * stored yet. * In: * path: path to be stored; * Returns: * nothing. *****/ static void addPath(String path) { int i = 0; /* see if the path has already been added */ for(i = 0; i < max_paths; i++) if(!(strcmp((char*)(paths[i]), path))) return; /* store this path */ if(max_paths < MAX_PATHS) { strcpy((char*)(paths[max_paths]), path); max_paths++; } } /***** * Follow symbolic links (if any) to translate filename into the name of the * real file that it represents. Returns TRUE if the call was successful, * meaning the links were translated successfully, or the file was not * linked to begin with (or there was no file). Returns false if some * error prevented the call from determining if there were symbolic links * to process, or there was an error in processing them. The error * can be read from the unix global variable errno. *****/ Boolean followSymLinks(String filename) { /***** * FIXME * * readlink doesn't seem to do anything at all on Linux 2.0.27, * libc 5.3.12 *****/ int cc; char buf[1024]; cc = readlink(filename, buf, 1024); if (cc == -1) { #ifdef __sgi if (errno == EINVAL || errno == ENOENT || errno == ENXIO) #else if (errno == EINVAL || errno == ENOENT) #endif /* no error, just not a symbolic link, or no file */ return(True); else return(False); } else { buf[cc] = '\0'; strcpy(filename, buf); return(True); } } /***** * Name: resolveFile * Return Type: String * Description: checks if the given file exists on the local file system * In: * filename: file to check * Returns: * a full filename when the file exists. NULL if it doesn't. * Note: * This routine tries three things to check if a file exists on the local * file system: * 1. if "filename" is absolute, it is assumed the file exists and is * accessible; * 2. checks whether "filename" can be found in the path of the current * document; * 3. sees if "filename" can be found in the list of stored paths. * When a file has been found, it is checked if this is a regular file, * and if so it is transformed into a fully qualified pathname (with * relative paths fully resolved). *****/ static String resolveFile(String filename) { static String ret_val; char tmp[1024]; /* throw out http:// stuff */ if(!(strncasecmp(filename, "http://", 7))) return(NULL); Debug(("resolveFile, looking for %s\n", filename)); ret_val = NULL; /***** * If this is an absolute path, check if it's really a valid file * (or directory). This allows us to recognize chrooted files when * browsing the local web directory. *****/ if(filename[0] == '/') { if(!(access(filename, R_OK))) ret_val = strdup(filename); else /* a fake path, strip of the leading / */ sprintf(tmp, "%s", &filename[1]); } else { strcpy(tmp, filename); tmp[strlen(filename)] = '\0'; /* NULL terminate */ } if(ret_val == NULL) { char real_file[1024]; int i; /***** * search the paths visited so far. Do it top to bottom as the * last visited path is inserted in the last slot. Quite usefull * when looking for images or links in the current document. *****/ for(i = max_paths-1; i >= 0 ; i--) { sprintf(real_file, "%s%s", (char*)(paths[i]), tmp); /* check if we have access to this thing */ if(!(access(real_file, R_OK))) { struct stat statb; /***** * We seem to have some access rights, make sure this * is a regular file *****/ if(stat(real_file, &statb) == -1) { perror(filename); break; } else if(S_ISDIR(statb.st_mode)) { /***** * It's a dir. First check for index.html then * for Welcome.html. *****/ int len = strlen(real_file)-1; strcat(real_file, real_file[len] == '/' ? "index.html\0" : "/index.html\0"); if(!(access(real_file, R_OK))) { ret_val = strdup(real_file); break; } real_file[len+1] = '\0'; strcat(real_file, real_file[len] == '/' ? "Welcome.html\0" : "/Welcome.html\0"); if(!(access(real_file, R_OK))) { ret_val = strdup(real_file); break; } /* no file in here, too bad */ break; } else if(!S_ISREG(statb.st_mode)) { fprintf(stderr, "%s: not a regular file\n", filename); break; } ret_val = strdup(real_file); break; } } } if(ret_val == NULL) { sprintf(tmp, "%s:\ncannot display: unable to locate file.", filename); XMessage(toplevel, tmp); } else { /* clean out relative path stuff and add the path to the path index. */ char fname[1024], pname[1024]; (void)parseFilename(ret_val, fname, pname); addPath(pname); /* * resolve symbolic links as well (prevents object cache from going * haywire by having two different objects with cross-linked * mappings) */ #if 0 (void)followSymLinks(pname); #endif /***** * big chance parseFilename compressed relative paths out of ret_val, * do it again. We need to reallocate as the size of the fully * resolved path can exceed the current length. *****/ ret_val = (String)realloc(ret_val, strlen(pname) + strlen(fname) + 1); sprintf(ret_val, "%s%s", pname, fname); } Debug(("resolveFile, found as %s\n", ret_val ? ret_val : "(not found)")); return(ret_val); } /***** * Name: getMimeType * Return Type: int * Description: make a guess at the mime type of a document by looking at * the extension of the given document. * In: * file: file for which to get a mime-type; * Returns: * mime type of the given file. *****/ static int getMimeType(String file) { String chPtr; unsigned char img_type; if((chPtr = strstr(file, ".")) != NULL) { String start; /* first check if this is plain HTML or not */ for(start = &file[strlen(file)-1]; *start && *start != '.'; start--); if(!strcasecmp(start, ".html") || !strcasecmp(start, ".htm")) return(MIME_HTML); if(!strcasecmp(start, ".htmlp")) return(MIME_HTML_PERFECT); } /* something else then? */ /* check if this is an image XmHTML knows of */ if((img_type = XmHTMLImageGetType(file, NULL, 0)) == IMAGE_ERROR) return(MIME_ERR); /***** * Not an image we know of, get first line in file and see if it's * html anyway *****/ if(img_type == IMAGE_UNKNOWN) { FILE *fp; char buf[128]; /* open file */ if((fp = fopen(file, "r")) == NULL) return(MIME_ERR); /* read first line in file */ if((chPtr = fgets(buf, 128, fp)) == NULL) { /* close again */ fclose(fp); return(MIME_ERR); } /* close again */ fclose(fp); /* see if it contains any of these strings */ if(my_strcasestr(buf, "", XtNiconName, "", NULL); } XtSetSensitive(reload, True); setBusy(False); return(True); } /***** * Name: getInfoSize * Return Type: int * Description: returns the size of the given XmImageInfo structure. * In: * call_data: ptr to a XmImageInfo structure; * client_data: data registered when we called initCache. * Returns: * size of the given XmImageInfo structure. * Note: * This function is used both by us and the caching routines. *****/ static int getInfoSize(XtPointer call_data, XtPointer client_data) { int size = 0; XmImageInfo *frame = (XmImageInfo*)call_data; while(frame != NULL) { size += sizeof(XmImageInfo); size += frame->width*frame->height; /* raw image data */ /* clipmask size. The clipmask is a bitmap of depth 1 */ if(frame->clip) { int clipsize; clipsize = frame->width; /* make it byte-aligned */ while((clipsize % 8)) clipsize++; /* this many bytes on a row */ clipsize /= 8; /* and this many rows */ clipsize *= frame->height; size += clipsize; } /* reds, greens and blues */ size += 3*frame->ncolors*sizeof(Dimension); frame = frame->frame; /* next frame of this image (if any) */ } return(size); } /***** * Name: getDocFromCache * Return Type: DocumentCache* * Description: retrieves a document from the document cache. * In: * file: filename of document to be retrieved. * Returns: * nothing. *****/ static DocumentCache* getDocFromCache(String file) { int i; for(i = 0; i < last_doc; i++) { if(!(strcmp(doc_cache[i].file, file))) return(&doc_cache[i]); } return(NULL); } /***** * Name: storeDocInCache * Return Type: void * Description: stores the given document in the document cache. * In: * file: filename of document to be stored; * Returns: * nothing. *****/ static void storeDocInCache(String file) { char foo[128], pname[1024]; if(last_doc == MAX_HISTORY_ITEMS) { int i; /* free hrefs */ for(i = 0; i < doc_cache[0].nrefs; i++) { if(doc_cache[0].refs[i]) free(doc_cache[0].refs[i]); doc_cache[0].refs[i] = NULL; } /* free image url's */ for(i = 0; i < doc_cache[0].nimages; i++) { free(doc_cache[0].images[i]); doc_cache[0].images[i] = NULL; } /* free visited anchor list */ for(i = 0; i < doc_cache[0].nvisited; i++) { if(doc_cache[0].visited[i]) free(doc_cache[0].visited[i]); doc_cache[0].visited[i] = NULL; } /* free file and path fields */ free(doc_cache[0].file); free(doc_cache[0].path); /* move everything downward */ for(i = 0; i < MAX_HISTORY_ITEMS-1; i++) doc_cache[i] = doc_cache[i+1]; last_doc = MAX_HISTORY_ITEMS - 1; } Debug(("Storing document %s in document cache\n", file)); current_doc = last_doc; doc_cache[current_doc].nrefs = 0; doc_cache[current_doc].nvisited = 0; doc_cache[current_doc].nimages = 0; doc_cache[current_doc].file = strdup(file); /* get path to this file */ (void)parseFilename(file, foo, pname); /* and store it */ doc_cache[current_doc].path = strdup(pname); last_doc++; } /***** * Name: removeDocFromCache * Return Type: void * Description: removes a document from the document cache * In: * doc: id of document to be removed; * Returns: * nothing. *****/ static void removeDocFromCache(int doc) { DocumentCache *this_doc; int i; this_doc = &doc_cache[doc]; Debug(("Removing document %s from document cache\n", this_doc->file)); /* remove all document url's */ for(i = 0; i < this_doc->nrefs; i++) { if(this_doc->refs[i]) free(this_doc->refs[i]); this_doc->refs[i] = NULL; } /* free visited anchor list */ for(i = 0; i < this_doc->nvisited; i++) { if(this_doc->visited[i]) free(this_doc->visited[i]); this_doc->visited[i] = NULL; } /* and remove all image url's */ for(i = 0; i < this_doc->nimages; i++) { /* remove image cache entry for this image */ removeURLObjectFromCache(this_doc->images[i]); free(this_doc->images[i]); } /***** * Update image cache (clears out all objects with a reference count * of zero). *****/ pruneObjectCache(); this_doc->nrefs = 0; this_doc->nvisited = 0; this_doc->nimages = 0; free(this_doc->file); free(this_doc->path); } /***** * Name: storeInHistory * Return Type: void * Description: stores the given href in the history list * In: * file: name of document * loc: value of named anchor in file. * Returns: * nothing. *****/ static void storeInHistory(String file, String loc) { int i; DocumentCache *this_doc = NULL; /* sanity check */ if(file == NULL && loc == NULL) return; /* if file is NULL we are for sure in the current document */ if(file == NULL) this_doc = &doc_cache[current_doc]; else { /**** * This is a new file. If we have any documents on the stack, * remove them. ****/ if(last_doc) { for(i = current_doc+1; i < last_doc; i++) removeDocFromCache(i); last_doc = current_doc+1; } /* update refs for current document */ this_doc = &doc_cache[current_doc]; if(this_doc->nrefs) { /* remove any existing references above the current one */ for(i = this_doc->current_ref+1; i < this_doc->nrefs; i++) { if(this_doc->refs[i]) { free(this_doc->refs[i]); this_doc->refs[i] = NULL; } } this_doc->nrefs = this_doc->current_ref + 1; } if((this_doc = getDocFromCache(file)) == NULL) { storeDocInCache(file); this_doc = &doc_cache[current_doc]; } } if(this_doc->nrefs == MAX_HISTORY_ITEMS) { /* free up the first item */ if(this_doc->refs[0]) free(this_doc->refs[0]); /* move everything downward */ for(i = 0; i < MAX_HISTORY_ITEMS - 1; i++) this_doc->refs[i] = this_doc->refs[i+1]; this_doc->nrefs = MAX_HISTORY_ITEMS-1; } /* sanity check */ if(loc) this_doc->refs[this_doc->nrefs] = strdup(loc); this_doc->current_ref = this_doc->nrefs; this_doc->nrefs++; /* set button sensitivity */ XtSetSensitive(back, current_doc || this_doc->current_ref ? True : False); XtSetSensitive(forward, False); } /***** * Name: storeAnchor * Return Type: void * Description: stores the given href in the visited anchor list of current * document. * In: * href: value to store. * Returns: * nothing. *****/ static void storeAnchor(String href) { int i; DocumentCache *this_doc = NULL; /* sanity check */ if(href == NULL) return; /* pick up current document */ this_doc = &doc_cache[current_doc]; /* check if this location is already present in the visited list */ for(i = 0; i < this_doc->nvisited; i++) if(!(strcmp(this_doc->visited[i], href))) return; /* not present yet, store in visited anchor list */ /* move everything down if list is full */ if(this_doc->nvisited == MAX_HISTORY_ITEMS) { /* free up the first item */ if(this_doc->visited[0]) free(this_doc->visited[0]); /* move everything downward */ for(i = 0; i < MAX_HISTORY_ITEMS - 1; i++) this_doc->visited[i] = this_doc->visited[i+1]; this_doc->nvisited = MAX_HISTORY_ITEMS-1; } /* store this item */ this_doc->visited[this_doc->nvisited] = strdup(href); this_doc->nvisited++; } /***** * Name: loadOrJump * Return Type: void * Description: checks the given href for a file and a possible jump to * a named anchor in this file. * In: * file: name of file to load * loc: location in file to jump to. * store: history storage * Returns: * True upon success, False on failure (file load failed) *****/ static Boolean loadAndOrJump(String file, String loc, Boolean store) { static String prev_file; /* only load a file if it isn't the current one */ if(prev_file == NULL || (file && strcmp(file, prev_file))) { /* do nothing more if the load fails */ if(!(getAndSetFile(file, loc, store))) return(False); } if(prev_file) free(prev_file); prev_file = strdup(file); /* jump to the requested anchor in this file or to top of the document */ if(loc) XmHTMLAnchorScrollToName(html_widgets[0].html, loc); else XmHTMLTextScrollToLine(html_widgets[0].html, 0); return(True); } /***** * Name: readFile * Return Type: void * Description: XmNokCallback handler for the fileSelectionDialog: retrieves * the entered filename and loads it. * In: * w: widget * dialog: widget id of the fileSelectionDialog * cbs: callback data * Returns: * nothing. *****/ static void readFile(Widget widget, Widget dialog, XmFileSelectionBoxCallbackStruct *cbs) { String filename, file; int item; /* remove the fileSelectionDialog */ XtPopdown(XtParent(dialog)); /* get the entered filename */ XmStringGetLtoR(cbs->value, XmSTRING_DEFAULT_CHARSET, &filename); /* sanity check */ if(!filename || !*filename) { if(filename) XtFree(filename); return; } /* get item data */ XtVaGetValues(dialog, XmNuserData, &item, NULL); if(item == FILE_OPEN) { /* find the file */ file = resolveFile(filename); XtFree(filename); if(file == NULL) return; /* load the file, will also update the document cache */ loadAndOrJump(file, NULL, True); XFlush(XtDisplay(widget)); free(file); } else if(item == FILE_SAVEAS) { FILE *fp; /* get parser output */ String buffer = XmHTMLTextGetString(html_widgets[0].html); if(buffer) { if((fp = fopen(filename, "w")) == NULL) perror(filename); else { fputs(buffer, fp); fputs("\n", fp); fclose(fp); } XtFree(buffer); } XtFree(filename); } } /***** * Name: fileCB * Return Type: void * Description: callback for the File->open menu item. Pops up a * fileSelection dialog. * In: * widget: widget id * item: id of selected menu item. * Returns: * nothing. *****/ static void fileCB(Widget widget, int item) { static Widget dialog; switch(item) { case FILE_QUIT: { int i, j; /* clear all text */ XmHTMLTextSetString(html_widgets[0].html, ""); /* * Just in case we have been using a palette, clear it by * disabling dithering. */ XtVaSetValues(html_widgets[0].html, XmNimageMapToPalette, XmDISABLED, NULL); /* kill frame widgets */ for(i = 1; i < MAX_HTML_WIDGETS;i++) { if(html_widgets[i].html) XtDestroyWidget(html_widgets[i].html); if(html_widgets[i].name) free(html_widgets[i].name); if(html_widgets[i].src) free(html_widgets[i].src); } #ifdef DEBUG /* clear fonts */ _XmHTMLUnloadFonts((XmHTMLWidget)html_widgets[0].html); #endif /* kill all doc stuff */ for(j = 0; j < last_doc; j++) { /* free hrefs */ for(i = 0; i < doc_cache[j].nrefs; i++) { if(doc_cache[j].refs[i]) free(doc_cache[j].refs[i]); doc_cache[j].refs[i] = NULL; } /* free image url's */ for(i = 0; i < doc_cache[j].nimages; i++) { free(doc_cache[j].images[i]); doc_cache[j].images[i] = NULL; } /* free visited anchor list */ for(i = 0; i < doc_cache[j].nvisited; i++) { if(doc_cache[j].visited[i]) free(doc_cache[j].visited[i]); doc_cache[j].visited[i] = NULL; } /* free file and path fields */ free(doc_cache[j].file); free(doc_cache[j].path); } /* kill all images */ killImages(); printf("Bye!\n"); exit(EXIT_SUCCESS); } break; case FILE_RAISE: XRaiseWindow(XtDisplay(toplevel), XtWindow(toplevel)); return; case FILE_LOWER: XLowerWindow(XtDisplay(toplevel), XtWindow(toplevel)); return; case FILE_RELOAD: { Cardinal topline = 0; /* get current text position */ XtVaGetValues(html_widgets[0].html, XmNtopLine, &topline, NULL); /* clear current contents to force a *real* reload */ XmHTMLTextSetString(html_widgets[0].html, NULL); /* reload the current file */ getAndSetFile(doc_cache[current_doc].file, NULL, False); /* and restore the current text position. */ XmHTMLTextScrollToLine(html_widgets[0].html, topline); } return; case FILE_INFO: docInfoCB(False); return; case FILE_VIEW: viewSource(toplevel); return; default: break; } if(!dialog) { Arg args[1]; Widget menu = XtParent(widget); XmString xms; /* set the file selection pattern */ xms = XmStringCreateLocalized("*.html"); XtSetArg(args[0], XmNpattern, xms); /* create the dialog */ dialog = XmCreateFileSelectionDialog(menu, "fileOpen", args, 1); /* no longer needed, free it */ XmStringFree(xms); /* * register the dialog itself as callback data, so we can pop it down * when pressed on the OK button. */ XtAddCallback(dialog, XmNokCallback, (XtCallbackProc)readFile, dialog); XtAddCallback(dialog, XmNcancelCallback, (XtCallbackProc)XtUnmanageChild, NULL); } if(item == FILE_SAVEAS) XtVaSetValues(XtParent(dialog), XmNtitle, "Save Document As", NULL); else XtVaSetValues(XtParent(dialog), XmNtitle, "Open a File", NULL); /* save item no as user data */ XtVaSetValues(dialog, XmNuserData, (XtPointer)item, NULL); XtManageChild(dialog); XtPopup(XtParent(dialog), XtGrabNone); XMapRaised(XtDisplay(dialog), XtWindow(XtParent(dialog))); } /***** * Name: armCB * Return Type: void * Description: XmHTML's XmNarmCallback handler * In: * w: widget id; * arg1: client_data, unused; * href_data: callback data; * Returns: * nothing. * Note: * this routine is only used when this example is being run with the -root * option and propagates the received event (available as href_data->event) * to the window manager. *****/ static void armCB(Widget w, XtPointer arg1, XmAnyCallbackStruct *href_data) { XButtonEvent *event; /* security check, armCallback can be added using other methods * than XtAddCallback (editres for example) */ if(root_window) { event = (XButtonEvent*)href_data->event; event->window = DefaultRootWindow(XtDisplay(w)); event->root = DefaultRootWindow(XtDisplay(w)); event->subwindow = DefaultRootWindow(XtDisplay(w)); event->send_event = True; XUngrabPointer(XtDisplay(w), CurrentTime); XSendEvent(XtDisplay(w), DefaultRootWindow(XtDisplay(w)), True, ButtonPressMask, (XEvent *)event); XFlush(XtDisplay(w)); } } static void callClient(URLType url_type, String url) { if(external_client) { char cmd[1024]; if(url_type == ANCHOR_MAILTO) sprintf(cmd, "netscape -remote 'mailto(%s)'", url); if(url_type == ANCHOR_NEWS) sprintf(cmd, "netscape -remote 'news(%s)'", url); else sprintf(cmd, "netscape -remote 'openURL(%s)'", url); if(!(fork())) { if(execl("/bin/sh", "/bin/sh", "-c", cmd, NULL) == -1) { fprintf(stderr, "execl failed (%s)", strerror(errno)); exit(100); } } } } /***** * Name: anchorCB * Return Type: void * Description: XmNactivateCallback handler for the XmHTML widget * In: * w: html widget * arg1: client_data, unused * href_data: anchor data * Returns: * nothing. *****/ static void anchorCB(Widget w, XtPointer arg1, XmHTMLAnchorPtr href_data) { /* see if we have been called with a valid reason */ if(href_data->reason != XmCR_ACTIVATE) return; switch(href_data->url_type) { /* a named anchor */ case ANCHOR_JUMP: { int id; /* see if XmHTML knows this anchor */ if((id = XmHTMLAnchorGetId(w, href_data->href)) != -1) { /* store href in history and visited anchor list... */ storeInHistory(NULL, href_data->href); storeAnchor(href_data->href); /* ...and let XmHTML jump and mark as visited */ href_data->doit = True; href_data->visited = True; return; } return; } break; /* a local file with a possible ID jump */ case ANCHOR_FILE_LOCAL: { String chPtr, file = NULL, loc = NULL; /* store href in visited anchor list */ storeAnchor(href_data->href); /* first see if this anchor contains a jump */ if((chPtr = strstr(href_data->href, "#")) != NULL) { char tmp[1024]; strncpy(tmp, href_data->href, chPtr - href_data->href); tmp[chPtr - href_data->href] = '\0'; /* try to find the file */ file = resolveFile(tmp); } else file = resolveFile(href_data->href); if(file == NULL) return; /***** * All members in the XmHTMLAnchorCallbackStruct are * *pointers* to the contents of the current document. * So, if we will be changing the document, any members of this * structure become INVALID (in this case, any jump address to * a location in a different file). Therefore we *must* save * the members we might need after the document has changed. ******/ if(chPtr) loc = strdup(chPtr); /***** * If we have a target, call the frame loader, else call the * plain document loader. *****/ if(href_data->target) jumpToFrame(file, loc, href_data->target, True); else loadAndOrJump(file, loc, True); free(file); if(loc) free(loc); } break; case ANCHOR_EXEC: if(root_window || allow_exec) { char *ptr; char *cmd=NULL; Debug(("execute: %s\n", href_data->href)); if((ptr = strstr(href_data->href, ":")) != NULL) { ptr++; cmd = strdup(ptr); cmd[strlen(cmd)] = '\0'; if(!(fork())) { if(execl("/bin/sh", "/bin/sh", "-c", cmd, NULL) == -1) { fprintf(stderr, "execl failed (%s)", strerror(errno)); exit(100); } } free(cmd); } } break; /* all other types are unsupported */ case ANCHOR_FILE_REMOTE: fprintf(stderr, "fetch remote file: %s\n", href_data->href); callClient(href_data->url_type, href_data->href); break; case ANCHOR_FTP: fprintf(stderr, "fetch ftp file: %s\n", href_data->href); callClient(href_data->url_type, href_data->href); break; case ANCHOR_HTTP: fprintf(stderr, "fetch http file: %s\n", href_data->href); callClient(href_data->url_type, href_data->href); break; case ANCHOR_GOPHER: fprintf(stderr, "gopher: %s\n", href_data->href); callClient(href_data->url_type, href_data->href); break; case ANCHOR_WAIS: fprintf(stderr, "wais: %s\n", href_data->href); callClient(href_data->url_type, href_data->href); break; case ANCHOR_NEWS: fprintf(stderr, "open newsgroup: %s\n", href_data->href); callClient(href_data->url_type, href_data->href); break; case ANCHOR_TELNET: fprintf(stderr, "open telnet connection: %s\n", href_data->href); callClient(href_data->url_type, href_data->href); break; case ANCHOR_MAILTO: fprintf(stderr, "email to: %s\n", href_data->href); callClient(href_data->url_type, href_data->href); break; case ANCHOR_UNKNOWN: default: fprintf(stderr, "don't know this type of url: %s\n", href_data->href); break; } } /***** * Name: setFrameText * Return Type: void * Description: loads the given file in the given HTML frame; * In: * widget: XmHTML widget id in which to load the file; * filename: file to be loaded; * loc: selected position in the file. If non-NULL this routine will * scroll to the given location. * Returns: * nothing. *****/ static void setFrameText(Widget frame, String filename, String loc) { setBusy(True); if(filename) { String buf, file, mime; file = resolveFile(filename); if(file == NULL || (buf = loadFile(file, &mime)) == NULL) { setBusy(False); return; } /* kill of any outstanding progressive image loading contexts */ XmHTMLImageProgressiveKill(frame); /* set the text in the widget */ XtVaSetValues(frame, XmNvalue, buf, XmNmimeType, mime, NULL); free(buf); free(file); } /* jump to the requested anchor in this file or to top of the document */ if(loc) XmHTMLAnchorScrollToName(frame, loc); else XmHTMLTextScrollToLine(frame, 0); setBusy(False); } /***** * Name: jumpToFrame * Return Type: void * Description: loads the contents of the given file in a named frame * In: * filename: name of file to load; * loc: url of file; * target: name of frame in which to load the file; * store: flag for history storage; * Returns: * nothing. *****/ static void jumpToFrame(String filename, String loc, String target, Boolean store) { int i; /* html_widgets[0] is the master XmHTML Widget, never framed */ for(i = 1; i < MAX_HTML_WIDGETS; i++) { if(html_widgets[i].active && !(strcmp(html_widgets[i].name, target))) { /* * Load new file into frame if it's not the same as the current * src value for this frame. */ if(html_widgets[i].src && strcmp(html_widgets[i].src, filename)) { free(html_widgets[i].src); html_widgets[i].src = strdup(filename); setFrameText(html_widgets[i].html, filename, loc); } else /* same file, jump to requested location */ setFrameText(html_widgets[i].html, NULL, loc); return; } } /* frame not found, use the toplevel HTML widget */ if(i == MAX_HTML_WIDGETS) loadAndOrJump(filename, loc, store); } /***** * Name: focusCB * Return Type: void * Description: callback for XmHTML's focusCallback and losingFocusCallback * In: * w: widget id; * client_..: data registered on this callback, unused; * cbs: callback data provided by calling widget; * Returns: * *****/ static void focusCB(Widget w, XtPointer client_data, XmAnyCallbackStruct *cbs) { if(cbs->reason == XmCR_FOCUS) { /* set active widget */ } else if(cbs->reason == XmCR_LOSING_FOCUS) { } } /* external gif decoder */ #include "gif_decode.c" /***** * Name: frameCB * Return Type: void * Description: callback for XmHTML's XmNframeCallback * In: * w: widget id; * arg1: client_data, unused; * cbs: data about the frame being created/destroyed/notified of * creation. * Returns: * nothing. *****/ static void frameCB(Widget w, XtPointer arg1, XmHTMLFramePtr cbs) { int i; if(cbs->reason == XmCR_HTML_FRAME) { Widget html = cbs->html; /* find the first free slot where we can insert this frame */ for(i = 0; i < MAX_HTML_WIDGETS;i++) if(!html_widgets[i].active) break; /* a frame always has a name */ html_widgets[i].name = strdup(cbs->name); if(cbs->src) html_widgets[i].src = resolveFile(cbs->src); /* anchor callback */ XtAddCallback(html, XmNactivateCallback, (XtCallbackProc)anchorCB, NULL); /* arm callback */ if(root_window) XtAddCallback(html, XmNarmCallback, (XtCallbackProc)armCB, NULL); /* anchor tracking callback */ XtAddCallback(html, XmNanchorTrackCallback, (XtCallbackProc)trackCB, NULL); /* HTML document callback */ XtAddCallback(html, XmNdocumentCallback, (XtCallbackProc)docCB, NULL); /* focus callbacks so we can track which widget is current */ XtAddCallback(html, XmNfocusCallback, (XtCallbackProc)focusCB, NULL); XtAddCallback(html, XmNlosingFocusCallback, (XtCallbackProc)focusCB, NULL); /* set other things we want to have */ XtVaSetValues(html, XmNanchorVisitedProc, testAnchor, XmNimageProc, loadImage, XmNprogressiveReadProc, getImageData, XmNprogressiveEndProc, endImageData, #ifdef HAVE_GIF_CODEC XmNdecodeGIFProc, decodeGIFImage, #endif /* propagate current defaults */ XmNanchorButtons, html_config[OPTIONS_ANCHOR_BUTTONS].value, XmNhighlightOnEnter, html_config[OPTIONS_HIGHLIGHT_ON_ENTER].value, XmNenableBadHTMLWarnings, html_config[OPTIONS_DISABLE_WARNINGS].value, XmNstrictHTMLChecking, html_config[OPTIONS_ENABLE_STRICT_HTML32].value, XmNenableBodyColors, html_config[OPTIONS_ENABLE_BODYCOLORS].value, XmNenableBodyImages, html_config[OPTIONS_ENABLE_BODYIMAGES].value, XmNenableDocumentColors, html_config[OPTIONS_ENABLE_DOCUMENT_COLORS].value, XmNenableDocumentFonts, html_config[OPTIONS_ENABLE_DOCUMENT_FONTS].value, XmNenableOutlining, html_config[OPTIONS_ENABLE_OUTLINING].value, XmNfreezeAnimations, html_config[OPTIONS_FREEZE_ANIMATIONS].value, XmNimageEnable, html_config[OPTIONS_ENABLE_IMAGES].value, NULL); /* store widget id */ html_widgets[i].html = html; html_widgets[i].active = True; /* set source text */ setFrameText(html_widgets[i].html, html_widgets[i].src, NULL); return; } if(cbs->reason == XmCR_HTML_FRAMECREATE) { /* see if we have an inactive frame in our frame cache */ for(i = 0; i < MAX_HTML_WIDGETS;i++) if(!html_widgets[i].active && !html_widgets[i].used) break; /* we have an available slot */ if(i != MAX_HTML_WIDGETS && html_widgets[i].html != NULL) { cbs->doit = False; cbs->html = html_widgets[i].html; /* this slot is being used */ html_widgets[i].used = True; } /* * this is the appropriate place for doing frame reuse: set * the doit field in the callback structure to False and set the * id of a *HTML* widget in the html field. */ return; } if(cbs->reason == XmCR_HTML_FRAMEDESTROY) { int freecount = 0; /* * this is the appropriate place for keeping this widget: just set * the doit field to false, update the frame cache (if this frame is * going to be reused, it's name and src value will probably change). */ for(i = 0; i < MAX_HTML_WIDGETS; i++) { if(!html_widgets[i].active) freecount++; if(html_widgets[i].html == cbs->html) { freecount++; /* a frame has always got a name */ free(html_widgets[i].name); if(html_widgets[i].src) free(html_widgets[i].src); html_widgets[i].name = NULL; html_widgets[i].src = NULL; /* we keep up to three frames in memory */ if(freecount > 3) html_widgets[i].html = NULL; html_widgets[i].active = False; html_widgets[i].used = False; break; } } if(freecount < 4) cbs->doit = False; return; } /* do nothing */ return; } /***** * Name: trackCB * Return Type: void * Description: displays the URL when the pointer is moved over an anchor * In: * html: owner of this callback * arg1: client_data, unused * href_data: call_data, anchor callback structure. * Returns: * nothing *****/ static void trackCB(Widget w, XtPointer arg1, XmHTMLAnchorPtr href_data) { XmString xms; static char anchor_label[128]; static Boolean cleared = False; XmHTMLInfoPtr info; XButtonEvent event; int len; /* see if we have been called with a valid reason */ if(href_data->reason != XmCR_HTML_ANCHORTRACK) return; /* a valid anchor, eg, moving into an anchor */ if(href_data->href) { /* see if the new label isn't the previous label */ if(anchor_label[0] != '\0') { /* first see if an alternate title has been given */ if(href_data->title && !strcmp(anchor_label, href_data->title)) return; if(!strcmp(anchor_label, href_data->href)) return; } /* create a new label */ if(href_data->title) { strncpy(anchor_label, href_data->title, 128); len = strlen(href_data->title); } else { strncpy(anchor_label, href_data->href, 128); len = strlen(href_data->href); } /* don't overflow */ anchor_label[len > 127 ? 127 : len] = '\0'; xms = XmStringCreateLocalized(anchor_label); XtVaSetValues(label, XmNlabelString, xms, NULL); XmStringFree(xms); cleared = False; } /* a valid anchor, eg, moving away from an anchor */ else { anchor_label[0] = '\0'; /* invalidate previous anchor selection */ if(!cleared) { /* remove previous label */ xms = XmStringCreateLocalized(" "); XtVaSetValues(label, XmNlabelString, xms, NULL); XmStringFree(xms); } cleared = True; } /* image dialog not available, ignore */ if(!html_config[OPTIONS_FANCY_TRACKING].value || !image_dialog || !XtIsManaged(image_dialog)) return; /* get the info for the selected position */ event = href_data->event->xbutton; info = XmHTMLXYToInfo(w, event.x, event.y); /* not over an image anchor or still the same anchor, ignore */ if(info == NULL || info->image == NULL || (preview_image && !strcmp(info->image->url, preview_image->file))) return; /* We moved onto a new image, flip it */ showImageInfo(info->image); } /***** * Name: metaListCB * Return Type: void * Description: callback for the list in the document information dialog. * Displays the data associated with the selected item in the * list. * In: * w: list widget id; * edit: widget id of text widget in which data will be displayed. * Returns: * nothing. *****/ static void metaListCB(Widget w, Widget edit) { int *pos_list; /* selected list position */ int pos_cnt, selected; /* no of selected items */ XmHTMLMetaDataPtr meta_data = NULL; if(!(XmListGetSelectedPos(w, &pos_list, &pos_cnt))) /* no item selected */ return; selected = pos_list[0]; /* list positions start at 1 instead of zero, so be sure to adjust */ selected--; /* get meta data out of the list's userData */ XtVaGetValues(w, XmNuserData, &meta_data, NULL); /* very serious error */ if(meta_data == NULL) { fprintf(stderr, "Could not retrieve meta-data from userData field!\n"); free(pos_list); return; } /* and put the selected string in the edit field */ XmTextSetString(edit, meta_data[selected].content); free(pos_list); } static String getCharset(String content) { String ptr, start; int len = 0; static char this_set[128]; /* * We possible have a charset spec in here. Check * for it */ if((ptr = strstr(content, "charset")) != NULL) { ptr+= 7; /* move past "charset" */ /* skip until we hit = */ for(;*ptr && *ptr != '='; ptr++); ptr++; /* skip all spaces */ for(;*ptr && isspace(*ptr); ptr++); /* * now count how many chars we have. The charset spec is terminated * by a a quote */ start = ptr; while(*ptr && *ptr != '\"') { len++; ptr++; } if(len) { strncpy(this_set, start, len); this_set[len] = '\0'; /* nullify */ /* * if we don't have a - in charset, we *must* append -* to make * it a valid XmNcharset spec. */ if(!(strstr(this_set, "-"))) strcat(this_set, "-*"); return(this_set); } } return(NULL); } static void viewSource(Widget w) { static Widget viewer, html; String source; if(viewer == NULL) { viewer = XmCreateFormDialog(toplevel, "DocSource", NULL, 0); XtVaSetValues(XtParent(viewer), XtNtitle, "Document Source", NULL); html = XtVaCreateManagedWidget("sourceView", xmHTMLWidgetClass, viewer, XmNtopAttachment, XmATTACH_FORM, XmNleftAttachment, XmATTACH_FORM, XmNbottomAttachment, XmATTACH_FORM, XmNrightAttachment, XmATTACH_FORM, XmNmarginWidth, 5, XmNmarginHeight, 5, NULL); } if((source = XmHTMLTextGetSource(html_widgets[0].html)) == NULL) return; XtVaSetValues(html, XmNvalue, source, XmNmimeType, "text/plain", NULL); /* put on screen */ XtManageChild(viewer); XMapRaised(XtDisplay(viewer), XtWindow(viewer)); } /***** * Name: docInfoCB * Return Type: void * Description: shows a dialog with document information. This routine gets * activated when the file->document info menu item is selected. * In: * font_only: check only for a possible font spec in the meta information. * Returns: * nothing. *****/ static void docInfoCB(Boolean font_only) { static Widget title_label, author_label, base_label, doctype_label; static Widget list, edit; XmString xms; static XmHTMLHeadAttributes head_info; Boolean have_info = False; String tmp; int i, argc = 0; Arg args[5]; if(font_only) { char this_font[128]; String this_charset = NULL; this_font[0] = '\0'; /* get meta info for a charset and/or font spec */ if(XmHTMLGetHeadAttributes(html_widgets[0].html, &head_info, HeadMeta)) { if(head_info.num_meta) { for(i = 0; i < head_info.num_meta; i++) { tmp = (head_info.meta[i].http_equiv ? head_info.meta[i].http_equiv : head_info.meta[i].name); if(!strcmp(tmp, "font")) { sprintf(this_font, "*-%s-normal-*", head_info.meta[i].content); } else if(!strcmp(tmp, "content-type") && this_charset == NULL) { this_charset = getCharset(head_info.meta[i].content); } } } } /***** * have we been told to set a new font? * Please note that in a real world application it should be checked * if the requested font is available. XmHTML silently ignores this * kind of errors. *****/ if(*this_font) { /* font changed */ if(strcmp(this_font, current_font)) { strcpy(current_font, this_font); current_font[strlen(this_font)] = '\0'; Debug(("docInfoCB, setting XmNfontFamily to %s\n", current_font)); XtSetArg(args[argc], XmNfontFamily, current_font); argc++; } /* still the same, don't touch it */ } else if(strcmp(current_font, default_font)) { /* reset default font */ strcpy(current_font, default_font); current_font[strlen(default_font)] = '\0'; Debug(("docInfoCB, setting XmNfontFamily to %s\n", current_font)); XtSetArg(args[argc], XmNfontFamily, current_font); argc++; } /***** * have we been told to set a new character set? * Please note that a real world app *MUST* check if the requested * character set is available. XmHTML silently ignores any errors * resulting from an invalid charset, it just falls back to whatever * the X server provides. * To make these checks really consistent, it should also be verified * that the current font family is still valid if the charset is * changed. XmHTML's font allocation routines will almost *never* * fail on font allocations: if a font can not be found in the requested * character set it will use the default charset supplied by the X * server. If font allocation still fails after this, it will wildcard * the fontfamily and try again (this is done so XmHTML will always have * a default font available). If this also fails (which is almost * impossible) XmHTML will exit your application: if it can't find * any fonts at all, why keep on running? *****/ if(this_charset) { /* charset changed */ if(strcmp(current_charset, this_charset)) { /* we currently only known koi8 and iso8859-1 */ if(strstr(this_charset, "koi8")) { /* save charset */ strcpy(current_charset, this_charset); current_charset[strlen(this_charset)] = '\0'; /* koi8 has cronyx for its foundry */ strcpy(current_font, "cronyx-times-*-*\0"); argc = 0; /* overrides a fontspec */ Debug(("docInfoCB, setting XmNcharset to %s\n", current_charset)); XtSetArg(args[argc], XmNfontFamily, current_font); argc++; XtSetArg(args[argc], XmNcharset, current_charset); argc++; } else if(!strstr(this_charset, "iso8859-1")) { fprintf(stderr, "Warning: character set %s unsupported\n", current_charset); } } /* still the same, don't touch it */ } else if(strcmp(current_charset, default_charset)) { strcpy(current_charset, default_charset); current_charset[strlen(default_charset)] = '\0'; XtSetArg(args[argc], XmNcharset, current_charset); argc++; } /* plop'em in */ if(argc) XtSetValues(html_widgets[0].html, args, argc); /***** * Tell XmHTML to clear everything. This is not really required but * adviseable since XmHTML will only clear the fields that have * been requested. For example, in all subsequent calls to the above * XmHTMLGetHeadAttributes call, XmHTML will first clear the stored * meta info before filling it again. *****/ XmHTMLGetHeadAttributes(html_widgets[0].html, &head_info, HeadClear); return; } /***** * Get information from the current document. XmHTML will * take care of replacing the requested members when they have been used * before. When no is available this function returns false * Note: the value of the member is always returned if there * is one, even if GetHeadAttributes() returns False. *****/ have_info = XmHTMLGetHeadAttributes(html_widgets[0].html, &head_info, HeadDocType|HeadTitle|HeadBase|HeadMeta); if(!info_dialog) { Widget sep, rc1, rc2, fr1, fr2; Arg args[20]; int argc = 0; info_dialog = XmCreateFormDialog(toplevel, "documentAttributes", NULL, 0); XtVaSetValues(XtParent(info_dialog), XtNtitle, "Document Attributes", NULL); fr1 = XtVaCreateManagedWidget("documentTitleForm", xmFormWidgetClass, info_dialog, XmNtopAttachment, XmATTACH_FORM, XmNleftAttachment, XmATTACH_FORM, XmNrightAttachment, XmATTACH_FORM, NULL); sep = XtVaCreateManagedWidget("documentSeperator", xmSeparatorGadgetClass, info_dialog, XmNorientation, XmHORIZONTAL, XmNtopAttachment, XmATTACH_WIDGET, XmNtopWidget, fr1, XmNleftAttachment, XmATTACH_FORM, XmNrightAttachment, XmATTACH_FORM, XmNleftOffset, 10, XmNrightOffset, 10, XmNtopOffset, 10, NULL); fr2 = XtVaCreateManagedWidget("documentTitleForm", xmFormWidgetClass, info_dialog, XmNtopAttachment, XmATTACH_WIDGET, XmNtopWidget, sep, XmNleftAttachment, XmATTACH_FORM, XmNrightAttachment, XmATTACH_FORM, XmNbottomAttachment, XmATTACH_FORM, NULL); rc1 = XtVaCreateManagedWidget("documentTitleLeftRow", xmRowColumnWidgetClass, fr1, XmNtopAttachment, XmATTACH_FORM, XmNleftAttachment, XmATTACH_FORM, XmNbottomAttachment, XmATTACH_FORM, XmNpacking, XmPACK_COLUMN, XmNorientation, XmVERTICAL, XmNnumColumns, 1, XmNtopOffset, 10, XmNleftOffset, 10, XmNrightOffset, 10, NULL); XtVaCreateManagedWidget("Document Title:", xmLabelGadgetClass, rc1, XmNalignment, XmALIGNMENT_BEGINNING, NULL); XtVaCreateManagedWidget("Author:", xmLabelGadgetClass, rc1, XmNalignment, XmALIGNMENT_BEGINNING, NULL); XtVaCreateManagedWidget("Document type:", xmLabelGadgetClass, rc1, XmNalignment, XmALIGNMENT_BEGINNING, NULL); XtVaCreateManagedWidget("Base Location:", xmLabelGadgetClass, rc1, XmNalignment, XmALIGNMENT_BEGINNING, NULL); rc2 = XtVaCreateManagedWidget("documentTitleRightRow", xmRowColumnWidgetClass, fr1, XmNtopAttachment, XmATTACH_FORM, XmNleftAttachment, XmATTACH_WIDGET, XmNleftWidget, rc1, XmNrightAttachment, XmATTACH_FORM, XmNbottomAttachment, XmATTACH_FORM, XmNpacking, XmPACK_COLUMN, XmNorientation, XmVERTICAL, XmNnumColumns, 1, XmNtopOffset, 10, XmNleftOffset, 10, XmNrightOffset, 10, NULL); title_label = XtVaCreateManagedWidget("docTitle", xmLabelGadgetClass, rc2, XmNalignment, XmALIGNMENT_BEGINNING, NULL); author_label = XtVaCreateManagedWidget("docAuthor", xmLabelGadgetClass, rc2, XmNalignment, XmALIGNMENT_BEGINNING, NULL); doctype_label = XtVaCreateManagedWidget("docType", xmLabelGadgetClass, rc2, XmNalignment, XmALIGNMENT_BEGINNING, NULL); base_label = XtVaCreateManagedWidget("docBase", xmLabelGadgetClass, rc2, XmNalignment, XmALIGNMENT_BEGINNING, NULL); XtSetArg(args[argc], XmNlistSizePolicy, XmRESIZE_IF_POSSIBLE); argc++; XtSetArg(args[argc], XmNvisibleItemCount, 5); argc++; XtSetArg(args[argc], XmNleftAttachment, XmATTACH_FORM); argc++; XtSetArg(args[argc], XmNtopAttachment, XmATTACH_FORM); argc++; XtSetArg(args[argc], XmNbottomAttachment, XmATTACH_FORM); argc++; XtSetArg(args[argc], XmNleftOffset, 10); argc++; XtSetArg(args[argc], XmNtopOffset, 10); argc++; XtSetArg(args[argc], XmNbottomOffset, 10); argc++; list = XmCreateScrolledList(fr2, "metaList", args, argc); argc = 0; XtSetArg(args[argc], XmNscrollBarDisplayPolicy, XmAS_NEEDED); argc++; XtSetArg(args[argc], XmNscrollingPolicy, XmAUTOMATIC); argc++; XtSetArg(args[argc], XmNleftAttachment, XmATTACH_WIDGET); argc++; XtSetArg(args[argc], XmNleftWidget, list); argc++; XtSetArg(args[argc], XmNtopAttachment, XmATTACH_FORM); argc++; XtSetArg(args[argc], XmNbottomAttachment, XmATTACH_FORM); argc++; XtSetArg(args[argc], XmNrightAttachment, XmATTACH_FORM); argc++; XtSetArg(args[argc], XmNleftOffset, 10); argc++; XtSetArg(args[argc], XmNrightOffset, 10); argc++; XtSetArg(args[argc], XmNtopOffset, 10); argc++; XtSetArg(args[argc], XmNbottomOffset, 10); argc++; edit = XmCreateScrolledText(fr2, "metaEdit", args, argc); argc = 0; XtSetArg(args[argc], XmNeditable, False); argc++; XtSetArg(args[argc], XmNcolumns, 35); argc++; XtSetArg(args[argc], XmNrows, 5); argc++; XtSetArg(args[argc], XmNwordWrap, True); argc++; XtSetArg(args[argc], XmNeditMode, XmMULTI_LINE_EDIT); argc++; XtSetArg(args[argc], XmNscrollHorizontal, False); argc++; XtSetArg(args[argc], XmNscrollVertical, False); argc++; XtSetValues(edit, args, argc); /* single selection callback on the list */ XtAddCallback(list, XmNbrowseSelectionCallback, (XtCallbackProc)metaListCB, edit); XtManageChild(list); XtManageChild(edit); } /* delete all list items */ XmListDeleteAllItems(list); /* clear all text in the edit window */ XmTextSetString(edit, NULL); if(head_info.doctype) xms = XmStringCreateLocalized(head_info.doctype); else xms = XmStringCreateLocalized(""); XtVaSetValues(doctype_label, XmNlabelString, xms, NULL); XmStringFree(xms); if(have_info) { if(head_info.title) xms = XmStringCreateLocalized(head_info.title); else xms = XmStringCreateLocalized(""); XtVaSetValues(title_label, XmNlabelString, xms, NULL); XmStringFree(xms); if(head_info.base) xms = XmStringCreateLocalized(head_info.base); else xms = XmStringCreateLocalized(""); XtVaSetValues(base_label, XmNlabelString, xms, NULL); XmStringFree(xms); xms = NULL; /* * No need to check for font or charset in meta info, that's * already been done from within the documentCallback. */ if(head_info.num_meta) { for(i = 0; i < head_info.num_meta; i++) { tmp = (head_info.meta[i].http_equiv ? head_info.meta[i].http_equiv : head_info.meta[i].name); /* pick out author */ if(!strcmp(tmp, "author")) xms = XmStringCreateLocalized(head_info.meta[i].content); } } if(xms == NULL) xms = XmStringCreateLocalized(""); XtVaSetValues(author_label, XmNlabelString, xms, NULL); XmStringFree(xms); /* string table */ if(head_info.num_meta) { XmStringTable strs; strs =(XmStringTable)malloc(head_info.num_meta*sizeof(XmString*)); for(i = 0; i < head_info.num_meta; i++) { if(head_info.meta[i].http_equiv) strs[i] = XmStringCreateLocalized(head_info.meta[i].http_equiv); else strs[i] = XmStringCreateLocalized(head_info.meta[i].name); } XtVaSetValues(list, XmNitemCount, head_info.num_meta, XmNitems, strs, XmNuserData, (XtPointer)head_info.meta, NULL); for(i = 0; i < head_info.num_meta; i++) XmStringFree(strs[i]); free(strs); } } else { xms = XmStringCreateLocalized(""); XtVaSetValues(title_label, XmNlabelString, xms, NULL); XmStringFree(xms); xms = XmStringCreateLocalized(""); XtVaSetValues(author_label, XmNlabelString, xms, NULL); XmStringFree(xms); xms = XmStringCreateLocalized(""); XtVaSetValues(base_label, XmNlabelString, xms, NULL); XmStringFree(xms); } /* put on screen */ XtManageChild(info_dialog); XMapRaised(XtDisplay(info_dialog), XtWindow(info_dialog)); } /***** * Name: docCB * Return Type: void * Description: displays current document state as given by XmHTML. * In: * html: owner of this callback * arg1: client_data, unused * cbs: call_data, documentcallback structure. * Returns: * nothing * Note: * the XmNdocumentCallback is an excellent place for setting several * formatting resources: XmHTML triggers this callback when the parser * has finished, but *before* doing any formatting. As an example, we * check the information contained in the document head for a font * specification: we call docInfoCB with the font_only arg set to True. *****/ static void docCB(Widget w, XtPointer arg1, XmHTMLDocumentPtr cbs) { XmString xms; char doc_label[128]; Pixel my_pix = (Pixel)0; static Pixel red, green; /* see if we have been called with a valid reason */ if(cbs->reason != XmCR_HTML_DOCUMENT) return; /* * If we are being notified of the results of another pass on the loaded * document, only check whether the generated parser tree is balanced and * don't update the labels since the callback data is the result of a * modified document. * * XmHTML's document verification and repair routines are capable of * creating a verified, properly balanced and HTML conforming document * from even the most horrible non-HTML conforming documents! * * And when the XmNstrictHTMLChecking resource has been set to True, these * routines are also bound to make the document HTML 3.2 conformant as well. */ if(cbs->pass_level) { /* * Allow up to two iterations on the current document (remember that * the document has already been checked twice when pass_level == 1). * XmHTML sets the redo field to True whenever the parser tree is * unbalanced, so it needs to be set to False when the allowed number * of iterations has been reached. * * The results of displaying a document with an unbalanced parser tree * are undefined however and can lead to some weird markup results. */ if(!cbs->balanced && cbs->pass_level < 2) return; cbs->redo = False; /* done parsing, check if a font is given in the meta spec. */ docInfoCB(True); return; } if(!red) red = XmHTMLAllocColor((Widget)html_widgets[0].html, "Red", BlackPixelOfScreen(XtScreen(toplevel))); if(!green) green = XmHTMLAllocColor((Widget)html_widgets[0].html, "Green", WhitePixelOfScreen(XtScreen(toplevel))); if(cbs->html32) { sprintf(doc_label, "HTML 3.2"); my_pix = green; } else { sprintf(doc_label, "Bad HTML 3.2"); my_pix = red; } xms = XmStringCreateLocalized(doc_label); XtVaSetValues(html32, XmNbackground, my_pix, XmNlabelString, xms, NULL); XmStringFree(xms); if(cbs->verified) { sprintf(doc_label, "Verified"); my_pix = green; } else { sprintf(doc_label, "Unverified"); my_pix = red; } xms = XmStringCreateLocalized(doc_label); XtVaSetValues(verified, XmNbackground, my_pix, XmNlabelString, xms, NULL); XmStringFree(xms); /* * default processing here. If the parser tree isn't balanced * (cbs->balanced == False) and you want to prevent XmHTML from making * another pass on the current document, set the redo field to false. */ if(cbs->balanced) { /* check meta info for a possible font spec */ docInfoCB(True); } } /***** * Name: linkCB * Return Type: void * Description: XmHTML's XmNlinkCallback handler * In: * w: widget id; * arg1: client_data, unused; * cbs: link data found in current document. * Returns: * nothing, but copies the link data to an internal structure which is used * for displaying a site-navigation bar. *****/ static void linkCB(Widget w, XtPointer arg1, XmHTMLLinkPtr cbs) { int i, j; /* free previous document links */ for(i = 0; i < LINK_LAST; i++) { if(document_links[i].have_data) { /* we always have a href */ free(document_links[i].href); /* but title is optional */ if(document_links[i].title) free(document_links[i].title); document_links[i].href= NULL; document_links[i].title = NULL; } document_links[i].have_data = False; } /***** * Since this callback gets triggered for every document, this is also * the place for updating the info dialog (if it's up that is). *****/ if(info_dialog && XtIsManaged(info_dialog)) docInfoCB(False); /***** * if the current document doesn't have any links, unmanage the button * and dialog and return. *****/ if(cbs->num_link == 0) { /* might not have been created yet */ if(link_dialog != NULL) XtUnmanageChild(link_dialog); XtUnmanageChild(link_button); return; } /* store links in this document */ for(i = 0; i < LINK_LAST; i++) { for(j = 0; j < cbs->num_link; j++) { /* kludge for the mailto */ String tmp = (i == 0 ? "made" : link_labels[i]); /***** * url is mandatory and so is one of rel or rev. Although both * of them can be present, we prefer a rel over a rev. *****/ if(!cbs->link[j].url || (!cbs->link[j].rel && !cbs->link[j].rev)) continue; if((cbs->link[j].rel && my_strcasestr(cbs->link[j].rel, tmp)) || (cbs->link[j].rev && my_strcasestr(cbs->link[j].rev, tmp))) { document_links[i].have_data = True; document_links[i].link_type = cbs->link[j].rel ? 1 : 0; /* we always have this */ document_links[i].href = strdup(cbs->link[j].url); /* title is optional */ if(cbs->link[j].title) document_links[i].title = strdup(cbs->link[j].title); } } } /* if the dialog is already up, update the buttons */ if(link_dialog && XtIsManaged(link_dialog)) { for(i = 0; i < LINK_LAST; i++) { if(document_links[i].have_data) XtSetSensitive(link_buttons[i], True); else XtSetSensitive(link_buttons[i], False); } /* and make sure everything is up and displayed */ XmUpdateDisplay(link_dialog); } /* manage the button so user can see the site structure of this doc. */ XtManageChild(link_button); } /***** * Name: imageViewUnmapCB * Return Type: void * Description: callback for the XmNunmapCallback callback resource in the * image preview dialog (popped up by showImageInfo) * In: * w: widget being unmapped. * Returns: * nothing. *****/ static void imageViewUnmapCB(Widget w) { if(preview_image) XmImageDestroy(preview_image); preview_image = NULL; } /***** * Name: showImageInfo * Return Type: void * Description: display a dialog with some information about the selected * image * In: * info: info to be displayed * Returns: * nothing. * Note: * This routine also demonstrates how to use the XmImage data type for * creating a preview image from the given XmImageInfo structure. *****/ static void showImageInfo(XmImageInfo *info) { /* various labels */ static Widget url_label, size_label, dimension_label, npixels_label; static Widget type_label, pixmap_label, trans_label, depth_label; static Widget cache_label; XmString xms; Dimension w, h; char tmp[64]; static XmImageConfig config; /* for setting XmImage options */ Pixel bg_pixel = 0; int cache_size, nobjects; setBusy(True); if(!image_dialog) { Widget rc1, rc2; image_dialog = XmCreateFormDialog(toplevel, "imageView", NULL, 0); /* add an unmap callback so we can destroy the preview_image */ XtAddCallback(image_dialog, XmNunmapCallback, (XtCallbackProc)imageViewUnmapCB, NULL); XtVaSetValues(XtParent(image_dialog), XtNtitle, "View Image Details", NULL); /* a rowcol for the labels */ rc1 = XtVaCreateManagedWidget("imageRowColumn", xmRowColumnWidgetClass, image_dialog, XmNtopAttachment, XmATTACH_FORM, XmNleftAttachment, XmATTACH_FORM, XmNbottomAttachment, XmATTACH_FORM, XmNspacing, 0, XmNpacking, XmPACK_COLUMN, XmNorientation, XmVERTICAL, XmNnumColumns, 1, XmNtopOffset, 10, XmNleftOffset, 10, XmNbottomOffset, 10, NULL); XtVaCreateManagedWidget("Location:", xmLabelGadgetClass, rc1, XmNalignment, XmALIGNMENT_BEGINNING, NULL); XtVaCreateManagedWidget("Image Type:", xmLabelGadgetClass, rc1, XmNalignment, XmALIGNMENT_BEGINNING, NULL); XtVaCreateManagedWidget("Uncompressed Size:", xmLabelGadgetClass, rc1, XmNalignment, XmALIGNMENT_BEGINNING, NULL); XtVaCreateManagedWidget("Image Dimensions:", xmLabelGadgetClass, rc1, XmNalignment, XmALIGNMENT_BEGINNING, NULL); XtVaCreateManagedWidget("Number of Colors:", xmLabelGadgetClass, rc1, XmNalignment, XmALIGNMENT_BEGINNING, NULL); XtVaCreateManagedWidget("Depth:", xmLabelGadgetClass, rc1, XmNalignment, XmALIGNMENT_BEGINNING, NULL); XtVaCreateManagedWidget("Transparent:", xmLabelGadgetClass, rc1, XmNalignment, XmALIGNMENT_BEGINNING, NULL); XtVaCreateManagedWidget("Image Cache:", xmLabelGadgetClass, rc1, XmNalignment, XmALIGNMENT_BEGINNING, NULL); /* a rowcol for the fill-in fields */ rc2 = XtVaCreateManagedWidget("imageRowColumn", xmRowColumnWidgetClass, image_dialog, XmNtopAttachment, XmATTACH_FORM, XmNleftAttachment, XmATTACH_WIDGET, XmNleftWidget, rc1, XmNbottomAttachment, XmATTACH_FORM, XmNspacing, 0, XmNpacking, XmPACK_COLUMN, XmNorientation, XmVERTICAL, XmNnumColumns, 1, XmNtopOffset, 10, XmNleftOffset, 10, XmNbottomOffset, 10, NULL); xms = XmStringCreateLocalized(""); url_label = XtVaCreateManagedWidget("url", xmLabelGadgetClass, rc2, XmNalignment, XmALIGNMENT_END, XmNlabelString, xms, NULL); type_label = XtVaCreateManagedWidget("type", xmLabelGadgetClass, rc2, XmNalignment, XmALIGNMENT_END, XmNlabelString, xms, NULL); size_label = XtVaCreateManagedWidget("size", xmLabelGadgetClass, rc2, XmNalignment, XmALIGNMENT_END, XmNlabelString, xms, NULL); dimension_label = XtVaCreateManagedWidget("dimensions", xmLabelGadgetClass, rc2, XmNalignment, XmALIGNMENT_END, XmNlabelString, xms, NULL); npixels_label = XtVaCreateManagedWidget("npixels", xmLabelGadgetClass, rc2, XmNalignment, XmALIGNMENT_END, XmNlabelString, xms, NULL); depth_label = XtVaCreateManagedWidget("depth", xmLabelGadgetClass, rc2, XmNalignment, XmALIGNMENT_END, XmNlabelString, xms, NULL); trans_label = XtVaCreateManagedWidget("transparent", xmLabelGadgetClass, rc2, XmNalignment, XmALIGNMENT_END, XmNlabelString, xms, NULL); cache_label = XtVaCreateManagedWidget("cacheInfo", xmLabelGadgetClass, rc2, XmNalignment, XmALIGNMENT_END, XmNlabelString, xms, NULL); XmStringFree(xms); /* and the display area */ pixmap_label = XtVaCreateManagedWidget("imageViewer", xmLabelGadgetClass, image_dialog, XmNtopAttachment, XmATTACH_FORM, XmNleftAttachment, XmATTACH_WIDGET, XmNleftWidget, rc2, XmNrightAttachment, XmATTACH_FORM, XmNbottomAttachment, XmATTACH_FORM, XmNtopOffset, 10, XmNleftOffset, 10, XmNrightOffset, 10, XmNbottomOffset, 10, XmNlabelType, XmPIXMAP, NULL); } /* destroy an existing image. */ if(preview_image) { XmImageDestroy(preview_image); preview_image = NULL; } /***** * create the image we want to display. Try to scale it down to a 128x128 * icon and keep the aspect ratio. * swidth & sheight are the *real* image dimensions while * width & height are the current image dimensions. If they differ the * image has been scaled by XmHTML. For our preview image we use the * current image dimensions and show the scaling factor in the dialog box * itself. We could use the real dimensions but that would lead to a little * loss of data as the original data is no longer available. * * Note that we always set an explicit value for width & height: if they * left at zero, XmImageCreate will attempt to use the real image dimensions. *****/ w = info->width; h = info->height; /* sanity check */ if(h) { if(w > 128 || h > 128) { if(w > h) { h = (Dimension)(h*128/w); w = 128; } else { w = (Dimension)(w*128/h); h = 128; } } } /* * create an XmImage from the obtained XmImageInfo. We don't run animations * in the image info dialog, so we just want an image from the first frame * in an animation. Saves time and resources. * * XmImage configuration stuff we set in here: * - maximum image colors; * - enable quantization of images with more than allowed colors, and allow * Floyd-Steinberg dithering if it should be required; * - animation frame selection; * - background substitution; */ /* get background pixel */ XtVaGetValues(image_dialog, XmNbackground, &bg_pixel, NULL); /* set XmImageConfig options */ config.ncolors = 216; /* has no effect for CreateFromInfo */ config.which_frames = FirstFrame; config.bg_color = bg_pixel; config.flags = ImageQuantize|ImageFSDither|ImageBackground| ImageFrameSelect; /* create preview image */ preview_image = XmImageCreateFromInfo(toplevel, info, w, h, &config); /* create the various labels */ /* name/location of this image */ xms = XmStringCreateSimple(collapseURL(info->url)); XtVaSetValues(url_label, XmNlabelString, xms, NULL); XmStringFree(xms); /* type of image */ xms = XmStringCreateSimple(image_types[info->type]); XtVaSetValues(type_label, XmNlabelString, xms, NULL); XmStringFree(xms); /* in-memory size of this image/animation */ sprintf(tmp, "%i bytes", getInfoSize((XtPointer)info, NULL)); xms = XmStringCreateSimple(tmp); XtVaSetValues(size_label, XmNlabelString, xms, NULL); XmStringFree(xms); /* * Show image dimensions as specified within the image itself. * image->width and image->height contain the dimensions of the generated * pixmap, which will be the same as swidth and sheight *unless* we've * requested scaling in either direction. */ /* image was scaled */ if(info->swidth != info->width || info->sheight != info->height) sprintf(tmp, "%ix%i pixels (scaled from %ix%i)", info->width, info->height, info->swidth, info->sheight); else sprintf(tmp, "%ix%i pixels", info->swidth, info->sheight); xms = XmStringCreateSimple(tmp); XtVaSetValues(dimension_label, XmNlabelString, xms, NULL); XmStringFree(xms); if(preview_image) /* image has been quantized? */ if(preview_image->ncolors != preview_image->scolors) sprintf(tmp, "%i (reduced from %i)", preview_image->ncolors, preview_image->scolors); else sprintf(tmp, "%i", preview_image->ncolors); else sprintf(tmp, "%i", info->ncolors); xms = XmStringCreateSimple(tmp); XtVaSetValues(npixels_label, XmNlabelString, xms, NULL); XmStringFree(xms); /* image depth (bits per pixel) and colorclass */ switch(info->colorspace) { case XmIMAGE_COLORSPACE_GRAYSCALE: sprintf(tmp, "%i, Grayscale", info->depth); break; case XmIMAGE_COLORSPACE_INDEXED: sprintf(tmp, "%i, Indexed color", info->depth); break; case XmIMAGE_COLORSPACE_RGB: sprintf(tmp, "%i, TrueColor", info->depth); break; default: sprintf(tmp, "%i", info->depth); break; } xms = XmStringCreateSimple(tmp); XtVaSetValues(depth_label, XmNlabelString, xms, NULL); XmStringFree(xms); /* transparency */ switch(info->transparency) { case XmNONE: sprintf(tmp, "No"); break; case XmIMAGE_TRANSPARENCY_BG: sprintf(tmp, "Yes, using background substitution " "(pixel index %i)\n", info->bg); break; case XmIMAGE_TRANSPARENCY_ALPHA: sprintf(tmp, "Yes, using alpha channel"); break; default: sprintf(tmp, "Unknown"); } xms = XmStringCreateSimple(tmp); XtVaSetValues(trans_label, XmNlabelString, xms, NULL); XmStringFree(xms); /* cache info */ getCacheInfo(&cache_size, &nobjects); sprintf(tmp, "%i images cached (%i bytes)", nobjects, cache_size); xms = XmStringCreateSimple(tmp); XtVaSetValues(cache_label, XmNlabelString, xms, NULL); XmStringFree(xms); /* and the actual image */ if(preview_image) { XtVaSetValues(pixmap_label, XmNlabelType, XmPIXMAP, XmNlabelPixmap, preview_image->pixmap, NULL); } else { xms = XmStringCreateSimple("(unable to create image)"); XtVaSetValues(pixmap_label, XmNlabelType, XmSTRING, XmNlabelString, xms, NULL); XmStringFree(xms); } setBusy(False); /* put on screen */ XtManageChild(image_dialog); XMapRaised(XtDisplay(image_dialog), XtWindow(image_dialog)); } /***** * Name: infoPopupCB * Return Type: void * Description: callback for the menu-items in the popup menu * In: * w: widget id; * item: id of selected menu item. * Returns: * nothing. *****/ static void infoPopupCB(Widget w, int item) { XtPointer ref = NULL; /* get href, image name or image data from the widget's userdata */ XtVaGetValues(w, XmNuserData, &ref, NULL); /* fourth popup menu item doesn't carry userData */ if(ref == NULL && item != 3) { fprintf(stderr, "Failed to retrieve userData field from popup " "button %i\n", item); return; } switch(item) { case 0: /* follow link */ { /***** * We just compose a XmHTMLAnchorCallbackStruct and let anchorCB * do the loading. *****/ XmHTMLAnchorCallbackStruct cbs; cbs.reason = XmCR_ACTIVATE; cbs.event = NULL; cbs.url_type = XmHTMLGetURLType((String)ref); cbs.href = (String)ref; cbs.title = NULL; cbs.line = 0; cbs.target = NULL; cbs.doit = False; cbs.visited= False; /* and call the activate callback */ anchorCB(html_widgets[0].html, NULL, &cbs); } break; case 1: /* open image */ { String filename; if((filename = resolveFile((String)ref)) != NULL) { loadAndOrJump(filename, NULL, True); free(filename); } } break; case 2: /* view image details */ showImageInfo((XmImageInfo*)ref); break; case 3: /* toggle auto image tracking */ if(html_config[OPTIONS_FANCY_TRACKING].value) html_config[OPTIONS_FANCY_TRACKING].value = False; else html_config[OPTIONS_FANCY_TRACKING].value = True; /* reflect change in options menu as well */ XtVaSetValues(html_config[OPTIONS_FANCY_TRACKING].w, XmNset, html_config[OPTIONS_FANCY_TRACKING].value, NULL); break; default: fprintf(stderr, "Impossible popup menu selection!\n"); } } /***** * Name: collapseURL * Return Type: * Description: Copies up to 50 chars from the given url and puts in three * dots when the given url is larger than fifty chars. * In: * url: url to be collapsed or copied; * Returns: * copied url. *****/ static String collapseURL(String url) { static char name[51]; int len; if(url == NULL) return(""); len = strlen(url); if(len < 50) { strcpy(name, url); name[len] = '\0'; /* NULL terminate */ return(name); } /* copy first 11 chars */ strncpy(name, url, 11); /* put in a few dots */ name[11] = '.'; name[12] = '.'; name[13] = '.'; name[14] = '\0'; /* copy last 36 chars */ strcat(name, &url[len - 36]); name[50] = '\0'; /* NULL terminate */ return(name); } /***** * Name: infoCB * Return Type: void * Description: ButtonPressed handler for the workArea of a XmHTML widget. * In this case, a possible use of the XmHTMLXYToInfo resource * is demonstrated. * In: * w: widget id; * popup: popup menu widget id * event: location of button press. * Returns: * nothing. *****/ static void infoCB(Widget parent, Widget popup, XButtonPressedEvent *event) { XmString xms; char tmp[84]; /* max label width */ XmHTMLInfoPtr info; WidgetList children; Widget html_w; XUngrabPointer(XtDisplay(parent), CurrentTime); /* only button 3 */ if(event->button != 3) return; html_w = XtParent(parent); if(html_w == NULL || !XmIsHTML(html_w)) { fprintf(stderr, "%s parent gotten from XtParent(%s)\n", html_w == NULL ? "NULL" : "Invalid", XtName(parent)); return; } /* get the info for the selected position */ info = XmHTMLXYToInfo(html_w, event->x, event->y); /* no popup if no image and anchor */ if(info == NULL || (info->image == NULL && info->anchor == NULL)) return; XtVaGetValues(popup, XmNchildren, &children, NULL); /* unmanage all buttons */ XtUnmanageChild(children[0]); XtUnmanageChild(children[1]); XtUnmanageChild(children[2]); XtUnmanageChild(children[3]); /***** * Note on how to convey the href or image url's to the popup callbacks: * * All strings provided in the anchor and/or image field of this callback * are internal to XmHTML and are guarenteed to exist as long as the * current document is up. Knowing this, we can safely store these strings * in the userData field of the popup menu buttons. ****/ /* if we have an anchor, we copy the url to the label */ if(info->anchor) { sprintf(tmp, "Follow this link (%s)", collapseURL(info->anchor->href)); xms = XmStringCreateLocalized(tmp); XtVaSetValues(children[0], XmNlabelString, xms, XmNuserData, info->anchor->href, NULL); XmStringFree(xms); /* manage it */ XtManageChild(children[0]); } if(info->image) { sprintf(tmp, "Open this image (%s)", collapseURL(info->image->url)); xms = XmStringCreateLocalized(tmp); XtVaSetValues(children[1], XmNlabelString, xms, XmNuserData, info->image->url, NULL); XmStringFree(xms); /* manage it */ XtManageChild(children[1]); xms = XmStringCreateLocalized("View Image details"); XtVaSetValues(children[2], XmNlabelString, xms, XmNuserData, info->image, NULL); XmStringFree(xms); /* manage it */ XtManageChild(children[2]); /* set proper string for fancy image tracking */ if(html_config[OPTIONS_FANCY_TRACKING].value) xms = XmStringCreateLocalized("Disable Anchored Image Tracking"); else xms = XmStringCreateLocalized("Enable Anchored Image Tracking"); XtVaSetValues(children[3], XmNlabelString, xms, NULL); XmStringFree(xms); /* manage it */ XtManageChild(children[3]); } /* set correct menu position */ XmMenuPosition(popup, event); /* and show it */ XtManageChild(popup); } /***** * Name: navCB * Return Type: void * Description: callback for the buttons in the link dialog (displayed * by the linkButtonCB routine) * In: * w: widget id of selected button; * item: id of selected item; * Returns: * nothing. * Note: * This routine simply creates a XmHTMLAnchorCallbackStruct and calls * the anchorCB routine to let it handle navigation of the document * (which can include loading a new local or remote document, call a mail * application to mail something, download something, whatever). *****/ static void navCB(Widget w, int item) { static XmHTMLAnchorCallbackStruct cbs; /***** * We just compose a XmHTMLAnchorCallbackStruct and let anchorCB do the * loading. *****/ cbs.reason = XmCR_ACTIVATE; cbs.event = NULL; cbs.url_type = XmHTMLGetURLType(document_links[item].href); cbs.href = document_links[item].href; cbs.title = document_links[item].title; cbs.line = 0; cbs.target = NULL; cbs.doit = False; cbs.visited= False; if(document_links[item].link_type == 0) { cbs.rev = link_labels[item]; cbs.rel = NULL; } else { cbs.rel = link_labels[item]; cbs.rev = NULL; } /* and call the activate callback */ anchorCB(html_widgets[0].html, NULL, &cbs); } /***** * Name: linkButtonCB * Return Type: void * Description: displays a dialog with buttons to allow navigation of a * document using the information contained in the * section of a HTML document. * In: * w: widget id, unused; * arg1: client_data, unused; * arg2: call_data, unused; * Returns: * nothing. *****/ static void linkButtonCB(Widget w, XtPointer arg1, XtPointer arg2) { int i; if(!link_dialog) { Widget rc; link_dialog = XmCreateFormDialog(toplevel, "Preview", NULL, 0); XtVaSetValues(XtParent(link_dialog), XtNtitle, "Site Structure", NULL); /* a rowcol for the buttons */ rc = XtVaCreateManagedWidget("rowColumn", xmRowColumnWidgetClass, link_dialog, XmNtopAttachment, XmATTACH_FORM, XmNleftAttachment, XmATTACH_FORM, XmNrightAttachment, XmATTACH_FORM, XmNspacing, 0, XmNpacking, XmPACK_COLUMN, XmNorientation, XmVERTICAL, XmNnumColumns, 1, NULL); /* all buttons */ for(i = 0; i < LINK_LAST; i++) { link_buttons[i] = XtVaCreateManagedWidget(link_labels[i], xmPushButtonWidgetClass, rc, NULL); XtAddCallback(link_buttons[i], XmNactivateCallback, (XtCallbackProc)navCB, (XtPointer)i); } } /* Now see what buttons to activate. */ for(i = 0; i < LINK_LAST; i++) { if(document_links[i].have_data) XtSetSensitive(link_buttons[i], True); else XtSetSensitive(link_buttons[i], False); } /* * Keep the focus on the html widget so the navigation keys always * work. */ XmProcessTraversal(html_widgets[0].html, XmTRAVERSE_CURRENT); /* put on screen */ XtManageChild(link_dialog); XMapRaised(XtDisplay(link_dialog), XtWindow(link_dialog)); /* and make sure everything is up and displayed */ XmUpdateDisplay(link_dialog); } /***** * Name: historyCB * Return Type: void * Description: XmNactivateCallback handler for the back & forward buttons. * moves back or forward in the history. * In: * w: widget * button: button the triggered this callback. * Returns: * nothing. *****/ static void historyCB(Widget w, int button) { DocumentCache *this_doc = &doc_cache[current_doc]; /* back button has been pressed */ if(button == 0) { if(!XtIsSensitive(back)) return; /* current_ref == 0 -> top of file */ if(this_doc->current_ref) this_doc->current_ref--; /* reached bottom of current document, move to the previous one */ else { current_doc--; this_doc = &doc_cache[current_doc]; } if(loadAndOrJump(this_doc->file, this_doc->refs[this_doc->current_ref], False)) { XtSetSensitive(forward, True); XtSetSensitive(back, current_doc || this_doc->current_ref ? True : False); } else { if(current_doc) { current_doc--; historyCB(w, button); } } } /* forward button has been pressed */ else { if(!XtIsSensitive(forward)) return; this_doc->current_ref++; /* reached max of this document, move to the next doc */ if(this_doc->current_ref == this_doc->nrefs) { this_doc->current_ref--; current_doc++; this_doc = &doc_cache[current_doc]; } if(loadAndOrJump(this_doc->file, this_doc->refs[this_doc->current_ref], False)) { XtSetSensitive(back, True); if(current_doc == last_doc - 1 && this_doc->current_ref == this_doc->nrefs - 1) XtSetSensitive(forward, False); } else { if(current_doc != last_doc-1) { current_doc++; historyCB(w, button); } } } /* * Keep the focus on the html widget so the navigation keys always * work. */ XmProcessTraversal(html_widgets[0].html, XmTRAVERSE_CURRENT); } /***** * Name: destroyCacheObject * Return Type: void * Description: gets called by the object caching routines when it wants to * destroy a cached object. * In: * call_data: object to be destroyed; * client_data:data we registered ourselves when we made the initObjectCache * call. * Returns: * nothing. *****/ static void destroyCacheObject(XtPointer call_data, XtPointer client_data) { Debug(("destroyCacheObject, called for %s\n", ((XmImageInfo*)call_data)->url)); XmHTMLImageFreeImageInfo((Widget)client_data, (XmImageInfo*)call_data); } /***** * Name: loadAnimation * Return Type: XmImageInfo * Description: load an animation consisting of multiple images * In: * names: comma separated list of images * Returns: * a list of XmImageInfo composing the animation * Note: * this is a fairly simple example on how to add support for images * that are not supported by the XmHTMLImageDefaultProc convenience * function. *****/ XmImageInfo* loadAnimation(char *names) { static XmImageInfo *all_frames; XmImageInfo *frame = NULL; String chPtr, name_buf, filename; int nframes = 0; name_buf = strdup(names); for(chPtr = strtok(name_buf, ","); chPtr != NULL; chPtr = strtok(NULL, ",")) { if((filename = resolveFile(chPtr)) == NULL) { free(name_buf); return(NULL); } if(nframes) { frame->frame = XmHTMLImageDefaultProc(html_widgets[0].html, filename, NULL, 0); frame = frame->frame; } else { all_frames = XmHTMLImageDefaultProc(html_widgets[0].html, filename, NULL, 0); frame = all_frames; } frame->timeout = animation_timeout; nframes++; free(filename); } free(name_buf); /* nframes is total no of frames in this animation */ all_frames->nframes = nframes; all_frames->timeout = animation_timeout; return(all_frames); } /***** * Name: flushImages * Return Type: void * Description: flushes all images (normal *and* delayed) to the currently * loaded document. * In: * w: widget id, unused; * Returns: * nothing. * Note: * This routine simply flushes all images in the currently loaded document. * When it finds an image that has been delayed, it loads the real image and * replaces the delayed image in the current document with the real image. * When all images have been updated, it calls XmHTMLRedisplay to force * XmHTML to do a re-computation of the document layout. *****/ static void flushImages(Widget w) { int i; static XmImageInfo *image, *new_image; String url, filename; DocumentCache *this_doc; XmImageStatus status = XmIMAGE_OK, retval = XmIMAGE_OK; setBusy(True); /* get current document */ this_doc = &doc_cache[current_doc]; for(i = 0; i < this_doc->nimages; i++) { url = this_doc->images[i]; /* get image to update */ if((image = (XmImageInfo*)getURLObjectFromCache(url)) == NULL) continue; /* only do this when the image hasn't been loaded yet */ if(image->options & XmIMAGE_DELAYED && (filename = resolveFile(url)) != NULL) { if(strstr(url, ",")) new_image = loadAnimation(url); else { new_image = XmHTMLImageDefaultProc(html_widgets[0].html, filename, NULL, 0); free(filename); } if(new_image) { /* don't let XmHTML free it */ new_image->options &= ~(XmIMAGE_DEFERRED_FREE) & ~(XmIMAGE_DELAYED); /***** * Replace it. XmHTMLImageReplace returns a statuscode * indicating success of the action. If XmIMAGE_OK is returned, * no call to XmHTMLRedisplay is necessary, the dimensions are * the same as specified in the document. It returns * XmIMAGE_ALMOST if a recomputation of document layout is * necessary, and something else if an error occured: * XmIMAGE_ERROR: the widget arg is not a XmHTML widget or * pixmap creation failed; * XmIMAGE_BAD if image and/or new_image is NULL; * XmIMAGE_UNKNOWN if image is unbound to an internal image. *****/ retval = XmHTMLImageReplace(html_widgets[0].html, image, new_image); /* update private cache */ replaceObjectInCache((XtPointer)image, (XtPointer)new_image); /* and destroy previous image data */ XmHTMLImageFreeImageInfo((Widget)html_widgets[0].html, image); } } else /* same note as above applies */ retval = XmHTMLImageUpdate(html_widgets[0].html, image); /* store return value of ImageReplace and/or ImageUpdate */ if(retval == XmIMAGE_ALMOST) status = retval; } /* force a reformat and redisplay of the current document if required */ if(status == XmIMAGE_ALMOST) XmHTMLRedisplay(html_widgets[0].html); /* keep focus on the html widget */ XmProcessTraversal(html_widgets[0].html, XmTRAVERSE_CURRENT); setBusy(False); } /***** * Name: killImages * Return Type: void * Description: kills all images (normal *and* delayed) of all documents. * Should be called at exit. * In: * nothing. * Returns: * nothing. *****/ static void killImages(void) { setBusy(True); destroyObjectCache(); setBusy(False); } static void progressiveButtonCB(Widget w, int reset) { int curr_state = 0; String label; XmString xms; if(reset) { if(XtIsManaged(prg_button)) XtUnmanageChild(prg_button); /* set new label and global PLC state */ xms = XmStringCreateLocalized("Suspend Image Load"); XtVaSetValues(prg_button, XmNlabelString, xms, XmNuserData, (XtPointer)STREAM_OK, NULL); XmStringFree(xms); return; } /* get current progressive image loading state */ XtVaGetValues(w, XmNuserData, &curr_state, NULL); switch(curr_state) { case STREAM_OK: XmHTMLImageProgressiveSuspend(html_widgets[0].html); curr_state = STREAM_SUSPEND; label = "Continue Image Load"; break; case STREAM_SUSPEND: XmHTMLImageProgressiveContinue(html_widgets[0].html); curr_state = STREAM_OK; label = "Suspend Image Load"; break; default: fprintf(stderr, "Oops, unknown button state in " "progressiveButtonCB\n"); return; } /* set new label and global PLC state */ xms = XmStringCreateLocalized(label); XtVaSetValues(prg_button, XmNlabelString, xms, XmNuserData, (XtPointer)curr_state, NULL); XmStringFree(xms); } /***** * Name: getImageData * Return Type: int * Description: XmHTMLGetDataProc method. Called when we are to * load images progressively. * In: * stream: Progressive Load Context stream object * buffer: destination buffer. * Returns: * STREAM_END when we have run out of data, number of bytes copied into the * buffer otherwise. * Note: * This routine is an example implementation of how to write a * XmHTMLGetDataProc method. As this program doesn't have networking * capabilities, we mimic a connection by providing the data requested in * small chunks (which is a command line option: prg_skip [number of bytes]). * The stream argument is a structure containing a minimum and maximum byte * count (the min_out and max_out fields), a number representing the number * of bytes used by XmHTML (the total_in field) and user_data registered * with the object that is being loaded progressively. * * If this routine returns data, it must *always* be a number between * min_out and max_out (including min_out and max_out). Returning less is * not an advisable thing to do (it will cause an additional call immediatly) * and returning more *can* cause an error (by overflowing the buffer). * * You can let XmHTML expand it's internal buffers by setting the max_out * field to the size you want the buffer to have and returning STREAM_RESIZE. * XmHTML will then try to resize its internal buffers to the requested size * and call this routine again immediatly. When a buffer has been resized, it * is very likely that XmHTML will backtrack to an appropriate starting point, * so be sure to check and use the total_in field of the stream arg when * returning data. * * If you want to abort progressive loading, you can return STREAM_ABORT. * This will cause XmHTML to terminate the progressive load for the given * object (which involves a call to any installed XmHTMLProgressiveEndData * method). Example use of this could be an ``Abort'' button. An alternative * method is to use the XmHTMLImageProgressiveKill() convenience routine as * shown in the getAndSetFile() routine above. * * Also note that returning 0 is equivalent to returning STREAM_END (which * is defined as being 0). * * As a final note, XmHTML will ignore any bytes copied into the buffer * if you return any of the STREAM_ codes. *****/ static int getImageData(XmHTMLPLCStream *stream, XtPointer buffer) { ImageBuffer *ib = (ImageBuffer*)stream->user_data; int len; Debug(("getImageData, request made for %s\n", ib->file)); Debug(("getImageData, XmHTML already has %i bytes\n", stream->total_in)); /* no more data available, everything has been copied */ if(ib->next >= ib->size) return(STREAM_END); /* * Maximum no of bytes we can return. ib->size contains the total size of * the image data, and total_in contains the number of bytes that have * already been used by XmHTML so far. * total_in may differ from ib->next due to backtracking of the calling PLC. */ len = ib->size - stream->total_in; /* * If you want to flush all data you've got to XmHTML but max_out is too * small, you can do something like this: * if(len > stream->max_out) * { * stream->max_out = len; * return(STREAM_RESIZE); * } * As noted above, XmHTML will then resize it's internal buffers to fit * the requested size and call this routine again. Before I forget, the * default size of the internal buffers is 2K. * And no, setting max_out to 0 will not cause XmHTML to choke. It will * simply ignore it (and issue a blatant warning message accusing you of * being a bad programmer :-). */ if(progressive_data_inc) { /* increment if not yet done for this pass */ if(!ib->may_free) { progressive_data_skip += progressive_data_inc; Debug(("getImageData, incrementing buffer size to %i bytes\n", progressive_data_skip)); stream->max_out = progressive_data_skip; ib->may_free = True; return(STREAM_RESIZE); } else /* already incremented, copy data for this pass */ ib->may_free = False; } /* provide the minimum if our skip is too small */ if(len < stream->min_out || progressive_data_skip < stream->min_out) len = stream->min_out; else len = progressive_data_skip; /* final sanity */ if(len + stream->total_in > ib->size) len = ib->size - stream->total_in; /* more bytes available than minimally requested, we can copy */ if(len >= stream->min_out) { /* but don't exceed the maximum allowable amount to return */ if(len > stream->max_out) len = stream->max_out; Debug(("getImageData, returning %i bytes (min_out = %i, " "max_out = %i)\n", len, stream->min_out, stream->max_out)); memcpy((char*)buffer, ib->buffer + stream->total_in, len); ib->next = stream->total_in + len; return(len); } /* * some sort of error, XmHTML requested data beyond the end of the file, * so we just return STREAM_END here and let XmHTML decide what to do. */ return(STREAM_END); } static void endImageData(XmHTMLPLCStream *stream, XtPointer data, int type, Boolean ok) { XmImageInfo *image = (XmImageInfo*)data; ImageBuffer *ib; /* * XmHTML signals us that there are no more images being loaded * progressively. Remove ``Suspend Image Load'' button. * Beware: this is the only case in which stream is NULL. */ if(type == XmPLC_FINISHED) { XtSetSensitive(prg_button, False); XtUnmanageChild(prg_button); return; } ib = (ImageBuffer*)stream->user_data; /* * To keep the cache size in sync, we update the cached image by replacing * it. As we will be replacing the same object, the only effect this call * will have is that the sizeObjectProc will be called. */ if(ok) replaceObjectInCache((XtPointer)image, (XtPointer)image); else { /* incomplete image, remove it from the image cache */ removeObjectFromCache(ib->file); cleanObjectCache(); } Debug(("endImageData, called for %s, ok = %s\n", ib->file, ok ? "True" : "False")); free(ib->file); free(ib->buffer); free(ib); } /***** * Name: loadImage * Return Type: XmImageInfo * Description: XmHTMLimageProc handler * In: * w: HTML widget id * url: src value of an img element. * Returns: * return value from the HTML widget imageDefaultProc * Note: * this is a very simple example of how to respond to requests for images: * XmHTML calls this routine with the name (or location or whatever the * src value is) of an image to load. All you need to do is get the full * name of the image requested and call the imageDefaultProc and let XmHTML * handle the actual loading. * This is also the place to fetch remote images, implement an imagecache * or add support for images not supported by the imageDefaultProc. *****/ static XmImageInfo* loadImage(Widget w, String url) { String filename = NULL; XmImageInfo *image = NULL; DocumentCache *this_doc; int i; /* get current document */ this_doc = &doc_cache[current_doc]; Debug(("Requested to load image %s\n", url)); /***** * get full path for this url. The "," strstr is used to check if this * image is an animation consisting of a list of comma-separated images * (see examples/test-pages/animation?.html for an example) *****/ if((filename = resolveFile(url)) == NULL) if(!strstr(url, ",")) return(NULL); /* * add this image to this document's image list. * First check if we haven't got it already, can only happen when the * document is reloaded (XmHTML doesn't load identical images). * Use original URL for this. */ for(i = 0; i < this_doc->nimages; i++) { if(!(strcmp(this_doc->images[i], url))) break; } /* don't have it yet */ if(i == this_doc->nimages) { /* see if it can hold any more images */ if(this_doc->nimages != MAX_IMAGE_ITEMS) { this_doc->images[this_doc->nimages++] = strdup(url); i = this_doc->nimages; } else { char buf[128]; sprintf(buf, "This document contains more than %i images,\n" "Only the first %i will be shown.", MAX_IMAGE_ITEMS, MAX_IMAGE_ITEMS); XMessage(toplevel, buf); return(NULL); } } /* now check if we have this image already available */ if((image = (XmImageInfo*)getObjectFromCache(filename, url)) != NULL) { /* * If i isn't equal to the current no of images for the current * document, the requested image has already been loaded once for this * document, so we do not have to store it again. */ /* call storeImage again as we might be using a different URL */ if(i == this_doc->nimages) storeObjectInCache((XtPointer)image, filename ? filename : url, url); if(filename) free(filename); return(image); } if(filename || (strstr(url, ",")) != NULL) { /* test delayed image loading */ if(html_config[OPTIONS_AUTO_IMAGE_LOAD].value) { if(strstr(url, ",")) image = loadAnimation(url); else { if(!progressive_images) image = XmHTMLImageDefaultProc(w, filename, NULL, 0); else { unsigned char img_type; img_type = XmHTMLImageGetType(filename, NULL, 0); if(img_type != IMAGE_ERROR && img_type != IMAGE_UNKNOWN && img_type != IMAGE_XPM && img_type != IMAGE_PNG) { FILE *file; static ImageBuffer *ib; /* open the given file */ if((file = fopen(filename, "r")) == NULL) { perror(filename); return(NULL); } /* * We load the image data into an ImageBuffer (which * we will be using in the get_data() function. */ ib = (ImageBuffer*)malloc(sizeof(ImageBuffer)); ib->file = strdup(filename); ib->next = 0; /* see how large this file is */ fseek(file, 0, SEEK_END); ib->size = ftell(file); rewind(file); /* allocate a buffer to contain the entire image */ ib->buffer = malloc(ib->size+1); /* now read the contents of this file */ if((fread(ib->buffer, 1, ib->size, file)) != ib->size) printf("Warning: did not read entire file!\n"); ib->buffer[ib->size] = '\0'; /* sanity */ /* create an empty ImageInfo */ image = (XmImageInfo*)malloc(sizeof(XmImageInfo)); memset(image, 0, sizeof(XmImageInfo)); /* set the Progressive bit and allow scaling */ image->options =XmIMAGE_PROGRESSIVE|XmIMAGE_ALLOW_SCALE; image->url = strdup(filename); /* set file buffer as user data for this image */ image->user_data = (XtPointer)ib; /* make the progressive image loading button visible */ if(!XtIsManaged(prg_button)) XtManageChild(prg_button); XtSetSensitive(prg_button, True); /* all done! */ } else image = XmHTMLImageDefaultProc(w, filename, NULL, 0); } } /* failed, too bad */ if(!image) return(NULL); /* don't let XmHTML free it */ image->options &= ~(XmIMAGE_DEFERRED_FREE); } else { image = (XmImageInfo*)malloc(sizeof(XmImageInfo)); memset(image, 0, sizeof(XmImageInfo)); image->options = XmIMAGE_DELAYED|XmIMAGE_ALLOW_SCALE; image->url = strdup(filename ? filename : url); } /* store in the cache */ storeObjectInCache((XtPointer)image, filename ? filename : url, url); } if(filename) free(filename); return(image); } /***** * Name: testAnchor * Return Type: int * Description: XmNanchorVisitedProc procedure * In: * w: widget * href: href to test * Returns: * True when the given href has already been visited, false otherwise. * Note: * This is quite inefficient. In fact, the whole history scheme is * inefficient, but then again, this is only an example and not a full * featured browser ;-) *****/ static int testAnchor(Widget w, String href) { int i, j; /* walk each document */ for(i = 0 ; i < last_doc; i++) { /* and walk the history list of each document */ for(j = 0; j < doc_cache[i].nvisited; j++) if(doc_cache[i].visited[j] && !strcmp(doc_cache[i].visited[j], href)) return(True); } /* we don't know it */ return(False); } /***** * Name: aboutCB * Return Type: void * Description: displays an ``About'' dialog when the help->about menu item * is selected. * In: * widget: menubutton widget id * client_data: unused * call_data: unused * Returns: * nothing *****/ static void aboutCB(Widget widget, XtPointer client_data, XtPointer call_data) { char label[256]; sprintf(label, "A Simple HTML browser using\n" "%s\n", XmHTMLVERSION_STRING); XMessage(toplevel, label); } static void optionsCB(Widget w, int item) { XmString label; Boolean set = False; int i, argc = 0; Arg args[4]; switch(item) { /* * These are seven XmHTML On/Off resources so we can treat them * all in the same manner */ case OPTIONS_ANCHOR_BUTTONS: case OPTIONS_HIGHLIGHT_ON_ENTER: case OPTIONS_ENABLE_STRICT_HTML32: case OPTIONS_ENABLE_BODYCOLORS: case OPTIONS_ENABLE_BODYIMAGES: case OPTIONS_ENABLE_DOCUMENT_COLORS: case OPTIONS_ENABLE_DOCUMENT_FONTS: case OPTIONS_ENABLE_IMAGES: case OPTIONS_ENABLE_OUTLINING: case OPTIONS_DISABLE_WARNINGS: case OPTIONS_FREEZE_ANIMATIONS: /* get value */ XtVaGetValues(w, XmNset, &set, NULL); /* check if changed */ if(set == html_config[item].value) break; /* store new value */ html_config[item].value = set; Debug(("optionsCB, setting value for resource %s to %s\n", html_config[item].name, set ? "True" : "False")); /* set new value */ XtSetArg(args[argc], html_config[item].name, html_config[item].value); argc++; /* * if global image support has been toggled, toggle other buttons * as well. */ if(item == OPTIONS_ENABLE_IMAGES) { XtSetSensitive(html_config[OPTIONS_ENABLE_BODYIMAGES].w, set); XtSetSensitive(html_config[OPTIONS_AUTO_IMAGE_LOAD].w, set); XtSetSensitive(load_images, set); /* set corresponding resources */ XtSetArg(args[argc], html_config[OPTIONS_ENABLE_BODYIMAGES].name, set ? html_config[OPTIONS_ENABLE_BODYIMAGES].value:False); argc++; } /* propagate changes down to all active HTML widgets */ for(i = 0; i < MAX_HTML_WIDGETS; i++) { if(html_widgets[i].active) XtSetValues(html_widgets[i].html, args, argc); } break; case OPTIONS_AUTO_IMAGE_LOAD: case OPTIONS_FANCY_TRACKING: /* get value */ XtVaGetValues(w, XmNset, &set, NULL); /* check if changed */ if(set == html_config[item].value) break; /* store new value */ html_config[item].value = set; Debug(("optionsCB, setting value for %s to %s\n", html_config[item].name, set ? "True" : "False")); /* change label as well */ if(set) label = XmStringCreateLocalized("Reload Images"); else label = XmStringCreateLocalized("Load Images"); XtVaSetValues(load_images, XmNlabelString, label, NULL); XmStringFree(label); break; case OPTIONS_ANCHOR: fprintf(stderr, "Configure anchor appearance (not ready)\n"); break; case OPTIONS_FONTS: fprintf(stderr, "Configure document fonts (not ready).\n"); break; case OPTIONS_BODY: fprintf(stderr, "Configure default body settings (not ready).\n"); break; case OPTIONS_IMAGE: fprintf(stderr, "Configure default image settings (not ready).\n"); break; default: fprintf(stderr, "optionsCB: impossible menu selection " "(item = %i)\n", item); } } /***** * Name: main * Return Type: int * Description: main for example 2 * In: * argc: no of command line arguments * argv: array of command line arguments * Returns: * EXIT_FAILURE when an error occurs, EXIT_SUCCESS otherwise *****/ int main(int argc, char **argv) { Widget main_w, menubar, menu, rc, form, sep, fr, pb, popup_menu = NULL; Widget option_menu; Widget frame = NULL; /* shutup compiler */ Display *display = NULL; /* shutup compiler */ XmString file, load, quit, help, about, save; XmString raise = NULL, lower = NULL; /* shutup compiler */ XmString reloadfile, docinfo; XmString xms; int i; String use_file = NULL, font = NULL, charset = NULL; Visual *visual = NULL; int depth = 0; Colormap colormap = 0; Boolean arg_error = False; fprintf(stderr, "%s, %i\n", XmHTMLVERSION_STRING, XmHTMLGetVersion()); root_window = False; progressive_images = False; /* set the debugging levels */ _XmHTMLSetDebugLevels(&argc, argv); for(i = 1; i < argc; i++) { if(argv[i][0] == '-') { switch(argv[i][1]) { case 'a': if(!(strcmp(argv[i], "-allow_exec"))) allow_exec = True; else if(!(strcmp(argv[i], "-animation_timeout"))) { i++; if(i < argc && argv[i][0] != '-') { if((animation_timeout= atoi(argv[i])) < 50) { fprintf(stderr, "animation_delay to small, " "reset to 50\n"); animation_timeout = 50; } } else { fprintf(stderr, "Error: missing timeout for " "-animation_timeout argument\n"); exit(EXIT_FAILURE); } } else arg_error = True; break; #ifdef DEBUG case 'd': if(!(strcmp(argv[i], "-debug"))) debug = True; else arg_error = True; break; #endif case 'i': if(!(strcmp(argv[i], "-images_delayed"))) html_config[OPTIONS_AUTO_IMAGE_LOAD].value = False; else arg_error = True; break; case 'h': fprintf(stderr, "%s\n", usage); exit(EXIT_SUCCESS); case 'n': if(!(strcmp(argv[i], "-netscape"))) external_client = True; else if(!(strcmp(argv[i], "-noframe"))) noframe = True; else arg_error = True; break; case 'p': if(!(strcmp(argv[i], "-progressive"))) progressive_images = True; else if (!(strcmp(argv[i], "-prg_skip"))) { i++; if(i < argc && argv[i][0] != '-') { if((progressive_data_skip = atoi(argv[i])) <= 0) { fprintf(stderr, "prg_skip to small, reset to " "%i\n", MAX_PROGRESSIVE_DATA_SKIP); progressive_data_skip = MAX_PROGRESSIVE_DATA_SKIP; } } else { fprintf(stderr, "Error: missing number for " "-prg_skip argument\n"); exit(EXIT_FAILURE); } } else if (!(strcmp(argv[i], "-prg_inc"))) { i++; if(i < argc && argv[i][0] != '-') { if((progressive_data_inc = atoi(argv[i])) <= 0) { fprintf(stderr, "prg_inc to small, ignored."); progressive_data_inc = 0; } else progressive_data_skip = 256; } else { fprintf(stderr, "Error: missing number for " "-prg_skip argument\n"); exit(EXIT_FAILURE); } } else arg_error = True; break; case 'r': if(!(strcmp(argv[i], "-root"))) root_window = True; else arg_error = True; break; case '-': if(!(strcmp(argv[i], "--help"))) { fprintf(stderr, "%s\n", usage); exit(EXIT_SUCCESS); } /* end-of-arguments, X11 opts follow */ else if(!(strcmp(argv[i], "--"))) i = argc; break; default: arg_error = True; break; } if(arg_error) { fprintf(stderr, "%s: unknown argument %s\n\n%s\n", argv[0], argv[i], usage); exit(EXIT_FAILURE); } } else use_file = argv[i]; } /* set current working directory as the first path to search */ #ifdef _POSIX_SOURCE getcwd((char*)(paths[0]), sizeof(paths[0])); #else getwd((char*)(paths[0])); #endif strcat((char*)(paths[0]), "/"); max_paths = 1; toplevel = XtVaAppInitialize(&context, APP_CLASS, NULL, 0, &argc, argv, appFallbackResources, NULL, NULL); /* check if visual, depth or colormap have been given */ if(getStartupVisual(toplevel, &visual, &depth, &colormap)) XtVaSetValues(toplevel, XmNvisual, visual, XmNdepth, depth, XmNcolormap, colormap, NULL); if(root_window) { display = XtDisplay(toplevel); XtVaSetValues(toplevel, XmNoverrideRedirect, True, XmNx, 0, XmNy, 0, XmNheight, DisplayHeight(display, DefaultScreen(display)), XmNwidth, DisplayWidth(display, DefaultScreen(display)), NULL); } /* create mainWindow */ main_w = XtVaCreateManagedWidget("mainWindow", xmMainWindowWidgetClass, toplevel, NULL); /* simple menubar with a File and Help menu */ menubar = XmCreateMenuBar(main_w, "menubar", NULL, 0); /* create file menu */ menu = XmCreatePulldownMenu(menubar, "filePulldown", NULL, 0); /* create file button */ file = XmStringCreateLocalized("File"); XtVaCreateManagedWidget("File", xmCascadeButtonWidgetClass, menubar, XmNlabelString, file, XmNmnemonic, 'F', XmNsubMenuId, menu, NULL); XmStringFree(file); /* don't need it anymore */ /* file menu items */ load = XmStringCreateLocalized("Open"); quit = XmStringCreateLocalized("Quit"); save = XmStringCreateLocalized("Save As.."); reloadfile = XmStringCreateLocalized("Reload"); docinfo = XmStringCreateLocalized("View Document Info"); help = XmStringCreateLocalized("View Document Source"); if(root_window) { raise = XmStringCreateLocalized("Raise"); lower = XmStringCreateLocalized("Lower"); } pb = XtVaCreateManagedWidget("Open", xmPushButtonGadgetClass, menu, XmNlabelString, load, NULL); XtAddCallback(pb, XmNactivateCallback, (XtCallbackProc)fileCB, (XtPointer)FILE_OPEN); reload = XtVaCreateManagedWidget("Reload", xmPushButtonGadgetClass, menu, XmNlabelString, reloadfile, XmNsensitive, False, NULL); XtAddCallback(reload, XmNactivateCallback, (XtCallbackProc)fileCB, (XtPointer)FILE_RELOAD); if(root_window) { pb = XtVaCreateManagedWidget("Raise", xmPushButtonGadgetClass, menu, XmNlabelString, raise, NULL); XtAddCallback(pb, XmNactivateCallback, (XtCallbackProc)fileCB, (XtPointer)FILE_RAISE); pb = XtVaCreateManagedWidget("Lower", xmPushButtonGadgetClass, menu, XmNlabelString, lower, NULL); XtAddCallback(pb, XmNactivateCallback, (XtCallbackProc)fileCB, (XtPointer)FILE_LOWER); } XtVaCreateManagedWidget("separator", xmSeparatorGadgetClass, menu, NULL); pb = XtVaCreateManagedWidget("SaveAs", xmPushButtonGadgetClass, menu, XmNlabelString, save, NULL); XtAddCallback(pb, XmNactivateCallback, (XtCallbackProc)fileCB, (XtPointer)FILE_SAVEAS); XtVaCreateManagedWidget("separator", xmSeparatorGadgetClass, menu, NULL); pb = XtVaCreateManagedWidget("SourceView", xmPushButtonGadgetClass, menu, XmNlabelString, help, NULL); XtAddCallback(pb, XmNactivateCallback, (XtCallbackProc)fileCB, (XtPointer)FILE_VIEW); pb = XtVaCreateManagedWidget("DocInfo", xmPushButtonGadgetClass, menu, XmNlabelString, docinfo, NULL); XtAddCallback(pb, XmNactivateCallback, (XtCallbackProc)fileCB, (XtPointer)FILE_INFO); XtVaCreateManagedWidget("separator", xmSeparatorGadgetClass, menu, NULL); pb = XtVaCreateManagedWidget("Quit", xmPushButtonGadgetClass, menu, XmNlabelString, quit, NULL); XtAddCallback(pb, XmNactivateCallback, (XtCallbackProc)fileCB, (XtPointer)FILE_QUIT); XmStringFree(load); XmStringFree(save); XmStringFree(quit); XmStringFree(docinfo); XmStringFree(reloadfile); XmStringFree(help); if(root_window) { XmStringFree(raise); XmStringFree(lower); } /* create Options menu */ option_menu = XmCreatePulldownMenu(menubar, "optionsPulldown", NULL, 0); /* create options button */ help = XmStringCreateLocalized("Options"); pb = XtVaCreateManagedWidget("Options", xmCascadeButtonWidgetClass, menubar, XmNlabelString, help, XmNmnemonic, 'O', XmNsubMenuId, option_menu, NULL); XmStringFree(help); /* don't need it anymore */ /* option menu items */ /* anchor styles */ pb = XtVaCreateManagedWidget("Anchor Appearance", xmPushButtonGadgetClass, option_menu, NULL); XtAddCallback(pb, XmNactivateCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_ANCHOR); /* default fonts */ pb = XtVaCreateManagedWidget("Default Fonts", xmPushButtonGadgetClass, option_menu, NULL); XtAddCallback(pb, XmNactivateCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_FONTS); /* default body configuration: background image & colors */ pb = XtVaCreateManagedWidget("Body Defaults", xmPushButtonGadgetClass, option_menu, NULL); XtAddCallback(pb, XmNactivateCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_BODY); /* image configuration. no of colors, gamma & progressive configuration */ pb = XtVaCreateManagedWidget("Images/Colors", xmPushButtonGadgetClass, option_menu, NULL); XtAddCallback(pb, XmNactivateCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_IMAGE); /* * A separator. The toggle buttons come when a XmHTML widget has been * created (so we can get the default values). */ XtVaCreateManagedWidget("separator", xmSeparatorGadgetClass, option_menu, NULL); /* create help menu */ menu = XmCreatePulldownMenu(menubar, "helpPulldown", NULL, 0); /* create help button */ help = XmStringCreateLocalized("Help"); pb = XtVaCreateManagedWidget("Help", xmCascadeButtonWidgetClass, menubar, XmNlabelString, help, XmNmnemonic, 'H', XmNsubMenuId, menu, NULL); XmStringFree(help); /* don't need it anymore */ /* tell Motif this is the help menu */ XtVaSetValues(menubar, XmNmenuHelpWidget, pb, NULL); /* help menu items */ about = XmStringCreateLocalized("About"); pb = XtVaCreateManagedWidget("About", xmPushButtonGadgetClass, menu, XmNlabelString, about, NULL); XtAddCallback(pb, XmNactivateCallback, (XtCallbackProc)aboutCB, NULL); XmStringFree(about); /* menubar is done */ XtManageChild(menubar); /* form as the workarea */ form = XtVaCreateWidget("form", xmFormWidgetClass, main_w, XmNresizePolicy, XmRESIZE_ANY, NULL); /* a rowcol for the buttons */ rc = XtVaCreateWidget("rowColumn", xmRowColumnWidgetClass, form, XmNtopAttachment, XmATTACH_FORM, XmNleftAttachment, XmATTACH_FORM, XmNrightAttachment, XmATTACH_FORM, XmNadjustLast, False, XmNspacing, 0, XmNpacking, XmPACK_COLUMN, XmNorientation, XmVERTICAL, XmNnumColumns, 4, NULL); /* the back button */ back = XtVaCreateManagedWidget("Back", xmPushButtonWidgetClass, rc, XmNsensitive, False, NULL); /* callback */ XtAddCallback(back , XmNactivateCallback, (XtCallbackProc)historyCB, (XtPointer)0); /* the forward button */ forward = XtVaCreateManagedWidget("Forward", xmPushButtonWidgetClass, rc, XmNsensitive, False, NULL); /* callback */ XtAddCallback(forward, XmNactivateCallback, (XtCallbackProc)historyCB, (XtPointer)1); /* load images button */ load_images = XtVaCreateManagedWidget( (html_config[OPTIONS_AUTO_IMAGE_LOAD].value ? "Reload Images" : "Load Images"), xmPushButtonWidgetClass, rc, XmNsensitive, False, NULL); /* callback */ XtAddCallback(load_images, XmNactivateCallback, (XtCallbackProc)flushImages, NULL); /* the Link button */ link_button = XtCreateWidget("Document Links", xmPushButtonWidgetClass, rc, NULL, 0); /* callback */ XtAddCallback(link_button, XmNactivateCallback, (XtCallbackProc)linkButtonCB, NULL); /* suspend/activate/kill progressive image loading */ if(progressive_images) { prg_button = XtVaCreateWidget("Suspend Image Load", xmPushButtonWidgetClass, rc, XmNuserData, (XtPointer)STREAM_OK, XmNsensitive, False, NULL); /* callback */ XtAddCallback(prg_button, XmNactivateCallback, (XtCallbackProc)progressiveButtonCB, (XtPointer)0); } /* done with buttonbar, manage it */ XtManageChild(rc); /* a separator between the rc and the html widget */ sep = XtVaCreateManagedWidget("separator", xmSeparatorGadgetClass, form, XmNtopAttachment, XmATTACH_WIDGET, XmNtopWidget, rc, XmNleftAttachment, XmATTACH_FORM, XmNrightAttachment, XmATTACH_FORM, XmNorientation, XmHORIZONTAL, NULL); /* create a frame as the html container */ if(!noframe) frame = XtVaCreateManagedWidget("frame", xmFrameWidgetClass, form, XmNtopAttachment, XmATTACH_WIDGET, XmNtopWidget, sep, XmNtopOffset, 10, XmNleftAttachment, XmATTACH_FORM, XmNleftOffset, 10, XmNrightAttachment, XmATTACH_FORM, XmNrightOffset, 10, XmNshadowType, XmSHADOW_IN, NULL); /* create the HTML widget using the default resources */ if(root_window) { /* create the HTML widget using the default resources */ html_widgets[0].html = XtVaCreateManagedWidget("html", xmHTMLWidgetClass, (noframe ? form : frame), XmNanchorVisitedProc, testAnchor, XmNimageProc, loadImage, XmNimageEnable, True, XmNprogressiveReadProc, getImageData, XmNprogressiveEndProc, endImageData, #ifdef HAVE_GIF_CODEC XmNdecodeGIFProc, decodeGIFImage, #endif /* * Is isn't the way to this, but then again ... * it's just an example! */ XmNheight, DisplayHeight(display, DefaultScreen(display))-125, XmNwidth, DisplayWidth(display, DefaultScreen(display))-50, NULL); /* arm callback for passing events to the window manager */ XtAddCallback(html_widgets[0].html, XmNarmCallback, (XtCallbackProc)armCB, NULL); } else { html_widgets[0].html = XtVaCreateManagedWidget("html", xmHTMLWidgetClass, (noframe ? form : frame), XmNanchorVisitedProc, testAnchor, XmNimageProc, loadImage, XmNimageEnable, True, XmNprogressiveReadProc, getImageData, XmNprogressiveEndProc, endImageData, #ifdef HAVE_GIF_CODEC XmNdecodeGIFProc, decodeGIFImage, #endif NULL); } html_widgets[0].active = True; html_widgets[0].used = True; XtVaSetValues(noframe ? html_widgets[0].html: frame, XmNtopAttachment, XmATTACH_WIDGET, XmNtopWidget, sep, XmNtopOffset, 10, XmNleftAttachment, XmATTACH_FORM, XmNleftOffset, 10, XmNrightAttachment, XmATTACH_FORM, XmNrightOffset, 10, NULL); /* anchor activation callback */ XtAddCallback(html_widgets[0].html, XmNactivateCallback, (XtCallbackProc)anchorCB, NULL); /* anchor tracking callback */ XtAddCallback(html_widgets[0].html, XmNanchorTrackCallback, (XtCallbackProc)trackCB, NULL); /* HTML frame callback */ XtAddCallback(html_widgets[0].html, XmNframeCallback, (XtCallbackProc)frameCB, NULL); /* set the HTML document callback */ XtAddCallback(html_widgets[0].html, XmNdocumentCallback, (XtCallbackProc)docCB, NULL); /* link callback for site structure */ XtAddCallback(html_widgets[0].html, XmNlinkCallback, (XtCallbackProc)linkCB, NULL); /* focus callbacks so we can track which widget is current */ XtAddCallback(html_widgets[0].html, XmNfocusCallback, (XtCallbackProc)focusCB, NULL); XtAddCallback(html_widgets[0].html, XmNlosingFocusCallback, (XtCallbackProc)focusCB, NULL); /* * now create the togglebuttons for XmHTML On/Off resources: * XmNenableBadHTMLWarnings * XmNstrictHTMLChecking * XmNenableBodyColors * XmNenableBodyImages * XmNenableDocumentColors * XmNenableDocumentFonts * XmNimageEnable * XmNenableOutlining * XmNfreezeAnimations * Also get the current default values for fontFamily and charset, we * want to honor the meta HTTP-EQUIV Content-Type info when it contains * a charset specification, and we also treat a meta NAME font spec * to set the basefont of a document (a kludge for some sort of style * control). */ /* get widget defaults */ XtVaGetValues(html_widgets[0].html, XmNanchorButtons, &html_config[OPTIONS_ANCHOR_BUTTONS].value, XmNhighlightOnEnter, &html_config[OPTIONS_HIGHLIGHT_ON_ENTER].value, XmNenableBadHTMLWarnings, &html_config[OPTIONS_DISABLE_WARNINGS].value, XmNstrictHTMLChecking, &html_config[OPTIONS_ENABLE_STRICT_HTML32].value, XmNenableBodyColors, &html_config[OPTIONS_ENABLE_BODYCOLORS].value, XmNenableBodyImages, &html_config[OPTIONS_ENABLE_BODYIMAGES].value, XmNenableDocumentColors, &html_config[OPTIONS_ENABLE_DOCUMENT_COLORS].value, XmNenableDocumentFonts, &html_config[OPTIONS_ENABLE_DOCUMENT_FONTS].value, XmNenableOutlining, &html_config[OPTIONS_ENABLE_OUTLINING].value, XmNfreezeAnimations, &html_config[OPTIONS_FREEZE_ANIMATIONS].value, XmNimageEnable, &html_config[OPTIONS_ENABLE_IMAGES].value, XmNfontFamily, &font, XmNcharset, &charset, NULL); /* save default font information */ strcpy(default_font, font); strcpy(current_font, font); /* save default charset information */ strcpy(default_charset, charset); strcpy(current_charset, charset); /* Anchor stuff */ pb = XtVaCreateManagedWidget("Buttoned Anchors", xmToggleButtonGadgetClass, option_menu, XmNset, html_config[OPTIONS_ANCHOR_BUTTONS].value, NULL); XtAddCallback(pb, XmNvalueChangedCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_ANCHOR_BUTTONS); html_config[OPTIONS_ANCHOR_BUTTONS].w = pb; pb = XtVaCreateManagedWidget("Highlight Anchors", xmToggleButtonGadgetClass, option_menu, XmNset, html_config[OPTIONS_HIGHLIGHT_ON_ENTER].value, NULL); XtAddCallback(pb, XmNvalueChangedCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_HIGHLIGHT_ON_ENTER); html_config[OPTIONS_HIGHLIGHT_ON_ENTER].w = pb; /* document body options */ pb = XtVaCreateManagedWidget("Enable Body Colors", xmToggleButtonGadgetClass, option_menu, XmNset, html_config[OPTIONS_ENABLE_BODYCOLORS].value, NULL); XtAddCallback(pb, XmNvalueChangedCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_ENABLE_BODYCOLORS); html_config[OPTIONS_ENABLE_BODYCOLORS].w = pb; pb = XtVaCreateManagedWidget("Enable Body Images", xmToggleButtonGadgetClass, option_menu, XmNset, html_config[OPTIONS_ENABLE_BODYIMAGES].value, NULL); XtAddCallback(pb, XmNvalueChangedCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_ENABLE_BODYIMAGES); html_config[OPTIONS_ENABLE_BODYIMAGES].w = pb; pb = XtVaCreateManagedWidget("Enable Color tag support", xmToggleButtonGadgetClass, option_menu, XmNset, html_config[OPTIONS_ENABLE_DOCUMENT_COLORS].value, NULL); XtAddCallback(pb, XmNvalueChangedCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_ENABLE_DOCUMENT_COLORS); html_config[OPTIONS_ENABLE_DOCUMENT_COLORS].w = pb; pb = XtVaCreateManagedWidget("Allow Font Switching", xmToggleButtonGadgetClass, option_menu, XmNset, html_config[OPTIONS_ENABLE_DOCUMENT_FONTS].value, NULL); XtAddCallback(pb, XmNvalueChangedCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_ENABLE_DOCUMENT_FONTS); html_config[OPTIONS_ENABLE_DOCUMENT_FONTS].w = pb; pb = XtVaCreateManagedWidget("Enable Text Outlining", xmToggleButtonGadgetClass, option_menu, XmNset, html_config[OPTIONS_ENABLE_OUTLINING].value, NULL); XtAddCallback(pb, XmNvalueChangedCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_ENABLE_OUTLINING); html_config[OPTIONS_ENABLE_OUTLINING].w = pb; pb = XtVaCreateManagedWidget("Anchored Image Tracking", xmToggleButtonGadgetClass, option_menu, XmNset, html_config[OPTIONS_FANCY_TRACKING].value, NULL); XtAddCallback(pb, XmNvalueChangedCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_FANCY_TRACKING); html_config[OPTIONS_FANCY_TRACKING].w = pb; XtVaCreateManagedWidget("separator", xmSeparatorGadgetClass, option_menu, NULL); pb = XtVaCreateManagedWidget("Bad HTML Warnings", xmToggleButtonGadgetClass, option_menu, XmNset, html_config[OPTIONS_DISABLE_WARNINGS].value, NULL); XtAddCallback(pb, XmNvalueChangedCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_DISABLE_WARNINGS); html_config[OPTIONS_DISABLE_WARNINGS].w = pb; pb = XtVaCreateManagedWidget("Strict HTML3.2 checking", xmToggleButtonGadgetClass, option_menu, XmNset, html_config[OPTIONS_ENABLE_STRICT_HTML32].value, NULL); XtAddCallback(pb, XmNvalueChangedCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_ENABLE_STRICT_HTML32); html_config[OPTIONS_ENABLE_STRICT_HTML32].w = pb; XtVaCreateManagedWidget("separator", xmSeparatorGadgetClass, option_menu, NULL); pb = XtVaCreateManagedWidget("Freeze Animations", xmToggleButtonGadgetClass, option_menu, XmNset, html_config[OPTIONS_FREEZE_ANIMATIONS].value, NULL); XtAddCallback(pb, XmNvalueChangedCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_FREEZE_ANIMATIONS); html_config[OPTIONS_FREEZE_ANIMATIONS].w = pb; XtVaCreateManagedWidget("separator", xmSeparatorGadgetClass, option_menu, NULL); pb = XtVaCreateManagedWidget("Enable Image Support", xmToggleButtonGadgetClass, option_menu, XmNset, html_config[OPTIONS_ENABLE_IMAGES].value, NULL); XtAddCallback(pb, XmNvalueChangedCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_ENABLE_IMAGES); html_config[OPTIONS_ENABLE_IMAGES].w = pb; pb = XtVaCreateManagedWidget("Auto Load Images", xmToggleButtonGadgetClass, option_menu, XmNset, html_config[OPTIONS_AUTO_IMAGE_LOAD].value, NULL); XtAddCallback(pb, XmNvalueChangedCallback, (XtCallbackProc)optionsCB, (XtPointer)OPTIONS_AUTO_IMAGE_LOAD); html_config[OPTIONS_AUTO_IMAGE_LOAD].w = pb; /***** * only add debug options when debug is defined during compilation * Note: resources involved are *always* available, but only get honored * when XmHTML was defined with DEBUG defined. *****/ #ifdef DEBUG DebugAddMenu(html_widgets[0].html, menubar, "Debug Options"); #endif /* if image support is disabled, make all image related buttons insens. */ if(!html_config[OPTIONS_ENABLE_IMAGES].value) { XtSetSensitive(html_config[OPTIONS_ENABLE_BODYIMAGES].w, False); XtSetSensitive(html_config[OPTIONS_AUTO_IMAGE_LOAD].w, False); XtSetSensitive(load_images, False); } /* a frame in which to display current hrefs */ fr = XtVaCreateManagedWidget("docInfoForm", xmFrameWidgetClass, form, XmNshadowType, XmSHADOW_IN, XmNleftAttachment, XmATTACH_FORM, XmNrightAttachment, XmATTACH_FORM, XmNbottomAttachment,XmATTACH_FORM, XmNrightOffset, 10, XmNleftOffset, 10, XmNbottomOffset, 10, NULL); pb = XtVaCreateManagedWidget("anchorRefFrame", xmFormWidgetClass, fr, NULL); /* create a label to contain name of current anchor */ xms = XmStringCreateLocalized(""); label = XtVaCreateManagedWidget("anchorRef", xmLabelGadgetClass, pb, XmNalignment, XmALIGNMENT_BEGINNING, XmNlabelString, xms, XmNtopAttachment, XmATTACH_FORM, XmNleftAttachment, XmATTACH_FORM, NULL); XmStringFree(xms); /* a label to indicate whether parser succeeded with verification */ xms = XmStringCreateLocalized(" "); verified = XtVaCreateManagedWidget("documentVerified", xmLabelGadgetClass, pb, XmNalignment, XmALIGNMENT_CENTER, XmNlabelString, xms, XmNtopAttachment, XmATTACH_FORM, XmNrightAttachment, XmATTACH_FORM, NULL); /* a label to indicate HTML 3.2 conformance */ html32 = XtVaCreateManagedWidget("documentHTML32", xmLabelGadgetClass, pb, XmNalignment, XmALIGNMENT_CENTER, XmNlabelString, xms, XmNtopAttachment, XmATTACH_FORM, XmNrightAttachment, XmATTACH_WIDGET, XmNrightWidget, verified, XmNrightOffset, 5, XmNleftOffset, 10, NULL); XmStringFree(xms); /* set final constraints on the html container */ XtVaSetValues((noframe ? html_widgets[0].html : frame), XmNbottomAttachment, XmATTACH_WIDGET, XmNbottomWidget, fr, XmNbottomOffset, 10, NULL); /* store the html widget as callback data for the file menu */ XtVaSetValues(menu, XmNuserData, html_widgets[0].html, NULL); /* all widgets have been created, manage them */ XtManageChild(form); XmMainWindowSetAreas(main_w, menubar, NULL, NULL, NULL, form); /***** * Popup menu for displaying image/link information * button 0: follow this link * button 1: open this image * button 2: show image info * button 3: enable/disable auto image tracking *****/ XtVaGetValues(html_widgets[0].html, XmNworkWindow, &pb, NULL); popup_menu = XmCreatePopupMenu(pb, "infoPopup", NULL, 0); for(i = 0; i < 4; i++) { pb = XtVaCreateManagedWidget("popupButton", xmPushButtonGadgetClass, popup_menu, NULL); XtAddCallback(pb, XmNactivateCallback, (XtCallbackProc)infoPopupCB, (XtPointer)i); } /* * An event handler that needs something of XmHTML should attach * itself to the workWindow */ pb = NULL; XtVaGetValues(html_widgets[0].html, XmNworkWindow, &pb, NULL); /* * Initialize object caching which we will be using for image caching. * We set a cache of 5MB (= 5242880 bytes). * Since the objects will all be of type XmImageInfo we need to have a * handle to the HTML widget using this cache. * Currently, the cache is global for *all* HTML widgets which could * give problems when images in frames are cached: the widget id provided * will *not* be the one which was used to create the XmImage for a frame. * Maybe I should move the client_data argument to the cleanObjectCache * function instead of having it here, or provide a sort of cacheContext. * (or even make it into an X Object). */ initCache(5*1024*1024, (cleanObjectProc)destroyCacheObject, (sizeObjectProc)getInfoSize, (XtPointer)html_widgets[0].html); /* and the event handler */ XtAddEventHandler(pb, ButtonPressMask, 0, (XtEventHandler)infoCB, popup_menu); XtRealizeWidget(toplevel); /* put it beneath all other windows so it will become the root */ if(root_window) XLowerWindow(XtDisplay(toplevel), XtWindow(toplevel)); /* The HTML widget has the focus */ XmProcessTraversal(html_widgets[0].html, XmTRAVERSE_CURRENT); /* if we have a file, load it */ if(use_file) { String filename; /* get full filename */ if((filename = resolveFile(use_file)) != NULL) { /* load the file, will also update the document cache */ loadAndOrJump(filename, NULL, True); free(filename); } } /* enter the event loop */ XtAppMainLoop(context); /* never reached, but keeps compiler happy */ exit(EXIT_SUCCESS); }