fchat-rising/learn/matcher.ts

906 lines
31 KiB
TypeScript
Raw Normal View History

2020-10-04 21:30:54 +00:00
/* eslint-disable no-null-keyword, max-file-line-count */
2019-06-29 01:37:41 +00:00
import * as _ from 'lodash';
2019-07-07 21:44:32 +00:00
import { Character, CharacterInfotag } from '../interfaces';
2020-10-04 21:30:54 +00:00
import log from 'electron-log'; //tslint:disable-line:match-default-export-name
2020-10-24 20:47:50 +00:00
// tslint:disable-next-line ban-ts-ignore
// @ts-ignore
2020-10-24 19:43:25 +00:00
import anyAscii from 'any-ascii';
2020-10-04 21:30:54 +00:00
import {
BodyType, fchatGenderMap,
FurryPreference,
Gender, genderKinkMapping,
Kink,
kinkMapping,
2020-10-18 22:52:56 +00:00
KinkPreference, likelyHuman, mammalSpecies, nonAnthroSpecies,
2020-10-04 21:30:54 +00:00
Orientation,
2020-10-18 22:52:56 +00:00
Species, SpeciesMap, speciesMapping, SpeciesMappingCache,
2020-10-04 21:30:54 +00:00
speciesNames,
SubDomRole,
TagId
} from './matcher-types';
2019-06-29 01:37:41 +00:00
2019-07-07 01:37:15 +00:00
2019-06-29 01:37:41 +00:00
export interface MatchReport {
2019-06-29 20:59:29 +00:00
you: MatchResult;
them: MatchResult;
2020-10-04 21:30:54 +00:00
youMultiSpecies: boolean;
themMultiSpecies: boolean;
2020-10-24 23:17:31 +00:00
merged: MatchResultScores;
2020-10-04 21:30:54 +00:00
score: Scoring | null;
2019-06-29 20:59:29 +00:00
}
export interface MatchResultCharacterInfo {
species: Species | null;
gender: Gender | null;
orientation: Orientation | null;
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
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;
2019-07-07 21:44:32 +00:00
total: number;
yourAnalysis: CharacterAnalysis;
theirAnalysis: CharacterAnalysis;
2019-06-29 20:59:29 +00:00
}
export enum Scoring {
MATCH = 1,
WEAK_MATCH = 0.5,
NEUTRAL = 0,
2019-06-29 22:24:44 +00:00
WEAK_MISMATCH = -0.5,
2019-06-29 20:59:29 +00:00
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'
};
2020-10-24 23:17:31 +00:00
const scoreIcons: ScoreClassMap = {
[Scoring.MATCH]: 'fas fa-heart',
[Scoring.WEAK_MATCH]: 'fas fa-thumbs-up',
[Scoring.NEUTRAL]: 'fas fa-meh',
[Scoring.WEAK_MISMATCH]: 'fas fa-question-circle',
[Scoring.MISMATCH]: 'fas fa-heart-broken'
};
2019-06-29 20:59:29 +00:00
export class Score {
readonly score: Scoring;
readonly description: string;
2020-10-24 23:17:31 +00:00
readonly shortDesc: string;
2019-06-29 20:59:29 +00:00
2020-10-24 23:17:31 +00:00
constructor(score: Scoring, description: string = '', shortDesc: string = '') {
2019-06-29 20:59:29 +00:00
if ((score !== Scoring.NEUTRAL) && (description === ''))
throw new Error('Description must be provided if score is not neutral');
this.score = score;
this.description = description;
2020-10-24 23:17:31 +00:00
this.shortDesc = shortDesc;
2019-06-29 20:59:29 +00:00
}
getRecommendedClass(): string {
2019-07-07 01:37:15 +00:00
return Score.getClasses(this.score);
}
2020-10-24 23:17:31 +00:00
getRecommendedIcon(): string {
return Score.getIcon(this.score);
}
2019-07-07 01:37:15 +00:00
static getClasses(score: Scoring): string {
return scoreClasses[score];
2019-06-29 20:59:29 +00:00
}
2020-10-24 23:17:31 +00:00
static getIcon(score: Scoring): string {
return scoreIcons[score];
}
2019-06-29 20:59:29 +00:00
}
2020-10-04 21:30:54 +00:00
export interface CharacterAnalysisVariation {
readonly character: Character;
readonly analysis: CharacterAnalysis;
}
2019-07-07 21:44:32 +00:00
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;
2019-10-27 18:58:41 +00:00
readonly subDomRole: SubDomRole | null;
2019-07-07 21:44:32 +00:00
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);
2019-10-27 18:58:41 +00:00
this.subDomRole = Matcher.getTagValueList(TagId.SubDomRole, c);
2019-07-07 21:44:32 +00:00
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);
}
}
2019-06-29 20:59:29 +00:00
/**
* 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
*/
2019-06-29 01:37:41 +00:00
export class Matcher {
2019-07-07 21:44:32 +00:00
readonly you: Character;
readonly them: Character;
readonly yourAnalysis: CharacterAnalysis;
readonly theirAnalysis: CharacterAnalysis;
2019-06-29 01:37:41 +00:00
2019-07-07 21:44:32 +00:00
constructor(you: Character, them: Character, yourAnalysis?: CharacterAnalysis, theirAnalysis?: CharacterAnalysis) {
2019-06-29 01:37:41 +00:00
this.you = you;
this.them = them;
2019-07-07 21:44:32 +00:00
this.yourAnalysis = yourAnalysis || new CharacterAnalysis(you);
this.theirAnalysis = theirAnalysis || new CharacterAnalysis(them);
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
static generateReport(you: Character, them: Character): MatchReport {
2019-07-07 21:44:32 +00:00
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);
2019-06-29 20:59:29 +00:00
2020-10-24 23:17:31 +00:00
const youThemMatch = youThem.match();
const themYouMatch = themYou.match();
2020-10-04 21:30:54 +00:00
const report: MatchReport = {
2020-10-24 23:17:31 +00:00
you: youThemMatch,
them: themYouMatch,
2020-10-04 21:30:54 +00:00
youMultiSpecies: false,
themMultiSpecies: false,
2020-10-24 23:17:31 +00:00
merged: Matcher.mergeResults(youThemMatch, themYouMatch),
2020-10-04 21:30:54 +00:00
score: null
2019-06-29 20:59:29 +00:00
};
2020-10-04 21:30:54 +00:00
report.score = Matcher.calculateReportScore(report);
return report;
}
static identifyBestMatchReport(you: Character, them: Character): MatchReport {
const reportStartTime = Date.now();
const yourCharacterAnalyses = Matcher.generateAnalysisVariations(you);
const theirCharacterAnalyses = Matcher.generateAnalysisVariations(them);
let bestScore = null;
2020-10-04 21:48:48 +00:00
let bestScoreLevelCount = -10000;
2020-10-04 21:30:54 +00:00
let bestReport: MatchReport;
for(const yourAnalysis of yourCharacterAnalyses) {
for (const theirAnalysis of theirCharacterAnalyses) {
const youThem = new Matcher(yourAnalysis.character, theirAnalysis.character, yourAnalysis.analysis, theirAnalysis.analysis);
const themYou = new Matcher(theirAnalysis.character, yourAnalysis.character, theirAnalysis.analysis, yourAnalysis.analysis);
2020-10-24 23:17:31 +00:00
const youThemMatch = youThem.match();
const themYouMatch = themYou.match();
2020-10-04 21:30:54 +00:00
const report: MatchReport = {
2020-10-24 23:17:31 +00:00
you: youThemMatch,
them: themYouMatch,
2020-10-04 21:30:54 +00:00
youMultiSpecies: (yourCharacterAnalyses.length > 1),
themMultiSpecies: (theirCharacterAnalyses.length > 1),
2020-10-24 23:17:31 +00:00
merged: Matcher.mergeResults(youThemMatch, themYouMatch),
2020-10-04 21:30:54 +00:00
score: null
};
report.score = Matcher.calculateReportScore(report);
const scoreLevelCount = Matcher.countScoresAtLevel(report, report.score);
if (
(bestScore === null)
|| (
(report.score !== null)
&& (report.score >= bestScore)
&& (scoreLevelCount !== null)
2020-10-04 21:48:48 +00:00
&& (report.score * scoreLevelCount > bestScoreLevelCount)
2020-10-04 21:30:54 +00:00
)
) {
bestScore = report.score;
2020-10-04 21:48:48 +00:00
bestScoreLevelCount = ((scoreLevelCount !== null) && (report.score !== null)) ? report.score * scoreLevelCount : -1000;
2020-10-04 21:30:54 +00:00
bestReport = report;
}
}
}
log.debug('report.identify.best', {buildTime: Date.now() - reportStartTime});
return bestReport!;
}
2020-10-24 23:17:31 +00:00
// tslint:disable-next-line
private static mergeResultScores(scores: MatchResultScores, results: MatchResultScores): void {
_.each(scores, (v: Score, k: any) => {
if (
// tslint:disable-next-line no-unsafe-any
((!(k in results)) || (v.score < results[k].score))
&& (v.score !== Scoring.NEUTRAL)
) {
results[k] = v;
}
}
);
}
static mergeResults(you: MatchResult, them: MatchResult): MatchResultScores {
const results: MatchResultScores = {} as any;
Matcher.mergeResultScores(you.scores, results);
Matcher.mergeResultScores(them.scores, results);
return results;
}
2020-10-04 21:30:54 +00:00
static generateAnalysisVariations(c: Character): CharacterAnalysisVariation[] {
const speciesOptions = Matcher.getAllSpeciesAsStr(c);
if (speciesOptions.length === 0) {
speciesOptions.push('');
}
return _.map(
speciesOptions,
(species) => {
const nc = _.cloneDeep(c);
nc.infotags[TagId.Species] = { string: species };
return { character: nc, analysis: new CharacterAnalysis(nc) };
}
);
}
static countScoresAtLevel(m: MatchReport, scoreLevel: Scoring | null): number | null {
if (scoreLevel === null) {
return null;
}
const yourScores = _.values(m.you.scores);
const theirScores = _.values(m.them.scores);
return _.reduce(
_.concat(yourScores, theirScores),
(accum: number, score: Score) => accum + (score.score === scoreLevel ? 1 : 0),
0
);
}
static calculateReportScore(m: MatchReport): Scoring | null {
const yourScores = _.values(m.you.scores);
const theirScores = _.values(m.them.scores);
const finalScore = _.reduce(
_.concat(yourScores, theirScores),
(accum: Scoring | null, score: Score) => {
if (accum === null) {
return (score.score !== Scoring.NEUTRAL) ? score.score : null;
}
return (score.score === Scoring.NEUTRAL) ? accum : Math.min(accum, score.score);
},
null
);
if ((finalScore !== null) && (finalScore > 0)) {
// Manage edge cases where high score may not be ideal
// Nothing to score
if ((yourScores.length === 0) || (theirScores.length === 0)) {
// can't know
return Scoring.NEUTRAL;
}
// Only neutral scores given
if (
(_.every(yourScores, (n: Scoring) => n === Scoring.NEUTRAL)) ||
(_.every(theirScores, (n: Scoring) => n === Scoring.NEUTRAL))
) {
return Scoring.NEUTRAL;
}
}
// console.log('Profile score', c.character.name, score, m.you.total, m.them.total,
// m.you.total + m.them.total, m.you.total * m.them.total);
return (finalScore === null) ? Scoring.NEUTRAL : finalScore;
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
match(): MatchResult {
2019-07-07 21:44:32 +00:00
const data: MatchResult = {
2019-06-29 20:59:29 +00:00
you: this.you,
them: this.them,
2019-07-07 21:44:32 +00:00
yourAnalysis: this.yourAnalysis,
theirAnalysis: this.theirAnalysis,
2019-07-07 01:37:15 +00:00
total: 0,
2019-06-29 20:59:29 +00:00
scores: {
[TagId.Orientation]: this.resolveOrientationScore(),
[TagId.Gender]: this.resolveGenderScore(),
[TagId.Age]: this.resolveAgeScore(),
[TagId.FurryPreference]: this.resolveFurryPairingsScore(),
2019-10-27 18:58:41 +00:00
[TagId.Species]: this.resolveSpeciesScore(),
[TagId.SubDomRole]: this.resolveSubDomScore()
2019-06-29 20:59:29 +00:00
},
info: {
species: Matcher.species(this.you),
gender: Matcher.getTagValueList(TagId.Gender, this.you),
2019-07-06 16:49:19 +00:00
orientation: Matcher.getTagValueList(TagId.Orientation, this.you)
2019-06-29 20:59:29 +00:00
}
2019-07-07 01:37:15 +00:00
};
data.total = _.reduce(
data.scores,
(accum: number, s: Score) => (accum + s.score),
0
);
return data;
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
private resolveOrientationScore(): Score {
// Question: If someone identifies themselves as 'straight cuntboy', how should they be matched? like a straight female?
2019-07-07 21:44:32 +00:00
return Matcher.scoreOrientationByGender(this.yourAnalysis.gender, this.yourAnalysis.orientation, this.theirAnalysis.gender);
2019-07-07 01:37:15 +00:00
}
2019-07-07 21:44:32 +00:00
static scoreOrientationByGender(yourGender: Gender | null, yourOrientation: Orientation | null, theirGender: Gender | null): Score {
if ((yourGender === null) || (theirGender === null) || (yourOrientation === null))
return new Score(Scoring.NEUTRAL);
2019-07-07 01:37:15 +00:00
2019-06-29 20:59:29 +00:00
// CIS
2019-07-06 16:49:19 +00:00
// tslint:disable-next-line curly
2019-07-07 21:44:32 +00:00
if (Matcher.isCisGender(yourGender)) {
2019-06-29 20:59:29 +00:00
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>');
}
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
return new Score(Scoring.NEUTRAL);
2019-06-29 01:37:41 +00:00
}
2019-07-06 16:49:19 +00:00
static formatKinkScore(score: KinkPreference, description: string): Score {
2019-06-29 20:59:29 +00:00
if (score === KinkPreference.No)
return new Score(Scoring.MISMATCH, `No <span>${description}</span>`);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (score === KinkPreference.Maybe)
2020-06-30 16:51:56 +00:00
return new Score(Scoring.WEAK_MISMATCH, `Hesitant about <span>${description}</span>`);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (score === KinkPreference.Yes)
return new Score(Scoring.WEAK_MATCH, `Likes <span>${description}</span>`);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (score === KinkPreference.Favorite)
return new Score(Scoring.MATCH, `Loves <span>${description}</span>`);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
return new Score(Scoring.NEUTRAL);
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
private resolveSpeciesScore(): Score {
const you = this.you;
2019-07-07 21:44:32 +00:00
const theirAnalysis = this.theirAnalysis;
const theirSpecies = theirAnalysis.species;
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (theirSpecies === null)
return new Score(Scoring.NEUTRAL);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
const speciesScore = Matcher.getKinkSpeciesPreference(you, theirSpecies);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (speciesScore !== null) {
2020-10-18 22:52:56 +00:00
// console.log(this.them.name, speciesScore, theirSpecies);
2019-06-29 20:59:29 +00:00
const speciesName = speciesNames[theirSpecies] || `${Species[theirSpecies].toLowerCase()}s`;
2019-06-29 01:37:41 +00:00
2019-07-06 16:49:19 +00:00
return Matcher.formatKinkScore(speciesScore, speciesName);
2019-06-29 01:37:41 +00:00
}
2019-07-07 21:44:32 +00:00
if (theirAnalysis.isAnthro) {
2019-07-02 01:03:28 +00:00
const anthroScore = Matcher.getKinkPreference(you, Kink.AnthroCharacters);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (anthroScore !== null)
2019-07-06 16:49:19 +00:00
return Matcher.formatKinkScore(anthroScore, 'anthros');
2019-06-29 01:37:41 +00:00
}
2019-07-07 21:44:32 +00:00
if (theirAnalysis.isMammal) {
2019-07-02 01:03:28 +00:00
const mammalScore = Matcher.getKinkPreference(you, Kink.Mammals);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (mammalScore !== null)
2019-07-06 16:49:19 +00:00
return Matcher.formatKinkScore(mammalScore, 'mammals');
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
return new Score(Scoring.NEUTRAL);
2019-06-29 01:37:41 +00:00
}
2019-07-07 21:44:32 +00:00
2019-06-29 20:59:29 +00:00
formatScoring(score: Scoring, description: string): Score {
let type = '';
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
switch (score) {
case Scoring.MISMATCH:
type = 'No';
break;
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
case Scoring.WEAK_MISMATCH:
2020-06-30 16:51:56 +00:00
type = 'Hesitant about';
2019-06-29 20:59:29 +00:00
break;
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
case Scoring.WEAK_MATCH:
type = 'Likes';
break;
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
case Scoring.MATCH:
type = 'Loves';
break;
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
return new Score(score, `${type} <span>${description}</span>`);
}
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
private resolveFurryPairingsScore(): Score {
const you = this.you;
2019-07-07 21:44:32 +00:00
const theyAreAnthro = this.theirAnalysis.isAnthro;
const theyAreHuman = this.theirAnalysis.isHuman;
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
const score = theyAreAnthro
? Matcher.furryLikeabilityScore(you)
: (theyAreHuman ? Matcher.humanLikeabilityScore(you) : Scoring.NEUTRAL);
2019-06-29 01:37:41 +00:00
2019-06-29 22:24:44 +00:00
if (score === Scoring.WEAK_MATCH)
2019-07-06 16:49:19 +00:00
return new Score(
score,
theyAreAnthro
2020-03-15 02:31:28 +00:00
? 'Prefers <span>humans</span>, ok with anthros'
: 'Prefers <span>anthros</span>, ok with humans'
2019-07-06 16:49:19 +00:00
);
2019-06-29 22:24:44 +00:00
2019-06-29 20:59:29 +00:00
return this.formatScoring(score, theyAreAnthro ? 'furry pairings' : theyAreHuman ? 'human pairings' : '');
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
static furryLikeabilityScore(c: Character): Scoring {
const furryPreference = Matcher.getTagValueList(TagId.FurryPreference, c);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (
(furryPreference === FurryPreference.FursAndHumans) ||
(furryPreference === FurryPreference.FurriesPreferredHumansOk) ||
(furryPreference === FurryPreference.FurriesOnly)
)
return Scoring.MATCH;
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (furryPreference === FurryPreference.HumansPreferredFurriesOk)
return Scoring.WEAK_MATCH;
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (furryPreference === FurryPreference.HumansOnly)
return Scoring.MISMATCH;
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
return Scoring.NEUTRAL;
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
static humanLikeabilityScore(c: Character): Scoring {
const humanPreference = Matcher.getTagValueList(TagId.FurryPreference, c);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (
(humanPreference === FurryPreference.FursAndHumans)
|| (humanPreference === FurryPreference.HumansPreferredFurriesOk)
|| (humanPreference === FurryPreference.HumansOnly)
)
return Scoring.MATCH;
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (humanPreference === FurryPreference.FurriesPreferredHumansOk)
return Scoring.WEAK_MATCH;
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (humanPreference === FurryPreference.FurriesOnly)
return Scoring.MISMATCH;
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
return Scoring.NEUTRAL;
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
private resolveAgeScore(): Score {
2019-06-29 01:37:41 +00:00
const you = this.you;
2019-07-07 21:44:32 +00:00
const theirAge = this.theirAnalysis.age;
2019-06-29 01:37:41 +00:00
2019-07-07 21:44:32 +00:00
if (theirAge === null)
2019-06-29 20:59:29 +00:00
return new Score(Scoring.NEUTRAL);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
const ageplayScore = Matcher.getKinkPreference(you, Kink.Ageplay);
const underageScore = Matcher.getKinkPreference(you, Kink.UnderageCharacters);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if ((theirAge < 16) && (ageplayScore !== null))
2019-07-06 16:49:19 +00:00
return Matcher.formatKinkScore(ageplayScore, `ages of ${theirAge}`);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if ((theirAge < 16) && (ageplayScore === null))
2019-07-06 16:49:19 +00:00
return Matcher.formatKinkScore(KinkPreference.No, `ages of ${theirAge}`);
2019-06-29 01:37:41 +00:00
2019-09-24 19:53:43 +00:00
if ((theirAge < 18) && (theirAge >= 16) && (underageScore !== null))
2019-07-06 16:49:19 +00:00
return Matcher.formatKinkScore(underageScore, `ages of ${theirAge}`);
2019-06-29 01:37:41 +00:00
2019-07-07 21:44:32 +00:00
const yourAge = this.yourAnalysis.age;
2019-07-11 00:36:11 +00:00
if ((yourAge !== null) && (yourAge > 0) && (theirAge > 0) && (yourAge <= 80) && (theirAge <= 80)) {
2019-06-29 20:59:29 +00:00
const olderCharactersScore = Matcher.getKinkPreference(you, Kink.OlderCharacters);
const youngerCharactersScore = Matcher.getKinkPreference(you, Kink.YoungerCharacters);
2019-07-11 00:38:09 +00:00
const ageDifference = Math.abs(yourAge - theirAge);
2019-06-29 01:37:41 +00:00
2020-07-01 14:33:29 +00:00
if ((yourAge < theirAge) && (olderCharactersScore !== null) && (ageDifference >= 8))
2019-07-06 16:49:19 +00:00
return Matcher.formatKinkScore(olderCharactersScore, 'older characters');
2019-06-29 01:37:41 +00:00
2020-07-01 14:33:29 +00:00
if ((yourAge > theirAge) && (youngerCharactersScore !== null) && (ageDifference >= 8))
2019-07-06 16:49:19 +00:00
return Matcher.formatKinkScore(youngerCharactersScore, 'younger characters');
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
return new Score(Scoring.NEUTRAL);
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
private resolveGenderScore(): Score {
const you = this.you;
2019-07-07 21:44:32 +00:00
const theirGender = this.theirAnalysis.gender;
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (theirGender === null)
return new Score(Scoring.NEUTRAL);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
const genderName = `${Gender[theirGender].toLowerCase()}s`;
const genderKinkScore = Matcher.getKinkGenderPreference(you, theirGender);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (genderKinkScore !== null)
2019-07-06 16:49:19 +00:00
return Matcher.formatKinkScore(genderKinkScore, genderName);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
return new Score(Scoring.NEUTRAL);
2019-06-29 01:37:41 +00:00
}
2019-10-27 18:58:41 +00:00
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>');
2020-06-30 16:51:56 +00:00
return new Score(Scoring.WEAK_MISMATCH, 'Hesitant about <span>dominants</span>');
2020-03-15 14:02:31 +00:00
}
if ((yourSubDomRole === SubDomRole.AlwaysSubmissive) || (yourSubDomRole === SubDomRole.UsuallySubmissive)) {
2019-10-27 18:58:41 +00:00
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>');
2020-06-30 16:51:56 +00:00
return new Score(Scoring.WEAK_MISMATCH, 'Hesitant about <span>submissives</span>');
2019-10-27 18:58:41 +00:00
}
// 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);
}
2019-06-29 20:59:29 +00:00
static getTagValue(tagId: number, c: Character): CharacterInfotag | undefined {
return c.infotags[tagId];
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
static getTagValueList(tagId: number, c: Character): number | null {
2020-03-15 14:02:31 +00:00
const t = Matcher.getTagValue(tagId, c);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if ((!t) || (!t.list))
2019-06-29 01:37:41 +00:00
return null;
2019-06-29 20:59:29 +00:00
return t.list;
2019-06-29 01:37:41 +00:00
}
2019-07-07 21:44:32 +00:00
static isCisGender(...genders: Gender[] | null[]): boolean {
2019-06-29 20:59:29 +00:00
return _.every(genders, (g: Gender) => ((g === Gender.Female) || (g === Gender.Male)));
2019-06-29 01:37:41 +00:00
}
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];
2019-06-29 01:37:41 +00:00
}
static getKinkGenderPreference(c: Character, gender: Gender): KinkPreference | null {
if (!(gender in genderKinkMapping))
return null;
2020-03-15 14:02:31 +00:00
return Matcher.getKinkPreference(c, genderKinkMapping[gender]);
2019-06-29 01:37:41 +00:00
}
static getKinkSpeciesPreference(c: Character, species: Species): KinkPreference | null {
2020-03-15 14:02:31 +00:00
return Matcher.getKinkPreference(c, species);
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
static has(c: Character, kinkId: Kink): boolean {
2019-06-29 01:37:41 +00:00
const r = Matcher.getKinkPreference(c, kinkId);
2019-06-29 20:59:29 +00:00
return (r !== null);
2019-06-29 01:37:41 +00:00
}
2019-06-29 20:59:29 +00:00
static isMammal(c: Character): boolean | null {
const species = Matcher.species(c);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (species === null)
return null;
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
return (mammalSpecies.indexOf(species) >= 0);
2019-06-29 01:37:41 +00:00
}
static isAnthro(c: Character): boolean | null {
2020-03-15 14:02:31 +00:00
const bodyTypeId = Matcher.getTagValueList(TagId.BodyType, c);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (bodyTypeId === BodyType.Anthro)
2019-06-29 01:37:41 +00:00
return true;
2020-03-15 14:02:31 +00:00
const speciesId = Matcher.species(c);
2019-06-29 01:37:41 +00:00
if (!speciesId)
return null;
2019-06-29 20:59:29 +00:00
return (nonAnthroSpecies.indexOf(speciesId) < 0);
2019-06-29 01:37:41 +00:00
}
static isHuman(c: Character): boolean | null {
2020-03-15 14:02:31 +00:00
const bodyTypeId = Matcher.getTagValueList(TagId.BodyType, c);
2019-06-29 01:37:41 +00:00
2019-06-29 20:59:29 +00:00
if (bodyTypeId === BodyType.Human)
2019-06-29 01:37:41 +00:00
return true;
2020-03-15 14:02:31 +00:00
const speciesId = Matcher.species(c);
2019-06-29 01:37:41 +00:00
return (speciesId === Species.Human);
}
static species(c: Character): Species | null {
2020-03-15 14:02:31 +00:00
const mySpecies = Matcher.getTagValue(TagId.Species, c);
2019-06-29 01:37:41 +00:00
2020-04-11 17:46:57 +00:00
if ((!mySpecies) || (!mySpecies.string)) {
2019-06-29 01:37:41 +00:00
return Species.Human; // best guess
2020-04-11 17:46:57 +00:00
}
2019-06-29 01:37:41 +00:00
2020-10-18 22:52:56 +00:00
const s = Matcher.getMappedSpecies(mySpecies.string);
if (!s) {
2020-10-30 22:47:33 +00:00
log.debug('matcher.species.unknown', { character: c.name, species: mySpecies.string });
2020-10-18 22:52:56 +00:00
}
return s;
2020-10-04 21:30:54 +00:00
}
2020-10-18 22:52:56 +00:00
static generateSpeciesMappingCache(mapping: SpeciesMap): SpeciesMappingCache {
return _.mapValues(
mapping,
(keywords: string[]) => _.map(
keywords,
(keyword: string) => {
const keywordPlural = `${keyword}s`; // this is weak: elf -> elves doesn't occur
return {
keyword,
regexp: RegExp(`(^|\\b)(${keyword}|${keywordPlural})($|\\b)`)
};
}
)
);
}
private static speciesMappingCache?: SpeciesMappingCache;
private static likelyHumanCache?: SpeciesMappingCache;
2020-10-24 19:43:25 +00:00
private static matchMappedSpecies(species: string, mapping: SpeciesMappingCache, skipAscii: boolean = false): Species | null {
2020-10-04 21:30:54 +00:00
let foundSpeciesId: Species | null = null;
let match = '';
2020-10-24 19:43:25 +00:00
const finalSpecies = (skipAscii ? species : anyAscii(species)).toLowerCase().trim();
2019-06-29 01:37:41 +00:00
_.each(
2020-10-18 22:52:56 +00:00
mapping,
(matchers, speciesId: string) => {
2019-06-29 01:37:41 +00:00
_.each(
2020-10-18 22:52:56 +00:00
matchers,
(matcher) => {
// finalSpecies.indexOf(k) >= 0)
if ((matcher.keyword.length > match.length) && (matcher.regexp.test(finalSpecies))) {
match = matcher.keyword;
2019-07-06 16:49:19 +00:00
foundSpeciesId = parseInt(speciesId, 10);
2019-06-29 01:37:41 +00:00
}
}
);
}
);
2019-09-24 19:53:43 +00:00
return foundSpeciesId;
2019-06-29 01:37:41 +00:00
}
2019-07-07 01:37:15 +00:00
2020-10-18 22:52:56 +00:00
static getMappedSpecies(species: string): Species | null {
if (!Matcher.speciesMappingCache) {
Matcher.speciesMappingCache = Matcher.generateSpeciesMappingCache(speciesMapping);
}
if (!Matcher.likelyHumanCache) {
Matcher.likelyHumanCache = Matcher.generateSpeciesMappingCache(likelyHuman);
}
return Matcher.matchMappedSpecies(species, Matcher.speciesMappingCache)
2020-10-24 19:43:25 +00:00
|| Matcher.matchMappedSpecies(species, Matcher.speciesMappingCache, true)
|| Matcher.matchMappedSpecies(species, Matcher.likelyHumanCache)
|| Matcher.matchMappedSpecies(species, Matcher.likelyHumanCache, true);
2020-10-18 22:52:56 +00:00
}
2020-10-04 21:30:54 +00:00
static getAllSpecies(c: Character): Species[] {
const species = Matcher.getAllSpeciesAsStr(c);
return _.filter(_.map(species, (s) => Matcher.getMappedSpecies(s)), (s) => (s !== null)) as Species[];
}
static getAllSpeciesAsStr(c: Character): string[] {
const mySpecies = Matcher.getTagValue(TagId.Species, c);
if ((!mySpecies) || (!mySpecies.string)) {
return [];
}
2020-10-18 22:52:56 +00:00
const speciesStr = mySpecies.string.toLowerCase().replace(/optionally|alternatively/g, ',')
.replace(/[)(]/g, ' ').trim();
2020-10-04 21:30:54 +00:00
const matches = speciesStr.split(/[,]? or |,/);
return _.filter(_.map(matches, (m) => m.toLowerCase().trim()), (m) => (m !== ''));
}
2019-07-07 01:37:15 +00:00
static strToGender(fchatGenderStr: string | undefined): Gender | null {
if (fchatGenderStr === undefined) {
return null;
}
if (fchatGenderStr in fchatGenderMap) {
return fchatGenderMap[fchatGenderStr];
}
return null;
}
2019-06-29 01:37:41 +00:00
}