Potions of (in)competence now modify attributes by 5, not 1.

Add a squeaking hinge trap
Rename doobtraps() to triggerattachedtrap() for readbility.
Make using spanners able to set off traps.....
Make spanners able to unlock anything (chests, doors)
Allow (I)nteraction with adjacent objects, not just impassable ones (but limit what kind of objects this works with).
Remove CMD_CLOSE command ('s') - we can just use 'interact'.
Limit (I)nteractions to cells in front of you.
This commit is contained in:
Rob Pearce 2016-06-01 23:59:39 +10:00
parent 2ed1f07968
commit 89dbff4a63
10 changed files with 283 additions and 91 deletions

View File

@ -2008,7 +2008,7 @@ int attackob(lifeform_t *lf, object_t *o, object_t *wep, flag_t *damflag) {
object_t *oo; object_t *oo;
f = hasflag(o->flags, F_TRAPPED); f = hasflag(o->flags, F_TRAPPED);
if (f && pctchance(75)) { if (f && pctchance(75)) {
doobtraps(o, lf); triggerattachedtraps(o, lf);
} else { } else {
// if a trap didn't go off, you might break the lock // if a trap didn't go off, you might break the lock
f = hasflag(o->flags, F_LOCKED); f = hasflag(o->flags, F_LOCKED);

15
data.c
View File

@ -183,16 +183,16 @@ void initcommands(void) {
//addcommand(CMD_DROP, 'd', "Drop an item."); //addcommand(CMD_DROP, 'd', "Drop an item.");
addcommand(CMD_DROPMULTI, 'd', "Drop one or more items."); addcommand(CMD_DROPMULTI, 'd', "Drop one or more items.");
addcommand(CMD_EAT, 'e', "Eat something."); addcommand(CMD_EAT, 'e', "Eat something.");
addcommand(CMD_INTERACT, 'I', "Interact with adjacent objects."); addcommand(CMD_INTERACT, 'I', "Interact with a nearby object.");
addcommand(CMD_MAGIC, 'm', "Use magic or abilities."); addcommand(CMD_MAGIC, 'm', "Use magic or abilities.");
addcommand(CMD_MEMMAGIC, 'M', "Memorise a hotkey for magic or abilities."); addcommand(CMD_MEMMAGIC, 'M', "Memorise a hotkey for magic or abilities.");
addcommand(CMD_OFFER, 'O', "Offer a sacrifice to the gods."); addcommand(CMD_OFFER, 'O', "Offer a sacrifice to the gods.");
addcommand(CMD_OPERATE, 'o', "Operate a tool/wand/device, or fill a flask."); addcommand(CMD_OPERATE, 'o', "Operate a held tool/wand/device, or fill a flask.");
addcommand(CMD_POUR, 'P', "Pour a potion onto something."); addcommand(CMD_POUR, 'P', "Pour a potion onto something.");
addcommand(CMD_QUAFF, 'q', "Quaff (drink) a potion."); addcommand(CMD_QUAFF, 'q', "Quaff (drink) a potion.");
addcommand(CMD_READ, 'r', "Read a scroll/book."); addcommand(CMD_READ, 'r', "Read a scroll/book.");
addcommand(CMD_RESTFULL, 'R', "Rest until healed, or train your skills."); addcommand(CMD_RESTFULL, 'R', "Rest until healed, or train your skills.");
addcommand(CMD_CLOSE, 's', "Shut a door."); //addcommand(CMD_CLOSE, 's', "Shut a door.");
addcommand(CMD_THROW, 't', "Throw an object."); addcommand(CMD_THROW, 't', "Throw an object.");
addcommand(CMD_TAKEOFF, 'T', "Take off an item of clothing/jewelery."); addcommand(CMD_TAKEOFF, 'T', "Take off an item of clothing/jewelery.");
addcommand(CMD_WEILD, 'w', "Weild a weapon."); addcommand(CMD_WEILD, 'w', "Weild a weapon.");
@ -2923,6 +2923,14 @@ void initobjects(void) {
addflag(lastot->flags, F_SECRET, 70, NA, NA, NULL); addflag(lastot->flags, F_SECRET, 70, NA, NA, NULL);
addflag(lastot->flags, F_NOFEEL, B_TRUE, NA, NA, NULL); addflag(lastot->flags, F_NOFEEL, B_TRUE, NA, NA, NULL);
addot(OT_TRAPNOISE, "noise trap", "A hinge which squeaks loudly, attracting attention.", MT_NOTHING, 0, OC_TRAP, SZ_SMALL);
addflag(lastot->flags, F_OBJECTTRAP, B_TRUE, NA, NA, NULL);
addflag(lastot->flags, F_TRAP, 50, B_FALSE, NA, NULL);
addflag(lastot->flags, F_GLYPH, C_GREY, '^', NA, NULL);
addflag(lastot->flags, F_NOPICKUP, B_TRUE, NA, NA, NULL);
addflag(lastot->flags, F_NOBLESS, B_TRUE, NA, NA, NULL);
addot(OT_TRAPGAS, "gas trap", "A trap which releases poisonous gas.", MT_NOTHING, 0, OC_TRAP, SZ_SMALL); addot(OT_TRAPGAS, "gas trap", "A trap which releases poisonous gas.", MT_NOTHING, 0, OC_TRAP, SZ_SMALL);
addflag(lastot->flags, F_TRAP, 70, B_TRUE, NA, NULL); addflag(lastot->flags, F_TRAP, 70, B_TRUE, NA, NULL);
addflag(lastot->flags, F_RARITY, H_DUNGEON, 69, RR_UNCOMMON, NULL); addflag(lastot->flags, F_RARITY, H_DUNGEON, 69, RR_UNCOMMON, NULL);
@ -6495,6 +6503,7 @@ void initobjects(void) {
addflag(lastot->flags, F_ACCURACY, 75, NA, NA, NULL); addflag(lastot->flags, F_ACCURACY, 75, NA, NA, NULL);
addflag(lastot->flags, F_OPERABLE, B_TRUE, NA, NA, NULL); addflag(lastot->flags, F_OPERABLE, B_TRUE, NA, NA, NULL);
addflag(lastot->flags, F_OPERNEEDDIR, B_TRUE, NA, NA, "Use your spanner in which direction"); addflag(lastot->flags, F_OPERNEEDDIR, B_TRUE, NA, NA, "Use your spanner in which direction");
addflag(lastot->flags, F_OPERNEEDSKILL, SK_TECHUSAGE, PR_NOVICE, NA, NULL);
addflag(lastot->flags, F_HELPSDISARM, 5, NA, NA, NULL); addflag(lastot->flags, F_HELPSDISARM, 5, NA, NA, NULL);
addflag(lastot->flags, F_HELPSREPAIR, MT_METAL, 2, 15, NULL); addflag(lastot->flags, F_HELPSREPAIR, MT_METAL, 2, 15, NULL);
addflag(lastot->flags, F_USESSKILL, SK_CLUBS, NA, NA, NULL); addflag(lastot->flags, F_USESSKILL, SK_CLUBS, NA, NA, NULL);

12
defs.h
View File

@ -484,6 +484,7 @@ enum CELLCONDITION {
CC_EDIBLE, CC_EDIBLE,
CC_KNOWN, CC_KNOWN,
CC_IDENTIFIED, CC_IDENTIFIED,
CC_INTERACTABLE,
CC_MINAR, // arg CC_MINAR, // arg
CC_MAXAR, // arg CC_MAXAR, // arg
CC_MINDR, // arg CC_MINDR, // arg
@ -1747,6 +1748,7 @@ enum OBTYPE {
OT_TRAPLIGHTNING, OT_TRAPLIGHTNING,
OT_TRAPMINE, OT_TRAPMINE,
OT_TRAPNEEDLEP, OT_TRAPNEEDLEP,
OT_TRAPNOISE,
OT_TRAPPIT, OT_TRAPPIT,
OT_TRAPROCK, OT_TRAPROCK,
OT_TRAPSUMMON, OT_TRAPSUMMON,
@ -3122,6 +3124,11 @@ enum FLAG {
// v0 = B_IFNOTBLESSED (only targets an ob if unblessed) // v0 = B_IFNOTBLESSED (only targets an ob if unblessed)
F_OPERONOFF, // operating this will just turn it on/off F_OPERONOFF, // operating this will just turn it on/off
F_OPERUSECHARGE, // operating this will use 1 charge F_OPERUSECHARGE, // operating this will use 1 charge
F_OPERNEEDSKILL, // you need a certain skill to operate this.
// eg. you know what a spanner is without any tech skill, but you
// need skill to actually use it.
// v0 = skill that you need
// v1 = skill level you need it at
F_OPERNEEDTARGET, // need to ask for a target of type val0 when opering F_OPERNEEDTARGET, // need to ask for a target of type val0 when opering
// v1 is bitmask of: // v1 is bitmask of:
// TR_NEEDLOS, TR_NEEDLOF, TR_NONE // TR_NEEDLOS, TR_NEEDLOF, TR_NONE
@ -3219,7 +3226,7 @@ enum FLAG {
// check. // check.
// v2 = sc_dodge difficulty // v2 = sc_dodge difficulty
// if text == "ground", then this trap only goes off // if text == "ground", then this trap only goes off
// if you're on the ground (ie not flying) // if you're on the ground (ie not flying) - ie. a pressure plate
F_OBJECTTRAP, // this trap can go onto an object (door, chest, etc) F_OBJECTTRAP, // this trap can go onto an object (door, chest, etc)
F_DOORTRAPONLY, // this trap can only go on to doors F_DOORTRAPONLY, // this trap can only go on to doors
F_DOORFALLOB, // if this object is trapped with OT_TRAPDOORFALL, F_DOORFALLOB, // if this object is trapped with OT_TRAPDOORFALL,
@ -4598,6 +4605,7 @@ enum ERROR {
E_NOABIL, E_NOABIL,
E_NOMP, E_NOMP,
E_NOSTAM, E_NOSTAM,
E_NOSKILL,
E_NOSPELLS, E_NOSPELLS,
E_AVOIDOB, E_AVOIDOB,
E_FROZEN, E_FROZEN,
@ -4678,7 +4686,7 @@ enum COMMAND {
// //
CMD_AGAIN, CMD_AGAIN,
CMD_AIM, CMD_AIM,
CMD_CLOSE, //CMD_CLOSE,
CMD_COMMS, CMD_COMMS,
CMD_COMMSALL, CMD_COMMSALL,
CMD_DOWN, CMD_DOWN,

46
io.c
View File

@ -4495,6 +4495,7 @@ void doattackcell(int dir) {
} }
} }
/*
void doclose(void) { void doclose(void) {
int failed = B_TRUE; // default is to fail int failed = B_TRUE; // default is to fail
int dir; int dir;
@ -4545,6 +4546,7 @@ void doclose(void) {
} }
} }
} }
*/
void docommslf(lifeform_t *lf, char ch, lifeform_t *lf2, cell_t *targc) { void docommslf(lifeform_t *lf, char ch, lifeform_t *lf2, cell_t *targc) {
char lfname[BUFLEN], lfname2[BUFLEN]; char lfname[BUFLEN], lfname2[BUFLEN];
@ -7025,6 +7027,18 @@ char *makedesc_ob(object_t *o, char *retbuf) {
strncat(retbuf, buf, HUGEBUFLEN); strncat(retbuf, buf, HUGEBUFLEN);
} }
f = hasflag(o->flags, F_OPERABLE);
if (f) {
flag_t *f2;
f2 = hasflag(o->flags, F_OPERNEEDSKILL);
if (f2) {
sprintf(buf, "It requires %s %s skill to operate.\n", getskilllevelname(f2->val[1]), getskillname(f2->val[0]));
} else {
sprintf(buf, "It can be operated.\n");
}
strncat(retbuf, buf, HUGEBUFLEN);
}
// other weapon properties... // other weapon properties...
getflags(o->flags, retflag, &nretflags, F_CANBLOCK, F_NONE); getflags(o->flags, retflag, &nretflags, F_CANBLOCK, F_NONE);
for (i = 0; i < nretflags; i++) { for (i = 0; i < nretflags; i++) {
@ -9109,6 +9123,8 @@ void dohelp(char helpmode) {
restoregamewindows(); restoregamewindows();
} }
void dointeract(void) { void dointeract(void) {
cell_t *c; cell_t *c;
object_t *o; object_t *o;
@ -9121,7 +9137,7 @@ void dointeract(void) {
// operable objects here? // operable objects here?
for (o = player->cell->obpile->first; o ; o = o->next) { for (o = player->cell->obpile->first; o ; o = o->next) {
if (isoperable(o)) { if (isinteractable(o) && !isdoor(o, NULL)) { // can't open/close doors you're on top of
char obname[BUFLEN],buf[BUFLEN]; char obname[BUFLEN],buf[BUFLEN];
char verb[BUFLEN]; char verb[BUFLEN];
int ch; int ch;
@ -9146,6 +9162,14 @@ void dointeract(void) {
msg("Cancelled."); msg("Cancelled.");
return; return;
} }
// can only interact with things we're facing
if (getrelativedir(player, dir) != RD_FORWARDS) {
msg("You can only interact with things in front of you.");
return;
}
c = getcellindir(player->cell, dir); c = getcellindir(player->cell, dir);
if (!c) { if (!c) {
msg("Cancelled."); msg("Cancelled.");
@ -9157,21 +9181,14 @@ void dointeract(void) {
// get adjacent cells // get adjacent cells
// anything operable and impasssble? // anything operable and impasssble?
/*
initcondv(&cs, CC_HASFLAG, B_TRUE, F_IMPASSABLE, initcondv(&cs, CC_HASFLAG, B_TRUE, F_IMPASSABLE,
CC_OPERABLE, B_TRUE, NA, CC_OPERABLE, B_TRUE, NA,
CC_NONE); CC_NONE);
getobs(c->obpile, &cs, ob, &nobs); */
for (n = 0; n < nobs; n++) { initcondv(&cs, CC_INTERACTABLE, B_TRUE, NA,
char obname[BUFLEN];
getobname(ob[n], obname, ob[n]->amt);
addchoice(&prompt, ch, obname, NULL, ob[n], NULL);
if (ch == 'z') ch = 'A';
else ch++;
}
// open doors
initcondv(&cs, CC_HASFLAG, B_TRUE, F_DOOR,
CC_NONE); CC_NONE);
getobs(c->obpile, &cs, ob, &nobs); getobs(c->obpile, &cs, ob, &nobs);
for (n = 0; n < nobs; n++) { for (n = 0; n < nobs; n++) {
char obname[BUFLEN]; char obname[BUFLEN];
@ -9180,7 +9197,6 @@ void dointeract(void) {
if (ch == 'z') ch = 'A'; if (ch == 'z') ch = 'A';
else ch++; else ch++;
} }
o = NULL; o = NULL;
if (!prompt.nchoices) { if (!prompt.nchoices) {
@ -11123,17 +11139,19 @@ void handleinput(void) {
case CMD_NEXTTARGET: case CMD_NEXTTARGET:
donextguntarget(); donextguntarget();
break; break;
// object functions /*
case CMD_CLOSE: // close case CMD_CLOSE: // close
addflag(player->flags, F_LASTCMD, NA, NA, NA, temp); addflag(player->flags, F_LASTCMD, NA, NA, NA, temp);
doclose(); doclose();
break; break;
*/
case CMD_COMMS: // communicate case CMD_COMMS: // communicate
docomms(NULL); docomms(NULL);
break; break;
case CMD_COMMSALL: // communicate case CMD_COMMSALL: // communicate
docommsmulti(); docommsmulti();
break; break;
// object functions
case CMD_EAT: // eat case CMD_EAT: // eat
doeat(player->pack); doeat(player->pack);
break; break;

10
lf.c
View File

@ -1179,6 +1179,8 @@ int canopendoors(lifeform_t *lf) {
} }
int canoperate(lifeform_t *lf, object_t *o, enum ERROR *why) { int canoperate(lifeform_t *lf, object_t *o, enum ERROR *why) {
flag_t *f;
if (why) *why = E_OK; if (why) *why = E_OK;
if (lfhasflag(lf, F_RAGE)) { if (lfhasflag(lf, F_RAGE)) {
if (why) *why = E_RAGE; if (why) *why = E_RAGE;
@ -1194,6 +1196,14 @@ int canoperate(lifeform_t *lf, object_t *o, enum ERROR *why) {
return B_FALSE; return B_FALSE;
} }
} }
f = hasflag(o->flags, F_OPERNEEDSKILL);
if (f) {
if (getskill(lf, f->val[0]) < f->val[1]) {
if (why) *why = E_NOSKILL;
return B_FALSE;
}
}
return B_TRUE; return B_TRUE;
} }

4
move.c
View File

@ -1968,6 +1968,7 @@ int moveto(lifeform_t *lf, cell_t *newcell, int onpurpose, int dontclearmsg) {
for (o = newcell->obpile->first ; o ; o = nexto ) { for (o = newcell->obpile->first ; o ; o = nexto ) {
flag_t *f; flag_t *f;
nexto = o->next; nexto = o->next;
f = hasflag(o->flags, F_TRAP); f = hasflag(o->flags, F_TRAP);
if (f) { if (f) {
if (strstr(f->text, "ground") && isairborne(lf, NULL)) { if (strstr(f->text, "ground") && isairborne(lf, NULL)) {
@ -2174,7 +2175,7 @@ int opendoor(lifeform_t *lf, object_t *o) {
// trapped? // trapped?
if (lf && hasflag(o->flags, F_TRAPPED)) { if (lf && hasflag(o->flags, F_TRAPPED)) {
if (doobtraps(o, lf)) { if (triggerattachedtraps(o, lf)) {
if (isplayer(lf)) stoppathfinding(lf); if (isplayer(lf)) stoppathfinding(lf);
return B_TRUE; return B_TRUE;
} }
@ -2918,6 +2919,7 @@ int teleportto(lifeform_t *lf, cell_t *c, int wantsmoke) {
return B_FALSE; return B_FALSE;
} }
// trigger an actual trap object (ie. OC_TRAP)
void triggertrap(lifeform_t *lf, object_t *o, object_t *trapob, cell_t *where) { void triggertrap(lifeform_t *lf, object_t *o, object_t *trapob, cell_t *where) {
char triggerer[BUFLEN]; char triggerer[BUFLEN];
char trapname[BUFLEN]; char trapname[BUFLEN];

13
nexus.c
View File

@ -1684,6 +1684,19 @@ void initcondv(condset_t *cs, ...) {
va_end(args); va_end(args);
} }
char incletter(char *ch) {
int newchar;
if (*ch == '\0') {
newchar = 'a';
} else if (*ch == 'z') {
newchar = 'A';
} else {
newchar = (*ch) + 1;
}
*ch = newchar;
return *ch;
}
void inctime(long nunits) { void inctime(long nunits) {
curtime += (nunits*(TIMECONST)); curtime += (nunits*(TIMECONST));

View File

@ -24,6 +24,7 @@ char getpctletter(float num, float max);
int init(void); int init(void);
void initcond(condset_t *cs); void initcond(condset_t *cs);
void initcondv(condset_t *cs, ...); void initcondv(condset_t *cs, ...);
char incletter(char *ch);
void inctime(long nunits); void inctime(long nunits);
void calcbresnham(map_t *m, int x1, int y1, int x2, int y2, cell_t **retcell, int *numpixels); void calcbresnham(map_t *m, int x1, int y1, int x2, int y2, cell_t **retcell, int *numpixels);
void initbresnham(int x1, int y1, int x2, int y2, int *xinc1, int *yinc1, int *dinc1, int *xinc2, int *yinc2, int *dinc2, int *numpixels, int *d); void initbresnham(int x1, int y1, int x2, int y2, int *xinc1, int *yinc1, int *dinc1, int *xinc2, int *yinc2, int *dinc2, int *numpixels, int *d);

265
objects.c
View File

@ -3491,6 +3491,23 @@ void colourmatchob(object_t *o, lifeform_t *lf) {
} }
} }
// returns TRUE if user chose to cancel the action
int confirmknowntraps(object_t *o) {
if (hasflagval(o->flags, F_TRAPPED, NA, NA, B_TRUE, NULL)) {
if (getattrbracket(getattr(player, A_WIS), A_WIS, NULL) >= AT_AVERAGE) {
char ch;
char buf[BUFLEN],obname[BUFLEN];
getobname(o, obname, o->amt);
snprintf(buf, BUFLEN,"Really operate %s?", obname);
ch = askchar(buf,"yn","n", B_TRUE, B_FALSE);
if (ch != 'y') {
return B_TRUE;
}
}
}
return B_FALSE;
}
void copyobprops(object_t *dst, object_t *src) { void copyobprops(object_t *dst, object_t *src) {
dst->material = src->material; dst->material = src->material;
dst->weight = src->weight; dst->weight = src->weight;
@ -3790,7 +3807,8 @@ int doobdieconvert(object_t *o, int wantannounce) {
} }
int doobtraps(object_t *o, lifeform_t *lf) { // trigger any traps attached to a regular object (ie. a door with a trap)
int triggerattachedtraps(object_t *o, lifeform_t *lf) {
flag_t *f; flag_t *f;
f = hasflag(o->flags, F_TRAPPED); f = hasflag(o->flags, F_TRAPPED);
if (f) { if (f) {
@ -4768,7 +4786,7 @@ lifeform_t *getobcreatedby(object_t *o) {
} }
int getobjamdiff(int depth) { int getobjamdiff(int depth) {
return 100 + rnd(1,100); return 50 + rnd(0,depth*5);
} }
// returns the amount to adjust a skillcheck roll if you give object 'o' to lifeform 'lf' // returns the amount to adjust a skillcheck roll if you give object 'o' to lifeform 'lf'
@ -7627,6 +7645,18 @@ object_t *hasequippedobidon(obpile_t *op, enum OBTYPE oid, enum BODYPART bp) {
return NULL; return NULL;
} }
int hasinteractableflags(flagpile_t *fp) {
if (hasflag(fp, F_IMPASSABLE)) {
return B_TRUE;
} else if (hasflag(fp, F_DOOR)) {
return B_TRUE;
} else if (hasflag(fp, F_CONTAINER)) {
return B_TRUE;
}
return B_FALSE;
}
object_t *hasknownob(obpile_t *op, enum OBTYPE oid) { object_t *hasknownob(obpile_t *op, enum OBTYPE oid) {
object_t *o; object_t *o;
for (o = op->first ; o ; o = o->next) { for (o = op->first ; o ; o = o->next) {
@ -8203,6 +8233,21 @@ int isimpassableob(object_t *o, lifeform_t *lf, enum LFSIZE forcesize) {
return B_FALSE; return B_FALSE;
} }
// 'Interacting' is using things without holding them.
//
// It's fine to interact with a computer, door or treasure chest.
// It's NOT fine to interact with a wand, even though it is operable.
//
int isinteractable(object_t *o) {
if (isoperable(o) && hasinteractableflags(o->flags)) {
return B_TRUE;
} else if (isdoor(o, NULL)) {
return B_TRUE;
}
return B_FALSE;
}
int ismagicalobtype(objecttype_t *ot) { int ismagicalobtype(objecttype_t *ot) {
switch (ot->obclass->id) { switch (ot->obclass->id) {
case OC_SCROLL: case OC_SCROLL:
@ -10103,6 +10148,9 @@ int obmeetscondition(object_t *o, enum CELLCONDITION cond, int arg, int value) {
case CC_OPERABLE: case CC_OPERABLE:
if (isoperable(o)) check = B_TRUE; if (isoperable(o)) check = B_TRUE;
break; break;
case CC_INTERACTABLE:
if (isinteractable(o)) check = B_TRUE;
break;
case CC_POURABLE: case CC_POURABLE:
if (ispourable(o)) check = B_TRUE; if (ispourable(o)) check = B_TRUE;
break; break;
@ -10268,6 +10316,9 @@ int otmeetscondition(objecttype_t *ot, enum CELLCONDITION cond, int arg, int val
case CC_OPERABLE: case CC_OPERABLE:
if (hasflag(ot->flags, F_OPERABLE)) check = B_TRUE; if (hasflag(ot->flags, F_OPERABLE)) check = B_TRUE;
break; break;
case CC_INTERACTABLE:
if (hasinteractableflags(ot->flags)) check = B_TRUE;
break;
case CC_POURABLE: case CC_POURABLE:
if (hasflag(ot->flags, F_POURABLE)) check = B_TRUE; if (hasflag(ot->flags, F_POURABLE)) check = B_TRUE;
break; break;
@ -10524,20 +10575,21 @@ int operate(lifeform_t *lf, object_t *o, cell_t *where) {
} }
} }
getobname(o, obname, 1);
if (!canoperate(lf, o, &why)) { if (!canoperate(lf, o, &why)) {
if (isplayer(lf)) { if (isplayer(lf)) {
switch (why) { switch (why) {
case E_NOHANDS: msg("You lack the manual dexterity to operate this."); break; case E_NOHANDS: msg("You lack the manual dexterity to operate %s.", obname); break;
case E_RAGE: msg("You are too enraged to operate anything!"); break; case E_RAGE: msg("You are too enraged to operate anything!"); break;
case E_STUNNED: msg("You cannot operate anything while stunned."); break; case E_STUNNED: msg("You cannot operate anything while stunned."); break;
default: msg("For some reason, you can't operate this."); break; case E_NOSKILL: msg("You lack the skill to operate %s.", obname); break;
default: msg("For some reason, you can't operate %s.", obname); break;
} }
return B_TRUE; return B_TRUE;
} }
} }
getobname(o, obname, 1);
if ((isplayer(lf)) || cansee(player, lf)) { if ((isplayer(lf)) || cansee(player, lf)) {
playercansee = B_TRUE; playercansee = B_TRUE;
} else { } else {
@ -10575,16 +10627,9 @@ int operate(lifeform_t *lf, object_t *o, cell_t *where) {
// has known trap? // has known trap?
if (isplayer(lf)) { if (isplayer(lf)) {
if (hasflagval(o->flags, F_TRAPPED, NA, NA, B_TRUE, NULL)) { if (confirmknowntraps(o)) {
if (getattrbracket(getattr(lf, A_WIS), A_WIS, NULL) >= AT_AVERAGE) { msg("Cancelled.");
char ch; return B_TRUE;
snprintf(buf, BUFLEN,"Really operate %s?", obname);
ch = askchar(buf,"yn","n", B_TRUE, B_FALSE);
if (ch != 'y') {
msg("Cancelled.");
return B_TRUE;
}
}
} }
} }
@ -10680,13 +10725,17 @@ int operate(lifeform_t *lf, object_t *o, cell_t *where) {
char ch; char ch;
int dir; int dir;
// ask direction // ask direction
ch = askchar(f->text, "yuhjklbn-","-", B_FALSE, B_TRUE); ch = askchar(f->text, "yuhjklbn.-","-", B_FALSE, B_TRUE);
if ((ch == '-') || !ch) { if ((ch == '-') || !ch) {
msg("Cancelled."); msg("Cancelled.");
return B_TRUE; return B_TRUE;
} else { } else {
dir = chartodir(ch); dir = chartodir(ch);
where = getcellindir(lf->cell, dir); if (dir == D_NONE) {
where = lf->cell;
} else {
where = getcellindir(lf->cell, dir);
}
} }
} }
@ -10792,7 +10841,7 @@ int operate(lifeform_t *lf, object_t *o, cell_t *where) {
if (isplayer(lf)) maketried(o->type->id, NULL); if (isplayer(lf)) maketried(o->type->id, NULL);
// trapped? // trapped?
if (doobtraps(o, lf)) { if (triggerattachedtraps(o, lf)) {
return B_TRUE; return B_TRUE;
} }
@ -11913,59 +11962,134 @@ int operate(lifeform_t *lf, object_t *o, cell_t *where) {
int donesomething = B_FALSE; int donesomething = B_FALSE;
if (!where) { if (!where) {
if (isplayer(lf)) msg("There is nothing to use your spanner on there!"); if (isplayer(lf)) msg("There is nothing to use your spanner on there!");
} else if (where->lf) { } else if (where->lf && (where->lf != lf)) {
if (isplayer(lf)) msg("There is someone in your way!"); if (isplayer(lf)) msg("There is someone in your way!");
} else { } else {
object_t *o; int isopen;
char ch; object_t *o2;
char ch = '\0',qbuf[BUFLEN],spannername[BUFLEN];
flag_t *f; flag_t *f;
for (o = where->obpile->first ; o ; o = o->next) {
int isopen; getobname(o, spannername, 1);
// jammed doors
if (isdoor(o, &isopen)) { snprintf(qbuf, BUFLEN, "What will you use %s on?", spannername);
if (!isopen) { // ie. if closed. initprompt(&prompt, qbuf);
f = hasflag(o->flags, F_JAMMED); for (o2 = where->obpile->first ; o2 ; o2 = o2->next) {
if (f) { // can use the spanner on...
ch = askchar("The hinges seem jammed. Loosen them", "yn", "y", B_TRUE, B_FALSE); if (isdoor(o2, &isopen)) { // doors
if (ch == 'y') { getobname(o2, obname, 1);
char obname[BUFLEN]; addchoice(&prompt, incletter(&ch), obname, NULL, o2 , NULL);
getobname(o, obname, 1); } else if (hasflag(o2->flags, F_LOCKABLE)) {
msg("You loosen the hinges on %s.", obname); getobname(o2, obname, 1);
killflag(f); addchoice(&prompt, incletter(&ch), obname, NULL, o2 , NULL);
taketime(lf, getactspeed(lf));
donesomething = B_TRUE;
}
} else {
char obname[BUFLEN];
getobname(o, obname, 1);
snprintf(buf, BUFLEN, "Tighten the hinges on %s",obname);
ch = askchar(buf, "yn", "y", B_TRUE, B_FALSE);
if (ch == 'y') {
msg("You tighten the hinges on %s.", obname);
addflag(o->flags, F_JAMMED, 2, B_TRUE, 150, NULL);
taketime(lf, getactspeed(lf));
donesomething = B_TRUE;
}
}
}
}
// gratings
if ((o->type->id == OT_GRATINGFLOOR) && hasflag(o->flags, F_LOCKED)) {
char obname[BUFLEN];
getobname(o, obname, 1);
ch = askchar("It looks like you could remove the bolts. Do so", "yn", "y", B_TRUE, B_FALSE);
if (ch == 'y') {
msg("You remove the cover from %s.", obname);
killflagsofid(o->flags, F_LOCKED);
killflagsofid(o->flags, F_LOCKABLE);
taketime(lf, getactspeed(lf));
donesomething = B_TRUE;
}
} }
} }
}
if (!donesomething) { if (prompt.nchoices >= 1) {
if (isplayer(lf)) msg("There is nothing to use your spanner on there!"); int isopen;
if (prompt.nchoices == 1) {
o2 = (object_t *)prompt.choice[0].data;
} else {
prompt.maycancel = B_TRUE;
getchoice(&prompt);
o2 = (object_t *) prompt.result;
}
if (o2) {
if (isplayer(lf)) {
if (confirmknowntraps(o2)) {
msg("Cancelled.");
return B_TRUE;
}
}
snprintf(qbuf, BUFLEN, "What will you do to %s?", obname);
initprompt(&prompt, qbuf);
// figure out what you can do to the object
if (isdoor(o2, &isopen) && !isopen) {
getobname(o2, obname, 1);
f = hasflagknown(o2->flags, F_JAMMED);
if (f) {
// you know it's jammed...
snprintf(qbuf, BUFLEN, "Unjam it");
addchoice(&prompt, 'j', qbuf, NULL, o2 , NULL);
} else {
// you don't _think_ it's jammed...
snprintf(qbuf, BUFLEN, "Jam it");
addchoice(&prompt, 'j', qbuf, NULL, o2 , NULL);
}
}
f = hasflag(o2->flags, F_LOCKABLE);
if (f) {
snprintf(qbuf, BUFLEN, "Dismantle the lock");
addchoice(&prompt, 'l', qbuf, NULL, o2 , NULL);
}
if (prompt.nchoices >= 1) {
prompt.maycancel = B_TRUE;
ch = getchoice(&prompt);
if (ch != '\0') {
// traps go off
if (triggerattachedtraps(o2, lf)) {
return B_TRUE;
}
}
switch (ch) {
case 'j':
f = hasflagknown(o2->flags, F_JAMMED);
if (f) {
// unjam it
msg("You loosen the hinges on %s.", obname);
killflag(f);
taketime(lf, getactspeed(lf));
donesomething = B_TRUE;
} else {
// you don't _think_ it's jammed...
// try to jam it.
if (hasflag(o2->flags, F_JAMMED)) {
msg("The hinges on %s cannot be tightened any more.", obname);
f->known = B_TRUE;
taketime(lf, getactspeed(lf));
donesomething = B_TRUE;
} else {
msg("You tighten the hinges on %s.", obname);
addflag(o2->flags, F_JAMMED, 2, B_TRUE, 120, NULL); // very tight
taketime(lf, getactspeed(lf));
donesomething = B_TRUE;
}
}
break;
case 'l':
f = hasflagknown(o2->flags, F_LOCKED);
if (f) {
msg("You remove the locking bolts from %s.", obname);
killflagsofid(o2->flags, F_LOCKED);
killflagsofid(o2->flags, F_LOCKABLE);
taketime(lf, getactspeed(lf));
donesomething = B_TRUE;
} else {
// should never happen.
msg("%s is already unlocked.",obname);
}
break;
default:
msg("Cancelled.");
return B_TRUE;
}
} else {
msg("There doesn't seem to be much you can do to %s.",obname);
}
} else {
msg("Cancelled.");
return B_TRUE;
}
} else {
msg("There is nothing to use your spanner on there!");
}
} }
} else if (o->type->id == OT_STYPTIC) { } else if (o->type->id == OT_STYPTIC) {
flag_t *retflag[MAXCANDIDATES]; flag_t *retflag[MAXCANDIDATES];
@ -12690,9 +12814,9 @@ void potioneffects(lifeform_t *lf, enum OBTYPE oid, object_t *o, enum BLESSTYPE
break; break;
} }
if (potblessed == B_CURSED) { if (potblessed == B_CURSED) {
amt = -1; amt = -5;
} else { } else {
amt = 1; amt = 5;
} }
// select a random attribute // select a random attribute
if (potblessed == B_BLESSED) { if (potblessed == B_BLESSED) {
@ -16949,7 +17073,10 @@ void trapeffects(object_t *trapob, enum OBTYPE oid, cell_t *c, object_t *trapped
} }
} }
} else if (oid == OT_TRAPALARM) { } else if (oid == OT_TRAPALARM) {
// 50 is stupidly loud, intentionally
noise(c, NULL, NC_OTHER, 50, "a blaring siren!", "A blaring siren goes off!"); noise(c, NULL, NC_OTHER, 50, "a blaring siren!", "A blaring siren goes off!");
} else if (oid == OT_TRAPNOISE) {
noise(c, NULL, NC_OTHER, 3, "a loud squeak!", "The hinges squeak loudly!");
} else if ((oid == OT_TRAPARROW) || (oid == OT_TRAPARROWP)) { } else if ((oid == OT_TRAPARROW) || (oid == OT_TRAPARROWP)) {
int dir,bestdir = D_NONE; int dir,bestdir = D_NONE;
cell_t *src = NULL; cell_t *src = NULL;

View File

@ -2,6 +2,7 @@
#define __OBJECTS_H #define __OBJECTS_H
#include "defs.h" #include "defs.h"
brand_t *addbrand(enum BRAND id, char *suffix, enum BODYPART bp, enum BLESSTYPE blessed, int blesschance); brand_t *addbrand(enum BRAND id, char *suffix, enum BODYPART bp, enum BLESSTYPE blessed, int blesschance);
object_t *addemptyob(obpile_t *where, object_t *o); object_t *addemptyob(obpile_t *where, object_t *o);
hiddenname_t *addhiddenname(enum OBCLASS obclass, char *text); hiddenname_t *addhiddenname(enum OBCLASS obclass, char *text);
@ -40,6 +41,7 @@ int chargesknown(object_t *o);
int getcritprotection(object_t *o); int getcritprotection(object_t *o);
int checkobnames(char *haystack, char *needle); int checkobnames(char *haystack, char *needle);
void colourmatchob(object_t *o, lifeform_t *lf); void colourmatchob(object_t *o, lifeform_t *lf);
int confirmknowntraps(object_t *o);
void copyobprops(object_t *dst, object_t *src); void copyobprops(object_t *dst, object_t *src);
int counthiddennames(enum OBCLASS ocid, char *text); int counthiddennames(enum OBCLASS ocid, char *text);
int countmoney(obpile_t *op); int countmoney(obpile_t *op);
@ -54,7 +56,7 @@ int curseob(object_t *o);
void damageallobs(object_t *srcob, obpile_t *op, int howmuch, int damtype, lifeform_t *srclf); void damageallobs(object_t *srcob, obpile_t *op, int howmuch, int damtype, lifeform_t *srclf);
void dodecay(object_t *o); void dodecay(object_t *o);
int doobdieconvert(object_t *o, int wantannounce); int doobdieconvert(object_t *o, int wantannounce);
int doobtraps(object_t *o, lifeform_t *lf); int triggerattachedtraps(object_t *o, lifeform_t *lf);
void dumpobs(int ntogen); void dumpobs(int ntogen);
void dumpobrarity(void); void dumpobrarity(void);
void explodeob(object_t *o, flag_t *f, int bigness, lifeform_t *causedby); void explodeob(object_t *o, flag_t *f, int bigness, lifeform_t *causedby);
@ -172,6 +174,7 @@ enum BODYPART getweildloc(object_t *o, lifeform_t *lf, enum BODYPART *otherloc,
int hasedibleob(obpile_t *op); int hasedibleob(obpile_t *op);
object_t *hasequippedobid(obpile_t *op, enum OBTYPE oid); object_t *hasequippedobid(obpile_t *op, enum OBTYPE oid);
object_t *hasequippedobidon(obpile_t *op, enum OBTYPE oid, enum BODYPART bp); object_t *hasequippedobidon(obpile_t *op, enum OBTYPE oid, enum BODYPART bp);
int hasinteractableflags(flagpile_t *fp);
object_t *hasknownob(obpile_t *op, enum OBTYPE oid); object_t *hasknownob(obpile_t *op, enum OBTYPE oid);
object_t *hasob(obpile_t *op, enum OBTYPE oid); object_t *hasob(obpile_t *op, enum OBTYPE oid);
object_t *hasobletter(obpile_t *op, char letter); object_t *hasobletter(obpile_t *op, char letter);
@ -213,6 +216,7 @@ int isknownot(objecttype_t *ot);
int isheavyweapon(object_t *o); int isheavyweapon(object_t *o);
int isidentified(object_t *o); int isidentified(object_t *o);
int isimpassableob(object_t *o, lifeform_t *lf, enum LFSIZE forcesize); int isimpassableob(object_t *o, lifeform_t *lf, enum LFSIZE forcesize);
int isinteractable(object_t *o);
int ismagicalobtype(objecttype_t *ot); int ismagicalobtype(objecttype_t *ot);
int ismagical(object_t *o); int ismagical(object_t *o);
int ismeleeweapon(object_t *o); int ismeleeweapon(object_t *o);