895 lines
30 KiB
TypeScript
895 lines
30 KiB
TypeScript
import * as _ from 'lodash';
|
|
import { Character, CharacterInfotag } from '../interfaces';
|
|
|
|
/* eslint-disable no-null-keyword */
|
|
|
|
export enum TagId {
|
|
Age = 1,
|
|
Orientation = 2,
|
|
Gender = 3,
|
|
Build = 13,
|
|
FurryPreference = 29,
|
|
SubDomRole = 15,
|
|
Position = 41,
|
|
BodyType = 51,
|
|
ApparentAge = 64,
|
|
RelationshipStatus = 42,
|
|
Species = 9,
|
|
LanguagePreference = 49
|
|
}
|
|
|
|
export enum Gender {
|
|
Male = 1,
|
|
Female = 2,
|
|
Transgender = 3,
|
|
Herm = 32,
|
|
MaleHerm = 51,
|
|
Cuntboy = 69,
|
|
None = 105,
|
|
Shemale = 141
|
|
}
|
|
|
|
export enum SubDomRole {
|
|
AlwaysSubmissive = 7,
|
|
UsuallySubmissive = 8,
|
|
Switch = 9,
|
|
UsuallyDominant = 10,
|
|
AlwaysDominant = 11
|
|
}
|
|
|
|
export enum Orientation {
|
|
Straight = 4,
|
|
Gay = 5,
|
|
Bisexual = 6,
|
|
Asexual = 7,
|
|
Unsure = 8,
|
|
BiMalePreference = 89,
|
|
BiFemalePreference = 90,
|
|
Pansexual = 127,
|
|
BiCurious = 128
|
|
}
|
|
|
|
export enum BodyType {
|
|
Anthro = 122,
|
|
Feral = 121,
|
|
Morphable = 123,
|
|
Varies = 124,
|
|
Other = 125,
|
|
Androgynous = 126,
|
|
Human = 143,
|
|
Taur = 145
|
|
}
|
|
|
|
export enum KinkPreference {
|
|
Favorite = 1,
|
|
Yes = 0.5,
|
|
Maybe = -0.5,
|
|
No = -1
|
|
}
|
|
|
|
enum Kink {
|
|
Females = 554,
|
|
MaleHerms = 552,
|
|
Males = 553,
|
|
Transgenders = 551,
|
|
Herms = 132,
|
|
Shemales = 356,
|
|
Cuntboys = 231,
|
|
|
|
OlderCharacters = 109,
|
|
YoungerCharacters = 197,
|
|
Ageplay = 196,
|
|
UnderageCharacters = 207,
|
|
|
|
RoleReversal = 408,
|
|
|
|
AnthroCharacters = 587,
|
|
Humans = 609,
|
|
|
|
Mammals = 224
|
|
}
|
|
|
|
export enum FurryPreference {
|
|
FurriesOnly = 39,
|
|
FursAndHumans = 40,
|
|
HumansOnly = 41,
|
|
HumansPreferredFurriesOk = 150,
|
|
FurriesPreferredHumansOk = 149
|
|
}
|
|
|
|
interface GenderKinkIdMap {
|
|
[key: number]: Kink
|
|
}
|
|
|
|
const genderKinkMapping: GenderKinkIdMap = {
|
|
[Gender.Female]: Kink.Females,
|
|
[Gender.Male]: Kink.Males,
|
|
[Gender.Cuntboy]: Kink.Cuntboys,
|
|
[Gender.Herm]: Kink.Herms,
|
|
[Gender.MaleHerm]: Kink.MaleHerms,
|
|
[Gender.Shemale]: Kink.Shemales,
|
|
[Gender.Transgender]: Kink.Transgenders
|
|
};
|
|
|
|
// if no species and 'no furry characters', === human
|
|
// if no species and dislike 'anthro characters' === human
|
|
|
|
export enum Species {
|
|
Human = 609,
|
|
Humanoid = 131,
|
|
Bovine = 318,
|
|
Equine = 236,
|
|
Feline = 212,
|
|
Canine = 226,
|
|
Caprinae = 558,
|
|
Demon = 7,
|
|
Divinity = 530,
|
|
Vulpine = 213,
|
|
Avian = 215,
|
|
Amphibian = 223,
|
|
Cervine = 227,
|
|
Insect = 237,
|
|
Lapine = 214,
|
|
Musteline = 328,
|
|
Dragon = 228,
|
|
Procyon = 325,
|
|
Rodent = 283,
|
|
Ursine = 326,
|
|
MarineMammal = 309,
|
|
Primate = 613,
|
|
Elf = 611,
|
|
Orc = 615,
|
|
Fish = 608,
|
|
Reptile = 225,
|
|
Marsupial = 322,
|
|
Anthro = 587,
|
|
Robot = 161,
|
|
Hyaenidae = 321,
|
|
Mephitidae = 323,
|
|
Bat = 451,
|
|
Alien = 281,
|
|
Dinosaur = 610,
|
|
Pokemon = 504,
|
|
Fae = 612,
|
|
Taur = 68,
|
|
Vampire = 182,
|
|
Naga = 619,
|
|
Monster = 483,
|
|
|
|
Minotaur = 12121212,
|
|
Giraffe = 13131313,
|
|
Rhinoceros = 14141414
|
|
}
|
|
|
|
const nonAnthroSpecies = [
|
|
Species.Human, Species.Elf, Species.Orc, Species.Humanoid,
|
|
Species.Demon, Species.Divinity, Species.Alien, Species.Robot,
|
|
Species.Fae, Species.Vampire
|
|
];
|
|
|
|
const mammalSpecies = [Species.Equine, Species.Feline, Species.Canine, Species.Vulpine, Species.Cervine, Species.Lapine,
|
|
Species.Musteline, Species.Procyon, Species.Rodent, Species.Ursine, Species.MarineMammal, Species.Primate,
|
|
Species.Anthro, Species.Bovine, Species.Caprinae, Species.Marsupial, Species.Hyaenidae, Species.Minotaur,
|
|
Species.Bat, Species.Mephitidae, Species.Taur, Species.Giraffe, Species.Rhinoceros];
|
|
|
|
interface SpeciesMap {
|
|
[key: number]: string[];
|
|
}
|
|
|
|
interface SpeciesStrMap {
|
|
[key: number]: string;
|
|
}
|
|
|
|
export const speciesNames: SpeciesStrMap = {
|
|
[Species.MarineMammal]: 'marine mammals',
|
|
[Species.Elf]: 'elves',
|
|
[Species.Fish]: 'fishes',
|
|
[Species.Mephitidae]: 'mephitis',
|
|
[Species.Rhinoceros]: 'rhinoceros'
|
|
};
|
|
|
|
const speciesMapping: SpeciesMap = {
|
|
[Species.Human]: ['human', 'humanoid', 'angel', 'android', 'african american', 'africanamerican', 'woman', 'dothraki', 'homo sapien', 'homosapien', 'homosapian', 'hooman', 'hoomin', 'hooomin'],
|
|
[Species.Humanoid]: ['satyr', 'gnome', 'dwarf', 'halfling', 'tiefling', 'humanoid'],
|
|
[Species.Equine]: ['horse', 'stallion', 'mare', 'filly', 'equine', 'shire', 'donkey', 'mule', 'zebra', 'pony', 'unicorn', 'clydesdale', 'shire',
|
|
'appaloosa', 'friesian', 'draft', 'draught', 'alicorn', 'amazon', 'amazonian', 'horsie', 'hoss', 'pegasus', 'colt', 'filly'],
|
|
[Species.Feline]: ['cat', 'kitten', 'catgirl', 'neko', 'tiger', 'puma', 'lion', 'lioness',
|
|
'tigress', 'feline', 'jaguar', 'cheetah', 'lynx', 'leopard', 'cougar', 'kitty', 'migote', 'miqo\'te', 'miqote', 'ocelot',
|
|
'sabertooth', 'saber tooth', 'tabby', 'liger'],
|
|
[Species.Canine]: ['dog', 'wolf', 'dingo', 'coyote', 'jackal', 'canine', 'doberman', 'husky', 'hound', 'akita', 'pitbull', 'pit bull', 'terrier',
|
|
'bull terrier', 'australian shepherd', 'australian shepard', 'german shepherd', 'german shepard', 'malinois', 'woof', 'labrador', 'collie',
|
|
'canis', 'canid', 'chihuahua', 'poodle', 'chinchilla', 'chowchow', 'corgi', 'anubis', 'anubian', 'dalmatian', 'inumimi', 'lupine', 'malamute', 'mastiff',
|
|
'mutt', 'rottweiler', 'shih tzu', 'worgen'],
|
|
[Species.Vulpine]: ['fox', 'fennec', 'kitsune', 'vulpine', 'vixen'],
|
|
[Species.Avian]: ['bird', 'gryphon', 'phoenix', 'roc', 'chimera', 'avian', 'albatross', 'cockatiel', 'dove', 'eagle', 'owl', 'penguin', 'raven'],
|
|
[Species.Amphibian]: ['salamander', 'frog', 'toad', 'newt', 'amphibian'],
|
|
[Species.Cervine]: ['deer', 'elk', 'moose', 'cervid', 'cervine', 'caribou', 'reindeer', 'doe', 'stag'],
|
|
[Species.Insect]: ['bee', 'wasp', 'spider', 'scorpion', 'ant', 'insect'],
|
|
[Species.Lapine]: ['bunny', 'rabbit', 'hare', 'lapine'],
|
|
[Species.Dragon]: ['dragon', 'drake', 'wyvern', 'draconian'],
|
|
[Species.Demon]: ['demon', 'daemon', 'deamon', 'demoness', 'demonkin', 'devil', 'succubus', 'incubus', 'baphomet'],
|
|
[Species.Musteline]: ['mink', 'ferret', 'weasel', 'stoat', 'otter', 'wolverine', 'marten', 'musteline'],
|
|
[Species.Procyon]: ['raccoon', 'racoon', 'coatimund', 'longtail', 'procyon'],
|
|
[Species.Rodent]: ['rat', 'mouse', 'chipmunk', 'squirrel', 'rodent', 'maus'],
|
|
[Species.Ursine]: ['bear', 'panda', 'black bear', 'brown bear', 'polar bear', 'ursine'],
|
|
[Species.MarineMammal]: ['whale', 'killer whale', 'dolphin'],
|
|
[Species.Primate]: ['monkey', 'ape', 'chimp', 'chimpanzee', 'gorilla', 'lemur', 'silverback'],
|
|
[Species.Divinity]: ['god', 'goddess', 'demigod', 'demigoddess', 'demi-god', 'demi-goddess'],
|
|
[Species.Elf]: ['elf', 'e l f', 'drow', 'draenei', 'draenai', 'kaldorei', 'sindorei'],
|
|
[Species.Fish]: ['fish', 'shark', 'great white', 'sergal', 'elven'],
|
|
[Species.Orc]: ['orc'],
|
|
[Species.Reptile]: ['chameleon', 'anole', 'alligator', 'aligator', 'snake', 'crocodile', 'lizard', 'gator', 'gecko', 'reptile', 'reptilian'],
|
|
[Species.Anthro]: ['anthro', 'anthropomorphic'],
|
|
[Species.Bovine]: ['cow', 'bovine', 'bison', 'antelope', 'gazelle', 'oryx', 'black angus', 'bull', 'ox'],
|
|
[Species.Caprinae]: ['sheep', 'goat', 'ibex', 'takin', 'bharal', 'goral', 'serow', 'lamb'],
|
|
[Species.Marsupial]: ['opossum', 'possum', 'kangaroo', 'roo', 'koala', 'wombat'],
|
|
[Species.Hyaenidae]: ['hyena'],
|
|
[Species.Minotaur]: ['minotaur', 'tauren'],
|
|
[Species.Bat]: ['bat'],
|
|
[Species.Alien]: ['alien', 'krogan', 'xenomorph'],
|
|
[Species.Mephitidae]: ['skunk'],
|
|
[Species.Robot]: ['android', 'robot', 'cyborg'],
|
|
[Species.Dinosaur]: ['saurus', 'deathclaw', 'dinosaur', 'raptor', 'trex', 't-rex'],
|
|
[Species.Pokemon]: ['charizard', 'charmander', 'pikachu', 'digimon', 'renamon', 'eevee', 'gardevoir', 'absol', 'aggron', 'jolteon', 'lopunny'],
|
|
[Species.Fae]: ['fairy', 'fae', 'imp', 'elemental'],
|
|
[Species.Taur]: ['chakat', 'centaur', 'equitaur'],
|
|
[Species.Vampire]: ['vampyre', 'vampire', 'dhampir', 'daywalker'],
|
|
[Species.Naga]: ['naga', 'lamia'],
|
|
[Species.Monster]: ['gnoll', 'goblin', 'kobold', 'monster', 'troll', 'illithid', 'golem', 'basilisk'],
|
|
[Species.Giraffe]: ['giraffe'],
|
|
[Species.Rhinoceros]: ['rhino', 'rhinoceros']
|
|
};
|
|
|
|
|
|
interface FchatGenderMap {
|
|
[key: string]: Gender;
|
|
}
|
|
|
|
const fchatGenderMap: FchatGenderMap = {
|
|
None: Gender.None,
|
|
Male: Gender.Male,
|
|
Female: Gender.Female,
|
|
Shemale: Gender.Shemale,
|
|
Herm: Gender.Herm,
|
|
'Male-Herm': Gender.MaleHerm,
|
|
'Cunt-Boy': Gender.Cuntboy,
|
|
Transgender: Gender.Transgender
|
|
};
|
|
|
|
interface KinkPreferenceMap {
|
|
[key: string]: KinkPreference;
|
|
}
|
|
|
|
const kinkMapping: KinkPreferenceMap = {
|
|
favorite: KinkPreference.Favorite,
|
|
yes: KinkPreference.Yes,
|
|
maybe: KinkPreference.Maybe,
|
|
no: KinkPreference.No
|
|
};
|
|
|
|
export interface MatchReport {
|
|
you: MatchResult;
|
|
them: MatchResult;
|
|
}
|
|
|
|
export interface MatchResultCharacterInfo {
|
|
species: Species | null;
|
|
gender: Gender | null;
|
|
orientation: Orientation | null;
|
|
}
|
|
|
|
export interface MatchResultScores {
|
|
[key: number]: Score;
|
|
[TagId.Orientation]: Score;
|
|
[TagId.Gender]: Score;
|
|
[TagId.Age]: Score;
|
|
[TagId.FurryPreference]: Score;
|
|
[TagId.Species]: Score;
|
|
}
|
|
|
|
export interface MatchResult {
|
|
you: Character,
|
|
them: Character,
|
|
scores: MatchResultScores;
|
|
info: MatchResultCharacterInfo;
|
|
total: number;
|
|
|
|
yourAnalysis: CharacterAnalysis;
|
|
theirAnalysis: CharacterAnalysis;
|
|
}
|
|
|
|
export enum Scoring {
|
|
MATCH = 1,
|
|
WEAK_MATCH = 0.5,
|
|
NEUTRAL = 0,
|
|
WEAK_MISMATCH = -0.5,
|
|
MISMATCH = -1
|
|
}
|
|
|
|
export interface ScoreClassMap {
|
|
[key: number]: string;
|
|
}
|
|
|
|
const scoreClasses: ScoreClassMap = {
|
|
[Scoring.MATCH]: 'match',
|
|
[Scoring.WEAK_MATCH]: 'weak-match',
|
|
[Scoring.NEUTRAL]: 'neutral',
|
|
[Scoring.WEAK_MISMATCH]: 'weak-mismatch',
|
|
[Scoring.MISMATCH]: 'mismatch'
|
|
};
|
|
|
|
export class Score {
|
|
readonly score: Scoring;
|
|
readonly description: string;
|
|
|
|
constructor(score: Scoring, description: string = '') {
|
|
if ((score !== Scoring.NEUTRAL) && (description === ''))
|
|
throw new Error('Description must be provided if score is not neutral');
|
|
|
|
this.score = score;
|
|
this.description = description;
|
|
}
|
|
|
|
getRecommendedClass(): string {
|
|
return Score.getClasses(this.score);
|
|
}
|
|
|
|
static getClasses(score: Scoring): string {
|
|
return scoreClasses[score];
|
|
}
|
|
}
|
|
|
|
|
|
export class CharacterAnalysis {
|
|
readonly character: Character;
|
|
|
|
readonly gender: Gender | null;
|
|
readonly orientation: Orientation | null;
|
|
readonly species: Species | null;
|
|
readonly furryPreference: FurryPreference | null;
|
|
readonly age: number | null;
|
|
readonly subDomRole: SubDomRole | null;
|
|
|
|
readonly isAnthro: boolean | null;
|
|
readonly isHuman: boolean | null;
|
|
readonly isMammal: boolean | null;
|
|
|
|
constructor(c: Character) {
|
|
this.character = c;
|
|
|
|
this.gender = Matcher.getTagValueList(TagId.Gender, c);
|
|
this.orientation = Matcher.getTagValueList(TagId.Orientation, c);
|
|
this.species = Matcher.species(c);
|
|
this.furryPreference = Matcher.getTagValueList(TagId.FurryPreference, c);
|
|
this.subDomRole = Matcher.getTagValueList(TagId.SubDomRole, c);
|
|
|
|
const ageTag = Matcher.getTagValue(TagId.Age, c);
|
|
|
|
this.age = ((ageTag) && (ageTag.string)) ? parseInt(ageTag.string, 10) : null;
|
|
|
|
this.isAnthro = Matcher.isAnthro(c);
|
|
this.isHuman = Matcher.isHuman(c);
|
|
this.isMammal = Matcher.isMammal(c);
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Answers the question: What YOU think about THEM
|
|
* Never what THEY think about YOU
|
|
*
|
|
* So, when comparing two characters, you have to run it twice (you, them / them, you)
|
|
* to get the full picture
|
|
*/
|
|
export class Matcher {
|
|
readonly you: Character;
|
|
readonly them: Character;
|
|
|
|
readonly yourAnalysis: CharacterAnalysis;
|
|
readonly theirAnalysis: CharacterAnalysis;
|
|
|
|
|
|
constructor(you: Character, them: Character, yourAnalysis?: CharacterAnalysis, theirAnalysis?: CharacterAnalysis) {
|
|
this.you = you;
|
|
this.them = them;
|
|
|
|
this.yourAnalysis = yourAnalysis || new CharacterAnalysis(you);
|
|
this.theirAnalysis = theirAnalysis || new CharacterAnalysis(them);
|
|
}
|
|
|
|
static generateReport(you: Character, them: Character): MatchReport {
|
|
const yourAnalysis = new CharacterAnalysis(you);
|
|
const theirAnalysis = new CharacterAnalysis(them);
|
|
|
|
const youThem = new Matcher(you, them, yourAnalysis, theirAnalysis);
|
|
const themYou = new Matcher(them, you, theirAnalysis, yourAnalysis);
|
|
|
|
return {
|
|
you: youThem.match(),
|
|
them: themYou.match()
|
|
};
|
|
}
|
|
|
|
match(): MatchResult {
|
|
const data: MatchResult = {
|
|
you: this.you,
|
|
them: this.them,
|
|
|
|
yourAnalysis: this.yourAnalysis,
|
|
theirAnalysis: this.theirAnalysis,
|
|
|
|
total: 0,
|
|
|
|
scores: {
|
|
[TagId.Orientation]: this.resolveOrientationScore(),
|
|
[TagId.Gender]: this.resolveGenderScore(),
|
|
[TagId.Age]: this.resolveAgeScore(),
|
|
[TagId.FurryPreference]: this.resolveFurryPairingsScore(),
|
|
[TagId.Species]: this.resolveSpeciesScore(),
|
|
[TagId.SubDomRole]: this.resolveSubDomScore()
|
|
},
|
|
|
|
info: {
|
|
species: Matcher.species(this.you),
|
|
gender: Matcher.getTagValueList(TagId.Gender, this.you),
|
|
orientation: Matcher.getTagValueList(TagId.Orientation, this.you)
|
|
}
|
|
};
|
|
|
|
data.total = _.reduce(
|
|
data.scores,
|
|
(accum: number, s: Score) => (accum + s.score),
|
|
0
|
|
);
|
|
|
|
return data;
|
|
}
|
|
|
|
private resolveOrientationScore(): Score {
|
|
// Question: If someone identifies themselves as 'straight cuntboy', how should they be matched? like a straight female?
|
|
|
|
return Matcher.scoreOrientationByGender(this.yourAnalysis.gender, this.yourAnalysis.orientation, this.theirAnalysis.gender);
|
|
}
|
|
|
|
|
|
static scoreOrientationByGender(yourGender: Gender | null, yourOrientation: Orientation | null, theirGender: Gender | null): Score {
|
|
if ((yourGender === null) || (theirGender === null) || (yourOrientation === null))
|
|
return new Score(Scoring.NEUTRAL);
|
|
|
|
// CIS
|
|
// tslint:disable-next-line curly
|
|
if (Matcher.isCisGender(yourGender)) {
|
|
if (yourGender === theirGender) {
|
|
// same sex CIS
|
|
if (yourOrientation === Orientation.Straight)
|
|
return new Score(Scoring.MISMATCH, 'No <span>same sex</span>');
|
|
|
|
if (
|
|
(yourOrientation === Orientation.Gay)
|
|
|| (yourOrientation === Orientation.Bisexual)
|
|
|| (yourOrientation === Orientation.Pansexual)
|
|
|| ((yourOrientation === Orientation.BiFemalePreference) && (theirGender === Gender.Female))
|
|
|| ((yourOrientation === Orientation.BiMalePreference) && (theirGender === Gender.Male))
|
|
)
|
|
return new Score(Scoring.MATCH, 'Loves <span>same sex</span>');
|
|
|
|
if (
|
|
(yourOrientation === Orientation.BiCurious)
|
|
|| ((yourOrientation === Orientation.BiFemalePreference) && (theirGender === Gender.Male))
|
|
|| ((yourOrientation === Orientation.BiMalePreference) && (theirGender === Gender.Female))
|
|
)
|
|
return new Score(Scoring.WEAK_MATCH, 'Likes <span>same sex</span>');
|
|
} else if (Matcher.isCisGender(theirGender)) {
|
|
// straight CIS
|
|
if (yourOrientation === Orientation.Gay)
|
|
return new Score(Scoring.MISMATCH, 'No <span>opposite sex</span>');
|
|
|
|
if (
|
|
(yourOrientation === Orientation.Straight)
|
|
|| (yourOrientation === Orientation.Bisexual)
|
|
|| (yourOrientation === Orientation.BiCurious)
|
|
|| (yourOrientation === Orientation.Pansexual)
|
|
|| ((yourOrientation === Orientation.BiFemalePreference) && (theirGender === Gender.Female))
|
|
|| ((yourOrientation === Orientation.BiMalePreference) && (theirGender === Gender.Male))
|
|
)
|
|
return new Score(Scoring.MATCH, 'Loves <span>opposite sex</span>');
|
|
|
|
if (
|
|
((yourOrientation === Orientation.BiFemalePreference) && (theirGender === Gender.Male))
|
|
|| ((yourOrientation === Orientation.BiMalePreference) && (theirGender === Gender.Female))
|
|
)
|
|
return new Score(Scoring.WEAK_MATCH, 'Likes <span>opposite sex</span>');
|
|
}
|
|
}
|
|
|
|
return new Score(Scoring.NEUTRAL);
|
|
}
|
|
|
|
|
|
static formatKinkScore(score: KinkPreference, description: string): Score {
|
|
if (score === KinkPreference.No)
|
|
return new Score(Scoring.MISMATCH, `No <span>${description}</span>`);
|
|
|
|
if (score === KinkPreference.Maybe)
|
|
return new Score(Scoring.WEAK_MISMATCH, `Hesitant about <span>${description}</span>`);
|
|
|
|
if (score === KinkPreference.Yes)
|
|
return new Score(Scoring.WEAK_MATCH, `Likes <span>${description}</span>`);
|
|
|
|
if (score === KinkPreference.Favorite)
|
|
return new Score(Scoring.MATCH, `Loves <span>${description}</span>`);
|
|
|
|
return new Score(Scoring.NEUTRAL);
|
|
}
|
|
|
|
private resolveSpeciesScore(): Score {
|
|
const you = this.you;
|
|
const theirAnalysis = this.theirAnalysis;
|
|
const theirSpecies = theirAnalysis.species;
|
|
|
|
if (theirSpecies === null)
|
|
return new Score(Scoring.NEUTRAL);
|
|
|
|
const speciesScore = Matcher.getKinkSpeciesPreference(you, theirSpecies);
|
|
|
|
if (speciesScore !== null) {
|
|
const speciesName = speciesNames[theirSpecies] || `${Species[theirSpecies].toLowerCase()}s`;
|
|
|
|
return Matcher.formatKinkScore(speciesScore, speciesName);
|
|
}
|
|
|
|
if (theirAnalysis.isAnthro) {
|
|
const anthroScore = Matcher.getKinkPreference(you, Kink.AnthroCharacters);
|
|
|
|
if (anthroScore !== null)
|
|
return Matcher.formatKinkScore(anthroScore, 'anthros');
|
|
}
|
|
|
|
if (theirAnalysis.isMammal) {
|
|
const mammalScore = Matcher.getKinkPreference(you, Kink.Mammals);
|
|
|
|
if (mammalScore !== null)
|
|
return Matcher.formatKinkScore(mammalScore, 'mammals');
|
|
}
|
|
|
|
return new Score(Scoring.NEUTRAL);
|
|
}
|
|
|
|
|
|
formatScoring(score: Scoring, description: string): Score {
|
|
let type = '';
|
|
|
|
switch (score) {
|
|
case Scoring.MISMATCH:
|
|
type = 'No';
|
|
break;
|
|
|
|
case Scoring.WEAK_MISMATCH:
|
|
type = 'Hesitant about';
|
|
break;
|
|
|
|
case Scoring.WEAK_MATCH:
|
|
type = 'Likes';
|
|
break;
|
|
|
|
case Scoring.MATCH:
|
|
type = 'Loves';
|
|
break;
|
|
}
|
|
|
|
return new Score(score, `${type} <span>${description}</span>`);
|
|
}
|
|
|
|
private resolveFurryPairingsScore(): Score {
|
|
const you = this.you;
|
|
const theyAreAnthro = this.theirAnalysis.isAnthro;
|
|
const theyAreHuman = this.theirAnalysis.isHuman;
|
|
|
|
const score = theyAreAnthro
|
|
? Matcher.furryLikeabilityScore(you)
|
|
: (theyAreHuman ? Matcher.humanLikeabilityScore(you) : Scoring.NEUTRAL);
|
|
|
|
if (score === Scoring.WEAK_MATCH)
|
|
return new Score(
|
|
score,
|
|
theyAreAnthro
|
|
? 'Prefers <span>humans</span>, ok with anthros'
|
|
: 'Prefers <span>anthros</span>, ok with humans'
|
|
);
|
|
|
|
return this.formatScoring(score, theyAreAnthro ? 'furry pairings' : theyAreHuman ? 'human pairings' : '');
|
|
}
|
|
|
|
static furryLikeabilityScore(c: Character): Scoring {
|
|
const furryPreference = Matcher.getTagValueList(TagId.FurryPreference, c);
|
|
|
|
if (
|
|
(furryPreference === FurryPreference.FursAndHumans) ||
|
|
(furryPreference === FurryPreference.FurriesPreferredHumansOk) ||
|
|
(furryPreference === FurryPreference.FurriesOnly)
|
|
)
|
|
return Scoring.MATCH;
|
|
|
|
if (furryPreference === FurryPreference.HumansPreferredFurriesOk)
|
|
return Scoring.WEAK_MATCH;
|
|
|
|
if (furryPreference === FurryPreference.HumansOnly)
|
|
return Scoring.MISMATCH;
|
|
|
|
return Scoring.NEUTRAL;
|
|
}
|
|
|
|
static humanLikeabilityScore(c: Character): Scoring {
|
|
const humanPreference = Matcher.getTagValueList(TagId.FurryPreference, c);
|
|
|
|
if (
|
|
(humanPreference === FurryPreference.FursAndHumans)
|
|
|| (humanPreference === FurryPreference.HumansPreferredFurriesOk)
|
|
|| (humanPreference === FurryPreference.HumansOnly)
|
|
)
|
|
return Scoring.MATCH;
|
|
|
|
if (humanPreference === FurryPreference.FurriesPreferredHumansOk)
|
|
return Scoring.WEAK_MATCH;
|
|
|
|
if (humanPreference === FurryPreference.FurriesOnly)
|
|
return Scoring.MISMATCH;
|
|
|
|
return Scoring.NEUTRAL;
|
|
}
|
|
|
|
private resolveAgeScore(): Score {
|
|
const you = this.you;
|
|
const theirAge = this.theirAnalysis.age;
|
|
|
|
if (theirAge === null)
|
|
return new Score(Scoring.NEUTRAL);
|
|
|
|
const ageplayScore = Matcher.getKinkPreference(you, Kink.Ageplay);
|
|
const underageScore = Matcher.getKinkPreference(you, Kink.UnderageCharacters);
|
|
|
|
if ((theirAge < 16) && (ageplayScore !== null))
|
|
return Matcher.formatKinkScore(ageplayScore, `ages of ${theirAge}`);
|
|
|
|
if ((theirAge < 16) && (ageplayScore === null))
|
|
return Matcher.formatKinkScore(KinkPreference.No, `ages of ${theirAge}`);
|
|
|
|
if ((theirAge < 18) && (theirAge >= 16) && (underageScore !== null))
|
|
return Matcher.formatKinkScore(underageScore, `ages of ${theirAge}`);
|
|
|
|
const yourAge = this.yourAnalysis.age;
|
|
|
|
if ((yourAge !== null) && (yourAge > 0) && (theirAge > 0) && (yourAge <= 80) && (theirAge <= 80)) {
|
|
const olderCharactersScore = Matcher.getKinkPreference(you, Kink.OlderCharacters);
|
|
const youngerCharactersScore = Matcher.getKinkPreference(you, Kink.YoungerCharacters);
|
|
const ageDifference = Math.abs(yourAge - theirAge);
|
|
|
|
if ((yourAge < theirAge) && (olderCharactersScore !== null) && (ageDifference >= 4))
|
|
return Matcher.formatKinkScore(olderCharactersScore, 'older characters');
|
|
|
|
if ((yourAge > theirAge) && (youngerCharactersScore !== null) && (ageDifference <= 4))
|
|
return Matcher.formatKinkScore(youngerCharactersScore, 'younger characters');
|
|
}
|
|
|
|
return new Score(Scoring.NEUTRAL);
|
|
}
|
|
|
|
private resolveGenderScore(): Score {
|
|
const you = this.you;
|
|
const theirGender = this.theirAnalysis.gender;
|
|
|
|
if (theirGender === null)
|
|
return new Score(Scoring.NEUTRAL);
|
|
|
|
const genderName = `${Gender[theirGender].toLowerCase()}s`;
|
|
const genderKinkScore = Matcher.getKinkGenderPreference(you, theirGender);
|
|
|
|
if (genderKinkScore !== null)
|
|
return Matcher.formatKinkScore(genderKinkScore, genderName);
|
|
|
|
return new Score(Scoring.NEUTRAL);
|
|
}
|
|
|
|
|
|
private resolveSubDomScore(): Score {
|
|
const you = this.you;
|
|
const yourSubDomRole = this.yourAnalysis.subDomRole;
|
|
const theirSubDomRole = this.theirAnalysis.subDomRole;
|
|
const yourRoleReversalPreference = Matcher.getKinkPreference(you, Kink.RoleReversal);
|
|
|
|
if ((!yourSubDomRole) || (!theirSubDomRole))
|
|
return new Score(Scoring.NEUTRAL);
|
|
|
|
if ((yourSubDomRole === SubDomRole.AlwaysDominant) || (yourSubDomRole === SubDomRole.UsuallyDominant)) {
|
|
if (theirSubDomRole === SubDomRole.Switch)
|
|
return new Score(Scoring.WEAK_MATCH, `Likes <span>switches</span>`);
|
|
|
|
if ((theirSubDomRole === SubDomRole.AlwaysSubmissive) || (theirSubDomRole === SubDomRole.UsuallySubmissive))
|
|
return new Score(Scoring.MATCH, `Loves <span>submissives</span>`);
|
|
|
|
if (yourRoleReversalPreference === KinkPreference.Favorite)
|
|
return new Score(Scoring.MATCH, `Loves <span>role reversal</span>`);
|
|
|
|
if (yourRoleReversalPreference === KinkPreference.Yes)
|
|
return new Score(Scoring.MATCH, `Likes <span>role reversal</span>`);
|
|
|
|
if ((yourSubDomRole === SubDomRole.AlwaysDominant) && (theirSubDomRole === SubDomRole.AlwaysDominant))
|
|
return new Score(Scoring.MISMATCH, 'No <span>dominants</span>');
|
|
|
|
return new Score(Scoring.WEAK_MISMATCH, 'Hesitant about <span>dominants</span>');
|
|
}
|
|
|
|
if ((yourSubDomRole === SubDomRole.AlwaysSubmissive) || (yourSubDomRole === SubDomRole.UsuallySubmissive)) {
|
|
if (theirSubDomRole === SubDomRole.Switch)
|
|
return new Score(Scoring.WEAK_MATCH, `Likes <span>switches</span>`);
|
|
|
|
if ((theirSubDomRole === SubDomRole.AlwaysDominant) || (theirSubDomRole === SubDomRole.UsuallyDominant))
|
|
return new Score(Scoring.MATCH, `Loves <span>dominants</span>`);
|
|
|
|
if (yourRoleReversalPreference === KinkPreference.Favorite)
|
|
return new Score(Scoring.MATCH, `Loves <span>role reversal</span>`);
|
|
|
|
if (yourRoleReversalPreference === KinkPreference.Yes)
|
|
return new Score(Scoring.MATCH, `Likes <span>role reversal</span>`);
|
|
|
|
if ((yourSubDomRole === SubDomRole.AlwaysSubmissive) && (theirSubDomRole === SubDomRole.AlwaysSubmissive))
|
|
return new Score(Scoring.MISMATCH, 'No <span>submissives</span>');
|
|
|
|
return new Score(Scoring.WEAK_MISMATCH, 'Hesitant about <span>submissives</span>');
|
|
}
|
|
|
|
// You must be a switch
|
|
if (theirSubDomRole === SubDomRole.Switch)
|
|
return new Score(Scoring.MATCH, `Loves <span>switches</span>`);
|
|
|
|
// if (yourRoleReversalPreference === KinkPreference.Favorite)
|
|
// return new Score(Scoring.MATCH, `Loves <span>role reversal</span>`);
|
|
//
|
|
// if (yourRoleReversalPreference === KinkPreference.Yes)
|
|
// return new Score(Scoring.MATCH, `Likes <span>role reversal</span>`);
|
|
|
|
if ((theirSubDomRole === SubDomRole.AlwaysDominant) || (theirSubDomRole === SubDomRole.UsuallyDominant))
|
|
return new Score(Scoring.MATCH, `Loves <span>dominants</span>`);
|
|
|
|
if ((theirSubDomRole === SubDomRole.AlwaysSubmissive) || (theirSubDomRole === SubDomRole.UsuallySubmissive))
|
|
return new Score(Scoring.MATCH, `Loves <span>submissives</span>`);
|
|
|
|
return new Score(Scoring.NEUTRAL);
|
|
}
|
|
|
|
|
|
static getTagValue(tagId: number, c: Character): CharacterInfotag | undefined {
|
|
return c.infotags[tagId];
|
|
}
|
|
|
|
static getTagValueList(tagId: number, c: Character): number | null {
|
|
const t = Matcher.getTagValue(tagId, c);
|
|
|
|
if ((!t) || (!t.list))
|
|
return null;
|
|
|
|
return t.list;
|
|
}
|
|
|
|
static isCisGender(...genders: Gender[] | null[]): boolean {
|
|
return _.every(genders, (g: Gender) => ((g === Gender.Female) || (g === Gender.Male)));
|
|
}
|
|
|
|
static getKinkPreference(c: Character, kinkId: number): KinkPreference | null {
|
|
if (!(kinkId in c.kinks))
|
|
return null;
|
|
|
|
const kinkVal = c.kinks[kinkId];
|
|
|
|
if (kinkVal === undefined) {
|
|
return null;
|
|
}
|
|
|
|
if (typeof kinkVal === 'string') {
|
|
return kinkMapping[c.kinks[kinkId] as string];
|
|
}
|
|
|
|
const custom = c.customs[kinkVal];
|
|
|
|
if (!custom) {
|
|
return null;
|
|
}
|
|
|
|
return kinkMapping[custom.choice];
|
|
}
|
|
|
|
static getKinkGenderPreference(c: Character, gender: Gender): KinkPreference | null {
|
|
if (!(gender in genderKinkMapping))
|
|
return null;
|
|
|
|
return Matcher.getKinkPreference(c, genderKinkMapping[gender]);
|
|
}
|
|
|
|
static getKinkSpeciesPreference(c: Character, species: Species): KinkPreference | null {
|
|
return Matcher.getKinkPreference(c, species);
|
|
}
|
|
|
|
static has(c: Character, kinkId: Kink): boolean {
|
|
const r = Matcher.getKinkPreference(c, kinkId);
|
|
|
|
return (r !== null);
|
|
}
|
|
|
|
static isMammal(c: Character): boolean | null {
|
|
const species = Matcher.species(c);
|
|
|
|
if (species === null)
|
|
return null;
|
|
|
|
return (mammalSpecies.indexOf(species) >= 0);
|
|
}
|
|
|
|
static isAnthro(c: Character): boolean | null {
|
|
const bodyTypeId = Matcher.getTagValueList(TagId.BodyType, c);
|
|
|
|
if (bodyTypeId === BodyType.Anthro)
|
|
return true;
|
|
|
|
const speciesId = Matcher.species(c);
|
|
|
|
if (!speciesId)
|
|
return null;
|
|
|
|
return (nonAnthroSpecies.indexOf(speciesId) < 0);
|
|
}
|
|
|
|
static isHuman(c: Character): boolean | null {
|
|
const bodyTypeId = Matcher.getTagValueList(TagId.BodyType, c);
|
|
|
|
if (bodyTypeId === BodyType.Human)
|
|
return true;
|
|
|
|
const speciesId = Matcher.species(c);
|
|
|
|
return (speciesId === Species.Human);
|
|
}
|
|
|
|
static species(c: Character): Species | null {
|
|
let foundSpeciesId: Species | null = null;
|
|
let match = '';
|
|
|
|
const mySpecies = Matcher.getTagValue(TagId.Species, c);
|
|
|
|
if ((!mySpecies) || (!mySpecies.string)) {
|
|
return Species.Human; // best guess
|
|
}
|
|
|
|
const finalSpecies = mySpecies.string.toLowerCase();
|
|
|
|
_.each(
|
|
speciesMapping,
|
|
(keywords: string[], speciesId: string) => {
|
|
_.each(
|
|
keywords,
|
|
(k: string) => {
|
|
if ((k.length > match.length) && (finalSpecies.indexOf(k) >= 0)) {
|
|
match = k;
|
|
foundSpeciesId = parseInt(speciesId, 10);
|
|
}
|
|
}
|
|
);
|
|
}
|
|
);
|
|
|
|
return foundSpeciesId;
|
|
}
|
|
|
|
|
|
static strToGender(fchatGenderStr: string | undefined): Gender | null {
|
|
if (fchatGenderStr === undefined) {
|
|
return null;
|
|
}
|
|
|
|
if (fchatGenderStr in fchatGenderMap) {
|
|
return fchatGenderMap[fchatGenderStr];
|
|
}
|
|
|
|
return null;
|
|
}
|
|
}
|