merentha_fluffos_v2/
merentha_fluffos_v2/bin/
merentha_fluffos_v2/fluffos-2.9-ds2.03/
merentha_fluffos_v2/fluffos-2.9-ds2.03/ChangeLog.old/
merentha_fluffos_v2/fluffos-2.9-ds2.03/Win32/
merentha_fluffos_v2/fluffos-2.9-ds2.03/compat/
merentha_fluffos_v2/fluffos-2.9-ds2.03/compat/simuls/
merentha_fluffos_v2/fluffos-2.9-ds2.03/include/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/clone/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/command/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/data/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/etc/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/include/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/inherit/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/inherit/master/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/log/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/single/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/single/tests/compiler/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/single/tests/efuns/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/single/tests/operators/
merentha_fluffos_v2/fluffos-2.9-ds2.03/testsuite/u/
merentha_fluffos_v2/fluffos-2.9-ds2.03/tmp/
merentha_fluffos_v2/fluffos-2.9-ds2.03/windows/
merentha_fluffos_v2/lib/cfg/
merentha_fluffos_v2/lib/cfg/races/
merentha_fluffos_v2/lib/cmds/abilities/
merentha_fluffos_v2/lib/cmds/actions/
merentha_fluffos_v2/lib/cmds/spells/
merentha_fluffos_v2/lib/daemon/include/
merentha_fluffos_v2/lib/daemon/services/
merentha_fluffos_v2/lib/doc/
merentha_fluffos_v2/lib/doc/building/
merentha_fluffos_v2/lib/doc/help/classes/
merentha_fluffos_v2/lib/doc/help/general/
merentha_fluffos_v2/lib/doc/help/races/
merentha_fluffos_v2/lib/doc/help/skills/
merentha_fluffos_v2/lib/doc/help/stats/
merentha_fluffos_v2/lib/doc/man/efuns/
merentha_fluffos_v2/lib/doc/man/lfuns/
merentha_fluffos_v2/lib/doc/news/
merentha_fluffos_v2/lib/doc/old/
merentha_fluffos_v2/lib/doc/old/concepts/
merentha_fluffos_v2/lib/doc/old/lpc/constructs/
merentha_fluffos_v2/lib/doc/old/lpc/types/
merentha_fluffos_v2/lib/domains/ROOMS/
merentha_fluffos_v2/lib/domains/obj/armour/
merentha_fluffos_v2/lib/domains/obj/monsters/
merentha_fluffos_v2/lib/domains/obj/other/
merentha_fluffos_v2/lib/domains/obj/weapons/
merentha_fluffos_v2/lib/realms/petrarch/
merentha_fluffos_v2/lib/save/daemons/
merentha_fluffos_v2/lib/save/rid/
merentha_fluffos_v2/lib/save/users/a/
merentha_fluffos_v2/lib/save/users/p/
merentha_fluffos_v2/lib/save/users/t/
merentha_fluffos_v2/lib/std/login/
merentha_fluffos_v2/lib/std/obj/
merentha_fluffos_v2/win32/
message

message is a new efun in MudOS designed to make communication efuns more 
generic and to provide a standard way of talking to clients intelligently.

See the message() efun doc for details on how the efun works.

Probably the most important element of this function is the "class".
If used properly, you could use this field to implement a very simple
version of earmuffs, or to communicate intelligently with a custom
client.  The class defines the type of message that the string
contains.  Initial simple implementations would have the classes
"shout", "say", "write", "tell_object" (which would be generated by
simul_efuns of the same name that replace the more traditional efuns).

Given this, let's say that you wanted to implment a quick and easy
earmuff ability. (the ability to mask shouts) In your user (player)
object, you would have the function receive_message.  Here's the
simplest implementation possible:

<pre>
void receive_message (string msg, mixed class)
{
	receive(msg);
}
</pre>

This simply takes all messages generated by the message efun and
displays them to the user.  However, you could imagine a simple
earmuffs implementation on top of this:

<pre>
string *muffle = ({});

int muffle_class (string arg)
{
	muffle += ({arg});
}

void receive_message (string msg, string class)
{
	if (member_array(class,string) == -1)
		receive(msg);
}
</pre>

Now you can see, that if a particular class is muffled (say, "shout"
for example), the text never gets displayed, but in other cases it
does.

However, not all uses of the shout() efun are really shouts, in the
traditional mud sense.  For example, let's say that the admin of the
mud wants to send a message to all users telling them that the system
is going to be shut down in 5 minutes.  To do this, they might use
echo, which in turn uses the shout() efun.  So all users who had
"shout" muffled would miss this important message. This means that a
broader number of classes is really needed to make message() truly
useful.  For the example given, let's say we make a new "broadcast"
class.  This message class would be used for important announcements
that everyone should hear.  Perhaps a restriction could even be made
so that muffle prohibited blocking this class.

Let's look at another example.  What if you're tired of all of the
millions of emotes (soul commands) that clutter your screen?  Wouldn't
it be nice to just muffle those?  Well, obviously a new "emote" class
is needed that all soul functions use.  Now you might be thinking to
yourself "hey... I don't want to have to use this really complex
message() efun every time I write a soul command.  write() and say()
are very simple, and I like using those."  Well, I couldn't agree with
you more.  To combat that problem, I make a simul_efun for each type
of commonly used message class. Like emote for example.  I made a new
simul_efun called emote() that in fact made writing soul commands
easier, and used message() with the "emote" class.  I won't show you
the code for the emote simul_efun, but here's the basic idea:

<pre>
varargs int emote (object emoter, string self_message, string
	other_message, mixed emotee, string target_message, string modifier);

emoter - the object doing the emoting
self_message - the message displayed to the emoter
other_message - the message displayed to the whole room
emotee - the target of the emote (i.e. kick huthar)
target_message - the message displayed to the emotee
string modifier - any extra modifier to tack on to the end of the
	emote string. (i.e. adverbs: smiles happily, cheerfully, etc.) - only
	really complex soul commands need this (if they want to be able to
	control multiple modifiers to a single soul command)
</pre>

At this point, some might be thinking, "ok... so you can do very
powerful selective muffling, big deal.  this seems like a lot of work
for nothing."  Good point.  Muffling was just a simple neat thing you
could do with message now.  Most of the real advantages from message
will come a bit down the line when someone gets around to writing a
smart client program.  Here's how that will work.

Basically the idea is to separate all of the messages sent to a user
by the content.  So you have a "combat" class, and a "stat" class, and
a "room_description" class, and a "help" class as some examples.
Before I get started, let's write a new version of receive_message().

<pre>
int has_smart_client;

void receive_message (string msg, string class)
{
	if (member_array(class,muffle) == -1) {
		if (has_smart_client) 
			receive (class + " : " + msg);
		else
			receive (msg);
	}
}
</pre>

Ok. Let's look at what this does.  If the user object has defined
has_smart_client to be > 0, then it prepends all messages with the
class name as well.  So if you were to write a smart client that
parsed messages like that, you could make it redirect room
descriptions into one window, conversation into another, combat into
yet another area, etc.  You could make a status line that always kept
your current room name (since it got passed as class "room_name" when
you enetered the room).  You could make the heart_beat, pass a class
"status" message which gives a constant readout of your hit points in
your status line.  All of this would be transparent to the end user.
It would just work.

In addition, you could do a simply graphical client using the same
technique.  The BSX graphical mud / client could easily be implemented
on top of MudOS using message().  Or you could pass around small
bitmaps rather than the polygon-based line drawings of BSX.  The
possibilities are pretty wide open.

There is at least one major flaw with this argument so far.  Since
everyone has to implement this message protocol themselves, and since
nobody has written a smart client to take advantage of the protocol,
then when a client comes out, what's to guarantee that your mudlib
will even work with it?  Well, that's actually most of the point to
this document.  I'd like to outline a simple protocol that I hope
everyone will adopt, so that when a client finally comes out, it will
work with all mudlibs that adhere to this protocol.

(Note: It's now several years later, and AFAIK noone uses this protocol.
AMCP also exists and is supported by some mudlibs, but the only client is
very minimal.  -Beek)

The protocol:

all messages sent to the smart client are in the form

"class:msg_length:msg"

msg_len is the length of the msg string.  This is put in so that the
client can always know when it has received the entire message, and
when one starts and ends.

The following list of classes should be used and a client should be
able to parse and use any messages using these classes.

<pre>
say			use of the "say" command or its equivalent
shout			use of the "shout" command or its equivalent
tell			use of the "tell" command or its equivalent
emote			a soul command or emote
broadcast		a broadcast message to everyone on the mud
combat			generic combat messages
combat_self		combat messages generated by the user's own attack
combat_other		combat messages generated by others
combat_*		all other specific combat messages
room_description	a long description of a room or location
room_name		a short name for a room or location
inventory		what you're carrying
item_description	a long description of the item
status			generic status messages
status_hp		current hit points
status_sp		current spell points
status_sobriety		current state of drunkeness
status_*		all other specific status messages
score			generic score messages
score_exp		experience points
score_money		the amount of coins or other money
developer		a broadcast message to all wizards/developers
class_fighter		a message to all fighters
class_mage		a message to all mages
class_thief		a message to all thieves
class_priest		a message to all priests
class_*			a message to the class specified
race_human		a message to all humans
race_elf		a message to all elves
race_dwarf		a message to all dwarves
race_*			a message to the race specified

*** optional classes to implement ***
bitmap			a generic bitmap message
bitmap_*		a specific type of bitmap
drawing			a generic drawing message
drawing_*		a specific type of drawing
</pre>