Smart filters and M1 build
This commit is contained in:
		
							parent
							
								
									65ab5ffa32
								
							
						
					
					
						commit
						bbc2ca2f83
					
				| @ -1,7 +1,8 @@ | |||||||
| # Changelog | # Changelog | ||||||
| 
 | 
 | ||||||
| ## Canary | ## 1.17.0 | ||||||
| * Added a way to hide/filter out characters, messages, and ads (Settings > Identity Politics) | * Added a way to hide/filter out characters, messages, and ads (Settings > Smart Filters) | ||||||
|  | * Added MacOS M1 build | ||||||
| 
 | 
 | ||||||
| ## 1.16.2 | ## 1.16.2 | ||||||
| * Fixed broken auto-ads | * Fixed broken auto-ads | ||||||
|  | |||||||
| @ -1,7 +1,7 @@ | |||||||
| # Download | # Download | ||||||
| [Windows](https://github.com/mrstallion/fchat-rising/releases/download/v1.16.2/F-Chat-Rising-1.16.2-win.exe) (75 MB) | [Windows](https://github.com/mrstallion/fchat-rising/releases/download/v1.17.0/F-Chat-Rising-1.17.0-win.exe) (75 MB) | ||||||
| | [MacOS](https://github.com/mrstallion/fchat-rising/releases/download/v1.16.2/F-Chat-Rising-1.16.2-macos.dmg) (76 MB) | | [MacOS](https://github.com/mrstallion/fchat-rising/releases/download/v1.17.0/F-Chat-Rising-1.17.0-macos.dmg) (76 MB) | ||||||
| | [Linux](https://github.com/mrstallion/fchat-rising/releases/download/v1.16.2/F-Chat-Rising-1.16.2-linux.AppImage) (76 MB) | | [Linux](https://github.com/mrstallion/fchat-rising/releases/download/v1.17.0/F-Chat-Rising-1.17.0-linux.AppImage) (76 MB) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| # F-Chat Rising | # F-Chat Rising | ||||||
|  | |||||||
| @ -270,7 +270,7 @@ | |||||||
|             core.connection.onMessage('FKS', async (data) => { |             core.connection.onMessage('FKS', async (data) => { | ||||||
|                 const results = data.characters.map((x) => ({ character: core.characters.get(x), profile: null })) |                 const results = data.characters.map((x) => ({ character: core.characters.get(x), profile: null })) | ||||||
|                     .filter((x) => core.state.hiddenUsers.indexOf(x.character.name) === -1 && !x.character.isIgnored) |                     .filter((x) => core.state.hiddenUsers.indexOf(x.character.name) === -1 && !x.character.isIgnored) | ||||||
|                     .filter((x) => this.isSpeciesMatch(x) && this.isBodyTypeMatch(x) && this.isSmartFilterMatch(x)) |                     .filter((x) => this.isSpeciesMatch(x) && this.isBodyTypeMatch(x) && !this.isSmartFiltered(x)) | ||||||
|                     .sort(sort); |                     .sort(sort); | ||||||
| 
 | 
 | ||||||
|                 // pre-warm cache |                 // pre-warm cache | ||||||
| @ -348,7 +348,7 @@ | |||||||
|         private resort(results = this.results) { |         private resort(results = this.results) { | ||||||
|           this.results = (_.filter( |           this.results = (_.filter( | ||||||
|               results, |               results, | ||||||
|               (x) => this.isSpeciesMatch(x) && this.isBodyTypeMatch(x) |               (x) => this.isSpeciesMatch(x) && this.isBodyTypeMatch(x) && !this.isSmartFiltered(x) | ||||||
|           ) as SearchResult[]).sort(sort); |           ) as SearchResult[]).sort(sort); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
| @ -399,12 +399,12 @@ | |||||||
|             return this.data.bodytypes.indexOf(bodytype!.value) > -1 |             return this.data.bodytypes.indexOf(bodytype!.value) > -1 | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         isSmartFilterMatch(result: SearchResult) { |         isSmartFiltered(result: SearchResult) { | ||||||
|           if (!core.state.settings.risingFilter.hideSearchResults) { |           if (!core.state.settings.risingFilter.hideSearchResults) { | ||||||
|             return true; |             return false; | ||||||
|           } |           } | ||||||
| 
 | 
 | ||||||
|           return result.profile ? !result.profile?.match.isFiltered : true; |           return !!result.profile?.match.isFiltered; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         getSpeciesOptions(): SearchSpecies[] { |         getSpeciesOptions(): SearchSpecies[] { | ||||||
|  | |||||||
| @ -1,7 +1,7 @@ | |||||||
| <template> | <template> | ||||||
|     <modal :action="l('settings.action')" @submit="submit" @open="load()" id="settings" dialogClass="w-100"> |     <modal :action="l('settings.action')" @submit="submit" @open="load()" id="settings" dialogClass="w-100"> | ||||||
|         <tabs style="flex-shrink:0;margin-bottom:10px" v-model="selectedTab" |         <tabs style="flex-shrink:0;margin-bottom:10px" v-model="selectedTab" | ||||||
|             :tabs="[l('settings.tabs.general'), l('settings.tabs.notifications'), 'F-Chat Rising 🦄', 'Identity Politics 🦄', l('settings.tabs.hideAds'), l('settings.tabs.import')]"></tabs> |             :tabs="[l('settings.tabs.general'), l('settings.tabs.notifications'), 'F-Chat Rising 🦄', 'Smart Filters 🦄', l('settings.tabs.hideAds'), l('settings.tabs.import')]"></tabs> | ||||||
|         <div v-show="selectedTab === '0'"> |         <div v-show="selectedTab === '0'"> | ||||||
|             <div class="form-group"> |             <div class="form-group"> | ||||||
|                 <label class="control-label" for="disallowedTags">{{l('settings.disallowedTags')}}</label> |                 <label class="control-label" for="disallowedTags">{{l('settings.disallowedTags')}}</label> | ||||||
| @ -202,6 +202,14 @@ | |||||||
|         </div> |         </div> | ||||||
| 
 | 
 | ||||||
|         <div v-show="selectedTab === '3'"> |         <div v-show="selectedTab === '3'"> | ||||||
|  |             <div class="warning"> | ||||||
|  |               <h5>Danger Zone!</h5> | ||||||
|  |               <div>By activating filtering, you may no longer be able to see or receive all messages from F-Chat. | ||||||
|  |               Filters do not apply to friends or bookmarked characters.</div> | ||||||
|  | 
 | ||||||
|  |               <div>Beta version. Some of these features and behaviors may be removed or significantly changed in the future.</div> | ||||||
|  |             </div> | ||||||
|  | 
 | ||||||
|             <h5>Visibility</h5> |             <h5>Visibility</h5> | ||||||
| 
 | 
 | ||||||
|             <div class="form-group filters"> |             <div class="form-group filters"> | ||||||
| @ -235,9 +243,9 @@ | |||||||
|                     Hide <b>private messages</b> (PMs) from matching characters |                     Hide <b>private messages</b> (PMs) from matching characters | ||||||
|                 </label> |                 </label> | ||||||
| 
 | 
 | ||||||
|                 <label class="control-label" for="risingFilter.penalizeMatches"> |                 <label class="control-label" for="risingFilter.showFilterIcon"> | ||||||
|                     <input type="checkbox" id="risingFilter.penalizeMatches" v-model="risingFilter.penalizeMatches"/> |                     <input type="checkbox" id="risingFilter.showFilterIcon" v-model="risingFilter.showFilterIcon"/> | ||||||
|                     Penalize <b>match scores</b> for matching characters |                     Show <b>filter icon</b> on matching characters | ||||||
|                 </label> |                 </label> | ||||||
|             </div> |             </div> | ||||||
| 
 | 
 | ||||||
| @ -246,17 +254,27 @@ | |||||||
|                     <input type="checkbox" id="risingFilter.autoReply" v-model="risingFilter.autoReply"/> |                     <input type="checkbox" id="risingFilter.autoReply" v-model="risingFilter.autoReply"/> | ||||||
|                     Send an automatic 'no thank you' response to matching characters if they message you |                     Send an automatic 'no thank you' response to matching characters if they message you | ||||||
|                 </label> |                 </label> | ||||||
|  | 
 | ||||||
|  |                 <label class="control-label" for="risingFilter.penalizeMatches"> | ||||||
|  |                     <input type="checkbox" id="risingFilter.penalizeMatches" v-model="risingFilter.penalizeMatches"/> | ||||||
|  |                     Penalize <b>match scores</b> for matching characters | ||||||
|  |                 </label> | ||||||
|  | 
 | ||||||
|  |                 <label class="control-label" for="risingFilter.rewardNonMatches"> | ||||||
|  |                     <input type="checkbox" id="risingFilter.rewardNonMatches" v-model="risingFilter.rewardNonMatches"/> | ||||||
|  |                     Increase <b>match scores</b> for non-matching characters | ||||||
|  |                 </label> | ||||||
|             </div> |             </div> | ||||||
| 
 | 
 | ||||||
|             <h5>Character Age Match</h5> |             <h5>Character Age Match</h5> | ||||||
|             <div class="form-group">Leave empty for no limit.</div> |             <div class="form-group">Leave empty for no limit.</div> | ||||||
| 
 | 
 | ||||||
|             <div class="form-group"> |             <div class="form-group"> | ||||||
|                 <label class="control-label" for="risingFilter.minAge">Characters younger than</label> |                 <label class="control-label" for="risingFilter.minAge">Characters younger than (years)</label> | ||||||
|                 <input id="risingFilter.minAge" type="number" class="form-control" v-model="risingFilter.minAge"/> |                 <input id="risingFilter.minAge" type="number" class="form-control" v-model="risingFilter.minAge" placeholder="Enter age" /> | ||||||
| 
 | 
 | ||||||
|                 <label class="control-label" for="risingFilter.maxAge">Characters older than</label> |                 <label class="control-label" for="risingFilter.maxAge">Characters older than (years)</label> | ||||||
|                 <input id="risingFilter.maxAge" type="number" class="form-control" v-model="risingFilter.maxAge"/> |                 <input id="risingFilter.maxAge" type="number" class="form-control" v-model="risingFilter.maxAge" placeholder="Enter age" /> | ||||||
|             </div> |             </div> | ||||||
| 
 | 
 | ||||||
|             <h5>Type Match</h5> |             <h5>Type Match</h5> | ||||||
| @ -271,7 +289,7 @@ | |||||||
|             <div class="form-group">Filters are not applied to these character names. Separate names with a linefeed.</div> |             <div class="form-group">Filters are not applied to these character names. Separate names with a linefeed.</div> | ||||||
| 
 | 
 | ||||||
|             <div class="form-group"> |             <div class="form-group"> | ||||||
|                 <textarea class="form-control" :value="getExceptionList()" @change="(v) => setExceptionList(v)"></textarea> |                 <textarea class="form-control" :value="getExceptionList()" @change="(v) => setExceptionList(v)" placeholder="Enter names"></textarea> | ||||||
|             </div> |             </div> | ||||||
|         </div> |         </div> | ||||||
| 
 | 
 | ||||||
| @ -510,7 +528,7 @@ | |||||||
|     } |     } | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <style> | <style lang="scss"> | ||||||
|     #settings .form-group { |     #settings .form-group { | ||||||
|         margin-left: 0; |         margin-left: 0; | ||||||
|         margin-right: 0; |         margin-right: 0; | ||||||
| @ -522,4 +540,23 @@ | |||||||
|       margin-left: 5px; |       margin-left: 5px; | ||||||
|       list-style: none; |       list-style: none; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     #settings .warning { | ||||||
|  |       border: 1px solid var(--warning); | ||||||
|  |       padding: 10px; | ||||||
|  |       margin-bottom: 20px; | ||||||
|  |       border-radius: 3px; | ||||||
|  | 
 | ||||||
|  |       div { | ||||||
|  |         margin-top: 10px; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #settings .form-group.filters.age label { | ||||||
|  |       padding-top: 10px; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #settings .form-group.filters.age  input { | ||||||
|  |       margin-left: 5px; | ||||||
|  |     } | ||||||
| </style> | </style> | ||||||
|  | |||||||
| @ -1,5 +1,5 @@ | |||||||
| <!-- Linebreaks inside this template will break BBCode views --> | <!-- Linebreaks inside this template will break BBCode views --> | ||||||
| <template><span :class="userClass" v-bind:bbcodeTag.prop="'user'" v-bind:character.prop="character" v-bind:channel.prop="channel" @mouseover.prevent="show()" @mouseenter.prevent="show()" @mouseleave.prevent="dismiss()" @click.middle.prevent.stop="toggleStickyness()" @click.right.passive="dismiss(true)" @click.left.passive="dismiss(true)"><span v-if="!!statusClass" :class="statusClass"></span><span v-if="!!rankIcon" :class="rankIcon"></span>{{character.name}}<span v-if="!!matchClass" :class="matchClass">{{getMatchScoreTitle(matchScore)}}</span></span></template> | <template><span :class="userClass" v-bind:bbcodeTag.prop="'user'" v-bind:character.prop="character" v-bind:channel.prop="channel" @mouseover.prevent="show()" @mouseenter.prevent="show()" @mouseleave.prevent="dismiss()" @click.middle.prevent.stop="toggleStickyness()" @click.right.passive="dismiss(true)" @click.left.passive="dismiss(true)"><span v-if="!!statusClass" :class="statusClass"></span><span v-if="!!rankIcon" :class="rankIcon"></span><span v-if="!!smartFilterIcon" :class="smartFilterIcon"></span>{{character.name}}<span v-if="!!matchClass" :class="matchClass">{{getMatchScoreTitle(matchScore)}}</span></span></template> | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| <script lang="ts"> | <script lang="ts"> | ||||||
| @ -36,6 +36,7 @@ export function getStatusIcon(status: Character.Status): string { | |||||||
| 
 | 
 | ||||||
| export interface StatusClasses { | export interface StatusClasses { | ||||||
|   rankIcon: string | null; |   rankIcon: string | null; | ||||||
|  |   smartFilterIcon: string | null; | ||||||
|   statusClass: string | null; |   statusClass: string | null; | ||||||
|   matchClass: string | null; |   matchClass: string | null; | ||||||
|   matchScore: number | string | null; |   matchScore: number | string | null; | ||||||
| @ -54,6 +55,7 @@ export function getStatusClasses( | |||||||
|     let statusClass = null; |     let statusClass = null; | ||||||
|     let matchClass = null; |     let matchClass = null; | ||||||
|     let matchScore = null; |     let matchScore = null; | ||||||
|  |     let smartFilterIcon: string | null = null; | ||||||
| 
 | 
 | ||||||
|     if(character.isChatOp) { |     if(character.isChatOp) { | ||||||
|         rankIcon = 'far fa-gem'; |         rankIcon = 'far fa-gem'; | ||||||
| @ -68,10 +70,17 @@ export function getStatusClasses( | |||||||
|     if ((showStatus) || (character.status === 'crown')) |     if ((showStatus) || (character.status === 'crown')) | ||||||
|         statusClass = `fa-fw ${getStatusIcon(character.status)}`; |         statusClass = `fa-fw ${getStatusIcon(character.status)}`; | ||||||
| 
 | 
 | ||||||
|     if ((core.state.settings.risingAdScore) && (showMatch)) { |     const cache = ((showMatch) && ((core.state.settings.risingAdScore) || (core.state.settings.risingFilter.showFilterIcon))) | ||||||
|         const cache = core.cache.profileCache.getSync(character.name); |       ? core.cache.profileCache.getSync(character.name) | ||||||
|  |       : undefined; | ||||||
| 
 | 
 | ||||||
|         if (cache) { |     // undefined == not interested | ||||||
|  |     // null == no cache hit | ||||||
|  |     if (cache === null) { | ||||||
|  |         void core.cache.addProfile(character.name); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if ((core.state.settings.risingAdScore) && (showMatch) && (cache)) { | ||||||
|         if ((cache.match.searchScore >= kinkMatchWeights.unicornThreshold) && (cache.match.matchScore === Scoring.MATCH)) { |         if ((cache.match.searchScore >= kinkMatchWeights.unicornThreshold) && (cache.match.matchScore === Scoring.MATCH)) { | ||||||
|           matchClass = 'match-found unicorn'; |           matchClass = 'match-found unicorn'; | ||||||
|           matchScore = 'unicorn'; |           matchScore = 'unicorn'; | ||||||
| @ -79,10 +88,10 @@ export function getStatusClasses( | |||||||
|           matchClass = `match-found ${Score.getClasses(cache.match.matchScore)}`; |           matchClass = `match-found ${Score.getClasses(cache.match.matchScore)}`; | ||||||
|           matchScore = cache.match.matchScore; |           matchScore = cache.match.matchScore; | ||||||
|         } |         } | ||||||
|         } else { |  | ||||||
|             /* tslint:disable-next-line no-floating-promises */ |  | ||||||
|             core.cache.addProfile(character.name); |  | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     if (core.state.settings.risingFilter.showFilterIcon && cache?.match.isFiltered) { | ||||||
|  |       smartFilterIcon = 'user-filter fas fa-filter'; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     const gender = character.gender !== undefined ? character.gender.toLowerCase() : 'none'; |     const gender = character.gender !== undefined ? character.gender.toLowerCase() : 'none'; | ||||||
| @ -98,6 +107,7 @@ export function getStatusClasses( | |||||||
|       matchClass, |       matchClass, | ||||||
|       matchScore, |       matchScore, | ||||||
|       userClass, |       userClass, | ||||||
|  |       smartFilterIcon, | ||||||
|       isBookmark |       isBookmark | ||||||
|     }; |     }; | ||||||
| } | } | ||||||
| @ -130,6 +140,7 @@ export default class UserView extends Vue { | |||||||
|     userClass = ''; |     userClass = ''; | ||||||
| 
 | 
 | ||||||
|     rankIcon: string | null = null; |     rankIcon: string | null = null; | ||||||
|  |     smartFilterIcon: string | null = null; | ||||||
|     statusClass: string | null = null; |     statusClass: string | null = null; | ||||||
|     matchClass: string | null = null; |     matchClass: string | null = null; | ||||||
|     matchScore: number | string | null = null; |     matchScore: number | string | null = null; | ||||||
| @ -198,6 +209,7 @@ export default class UserView extends Vue { | |||||||
|       const res = getStatusClasses(this.character, this.channel, !!this.showStatus, !!this.bookmark, !!this.match); |       const res = getStatusClasses(this.character, this.channel, !!this.showStatus, !!this.bookmark, !!this.match); | ||||||
| 
 | 
 | ||||||
|       this.rankIcon = res.rankIcon; |       this.rankIcon = res.rankIcon; | ||||||
|  |       this.smartFilterIcon = res.smartFilterIcon; | ||||||
|       this.statusClass = res.statusClass; |       this.statusClass = res.statusClass; | ||||||
|       this.matchClass = res.matchClass; |       this.matchClass = res.matchClass; | ||||||
|       this.matchScore = res.matchScore; |       this.matchScore = res.matchScore; | ||||||
|  | |||||||
| @ -56,23 +56,28 @@ export class Settings implements ISettings { | |||||||
|     risingColorblindMode = false; |     risingColorblindMode = false; | ||||||
| 
 | 
 | ||||||
|     risingFilter = { |     risingFilter = { | ||||||
|         hideAds: false, |         hideAds: true, | ||||||
|         hideSearchResults: false, |         hideSearchResults: true, | ||||||
|         hideChannelMembers: false, |         hideChannelMembers: false, | ||||||
|         hidePublicChannelMessages: false, |         hidePublicChannelMessages: false, | ||||||
|         hidePrivateChannelMessages: false, |         hidePrivateChannelMessages: false, | ||||||
|         hidePrivateMessages: false, |         hidePrivateMessages: false, | ||||||
|         penalizeMatches: false, |         showFilterIcon: true, | ||||||
|  |         penalizeMatches: true, | ||||||
|  |         rewardNonMatches: false, | ||||||
|         autoReply: true, |         autoReply: true, | ||||||
|         minAge: null, |         minAge: null, | ||||||
|         maxAge: null, |         maxAge: null, | ||||||
|         smartFilters: { |         smartFilters: { | ||||||
|             ageplay: false, |             ageplay: false, | ||||||
|             anthro: false, |             anthro: false, | ||||||
|  |             female: false, | ||||||
|             feral: false, |             feral: false, | ||||||
|             human: false, |             human: false, | ||||||
|             hyper: false, |             hyper: false, | ||||||
|             incest: false, |             incest: false, | ||||||
|  |             intersex: false, | ||||||
|  |             male: false, | ||||||
|             microMacro: false, |             microMacro: false, | ||||||
|             obesity: false, |             obesity: false, | ||||||
|             pokemon: false, |             pokemon: false, | ||||||
|  | |||||||
| @ -13,7 +13,6 @@ import throat from 'throat'; | |||||||
| import Bluebird from 'bluebird'; | import Bluebird from 'bluebird'; | ||||||
| import log from 'electron-log'; | import log from 'electron-log'; | ||||||
| import isChannel = Interfaces.isChannel; | import isChannel = Interfaces.isChannel; | ||||||
| import isPrivate = Interfaces.isPrivate; //tslint:disable-line:match-default-export-name
 |  | ||||||
| 
 | 
 | ||||||
| function createMessage(this: any, type: MessageType, sender: Character, text: string, time?: Date): Message { | function createMessage(this: any, type: MessageType, sender: Character, text: string, time?: Date): Message { | ||||||
|     if(type === MessageType.Message && isAction(text)) { |     if(type === MessageType.Message && isAction(text)) { | ||||||
| @ -536,10 +535,16 @@ class State implements Interfaces.State { | |||||||
|             this.channelConversations.some((x) => x.unread === Interfaces.UnreadState.Mention); |             this.channelConversations.some((x) => x.unread === Interfaces.UnreadState.Mention); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     getPrivate(character: Character): PrivateConversation { |     getPrivate(character: Character): PrivateConversation; | ||||||
|  |     getPrivate(character: Character, noCreate: boolean = false): PrivateConversation | undefined { | ||||||
|         const key = character.name.toLowerCase(); |         const key = character.name.toLowerCase(); | ||||||
|         let conv = state.privateMap[key]; |         let conv = state.privateMap[key]; | ||||||
|         if(conv !== undefined) return conv; |         if(conv !== undefined) return conv; | ||||||
|  | 
 | ||||||
|  |         if (noCreate) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         conv = new PrivateConversation(character); |         conv = new PrivateConversation(character); | ||||||
|         this.privateConversations.push(conv); |         this.privateConversations.push(conv); | ||||||
|         this.privateMap[key] = conv; |         this.privateMap[key] = conv; | ||||||
| @ -613,6 +618,55 @@ function isOfInterest(this: any, character: Character): boolean { | |||||||
|     return character.isFriend || character.isBookmarked || state.privateMap[character.name.toLowerCase()] !== undefined; |     return character.isFriend || character.isBookmarked || state.privateMap[character.name.toLowerCase()] !== undefined; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | async function testSmartFilterForPrivateMessage(fromChar: Character.Character): Promise<boolean> { | ||||||
|  |     const cachedProfile = core.cache.profileCache.getSync(fromChar.name) || await core.cache.profileCache.get(fromChar.name); | ||||||
|  | 
 | ||||||
|  |     if ( | ||||||
|  |         cachedProfile && | ||||||
|  |         cachedProfile.match.isFiltered && | ||||||
|  |         core.state.settings.risingFilter.autoReply && | ||||||
|  |         !cachedProfile.match.autoResponded | ||||||
|  |     ) { | ||||||
|  |         cachedProfile.match.autoResponded = true; | ||||||
|  | 
 | ||||||
|  |         log.debug('filter.autoresponse', { name: fromChar.name }); | ||||||
|  | 
 | ||||||
|  |         void Conversation.conversationThroat( | ||||||
|  |           async() => { | ||||||
|  |                 await Conversation.testPostDelay(); | ||||||
|  | 
 | ||||||
|  |               // tslint:disable-next-line:prefer-template
 | ||||||
|  |                 const m = '[Automated message] Sorry, the player of this character has indicated that they are not interested in characters matching your profile. They will not see your message.\n\n' + | ||||||
|  |                     'Need a filter for yourself? Try out [url=https://mrstallion.github.io/fchat-rising/]F-Chat Rising[/url]'; | ||||||
|  | 
 | ||||||
|  |                 core.connection.send('PRI', {recipient: fromChar.name, message: m}); | ||||||
|  |                 core.cache.markLastPostTime(); | ||||||
|  |               } | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (cachedProfile && cachedProfile.match.isFiltered && core.state.settings.risingFilter.hidePrivateMessages) { | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async function testSmartFilterForChannel(fromChar: Character.Character, conversation: ChannelConversation): Promise<boolean> { | ||||||
|  |     if ( | ||||||
|  |         (isChannel(conversation) && conversation.channel.owner === '' && core.state.settings.risingFilter.hidePublicChannelMessages) || | ||||||
|  |         (isChannel(conversation) && conversation.channel.owner !== '' && core.state.settings.risingFilter.hidePrivateChannelMessages) | ||||||
|  |     ) { | ||||||
|  |         const cachedProfile = core.cache.profileCache.getSync(fromChar.name) || await core.cache.profileCache.get(fromChar.name); | ||||||
|  | 
 | ||||||
|  |         if (cachedProfile && cachedProfile.match.isFiltered && !fromChar.isChatOp) { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| export default function(this: any): Interfaces.State { | export default function(this: any): Interfaces.State { | ||||||
|     state = new State(); |     state = new State(); | ||||||
|     window.addEventListener('focus', () => { |     window.addEventListener('focus', () => { | ||||||
| @ -679,12 +733,17 @@ export default function(this: any): Interfaces.State { | |||||||
|             await conv.addMessage(new EventMessage(text)); |             await conv.addMessage(new EventMessage(text)); | ||||||
|         } |         } | ||||||
|     }); |     }); | ||||||
| 
 |  | ||||||
|     connection.onMessage('PRI', async(data, time) => { |     connection.onMessage('PRI', async(data, time) => { | ||||||
|         const char = core.characters.get(data.character); |         const char = core.characters.get(data.character); | ||||||
|         if(char.isIgnored) return connection.send('IGN', {action: 'notify', character: data.character}); |         if(char.isIgnored) return connection.send('IGN', {action: 'notify', character: data.character}); | ||||||
|         const message = createMessage(MessageType.Message, char, decodeHTML(data.message), time); |         const message = createMessage(MessageType.Message, char, decodeHTML(data.message), time); | ||||||
|  | 
 | ||||||
|  |         if (await testSmartFilterForPrivateMessage(char) === true) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         EventBus.$emit('private-message', { message }); |         EventBus.$emit('private-message', { message }); | ||||||
|  | 
 | ||||||
|         const conv = state.getPrivate(char); |         const conv = state.getPrivate(char); | ||||||
|         await conv.addMessage(message); |         await conv.addMessage(message); | ||||||
|     }); |     }); | ||||||
| @ -695,39 +754,13 @@ export default function(this: any): Interfaces.State { | |||||||
|         if(char.isIgnored) return; |         if(char.isIgnored) return; | ||||||
| 
 | 
 | ||||||
|         const message = createMessage(MessageType.Message, char, decodeHTML(data.message), time); |         const message = createMessage(MessageType.Message, char, decodeHTML(data.message), time); | ||||||
|         EventBus.$emit('channel-message', { message, channel: conversation }); |  | ||||||
|         await conversation.addMessage(message); |  | ||||||
| // message.type === MessageType.Message
 |  | ||||||
|         if ( |  | ||||||
|             (isPrivate(conversation) && core.state.settings.risingFilter.hidePrivateMessages) || |  | ||||||
|             (isChannel(conversation) && conversation.channel.owner === '' && core.state.settings.risingFilter.hidePublicChannelMessages) || |  | ||||||
|             (isChannel(conversation) && conversation.channel.owner !== '' && core.state.settings.risingFilter.hidePrivateChannelMessages) |  | ||||||
|         ) { |  | ||||||
|             const cachedProfile = core.cache.profileCache.getSync(char.name) || await core.cache.profileCache.get(char.name); |  | ||||||
| 
 | 
 | ||||||
|             if (cachedProfile && isPrivate(conversation) && core.state.settings.risingFilter.autoReply && !cachedProfile.match.autoResponded) { |         if (await testSmartFilterForChannel(char, conversation) === true) { | ||||||
|                 cachedProfile.match.autoResponded = true; |  | ||||||
| 
 |  | ||||||
|                 log.debug('filter.autoresponse', { name: char.name }); |  | ||||||
| 
 |  | ||||||
|                 void Conversation.conversationThroat( |  | ||||||
|                   async() => { |  | ||||||
|                         await Conversation.testPostDelay(); |  | ||||||
| 
 |  | ||||||
|                       // tslint:disable-next-line:prefer-template
 |  | ||||||
|                         const m = '[Automated message] Sorry, the player of this character has indicated that they are not interested in characters matching your profile. They will not see your message.\n\n' + |  | ||||||
|                             'Need a filter for yourself? Try out [url=https://mrstallion.github.io/fchat-rising/]F-Chat Rising[/url]'; |  | ||||||
| 
 |  | ||||||
|                         core.connection.send('PRI', {recipient: char.name, message: m}); |  | ||||||
|                         core.cache.markLastPostTime(); |  | ||||||
|                       } |  | ||||||
|                 ); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (cachedProfile && cachedProfile.match.isFiltered) { |  | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         } | 
 | ||||||
|  |         await conversation.addMessage(message); | ||||||
|  |         EventBus.$emit('channel-message', { message, channel: conversation }); | ||||||
| 
 | 
 | ||||||
|         const words = conversation.settings.highlightWords.slice(); |         const words = conversation.settings.highlightWords.slice(); | ||||||
|         if(conversation.settings.defaultHighlights) words.push(...core.state.settings.highlightWords); |         if(conversation.settings.defaultHighlights) words.push(...core.state.settings.highlightWords); | ||||||
|  | |||||||
| @ -26,6 +26,14 @@ | |||||||
| 
 | 
 | ||||||
|         <match-tags v-if="match" :match="match"></match-tags> |         <match-tags v-if="match" :match="match"></match-tags> | ||||||
| 
 | 
 | ||||||
|  |         <div class="filter-matches" v-if="smartFilterIsFiltered"> | ||||||
|  |           <h4>Smart Filter Matches</h4> | ||||||
|  | 
 | ||||||
|  |           <span class="tags"> | ||||||
|  |             <span v-for="filterName in smartFilterDetails" class="smart-filter-tag" :class="filterName">{{ (smartFilterLabels[filterName] || {}).name }}</span> | ||||||
|  |           </span> | ||||||
|  |         </div> | ||||||
|  | 
 | ||||||
| <!--        <div v-if="customs">--> | <!--        <div v-if="customs">--> | ||||||
| <!--          <span v-for="c in customs" :class="Score.getClasses(c.score)">{{c.name}}</span>--> | <!--          <span v-for="c in customs" :class="Score.getClasses(c.score)">{{c.name}}</span>--> | ||||||
| <!--        </div>--> | <!--        </div>--> | ||||||
| @ -72,6 +80,8 @@ import { | |||||||
| import { BBCodeView } from '../../bbcode/view'; | import { BBCodeView } from '../../bbcode/view'; | ||||||
| import { EventBus } from './event-bus'; | import { EventBus } from './event-bus'; | ||||||
| import { Character, CustomKink } from '../../interfaces'; | import { Character, CustomKink } from '../../interfaces'; | ||||||
|  | import { matchesSmartFilters, testSmartFilters } from '../../learn/filter/smart-filter'; | ||||||
|  | import { smartFilterTypes } from '../../learn/filter/types'; | ||||||
| 
 | 
 | ||||||
| interface CustomKinkWithScore extends CustomKink { | interface CustomKinkWithScore extends CustomKink { | ||||||
|   score: number; |   score: number; | ||||||
| @ -97,6 +107,15 @@ export default class CharacterPreview extends Vue { | |||||||
|   latestAd?: AdCachedPosting; |   latestAd?: AdCachedPosting; | ||||||
|   statusMessage?: string; |   statusMessage?: string; | ||||||
| 
 | 
 | ||||||
|  |   smartFilterIsFiltered?: boolean; | ||||||
|  |   smartFilterDetails?: string[]; | ||||||
|  | 
 | ||||||
|  |   smartFilterLabels: Record<string, { name: string }> = { | ||||||
|  |     ...smartFilterTypes, | ||||||
|  |     ageMin: { name: 'Min age' }, | ||||||
|  |     ageMax: { name: 'Max age' } | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|   age?: string; |   age?: string; | ||||||
|   sexualOrientation?: string; |   sexualOrientation?: string; | ||||||
|   species?: string; |   species?: string; | ||||||
| @ -170,6 +189,9 @@ export default class CharacterPreview extends Vue { | |||||||
|     this.customs = undefined; |     this.customs = undefined; | ||||||
|     this.ownCharacter = core.characters.ownProfile; |     this.ownCharacter = core.characters.ownProfile; | ||||||
| 
 | 
 | ||||||
|  |     this.smartFilterIsFiltered = false; | ||||||
|  |     this.smartFilterDetails = []; | ||||||
|  | 
 | ||||||
|     this.updateOnlineStatus(); |     this.updateOnlineStatus(); | ||||||
|     this.updateAdStatus(); |     this.updateAdStatus(); | ||||||
| 
 | 
 | ||||||
| @ -177,11 +199,36 @@ export default class CharacterPreview extends Vue { | |||||||
|       this.character = await this.getCharacterData(characterName); |       this.character = await this.getCharacterData(characterName); | ||||||
|       this.match = Matcher.identifyBestMatchReport(this.ownCharacter!.character, this.character!.character); |       this.match = Matcher.identifyBestMatchReport(this.ownCharacter!.character, this.character!.character); | ||||||
| 
 | 
 | ||||||
|  |       this.updateSmartFilterReport(); | ||||||
|       this.updateCustoms(); |       this.updateCustoms(); | ||||||
|       this.updateDetails(); |       this.updateDetails(); | ||||||
|     }, 0); |     }, 0); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   updateSmartFilterReport() { | ||||||
|  |       if (!this.character) { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       this.smartFilterIsFiltered = matchesSmartFilters(this.character.character, core.state.settings.risingFilter); | ||||||
|  |       this.smartFilterDetails = []; | ||||||
|  | 
 | ||||||
|  |       if (!this.smartFilterIsFiltered) { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       const results = testSmartFilters(this.character.character, core.state.settings.risingFilter); | ||||||
|  | 
 | ||||||
|  |       if (!results) { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       this.smartFilterDetails = [ | ||||||
|  |           ..._.map(_.filter(_.toPairs(results.ageCheck), (v) => v[1]), (v) => v[0]), | ||||||
|  |           ..._.map(_.filter(_.toPairs(results.filters), (v) => v[1].isFiltered), (v: any) => v[0]) | ||||||
|  |       ]; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   updateOnlineStatus(): void { |   updateOnlineStatus(): void { | ||||||
|     this.onlineCharacter = core.characters.get(this.characterName!); |     this.onlineCharacter = core.characters.get(this.characterName!); | ||||||
| 
 | 
 | ||||||
| @ -384,7 +431,8 @@ export default class CharacterPreview extends Vue { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     .status-message, |     .status-message, | ||||||
|     .latest-ad-message { |     .latest-ad-message, | ||||||
|  |     .filter-matches { | ||||||
|       display: block; |       display: block; | ||||||
|       background-color: rgba(0,0,0,0.2); |       background-color: rgba(0,0,0,0.2); | ||||||
|       padding: 10px; |       padding: 10px; | ||||||
| @ -392,6 +440,23 @@ export default class CharacterPreview extends Vue { | |||||||
|       margin-top: 1.3rem; |       margin-top: 1.3rem; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     .filter-matches { | ||||||
|  |       .tags { | ||||||
|  |         margin-top: 10px; | ||||||
|  |         display: block; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       .smart-filter-tag { | ||||||
|  |           display: inline-block; | ||||||
|  |           color: var(--messageTimeFgColor); | ||||||
|  |           margin-right: 4px; | ||||||
|  |           background-color: var(--messageTimeBgColor); | ||||||
|  |           border-radius: 2px; | ||||||
|  |           padding-left: 3px; | ||||||
|  |           padding-right: 3px; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     .character-avatar { |     .character-avatar { | ||||||
|       width: 100%; |       width: 100%; | ||||||
|       height: auto; |       height: auto; | ||||||
|  | |||||||
| @ -50,7 +50,7 @@ theme: jekyll-theme-slate | |||||||
| changelog: https://github.com/mrstallion/fchat-rising/blob/master/CHANGELOG.md | changelog: https://github.com/mrstallion/fchat-rising/blob/master/CHANGELOG.md | ||||||
| 
 | 
 | ||||||
| download: | download: | ||||||
|   version: 1.16.2 |   version: 1.17.0 | ||||||
| 
 | 
 | ||||||
|   url: https://github.com/mrstallion/fchat-rising/releases/download/v%VERSION%/F-Chat-Rising-%VERSION%-%PLATFORM_TAIL% |   url: https://github.com/mrstallion/fchat-rising/releases/download/v%VERSION%/F-Chat-Rising-%VERSION%-%PLATFORM_TAIL% | ||||||
| 
 | 
 | ||||||
| @ -58,14 +58,19 @@ download: | |||||||
|     - type: win |     - type: win | ||||||
|       name: Windows |       name: Windows | ||||||
|       tail: win.exe |       tail: win.exe | ||||||
|       size: 85 MB |       size: 84 MB | ||||||
| 
 | 
 | ||||||
|     - type: mac |     - type: mac | ||||||
|       name: MacOS |       name: MacOS (Intel) | ||||||
|       tail: macos.dmg |       tail: macos-intel.dmg | ||||||
|       size: 82 MB |       size: 80 MB | ||||||
|       instructions: ./macos-install |       instructions: ./macos-install | ||||||
| 
 | 
 | ||||||
|  |     - type: mac | ||||||
|  |       name: MacOS (M1) | ||||||
|  |       tail: macos-m1.dmg | ||||||
|  |       size: 83 MB | ||||||
|  | 
 | ||||||
|     - type: linux |     - type: linux | ||||||
|       name: Linux |       name: Linux | ||||||
|       tail: linux.AppImage |       tail: linux.AppImage | ||||||
|  | |||||||
| @ -53,7 +53,8 @@ require('electron-packager')({ | |||||||
|     icon: path.join(__dirname, 'build', 'icon'), |     icon: path.join(__dirname, 'build', 'icon'), | ||||||
|     ignore: ['\.map$'], |     ignore: ['\.map$'], | ||||||
|     osxSign: process.argv.length > 2 ? {identity: process.argv[2]} : false, |     osxSign: process.argv.length > 2 ? {identity: process.argv[2]} : false, | ||||||
|     prune: false |     prune: false, | ||||||
|  |     arch: process.platform === 'darwin' ? ['x64', 'arm64'] : undefined | ||||||
| }).then((appPaths) => { | }).then((appPaths) => { | ||||||
|     if (process.env.SKIP_INSTALLER) { |     if (process.env.SKIP_INSTALLER) { | ||||||
|         return; |         return; | ||||||
| @ -84,12 +85,16 @@ require('electron-packager')({ | |||||||
|         }).catch((e) => console.error(`Error while creating installer: ${e.message}`)); |         }).catch((e) => console.error(`Error while creating installer: ${e.message}`)); | ||||||
|     } else if(process.platform === 'darwin') { |     } else if(process.platform === 'darwin') { | ||||||
|         console.log('Creating Mac DMG'); |         console.log('Creating Mac DMG'); | ||||||
|         const target = path.join(distDir, `F-Chat Rising.dmg`); | 
 | ||||||
|  |         _.each([{ name: 'Intel', path: appPaths[0] }, { name: 'M1', path: appPaths[1] }], (arch) => { | ||||||
|  |             console.log(arch.name, arch.path); | ||||||
|  | 
 | ||||||
|  |             const target = path.join(distDir, `F-Chat Rising ${arch.name}.dmg`); | ||||||
|             if(fs.existsSync(target)) fs.unlinkSync(target); |             if(fs.existsSync(target)) fs.unlinkSync(target); | ||||||
|         const appPath = path.join(appPaths[0], 'F-Chat.app'); |             const appPath = path.join(arch.path, 'F-Chat.app'); | ||||||
|             if(process.argv.length <= 2) console.warn('Warning: Creating unsigned DMG'); |             if(process.argv.length <= 2) console.warn('Warning: Creating unsigned DMG'); | ||||||
|             require('appdmg')({ |             require('appdmg')({ | ||||||
|             basepath: appPaths[0], |                 basepath: arch.path, | ||||||
|                 target, |                 target, | ||||||
|                 specification: { |                 specification: { | ||||||
|                     title: 'F-Chat Rising', |                     title: 'F-Chat Rising', | ||||||
| @ -101,16 +106,17 @@ require('electron-packager')({ | |||||||
|                     } : undefined |                     } : undefined | ||||||
|                 } |                 } | ||||||
|             }).on('error', console.error); |             }).on('error', console.error); | ||||||
|         const zipName = `F-Chat_Rising_${pkg.version}.zip`; |             const zipName = `F-Chat_Rising_${arch.name}_${pkg.version}.zip`; | ||||||
|             const zipPath = path.join(distDir, zipName); |             const zipPath = path.join(distDir, zipName); | ||||||
|             if(fs.existsSync(zipPath)) fs.unlinkSync(zipPath); |             if(fs.existsSync(zipPath)) fs.unlinkSync(zipPath); | ||||||
|         const child = child_process.spawn('zip', ['-r', '-y', '-9', zipPath, 'F-Chat.app'], {cwd: appPaths[0]}); |             const child = child_process.spawn('zip', ['-r', '-y', '-9', zipPath, 'F-Chat.app'], {cwd: arch.path}); | ||||||
|             child.stdout.on('data', () => {}); |             child.stdout.on('data', () => {}); | ||||||
|             child.stderr.on('data', (data) => console.error(data.toString())); |             child.stderr.on('data', (data) => console.error(data.toString())); | ||||||
|             fs.writeFileSync(path.join(distDir, 'updates.json'), JSON.stringify({ |             fs.writeFileSync(path.join(distDir, 'updates.json'), JSON.stringify({ | ||||||
|                 releases: [{version: pkg.version, updateTo: {url: 'https://client.f-list.net/darwin/' + zipName}}], |                 releases: [{version: pkg.version, updateTo: {url: 'https://client.f-list.net/darwin/' + zipName}}], | ||||||
|                 currentRelease: pkg.version |                 currentRelease: pkg.version | ||||||
|             })); |             })); | ||||||
|  |         }); | ||||||
|     } else { |     } else { | ||||||
|         console.log('Creating Linux AppImage'); |         console.log('Creating Linux AppImage'); | ||||||
|         fs.renameSync(path.join(appPaths[0], 'F-Chat'), path.join(appPaths[0], 'AppRun')); |         fs.renameSync(path.join(appPaths[0], 'F-Chat'), path.join(appPaths[0], 'AppRun')); | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "fchat", |   "name": "fchat", | ||||||
|   "version": "1.16.2", |   "version": "1.17.0", | ||||||
|   "author": "The F-List Team and Mister Stallion (Esq.)", |   "author": "The F-List Team and Mister Stallion (Esq.)", | ||||||
|   "description": "F-List.net Chat Client", |   "description": "F-List.net Chat Client", | ||||||
|   "main": "main.js", |   "main": "main.js", | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| import _ from 'lodash'; | import _ from 'lodash'; | ||||||
| import { Matcher } from '../matcher'; | import { Matcher } from '../matcher'; | ||||||
| import { BodyType, Build, Kink, Species, TagId } from '../matcher-types'; | import { BodyType, Build, Gender, Kink, Species, TagId } from '../matcher-types'; | ||||||
| import { SmartFilterSelection, SmartFilterSettings } from './types'; | import { SmartFilterSelection, SmartFilterSettings } from './types'; | ||||||
| import { Character } from '../../interfaces'; | import { Character } from '../../interfaces'; | ||||||
| import log from 'electron-log'; | import log from 'electron-log'; | ||||||
| @ -8,32 +8,44 @@ import core from '../../chat/core'; | |||||||
| 
 | 
 | ||||||
| export interface SmartFilterOpts { | export interface SmartFilterOpts { | ||||||
|   name: string; |   name: string; | ||||||
|   kinks?: Kink[], |   kinks?: Kink[]; | ||||||
|   bodyTypes?: BodyType[], |   bodyTypes?: BodyType[]; | ||||||
|   builds?: Build[], |   builds?: Build[]; | ||||||
|   species?: Species[] |   species?: Species[]; | ||||||
|  |   genders?: Gender[]; | ||||||
|   isAnthro?: boolean; |   isAnthro?: boolean; | ||||||
|   isHuman?: boolean; |   isHuman?: boolean; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | export interface SmartFilterTestResult { | ||||||
|  |   isFiltered: boolean; | ||||||
|  |   builds: boolean; | ||||||
|  |   bodyTypes: boolean; | ||||||
|  |   species: boolean; | ||||||
|  |   genders: boolean; | ||||||
|  |   isAnthro: boolean; | ||||||
|  |   isHuman: boolean; | ||||||
|  |   kinks: boolean; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| export class SmartFilter { | export class SmartFilter { | ||||||
|   constructor(private opts: SmartFilterOpts) {} |   constructor(private opts: SmartFilterOpts) {} | ||||||
| 
 | 
 | ||||||
|   test(c: Character): boolean { |   test(c: Character): SmartFilterTestResult { | ||||||
|     const builds = this.testBuilds(c); |     const builds = this.testBuilds(c); | ||||||
|     const bodyTypes = this.testBodyTypes(c); |     const bodyTypes = this.testBodyTypes(c); | ||||||
|     const species = this.testSpecies(c); |     const species = this.testSpecies(c); | ||||||
|     const isAnthro = this.testIsAnthro(c); |     const isAnthro = this.testIsAnthro(c); | ||||||
|     const isHuman = this.testIsHuman(c); |     const isHuman = this.testIsHuman(c); | ||||||
|     const kinks = this.testKinks(c); |     const kinks = this.testKinks(c); | ||||||
|  |     const genders = this.testGenders(c); | ||||||
| 
 | 
 | ||||||
|     const result = builds || bodyTypes || species || isAnthro || isHuman || kinks; |     const isFiltered = builds || bodyTypes || species || isAnthro || isHuman || kinks || genders; | ||||||
|  |     const result = { isFiltered, builds, bodyTypes, species, isAnthro, isHuman, kinks, genders }; | ||||||
| 
 | 
 | ||||||
|     log.debug('smart-filter.test', |     log.silly('smart-filter.test', { name: c.name, filterName: this.opts.name, result }); | ||||||
|       { name: c.name, filterName: this.opts.name, result, builds, bodyTypes, species, isAnthro, isHuman, kinks }); |  | ||||||
| 
 | 
 | ||||||
|     return this.testBuilds(c) || this.testBodyTypes(c) || this.testSpecies(c) || this.testIsAnthro(c) || this.testIsHuman(c) || |     return result; | ||||||
|       this.testKinks(c); |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   testKinks(c: Character): boolean { |   testKinks(c: Character): boolean { | ||||||
| @ -65,6 +77,16 @@ export class SmartFilter { | |||||||
|     return !!build && !!_.find(this.opts.builds || [], build); |     return !!build && !!_.find(this.opts.builds || [], build); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   testGenders(c: Character): boolean { | ||||||
|  |     if (!this.opts.genders) { | ||||||
|  |       return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const gender = Matcher.getTagValueList(TagId.Gender, c); | ||||||
|  | 
 | ||||||
|  |     return !!gender && !!_.find(this.opts.genders || [], gender); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   testBodyTypes(c: Character): boolean { |   testBodyTypes(c: Character): boolean { | ||||||
|     if (!this.opts.bodyTypes) { |     if (!this.opts.bodyTypes) { | ||||||
|       return false; |       return false; | ||||||
| @ -94,11 +116,11 @@ export class SmartFilter { | |||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| export type SmartFilterCollection = { | export type SmartFilterCollection = { | ||||||
|   [key in keyof SmartFilterSelection]: SmartFilter; |   [key in keyof SmartFilterSelection]: SmartFilter; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| export const smartFilters: SmartFilterCollection = { | export const smartFilters: SmartFilterCollection = { | ||||||
|   ageplay: new SmartFilter({ |   ageplay: new SmartFilter({ | ||||||
|     name: 'ageplay', |     name: 'ageplay', | ||||||
| @ -110,6 +132,11 @@ export const smartFilters: SmartFilterCollection = { | |||||||
|     isAnthro: true |     isAnthro: true | ||||||
|   }), |   }), | ||||||
| 
 | 
 | ||||||
|  |   female: new SmartFilter({ | ||||||
|  |     name: 'female', | ||||||
|  |     genders: [Gender.Female] | ||||||
|  |   }), | ||||||
|  | 
 | ||||||
|   feral: new SmartFilter({ |   feral: new SmartFilter({ | ||||||
|     name: 'feral', |     name: 'feral', | ||||||
|     bodyTypes: [BodyType.Feral] |     bodyTypes: [BodyType.Feral] | ||||||
| @ -140,6 +167,16 @@ export const smartFilters: SmartFilterCollection = { | |||||||
|     kinks: [Kink.Incest, Kink.IncestParental, Kink.IncestSiblings, Kink.ParentChildPlay, Kink.ForcedIncest] |     kinks: [Kink.Incest, Kink.IncestParental, Kink.IncestSiblings, Kink.ParentChildPlay, Kink.ForcedIncest] | ||||||
|   }), |   }), | ||||||
| 
 | 
 | ||||||
|  |   intersex: new SmartFilter({ | ||||||
|  |     name: 'intersex', | ||||||
|  |     genders: [Gender.Transgender, Gender.Herm, Gender.MaleHerm, Gender.Cuntboy, Gender.Shemale] | ||||||
|  |   }), | ||||||
|  | 
 | ||||||
|  |   male: new SmartFilter({ | ||||||
|  |     name: 'male', | ||||||
|  |     genders: [Gender.Male] | ||||||
|  |   }), | ||||||
|  | 
 | ||||||
|   microMacro: new SmartFilter({ |   microMacro: new SmartFilter({ | ||||||
|     name: 'microMacro', |     name: 'microMacro', | ||||||
|     kinks: [Kink.MacroAsses, Kink.MacroBalls, Kink.MacroBreasts, Kink.MacroCocks, Kink.Macrophilia, Kink.MegaMacro, Kink.Microphilia, |     kinks: [Kink.MacroAsses, Kink.MacroBalls, Kink.MacroBreasts, Kink.MacroCocks, Kink.Macrophilia, Kink.MegaMacro, Kink.Microphilia, | ||||||
| @ -205,31 +242,59 @@ export const smartFilters: SmartFilterCollection = { | |||||||
|   }) |   }) | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| 
 | export function testSmartFilters(c: Character, opts: SmartFilterSettings): { | ||||||
| export function matchesSmartFilters(c: Character, opts: SmartFilterSettings): boolean { |   ageCheck: { ageMin: boolean; ageMax: boolean }; | ||||||
|  |   filters: { [key in keyof SmartFilterCollection]: SmartFilterTestResult } | ||||||
|  | } | null { | ||||||
|   if (c.name === core.characters.ownCharacter.name) { |   if (c.name === core.characters.ownCharacter.name) { | ||||||
|     return false; |     return null; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   if (core.characters.get(c.name)?.isChatOp) { |   const coreCharacter = core.characters.get(c.name); | ||||||
|     return false; | 
 | ||||||
|  |   if (coreCharacter?.isChatOp || coreCharacter?.isBookmarked || coreCharacter?.isFriend) { | ||||||
|  |     return null; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   if (opts.exceptionNames.includes(c.name)) { |   if (opts.exceptionNames.includes(c.name)) { | ||||||
|     log.debug('smart-filter.exception', { name: c.name }); |     log.debug('smart-filter.exception', { name: c.name }); | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const ageCheck = { ageMin: false, ageMax: false }; | ||||||
|  | 
 | ||||||
|  |   if (opts.minAge !== null || opts.maxAge !== null) { | ||||||
|  |     const age = Matcher.age(c) || Matcher.apparentAge(c)?.min || null; | ||||||
|  | 
 | ||||||
|  |     if (age !== null) { | ||||||
|  |       if (opts.minAge !== null && age < opts.minAge) { | ||||||
|  |         log.debug('smart-filter.age.min', { name: c.name, age, minAge: opts.minAge }); | ||||||
|  |         ageCheck.ageMin = true; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       if (opts.maxAge !== null && age > opts.maxAge) { | ||||||
|  |         log.debug('smart-filter.age.max', { name: c.name, age, maxAge: opts.maxAge }); | ||||||
|  |         ageCheck.ageMax = true; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return { | ||||||
|  |     ageCheck, | ||||||
|  |     filters: _.mapValues(smartFilters, (f, k) => (opts.smartFilters as any)[k] && f.test(c)) | ||||||
|  |   }; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function matchesSmartFilters(c: Character, opts: SmartFilterSettings): boolean { | ||||||
|  |   const match = testSmartFilters(c, opts); | ||||||
|  | 
 | ||||||
|  |   if (!match) { | ||||||
|     return false; |     return false; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   if (opts.minAge !== null || opts.maxAge !== null) { |   if (match.ageCheck.ageMax || match.ageCheck.ageMin) { | ||||||
|     const age = Matcher.age(c); |  | ||||||
| 
 |  | ||||||
|     if (age !== null) { |  | ||||||
|       if ((opts.minAge !== null && age < opts.minAge) || (opts.maxAge !== null && age > opts.maxAge)) { |  | ||||||
|         log.debug('smart-filter.age', { name: c.name, age, minAge: opts.minAge, maxAge: opts.maxAge }); |  | ||||||
|     return true; |     return true; | ||||||
|   } |   } | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 | 
 | ||||||
|   return !_.every(opts.smartFilters, (fs, filterName) => !fs || !(smartFilters as any)[filterName].test(c)); |   return !_.every(match.filters, (filterResult) => !filterResult.isFiltered); | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,15 +1,14 @@ | |||||||
| // <!--                [Automated message] Sorry, the player of this character has indicated that they are not interested in characters matching your profile.-->
 |  | ||||||
| // <!--                Need a filter for yourself? Try out [F-Chat Rising](https://mrstallion.github.io/fchat-rising/)!-->
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| export const smartFilterTypes = { | export const smartFilterTypes = { | ||||||
|   ageplay: { name: 'Ageplay' }, |   ageplay: { name: 'Ageplay' }, | ||||||
|   anthro: { name: 'Anthros' }, |   anthro: { name: 'Anthros' }, | ||||||
|  |   female: { name: 'Females' }, | ||||||
|   feral: { name: 'Ferals' }, |   feral: { name: 'Ferals' }, | ||||||
|   gore: { name: 'Gore/torture/death' }, |   gore: { name: 'Gore/torture/death' }, | ||||||
|   human: { name: 'Humans' }, |   human: { name: 'Humans' }, | ||||||
|   hyper: { name: 'Hyper' }, |   hyper: { name: 'Hyper' }, | ||||||
|   incest: { name: 'Incest' }, |   incest: { name: 'Incest' }, | ||||||
|  |   intersex: { name: 'Intersex' }, | ||||||
|  |   male: { name: 'Males' }, | ||||||
|   microMacro: { name: 'Micro/macro' }, |   microMacro: { name: 'Micro/macro' }, | ||||||
|   obesity: { name: 'Obesity' }, |   obesity: { name: 'Obesity' }, | ||||||
|   pokemon: { name: 'Pokemons/Digimons' }, |   pokemon: { name: 'Pokemons/Digimons' }, | ||||||
| @ -36,7 +35,9 @@ export interface SmartFilterSettings { | |||||||
|   hidePrivateChannelMessages: boolean; |   hidePrivateChannelMessages: boolean; | ||||||
|   hidePrivateMessages: boolean; |   hidePrivateMessages: boolean; | ||||||
|   penalizeMatches: boolean; |   penalizeMatches: boolean; | ||||||
|  |   rewardNonMatches: boolean; | ||||||
|   autoReply: boolean; |   autoReply: boolean; | ||||||
|  |   showFilterIcon: boolean; | ||||||
| 
 | 
 | ||||||
|   minAge: number | null; |   minAge: number | null; | ||||||
|   maxAge: number | null; |   maxAge: number | null; | ||||||
|  | |||||||
| @ -1237,11 +1237,59 @@ export class Matcher { | |||||||
| 
 | 
 | ||||||
|     static age(c: Character): number | null { |     static age(c: Character): number | null { | ||||||
|         const rawAge = Matcher.getTagValue(TagId.Age, c); |         const rawAge = Matcher.getTagValue(TagId.Age, c); | ||||||
|         const age = ((rawAge) && (rawAge.string)) ? parseInt(rawAge.string, 10) : null; | 
 | ||||||
|  |         if (!rawAge || !rawAge.string) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         const ageStr = rawAge.string.toLowerCase().trim(); | ||||||
|  | 
 | ||||||
|  |         if ((ageStr.indexOf('shota') >= 0) || (ageStr.indexOf('loli') >= 0) || (ageStr.indexOf('lolli') >= 0)) { | ||||||
|  |             return 10; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         const age = parseInt(rawAge.string, 10); | ||||||
| 
 | 
 | ||||||
|         return age && !Number.isNaN(age) && Number.isFinite(age) ? age : null; |         return age && !Number.isNaN(age) && Number.isFinite(age) ? age : null; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     static apparentAge(c: Character): { min: number, max: number } | null { | ||||||
|  |         const rawAge = Matcher.getTagValue(TagId.ApparentAge, c); | ||||||
|  | 
 | ||||||
|  |         if ((!rawAge) || (!rawAge.string)) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         const ageStr = rawAge.string.trim().toLowerCase(); | ||||||
|  | 
 | ||||||
|  |         if (ageStr === '') { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // '18'
 | ||||||
|  |         if (/^[0-9]+$/.exec(ageStr)) { | ||||||
|  |             const val = parseInt(rawAge.string, 10); | ||||||
|  | 
 | ||||||
|  |             return { min: val, max: val }; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // '18-22'
 | ||||||
|  |         const rangeMatch = ageStr.match(/^([0-9]+)-([0-9]+)$/); | ||||||
|  | 
 | ||||||
|  |         if (rangeMatch) { | ||||||
|  |             const v1 = parseInt(rangeMatch[1], 10); | ||||||
|  |             const v2 = parseInt(rangeMatch[2], 10); | ||||||
|  | 
 | ||||||
|  |             return { min: Math.min(v1, v2), max: Math.max(v1, v2) }; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if ((ageStr.indexOf('shota') >= 0) || (ageStr.indexOf('loli') >= 0) || (ageStr.indexOf('lolli') >= 0)) { | ||||||
|  |             return { min: 10, max: 10 }; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     static calculateSearchScoreForMatch( |     static calculateSearchScoreForMatch( | ||||||
|         score: Scoring, |         score: Scoring, | ||||||
|         match: MatchReport, |         match: MatchReport, | ||||||
|  | |||||||
| @ -161,10 +161,11 @@ export class ProfileCache extends AsyncCache<CharacterCacheRecord> { | |||||||
|         // const totalScoreDimensions = match ? Matcher.countScoresTotal(match) : 0;
 |         // const totalScoreDimensions = match ? Matcher.countScoresTotal(match) : 0;
 | ||||||
|         // const dimensionsAtScoreLevel = match ? (Matcher.countScoresAtLevel(match, score) || 0) : 0;
 |         // const dimensionsAtScoreLevel = match ? (Matcher.countScoresAtLevel(match, score) || 0) : 0;
 | ||||||
|         // const dimensionsAboveScoreLevel = match ? (Matcher.countScoresAboveLevel(match, Math.max(score, Scoring.WEAK_MATCH))) : 0;
 |         // const dimensionsAboveScoreLevel = match ? (Matcher.countScoresAboveLevel(match, Math.max(score, Scoring.WEAK_MATCH))) : 0;
 | ||||||
|         const isFiltered = matchesSmartFilters(c.character, core.state.settings.risingFilter); |         const risingFilter = core.state.settings.risingFilter; | ||||||
|  |         const isFiltered = matchesSmartFilters(c.character, risingFilter); | ||||||
| 
 | 
 | ||||||
|         const searchScore = match |         const searchScore = match | ||||||
|             ? Matcher.calculateSearchScoreForMatch(score, match, isFiltered && core.state.settings.risingFilter.penalizeMatches ? -2 : 0) |             ? Matcher.calculateSearchScoreForMatch(score, match, (isFiltered && risingFilter.penalizeMatches) ? -2 : (!isFiltered && risingFilter.rewardNonMatches) ? 1 : 0) | ||||||
|             : 0; |             : 0; | ||||||
| 
 | 
 | ||||||
|         const matchDetails = { matchScore: score, searchScore, isFiltered }; |         const matchDetails = { matchScore: score, searchScore, isFiltered }; | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "f-list-rising", |   "name": "f-list-rising", | ||||||
|   "version": "1.16.2", |   "version": "1.17.0", | ||||||
|   "author": "The F-List Team and and Mister Stallion (Esq.)", |   "author": "The F-List Team and and Mister Stallion (Esq.)", | ||||||
|   "description": "A heavily modded F-Chat 3.0 client for F-List", |   "description": "A heavily modded F-Chat 3.0 client for F-List", | ||||||
|   "license": "MIT", |   "license": "MIT", | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user