Scrolling Game Development Kit Forum
SGDK Version 1 => Script => Topic started by: eric22222 on 2006-07-08, 04:09:16 AM
-
Hello, all! Seems Malaysia has more internet than I expected it would. Anyway, I've had more free time than I expected I would and want to put in some effort to learning VBScript. I pulled up the scripting, um, docutorial, and tried reading through. I'm lost. I'll definately need a little help on this.
So the script is sort of applied to the whole game whenever an "event" meets certain criteria, right? Like, a special function is activated, or the player interacts with a type of tile.
I'm also a bit confused on the whole "sub"/"subroutine" thing if anyone can clear up what that does...
Here's the example framework:
Sub Player_OnControllerMove(OldActions, NewActions)
'Your code here.
End Sub
HostObj.SinkObjectEvents ProjectObj.GamePlayer, "Player"
HostObj.ConnectEventsNow()
ProjectObj.GamePlayer.Play 16
So I think the OldActions/NewActions parts are replaced with 8 bits, each one representing a button (up, left, etc). I'm not sure what kind of stuff would be code, but I think I get that "End Sub" closes out whatever a "Sub" is. The rest of it completely throws me for a loop... Except the last line. I think I get that it starts the game.
Don't think I'm not reading through all the info. It just isn't making since to me yet...
-
You should probably also look at some example code. Have you looked at the code included in the docutorial yet (the script for GoldYoink)?
Since a computer represents everything in bits, 8 bits just means a number between 0 and 255 (inclusive). The example code you posted could be copied and pasted exactly as you showed into a script file and it would work. It wouldn't do anything besides play the game, but it would work. That's because OldActions and NewActions are variable names that will receive the numbers. So you could add code after the "Your code here" comment line that does something with these variables. Surely there's some example code that demonstrates this, and I think the Docutorial walks through some details too. But OldActions and NewActions are just variables whose values are supplied by GameDev to tell the script what the state of the controller was before and what it is now by packing 8 bits of information into each of the two variables. 255 represents 8 bits that are all turned on amd 0 represents no bits turned on. 4 represents the third bit turned on. 5 represents the first and third bit turned on. But really all you have to understand is how you extract information from those numbers using "And" and the constants the define which bits are associated with which controller buttons/positions.
I think the scripting wizard "record and playback" feature will generate script that demonstrates how you can use the ControllerMove event (and record the movement for playback later).
-
Okay... so if I wanted to remake part of my game where touching a tile in the category "red" activates the function "swap_to_red", it would look something like this?
Sub Player_OnAfterMoveSprites()
If --something that checks if the player touched a tile in category in "red"-- Then
ProjectObj.GamePlayer.ActivateFunction ProjectObj.Maps("Map1").Specials("Swap_to_red")
End If
End Sub
HostObj.SinkObjectEvents ProjectObj.GamePlayer, "Player"
HostObj.ConnectEventsNow()
ProjectObj.GamePlayer.Play 16
I'm just piecing this together from the code of the tutorial project and the scripting reference, so I'm not sure on much... I found a few parts of the list that look like they would complete the code (Touch, OnTouchTile, and OnTileInteraction), though I'm not sure how I fit them to this.
Oh, and what does "Dim" mean? I think once I learn the basic vocabulary and syntax, I should have a good foothold in the learning curve. Maybe then I'll know what's going on. ???
-
1) If you want to activate a function when a tile is touched, you should put the code in the OnTileInteractionEvent instead of in the OnAfterMoveSprites event.
2) Dim "declares" the existence of a variable in a particular "scope". If the "Dim" for a variable is within a function, then that variable is only accessible within that function (if you "Dim" and use the same variable name in another function, it will be an independent value). If you Dim the variable at the top of the script (not inside a Function/Sub) then you can use that variable anywhere in the code (as long as you don't Dim another copy inside the function).
-
Okay, I think I'm picking up on this... Dim says "this variable will be used in this area." I think I'm starting to see what's going on. I'll hopefully be able to look at a bunch of code now and pick up on some stuff. If I have anymore questions, I'll put 'em here. Thanks! ;D
Edit: Almost forgot...
Can I define any combination of letters as a variable? Like, something that's self-explanatory enough for me to remember what refers to what?
Secondly, what does putting a number in parentheses after a variable do? Is that to define an initial value?
Thirdly: this is probably extremely obvious to you scripters, but I'm gonna need to humble myself and ask: does capitalization matter in script?
Thanks! (again!)
-
1) Microsoft's official documentation on variable declaration rules for VBScript is available in this page in their MSDN documentation (http://msdn.microsoft.com/library/en-us/script56/html/307a1831-25b4-4b12-8d8b-7e9dc8443c49.asp).
2) Parentheses are used with variables when the variable represents an array (multiple values indexed by number stored in one variable). VBScript Arrays are described on the same page in MSDN.
3) VBScript is not case-sensitive. "MyVar" will refer to the same variable as "myvar".
-
One addendum is that when comparing litteral strings, capitalization does matter. For example:
"Test1" = "tEsT1"
will return false.
-
Okay, I'm trying do some pretty simple stuff in a test project. Looking at all the sample script I had, I decided I'd try to activate a function whenever the player walks up to a wall on its left. Here's what I've pieced together:
Sub Player_OnControllerMove(OldActions, NewActions)
Dim Lyr, Spr, Def
Set Lyr = ProjectObj.Maps("Map1").MapLayer("Test")
Set Spr = Lyr.Sprite(0)
Set Def = Spr.rDef
If Def.SolidTest(Spr.X-1,Spr.Y+31) Then
ProjectObj.GamePlayer.ActivateFunction ProjectObj.Maps("Map1").Specials("Move up")
End If
End Sub
HostObj.SinkObjectEvents ProjectObj.GamePlayer, "Player"
HostObj.ConnectEventsNow()
ProjectObj.GamePlayer.Play 16
The function is just a relative teleport (up 5 pixels). Nothing really happens, but the game does run. So what am I missing here?
Also, I've deduced that Sprite(0) refers to the first sprite in the list, but is there a way to refer to the player sprite? I saw PlayerSprite in the reference, but it says I have to call InitPlayerSprite first.
-
As long as the game is being run at the time the call is made, you don't have to worry about InitPlayerSprite. You can just use ProjectObj.GamePlayer.PlayerSprite
Perhaps it would be a good idea to tell us exactly what you think each step is doing so we know what's going on. And, of course, you should test the "Move Up" special function to make sure it works from within the game normally (i.e. without script).
-
Yes, "Move Up" works as it should without script. So here's my take on what this whole thing is doing:
Sub Player_OnControllerMove(OldActions, NewActions)
This is what starts this section of code, running through it after the controller's position moves.
Dim Lyr, Spr, Def
This sets "Lyr", "Spr", and "Def" and variables.
Set Lyr = ProjectObj.Maps("Map1").MapLayer("Test")
Set Spr = Lyr.Sprite(0)
Set Def = Spr.rDef
This section defines "Lyr" as, well, the only layer I have in this project. The second line defines "Spr" as the first sprite on that layer, and the third line defines "Def" as "Spr.rDef", which is referencing the player sprite's sprite definition.
If Def.SolidTest(Spr.X-1,Spr.Y+31) Then
This part was a bit counter-intuitive. I know it's testing for solidity at the point one pixel left of the bottom-left corner, but I would've thought it would need something like "If Def.SolidTest(...) = True Then". But this is how I saw it in the tutorial project's script, so I followed suit.
ProjectObj.GamePlayer.ActivateFunction ProjectObj.Maps("Map1").Specials("Move up")
This part activates the function "Move Up" in Map1. That's just followed by the "Ends" and the three lines that I can only assume is needed in every code.
HostObj.SinkObjectEvents ProjectObj.GamePlayer, "Player"
HostObj.ConnectEventsNow()
ProjectObj.GamePlayer.Play 16
I'm not sure on the first one, the second one kind of puts it all together, and the third one starts the game. So... how'd I do?
Extra credit: Could I leave out all the Dim and Set parts if I changed the SolidTest line to:
If ProjectObj.GamePlayer.PlayerSprite.rDef.SolidTest(Spr.X-1,Spr.Y+31) Then
-
I tried your script and it works on my test project. Perhaps you didn't get the script hooked up to the project. Also, it only activates when I'm up against a wall *and* I push or release a key (because it only activates when OnControllerMove fires). Can you do something to verify that the script is running at all? For example, add this line above the SinkObjectEvents line:
MsgBox "Test"
To answer your other question, yes you can eliminate a lot of the code by performing more work on a single line like that. But your sample code still shows references to the Spr variable so you didn't entirely eliminate all the variable references there.
-
Oh... Right... I need to make sure the script is running with the project... My mistake :-[.
And I see what you mean about the Spr thing, that I missed some. Ok, thanks again guys!
-
A quick note on the "counter-intuitive" line:
an "if" statement controls the flow of a program by only executing the code inside the block when a condition evaluates to true. For example:
if true then
'do Something
end if
will always execute, since true always evaluates to true. So whatever your expression is, if it evaluates to true, the code inside the block will run when it reaches that point in the program. So saying "if (statement) = true" is equivalent to saying "if true = true" or "if false = true" which, if you know your boolean logic, is equivalent to "if true AND true" or "if false AND true" which then re-evaluates back to "if true" and "if false", the same as the original, without checking to see if it's true: "if (statement)"
-
Now that the basics are about covered, I'm moving into doing actually useful things. The still-untitled game that Adam and I are working on is going to have a whole bunch of moves that can be done (think Super Mario 64 or Banjo-Kazooie). I think a good use of script would be to clean up all those special functions we've used just to get a spear to be thrown through the air. Here's what we're going for: you press whatever button we decide on, the spear is tossed in the direction you're facing (high non-100 inertia from a 45-degree vector), the spear hits the ground and changes to spear_in_ground upon touching the ground tiles.
The main hurdle we had there was detecting when the spear had come in contact with the ground since only player interactions are supported without scripting (As far as I know). Our clever workaround was to actually swap the player to the spear. The original player was still controlled by the arrow keys, but the camera centered on the spear. It was actually pretty cool, but would be awful if you threw it down a pit and couldn't see the character anymore.
With a bit of script, I'm sure this could be alot easier. So... first question.
Sub DoFireButton0()
With ProjectObj.GamePlayer.PlayerSprite
Select Case .rDef.Template.StateType
Case STATE_LEFT_RIGHT
If (.CurState Mod 2) = 0 Then
ProjectObj.GamePlayer.ActivateFunction ProjectObj.Maps("MyMapName").Specials("Throw_spear_left")
Else
ProjectObj.GamePlayer.ActivateFunction ProjectObj.Maps("MyMapName").Specials("Throw_spear_right")
End If
End Select
End With
End Sub
I've modified part of the shooting script. I'm hoping I changed all the necessary info. Of course, this is only step one. Eventually, I'll be able to get the entire spear toss in script (hopefully). Maybe if I were to replace the lines that cause the functions with the acutal meat of the function (just make it create the sprite).
So is this right so far? Just to see which way the player is facing (it is a left/right state sprite).
-
using script, you can use two functions to find when a sprite has hit solid:
Sub Player_OnBeforeMoveSprites()
Sprite.ReactToSolid
If Sprite.bHitSolid Then
'Do something
End If
End Sub
Where Sprite is the sprite you want to check. It might also be easier to create the sprites in the script so you can keep track of them rather than calling a special function to do it.
-
Yes, I plan on creating the sprite in script. I'd like to handle a majority of that kind of stuff in script.
Now, bHitSolid will return true if the sprite hits any solid tile. Is there anything that can determine the category of tile the sprite hits? I'm going to be replacing a spear that hits ground with Spear_stuck, which the player can spring from or something. How would I do something different when a wall or hill is hit?
Also, when I ran this without any script, the spear would not recognize being on the ground whenever it was on a slant... I'm guessing it's because the spear doesn't fill the tile, and the corners are empty. I think I can get around by checking the pixels just outside the corners for solidity.
But first off, checking tile category for a sprite: how?
-
Sprites don't have tile categories, only tiles have categories. But maybe I misunderstood your last statement there so I'll just give you a hint in the right direction as far as tile categories go. No time to elaborate at the moment. You'll want to look into the TileGroup object and/or the Category object. The Category object is basically a wrapper for a TileGroup to give it a name, associate it with a tileset, and make it save-able in a .GDP file. But you can make your own TileGroup object and put tile index values into it to to help quickly determine whether a particular tile value on the map is in a specific set. The easy thing, I suppose, is to define the category in the IDE and that refer to it in script via ProjectObj.Groups(...). Then all you have to do is figure out which tile the sprite hit and check the value/index of that tile against the Category/TileGroup object to see if it's one of the tiles the spear can stick into (or you can check the individual tile indexes yourself in script).
To get a tile index at a specific coordinate in the map, use the Data property of the Layer object. That property is documented in BMDXCtls.hlp which is in the BMDXCtls source code download. It's basically just an array of tile values (bytes).
-
I guess my last statement was a bit misleading... I was trying to say "how do I check I sprite to see if it has come in contact with a tile in a certain category."
I'm sure your post if full of all sorts of help, I've just had to back through it a few times to understand all these scripty terms. What I'm hearing is that tile categories aren't really accessible through script, that they're actually in TileGroups. And I really don't know what IDE is.
All I really want to do is have the spear stick one way when it hits flat floor, another on a hill, and another against a wall. I'm thinking it would be WAY easier just to check for solidity at the points below the bottom two corners (if both are solid, it's flat, else it's a slant) and check the points on the side to see if it hits a wall. Maybe that's just natural human arrogance talking, but I think my way's better ;).
I guess I'm just having a hard time understanding all this new stuff. I would like to know what IDE is, though.
-
Oh yeah, your way should work. I just didn't know exactly what you needed. I thought you maybe needed more than just information about which direction it hit solid at.
To clear up a couple questions: Categories and TileGroups are both available in script. TileGroups are the data itself and Categories are the objects that contain them and link them into the project with a name and such. IDE refers to the "Integrated Development Environment" -- a term used for many development environments. I consider GameDev to be such an environment. When I say IDE, I'm just referring to GameDev.exe -- the environment where you can edit your project in a user-friendly way.
-
Ok, so I've hit another snag. How do I create a sprite in script? From my searches and CTLR-f's, the best I've come up with is this:
Dim SpearSprite
Set SpearSprite = Lyr.pMap.SpriteDefs(Spear_in_ground_test).MakeInstance
(I already have Lyr defined)
Beyond that, I have no clue. The post about creating the realistic explosions had a bit more after that, but it didn't seem to want to work either. So what am I missing here?
-
After creating the sprite, you need to add it to the layer. There should be examples of this in the GoldYoink script.
-
Ok, I either filled in the blanks wrong, or I got the wrong piece of script:
Dim SpearSprite
Set SpearSprite = Lyr.pMap.SpriteDefs(Spear_in_ground_test).MakeInstance
SpearSprite.Name = "Spear_in_ground"
Set SpearSprite.rPath = Test1
Set SpearSprite.rLayer = Lyr
Set SpearSprite.Template = oMap.SpriteTemplates("Spear_in_ground")
Test.AddSpriteDef HostObj.AsObject(SpearSprite)
It quits at the line Set SpearSprite.rPath. I'm completely unsure on this one... The Gold Yoink script didn't use MakeInstance or even AddSprite. I've also noticed that the script for realistic explosions seems like it's doing something different. It does use MakeInstance, and then some other stuff. I'm guessing there's more than one way to make a sprite.
-
You're combining a few things here. You're trying to make a sprite definition and a sprite instance at the same time. What you want to be doing is just making an instance of an existing Sprite Definition, which don't have names themselves, or paths or layers or templates. Those all exist as part of the sprite definition. Also, you need to add the sprite to the layer. I think all you need to do, if you already have the sprite defined, is:
Dim SpearSprite
Set SpearSprte = Lyr.pMap.SpriteDefs("Spear_in_ground_test").MakeInstance
Lyr.AddSprite(SpearSprite)
-
Ok, so I'm getting the "Object variable or with Block variable not set" error... It's pointing at the line "Lyr.AddSprite(SpearSprite)". I think I missed putting something that should be there earlier in the script... Here's that section of script:
Sub Player_OnBeforeMoveSprites()
Dim I
Dim Lyr, Spr, Def
Set Lyr = ProjectObj.Maps("Test").MapLayer("Main")
I = 1
Do While I<Lyr.SpriteCount
Set Spr = Lyr.Sprite(I)
Set Def = Spr.rDef
If Def.Name = "Spear_test_right" Or Def.Name = "Spear_test_left" Then
If Def.SolidTest(Spr.X, Spr.Y+32) Or Def.SolidTest(Spr.X+31, Spr.Y+32) Then
If Def.SolidTest(Spr.x+16, Spr.Y+32) Then
Lyr.RemoveSprite I
Dim SpearSprite
Set SpearSprte = Lyr.pMap.SpriteDefs("Spear_in_ground_test").MakeInstance
Lyr.AddSprite(SpearSprite)
Else
Lyr.RemoveSprite I
End If
End If
End If
I = I + 1
Loop
End Sub
And I don't know what you did Durnurd, but the concept of instance vs definition clicked with me! I'm now on the side where it's incredibly easy to understand. About 499 clicks to go before I'm good at script :).
-
The error you're getting now is because one of your SpearSprite variable references is misspelled. Put Option Explicit as the first line of (each) script to prevent these kinds of errors by forcing you to declare every variable you use. I say "(each)" script because it needs to be after each "#Split" if you used #Split in your script at all.
-
Oh... whoops.
Well, it works now. I just had to make it so that the spear in the ground would appear where the flying spear disappeared. And I figured it out! Thanks to that explosion script, I learned how to set the position of a sprite with .X and .Y.
Thanks guys! (And expect more cries for help)