From: PlasmaSheep Date: Thu, 15 Jul 2010 18:37:00 +0000 (+0000) Subject: Stupid, stupid redmine wiki. X-Git-Url: https://git.rm.cloudns.org/?a=commitdiff_plain;h=18618fb5f33992885bfd2e8672d677cb33e92081;p=xonotic%2Fxonotic.wiki.git Stupid, stupid redmine wiki. (Commit created by redmine exporter script from page "Introduction_to_QuakeC" version 3) --- diff --git a/Introduction_to_QuakeC.textile b/Introduction_to_QuakeC.textile index 512fa2a..5c15c73 100644 --- a/Introduction_to_QuakeC.textile +++ b/Introduction_to_QuakeC.textile @@ -52,7 +52,6 @@ To see what QuakeC looks like, here is an example: return nearest; } - (Note: _findchain_ is implemented in QuakeC for demonstration purposes only so one can see how to build a linked list, as this function is already built in to the engine and can be used directly) h2. Other resources @@ -73,23 +72,22 @@ h2. Declaring To declare a variable, the syntax is the same as in C: -float i; + float i; However, variables cannot be initialized in their declaration for historical reasons, and trying to do so would define a constant. Whenever a variable declaration could be interpreted as something else by the compiler, the _var_ keyword helps disambiguating. For example, -float(float a, float b) myfunc; + float(float a, float b) myfunc; is an old-style function declaration, while -var float(float a, float b) myfunc; + var float(float a, float b) myfunc; declares a variable of function type. An alternate and often more readable way to disambiguate variable declarations is using a _typedef_: - -typedef float(float, float) myfunc_t; - myfunc_t myfunc; + typedef float(float, float) myfunc_t; + myfunc_t myfunc; h2. Scope @@ -151,7 +149,7 @@ h2. entity The main object type in QuakeC is _entity_, a reference to an engine internal object. An _entity_ can be imagined as a huge struct, containing many _fields_. This is the only object type in the language. However, _fields_ can be added to the _entity_ type by the following syntax: -.float myfield; +//.float myfield;// and then all objects _e_ get a field that can be accessed like in _e.myfield_. @@ -162,54 +160,46 @@ If a field has not been set, it gets the usual zero value of the type when the o h2. fields A reference to such a field can be stored too, in a field variable. It is declared and used like - - -.float myfield; -// ... -// and in some function: -var .float myfieldvar; -myfieldvar = myfield; -e.myfieldvar = 42; - - -Field variables can be used as function parameters too - in that case you leave the _var_ keyword out, as it is not needed for disambiguation. +
+  .float myfield;
+  // ...
+  // and in some function:
+  var .float myfieldvar;
+  myfieldvar = myfield;
+  e.myfieldvar = 42;
+
Field variables can be used as function parameters too - in that case you leave the _var_ keyword out, as it is not needed for disambiguation. h2. functions Functions work just like in C: - -float sum3(float a, float b, float c) -{ - return a + b + c; -} - +
+  float sum3(float a, float b, float c)
+  {
+    return a + b + c;
+  }
However, the syntax to declare function pointers is simplified: - - -typedef float(float, float, float) op3func_t; -var float(float a, float b, float c) f; -op3func_t g; -f = sum3; -g = f; -print(ftos(g(1, 2, 3)), "\n"); // prints 6 - +
+  typedef float(float, float, float) op3func_t;
+  var float(float a, float b, float c) f;
+  op3func_t g;
+  f = sum3;
+  g = f;
+  print(ftos(g(1, 2, 3)), "\n"); // prints 6
Also note that the _var_ keyword is used again to disambiguate from a global function declaration. In original QuakeC by Id Software, this simplified function pointer syntax also was the only way to define functions (you may still encounter this in Xonotic code in a few places): - -float(float a, float b) sum2 = { - return a + b; -} - +
+  float(float a, float b) sum2 = {
+    return a + b;
+  }
A special kind of functions are built-in functions (defined by the engine). These are imported using so-called built-in numbers, with a syntax like - - -string strcat(string a, string b, ...) = #115; - +
+  string strcat(string a, string b, ...) = #115;
+
h2. void Just like in C, the _void_ type is a special placeholder type to declare that a function returns nothing. However, unlike in C, it is possible to declare variables of this type, although the only purpose of this is to declare a variable name without allocating space for it. The only occasion where this is used is the special _end_sys_globals_ and _end_sys_fields_ marker variables. @@ -219,44 +209,34 @@ h2. arrays As the QuakeC virtual machine provides no pointers or similar ways to handle arrays, array support is added by FTEQCC and very limited. Arrays can only be global, must have a fixed size (not dynamically allocated), and are a bit buggy and slow. Almost as great as in FORTRAN, except they can't be multidimensional either! You declare arrays like in C: - - -#define MAX_ASSASSINS 16 -entity assassins[MAX_ASSASSINS]; -#define BTREE_MAX_CHILDREN 5 -.entity btree_child[BTREE_MAX_CHILDREN]; -#define MAX_FLOATFIELDS 3 -var .float myfloatfields[MAX_FLOATFIELDS]; - - +
+  #define MAX_ASSASSINS 16
+  entity assassins[MAX_ASSASSINS];
+  #define BTREE_MAX_CHILDREN 5
+  .entity btree_child[BTREE_MAX_CHILDREN];
+  #define MAX_FLOATFIELDS 3
+  var .float myfloatfields[MAX_FLOATFIELDS];
+
The former is a global array of entities and can be used the usual way: - - -assassins[self.assassin_index] = self; - - +
+  assassins[self.assassin_index] = self;
+
The middle one is a global array of (allocated and constant) entity fields and **not** a field of array type (which does not exist), so its usage looks a bit strange: - -for(i = 0; i < BTREE_MAX_CHILDREN; ++i) - self.(btree_child[i]) = world; - + for(i = 0; i < BTREE_MAX_CHILDREN; ++i) + self.(btree_child[i]) = world; Note that this works: - -var .entity indexfield; -indexfield = btree_child[i]; -self.indexfield = world; - + var .entity indexfield; + indexfield = btree_child[i]; + self.indexfield = world; The latter one is a global array of (assignable) entity field variables, and looks very similar: - -myfloatfields[2] = health; -self.(myfloatfields[2]) = 0; -// equivalent to self.health = 0; - + myfloatfields[2] = health; + self.(myfloatfields[2]) = 0; + // equivalent to self.health = 0; Do not use arrays when you do not need to - using both arrays and function calls in the same expression can get messed up (**COMPILER BUG**), and arrays are slowly emulated using functions _ArrayGet*myfloatfields_ and _ArraySet*myfloatfields_ the compiler generates that internally do a binary search for the array index. @@ -268,30 +248,23 @@ h2. if not There is a second way to do a negated _if_: - -if not(expression) - ... - + if not(expression) + ... It compiles to slightly more efficient code than - -if(!expression) - ... - + if(!expression) + ... and has the notable difference that - -if not("") - ... - + if not("") + ... + will not execute (as //""_ counts as true in an _if// expression), but - if(!"") ... - will execute (as both //""_ and _string_null// is false when boolean operators are used on it)..