home *** CD-ROM | disk | FTP | other *** search
- =head1 NAME
-
- Tk2portableTk - how to make your B<Tk> source portable to other
- interpreted languages.
-
- =for category C Programming
-
- =head1 Author
-
- Ilya Zakharevich <ilya@math.ohio-state.edu> has contributed most of
- this document. Many thanks.
-
- =head1 DESCRIPTION
-
- B<PortableTk> is an attempt to make B<Tk> useful from other
- languages. Currently tk4.0 runs under Perl using this
- approach. Below, I<Lang> is the notation for an external language to
- which B<PortableTk> glues B<Tk> code.
-
- The main problem with using the code developed for B<TCL> with
- different languages is the absence of data types: almost anything is
- C<char*>. It makes automatic translation hopeless. However, if you
- C<typedef> several new symbols to be C<char*>, you can still use your
- code in B<TCL>, I<and> it will make the automatic translation
- possible.
-
- Another problem with the approach that "everything is a string" is
- impossibility to have a result that says "NotApplicable" without
- setting an error. Thus different B<Tk> command return different string
- values that mean "error happened", like C<"">, C<" "> or
- C<"??">. Other languages can be more flexible, so in B<portableTk> you
- should inform the compiler that what you want to return means "error"
- (see L<Setting variables>).
-
- Currently B<PortableTk> uses several different approachs
- to simplify translation: several B<TCL> functions that are especially
- dangerous to use are undefined, so you can easily find places that
- need to be updated to use Language-independent functions based on
- compiler warnings. Eventually a way to use these Language-independent
- functions under proper B<TCL> will be also provided. The end of this
- document provides a starting point for such a project.
-
- =head1 Structure of B<pTk>, porting your code
-
- B<pTk>, that is a port of B<Tk>, is very special with respect to porting
- of other code to B<portableTk>. The problem is that currently there is
- very little hope to merge the modifications back into B<Tk>, so a
- special strategy is needed to maintain this port. Do not use this
- strategy to port your own code.
-
- B<pTk> is produced from B<Tk> via a two-step process: first, some
- manual editing (the result is in the subdirectory C<mTk>), and second,
- automatic conversion by the C<munge> script (written in Perl). Thus the
- subdirectory C<pTk/mTk> contains code with minimal possible difference
- from the virgin B<Tk> code, so it is easier to merge(1) the
- differences between B<Tk> versions into modified code.
-
- It looks like the strategy for a portable code should be exactly
- opposite: starting from B<TCL>-based code, apply C<munge>, and then
- hand-edit the resulting code. Probably it is also possible to target
- your code to B<portableTk> from scratch, since this will make it
- possible to run it under a lot of I<Lang>uages.
-
- The only reason anyone would like to look into contents of C<pTk/mTk>
- directory is to find out which constructs are not supported by
- C<munge>. On the other hand, C<pTk> directory contains code that is
- conformant to B<portableTk>, so you can look there to find example code.
-
- C<munge> is the script that converts most common B<Tk> constructs to
- their C<portableTk> equivalent. For your code to qualify, you should
- follow B<Tk> conventions on indentation and names of variables, in
- particular, the array of arguments for the C<...CmdProc> should be
- called C<argv>.
-
- For details on what C<munge> can do, see
- L<Translation of some TCL functions>.
-
- =head1 B<PortableTk> API
-
- =head2 Checking what you are running under
-
- B<PortableTk> provides a symbol C<????>. If this symbol is defined,
- your source is compiled with it.
-
- =head2 New types of configuration options
-
- B<PortableTk> defines several new types of configuration options:
-
- TK_CONFIG_CALLBACK
- TK_CONFIG_LANGARG
- TK_CONFIG_SCALARVAR
- TK_CONFIG_HASHVAR
- TK_CONFIG_ARRAYVAR
- TK_CONFIG_IMAGE
-
- You should use them instead of TK_CONFIG_STRING whenever
- appropriate. This allows your application to receive a direct
- representation of the corresponding resource instead of the string
- representation, if this is possible under given language.
-
- ???? It looks like C<TK_CONFIG_IMAGE> and C<TK_CONFIG_SCALARVAR> set
- variables of type C<char*>.
-
- =head2 Language data
-
- The following data types are defined:
-
- =over 4
-
- =item C<Tcl_Obj *>
-
- is the main datatype of the language. This is a type that your C
- function gets pointers to for arguments when the corresponding I<Lang>
- function is called. The corresponding config type is
- C<TK_CONFIG_LANGARG>.
-
- This is also a type that keeps information about contents of I<Lang>
- variable.
-
- =item C<Var>
-
- Is a substitute for a C<char *> that contains name of variable. In
- I<Lang> it is an object that contains reference to another I<Lang>
- variable.
-
- =item C<LangResultSave>
-
- ????
-
- =item C<LangCallback>
-
- C<LangCallback*> a substitute for a C<char *> that contains command to
- call. The corresponding config type is C<TK_CONFIG_CALLBACK>.
-
- =item C<LangFreeProc>
-
- It is the type that the C<Lang_SplitList> sets. Before you call it,
- declare
-
- Args *args;
- LangFreeProc *freeProc = NULL;
- ...
- code = Lang_SplitList(interp, value,
- &argc, &args, &freeProc);
-
- After you use the split values, call
-
- if (args != NULL && freeProc) (*freeProc)(argc,args);
-
- It is not guaranteed that the C<args> can survive deletion of C<value>.
-
- =back
-
- =head2 Conversion
-
- The following macros and functions are used for conversion between
- strings and the additional types:
-
- LangCallback * LangMakeCallback(Tcl_Obj *)
- Tcl_Obj * LangCallbackArg(LangCallback *)
- char * LangString(Tcl_Obj *)
-
- After you use the result of LangCallbackArg(), you should free it with
- C<freeProc> C<LANG_DYNAMIC> (it is not guaranteed that any change of
- C<Tcl_Obj *> will not be reflected in <LangCallback>, so you cannot do
- LangSet...() in between, and you should reset it to C<NULL> if you
- want to do any further assignments to this C<Tcl_Obj *>).
-
- The following function returns the C<Tcl_Obj *> that is a reference to C<Var>:
-
- Tcl_Obj * LangVarArg(Var)
-
- ???? It is very anti-intuitive, I hope the name is changed.
-
- int LangCmpCallback(LangCallback *a,Tcl_Obj * b)
-
- (currently only a stub), and, at last,
-
- LangCallback * LangCopyCallback(LangCallback *)
-
- =head2 Callbacks
-
- Above we have seen the new datatype C<LangCallback> and the
- corresponding I<Config option> C<TK_CONFIG_CALLBACK>. The following
- functions are provided for manipulation of C<LangCallback>s:
-
- void LangFreeCallback(LangCallback *)
- int LangDoCallback(Tcl_Interp *,LangCallback *,
- int result,int argc, char *format,...)
-
- The argument C<format> of C<LangDoCallback> should contain a string that is
- suitable for C<sprintf> with optional arguments of C<LangDoCallback>.
- C<result> should be false if result of callback is not needed.
-
- int LangMethodCall(Tcl_Interp *,Tcl_Obj *,char *method,
- int result,int argc,...)
-
- ????
-
- Conceptually, C<LangCallback*> is a substitute for ubiquitous C<char *>
- in B<TCL>. So you should use C<LangFreeCallback> instead of C<ckfree>
- or C<free> if appropriate.
-
- =head2 Setting variables
-
- void LangFreeArg (Tcl_Obj *, Tcl_FreeProc *freeProc)
- Tcl_Obj * LangCopyArg (Tcl_Obj *);
- void Tcl_AppendArg (Tcl_Interp *interp, Tcl_Obj *)
- void LangSetString(Tcl_Obj * *, char *s)
- void LangSetDefault(Tcl_Obj * *, char *s)
-
- These two are equivalent unless s is an empty string. In this case
- C<LangSetDefault> behaves like C<LangSetString> with C<s==NULL>, i.e.,
- it sets the current value of the I<Lang> variable to be false.
-
- void LangSetInt(Tcl_Obj * *,int)
- void LangSetDouble(Tcl_Obj * *,double)
-
- The I<Lang> functions separate uninitialized and initialized data
- comparing data with C<NULL>. So the declaration for an C<Tcl_Obj *> should
- look like
-
- Tcl_Obj * arg = NULL;
-
- if you want to use this C<arg> with the above functions. After you are
- done, you should use C<LangFreeArg> with C<TCL_DYNAMIC> as C<freeProc>.
-
- =head2 Language functions
-
- Use
-
- =over 4
-
- =item C<int LangNull(Tcl_Obj *)>
-
- to check that an object is false;
-
- =item C<int LangStringMatch(char *string, Tcl_Obj * match)>
-
- ????
-
- =item C<void LangExit(int)>
-
- to make a proper shutdown;
-
- =item C<int LangEval(Tcl_Interp *interp, char *cmd, int global)>
-
- to call I<Lang> C<eval>;
-
- =item C<void Lang_SetErrorCode(Tcl_Interp *interp,char *code)>
-
- =item C<char *Lang_GetErrorCode(Tcl_Interp *interp)>
-
- =item C<char *Lang_GetErrorInfo(Tcl_Interp *interp)>
-
- =item C<void LangCloseHandler(Tcl_Interp *interp,Tcl_Obj * arg,FILE *f,Lang_FileCloseProc *proc)>
-
- currently stubs only;
-
- =item C<int LangSaveVar(Tcl_Interp *,Tcl_Obj * arg,Var *varPtr,int type)>
-
- to save the structure C<arg> into I<Lang> variable C<*varPtr>;
-
- =item C<void LangFreeVar(Var var)>
-
- to free the result;
-
- =item C<int LangEventCallback(Tcl_Interp *,LangCallback *,XEvent *,KeySym)>
-
- ????
-
- =item C<int LangEventHook(int flags)>
-
- =item C<void LangBadFile(int fd)>
-
- =item C<int LangCmpConfig(char *spec, char *arg, size_t length)>
-
- unsupported????;
-
- =item C<void Tcl_AppendArg (Tcl_Interp *interp, Tcl_Obj *)>
-
- =back
-
- Another useful construction is
-
- Tcl_Obj * variable = LangFindVar(interp, Tk_Window tkwin, char *name);
-
- After using the above function, you should call
-
- LangFreeVar(Var variable);
-
- ???? Note discrepancy in types!
-
- If you want to find the value of a variable (of type C<Tcl_Obj *>) given the
- variable name, use C<Tcl_GetVar(interp, varName, flags)>. If you are
- interested in the string value of this variable, use
- C<LangString(Tcl_GetVar(...))>.
-
- To get a B<C> array of C<Tcl_Obj *> of length C<n>, use
-
- Tcl_Obj * *args = LangAllocVec(n);
- ...
- LangFreeVec(n,args);
-
- You can set the values of the C<Tcl_Obj *>s using C<LangSet...> functions,
- and get string value using C<LangString>.
-
- If you want to merge an array of C<Tcl_Obj *>s into one C<Tcl_Obj *> (that will
- be an array variable), use
-
- result = Tcl_Merge(listLength, list);
-
- =head2 Translation of some TCL functions
-
- We mark items that can be dealt with by C<munge> by I<Autoconverted>.
-
- =over 4
-
- =item C<Tcl_AppendResult>
-
- does not take C<(char*)NULL>, but C<NULL> as delimiter. I<Autoconverted>.
-
- =item C<Tcl_CreateCommand>, C<Tcl_DeleteCommand>
-
- C<Tk_CreateWidget>, C<Tk_DeleteWidget>, the second argument is the
- window itself, not the pathname. I<Autoconverted>.
-
- =item C<sprintf(interp-E<gt>result, "%d %d %d %d",...)>
-
- C<Tcl_IntResults(interp,4,0,...)>. I<Autoconverted>.
-
- =item C<interp-E<gt>result = "1";>
-
- C<Tcl_SetResult(interp,"1", TCL_STATIC)>. I<Autoconverted>.
-
- =item Reading C<interp-E<gt>result>
-
- C<Tcl_GetResult(interp)>. I<Autoconverted>.
-
- =item C<interp-E<gt>result = Tk_PathName(textPtr-E<gt>tkwin);>
-
- C<Tk_WidgetResult(interp,textPtr-E<gt>tkwin)>. I<Autoconverted>.
-
- =item Sequence C<Tcl_PrintDouble, Tcl_PrintDouble, ..., Tcl_AppendResult>
-
- Use a single command
-
- void Tcl_DoubleResults(Tcl_Interp *interp, int append,
- int argc,...);
-
- C<append> governs whether it is required to clear the result first.
-
- A similar command for C<int> arguments is C<Tcl_IntResults>.
-
- =item C<Tcl_SplitList>
-
- Use C<Lang_SplitList> (see the description above).
-
- =back
-
- =head1 Translation back to TCL
-
- To use your B<portableTk> program with B<TCL>, put
-
- #include "ptcl.h"
-
- I<before> inclusion of C<tk.h>, and link the resulting code with
- C<ptclGlue.c>.
-
- These files currently implement the following:
-
- =over 4
-
- =item Additional config types:
-
- TK_CONFIG_CALLBACK
- TK_CONFIG_LANGARG
- TK_CONFIG_SCALARVAR
- TK_CONFIG_HASHVAR
- TK_CONFIG_ARRAYVAR
- TK_CONFIG_IMAGE
-
- =item Types:
-
- Var, Tcl_Obj *, LangCallback, LangFreeProc.
-
- =item Functions and macros:
-
- Lang_SplitList, LangString, LangSetString, LangSetDefault,
- LangSetInt, LangSetDouble Tcl_ArgResult, LangCallbackArg,
- LangSaveVar, LangFreeVar,
- LangFreeSplitProc, LangFreeArg, Tcl_DoubleResults, Tcl_IntResults,
- LangDoCallback, Tk_WidgetResult, Tcl_CreateCommand,
- Tcl_DeleteCommand, Tcl_GetResult.
-
- =back
-
- Current implementation contains enough to make it possible to compile
- C<mTk/tkText*.[ch]> with the virgin B<Tk>.
-
- =head2 New types of events ????
-
- PortableTk defines following new types of events:
-
- TK_EVENTTYPE_NONE
- TK_EVENTTYPE_STRING
- TK_EVENTTYPE_NUMBER
- TK_EVENTTYPE_WINDOW
- TK_EVENTTYPE_ATOM
- TK_EVENTTYPE_DISPLAY
- TK_EVENTTYPE_DATA
-
- and a function
-
- char * Tk_EventInfo(int letter,
- Tk_Window tkwin, XEvent *eventPtr,
- KeySym keySym, int *numPtr, int *isNum, int *type,
- int num_size, char *numStorage)
-
- =head1 Checking for trouble
-
- If you start with working TCL code, you can start convertion using
- the above hints. Good indication that you are doing is OK is absence
- of C<sprintf> and C<sscanf> in your code (at least in the part that is
- working with interpreter).
-
- =head1 Additional API
-
- What is described here is not included into base B<portableTk>
- distribution. Currently it is coded in B<TCL> and as Perl macros (core
- is coded as functions, so theoretically you can use the same object
- files with different interpreted languages).
-
- =head2 C<ListFactory>
-
- Dynamic arrays in B<TCL> are used for two different purposes: to
- construct strings, and to construct lists. These two usages will have
- separate interfaces in other languages (since list is a different type
- from a string), so you should use a different interface in your code.
-
- The type for construction of dynamic lists is C<ListFactory>. The API
- below is a counterpart of the API for construction of dynamic lists
- in B<TCL>:
-
- void ListFactoryInit(ListFactory *)
- void ListFactoryFinish(ListFactory *)
- void ListFactoryFree(ListFactory *)
- Tcl_Obj * * ListFactoryArg(ListFactory *)
- void ListFactoryAppend(ListFactory *, Tcl_Obj * *arg)
- void ListFactoryAppendCopy(ListFactory *, Tcl_Obj * *arg)
- ListFactory * ListFactoryNewLevel(ListFactory *)
- ListFactory * ListFactoryEndLevel(ListFactory *)
- void ListFactoryResult(Tcl_Interp *, ListFactory *)
-
- The difference is that a call to C<ListFactoryFinish> should precede the
- actual usage of the value of C<ListFactory>, and there are two
- different ways to append an C<Tcl_Obj *> to a C<ListFactory>:
- ListFactoryAppendCopy() guarantees that the value of C<arg> is copied
- to the list, but ListFactoryAppend() may append to the list a
- reference to the current value of C<arg>. If you are not going to change
- the value of C<arg> after appending, the call to ListFactoryAppend may
- be quicker.
-
- As in B<TCL>, the call to ListFactoryFree() does not free the
- C<ListFactory>, only the objects it references.
-
- The functions ListFactoryNewLevel() and ListFactoryEndLevel() return a
- pointer to a C<ListFactory> to fill. The argument of
- ListFactoryEndLevel() cannot be used after a call to this function.
-
- =head2 DStrings
-
- Production of strings are still supported in B<portableTk>.
-
- =head2 Accessing C<Tcl_Obj *>s
-
- The following functions for getting a value of an C<Tcl_Obj *> I<may> be
- provided:
-
- double LangDouble(Tcl_Obj *)
- int LangInt(Tcl_Obj *)
- long LangLong(Tcl_Obj *)
- int LangIsList(Tcl_Obj * arg)
-
- The function LangIsList() is supported only partially under B<TCL>,
- since there is no data types. It checks whether there is a space
- inside the string C<arg>.
-
- =head2 Assigning numbers to C<Tcl_Obj *>s
-
- While LangSetDouble() and LangSetInt() are supported ways to assign
- numbers to assign an integer value to a variable, for the sake of
- efficiency under B<TCL> it is supposed that the destination of these
- commands was massaged before the call so it contains a long enough
- string to sprintf() the numbers inside it. If you are going to
- immediately use the resulting C<Tcl_Obj *>, the best way to do this is to
- declare a buffer in the beginning of a block by
-
- dArgBuffer;
-
- and assign this buffer to the C<Tcl_Obj *> by
-
- void LangSetDefaultBuffer(Tcl_Obj * *)
-
- You can also create the buffer(s) manually and assign them using
-
- void LangSetBuffer(Tcl_Obj * *, char *)
-
- This is the only choice if you need to assign numeric values to
- several C<Tcl_Obj *>s simultaneously. The advantage of the first approach is
- that the above declarations can be made C<nop>s in different languages.
-
- Note that if you apply C<LangSetDefaultBuffer> to an C<Tcl_Obj *> that
- contains some value, you can create a leak if you do not free that
- C<Tcl_Obj *> first. This is a non-problem in real languages, but can be a
- trouble in C<TCL>, unless you use only the above API.
-
- =head2 Creating new C<Tcl_Obj *>s
-
- The API for creating a new C<Tcl_Obj *> is
-
- void LangNewArg(Tcl_Obj * *, LangFreeProc *)
-
- The API for creating a new C<Tcl_Obj *> is absent. Just initialize C<Tcl_Obj *> to
- be C<NULL>, and apply one of C<LangSet...> methods.
-
- After you use this C<Tcl_Obj *>, it should be freed thusly:
-
- C<LangFreeArg(arg, freeProc)>.
-
- =head2 Evaluating a list
-
- Use
-
- int LangArgEval(Tcl_Interp *, Tcl_Obj * arg)
-
- Here C<arg> should be a list to evaluate, in particular, the first
- element should be a C<LangCallback> massaged to be an C<Tcl_Obj *>. The
- arguments can be send to the subroutine by reference or by value in
- different languages.
-
- =head2 Getting result as C<Tcl_Obj *>
-
- Use C<Tcl_ArgResult>. It is not guaranteed that result survives this
- operation, so the C<Tcl_Obj *> you get should be the only mean to access the
- data from this moment on. After you use this C<Tcl_Obj *>, you should free
- it with C<freeProc> C<LANG_DYNAMIC> (you can do LangSet...() in between).
-
- =cut
-
-