2015-07-06 19:53:09 +02:00
|
|
|
|
/*
|
2015-01-30 22:10:29 +01:00
|
|
|
|
Copyright (c) 1998-2015, Enno Rehling <enno@eressea.de>
|
2015-01-30 20:37:14 +01:00
|
|
|
|
Katja Zedel <katze@felidae.kn-bremen.de
|
|
|
|
|
Christian Schlittchen <corwin@amber.kn-bremen.de>
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
|
purpose with or without fee is hereby granted, provided that the above
|
|
|
|
|
copyright notice and this permission notice appear in all copies.
|
|
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
|
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
|
|
#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
|
|
|
|
}
|
|
|
|
|
|
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) {
|
2016-01-28 16:00:36 +01:00
|
|
|
|
log_warning("trying to create message of unknown type \"%s\"\n", name);
|
2015-10-15 08:29:20 +02:00
|
|
|
|
if (!mt_find("missing_feedback")) {
|
2016-11-13 16:09:58 +01:00
|
|
|
|
mt_register(mt_new_va("missing_feedback", "unit:unit",
|
2018-01-14 09:38:26 +01:00
|
|
|
|
"region:region", "command:order", "name:string", NULL));
|
2015-10-15 08:29:20 +02:00
|
|
|
|
}
|
2016-11-13 16:09:58 +01:00
|
|
|
|
return msg_message("missing_feedback", "name unit region command",
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
2018-01-14 11:25:44 +01:00
|
|
|
|
static int missing_message_mode;
|
|
|
|
|
|
|
|
|
|
void message_handle_missing(int mode) {
|
|
|
|
|
missing_message_mode = mode;
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-13 22:46:02 +01:00
|
|
|
|
static message *missing_message(const char *name) {
|
2018-01-14 11:25:44 +01:00
|
|
|
|
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")) {
|
2018-02-25 17:18:30 +01:00
|
|
|
|
mt_register(mt_new_va("missing_message", "name:string", MT_NEW_END));
|
2018-01-14 11:25:44 +01:00
|
|
|
|
}
|
|
|
|
|
return msg_message("missing_message", "name", name);
|
2018-01-13 22:46:02 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
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));
|
|
|
|
|
if (*pm == NULL) {
|
|
|
|
|
*pm = malloc(sizeof(message_list));
|
|
|
|
|
(*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;
|
|
|
|
|
}
|
|
|
|
|
}
|