2010-08-08 10:06:34 +02:00
|
|
|
#include <platform.h>
|
2014-06-09 18:54:48 +02:00
|
|
|
#include "messages.h"
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
/* kernel includes */
|
|
|
|
#include "building.h"
|
|
|
|
#include "faction.h"
|
|
|
|
#include "item.h"
|
|
|
|
#include "order.h"
|
|
|
|
#include "plane.h"
|
|
|
|
#include "region.h"
|
|
|
|
#include "unit.h"
|
|
|
|
|
|
|
|
/* util includes */
|
|
|
|
#include <util/base36.h>
|
|
|
|
#include <util/goodies.h>
|
|
|
|
#include <util/language.h>
|
2017-12-29 06:13:28 +01:00
|
|
|
#include <util/macros.h>
|
2010-08-08 10:06:34 +02:00
|
|
|
#include <util/message.h>
|
|
|
|
#include <util/nrmessage.h>
|
|
|
|
#include <util/crmessage.h>
|
|
|
|
#include <util/log.h>
|
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
#define MAXSTRLEN (4*DISPLAYSIZE+3)
|
|
|
|
|
|
|
|
static void
|
2011-03-07 08:02:35 +01:00
|
|
|
arg_set(variant args[], const message_type * mtype, const char *buffer,
|
2015-01-30 20:37:14 +01:00
|
|
|
variant v)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
int i;
|
|
|
|
for (i = 0; i != mtype->nparameters; ++i) {
|
|
|
|
if (!strcmp(buffer, mtype->pnames[i]))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (i != mtype->nparameters) {
|
|
|
|
args[i] = v;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
log_error("invalid parameter %s for message type %s\n", buffer,
|
|
|
|
mtype->name);
|
|
|
|
assert(!"program aborted.");
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2018-10-28 21:28:05 +01:00
|
|
|
static int missing_message_mode;
|
|
|
|
|
|
|
|
void message_handle_missing(int mode) {
|
|
|
|
missing_message_mode = mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
static message *missing_feedback(const char *name, const struct unit *u,
|
|
|
|
const struct region *r, struct order *ord)
|
|
|
|
{
|
|
|
|
if (missing_message_mode == MESSAGE_MISSING_ERROR) {
|
|
|
|
log_error("trying to create undefined feedback of type \"%s\"\n", name);
|
|
|
|
}
|
|
|
|
else if (missing_message_mode == MESSAGE_MISSING_REPLACE) {
|
2020-01-02 11:34:26 +01:00
|
|
|
log_warning("trying to create undefined message of type \"%s\"\n", name);
|
2018-10-28 21:28:05 +01:00
|
|
|
if (strcmp(name, "missing_feedback") != 0) {
|
|
|
|
if (!mt_find("missing_feedback")) {
|
|
|
|
mt_create_va(mt_new("missing_feedback", NULL), "unit:unit",
|
|
|
|
"region:region", "command:order", "name:string", MT_NEW_END);
|
|
|
|
}
|
|
|
|
return msg_message("missing_feedback", "unit region command name", u, r, ord, name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static message *missing_message(const char *name) {
|
|
|
|
if (missing_message_mode == MESSAGE_MISSING_ERROR) {
|
|
|
|
log_error("trying to create undefined message of type \"%s\"\n", name);
|
|
|
|
}
|
|
|
|
else if (missing_message_mode == MESSAGE_MISSING_REPLACE) {
|
|
|
|
log_warning("trying to create undefined message of type \"%s\"\n", name);
|
|
|
|
if (strcmp(name, "missing_message") != 0) {
|
|
|
|
if (!mt_find("missing_message")) {
|
|
|
|
mt_create_va(mt_new("missing_message", NULL), "name:string", MT_NEW_END);
|
|
|
|
}
|
|
|
|
return msg_message("missing_message", "name", name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
struct message *msg_feedback(const struct unit *u, struct order *ord,
|
2015-01-30 20:37:14 +01:00
|
|
|
const char *name, const char *sig, ...)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
const message_type *mtype = mt_find(name);
|
|
|
|
variant args[16];
|
|
|
|
variant var;
|
|
|
|
memset(args, 0, sizeof(args));
|
|
|
|
|
2017-09-30 19:44:39 +02:00
|
|
|
if (ord && is_silent(ord)) {
|
2017-03-25 11:36:29 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
if (!mtype) {
|
2018-10-28 21:28:05 +01:00
|
|
|
return missing_feedback(name, u, u->region, ord);
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
var.v = (void *)u;
|
|
|
|
arg_set(args, mtype, "unit", var);
|
|
|
|
var.v = (void *)u->region;
|
|
|
|
arg_set(args, mtype, "region", var);
|
|
|
|
var.v = (void *)ord;
|
|
|
|
arg_set(args, mtype, "command", var);
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2017-03-25 11:36:29 +01:00
|
|
|
if (sig) {
|
|
|
|
const char *ic = sig;
|
|
|
|
va_list marker;
|
2015-01-30 20:37:14 +01:00
|
|
|
|
2017-03-25 11:36:29 +01:00
|
|
|
va_start(marker, sig);
|
|
|
|
while (*ic && !isalnum(*ic))
|
|
|
|
ic++;
|
|
|
|
while (*ic) {
|
|
|
|
char paramname[64];
|
|
|
|
char *oc = paramname;
|
|
|
|
int i;
|
2015-01-30 20:37:14 +01:00
|
|
|
|
2017-03-25 11:36:29 +01:00
|
|
|
while (isalnum(*ic))
|
|
|
|
*oc++ = *ic++;
|
|
|
|
*oc = '\0';
|
|
|
|
|
|
|
|
for (i = 0; i != mtype->nparameters; ++i) {
|
|
|
|
if (!strcmp(paramname, mtype->pnames[i]))
|
|
|
|
break;
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
2017-03-25 11:36:29 +01:00
|
|
|
if (i != mtype->nparameters) {
|
|
|
|
if (mtype->types[i]->vtype == VAR_VOIDPTR) {
|
|
|
|
args[i].v = va_arg(marker, void *);
|
|
|
|
}
|
|
|
|
else if (mtype->types[i]->vtype == VAR_INT) {
|
|
|
|
args[i].i = va_arg(marker, int);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
assert(!"unknown variant type");
|
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
|
|
|
else {
|
2017-03-25 11:36:29 +01:00
|
|
|
log_error("invalid parameter %s for message type %s\n", paramname, mtype->name);
|
|
|
|
assert(!"program aborted.");
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
2017-03-25 11:36:29 +01:00
|
|
|
while (*ic && !isalnum(*ic))
|
|
|
|
ic++;
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
2017-03-25 11:36:29 +01:00
|
|
|
va_end(marker);
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
|
|
|
return msg_create(mtype, args);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
message *msg_message(const char *name, const char *sig, ...)
|
2015-01-30 20:37:14 +01:00
|
|
|
/* msg_message("oops_error", "unit region command", u, r, cmd) */
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
va_list vargs;
|
|
|
|
const message_type *mtype = mt_find(name);
|
|
|
|
char paramname[64];
|
|
|
|
const char *ic = sig;
|
2015-01-18 15:44:44 +01:00
|
|
|
int argnum=0;
|
2015-01-30 20:37:14 +01:00
|
|
|
variant args[16];
|
|
|
|
memset(args, 0, sizeof(args));
|
|
|
|
|
|
|
|
if (!mtype) {
|
2018-01-13 22:46:02 +01:00
|
|
|
return missing_message(name);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
va_start(vargs, sig);
|
2011-03-07 08:02:35 +01:00
|
|
|
while (*ic && !isalnum(*ic))
|
2015-01-30 20:37:14 +01:00
|
|
|
ic++;
|
|
|
|
while (*ic) {
|
|
|
|
char *oc = paramname;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
while (isalnum(*ic))
|
|
|
|
*oc++ = *ic++;
|
|
|
|
*oc = '\0';
|
|
|
|
|
|
|
|
for (i = 0; i != mtype->nparameters; ++i) {
|
|
|
|
if (!strcmp(paramname, mtype->pnames[i]))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (i != mtype->nparameters) {
|
|
|
|
if (mtype->types[i]->vtype == VAR_VOIDPTR) {
|
|
|
|
args[i].v = va_arg(vargs, void *);
|
|
|
|
}
|
|
|
|
else if (mtype->types[i]->vtype == VAR_INT) {
|
|
|
|
args[i].i = va_arg(vargs, int);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
assert(!"unknown variant type");
|
|
|
|
}
|
2015-01-18 15:44:44 +01:00
|
|
|
argnum++;
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
log_error("invalid parameter %s for message type %s\n", paramname, mtype->name);
|
|
|
|
}
|
|
|
|
while (*ic && !isalnum(*ic))
|
|
|
|
ic++;
|
|
|
|
}
|
|
|
|
va_end(vargs);
|
2015-01-18 15:44:44 +01:00
|
|
|
if (argnum != mtype->nparameters) {
|
|
|
|
log_error("not enough parameters for message type %s\n", mtype->name);
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2015-01-30 20:37:14 +01:00
|
|
|
return msg_create(mtype, args);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
caddmessage(region * r, faction * f, const char *s, msg_t mtype, int level)
|
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
message *m = NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
|
2017-01-10 16:31:05 +01:00
|
|
|
UNUSED_ARG(level);
|
2015-01-30 20:37:14 +01:00
|
|
|
switch (mtype) {
|
2010-08-08 10:06:34 +02:00
|
|
|
case MSG_BATTLE:
|
2015-01-30 20:37:14 +01:00
|
|
|
assert(0 || !"battle messages must not use addmessage");
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
case MSG_MAGIC:
|
|
|
|
case MSG_MESSAGE:
|
|
|
|
case MSG_EVENT:
|
2015-01-30 20:37:14 +01:00
|
|
|
/* Botschaften an REGION oder einzelne PARTEI */
|
|
|
|
m = msg_message("msg_event", "string", s);
|
|
|
|
if (!r) {
|
|
|
|
assert(f);
|
|
|
|
m = add_message(&f->msgs, m);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (f == NULL)
|
|
|
|
add_message(&r->msgs, m);
|
|
|
|
else
|
|
|
|
r_addmessage(r, f, m);
|
|
|
|
}
|
|
|
|
break;
|
2010-08-08 10:06:34 +02:00
|
|
|
default:
|
2015-01-30 20:37:14 +01:00
|
|
|
assert(!"invalid message class");
|
|
|
|
}
|
|
|
|
if (m)
|
|
|
|
msg_release(m);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
void addmessage(region * r, faction * f, const char *s, msg_t mtype, int level)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-01-30 20:37:14 +01:00
|
|
|
caddmessage(r, f, s, mtype, level);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2014-12-10 21:17:00 +01:00
|
|
|
message * msg_error(const unit * u, struct order *ord, int mno) {
|
|
|
|
static char msgname[20];
|
|
|
|
|
2018-02-25 18:10:09 +01:00
|
|
|
if (u->faction->flags & FFL_NPC) {
|
2017-03-25 10:37:13 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
2014-12-10 21:17:00 +01:00
|
|
|
sprintf(msgname, "error%d", mno);
|
|
|
|
return msg_feedback(u, ord, msgname, "");
|
|
|
|
}
|
|
|
|
|
2017-06-25 08:21:14 +02:00
|
|
|
void cmistake(const unit * u, struct order *ord, int mno, int mtype)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2017-06-25 08:21:14 +02:00
|
|
|
message * msg;
|
2017-01-10 16:31:05 +01:00
|
|
|
UNUSED_ARG(mtype);
|
2017-06-25 08:21:14 +02:00
|
|
|
msg = msg_error(u, ord, mno);
|
|
|
|
if (msg) {
|
|
|
|
ADDMSG(&u->faction->msgs, msg);
|
2017-03-25 10:37:13 +01:00
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2015-05-15 13:32:22 +02:00
|
|
|
void syntax_error(const struct unit *u, struct order *ord)
|
|
|
|
{
|
|
|
|
message * result;
|
|
|
|
result = msg_error(u, ord, 10);
|
2017-03-25 10:37:13 +01:00
|
|
|
if (result) {
|
|
|
|
ADDMSG(&u->faction->msgs, result);
|
|
|
|
}
|
2015-05-15 13:32:22 +02:00
|
|
|
}
|
|
|
|
|
2015-10-14 14:08:50 +02:00
|
|
|
void free_messagelist(mlist *msgs)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-08-05 12:19:17 +02:00
|
|
|
struct mlist **mlistptr;
|
2015-10-14 14:08:50 +02:00
|
|
|
for (mlistptr = &msgs; *mlistptr;) {
|
|
|
|
struct mlist *ml = *mlistptr;
|
|
|
|
*mlistptr = ml->next;
|
|
|
|
msg_release(ml->msg);
|
|
|
|
free(ml);
|
2015-01-30 20:37:14 +01:00
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
message *add_message(message_list ** pm, message * m)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2015-07-06 19:53:09 +02:00
|
|
|
assert(m && m->type);
|
2017-11-25 20:24:57 +01:00
|
|
|
if (m != NULL) {
|
2015-01-30 20:37:14 +01:00
|
|
|
struct mlist *mnew = malloc(sizeof(struct mlist));
|
2018-12-15 19:38:40 +01:00
|
|
|
if (!mnew) abort();
|
2015-01-30 20:37:14 +01:00
|
|
|
if (*pm == NULL) {
|
|
|
|
*pm = malloc(sizeof(message_list));
|
2018-12-15 19:38:40 +01:00
|
|
|
if (*pm == NULL) abort();
|
2015-01-30 20:37:14 +01:00
|
|
|
(*pm)->end = &(*pm)->begin;
|
|
|
|
}
|
|
|
|
mnew->msg = msg_addref(m);
|
|
|
|
mnew->next = NULL;
|
|
|
|
*((*pm)->end) = mnew;
|
|
|
|
(*pm)->end = &mnew->next;
|
2011-03-07 08:02:35 +01:00
|
|
|
}
|
2015-01-30 20:37:14 +01:00
|
|
|
return m;
|
2011-03-07 08:02:35 +01:00
|
|
|
}
|
2015-08-16 11:30:44 +02:00
|
|
|
|
|
|
|
struct mlist ** merge_messages(message_list *mlist, message_list *append) {
|
|
|
|
struct mlist **split = 0;
|
|
|
|
assert(mlist);
|
|
|
|
if (append) {
|
|
|
|
split = mlist->end;
|
|
|
|
*split = append->begin;
|
|
|
|
mlist->end = append->end;
|
|
|
|
}
|
|
|
|
return split;
|
|
|
|
}
|
|
|
|
|
|
|
|
void split_messages(message_list *mlist, struct mlist **split) {
|
|
|
|
assert(mlist);
|
|
|
|
if (split) {
|
|
|
|
*split = 0;
|
|
|
|
mlist->end = split;
|
|
|
|
}
|
|
|
|
}
|