/* vi: set ts=2: * * * Eressea PB(E)M host Copyright (C) 1998-2003 * Christian Schlittchen (corwin@amber.kn-bremen.de) * Katja Zedel (katze@felidae.kn-bremen.de) * Henning Peters (faroul@beyond.kn-bremen.de) * Enno Rehling (enno@eressea-pbem.de) * Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de) * * based on: * * Atlantis v1.0 13 September 1993 Copyright 1993 by Russell Wallace * Atlantis v1.7 Copyright 1996 by Alex Schröder * * This program may not be used, modified or distributed without * prior permission by the authors of Eressea. * This program may not be sold or used commercially without prior written * permission from the authors. */ #define TEACH_ALL 1 #define TEACH_FRIENDS #include #include "eressea.h" #include "study.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include /* util includes */ #include #include /* libc includes */ #include #include #include #include #include #include #define TEACHNUMBER 10 static skill_t getskill(const struct locale * lang) { return findskill(getstrtoken(), lang); } static magic_t findmagicskill(const char *s) { return (char) findstr(magietypen, s, MAXMAGIETYP); } magic_t getmagicskill(void) { return findmagicskill(getstrtoken()); } /* ------------------------------------------------------------- */ /* Vertraute und Kröten sind keine Migranten */ boolean is_migrant(unit *u) { if (u->race == u->faction->race) return false; if (fval(u->race, RCF_UNDEAD|RCF_ILLUSIONARY)) return false; if (is_familiar(u)) return false; if (u->race == new_race[RC_TOAD]) return false; return true; } /* ------------------------------------------------------------- */ boolean magic_lowskill(unit *u) { if (u->race == new_race[RC_TOAD]) return true; return false; } /* ------------------------------------------------------------- */ int study_cost(unit *u, skill_t talent) { int stufe, k = 50; switch (talent) { case SK_SPY: return 100; break; case SK_TACTICS: case SK_HERBALISM: case SK_ALCHEMY: return 200; break; case SK_MAGIC: /* Die Magiekosten betragen 50+Summe(50*Stufe) */ /* 'Stufe' ist dabei die nächste zu erreichende Stufe */ stufe = 1 + get_level(u, SK_MAGIC); return k*(1+((stufe+1)*stufe/2)); break; } return 0; } /* ------------------------------------------------------------- */ static void init_learning(struct attrib * a) { a->data.v = calloc(sizeof(teaching_info), 1); } static void done_learning(struct attrib * a) { free(a->data.v); } const attrib_type at_learning = { "learning", init_learning, done_learning, NULL, NULL, NULL, ATF_UNIQUE }; static int teach_unit(unit * teacher, unit * student, int nteaching, skill_t sk, boolean report, int * academy) { teaching_info * teach = NULL; attrib * a; int n; /* learning sind die Tage, die sie schon durch andere Lehrer zugute * geschrieben bekommen haben. Total darf dies nicht über 30 Tage pro Mann * steigen. * * n ist die Anzahl zusätzlich gelernter Tage. n darf max. die Differenz * von schon gelernten Tagen zum max(30 Tage pro Mann) betragen. */ if (magic_lowskill(student)){ cmistake(teacher, teacher->thisorder, 292, MSG_EVENT); return 0; } n = student->number * 30; a = a_find(student->attribs, &at_learning); if (a!=NULL) { teach = (teaching_info*)a->data.v; n -= teach->value; } n = min(n, nteaching); if (n != 0) { struct building * b = inside_building(teacher); const struct building_type * btype = b?b->type:NULL; int index = 0; if (teach==NULL) { a = a_add(&student->attribs, a_new(&at_learning)); teach = (teaching_info*)a->data.v; } else { while (teach->teachers[index] && index!=MAXTEACHERS) ++index; } if (indexteachers[index++] = teacher; if (indexteachers[index] = NULL; teach->value += n; /* Solange Akademien größenbeschränkt sind, sollte Lehrer und * Student auch in unterschiedlichen Gebäuden stehen dürfen */ if (btype == bt_find("academy") && student->building && student->building->type == bt_find("academy")) { int j = study_cost(student, sk); j = max(50, j * 2); /* kann Einheit das zahlen? */ if (get_pooled(student, oldresourcetype[R_SILVER], GET_DEFAULT, j) >= j) { /* Jeder Schüler zusätzlich +10 Tage wenn in Uni. */ teach->value += (n / 30) * 10; /* learning erhöhen */ /* Lehrer zusätzlich +1 Tag pro Schüler. */ if (academy) *academy += n; } /* sonst nehmen sie nicht am Unterricht teil */ } /* Teaching ist die Anzahl Leute, denen man noch was beibringen kann. Da * hier nicht n verwendet wird, werden die Leute gezählt und nicht die * effektiv gelernten Tage. -> FALSCH ? (ENNO) * * Eine Einheit A von 11 Mann mit Talent 0 profitiert vom ersten Lehrer B * also 10x30=300 tage, und der zweite Lehrer C lehrt für nur noch 1x30=30 * Tage (damit das Maximum von 11x30=330 nicht überschritten wird). * * Damit es aber in der Ausführung nicht auf die Reihenfolge drauf ankommt, * darf der zweite Lehrer C keine weiteren Einheiten D mehr lehren. Also * wird student 30 Tage gutgeschrieben, aber teaching sinkt auf 0 (300-11x30 <= * 0). * * Sonst träte dies auf: * * A: lernt B: lehrt A C: lehrt A D D: lernt * * Wenn B vor C dran ist, lehrt C nur 30 Tage an A (wie oben) und * 270 Tage an D. * * Ist C aber vor B dran, lehrt C 300 tage an A, und 0 tage an D, * und B lehrt auch 0 tage an A. * * Deswegen darf C D nie lehren dürfen. * * -> Das ist wirr. wer hat das entworfen? * Besser wäre, man macht erst vorab alle zuordnungen, und dann * die Talentänderung (enno). */ nteaching = max(0, nteaching - student->number * 30); } return n; } static void teach(unit * u, struct order * ord) { region * r = u->region; int teaching, i, j, count, academy=0; unit *u2; skill_t sk = NOSKILL; if ((u->race->flags & RCF_NOTEACH) || fval(u, UFL_WERE)) { cmistake(u, ord, 274, MSG_EVENT); return; } if (r->planep && fval(r->planep, PFL_NOTEACH)) { cmistake(u, ord, 273, MSG_EVENT); return; } teaching = u->number * 30 * TEACHNUMBER; if ((i = get_effect(u, oldpotiontype[P_FOOL])) > 0) { /* Trank "Dumpfbackenbrot" */ i = min(i, u->number * TEACHNUMBER); /* Trank wirkt pro Schüler, nicht pro Lehrer */ teaching -= i * 30; change_effect(u, oldpotiontype[P_FOOL], -i); j = teaching / 30; ADDMSG(&u->faction->msgs, msg_message("teachdumb", "teacher amount", u, j)); } if (teaching == 0) return; u2 = 0; count = 0; init_tokens(ord); skip_token(); #if TEACH_ALL if (getparam(u->faction->locale)==P_ANY) { unit * student = r->units; skill_t teachskill[MAXSKILLS]; int i = 0; do { sk = getskill(u->faction->locale); teachskill[i++]=sk; } while (sk!=NOSKILL); while (teaching && student) { if (student->faction == u->faction) { #ifdef NEW_DAEMONHUNGER_RULE if (LongHunger(student)) continue; #else if (fval(student, UFL_HUNGER)) continue; #endif if (get_keyword(student->thisorder) == K_STUDY) { /* Input ist nun von student->thisorder !! */ init_tokens(student->thisorder); skip_token(); sk = getskill(student->faction->locale); if (sk!=NOSKILL && teachskill[0]!=NOSKILL) { for (i=0;teachskill[i]!=NOSKILL;++i) if (sk==teachskill[i]) break; sk = teachskill[i]; } if (sk != NOSKILL && eff_skill_study(u, sk, r)-TEACHDIFFERENCE > eff_skill_study(student, sk, r)) { teaching -= teach_unit(u, student, teaching, sk, true, &academy); } } } student = student->next; } #ifdef TEACH_FRIENDS while (teaching && student) { if (student->faction != u->faction && alliedunit(u, student->faction, HELP_GUARD)) { #ifdef NEW_DAEMONHUNGER_RULE if (LongHunger(student)) continue; #else if (fval(student, UFL_HUNGER)) continue; #endif if (get_keyword(student->thisorder) == K_STUDY) { /* Input ist nun von student->thisorder !! */ init_tokens(student->thisorder); skip_token(); sk = getskill(student->faction->locale); if (sk != NOSKILL && eff_skill_study(u, sk, r)-TEACHDIFFERENCE >= eff_skill(student, sk, r)) { teaching -= teach_unit(u, student, teaching, sk, true, &academy); } } } student = student->next; } #endif } else #endif { static char zOrder[BUFSIZE]; order * new_order; init_tokens(ord); skip_token(); strcpy(zOrder, locale_string(u->faction->locale, keywords[K_TEACH])); while (!parser_end()) { unit * u2 = getunit(r, u->faction); ++count; /* Falls die Unit nicht gefunden wird, Fehler melden */ if (!u2) { const char * token; /* Finde den string, der den Fehler verursacht hat */ parser_pushstate(); init_tokens(ord); skip_token(); for (j=0; j!=count-1; ++j) { /* skip over the first 'count' units */ getunit(r, u->faction); } token = getstrtoken(); /* Beginne die Fehlermeldung */ strcpy(buf, "Die Einheit '"); if (findparam(token, u->faction->locale) == P_TEMP) { /* Für: "Die Einheit 'TEMP ZET' wurde nicht gefunden" oder "Die Einheit * 'TEMP' wurde nicht gefunden" */ scat(token); token = getstrtoken(); if (*token) scat(" "); } scat(token); scat("' wurde nicht gefunden"); mistake(u, ord, buf, MSG_EVENT); parser_popstate(); continue; } /* Neuen Befehl zusammenbauen. TEMP-Einheiten werden automatisch in * ihre neuen Nummern übersetzt. */ strcat(zOrder, " "); strcat(zOrder, unitid(u2)); if (get_keyword(u2->thisorder) != K_STUDY) { ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "teach_nolearn", "student", u2)); continue; } /* Input ist nun von u2->thisorder !! */ parser_pushstate(); init_tokens(u2->thisorder); skip_token(); sk = getskill(u2->faction->locale); parser_popstate(); if (sk == NOSKILL) { ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "teach_nolearn", "student", u2)); continue; } /* u is teacher, u2 is student */ if (eff_skill_study(u2, sk, r) > eff_skill_study(u, sk, r)-TEACHDIFFERENCE) { ADDMSG(&u->faction->msgs, msg_feedback(u, ord, "teach_asgood", "student", u2)); continue; } if (sk == SK_MAGIC) { /* ist der Magier schon spezialisiert, so versteht er nur noch * Lehrer seines Gebietes */ if (find_magetype(u2) != 0 && find_magetype(u) != find_magetype(u2)) { sprintf(buf, "%s versteht unsere Art von Magie nicht", unitname(u2)); mistake(u, ord, buf, MSG_EVENT); continue; } } teaching -= teach_unit(u, u2, teaching, sk, false, &academy); } new_order = parse_order(zOrder, u->faction->locale); #ifdef LASTORDER set_order(&u->lastorder, new_order); #else replace_order(&u->orders, ord, new_order); free_order(new_order); /* parse_order & set_order have each increased the refcount */ #endif } if (academy && sk!=NOSKILL) { academy = academy/30; /* anzahl gelehrter wochen, max. 10 */ learn_skill(u, sk, academy/30.0/TEACHNUMBER); } } /* ------------------------------------------------------------- */ int learn_cmd(unit * u, order * ord) { region *r = u->region; int p; magic_t mtyp; int l; int studycost, days; double multi = 1.0; attrib * a = NULL; teaching_info * teach = NULL; int money = 0; skill_t sk; int maxalchemy = 0; static int learn_newskills = -1; if (learn_newskills<0) { const char * str = get_param(global.parameters, "study.newskills"); if (str && strcmp(str, "false")==0) learn_newskills = 0; else learn_newskills = 1; } if (u->number==0) return 0; if (fval(r->terrain, SEA_REGION)) { /* sonderbehandlung aller die auf Ozeanen lernen können */ if (u->race!=new_race[RC_AQUARIAN] && !(u->race->flags & RCF_SWIM)) { return 0; } } if (fval(u, UFL_LONGACTION)) return 0; if (u->race == new_race[RC_INSECT] && r_insectstalled(r) && !is_cursed(u->attribs, C_KAELTESCHUTZ,0)) { return 0; } if (fval(u, UFL_LONGACTION)) { cmistake(u, ord, 52, MSG_PRODUCE); return 0; } if ((u->race->flags & RCF_NOLEARN) || fval(u, UFL_WERE)) { sprintf(buf, "%s können nichts lernen", LOC(default_locale, rc_name(u->race, 1))); mistake(u, ord, buf, MSG_EVENT); return 0; } init_tokens(ord); skip_token(); sk = getskill(u->faction->locale); if (sk < 0) { cmistake(u, ord, 77, MSG_EVENT); return 0; } if (SkillCap(sk) && SkillCap(sk) <= effskill(u, sk)) { cmistake(u, ord, 77, MSG_EVENT); return 0; } /* Hack: Talente mit Malus -99 können nicht gelernt werden */ if (u->race->bonus[sk] == -99) { cmistake(u, ord, 77, MSG_EVENT); return 0; } if (learn_newskills==0) { skill * sv = get_skill(u, sk); if (sv==NULL) { /* we can only learn skills we already have */ cmistake(u, ord, 77, MSG_EVENT); return 0; } } /* snotlings können Talente nur bis T8 lernen */ if (u->race == new_race[RC_SNOTLING]){ if (get_level(u, sk) >= 8){ cmistake(u, ord, 308, MSG_EVENT); return 0; } } p = studycost = study_cost(u, sk); a = a_find(u->attribs, &at_learning); if (a!=NULL) { teach = (teaching_info*)a->data.v; } /* keine kostenpflichtigen Talente für Migranten. Vertraute sind * keine Migranten, wird in is_migrant abgefangen. Vorsicht, * studycost darf hier noch nicht durch Akademie erhöht sein */ if (studycost > 0 && !ExpensiveMigrants() && is_migrant(u)) { sprintf(buf, "Migranten können keine kostenpflichtigen Talente lernen"); mistake(u, ord, buf, MSG_EVENT); return 0; } /* Akademie: */ { struct building * b = inside_building(u); const struct building_type * btype = b?b->type:NULL; if (btype == bt_find("academy")) { studycost = max(50, studycost * 2); } } if (sk == SK_MAGIC) { if (u->number > 1){ cmistake(u, ord, 106, MSG_MAGIC); return 0; } if (is_familiar(u)){ /* Vertraute zählen nicht zu den Magiern einer Partei, * können aber nur Graue Magie lernen */ mtyp = M_GRAU; if (!is_mage(u)) create_mage(u, mtyp); } else if (!has_skill(u, SK_MAGIC)) { /* Die Einheit ist noch kein Magier */ if (count_skill(u->faction, SK_MAGIC) + u->number > max_skill(u->faction, SK_MAGIC)) { sprintf(buf, "Es kann maximal %d Magier pro Partei geben", max_skill(u->faction, SK_MAGIC)); mistake(u, ord, buf, MSG_EVENT); return 0; } mtyp = getmagicskill(); if (mtyp == M_NONE || mtyp == M_GRAU) { /* wurde kein Magiegebiet angegeben, wird davon * ausgegangen, daß das normal gelernt werden soll */ if(u->faction->magiegebiet != 0) { mtyp = u->faction->magiegebiet; } else { /* Es wurde kein Magiegebiet angegeben und die Partei * hat noch keins gewählt. */ cmistake(u, ord, 178, MSG_MAGIC); return 0; } } if (mtyp != u->faction->magiegebiet){ /* Es wurde versucht, ein anderes Magiegebiet zu lernen * als das der Partei */ if (u->faction->magiegebiet != 0){ cmistake(u, ord, 179, MSG_MAGIC); return 0; } else { /* Lernt zum ersten mal Magie und legt damit das * Magiegebiet der Partei fest */ u->faction->magiegebiet = mtyp; } } if (!is_mage(u)) create_mage(u, mtyp); } else { /* ist schon ein Magier und kein Vertrauter */ if(u->faction->magiegebiet == 0){ /* die Partei hat noch kein Magiegebiet gewählt. */ mtyp = getmagicskill(); if (mtyp == M_NONE){ cmistake(u, ord, 178, MSG_MAGIC); return 0; } else { /* Legt damit das Magiegebiet der Partei fest */ u->faction->magiegebiet = mtyp; } } } } if (sk == SK_ALCHEMY) { maxalchemy = eff_skill(u, SK_ALCHEMY, r); if (has_skill(u, SK_ALCHEMY)==0 && count_skill(u->faction, SK_ALCHEMY) + u->number > max_skill(u->faction, SK_ALCHEMY)) { sprintf(buf, "Es kann maximal %d Alchemisten pro Partei geben", max_skill(u->faction, SK_ALCHEMY)); mistake(u, ord, buf, MSG_EVENT); return 0; } } if (studycost) { int cost = studycost * u->number; money = get_pooled(u, oldresourcetype[R_SILVER], GET_DEFAULT, cost); money = min(money, cost); } if (money < studycost * u->number) { studycost = p; /* Ohne Univertreurung */ money = min(money, studycost); if (p>0 && money < studycost * u->number) { #ifdef PARTIAL_STUDY cmistake(u, ord, 65, MSG_EVENT); multi = money / (double)(studycost * u->number); #else cmistake(u, ord, 65, MSG_EVENT); return 0; /* nein, Silber reicht auch so nicht */ #endif } } if (teach==NULL) { a = a_add(&u->attribs, a_new(&at_learning)); teach = (teaching_info*)a->data.v; teach->teachers[0] = 0; } if (money>0) { use_pooled(u, oldresourcetype[R_SILVER], GET_DEFAULT, money); ADDMSG(&u->faction->msgs, msg_message("studycost", "unit region cost skill", u, u->region, money, sk)); } if (get_effect(u, oldpotiontype[P_WISE])) { l = min(u->number, get_effect(u, oldpotiontype[P_WISE])); teach->value += l * 10; change_effect(u, oldpotiontype[P_WISE], -l); } if (get_effect(u, oldpotiontype[P_FOOL])) { l = min(u->number, get_effect(u, oldpotiontype[P_FOOL])); teach->value -= l * 30; change_effect(u, oldpotiontype[P_FOOL], -l); } #ifdef KARMA_MODULE l = fspecial(u->faction, FS_WARRIOR); if (l > 0) { if (sk == SK_CROSSBOW || sk == SK_LONGBOW || sk == SK_CATAPULT || sk == SK_MELEE || sk == SK_SPEAR || sk == SK_AUSDAUER || sk == SK_WEAPONLESS) { teach->value += u->number * 5 * (l+1); } else { teach->value -= u->number * 5 * (l+1); teach->value = max(0, teach->value); } } #endif /* KARMA_MODULE */ if (p != studycost) { /* ist_in_gebaeude(r, u, BT_UNIVERSITAET) == 1) { */ /* p ist Kosten ohne Uni, studycost mit; wenn * p!=studycost, ist die Einheit zwangsweise * in einer Uni */ teach->value += u->number * 10; } if (is_cursed(r->attribs, C_BADLEARN,0)) { teach->value -= u->number * 10; } days = (int)((u->number * 30 + teach->value) * multi); /* the artacademy currently improves the learning of entertainment of all units in the region, to be able to make it cumulative with with an academy */ if (sk == SK_ENTERTAINMENT && buildingtype_exists(r, bt_find("artacademy"))) { days *= 2; } if (fval(u, UFL_HUNGER)) days /= 2; while (days) { if (days>=u->number*30) { learn_skill(u, sk, 1.0); days -= u->number*30; } else { double chance = (double)days/u->number/30; learn_skill(u, sk, chance); days = 0; } } if (a!=NULL) { if (teach!=NULL) { int index = 0; while (teach->teachers[index] && index!=MAXTEACHERS) { unit * teacher = teach->teachers[index++]; if (teacher->faction != u->faction) { ADDMSG(&u->faction->msgs, msg_message("teach_student", "teacher student skill", teacher, u, sk)); ADDMSG(&teacher->faction->msgs, msg_message("teach_teacher", "teacher student skill level", teacher, u, sk, effskill(u, sk))); } } } a_remove(&u->attribs, a); a = NULL; } fset(u, UFL_LONGACTION|UFL_NOTMOVING); /* Anzeigen neuer Tränke */ /* Spruchlistenaktualiesierung ist in Regeneration */ if (sk == SK_ALCHEMY) { const potion_type * ptype; faction * f = u->faction; int skill = eff_skill(u, SK_ALCHEMY, r); if (skill>maxalchemy) { for (ptype=potiontypes; ptype; ptype=ptype->next) { if (skill == ptype->level * 2) { attrib * a = a_find(f->attribs, &at_showitem); while (a && a->type==&at_showitem && a->data.v != ptype) a=a->next; if (a==NULL || a->type!=&at_showitem) { a = a_add(&f->attribs, a_new(&at_showitem)); a->data.v = (void*) ptype->itype; } } } } } return 0; } void teaching(region *r) { /* das sind alles befehle, die 30 tage brauchen, und die in thisorder * stehen! von allen 30-tage befehlen wird einfach der letzte verwendet * (dosetdefaults). * * lehren vor lernen. */ unit *u; for (u = r->units; u; u = u->next) { if (u->race == new_race[RC_SPELL] || fval(u, UFL_LONGACTION)) continue; if (fval(r->terrain, SEA_REGION) && u->race != new_race[RC_AQUARIAN] && !(u->race->flags & RCF_SWIM)) continue; if (u->race == new_race[RC_INSECT] && r_insectstalled(r) && !is_cursed(u->attribs, C_KAELTESCHUTZ,0)) { continue; } switch (get_keyword(u->thisorder)) { case K_TEACH: if (fval(u, UFL_LONGACTION)) { cmistake(u, u->thisorder, 52, MSG_PRODUCE); continue; } else { static const curse_type * gbdream_ct = NULL; if (gbdream_ct==0) gbdream_ct = ct_find("gbdream"); if (gbdream_ct) { if (get_curse(u->region->attribs, gbdream_ct)) { ADDMSG(&u->faction->msgs, msg_feedback(u, u->thisorder, "gbdream_noteach", "")); continue; } } } teach(u, u->thisorder); break; } } }