phantasmal_dgd_v1/
phantasmal_dgd_v1/bin/
phantasmal_dgd_v1/doc/
phantasmal_dgd_v1/mud/doc/
phantasmal_dgd_v1/mud/doc/api/
phantasmal_dgd_v1/mud/doc/kernel/
phantasmal_dgd_v1/mud/doc/kernel/hook/
phantasmal_dgd_v1/mud/doc/kernel/lfun/
phantasmal_dgd_v1/mud/include/
phantasmal_dgd_v1/mud/include/kernel/
phantasmal_dgd_v1/mud/kernel/lib/
phantasmal_dgd_v1/mud/kernel/lib/api/
phantasmal_dgd_v1/mud/kernel/obj/
phantasmal_dgd_v1/mud/kernel/sys/
phantasmal_dgd_v1/mud/tmp/
phantasmal_dgd_v1/mud/usr/System/
phantasmal_dgd_v1/mud/usr/System/keys/
phantasmal_dgd_v1/mud/usr/System/obj/
phantasmal_dgd_v1/mud/usr/System/open/lib/
phantasmal_dgd_v1/mud/usr/common/data/
phantasmal_dgd_v1/mud/usr/common/lib/parsed/
phantasmal_dgd_v1/mud/usr/common/obj/telopt/
phantasmal_dgd_v1/mud/usr/common/obj/ustate/
phantasmal_dgd_v1/mud/usr/game/
phantasmal_dgd_v1/mud/usr/game/include/
phantasmal_dgd_v1/mud/usr/game/obj/
phantasmal_dgd_v1/mud/usr/game/object/
phantasmal_dgd_v1/mud/usr/game/object/stuff/
phantasmal_dgd_v1/mud/usr/game/sys/
phantasmal_dgd_v1/mud/usr/game/text/
phantasmal_dgd_v1/mud/usr/game/users/
phantasmal_dgd_v1/src/host/
phantasmal_dgd_v1/src/host/beos/
phantasmal_dgd_v1/src/host/mac/
phantasmal_dgd_v1/src/host/unix/
phantasmal_dgd_v1/src/host/win32/res/
phantasmal_dgd_v1/src/kfun/
phantasmal_dgd_v1/src/lpc/
phantasmal_dgd_v1/src/parser/
#include <kernel/kernel.h>

#include <phantasmal/lpc_names.h>
#include <phantasmal/tagd.h>

#include <type.h>

static mapping mobile_tags;
static mapping object_tags;

static void create(void) {
  mobile_tags = ([ ]);
  object_tags = ([ ]);
}

void upgraded(varargs int clone) {
  if(!COMMON() && !SYSTEM())
    return;
}

private void check_value_type(int type, mixed value) {
  if(typeof(value) == type)
    return;

  if(typeof(value) == T_NIL && (type != T_INT && type != T_FLOAT)) {
    return;
  }
  error("Type mismatch:  type " + typeof(value)
	+ " won't fit into a tag of type " + type + "!");
}

void new_mobile_tag(string name, int type,
		    varargs string get_function, string set_function) {
  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only game code can create new mobile tag types!");

  if(mobile_tags[name])
    error("Mobile tag type '" + name + "' already exists!");

  if(type <= T_NIL || type > T_MAPPING)
	error(type + " is not a valid type for a tag!");

  mobile_tags[name] = ({ type, get_function, set_function });
}

void new_object_tag(string name, int type,
		    varargs string get_function, string set_function,
		    int inherit_type) {
  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only game code can create new object tag types!");

  if(object_tags[name])
    error("Object tag type '" + name + "' already exists!");

  if(type <= T_NIL || type > T_MAPPING)
	error(type + " is not a valid type for a tag!");

  if(inherit_type < TAG_INHERIT_NONE || inherit_type > TAG_INHERIT_MAX)
    error("Don't recognize TAG_INHERIT number " + inherit_type + " as a valid type!");

  if(inherit_type == TAG_INHERIT_MERGE && type == T_OBJECT) {
	error("Can't automatically merge two or more objects.  Change object type or inherit type!");
  }

  object_tags[name] = ({ type, get_function, set_function, inherit_type });
}

mixed mobile_get_tag_value(object mobile, string name) {
  mixed *tag_arr;

  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only Game code can get mobile tag values!");

  if(function_object("set_number", mobile) != MOBILE)
    error("Can only call mobile_get_tag_value on mobiles!");

  tag_arr = mobile_tags[name];
  if(!tag_arr)
    error("No such mobile tag type as '" + name + "'!");

  return call_other(mobile, (tag_arr[1] ? tag_arr[1] : "get_tag"), name);
}

void mobile_set_tag_value(object mobile, string name, mixed value) {
  mixed *tag_arr;

  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only Game code can set mobile tag values!");

  if(function_object("set_number", mobile) != MOBILE)
    error("Can only call mobile_set_tag_value on mobiles!");

  tag_arr = mobile_tags[name];
  if(!tag_arr)
    error("No such mobile tag type as '" + name + "'!");

  check_value_type(tag_arr[0], value);

  call_other(mobile, (tag_arr[2] ? tag_arr[2] : "set_tag"), name, value);
}

mixed object_get_tag_value(object obj, string name) {
  mixed  *tag_arr;
  mixed   tag_val, tmp;
  object *parents;
  int     ctr;

  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only Game code can get object tag values!");

  if(function_object("set_number", obj) != OBJECT)
    error("Can only call object_get_tag_value on objects!");

  tag_arr = object_tags[name];
  if(!tag_arr)
    error("No such object tag type as '" + name + "'!");

  tag_val = call_other(obj, (tag_arr[1] ? tag_arr[1] : "get_tag"), name);
  if(tag_val) return tag_val;
  
  parents = obj->get_archetypes();
  if(!parents || !sizeof(parents)) {
	  return nil;
  }

  switch(tag_arr[3]) {
	case TAG_INHERIT_NONE:
	  return nil;
	case TAG_INHERIT_FIRST:
	  return object_get_tag_value(parents[0], name);
	case TAG_INHERIT_MERGE:
	  tag_val = (tag_arr[0] == T_INT ? 0
	             : (tag_arr[3] == T_FLOAT ? 0.0
			: (tag_arr[3] == T_STRING ? ""
			   : nil)));
	  for(ctr = 0; ctr < sizeof(parents); ctr++) {
		switch(tag_arr[0]) {
		  case T_INT:
		  case T_FLOAT:
		  case T_STRING:
		    tag_val += tmp;
		    break;
		  case T_ARRAY:
		  case T_MAPPING:
		  	tag_val = tag_val | tmp;
		  	break;
		}
	  }
  }
  return tag_val;
}

void object_set_tag_value(object obj, string name, mixed value) {
  mixed *tag_arr;

  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only Game code can set object tag values!");

  if(function_object("set_number", obj) != OBJECT)
    error("Can only call object_set_tag_value on objects!");

  tag_arr = object_tags[name];
  if(!tag_arr)
    error("No such object tag type as '" + name + "'!");

  check_value_type(tag_arr[0], value);

  call_other(obj, (tag_arr[2] ? tag_arr[2] : "set_tag"), name, value);
}

mixed get_tag_value(object tagged_object, string name) {
  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only game code may call get_tag_value!");

  switch(function_object("set_number", tagged_object)) {
  case OBJECT:
    return object_get_tag_value(tagged_object, name);
  case MOBILE:
    return mobile_get_tag_value(tagged_object, name);
  default:
    error("Don't recognized tagged object type in get_tag_value!");
  }
}

void set_tag_value(object tagged_object, string name, mixed value) {
  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only game code may call set_tag_value!");

  switch(function_object("set_number", tagged_object)) {
  case OBJECT:
    object_set_tag_value(tagged_object, name, value);
    return;
  case MOBILE:
    mobile_set_tag_value(tagged_object, name, value);
    return;
  default:
    error("Don't recognize tagged object type in set_tag_value!");
  }
}

string* mobile_tag_names(void) {
  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only game code may call mobile_tag_names!");

  return map_indices(mobile_tags);
}

string* object_tag_names(void) { 
  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only game code may call object_tag_names!");

  return map_indices(object_tags);
}

mixed* mobile_all_tags(object mobile) {
  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only game code may call mobile_all_tags!");

  return mobile->get_all_tags();
}

mixed* object_all_tags(object obj) {
  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only game code may call object_all_tags!");

  return obj->get_all_tags();
}

int mobile_tag_type(string tag_name) {
  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only game code may call mobile_tag_type!");

  if(mobile_tags[tag_name])
    return mobile_tags[tag_name][0];

  return -1;
}

int object_tag_type(string tag_name) {
  if(!GAME() && !COMMON() && !SYSTEM())
    error("Only game code may call object_tag_type!");

  if(object_tags[tag_name])
    return object_tags[tag_name][0];

  return -1;
}