1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148
|
/*
* ------------------------------------------------------------------------
* PACKAGE: [incr Tk]
* DESCRIPTION: Building mega-widgets with [incr Tcl]
*
* [incr Tk] provides a framework for building composite "mega-widgets"
* using [incr Tcl] classes. It defines a set of base classes that are
* specialized to create all other widgets.
*
* This file defines the initialization and facilities common to all
* mega-widgets.
*
* ========================================================================
* AUTHOR: Michael J. McLennan
* Bell Labs Innovations for Lucent Technologies
* mmclennan@lucent.com
* http://www.tcltk.com/itcl
* ========================================================================
* Copyright (c) 1993-1998 Lucent Technologies, Inc.
* ------------------------------------------------------------------------
* See the file "license.terms" for information on usage and redistribution
* of this file, and for a DISCLAIMER OF ALL WARRANTIES.
*/
#include "itkInt.h"
/*
* ------------------------------------------------------------------------
* Itk_ConfigBodyCmd()
*
* Replacement for the usual "itcl::configbody" command. Recognizes
* mega-widget options included in a class definition. Options are
* identified by their "switch" name, but without the "-" prefix:
*
* itcl::configbody <class>::<itkOption> <body>
*
* Handles bodies for public variables as well:
*
* itcl::configbody <class>::<publicVar> <body>
*
* If an <itkOption> is found, it has priority over public variables.
* If <body> has the form "@name" then it is treated as a reference
* to a C handling procedure; otherwise, it is taken as a body of
* Tcl statements.
*
* Returns TCL_OK/TCL_ERROR to indicate success/failure.
* ------------------------------------------------------------------------
*/
int
Itk_ConfigBodyCmd(
void *dummy, /* unused */
Tcl_Interp *interp, /* current interpreter */
int objc, /* number of arguments */
Tcl_Obj *const objv[]) /* argument objects */
{
int result = TCL_OK;
char *token;
const char *head;
const char *tail;
ItclClass *iclsPtr;
ItclMemberCode *mcode;
ItkClassOptTable *optTable;
Tcl_HashEntry *entry;
ItkClassOption *opt;
Tcl_DString buffer;
if (objc != 3) {
Tcl_WrongNumArgs(interp, 1, objv, "class::option body");
return TCL_ERROR;
}
/*
* Parse the member name "namesp::namesp::class::option".
* Make sure that a class name was specified, and that the
* class exists.
*/
token = Tcl_GetString(objv[1]);
Itcl_ParseNamespPath(token, &buffer, &head, &tail);
if (!head || *head == '\0') {
Tcl_AppendStringsToObj(Tcl_GetObjResult(interp),
"missing class specifier for body declaration \"", token, "\"",
(char*)NULL);
result = TCL_ERROR;
goto configBodyCmdDone;
}
iclsPtr = Itcl_FindClass(interp, head, /* autoload */ 1);
if (iclsPtr == NULL) {
result = TCL_ERROR;
goto configBodyCmdDone;
}
/*
* Look first for a configuration option with that name.
* If it is not found, assume the reference is for a public
* variable, and use the usual "configbody" implementation
* to handle it.
*/
optTable = Itk_FindClassOptTable(iclsPtr);
opt = NULL;
if (optTable) {
Tcl_DString optName;
Tcl_DStringInit(&optName);
Tcl_DStringAppend(&optName, "-", -1);
Tcl_DStringAppend(&optName, tail, -1);
entry = Tcl_FindHashEntry(&optTable->options,
Tcl_DStringValue(&optName));
if (entry) {
opt = (ItkClassOption*)Tcl_GetHashValue(entry);
}
Tcl_DStringFree(&optName);
}
if (opt == NULL) {
result = Itcl_ConfigBodyCmd(dummy, interp, objc, objv);
goto configBodyCmdDone;
}
/*
* Otherwise, change the implementation for this option.
*/
token = Tcl_GetString(objv[2]);
if (Itcl_CreateMemberCode(interp, iclsPtr, (char*)NULL, token,
&mcode) != TCL_OK) {
result = TCL_ERROR;
goto configBodyCmdDone;
}
Itcl_PreserveData((void *)mcode);
#ifdef NOTDEF
Itcl_EventuallyFree((void *)mcode, (Tcl_FreeProc *)Itcl_DeleteMemberCode);
#endif
if (opt->codePtr) {
Itcl_ReleaseData((void *)opt->codePtr);
}
opt->codePtr = mcode;
configBodyCmdDone:
Tcl_DStringFree(&buffer);
return result;
}
|