diff --git a/chat/UrlTagView.vue b/bbcode/UrlTagView.vue
similarity index 96%
rename from chat/UrlTagView.vue
rename to bbcode/UrlTagView.vue
index 8f2e693..0c65db4 100644
--- a/chat/UrlTagView.vue
+++ b/bbcode/UrlTagView.vue
@@ -20,7 +20,7 @@
@@ -420,4 +441,27 @@
}
}
}
+
+
+
+ .infotag {
+ &.match {
+ background-color: green;
+ }
+
+ &.mismatch {
+ background-color: red;
+ }
+
+
+ &.weakMatch {
+ background-color: rgba(0, 162, 0, 0.6);
+ }
+
+
+ &.weakMismatch {
+ background-color: rgba(255, 96, 0, 0.6);
+ }
+
+ }
\ No newline at end of file
diff --git a/site/character_page/infotag.vue b/site/character_page/infotag.vue
index f6bfebb..0f65845 100644
--- a/site/character_page/infotag.vue
+++ b/site/character_page/infotag.vue
@@ -10,9 +10,10 @@
import {Component, Prop} from '@f-list/vue-ts';
import Vue from 'vue';
import {formatContactLink, formatContactValue} from './contact_utils';
- import { Character, DisplayInfotag } from './interfaces';
+ import { DisplayInfotag } from './interfaces';
// import { Character as CharacterInfo } from '../../interfaces';
import {Store} from './data_store';
+ import { MatchReport, TagId } from './matcher';
@Component
@@ -21,40 +22,34 @@
private readonly infotag!: DisplayInfotag;
@Prop({required: true})
- private readonly selfCharacter!: Character;
+ private readonly characterMatch!: MatchReport;
get tagClasses() {
- const styles = {
+ const styles: any = {
infotag: true,
};
- //console.log('TAG', this.label, this.value, this.infotag);
+ console.log(`Infotag ${this.infotag.id}: ${this.label}`);
- return this.getCharacterCompatibilityStyles(styles, this.selfCharacter);
- }
+ if ((this.characterMatch) && (this.infotag.id in this.characterMatch)) {
+ const n = this.characterMatch[this.infotag.id];
+ console.log(`Found match [${this.infotag.id} === ${TagId[this.infotag.id]}]: ${n}`);
- getCharacterCompatibilityStyles(styles: any, a: Character) {
- if (a.character.name) {
- styles.infotag = true;
+ if (n >= 1)
+ styles.match = true;
+ else if(n >= 0.5)
+ styles.weakMatch = true;
+ else if(n === 0)
+ styles.neutral = true;
+ else if(n <= -1)
+ styles.mismatch = true;
+ else if(n <= -0.5)
+ styles.weakMismatch = true;
}
- // const c: CharacterInfo = this.selfCharacter.character;
- // const t = this.infotag;
-
- /* console.log(this.label, this.value, this.infotag.id, this.infotag, c.infotags);
-
- switch (t.id) {
- case InfotagView.TAGID_ORIENTATION:
- break;
-
- default:
- // do nothing;
- break;
- }*/
-
return styles;
}
diff --git a/site/character_page/infotags.vue b/site/character_page/infotags.vue
index d8d18a1..00f111b 100644
--- a/site/character_page/infotags.vue
+++ b/site/character_page/infotags.vue
@@ -3,7 +3,7 @@
@@ -16,6 +16,7 @@
import {Character, CONTACT_GROUP_ID, DisplayInfotag} from './interfaces';
import InfotagView from './infotag.vue';
+ import { MatchReport } from './matcher';
interface DisplayInfotagGroup {
id: number
@@ -31,7 +32,7 @@
@Prop({required: true})
private readonly character!: Character;
@Prop({required: true})
- readonly selfCharacter!: Character;
+ readonly characterMatch!: MatchReport;
get groupedInfotags(): DisplayInfotagGroup[] {
const groups = Store.kinks.infotag_groups;
diff --git a/site/character_page/matcher.ts b/site/character_page/matcher.ts
new file mode 100644
index 0000000..0f2a213
--- /dev/null
+++ b/site/character_page/matcher.ts
@@ -0,0 +1,676 @@
+import * as _ from 'lodash';
+import { Character } from '../../interfaces';
+
+export enum TagId {
+ Age = 1,
+ Orientation = 2,
+ Gender = 3,
+ Build = 13,
+ FurryPreference = 29,
+ BdsmRole = 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 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,
+ No = -1
+}
+
+type ScoringCallback = (you: Character, them: Character) => number;
+
+interface CompatibilityCollection {
+ [key: number]: ScoringCallback;
+}
+
+const orientationCompatibility: CompatibilityCollection = {
+ [Orientation.Straight]: (you: Character, them: Character) => Matcher.isCis(you, them) ? (Matcher.isSameSexCis(you, them) ? -1 : 1) : 0,
+ [Orientation.Gay]: (you: Character, them: Character) => Matcher.isCis(you, them) ? (Matcher.isSameSexCis(you, them) ? 1 : -1) : 0,
+ [Orientation.Bisexual]: (you: Character) => Matcher.isGenderedCis(you) ? 1 : 0,
+ [Orientation.Asexual]: () => 0,
+ [Orientation.Unsure]: () => 0,
+ [Orientation.BiMalePreference]: (you: Character, them: Character) => Matcher.isCis(you, them) ? (Matcher.isMaleCis(you) ? 1 : 0.5) : 0,
+ [Orientation.BiFemalePreference]: (you: Character, them: Character) => Matcher.isCis(you, them) ? (Matcher.isFemaleCis(you) ? 1 : 0.5) : 0,
+ [Orientation.Pansexual]: () => 1,
+ [Orientation.BiCurious]: (you: Character, them: Character) => Matcher.isCis(you, them) ? (Matcher.isSameSexCis(you, them) ? 0.5 : Matcher.isGenderedCis(you) ? 1 : 0) : 0
+};
+
+enum Kink {
+ Females = 554,
+ MaleHerms = 552,
+ Males = 553,
+ Transgenders = 551,
+ Herms = 132,
+ Shemales = 356,
+ Cuntboys = 231,
+
+ OlderCharacters = 109,
+ YoungerCharacters = 197,
+ Ageplay = 196,
+ UnderageCharacters = 207,
+
+ AnthroCharacters = 587,
+ Humans = 609
+}
+
+
+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 chareacters', === human
+ // if no species and dislike 'antho characters' === human
+
+ enum Species {
+ Human = 609,
+ Equine = 236,
+ Feline = 212,
+ Canine = 226,
+ Vulpine = 213,
+ Avian = 215,
+ Amphibian = 223,
+ Cervine = 227,
+ Insect = 237,
+ Lapine = 214,
+ Musteline = 328,
+ Dragon = 228,
+ Procyon = 325,
+ Rodent = 283,
+ Ursine = 326,
+ MarineMammal,
+ Primate = 613,
+ Elf = 611,
+ Orc = 615,
+ Fish = 608,
+ Reptile = 225,
+ Anthro = 587,
+ Minotaur = 121212
+ }
+
+const nonAnthroSpecies = [Species.Human, Species.Elf, Species.Orc];
+
+// const mammalSpecies = [Species.Human, Species.Equine, Species.Feline, Species.Canine, Species.Vulpine, Species.Cervine, Species.Lapine, Species.Musteline, Species.Rodent, Species.Ursine, Species.MarineMammal, Species.Primate, Species.Elf, Species.Orc, Species.Anthro, Species.Minotaur];
+
+interface SpeciesMap {
+ [key: number]: string[]
+}
+
+const speciesMapping: SpeciesMap = {
+ [Species.Human]: ['human', 'humanoid', 'angel', 'android'],
+ [Species.Equine]: ['horse', 'stallion', 'mare', 'filly', 'equine', 'shire', 'donkey', 'mule', 'zebra', 'centaur', 'pony' ],
+ [Species.Feline]: ['cat', 'kitten', 'catgirl', 'neko', 'tiger', 'puma', 'lion', 'lioness', 'tigress', 'feline', 'jaguar', 'cheetah', 'lynx', 'leopard'],
+ [Species.Canine]: ['dog', 'wolf', 'dingo', 'coyote', 'jackal', 'canine', 'doberman', 'husky'],
+ [Species.Vulpine]: ['fox', 'fennec', 'kitsune', 'vulpine', 'vixen'],
+ [Species.Avian]: ['bird', 'gryphon', 'phoenix', 'roc', 'chimera', 'avian'],
+ [Species.Amphibian]: ['salamander', 'frog', 'toad', 'newt'],
+ [Species.Cervine]: ['deer', 'elk', 'moose'],
+ [Species.Insect]: ['bee', 'wasp', 'spider', 'scorpion', 'ant', 'insect'],
+ [Species.Lapine]: ['bunny', 'rabbit', 'hare', 'lapine'],
+ [Species.Dragon]: ['dragon', 'drake', 'wyvern'],
+ [Species.Musteline]: ['mink', 'ferret', 'weasel', 'stoat', 'otter', 'wolverine', 'marten'],
+ [Species.Procyon]: ['raccoon', 'coatimund', 'longtail'],
+ [Species.Rodent]: ['rat', 'mouse', 'chipmunk', 'squirrel', 'rodent'],
+ [Species.Ursine]: ['bear', 'panda', 'black bear', 'brown bear', 'polar bear'],
+ [Species.MarineMammal]: ['whale', 'killer whale', 'dolphin'],
+ [Species.Primate]: ['monkey', 'ape', 'chimp', 'chimpanzee', 'gorilla'],
+ [Species.Elf]: ['elf'],
+ [Species.Fish]: ['fish', 'shark', 'great white'],
+ [Species.Orc]: ['orc'],
+ [Species.Reptile]: ['chameleon', 'anole', 'alligator', 'snake', 'crocodile', 'lizard'],
+ [Species.Anthro]: ['anthro', 'anthropomorphic'],
+ [Species.Minotaur]: ['minotaur']
+}
+
+interface KinkPreferenceMap {
+ [key: string]: KinkPreference;
+}
+
+const kinkMapping: KinkPreferenceMap = {
+ favorite: KinkPreference.Favorite,
+ yes: KinkPreference.Yes,
+ maybe: KinkPreference.Maybe,
+ no: KinkPreference.No
+};
+
+export interface MatchReport {
+ [key: number]: number;
+}
+
+export class Matcher {
+ you: Character;
+ them: Character;
+
+ constructor(you: Character, them: Character) {
+ this.you = you;
+ this.them = them;
+ }
+
+ match(): MatchReport {
+ return {
+ [TagId.Orientation]: this.resolveScore(TagId.Orientation, orientationCompatibility),
+ [TagId.Gender]: this.resolveGenderScore(),
+ [TagId.Age]: this.resolveAgeScore(),
+ [TagId.FurryPreference]: this.resolveFurryScore(),
+ [TagId.Species]: this.resolveSpeciesScore()
+ };
+ }
+
+ private resolveScore(tagId: number, compatibilityMap: any, you: Character = this.you, them: Character = this.them): number {
+ const v = Matcher.getTagValueList(tagId, this.them);
+
+ if ((!v) || (!(v in compatibilityMap)))
+ return 0;
+
+ return compatibilityMap[v](you, them);
+ }
+
+
+ private resolveSpeciesScore() {
+ const you = this.you;
+ const them = this.them;
+
+ const yourSpecies = Matcher.species(you);
+ const theirSpecies = Matcher.species(them);
+
+ if (
+ ((yourSpecies !== null) && (Matcher.hatesSpecies(them, yourSpecies))) ||
+ ((theirSpecies !== null) && (Matcher.hatesSpecies(you, theirSpecies)))
+ ) {
+ return -1;
+ }
+
+ if (
+ ((yourSpecies !== null) && (Matcher.maybeSpecies(them, yourSpecies))) ||
+ ((theirSpecies !== null) && (Matcher.maybeSpecies(you, theirSpecies)))
+ ) {
+ return -0.5;
+ }
+
+ if (
+ ((yourSpecies !== null) && (Matcher.likesSpecies(them, yourSpecies))) ||
+ ((theirSpecies !== null) && (Matcher.likesSpecies(you, theirSpecies)))
+ ) {
+ return 1;
+ }
+
+ return 0;
+ }
+
+
+ private resolveFurryScore() {
+ const you = this.you;
+ const them = this.them;
+
+ const youAreAnthro = Matcher.isAnthro(you);
+ const theyAreAnthro = Matcher.isAnthro(them);
+
+ const youAreHuman = Matcher.isHuman(you);
+ const theyAreHuman = Matcher.isHuman(them);
+
+ const yourScore = theyAreAnthro ? Matcher.furryLikeabilityScore(you) : theyAreHuman ? Matcher.humanLikeabilityScore(you) : 0;
+ const theirScore = youAreAnthro ? Matcher.furryLikeabilityScore(them) : youAreHuman ? Matcher.humanLikeabilityScore(them) : 0;
+
+ return Math.min(yourScore || 0, theirScore || 0);
+ }
+
+
+ static furryLikeabilityScore(c: Character): number | null {
+ const anthroKink = Matcher.getKinkPreference(c, Kink.AnthroCharacters);
+
+ if ((anthroKink === KinkPreference.Yes) || (anthroKink === KinkPreference.Favorite)) {
+ return 1;
+ }
+
+ if (anthroKink === KinkPreference.Maybe) {
+ return -0.5;
+ }
+
+ if (anthroKink === KinkPreference.No) {
+ return -1;
+ }
+
+ const furryPreference = Matcher.getTagValueList(TagId.FurryPreference, c);
+
+ if (
+ (furryPreference === FurryPreference.FursAndHumans) ||
+ (furryPreference === FurryPreference.FurriesPreferredHumansOk) ||
+ (furryPreference === FurryPreference.FurriesOnly)
+ ) {
+ return 1;
+ }
+
+ if (furryPreference === FurryPreference.HumansPreferredFurriesOk) {
+ return 0.5;
+ }
+
+ if (furryPreference === FurryPreference.HumansOnly) {
+ return -1;
+ }
+
+ return 0;
+ }
+
+
+ static humanLikeabilityScore(c: Character): number | null {
+ const humanKink = Matcher.getKinkPreference(c, Kink.Humans);
+
+ if ((humanKink === KinkPreference.Yes) || (humanKink === KinkPreference.Favorite)) {
+ return 1;
+ }
+
+ if (humanKink === KinkPreference.Maybe) {
+ return -0.5;
+ }
+
+ if (humanKink === KinkPreference.No) {
+ return -1;
+ }
+
+ const humanPreference = Matcher.getTagValueList(TagId.FurryPreference, c);
+
+ if (
+ (humanPreference === FurryPreference.FursAndHumans) ||
+ (humanPreference === FurryPreference.HumansPreferredFurriesOk) ||
+ (humanPreference === FurryPreference.HumansOnly)
+ ) {
+ return 1;
+ }
+
+ if (humanPreference === FurryPreference.FurriesPreferredHumansOk) {
+ return 0.5;
+ }
+
+ if (humanPreference === FurryPreference.FurriesOnly) {
+ return -1;
+ }
+
+ return 0;
+ }
+
+
+ static likesFurs(c: Character) {
+ const score = this.furryLikeabilityScore(c);
+
+ return (score !== null) ? (score > 0) : false;
+ }
+
+ static hatesFurs(c: Character) {
+ const score = this.furryLikeabilityScore(c);
+
+ return (score !== null) ? (score < 0) : false;
+ }
+
+
+ static likesHumans(c: Character) {
+ const score = this.humanLikeabilityScore(c);
+
+ return (score !== null) ? (score > 0) : false;
+ }
+
+
+ static hatesHumans(c: Character) {
+ const score = this.humanLikeabilityScore(c);
+
+ return (score !== null) ? (score < 0) : false;
+ }
+
+
+ private resolveAgeScore(): number {
+ const you = this.you;
+ const them = this.them;
+
+ const yourAgeTag = Matcher.getTagValue(TagId.Age, you);
+ const theirAgeTag = Matcher.getTagValue(TagId.Age, them);
+
+ if ((!yourAgeTag) || (!theirAgeTag)) {
+ return 0;
+ }
+
+ if ((!yourAgeTag.string) || (!theirAgeTag.string)) {
+ return 0;
+ }
+
+ const yourAge = parseInt(yourAgeTag.string, 10);
+ const theirAge = parseInt(theirAgeTag.string, 10);
+
+ if (
+ ((theirAge < 16) && (Matcher.hates(you, Kink.Ageplay))) ||
+ ((yourAge < 16) && (Matcher.hates(them, Kink.Ageplay))) ||
+ ((theirAge < 16) && (Matcher.has(you, Kink.Ageplay) === false)) ||
+ ((yourAge < 16) && (Matcher.has(them, Kink.Ageplay) === false)) ||
+ ((yourAge < theirAge) && (Matcher.hates(you, Kink.OlderCharacters))) ||
+ ((yourAge > theirAge) && (Matcher.hates(them, Kink.OlderCharacters))) ||
+ ((yourAge > theirAge) && (Matcher.hates(you, Kink.YoungerCharacters))) ||
+ ((yourAge < theirAge) && (Matcher.hates(them, Kink.YoungerCharacters))) ||
+ ((theirAge < 18) && (Matcher.hates(you, Kink.UnderageCharacters))) ||
+ ((yourAge < 18) && (Matcher.hates(them, Kink.UnderageCharacters)))
+ )
+ return -1;
+
+ if (
+ ((theirAge < 18) && (Matcher.likes(you, Kink.UnderageCharacters))) ||
+ ((yourAge < 18) && (Matcher.likes(them, Kink.UnderageCharacters))) ||
+ ((yourAge > theirAge) && (Matcher.likes(you, Kink.YoungerCharacters))) ||
+ ((yourAge < theirAge) && (Matcher.likes(them, Kink.YoungerCharacters))) ||
+ ((yourAge < theirAge) && (Matcher.likes(you, Kink.OlderCharacters))) ||
+ ((yourAge > theirAge) && (Matcher.likes(them, Kink.OlderCharacters))) ||
+ ((theirAge < 16) && (Matcher.likes(you, Kink.Ageplay))) ||
+ ((yourAge < 16) && (Matcher.likes(them, Kink.Ageplay)))
+ )
+ return 1;
+
+ return 0;
+ }
+
+ private resolveGenderScore() {
+ const you = this.you;
+ const them = this.them;
+
+ const yourGender = Matcher.getTagValueList(TagId.Gender, you);
+ const theirGender = Matcher.getTagValueList(TagId.Gender, them);
+
+ const yourGenderScore = Matcher.genderLikeabilityScore(them, yourGender);
+ const theirGenderScore = Matcher.genderLikeabilityScore(you, theirGender);
+
+ const yourFinalScore = (yourGenderScore !== null) ? yourGenderScore : this.resolveScore(TagId.Orientation, orientationCompatibility, you, them);
+ const theirFinalScore = (theirGenderScore !== null) ? theirGenderScore : this.resolveScore(TagId.Orientation, orientationCompatibility, them, you);
+
+ return Math.min(yourFinalScore, theirFinalScore);
+ }
+
+ static getTagValue(tagId: number, c: Character) {
+ return c.infotags[tagId];
+ }
+
+ static getTagValueList(tagId: number, c: Character): number | undefined {
+ const t = this.getTagValue(tagId, c);
+
+ if ((!t) || (!t.list)) {
+ return;
+ }
+
+ return t.list;
+ }
+
+ // Considers males and females only
+ static isSameSexCis(a: Character, b: Character): boolean {
+ const aGender = this.getTagValueList(TagId.Gender, a);
+ const bGender = this.getTagValueList(TagId.Gender, b);
+
+ if ((aGender !== Gender.Male) && (aGender !== Gender.Female)) {
+ return false;
+ }
+
+ return ((aGender !== undefined) && (aGender === bGender));
+ }
+
+ // Considers
+ static isGenderedCis(c: Character): boolean {
+ const gender = this.getTagValueList(TagId.Gender, c);
+
+ return ((!!gender) && (gender !== Gender.None));
+ }
+
+ static isMaleCis(c: Character): boolean {
+ const gender = this.getTagValueList(TagId.Gender, c);
+
+ return (gender === Gender.Male);
+ }
+
+ static isFemaleCis(c: Character): boolean {
+ const gender = this.getTagValueList(TagId.Gender, c);
+
+ return (gender === Gender.Female);
+ }
+
+ static isCis(...characters: Character[]): boolean {
+ return _.every(characters, (c: Character) => ((Matcher.isMaleCis(c)) || (Matcher.isFemaleCis(c))));
+ }
+
+
+ static genderLikeabilityScore(c: Character, gender?: Gender): number | null {
+ if (gender === undefined) {
+ return null;
+ }
+
+ const byKink = Matcher.getKinkGenderPreference(c, gender);
+
+ if (byKink !== null) {
+ if ((byKink === KinkPreference.Yes) || (byKink === KinkPreference.Favorite)) {
+ return 1;
+ }
+
+ if (byKink === KinkPreference.Maybe) {
+ return -0.5;
+ }
+
+ if (byKink === KinkPreference.No) {
+ return -1;
+ }
+ }
+
+ if (this.isCis(c)) {
+ if ((gender !== Gender.Female) && (gender !== Gender.Male)) {
+ return -1;
+ }
+ }
+
+ return null;
+ }
+
+ static likesGender(c: Character, gender: Gender): boolean | null {
+ const byKink = Matcher.getKinkGenderPreference(c, gender);
+
+ if (byKink !== null)
+ return ((byKink === KinkPreference.Yes) || (byKink === KinkPreference.Favorite));
+
+ if ((Matcher.isCis(c)) && ((gender === Gender.Male) || (gender === Gender.Female)))
+ return gender !== this.getTagValueList(TagId.Gender, c);
+
+ return null;
+ }
+
+ static dislikesGender(c: Character, gender: Gender): boolean | null {
+ const byKink = Matcher.getKinkGenderPreference(c, gender);
+
+ if (byKink !== null)
+ return (byKink === KinkPreference.No);
+
+ if ((Matcher.isCis(c)) && ((gender === Gender.Male) || (gender === Gender.Female)))
+ return gender === this.getTagValueList(TagId.Gender, c);
+
+ return null;
+ }
+
+ static maybeGender(c: Character, gender: Gender): boolean | null {
+ const byKink = Matcher.getKinkGenderPreference(c, gender);
+
+ if (byKink !== null)
+ return (byKink === KinkPreference.Maybe);
+
+ return null;
+ }
+
+ static getKinkPreference(c: Character, kinkId: number): KinkPreference | null {
+ if (!(kinkId in c.kinks))
+ return null;
+
+ return kinkMapping[c.kinks[kinkId] as string];
+ }
+
+ static getKinkGenderPreference(c: Character, gender: Gender): KinkPreference | null {
+ if (!(gender in genderKinkMapping))
+ return null;
+
+ return this.getKinkPreference(c, genderKinkMapping[gender]);
+ }
+
+ static getKinkSpeciesPreference(c: Character, species: Species): KinkPreference | null {
+ return this.getKinkPreference(c, species);
+ }
+
+ static likesSpecies(c: Character, species: Species): boolean | null {
+ const byKink = Matcher.getKinkSpeciesPreference(c, species);
+
+ if (byKink !== null)
+ return ((byKink === KinkPreference.Yes) || (byKink === KinkPreference.Favorite));
+
+ return null;
+ }
+
+ static maybeSpecies(c: Character, species: Species): boolean | null {
+ const byKink = Matcher.getKinkSpeciesPreference(c, species);
+
+ if (byKink !== null)
+ return (byKink === KinkPreference.Maybe);
+
+ return null;
+ }
+
+ static hatesSpecies(c: Character, species: Species): boolean | null {
+ const byKink = Matcher.getKinkSpeciesPreference(c, species);
+
+ if (byKink !== null)
+ return (byKink === KinkPreference.No);
+
+ return null;
+ }
+
+ static likes(c: Character, kinkId: Kink): boolean {
+ const r = Matcher.getKinkPreference(c, kinkId);
+
+ return ((r === KinkPreference.Favorite) || (r === KinkPreference.Yes));
+
+ }
+
+ static hates(c: Character, kinkId: Kink): boolean {
+ const r = Matcher.getKinkPreference(c, kinkId);
+
+ return (r === KinkPreference.No);
+
+ }
+
+ static has(c: Character, kinkId: Kink): boolean {
+ const r = Matcher.getKinkPreference(c, kinkId);
+
+ return (r !== null);
+ }
+
+ static isAnthro(c: Character): boolean | null {
+ const bodyTypeId = this.getTagValueList(TagId.BodyType, c);
+
+ if (bodyTypeId === BodyType.Anthro) {
+ return true;
+ }
+
+ const speciesId = this.species(c);
+
+ if (!speciesId)
+ return null;
+
+ return (nonAnthroSpecies.indexOf(parseInt(`${speciesId}`, 10)) < 0);
+ }
+
+ static isHuman(c: Character): boolean | null {
+ const bodyTypeId = this.getTagValueList(TagId.BodyType, c);
+
+ if (bodyTypeId === BodyType.Human) {
+ return true;
+ }
+
+ const speciesId = this.species(c);
+
+ return (speciesId === Species.Human);
+ }
+
+ static species(c: Character): Species | null {
+ let foundSpeciesId: Species | null = null;
+ let match = '';
+
+ const mySpecies = this.getTagValue(TagId.Species, c);
+
+ if ((!mySpecies) || (!mySpecies.string)) {
+ return Species.Human; // best guess
+ }
+
+ const finalSpecies = mySpecies.string.toLowerCase();
+
+ _.each(
+ speciesMapping as any,
+ (keywords: string[], speciesId: Species) => {
+ _.each(
+ keywords,
+ (k: string) => {
+ if ((k.length > match.length) && (finalSpecies.indexOf(k) >= 0)) {
+ match = k;
+ foundSpeciesId = speciesId;
+ }
+ }
+ );
+ }
+ );
+
+ return foundSpeciesId;
+ }
+}
diff --git a/site/character_page/sidebar.vue b/site/character_page/sidebar.vue
index 6f58aba..1e3f787 100644
--- a/site/character_page/sidebar.vue
+++ b/site/character_page/sidebar.vue
@@ -39,7 +39,7 @@
Online In Chat
-
+
@@ -101,6 +101,7 @@
import FriendDialog from './friend_dialog.vue';
import InfotagView from './infotag.vue';
import {Character, CONTACT_GROUP_ID, SharedStore} from './interfaces';
+ import { MatchReport } from './matcher';
import MemoDialog from './memo_dialog.vue';
import ReportDialog from './report_dialog.vue';
@@ -143,7 +144,7 @@
@Prop()
readonly oldApi?: true;
@Prop({required: true})
- readonly selfCharacter!: Character;
+ readonly characterMatch!: MatchReport;
readonly shared: SharedStore = Store;
readonly quickInfoIds: ReadonlyArray = [1, 3, 2, 49, 9, 29, 15, 41, 25]; // Do not sort these.