U61 (v1.0.1) - Script tutorial




Disclaimer
==========


  Before you start, please note that this document is:

  * *not* a Lua tutorial (Lua is the language use for scripting in U61). Of
    course at the end of this document I hope you'll be able to do some basic
    Lua scripting since Lua is quite quick to learn, but if you want to do
    advanced Lua coding, go and check the official sources. I'm myself not a
    great Lua programmer so I won't be able to help you a lot anyway.

  * *not* a complete reference of what is possible to do with U61. It only
    shows how to get started with script writing, but if you really want to do
    complicated things, please read the rest of this site.

  However, it should help you write your first script lines, and I have good
  reasons to think these are the hardest to code.

  You do not need any special programming skills before reading this document.
  If you are curious and wish to learn how to toy with u61 it should be enough,
  provided that you know at least how to move, copy and edit files on your
  system. Of course it makes things easier to know about programming, but
  honestly Lua is not a difficult language to start with.



Step 1: create a script file
============================


  Well, let's not reinvent the wheel for instance, we'll just copy an existing
  file. Writing a complete file from scratch might indeed be too hard for a
  start.

  If you got an "official" U61 package (from http://www.ufoot.org for instance)
  you already have some scripts that came with the game. By the time I write
  this document, only the UNIX platform is supported, so the script files
  should be in "/usr/share/script/u61" or "/usr/local/share/script/u61" or in
  the directory where you installed U61.

  Now copy "/usr/local/share/u61/script/classic.lua" into
  "~/.u61/script/test.lua".

  If now you launch U61, and go to "Game options / Rules" you should see "test"
  appear in the list, use the arrows to move the selection on it, and the start
  a new game. You should have the impression to play with the "classic" rules,
  but in fact you are using the "test.lua" script.



Step 2: create an error
=======================


  OK, now scripting allows lots of liberty in programming. One of the basic
  consequences for U61 is that you can code almost anything in a script, you
  are pretty sure you won't get a protection fault. In fact the worst thing
  that can happen (and it does happen) is an infinite loop. The result is that
  the game hangs and you have to kill it.

  So what I propose is to deliberately create an error in the script and see
  what happens. This will help you deal with errors later. So just edit your
  brand new test.lua script file with your favorite editor (vim and GNU Emacs
  seem to work good but I would never mean to influence you in the choice of
  the editor) and go to, say line 93. There you just add the following text:

  \""56zboub!

  Now relaunch U61, which should have kept test.lua as your default script, and
  you should get nothing on the player(s) map(s) but instead a message on the
  console (provided that you have launched U61 from a terminal) that says:

  lua error: unexpected token;
    last token read: `\' at line 93 in file `/home/ufoot/.u61/test.lua'
  Lua error reading script, code is 1.

  And this is it. You have the line number of the error and a short description
  of it. This should help debugging.

  It is not necessary to quit and re-run completely u61 before trying out a new
  script. Indeed the script is reloaded from the disk each time you start a new
  game (with "Quick start" for instance). So all you need to do is:

  * Launch U61

  * Select your script with "Game options / Rules"

  * Start a game with "Quick start", view what happens.

  * Edit your file, save it.

  * Re-start the game with "Escape / Stop this game / Quick start" and view how
    your modifications influence the gameplay.

  As a start, just remove the faulty text I just told you to add from the
  script, same it and re-start a new game without quitting U61. The error
  should disappear.



Step 3: learn Lua basics
========================


  As I said before, this document is not a Lua reference, and it won't replace
  the reading of the official manual, which is available on this site anyway.
  Still, you do not need to be a Lua wizzard to tweak U61's scripts, as I'm not
  myself a very bright Lua programmer. So this short passage is just to give
  you a quick start.

  In U61, I essentially use Lua functions, there are other aspects of Lua
  programming than functions, but I think functions are easy to understand, so
  I use them. To declare a function, do like this:

  function my_function(par1,par2)
      ...
  end

  As you see, it's easy to declare a function, you might use this function
  elsewhere by calling:

  my_function("hello","world")

  Within a function, you can declare local values. Local values have a scope
  which is limited to the function where they are declared. Any function may
  return a value, here's an example:

  function calculate(a,b)
      local temp
      temp=a+b
      return temp
  end

  As you may have guessed, this function performs an addition, you can call:

  c=calculate(1,2)

  and the value of "c" will be 3.

  Lua has some common controls like "if...then" or "while...do". Here is a
  short example:

  if val==0 then
      result="zero"
  elseif val  <  0 then
      result="negative"
  else
      result="positive"
  end

  And another one:

  while i>0 do
      result=result*2
      i=i-1
  end

  To compare numeric values, you can use the operators:

  * == (equals)

  * ~= (different)

  * >
  * (greater)

  * <
  * (lower)

  * >
  * = (greater or equal)

  * <
  * = (lower or equal)

  Now if you have already programmed in another language like C, Pascal or
  Basic you know that almost as much as I do about Lua. I can tell you it's
  enough to start hacking.



Step 4: Try to understand the existing scripts
==============================================


  OK, now we know how to modify a script randomly and generate errors. Great,
  but you might want to do more than that! Let's start by a - very - short
  analysys of our test.lua script file.

  "user_rotate_left" is called when the rotate left key is pressed. In U61
  0.2.2 the code for user_rotate_left for the classic.lua script is:

  function user_rotate_left()
      rotate_block_left()    
  end

  This means the action to perform when "user_rotate_left" is called is to call
  "rotate_block_left". So what's the difference between "user_rotate_left" and
  "rotate_block_left"? Well, they do exactly the same but "user_rotate_left" is
  a reserved name, this function being what I call a "user callback", since
  it's called by the C++ engine and it's the responsibility of the scripter to
  code it correctly. "rotate_block_left" is totally unknown from the C++
  engine, it's just a function I created because I found it convenient to have
  a function that rotates a block already coded and usable elsewhere in the
  script. If we look at the code of "rotate_block_left", we find:

  function rotate_block_left()
      local x
      local y
      local i 
  
      i = u61_get_nb_items()-1
      while i>=0 do
          x = u61_get_item_x(i) 
          y = u61_get_item_y(i) 
          x,y = y,-x
          u61_set_item_x(i,x)
          u61_set_item_y(i,y)
          i = i-1
      end
  end

  Well, we won't detail the way it works, but we'll still notice that it uses
  the functions with a name like "u61_fonction_name". These functions belong to
  what I call the "system api". It's a set of functions which allow the user to
  query and modify U61's world. For instance "u61_get_nb_items" returns the
  size of the current block, which with the classic rules is generally 4. But
  beware, it could be more if you had a nasty curse, and in a general manner,
  it's good to write scripts that are usable in various situations.



Step 5: Toy arround
===================


  Now, we'll as a first example change our "test.lua" file so that when the
  user rotates a block, the block goes down of one square. This will forbid
  abusive rotation of the block and make the game harder.

  Just edit test.lua and find the code for "user_rotate_left" and
  "user_rotate_right". It should look like:

  function user_rotate_left()
      rotate_block_left()    
  end
  
  function user_rotate_right()
      rotate_block_right()    
  end

  Just replace this by:

  function user_rotate_left()
      rotate_block_left()
      u61_set_block_y(u61_get_block_y()+1)
  end
  
  function user_rotate_right()
      rotate_block_right()    
      translate_block_y(1)
  end

  And it should work! You'll notice that the 2 functions are coded differently.
  With "user_rotate_left" I used the U61 system API directly, whereas with
  "user_rotate_right" I used an Lua function which encapsulates the calls to
  the system API. I personnally find it a good idea to make very small
  functions but you are free to do whatever you want. But I believe some
  functions such as the user callbacks should be as small as possible, for in
  the long run it should make the sharing and merging of script files easier.



To be continued...
==================


  Well, that's all for now, I hope to enrich this document soon.

