212 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			212 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C
		
	
	
	
| #include "xt.h"
 | |
| 
 | |
| #define MAX_CLASS	        128
 | |
| #define MAX_CALLBACK_PER_CLASS   10
 | |
| 
 | |
| typedef struct {
 | |
|     char *name;
 | |
|     int has_arg;
 | |
| } CALLBACK_INFO;
 | |
| 
 | |
| typedef struct {
 | |
|     WidgetClass class;
 | |
|     char *name;
 | |
|     CALLBACK_INFO cb[MAX_CALLBACK_PER_CLASS], *cblast;
 | |
|     XtResourceList sub_resources;
 | |
|     int num_resources;
 | |
| } CLASS_INFO;
 | |
| 
 | |
| static CLASS_INFO ctab[MAX_CLASS], *clast = ctab;
 | |
| 
 | |
| Generic_Predicate (Class)
 | |
| 
 | |
| Generic_Simple_Equal (Class, CLASS, wclass)
 | |
| 
 | |
| Generic_Print (Class, "#[class %s]", CLASS(x)->name)
 | |
| 
 | |
| Object Make_Class (class, name) WidgetClass class; char *name; {
 | |
|     Object c;
 | |
| 
 | |
|     c = Find_Object (T_Class, (GENERIC)0, Match_Xt_Obj, class);
 | |
|     if (Nullp (c)) {
 | |
| 	c = Alloc_Object (sizeof (struct S_Class), T_Class, 0);
 | |
| 	CLASS(c)->tag = Null;
 | |
| 	CLASS(c)->wclass = class;
 | |
| 	CLASS(c)->name = name;
 | |
| 	Register_Object (c, (GENERIC)0, (PFO)0, 0);
 | |
| 	/* See comment in Define_Class below */
 | |
| 	XtInitializeWidgetClass (class);
 | |
|     }
 | |
|     return c;
 | |
| }
 | |
| 
 | |
| Object Make_Widget_Class (class) WidgetClass class; {
 | |
|     register CLASS_INFO *p;
 | |
| 
 | |
|     for (p = ctab; p < clast; p++)
 | |
| 	if (p->class == class)
 | |
| 	    return Make_Class (class, p->name);
 | |
|     Primitive_Error ("undefined widget class ~s", Xt_Class_Name (class));
 | |
|     /*NOTREACHED*/
 | |
| }
 | |
| 
 | |
| static Object P_Find_Class (name) Object name; {
 | |
|     register CLASS_INFO *p;
 | |
|     register char *s = Get_Strsym (name);
 | |
| 
 | |
|     for (p = ctab; p < clast; p++) {
 | |
| 	if (streq (p->name, s))
 | |
| 	    return Make_Class (p->class, p->name);
 | |
|     }
 | |
|     Primitive_Error ("no such widget class: ~s", name);
 | |
|     /*NOTREACHED*/
 | |
| }
 | |
| 
 | |
| static Object P_Class_Existsp (name) Object name; {
 | |
|     register CLASS_INFO *p;
 | |
|     register char *s = Get_Strsym (name);
 | |
| 
 | |
|     for (p = ctab; p < clast; p++) {
 | |
| 	if (streq (p->name, s))
 | |
| 	    return True;
 | |
|     }
 | |
|     return False;
 | |
| }
 | |
| 
 | |
| char *Class_Name (class) WidgetClass class; {
 | |
|     register CLASS_INFO *p;
 | |
| 
 | |
|     for (p = ctab; p < clast && p->class != class; p++)
 | |
| 	;
 | |
|     if (p == clast)
 | |
| 	return "unknown";
 | |
|     return p->name;
 | |
| }
 | |
| 
 | |
| void Get_Sub_Resource_List (class, rp, np) WidgetClass class;
 | |
| 	XtResourceList *rp; Cardinal *np; {
 | |
|     register CLASS_INFO *p;
 | |
| 
 | |
|     for (p = ctab; p < clast && p->class != class; p++)
 | |
| 	;
 | |
|     if (p == clast)
 | |
| 	Primitive_Error ("undefined widget class ~s", Xt_Class_Name (class));
 | |
|     *np = p->num_resources;
 | |
|     *rp = p->sub_resources;
 | |
| }
 | |
| 
 | |
| static Object P_Class_Resources (c) Object c; {
 | |
|     Check_Type (c, T_Class);
 | |
|     return Get_Resources (CLASS(c)->wclass, XtGetResourceList, 1);
 | |
| }
 | |
| 
 | |
| static Object P_Class_Constraint_Resources (c) Object c; {
 | |
|     Check_Type (c, T_Class);
 | |
|     return Get_Resources (CLASS(c)->wclass, XtGetConstraintResourceList, 1);
 | |
| }
 | |
| 
 | |
| static Object P_Class_Sub_Resources (c) Object c; {
 | |
|     Check_Type (c, T_Class);
 | |
|     return Get_Resources (CLASS(c)->wclass, Get_Sub_Resource_List, 0);
 | |
| }
 | |
| 
 | |
| void Define_Class (name, class, r, nr) char *name; WidgetClass class;
 | |
| 	XtResourceList r; {
 | |
|     Set_Error_Tag ("define-class");
 | |
|     if (clast == ctab+MAX_CLASS)
 | |
| 	Primitive_Error ("too many widget classes");
 | |
|     /*
 | |
|      * The next line should read:
 | |
|      *    XtInitializeWidgetClass (class);
 | |
|      * However, there is a bug in Motif 1.1.4 that causes an application
 | |
|      * to drop core if the row-column widget class is initialized before
 | |
|      * the first vendor-shell widget has been created.
 | |
|      * Thus, we can't initialize any classes at this point; we will do
 | |
|      * it in Make_Class above instead.
 | |
|      * This essentially causes a class to be initialized the first time
 | |
|      * it is used.
 | |
|      */
 | |
|     clast->name = name;
 | |
|     clast->class = class;
 | |
|     clast->cb[0].name = XtNdestroyCallback;
 | |
|     clast->cb[0].has_arg = 0;
 | |
|     clast->cblast = clast->cb+1;
 | |
|     clast->sub_resources = r;
 | |
|     clast->num_resources = nr;
 | |
|     clast++;
 | |
| }
 | |
| 
 | |
| void Define_Callback (cl, s, has_arg) char *cl, *s; {
 | |
|     register CLASS_INFO *p;
 | |
| 
 | |
|     Set_Error_Tag ("define-callback");
 | |
|     for (p = ctab; p < clast; p++)
 | |
| 	if (streq (p->name, cl)) {
 | |
| 	    if (p->cblast == p->cb+MAX_CALLBACK_PER_CLASS)
 | |
| 		Primitive_Error ("too many callbacks for this class");
 | |
| 	    p->cblast->name = s;
 | |
| 	    p->cblast->has_arg = has_arg;
 | |
| 	    p->cblast++;
 | |
| 	    return;
 | |
| 	}
 | |
|     Primitive_Error ("undefined class");
 | |
| }
 | |
| 
 | |
| PFX2S Find_Callback_Converter (c, name, sname) WidgetClass c; char *name;
 | |
| 	Object sname; {
 | |
|     register CLASS_INFO *p;
 | |
|     register CALLBACK_INFO *q;
 | |
|     PFX2S conv;
 | |
| 
 | |
|     for (p = ctab; p < clast; p++)
 | |
| 	if (p->class == c) {
 | |
| 	    for (q = p->cb; q < p->cblast; q++)
 | |
| 		if (streq (q->name, name)) {
 | |
| 		    if (q->has_arg) {
 | |
| 			char s1[128], s2[128], msg[256];
 | |
| 
 | |
| 			/* First look for a class specific converter
 | |
| 			 * then for a general one.  Callback converters
 | |
| 			 * have a prefix "callback:" to avoid name conflicts
 | |
| 			 * with converters for normal resources.
 | |
| 			 */
 | |
| 			sprintf (s1, "callback:%s-%s", p->name, name);
 | |
| 			conv = Find_Converter_To_Scheme (s1);
 | |
| 			if (conv == 0) {
 | |
| 			    sprintf(s2, "callback:%s", name);
 | |
| 			    conv = Find_Converter_To_Scheme (s2);
 | |
| 			    if (conv == 0) {
 | |
| 				sprintf (msg,
 | |
| 				    "no callback converter for %s or %s",
 | |
| 					s1, s2, name);
 | |
| 				Primitive_Error (msg);
 | |
| 			    }
 | |
| 			}
 | |
| 			return conv;
 | |
| 		    } else return (PFX2S)0;
 | |
| 		}
 | |
| 	    Primitive_Error ("no such callback: ~s", sname);
 | |
| 	}
 | |
|     Primitive_Error ("undefined widget class ~s", Xt_Class_Name (c));
 | |
|     /*NOTREACHED*/
 | |
| }
 | |
| 
 | |
| elk_init_xt_class () {
 | |
|     Generic_Define (Class, "class", "class?");
 | |
|     Define_Primitive (P_Find_Class,        "find-class",        1, 1, EVAL);
 | |
|     Define_Primitive (P_Class_Resources,   "class-resources",   1, 1, EVAL);
 | |
|     Define_Primitive (P_Class_Constraint_Resources,
 | |
|                                "class-constraint-resources",    1, 1, EVAL);
 | |
|     Define_Primitive (P_Class_Sub_Resources,
 | |
| 			       "class-sub-resources",           1, 1, EVAL);
 | |
|     Define_Primitive (P_Class_Existsp,     "class-exists?",     1, 1, EVAL);
 | |
|     /*
 | |
|      * Doesn't work with Motif-1.1.0:
 | |
|      *
 | |
|     Define_Class ("simple", simpleWidgetClass, (XtResourceList)0, 0);
 | |
|      */
 | |
|     Define_Class ("core", widgetClass, (XtResourceList)0, 0);
 | |
|     Define_Class ("constraint", constraintWidgetClass, (XtResourceList)0, 0);
 | |
|     Define_Class ("composite", compositeWidgetClass, (XtResourceList)0, 0);
 | |
| }
 |