Zh/Authoring a Logical Entity
逻辑实体是最简单的实体类型,因为它们是不可见的,它们只负责处理与其他实体的输入输出。比如说math_counter
实体负责存储一个可以增加或相减的变量,其他的实体可以通过输入修改它的值,并从输出获取它的数值。
在这个教程中,我们将创建一种逻辑实体,它将存储一个变量,每次收到正确的输入的时候都会增加这个变量。变量达到一个设定好的数值后,这个实体就会引发一个输出。
创建源文件
此教程假设你使用Visual Studio或者Visual C++ Express。请参考编译器的选择。
在Server
工程内创建一个新的文件夹(“新建筛选器”),随意起一个名字。然后添加新建项,选择.cpp文件,并命名为” sdk_logicalentity”。这个文件将包含此教程所有的代码。
给每一个实体单独的.cpp可以减少负担,因为编译器只需要编译修改过的cpp文件,所以编译的速度会加快。
包含文件
每个.cpp文件都是和工程内的其他文件隔离开的。为了避免这样的情况,我们可以包含“头文件”,告诉编译器工程内的哪个部分是我们需要的。在这个极其简单的实体中,我们只需要包含一个文件:
#include "cbase.h"
cbase.h
提供了Valve设定的建立实体的基本设置,每个为起源所编写的.cpp文件都必须包含这个文件。字母”c”是起源中所使用的一个前缀,代表这个代码属于服务端(客户端一般是”c_”)。
类的声明
这一篇有点长,但是你看完之后会对c艹的运作方式有初步的了解。
几乎每一个运算过程都是由“对象”(objects)处理的。“类”(class)就是软件生成对象时所使用的框架。创建一种新的实体就意味着创建一个新的类。
class CMyLogicalEntity : public CLogicalEntity
{
public:
DECLARE_CLASS( CMyLogicalEntity, CLogicalEntity );
... // 不要把这一行也复制进去!
};
这就是“类的声明”,告诉c++编译器我们想让这个类的对象存储什么内容,运行怎样的函数。
代码的第一行,我们用class
指令声明了一个类。我们给它起名叫CMyLogicalEntity
,冒号后面告诉编译器,这个类是从已有的类CLogicalEntity
继承的,因为之前包含了cbase.h
,所以我们可以引用这个类。。
继承 的意思是这个类是建立在已有的类的基础上的。 CLogicalEntity
本身是比较简单的, 但是它所继承的类CBaseEntity
里,包含了大量的代码。由此可见如果没有继承的话,创建一个新的实体会把人折磨疯的。
class指令写完后,我们使用了一对花括号“{}”。花括号里面的代码就是一套组合;在本例子里,这个组合就是类本身。
接下来是修饰符”public:
”, 在public: 下的成员,是类的公有成员,这个类的成员可以被其他类直接访问,而private: 下的是私有成员,只有类的内部能访问,类的外部无法访问的。你应该在有必要的情况下才建立公有成员:请参考维基百科:封装。
第一条指令
我们现在可以设置指令了。DECLARE_CLASS
是一个 宏定义,编译的时候它将自动转换成创建新实体所需要的变量和函数。
圆括号里的内容我们称之为”参数”。这些参数传递了必要的信息。你也许已经注意到参数之间使用逗号“,”分隔。
行尾的分号“;”说明这一条指令已经结束。因为编译器基本无视换行和空格,所以需要分号来告诉它这一行已经结束。
声明的结束 (暂时)
请不要输入...
- 这只是代表这个地方还有其他的声明。我们先跳过这一步,用花括号和分号};
结束这个类的声明。

}
后面不用加 ;
,你只需要在声明类的时候这么做
声明一个 DATADESC,函数和构造函数
回到省略号的地方添加以下内容:
DECLARE_DATADESC();
//构造函数
CMyLogicalEntity ()
{
m_nCounter = 0;
}
// Input function
void InputTick( inputdata_t &inputData );
第一行是另一个宏,为等下的DATADESC列表做准备。第二行是注释,//
,让其他人了解你的代码(包括你自己,也许以后你会忘记)的含义。
第三行是类的”构造函数”。它是”函数”的一种:将一段经常需要使用的代码封装起来,在需要使用时可以直接调用。因为它有和类相同的名称,所以创建一个新的对象的时候这个构造函数就会被调用。在本例中,我们将m_nCounter初始化为0.变量是没有初始值的,如果在给它们赋值之前引用它们的话会发生不可预测的错误。
We've just "defined" the constructor in its entirety, but most functions (including constructors for larger classes) are too big to fit into a class declaration. So instead, we declare them and write their contents later on in the .cpp file.
Accordingly, the last line in this code snippet is a declaration of the function that will be called whenever the correct input is received and, if you remember back to our original goal, increment our stored value by 1. void
means that the function won't "return" a result to whatever called it, since the output we'll make goes via a different route (constructors are special cases that don't need a value type, since they are always void
).
We call this function InputTick
and between round brackets define the "parameters" it requires. In this case, one inputdata_t
value named &inputData
. This is information about input event that is automatically generated by the engine at run-time, including among other things its name, I/O arguments from the map (there won't be any), and the entity from which it originated.
The &
means that instead of pulling down all of the information about absolutely every detail of the input event, we instead record the location of that inputdata_t
instance in the system's memory. This "pointer" allows us to access the instance's data without having to copy it to our current function's location. It's similar to opening a file from a desktop shortcut instead of moving or copying it there.
There is no {}
pair for InputTick()
: they'll come when we write the actual contents of the function later. We need to use a semicolon in their place to mark the end of the command.
Private declarations
Now we are declaring private members. These won't be accessible anywhere outside this class unless we make them so in a public function (which we won't).
private:
int m_nThreshold; // Count at which to fire our output
int m_nCounter; // Internal counter
COutputEvent m_OnThreshold; // Output event when the counter reaches the threshold
The first two of these are "variables". They are buckets in the computer's physical memory that can be filled with a value, used in calculations, passed around, perhaps written to the disk, and emptied. They are "integer" or "int" variables which can be used to store whole numbers like -1, 0, 1, 2, 3, and so on. They can't store words or characters however, nor numbers with decimal points. C++ is very strict about these kinds of things!
m_OnThreshold
is an instance of COutputEvent
, a class that Valve have already written for you in cbase.h
. You'll use it in InputTick
to pass a value through the I/O system when our conditions are met. It's actually a variable like int
, but a custom one that doesn't come with the compiler. The only reason it isn't coloured blue as well is that Visual Studio doesn't recognise it.
A word about the names in this section. m_
means that the objects are members of the current class, while n
means that the value is numeric (i
for 'integer' could be used too). You don't have to follow these naming conventions, but they are very common.
Check back
That's the declaration finished. Check back over what you've written and make sure it's the same as the reference code. If it is, you're ready to move past that };
and into the body of the entity's code.
Linking the class to an entity name
We are now past the class declaration. In Visual Studio, you should be able to collapse that whole section by clicking on the second minus sign in the margin. Now, after the };
, enter this:
LINK_ENTITY_TO_CLASS( my_logical_entity, CMyLogicalEntity );
This is another macro (you can tell from the capitals), this time linking the C++ class CMyLogicalEntity
to the Source engine classname my_logical_entity. Classnames are used by the I/O system, by Hammer, and occasionally by programmers.

Data description table
The data description table is a series of macros that give Source metadata about the members of the class you mention in it. The comments in the code below should explain what each one does: DEFINE_FIELD
, for instance, ensures that the m_nCounter
value is stored in a saved game to prevent it from resetting to zero every time the game is reloaded. DEFINE_KEYFIELD
on the next line does the same job, and also opens the value up for editing in Hammer.
This code has examples of the string values mentioned in the last section. Unlike the other commands, these are simply data that the compiler accepts, stores, and blindly exposes to the engine. You can put anything you like between the quote marks and the code will compile, but when it comes to actually instancing the entity Source will be confused unless everything is correct.
// Start of our data description for the class
BEGIN_DATADESC( CMyLogicalEntity )
// For save/load
DEFINE_FIELD( m_nCounter, FIELD_INTEGER ),
// As above, and also links our member variable to a Hammer keyvalue
DEFINE_KEYFIELD( m_nThreshold, FIELD_INTEGER, "threshold" ),
// Links our input name from Hammer to our input member function
DEFINE_INPUTFUNC( FIELD_VOID, "Tick", InputTick ),
// Links our output member variable to the output name used by Hammer
DEFINE_OUTPUT( m_OnThreshold, "OnThreshold" ),
END_DATADESC()
You'll have noticed that while our input has a function, our output is merely the COutputEvent
variable we declared earlier. While an input requires processing when it arrives, and thus a function, an output will have been processed already by the time it leaves.
When writing your own datadesc tables, remember to include a comma after each DEFINE_*
command.

Creating the input function
Now we are going to "define" our input function, which will run when the entity receives the tick
input through the I/O system. Notice the addition of CMyLogicalEntity::
to the function name, which explicitly links the function definition to our class.
//-----------------------------------------------------------------------------
// Purpose: Handle a tick input from another entity
//-----------------------------------------------------------------------------
void CMyLogicalEntity::InputTick( inputdata_t &inputData )
{
// Increment our counter
m_nCounter++;
// See if we've met or crossed our threshold value
if ( m_nCounter >= m_nThreshold )
{
// Fire an output event
m_OnThreshold.FireOutput( inputData.pActivator, this );
// Reset our counter
m_nCounter = 0;
}
}
Here's what's going on:
m_nCounter++
- C++ shorthand for "increment by one". In full, it would be
m_nCounter = m_nCounter + 1;
. Another way of writing it ism_nCounter += 1;
. if ( m_nCounter >= m_nThreshold )
if
statements need a true/false equation to evaluate.>
means greater than, so combined with=
we have an "operator" that says "greater than or equal to". If the value of m_nCounter is greater than or equal to the value of m_nThreshold, execute whatever is inside this command.{ }
- This is a "nested" or "child" bracket pair, within the functions own "parent" bracket pair. It groups all of the commands the
if
statement will execute if it evaluates totrue
. There can be as many levels of nesting as is needed, but be sure to keep track of them all! m_OnThreshold.FireOutput
- You probably pasted this in. Delete it and type it out in full, and note what happens when you reach the dot. You are calling a public function of the
m_OnThreshold
instance of theCOutputEvent
class, and Visual Studio helps you by providing a list of the possible options. Keep typing to filter it, or use up/down and return to select an item. - If we were to call the function we're currently writing from another class, we would type
MyLogicalEntity1.InputTick(<arguments>)
, whereMyLogicalEntity1
is the name of the instance we were targeting. (Which means that we would need to have a pointer to an instance of the class - we can't just access the template!) ( inputData.pActivator, this )
- These are the arguments we are passing to the
COutputEvent
. They will be automatically generated by Source, and arrive at our function through the parameters we gave it between its()
s. - We're sending:
- The entity that started this whole chain off (in Source lingo the "activator").
- The entity that the call is being sent from, the "caller", which is "this" one.
- These are needed for targetname keywords.
m_nCounter = 0
- If we didn't reset our counter, the number would only keep on going up.
Congratulations, the entity is ready to be compiled - press F7 to start the process. If you receive any errors from Visual Studio, check the reference code!
We can't use this entity directly from the engine however - if you've ever made a map, you'll know there's a lot of I/O information that we haven't made room for in this code. Thankfully our DATADESC table allows Source to plug this entity into the wider I/O system automatically; so the only thing left to do now is tell Hammer about it.
Create the FGD entry
An FGD is a text file that defines both what entities your mod has available and what can be done with them. This isn't a perfect world and Hammer won't read from the mod code itself, so it's important to keep the FGD up to date.
If you don't already have an FGD, save an empty text file under your mod's folder as <modname>.fgd
. Then load Hammer and visit Tools > Options
. Making sure your mod is the active configuration, click the "Add" button and navigate to your FGD.
FGDs are a different topic, so this won't be covered in detail. Just paste the following code into your new file:
@include "base.fgd"
@PointClass base(Targetname) = my_logical_entity : "Tutorial logical entity."
[
threshold(integer) : "Threshold" : 1 : "Threshold value."
input Tick(void) : "Adds one tick to the entity's count."
output OnThreshold(void) : "Threshold was hit."
]
And there you have it! Your entity is now ready for use. See Hammer Entity Tool for adding it to a map (place it in one of Valve's sample maps if you're struggling to create your own one), and Inputs and Outputs for getting it doing things.
Using your newly created entity in Hammer
To use this entity in your mod, you will have to place it into a map using Hammer (unless you use/insert it programmatically elsewhere). This tutorial assumes you already have a map to work with and basic knowledge of Hammer; a very basic map made up of an empty room with an "info_player_start" in the middle works.
In this map, we'll place three items on the floor, along with a grenade. Each time the player grabs one of the items, our logical entity's counter will be incremented by 1. When it reaches a threshold of 3 (which we will set from Hammer), the grenade explodes. All we have to do is insert the objects/entities and define their relationships to each other. Simple enough, right?
Start by adding your entity from Hammer; select the Entity Tool (Shift+E), and on the right hand side, select "Entities" under "Categories" and "my_logical_entity" under "Objects".
Click any point in the map to choose the location of the entity to be created. Since this is a logical entity, it will be immaterial and invisible to the player, so its location is not that important; just put it where it won't bother you later.
Once the location has been pinpointed, right-click the selection and choose "Create Object" -- your logical entity will be added to the world.
What you'll immediately want to do is name this logical entity; right-click it in any viewport and choose "Properties" (Remember to pick the Selection Tool (Shift+S) first).
This brings up the Properties menu. There, you can already see two fields:
Name - the name of that particular instance of "my_logical_entity" in the map
Threshold - property that refers to the member variable m_nThreshold from our C++ class, which we called "threshold" in the data description table and exported as "Threshold" in the .fgd file for Hammer to use. This is the count at which the entity will send out an "OnThreshold" signal/event.
So, let's call this entity "MyFirstCounter". And let's set the "Threshold" to 3. (Click the variable name in the list on the left, enter the desired value in the field on the right).
Once everything is set, apply and close the Properties dialog.
Next, we'll place our items on the map.
First, add the grenade. Select the Entity Tool, choose the Object "npc_grenade" and create that new entity somewhere in your map. Go into its properties and give it the name "MyBoobyTrap".
Now, add one item the user can pick up. For instance, "item_ammo_pistol." Go into its Properties dialog, and give it the name "MyPistolAmmo". Then go to the second tab in the Properties dialog, Outputs, and add a new Output. Select the appropriate items in the dropdown boxes: OnPlayerPickup is obviously the event we want to use as a trigger, the target is "MyFirstCounter," which will be incremented thanks to the "Tick" input function (to be selected in the "Via this Input dropdown". This will tell the game to order the counter to run its "Tick" function, which increments the counter by 1, when the player picks up the pistol ammo clip.
Once everything is set up, apply and close the Properties dialog. Copy the pistol ammo and paste two or more extra copies. All their properties and inputs/outputs will be copied over, and they will share the same name.
Now, go back to your logical entity's Properties, and modify its Outputs so that reaching the counter's Threshold value causes the grenade to "Ignite" (a predefined function of Valve's npc_grenade entity). Your output function will be "OnThreshold", the target will be "MyBoobyTrap" and the input you want is "Ignite". You can also set a time delay, if you'd like. This will cause the Valve-defined Ignite function of the grenade to run as soon as the counter hits its threshold (OnThreshold event generated).
Everything is now in place. Select File > Run Map or press F9 (make sure you've compiled the project first), and compile the map. In the map itself, you will be able to pick up the pistol ammo and watch your grenade explode as soon as you grab the third one (shoot some bullets if your gun is too full to pick up ammo).