glx@6956: /* $Id$ */ glx@6956: glx@6956: /** @file newgrf_townname.cpp glx@6956: * Implementation of Action 0F "universal holder" structure and functions. glx@6956: * This file implements a linked-lists of townname generators, glx@6956: * holding everything that the newgrf action 0F will send over to OpenTTD. glx@6956: */ glx@6956: glx@6956: #include "stdafx.h" glx@6956: #include "openttd.h" glx@6956: #include "newgrf_townname.h" rubidium@8130: #include "core/alloc_func.hpp" rubidium@8214: #include "string_func.h" glx@6956: rubidium@8264: #include "table/strings.h" rubidium@8264: glx@6956: static GRFTownName *_grf_townnames = NULL; glx@6956: glx@6956: GRFTownName *GetGRFTownName(uint32 grfid) glx@6956: { glx@6956: GRFTownName *t = _grf_townnames; glx@6956: for (; t != NULL; t = t->next) { glx@6956: if (t->grfid == grfid) return t; glx@6956: } glx@6956: return NULL; glx@6956: } glx@6956: glx@6956: GRFTownName *AddGRFTownName(uint32 grfid) glx@6956: { glx@6956: GRFTownName *t = GetGRFTownName(grfid); glx@6956: if (t == NULL) { glx@6956: t = CallocT(1); glx@6956: t->grfid = grfid; glx@6956: t->next = _grf_townnames; glx@6956: _grf_townnames = t; glx@6956: } glx@6956: return t; glx@6956: } glx@6956: glx@6956: void DelGRFTownName(uint32 grfid) glx@6956: { glx@6956: GRFTownName *t = _grf_townnames; glx@6956: GRFTownName *p = NULL; glx@6956: for (;t != NULL; p = t, t = t->next) if (t->grfid == grfid) break; glx@6956: if (t != NULL) { glx@6956: for (int i = 0; i < 128; i++) { glx@6956: for (int j = 0; j < t->nbparts[i]; j++) { glx@6956: for (int k = 0; k < t->partlist[i][j].partcount; k++) { skidd13@7928: if (!HasBit(t->partlist[i][j].parts[k].prob, 7)) free(t->partlist[i][j].parts[k].data.text); glx@6956: } glx@6956: free(t->partlist[i][j].parts); glx@6956: } glx@6956: free(t->partlist[i]); glx@6956: } glx@6956: if (p != NULL) { glx@6956: p->next = t->next; glx@6956: } else { glx@6956: _grf_townnames = t->next; glx@6956: } glx@6956: free(t); glx@6956: } glx@6956: } glx@6956: glx@6956: static char *RandomPart(char *buf, GRFTownName *t, uint32 seed, byte id, const char *last) glx@6956: { glx@6956: assert(t != NULL); glx@6956: for (int i = 0; i < t->nbparts[id]; i++) { glx@6956: byte count = t->partlist[id][i].bitcount; glx@6956: uint16 maxprob = t->partlist[id][i].maxprob; glx@6956: uint32 r = (GB(seed, t->partlist[id][i].bitstart, count) * maxprob) >> count; glx@6956: for (int j = 0; j < t->partlist[id][i].partcount; j++) { glx@6956: byte prob = t->partlist[id][i].parts[j].prob; glx@6956: maxprob -= GB(prob, 0, 7); glx@6956: if (maxprob > r) continue; skidd13@7928: if (HasBit(prob, 7)) { glx@6956: buf = RandomPart(buf, t, seed, t->partlist[id][i].parts[j].data.id, last); glx@6956: } else { glx@6956: buf = strecat(buf, t->partlist[id][i].parts[j].data.text, last); glx@6956: } glx@6956: break; glx@6956: } glx@6956: } glx@6956: return buf; glx@6956: } glx@6956: glx@6956: char *GRFTownNameGenerate(char *buf, uint32 grfid, uint16 gen, uint32 seed, const char *last) glx@6956: { glx@6956: strecpy(buf, "", last); glx@6956: for (GRFTownName *t = _grf_townnames; t != NULL; t = t->next) { glx@6956: if (t->grfid == grfid) { glx@6956: assert(gen < t->nb_gen); glx@6956: buf = RandomPart(buf, t, seed, t->id[gen], last); glx@6956: break; glx@6956: } glx@6956: } glx@6956: return buf; glx@6956: } glx@6956: glx@6956: StringID *GetGRFTownNameList() glx@6956: { glx@6956: int nb_names = 0, n = 0; glx@6956: for (GRFTownName *t = _grf_townnames; t != NULL; t = t->next) nb_names += t->nb_gen; glx@6956: StringID *list = MallocT(nb_names + 1); glx@6956: for (GRFTownName *t = _grf_townnames; t != NULL; t = t->next) { glx@6956: for (int j = 0; j < t->nb_gen; j++) list[n++] = t->name[j]; glx@6956: } glx@6956: list[n] = INVALID_STRING_ID; glx@6956: return list; glx@6956: } glx@6956: glx@6956: void CleanUpGRFTownNames() glx@6956: { glx@6956: while (_grf_townnames != NULL) DelGRFTownName(_grf_townnames->grfid); glx@6956: } glx@6956: glx@6956: uint32 GetGRFTownNameId(int gen) glx@6956: { glx@6956: for (GRFTownName *t = _grf_townnames; t != NULL; t = t->next) { glx@6956: if (gen < t->nb_gen) return t->grfid; glx@6956: gen -= t->nb_gen; glx@6956: } glx@6956: /* Fallback to no NewGRF */ glx@6956: return 0; glx@6956: } glx@6956: glx@6956: uint16 GetGRFTownNameType(int gen) glx@6956: { glx@6956: for (GRFTownName *t = _grf_townnames; t != NULL; t = t->next) { glx@6956: if (gen < t->nb_gen) return gen; glx@6956: gen -= t->nb_gen; glx@6956: } glx@6956: /* Fallback to english original */ glx@6956: return SPECSTR_TOWNNAME_ENGLISH; glx@6956: }