413 lines
11 KiB
C

/* This file is part of the Project Athena Zephyr Notification System.
* It is one of the source files comprising zwgc, the Zephyr WindowGram
* client.
*
* Created by: Marc Horowitz <marc@athena.mit.edu>
*
* $Id$
*
* Copyright (c) 1989 by the Massachusetts Institute of Technology.
* For copying and distribution information, see the file
* "mit-copyright.h".
*/
#include <sysdep.h>
#if (!defined(lint) && !defined(SABER))
static const char rcsid_X_driver_c[] = "$Id$";
#endif
#include <zephyr/mit-copyright.h>
/****************************************************************************/
/* */
/* The X driver: */
/* */
/****************************************************************************/
#ifndef X_DISPLAY_MISSING
#include <zephyr/zephyr.h>
#include "new_string.h"
#include "X_driver.h"
#include <X11/Xresource.h>
#include "new_memory.h"
#include "formatter.h"
#include "mux.h"
#include "variables.h"
#include "error.h"
#include "X_gram.h"
#include "xselect.h"
#include "unsigned_long_dictionary.h"
#include "zephyr.h"
char *app_instance;
/*
* x_dpy - the display we are outputting to
*/
Display *x_dpy = NULL;
/****************************************************************************/
/* */
/* Code to deal with getting X resources: */
/* */
/****************************************************************************/
/*
*
*/
#ifndef APPNAME
#define APPNAME "zwgc"
#endif
/*
*
*/
#ifndef APPCLASS
#define APPCLASS "Zwgc"
#endif
/*
* x_resources - our X resources from application resources, command line,
* and user's X resources.
*/
static XrmDatabase x_resources = NULL;
/*
* Internal Routine:
*
* int convert_string_to_bool(string text)
* Effects: If text represents yes/true/on, return 1. If text
* representes no/false/off, return 0. Otherwise,
* returns -1.
*/
static int
convert_string_to_bool(string text)
{
if (!strcasecmp("yes", text) || !strcasecmp("y", text) ||
!strcasecmp("true", text) || !strcasecmp("t", text) ||
!strcasecmp("on", text))
return(1);
else if (!strcasecmp("no", text) || !strcasecmp("n", text) ||
!strcasecmp("false", text) || !strcasecmp("f", text) ||
!strcasecmp("off", text))
return(0);
else
return(-1);
}
/*
*
*/
char *
get_string_resource(string name,
string class)
{
string full_name, full_class;
int status;
char *type;
XrmValue value;
full_name = string_Concat(APPNAME, ".");
full_name = string_Concat2(full_name, name);
full_class = string_Concat(APPCLASS, ".");
full_class = string_Concat2(full_class, class);
status = XrmGetResource(x_resources, full_name, full_class, &type, &value);
free(full_name);
free(full_class);
if (status != True)
return(NULL);
if (string_Neq(type, "String"))
return(NULL);
return(value.addr);
}
/*
*
*/
int
get_bool_resource(string name,
string class,
int default_value)
{
int result;
char *temp;
if (!(temp = get_string_resource(name, class)))
return(default_value);
result = convert_string_to_bool(temp);
if (result == -1)
result = default_value;
return(result);
}
static unsigned_long_dictionary color_dict = NULL;
/* Requires: name points to color name or hex string. name must be free'd
* eventually by the caller.
* Effects: returns unsigned long pixel value, or default if the
* color is not known by the server. If name is NULL, returns
* default;
*
* comment: caches return values from X server round trips. If name does
* not resolve, this fact is NOT cached, and will result in a round
* trip each time.
*/
unsigned long
x_string_to_color(char *name,
unsigned long def)
{
unsigned_long_dictionary_binding *binding;
int exists;
XColor xc;
if (name == NULL)
return(def);
binding = unsigned_long_dictionary_Define(color_dict,name,&exists);
if (exists) {
return((unsigned long) binding->value);
} else {
if (XParseColor(x_dpy,
DefaultColormapOfScreen(DefaultScreenOfDisplay(x_dpy)),
name,&xc)) {
if (XAllocColor(x_dpy,
DefaultColormapOfScreen(DefaultScreenOfDisplay(x_dpy)),
&xc)) {
binding->value = (unsigned long) xc.pixel;
return(xc.pixel);
} else {
ERROR2("Error in XAllocColor on \"%s\": using default color\n",
name);
}
} else {
ERROR2("Error in XParseColor on \"%s\": using default color\n",
name);
}
unsigned_long_dictionary_Delete(color_dict,binding);
return(def);
}
/*NOTREACHED*/
}
/*
* Standard X Toolkit command line options:
*/
static XrmOptionDescRec cmd_options[] = {
{"+rv", "*reverseVideo", XrmoptionNoArg, (XPointer) "off"},
{"+synchronous", "*synchronous", XrmoptionNoArg, (XPointer) "off"},
{"-background", "*background", XrmoptionSepArg, (XPointer) NULL},
{"-bd", "*borderColor", XrmoptionSepArg, (XPointer) NULL},
{"-bg", "*background", XrmoptionSepArg, (XPointer) NULL},
{"-bordercolor", "*borderColor", XrmoptionSepArg, (XPointer) NULL},
{"-borderwidth", ".borderWidth", XrmoptionSepArg, (XPointer) NULL},
{"-bw", ".borderWidth", XrmoptionSepArg, (XPointer) NULL},
{"-display", ".display", XrmoptionSepArg, (XPointer) NULL},
{"-fg", "*foreground", XrmoptionSepArg, (XPointer) NULL},
{"-fn", "*font", XrmoptionSepArg, (XPointer) NULL},
{"-font", "*font", XrmoptionSepArg, (XPointer) NULL},
{"-foreground", "*foreground", XrmoptionSepArg, (XPointer) NULL},
{"-geometry", ".geometry", XrmoptionSepArg, (XPointer) NULL},
{"-iconname", ".iconName", XrmoptionSepArg, (XPointer) NULL},
#ifdef CMU_ZWGCPLUS
{"-lifespan", "*lifespan", XrmoptionSepArg, (XPointer) NULL},
#endif
{"-name", ".name", XrmoptionSepArg, (XPointer) NULL},
{"-reverse", "*reverseVideo", XrmoptionNoArg, (XPointer) "on"},
{"-rv", "*reverseVideo", XrmoptionNoArg, (XPointer) "on"},
{"-transient", "*transient", XrmoptionNoArg, (XPointer) "on"},
{"-synchronous", "*synchronous", XrmoptionNoArg, (XPointer) "on"},
{"-title", ".title", XrmoptionSepArg, (XPointer) NULL},
{"-xrm", NULL, XrmoptionResArg, (XPointer) NULL} };
#define NUMBER_OF_OPTIONS ((sizeof (cmd_options))/ sizeof(cmd_options[0]))
/*
*
*/
static int
open_display_and_load_resources(int *pargc,
char **argv)
{
XrmDatabase temp_db1, temp_db2, temp_db3;
char *filename, *res, *xdef;
char dbasename[128];
/* Initialize X resource manager: */
XrmInitialize();
/*
* Parse X toolkit command line arguments (including -display)
* into resources:
*/
XrmParseCommand(&x_resources, cmd_options, NUMBER_OF_OPTIONS, APPNAME,
pargc, argv);
/*
* Try and open the display using the display specified if given.
* If can't open the display, return an error code.
*/
x_dpy = XOpenDisplay(get_string_resource("display", "display"));
if (!x_dpy)
return(1);
/* Read in our application-specific resources: */
sprintf(dbasename, "%s/zephyr/zwgc_resources", DATADIR);
temp_db1 = XrmGetFileDatabase(dbasename);
/*
* Get resources from the just opened display:
*/
xdef = XResourceManagerString(x_dpy);
if (xdef)
temp_db2 = XrmGetStringDatabase(xdef);
else
temp_db2 = NULL;
/*
* Merge the 4 sets of resources together such that when searching
* for resources, they are checking in the following order:
* command arguments, XENVIRONMENT resources, server resources,
* application resources
*/
XrmMergeDatabases(temp_db2, &temp_db1);
#if XlibSpecificationRelease > 4
/* X11 R5 per-screen resources */
res = XScreenResourceString (DefaultScreenOfDisplay (x_dpy));
if (res != NULL)
XrmMergeDatabases(XrmGetStringDatabase(res), &temp_db1);
#endif
/*
* Get XENVIRONMENT resources, if they exist, and merge
*/
filename = getenv("XENVIRONMENT");
if (filename)
{
temp_db3 = XrmGetFileDatabase(filename);
XrmMergeDatabases(temp_db3, &temp_db1);
}
XrmMergeDatabases(x_resources, &temp_db1);
x_resources = temp_db1;
return(0);
}
/*
* X_driver_ioerror: called by Xlib in case of an X IO error.
* Shouldn't return (according to man page).
*
* on IO error, we clean up and exit.
*
* XXX it would be better to set mux_end_loop_p, but we can't return to
* get there (Xlib will exit if this routine returns).
*
*/
static int
X_driver_ioerror(Display *display)
{
ERROR2("X IO error on display '%s'--exiting\n", DisplayString(display));
finalize_zephyr();
exit(1);
return 1;
}
/****************************************************************************/
/* */
/* Code to deal with initializing the driver: */
/* */
/****************************************************************************/
/*ARGSUSED*/
int
X_driver_init(char *drivername,
char notfirst,
int *pargc,
char **argv)
{
string temp;
int is_sync;
/*
* Attempt to open display and read resources, including from the
* command line. If fail, exit with error code, disabling this
* driver:
*/
if (open_display_and_load_resources(pargc, argv)) {
ERROR("Unable to open X display -- disabling X driver.\n");
return(1);
}
XSetIOErrorHandler(X_driver_ioerror);
/*
* For now, set some useful variables using resources:
*/
is_sync = get_bool_resource("synchronous", "Synchronous", 0);
if (is_sync)
XSynchronize(x_dpy, is_sync);
temp = get_string_resource("geometry", "Geometry");
if (temp)
var_set_variable("default_X_geometry", temp);
temp=strrchr(argv[0],'/');
app_instance=string_Copy(temp?temp+1:argv[0]);
color_dict = unsigned_long_dictionary_Create(37);
xshowinit();
x_gram_init(x_dpy);
xicccmInitAtoms(x_dpy);
mux_add_input_source(ConnectionNumber(x_dpy),
(void(*)(void *))x_get_input, x_dpy);
return(0);
}
/****************************************************************************/
/* */
/* The display routine itself: */
/* */
/****************************************************************************/
char *
X_driver(string text)
{
string text_copy;
desctype *desc;
int numstr, numnl;
text_copy = string_Copy(text);
desc = disp_get_cmds(text_copy, &numstr, &numnl);
xshow(x_dpy, desc, numstr, numnl);
free(text_copy);
free_desc(desc);
return(NULL);
}
#endif /* X_DISPLAY_MISSING */