640 lines
20 KiB
TypeScript
640 lines
20 KiB
TypeScript
/**
|
|
* Copyright (c) 2021 lingdocs.com
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
*/
|
|
|
|
export type PsStringField = "p" | "f";
|
|
export type PsString = {
|
|
[k in PsStringField]: string;
|
|
} & {
|
|
e?: string;
|
|
};
|
|
export type PsJSX = { p: JSX.Element, f: JSX.Element, e?: JSX.Element | string };
|
|
|
|
export type DictionaryInfo = {
|
|
title: string;
|
|
license: string;
|
|
release: number;
|
|
numberOfEntries: number;
|
|
url: string;
|
|
infoUrl: string;
|
|
}
|
|
|
|
export type Dictionary = {
|
|
info: DictionaryInfo;
|
|
entries: DictionaryEntry[];
|
|
}
|
|
|
|
// TODO: BETTER TYPING OF THIS WITH RECORD TO MAKE SURE THAT THE FIELDS LINE UP
|
|
export type DictionaryEntry = {
|
|
// BASE REQUIRED INFO
|
|
/** timestamp - used for word id */
|
|
ts: number;
|
|
/** Pashto alphabetical index */
|
|
i: number;
|
|
/** entry in Pashto */
|
|
p: string;
|
|
/** entry in Phonetics */
|
|
f: string;
|
|
/** entry in simplified phonetics */
|
|
g: string;
|
|
/** entry in English */
|
|
e: string;
|
|
|
|
// PART OF SPEECH AND LINK INFO
|
|
/** part of speech info */
|
|
c?: string;
|
|
/** link - timestamp of related word */
|
|
l?: number;
|
|
|
|
// INFLECTION INFO
|
|
/** first masculine irregular inflection in Pashto */
|
|
infap?: string;
|
|
/** first masculine irregular inflection in Phonetics */
|
|
infaf?: string;
|
|
/** base for second masculine / feminine irregular inflection in Pashto */
|
|
infbp?: string;
|
|
/** base for second masculine / feminine irregular inflection in Phonetics */
|
|
infbf?: string;
|
|
/** entry does not inflect? */
|
|
noInf?: boolean;
|
|
|
|
// PLURAL INFO
|
|
/** Arabic plural in Pashto */
|
|
app?: string;
|
|
/** Arabic plural in Phonetics */
|
|
apf?: string;
|
|
/** Pashto irregular plural in Pashto */
|
|
ppp?: string;
|
|
/** Pashto irregular plural in phonetics */
|
|
ppf?: string;
|
|
|
|
// VERB INFO
|
|
/** imperfective (present) stem in Pashto */
|
|
psp?: string;
|
|
/** imperfective (present) stem in Phonetics */
|
|
psf?: string;
|
|
/** perfective (subjunctive) stem in Pashto */
|
|
ssp?: string;
|
|
/** perfective (subjunctive) stem in Phonetics */
|
|
ssf?: string;
|
|
/** perfective root in Pashto */
|
|
prp?: string;
|
|
/** perfective root in Phonetics */
|
|
prf?: string;
|
|
/** past participle in Pashto */
|
|
pprtp?: string;
|
|
/** past participle in Phonetics */
|
|
pprtf?: string;
|
|
/** The idiosyncratic third person singular masc. short past in Pashto */
|
|
tppp?: string;
|
|
/** The idiosyncratic third person singular masc. short past in Phonetics */
|
|
tppf?: string;
|
|
/** intransitive short version is available like ګرځېږي and ګرځي */
|
|
shortIntrans?: boolean;
|
|
/** does not take a و - oo perfective verb prefix? */
|
|
noOo?: boolean;
|
|
/** takes a seperate و - oo perfective verb prefix? */
|
|
sepOo?: boolean;
|
|
/** Pashto separation point for seperable verbs */
|
|
separationAtP?: number;
|
|
/** Phonetics separation point for seperable verbs */
|
|
separationAtF?: number;
|
|
|
|
// PHONETICS - PASHTO - DIACRITICS INFO
|
|
/** Is an exception to the rules of diacritics for Pashto/Phonetics */
|
|
diacExcept?: boolean;
|
|
|
|
/** the English conjugations of a verb comma seperated set of 5 ie. "see,sees,seeing,saw,seen" or single word ie. "walk" if regular - or the english singular version of a noun */
|
|
ec?: string;
|
|
/** the English partical of a English phrasal verb - or the english irregular plural of a noun */
|
|
ep?: string;
|
|
}
|
|
|
|
export type DictionaryEntryNoFVars = DictionaryEntry & { __brand: "name for a dictionary entry with all the phonetics variations removed" };
|
|
export type PsStringNoFVars = PsString & { __brand: "name for a ps string with all the phonetics variations removed" };
|
|
export type FStringNoFVars = string & { __brand: "name for a phonetics string with all the phonetics variations removed" };
|
|
|
|
export type DictionaryEntryTextField = "p" | "f" | "e" | "c" | "infap" | "infaf" | "infbp" | "infbf" | "app" | "apf" | "ppp" | "ppf" | "psp" | "psf" | "ssp" | "ssf" | "prp" | "prf" | "pprtp" | "pprtf" | "tppp" | "tppf" | "ec" | "ep";
|
|
export type DictionaryEntryBooleanField = "noInf" | "shortIntrans" | "noOo" | "sepOo" | "diacExcept";
|
|
export type DictionaryEntryNumberField = "ts" | "i" | "l" | "separationAtP" | "separationAtF";
|
|
export type DictionaryEntryField = DictionaryEntryTextField | DictionaryEntryBooleanField | DictionaryEntryNumberField;
|
|
|
|
export type DictionaryEntryError = {
|
|
errors: string[],
|
|
p: string,
|
|
f: string,
|
|
e: string,
|
|
ts: number,
|
|
erroneousFields: DictionaryEntryField[],
|
|
}
|
|
|
|
export type Spelling = "Afghan" | "Pakistani ی" | "Pakistani ي";
|
|
|
|
export type TextOptions = {
|
|
pTextSize: "normal" | "larger" | "largest";
|
|
phonetics: "lingdocs" | "ipa" | "alalc" | "none";
|
|
dialect: "standard" | "peshawer" | "southern";
|
|
spelling: Spelling;
|
|
diacritics: boolean;
|
|
}
|
|
|
|
export enum Person {
|
|
FirstSingMale = 0,
|
|
FirstSingFemale,
|
|
SecondSingMale,
|
|
SecondSingFemale,
|
|
ThirdSingMale,
|
|
ThirdSingFemale,
|
|
FirstPlurMale,
|
|
FirstPlurFemale,
|
|
SecondPlurMale,
|
|
SecondPlurFemale,
|
|
ThirdPlurMale,
|
|
ThirdPlurFemale,
|
|
}
|
|
|
|
// INPUT
|
|
// all information to be passed to conjugating functions
|
|
export type VerbInfoBase = {
|
|
transitivity: Transitivity;
|
|
yulEnding: boolean | null;
|
|
stem: VerbStemSet;
|
|
root: VerbRootSet;
|
|
participle: ParticipleSet;
|
|
idiosyncraticThirdMascSing?: ShortThirdPersFormSet;
|
|
}
|
|
|
|
export type SimpleVerbInfo = VerbInfoBase & {
|
|
type: "simple";
|
|
}
|
|
|
|
export type StativeCompoundVerbInfo = VerbInfoBase & {
|
|
type: "stative compound"
|
|
complement: UnisexInflections;
|
|
}
|
|
|
|
export type GenerativeStativeCompoundVerbInfo = VerbInfoBase & {
|
|
type: "generative stative compound"
|
|
objComplement: ObjComplement,
|
|
singularForm?: GenerativeStativeCompoundVerbInfo,
|
|
// TODO: Could add intransitive form 🤪
|
|
}
|
|
|
|
export type DynamicCompoundVerbInfo = VerbInfoBase & {
|
|
type: "dynamic compound";
|
|
objComplement: ObjComplement;
|
|
auxVerb: DictionaryEntry;
|
|
auxVerbComplement?: DictionaryEntry;
|
|
singularForm?: DynamicCompoundVerbInfo;
|
|
intransitiveForm?: DynamicCompoundVerbInfo;
|
|
}
|
|
|
|
export type ObjComplement = {
|
|
entry: DictionaryEntry;
|
|
plural?: PsString;
|
|
person: Person;
|
|
}
|
|
|
|
export type NonComboVerbInfo = SimpleVerbInfo |
|
|
StativeCompoundVerbInfo | DynamicCompoundVerbInfo | GenerativeStativeCompoundVerbInfo;
|
|
|
|
export type VerbInfo = NonComboVerbInfo | {
|
|
type: "transitive or grammatically transitive simple";
|
|
transitive: SimpleVerbInfo;
|
|
grammaticallyTransitive: SimpleVerbInfo;
|
|
} | {
|
|
type: "dynamic or stative compound";
|
|
transitivity: Transitivity;
|
|
stative: StativeCompoundVerbInfo;
|
|
dynamic: DynamicCompoundVerbInfo;
|
|
} | {
|
|
type: "dynamic or generative stative compound";
|
|
transitivity: Transitivity;
|
|
stative: GenerativeStativeCompoundVerbInfo;
|
|
dynamic: DynamicCompoundVerbInfo;
|
|
}
|
|
|
|
export type Transitivity = "transitive" | "intransitive" | "grammatically transitive";
|
|
|
|
export type SplitInfo = FullForm<[PsString, PsString]>;
|
|
|
|
export type VerbStemSet = {
|
|
perfective: FullForm<PsString>;
|
|
imperfective: FullForm<PsString>;
|
|
perfectiveSplit?: SplitInfo;
|
|
}
|
|
|
|
export type VerbRootSet = {
|
|
perfective: OptionalPersonInflections<LengthOptions<PsString>>;
|
|
imperfective: OptionalPersonInflections<LengthOptions<PsString>>;
|
|
perfectiveSplit?: SplitInfo;
|
|
}
|
|
|
|
export type ParticipleSet = {
|
|
present: FullForm<PsString>,
|
|
past: FullForm<PsString>,
|
|
}
|
|
|
|
export type ShortThirdPersFormSet = {
|
|
[K in Aspect]: PsString;
|
|
}
|
|
|
|
export type Aspect = "perfective" | "imperfective";
|
|
|
|
export type Length = "short" | "long" | "mini";
|
|
|
|
export type LengthOptions<T> = {
|
|
long: T;
|
|
short: T;
|
|
mini?: T;
|
|
}
|
|
|
|
export type PersonInflectionsField = "mascSing" | "mascPlur" | "femSing" | "femPlur";
|
|
export type OptionalPersonInflections<T> = {
|
|
[K in PersonInflectionsField]: T;
|
|
} | T;
|
|
|
|
export type SingleOrLengthOpts<T> = T | LengthOptions<T>;
|
|
|
|
export type VerbConjugation = {
|
|
info: NonComboVerbInfo,
|
|
// INFINITIVE = info.root.imperfective.long
|
|
imperfective: AspectContent; // --╖ ASPECT = "imperfective"
|
|
perfective: AspectContent; // --╜ ASPECT = "perfective"
|
|
hypothetical: VerbForm; // INFINITIVE - ul + aay
|
|
participle: ParticipleContent;
|
|
perfect: PerfectContent; // PPART = PAST PARTICIPLE (plus spectial short forms)
|
|
passive?: PassiveContent; // only on transitive verbs
|
|
singularForm?: VerbConjugation;
|
|
}
|
|
|
|
export type VerbOutput = VerbConjugation | {
|
|
info: VerbInfo,
|
|
stative: VerbConjugation,
|
|
dynamic: VerbConjugation,
|
|
} | {
|
|
info: VerbInfo,
|
|
transitive: VerbConjugation,
|
|
grammaticallyTransitive: VerbConjugation,
|
|
};
|
|
|
|
export type PassiveContent = {
|
|
imperfective: AspectContentPassive // --╖ ASPECT = "imperfective"
|
|
perfective: AspectContentPassive // --╜ ASPECT = "perfective"
|
|
perfect: PerfectContent; // PPART INFINITIVE + kedulStat perfect.pastParticiple
|
|
// TODO: ADD PARTICIPLE
|
|
}
|
|
|
|
// ASPECT -> AspectContent
|
|
export type AspectContent = {
|
|
// STEM = info.stem[ASPECT]
|
|
// ROOT = info.root[ASPECT]
|
|
nonImperative: VerbForm; // STEM + pres ending
|
|
future: VerbForm; // به + this.nonImperative
|
|
imperative?: ImperativeForm; // STEM + imperative ending
|
|
// -- optional because not used for intransitive verison of kawul dynamic compounds
|
|
past: VerbForm; // ROOT + past ending
|
|
habitualPast: VerbForm; // ba + past
|
|
modal: ModalContent;
|
|
}
|
|
|
|
export type ModalContent = {
|
|
nonImperative: VerbForm; // ROOT + ey + kedulStat.perfective.nonImperative
|
|
future: VerbForm; // به + this.nonImperative
|
|
past: VerbForm; // ROOT + ey + kedulStat.perfective.past
|
|
habitualPast: VerbForm; // ba + past
|
|
hypotheticalPast: VerbForm; // ROOT + ey + shw + ey
|
|
}
|
|
|
|
// ASPECT -> AspectContentPssive
|
|
export type AspectContentPassive = {
|
|
// ROOT = info.root[ASPECT]
|
|
nonImperative: VerbForm; // ROOT LONG + kedulStat[ASPECT].nonImperative
|
|
future: VerbForm; // ba + this.nonImperative
|
|
past: VerbForm; // ROOT LONG + kedulStat[ASPECT].past
|
|
habitualPast: VerbForm;
|
|
modal: ModalContent,
|
|
}
|
|
|
|
export type ParticipleForm = SingleOrLengthOpts<UnisexInflections> | SingleOrLengthOpts<PsString>;
|
|
|
|
export type ParticipleContent = {
|
|
past: SingleOrLengthOpts<UnisexInflections> | SingleOrLengthOpts<PsString>,
|
|
// TODO: Should this ever have an object matrix??
|
|
present: SingleOrLengthOpts<UnisexInflections>,
|
|
}
|
|
|
|
// PPART -> PerfectContent
|
|
export type PerfectContent = {
|
|
halfPerfect: VerbForm; // PPART
|
|
past: VerbForm; // PPART + equative.past
|
|
present: VerbForm; // PPART + equative.prest
|
|
habitual: VerbForm; // PPART + equative.habit
|
|
subjunctive: VerbForm; // PPART + equative.subj
|
|
future: VerbForm; // ba + PPART + equative.subj
|
|
affirmational: VerbForm; // ba + PPART + equative.past
|
|
pastSubjunctiveHypothetical: VerbForm; // PPART + waay
|
|
}
|
|
|
|
// Plain, 1st, and 2nd Inflection
|
|
export type InflectionSet = ArrayFixed<ArrayOneOrMore<PsString>, 3>;
|
|
|
|
// Plural and Second Inflection
|
|
export type PluralInflectionSet = ArrayFixed<ArrayOneOrMore<PsString>, 2>
|
|
|
|
export type Gender = "masc" | "fem";
|
|
|
|
export type UnisexSet<T> = Record<Gender, T>;
|
|
export type GenderedSet<T> = UnisexSet<T> | Omit<UnisexSet<T>, "fem"> | Omit<UnisexSet<T>, "masc">;
|
|
export type UnisexInflections = UnisexSet<InflectionSet>;
|
|
|
|
export type Inflections = GenderedSet<InflectionSet>;
|
|
|
|
export type PluralInflections = GenderedSet<PluralInflectionSet>;
|
|
|
|
export type InflectorOutput = {
|
|
arabicPlural: PluralInflections,
|
|
plural?: PluralInflections,
|
|
inflections?: Inflections,
|
|
} | {
|
|
plural: PluralInflections,
|
|
arabicPlural?: PluralInflections,
|
|
inflections?: Inflections,
|
|
} | {
|
|
inflections: Inflections,
|
|
} | false;
|
|
|
|
export type PersonLine = [
|
|
/** singular form of person */
|
|
ArrayOneOrMore<PsString>,
|
|
/** plural form of person */
|
|
ArrayOneOrMore<PsString>,
|
|
];
|
|
|
|
/**
|
|
* The basic form of a verb conjugation
|
|
* Each line representing one person (singular and plural)
|
|
* 1st Person Male, 1st Person Female, 2nd Person Male etc...
|
|
*/
|
|
export type VerbBlock = [
|
|
PersonLine, // 1st Person Male
|
|
PersonLine, // 1st Person Female
|
|
PersonLine, // 2nd Person Male
|
|
PersonLine, // 2nd Person Female
|
|
PersonLine, // 3rd Person Male
|
|
PersonLine, // 3rd Person Female
|
|
];
|
|
|
|
export type EnglishBlock = [
|
|
[string, string],
|
|
[string, string],
|
|
[string, string],
|
|
[string, string],
|
|
[string, string],
|
|
[string, string],
|
|
];
|
|
|
|
export type ImperativeBlock = [
|
|
PersonLine, // 2nd Person Male
|
|
PersonLine, // 2nd Person Female
|
|
];
|
|
|
|
export type FullForm<T> = OptionalPersonInflections<SingleOrLengthOpts<T>>;
|
|
export type VerbForm = FullForm<VerbBlock>;
|
|
export type ImperativeForm = FullForm<ImperativeBlock>;
|
|
export type SentenceForm = SingleOrLengthOpts<ArrayOneOrMore<PsString>>;
|
|
|
|
export interface ArrayFixed<T, L extends number> extends Array<T> {
|
|
0: T;
|
|
length: L;
|
|
}
|
|
|
|
export type Wrapper<T> = T & { __brand: "wrapped" };
|
|
|
|
export type ArrayOneOrMore<T> = {
|
|
0: T
|
|
} & Array<T>
|
|
|
|
export type RootsOrStemsToHighlight = ("imperfective root" | "perfective root" | "imperfective stem" | "perfective stem" | "past participle")[];
|
|
|
|
/* i.e. ec: ["take", "takes", "taking", "took", "taken"], ep: out */
|
|
export type EnglishVerbConjugationEc = [string, string, string, string, string];
|
|
export type EnglishVerbConjugation = {
|
|
ec: EnglishVerbConjugationEc,
|
|
ep: string | undefined,
|
|
};
|
|
|
|
export type DisplayFormItem = DisplayForm | DisplayFormSubgroup | DisplayFormForSentence;
|
|
export type EnglishBuilder = (subject: Person, ec: EnglishVerbConjugationEc, neg: boolean) => string[];
|
|
|
|
export type DisplayForm = {
|
|
label: string,
|
|
aspect?: Aspect,
|
|
form: VerbForm | ImperativeForm | ParticipleForm | SentenceForm,
|
|
advanced?: boolean,
|
|
englishBuilder?: EnglishBuilder,
|
|
formula: React.ReactNode,
|
|
explanation: React.ReactNode,
|
|
sentence?: boolean,
|
|
passive?: boolean,
|
|
past?: boolean,
|
|
reorderWithNegative?: boolean,
|
|
}
|
|
|
|
export type DisplayFormForSentence = Omit<DisplayForm, "form"> & {
|
|
form: VerbForm,
|
|
secondPronounNeeded?: boolean,
|
|
}
|
|
|
|
// {
|
|
// label: string,
|
|
// aspect?: Aspect,
|
|
// form: VerbForm,
|
|
// advanced?: boolean,
|
|
// englishBuilder?: EnglishBuilder,
|
|
// formula: React.ReactNode,
|
|
// secondPronounNeeded?: boolean,
|
|
// explanation: React.ReactNode,
|
|
// sentence?: boolean,
|
|
// passive?: boolean,
|
|
// past?: boolean,
|
|
// reorderWithNegative?: boolean,
|
|
// }
|
|
|
|
export type DisplayFormSubgroup = {
|
|
label: string,
|
|
subgroup: string,
|
|
advanced?: boolean,
|
|
content: DisplayFormItem[],
|
|
}
|
|
|
|
export type AayTail = "ey" | "aay";
|
|
|
|
export type NounEntry = DictionaryEntry & { c: string } & { __brand: "a noun entry" };
|
|
export type MascNounEntry = NounEntry & { __brand2: "a masc noun entry" };
|
|
export type FemNounEntry = NounEntry & { __brand2: "a fem noun entry" };
|
|
export type UnisexNounEntry = MascNounEntry & { __brand3: "a unisex noun entry" };
|
|
export type AdverbEntry = DictionaryEntry & { c: string } & { __brand: "an adverb entry" };
|
|
export type LocativeAdverbEntry = AdverbEntry & { __brand2: "a locative adverb entry" };
|
|
export type AdjectiveEntry = DictionaryEntry & { c: string } & { __brand: "an adjective entry" };
|
|
export type VerbEntry = {
|
|
entry: DictionaryEntry & { __brand: "a verb entry" },
|
|
// TODO: the compliment could also be typed? Maybe?
|
|
complement?: DictionaryEntry,
|
|
};
|
|
|
|
export type SingularEntry<T extends NounEntry> = T & { __brand7: "a singular noun - as opposed to an always plural noun" };
|
|
export type PluralNounEntry<T extends NounEntry> = T & { __brand7: "a noun that is always plural" };
|
|
|
|
export type Pattern1Entry<T> = T & { __brand3: "basic inflection pattern" };
|
|
export type Pattern2Entry<T> = T & { __brand3: "ending in unstressed ی pattern" };
|
|
export type Pattern3Entry<T> = T & { __brand3: "ending in stressed ی pattern" };
|
|
export type Pattern4Entry<T> = T & { __brand3: "Pashtoon pattern" };
|
|
export type Pattern5Entry<T> = T & { __brand3: "short squish pattern" };
|
|
export type Pattern6FemEntry<T extends FemNounEntry> = T & { __brand3: "non anim. ending in ي" };
|
|
export type NonInflecting<T> = T & { __brand3: "non-inflecting" };
|
|
|
|
export type Entry = NounEntry | AdjectiveEntry | AdverbEntry | VerbEntry;
|
|
|
|
export type Words = {
|
|
nouns: NounEntry[],
|
|
adjectives: AdjectiveEntry[],
|
|
verbs: VerbEntry[],
|
|
adverbs: AdverbEntry[],
|
|
}
|
|
|
|
export type VPSelection = {
|
|
type: "VPSelection",
|
|
subject: NPSelection,
|
|
object: Exclude<VerbObject, undefined>,
|
|
verb: Exclude<VerbSelection, "object">,
|
|
};
|
|
|
|
// TODO: make this Rendered<VPSelection> with recursive Rendered<>
|
|
export type VPRendered = {
|
|
type: "VPRendered",
|
|
king: "subject" | "object",
|
|
servant: "subject" | "object" | undefined,
|
|
isPast: boolean,
|
|
isTransitive: boolean,
|
|
isCompound: "stative" | "dynamic" | false,
|
|
subject: Rendered<NPSelection>,
|
|
object: Rendered<NPSelection> | ObjectNP,
|
|
verb: VerbRendered,
|
|
englishBase?: string[],
|
|
}
|
|
|
|
export type VerbTense = "presentVerb"
|
|
| "subjunctiveVerb"
|
|
| "perfectiveFuture"
|
|
| "imperfectiveFuture"
|
|
| "perfectivePast"
|
|
| "imperfectivePast"
|
|
| "habitualPerfectivePast"
|
|
| "habitualImperfectivePast";
|
|
|
|
export type EquativeTense = "present" | "subjunctive" | "habitual" | "past" | "future" | "wouldBe" | "pastSubjunctive";
|
|
export type NounNumber = "singular" | "plural";
|
|
|
|
export type PerfectTense = `${EquativeTense} perfect`;
|
|
|
|
export type VerbSelection = {
|
|
type: "verb",
|
|
verb: VerbEntry,
|
|
dynAuxVerb?: VerbEntry,
|
|
object: VerbObject, // TODO: should have a locked in (but number changeable noun) here for dynamic compounds
|
|
transitivity: Transitivity,
|
|
isCompound: "stative" | "dynamic" | false,
|
|
voice: "active" | "passive",
|
|
changeTransitivity?: (t: "transitive" | "grammatically transitive") => VerbSelection,
|
|
changeStatDyn?: (t: "stative" | "dynamic") => VerbSelection,
|
|
changeVoice?: (v: "active" | "passive", subj?: NPSelection) => VerbSelection,
|
|
// TODO: changeStativeDynamic
|
|
// TODO: add in aspect element here??
|
|
negative: boolean,
|
|
} & ({
|
|
tense: VerbTense,
|
|
tenseCategory: "basic" | "modal",
|
|
} | {
|
|
tense: PerfectTense,
|
|
tenseCategory: "perfect"
|
|
});
|
|
|
|
export type VerbRendered = Omit<VerbSelection, "object"> & {
|
|
ps: {
|
|
head: PsString | undefined,
|
|
rest: SingleOrLengthOpts<
|
|
PsString[]
|
|
>,
|
|
},
|
|
hasBa: boolean,
|
|
person: Person,
|
|
};
|
|
|
|
export type VerbObject =
|
|
// transitive verb - object not selected yet
|
|
undefined |
|
|
// transitive verb - obect selected
|
|
NPSelection |
|
|
// grammatically transitive verb with unspoken 3rd pers masc plur entity
|
|
// or intransitive "none"
|
|
ObjectNP;
|
|
|
|
export type NPSelection = NounSelection | PronounSelection | ParticipleSelection;
|
|
|
|
export type NPType = "noun" | "pronoun" | "participle";
|
|
|
|
export type ObjectNP = "none" | Person.ThirdPlurMale;
|
|
|
|
// TODO require/import Person and PsString
|
|
export type NounSelection = {
|
|
type: "noun",
|
|
entry: NounEntry,
|
|
gender: Gender,
|
|
number: NounNumber,
|
|
dynamicComplement?: boolean,
|
|
// TODO: Implement
|
|
// adjectives: [],
|
|
// TODO: Implement
|
|
// possesor: NPSelection | undefined,
|
|
/* method only present if it's possible to change gender */
|
|
changeGender?: (gender: Gender) => NounSelection,
|
|
/* method only present if it's possible to change number */
|
|
changeNumber?: (number: NounNumber) => NounSelection,
|
|
};
|
|
|
|
// take an argument for subject/object in rendering English
|
|
export type PronounSelection = {
|
|
type: "pronoun",
|
|
person: Person,
|
|
distance: "near" | "far",
|
|
};
|
|
|
|
export type ParticipleSelection = {
|
|
type: "participle",
|
|
verb: VerbEntry,
|
|
};
|
|
|
|
// not object
|
|
// type Primitive = string | Function | number | boolean | Symbol | undefined | null;
|
|
// If T has key K ("user"), replace it
|
|
export type ReplaceKey<T, K extends string, R> = T extends Record<K, unknown> ? (Omit<T, K> & Record<K, R>) : T;
|
|
|
|
export type FormVersion = { removeKing: boolean, shrinkServant: boolean };
|
|
|
|
export type Rendered<T extends NPSelection> = ReplaceKey<
|
|
Omit<T, "changeGender" | "changeNumber" | "changeDistance">,
|
|
"e",
|
|
string
|
|
> & {
|
|
ps: PsString[],
|
|
e?: string,
|
|
inflected: boolean,
|
|
person: Person,
|
|
};
|
|
// TODO: recursive changing this down into the possesor etc.
|