1. 1 : import { Logger } from '../core/Logger.js';
  2. 2 : import { FuzzyRule } from './FuzzyRule.js';
  3. 3 : import { FuzzyVariable } from './FuzzyVariable.js';
  4. 4 :
  5. 5 : /**
  6. 6 : * Class for representing a fuzzy module. Instances of this class are used by
  7. 7 : * game entities for fuzzy inference. A fuzzy module is a collection of fuzzy variables
  8. 8 : * and the rules that operate on them.
  9. 9 : *
  10. 10 : * @author {@link https://github.com/Mugen87|Mugen87}
  11. 11 : */
  12. 12 : class FuzzyModule {
  13. 13 :
  14. 14 : /**
  15. 15 : * Constructs a new fuzzy module.
  16. 16 : */
  17. 17 : constructor() {
  18. 18 :
  19. 19 : /**
  20. 20 : * An array of the fuzzy rules.
  21. 21 : * @type {Array<FuzzyRule>}
  22. 22 : * @readonly
  23. 23 : */
  24. 24 : this.rules = new Array();
  25. 25 :
  26. 26 : /**
  27. 27 : * A map of FLVs.
  28. 28 : * @type {Map<String,FuzzyVariable>}
  29. 29 : * @readonly
  30. 30 : */
  31. 31 : this.flvs = new Map();
  32. 32 :
  33. 33 : }
  34. 34 :
  35. 35 : /**
  36. 36 : * Adds the given FLV under the given name to this fuzzy module.
  37. 37 : *
  38. 38 : * @param {String} name - The name of the FLV.
  39. 39 : * @param {FuzzyVariable} flv - The FLV to add.
  40. 40 : * @return {FuzzyModule} A reference to this fuzzy module.
  41. 41 : */
  42. 42 : addFLV( name, flv ) {
  43. 43 :
  44. 44 : this.flvs.set( name, flv );
  45. 45 :
  46. 46 : return this;
  47. 47 :
  48. 48 : }
  49. 49 :
  50. 50 : /**
  51. 51 : * Remove the FLV under the given name from this fuzzy module.
  52. 52 : *
  53. 53 : * @param {String} name - The name of the FLV to remove.
  54. 54 : * @return {FuzzyModule} A reference to this fuzzy module.
  55. 55 : */
  56. 56 : removeFLV( name ) {
  57. 57 :
  58. 58 : this.flvs.delete( name );
  59. 59 :
  60. 60 : return this;
  61. 61 :
  62. 62 : }
  63. 63 :
  64. 64 : /**
  65. 65 : * Adds the given fuzzy rule to this fuzzy module.
  66. 66 : *
  67. 67 : * @param {FuzzyRule} rule - The fuzzy rule to add.
  68. 68 : * @return {FuzzyModule} A reference to this fuzzy module.
  69. 69 : */
  70. 70 : addRule( rule ) {
  71. 71 :
  72. 72 : this.rules.push( rule );
  73. 73 :
  74. 74 : return this;
  75. 75 :
  76. 76 : }
  77. 77 :
  78. 78 : /**
  79. 79 : * Removes the given fuzzy rule from this fuzzy module.
  80. 80 : *
  81. 81 : * @param {FuzzyRule} rule - The fuzzy rule to remove.
  82. 82 : * @return {FuzzyModule} A reference to this fuzzy module.
  83. 83 : */
  84. 84 : removeRule( rule ) {
  85. 85 :
  86. 86 : const rules = this.rules;
  87. 87 :
  88. 88 : const index = rules.indexOf( rule );
  89. 89 : rules.splice( index, 1 );
  90. 90 :
  91. 91 : return this;
  92. 92 :
  93. 93 : }
  94. 94 :
  95. 95 : /**
  96. 96 : * Calls the fuzzify method of the defined FLV with the given value.
  97. 97 : *
  98. 98 : * @param {String} name - The name of the FLV
  99. 99 : * @param {Number} value - The crips value to fuzzify.
  100. 100 : * @return {FuzzyModule} A reference to this fuzzy module.
  101. 101 : */
  102. 102 : fuzzify( name, value ) {
  103. 103 :
  104. 104 : const flv = this.flvs.get( name );
  105. 105 :
  106. 106 : flv.fuzzify( value );
  107. 107 :
  108. 108 : return this;
  109. 109 :
  110. 110 : }
  111. 111 :
  112. 112 : /**
  113. 113 : * Given a fuzzy variable and a defuzzification method this returns a crisp value.
  114. 114 : *
  115. 115 : * @param {String} name - The name of the FLV
  116. 116 : * @param {String} type - The type of defuzzification.
  117. 117 : * @return {Number} The defuzzified, crips value.
  118. 118 : */
  119. 119 : defuzzify( name, type = FuzzyModule.DEFUZ_TYPE.MAXAV ) {
  120. 120 :
  121. 121 : const flvs = this.flvs;
  122. 122 : const rules = this.rules;
  123. 123 :
  124. 124 : this._initConsequences();
  125. 125 :
  126. 126 : for ( let i = 0, l = rules.length; i < l; i ++ ) {
  127. 127 :
  128. 128 : const rule = rules[ i ];
  129. 129 :
  130. 130 : rule.evaluate();
  131. 131 :
  132. 132 : }
  133. 133 :
  134. 134 : const flv = flvs.get( name );
  135. 135 :
  136. 136 : let value;
  137. 137 :
  138. 138 : switch ( type ) {
  139. 139 :
  140. 140 : case FuzzyModule.DEFUZ_TYPE.MAXAV:
  141. 141 : value = flv.defuzzifyMaxAv();
  142. 142 : break;
  143. 143 :
  144. 144 : case FuzzyModule.DEFUZ_TYPE.CENTROID:
  145. 145 : value = flv.defuzzifyCentroid();
  146. 146 : break;
  147. 147 :
  148. 148 : default:
  149. 149 : Logger.warn( 'YUKA.FuzzyModule: Unknown defuzzification method:', type );
  150. 150 : value = flv.defuzzifyMaxAv(); // use MaxAv as fallback
  151. 151 :
  152. 152 : }
  153. 153 :
  154. 154 : return value;
  155. 155 :
  156. 156 : }
  157. 157 :
  158. 158 : _initConsequences() {
  159. 159 :
  160. 160 : const rules = this.rules;
  161. 161 :
  162. 162 : // initializes the consequences of all rules.
  163. 163 :
  164. 164 : for ( let i = 0, l = rules.length; i < l; i ++ ) {
  165. 165 :
  166. 166 : const rule = rules[ i ];
  167. 167 :
  168. 168 : rule.initConsequence();
  169. 169 :
  170. 170 : }
  171. 171 :
  172. 172 : return this;
  173. 173 :
  174. 174 : }
  175. 175 :
  176. 176 : /**
  177. 177 : * Transforms this instance into a JSON object.
  178. 178 : *
  179. 179 : * @return {Object} The JSON object.
  180. 180 : */
  181. 181 : toJSON() {
  182. 182 :
  183. 183 : const json = {
  184. 184 : rules: new Array(),
  185. 185 : flvs: new Array()
  186. 186 : };
  187. 187 :
  188. 188 : // rules
  189. 189 :
  190. 190 : const rules = this.rules;
  191. 191 :
  192. 192 : for ( let i = 0, l = rules.length; i < l; i ++ ) {
  193. 193 :
  194. 194 : json.rules.push( rules[ i ].toJSON() );
  195. 195 :
  196. 196 : }
  197. 197 :
  198. 198 : // flvs
  199. 199 :
  200. 200 : const flvs = this.flvs;
  201. 201 :
  202. 202 : for ( let [ name, flv ] of flvs ) {
  203. 203 :
  204. 204 : json.flvs.push( { name: name, flv: flv.toJSON() } );
  205. 205 :
  206. 206 : }
  207. 207 :
  208. 208 : return json;
  209. 209 :
  210. 210 : }
  211. 211 :
  212. 212 : /**
  213. 213 : * Restores this instance from the given JSON object.
  214. 214 : *
  215. 215 : * @param {Object} json - The JSON object.
  216. 216 : * @return {FuzzyModule} A reference to this fuzzy module.
  217. 217 : */
  218. 218 : fromJSON( json ) {
  219. 219 :
  220. 220 : const fuzzySets = new Map(); // used for rules
  221. 221 :
  222. 222 : // flvs
  223. 223 :
  224. 224 : const flvsJSON = json.flvs;
  225. 225 :
  226. 226 : for ( let i = 0, l = flvsJSON.length; i < l; i ++ ) {
  227. 227 :
  228. 228 : const flvJSON = flvsJSON[ i ];
  229. 229 : const name = flvJSON.name;
  230. 230 : const flv = new FuzzyVariable().fromJSON( flvJSON.flv );
  231. 231 :
  232. 232 : this.addFLV( name, flv );
  233. 233 :
  234. 234 : for ( let fuzzySet of flv.fuzzySets ) {
  235. 235 :
  236. 236 : fuzzySets.set( fuzzySet.uuid, fuzzySet );
  237. 237 :
  238. 238 : }
  239. 239 :
  240. 240 : }
  241. 241 :
  242. 242 : // rules
  243. 243 :
  244. 244 : const rulesJSON = json.rules;
  245. 245 :
  246. 246 : for ( let i = 0, l = rulesJSON.length; i < l; i ++ ) {
  247. 247 :
  248. 248 : const ruleJSON = rulesJSON[ i ];
  249. 249 : const rule = new FuzzyRule().fromJSON( ruleJSON, fuzzySets );
  250. 250 :
  251. 251 : this.addRule( rule );
  252. 252 :
  253. 253 : }
  254. 254 :
  255. 255 : return this;
  256. 256 :
  257. 257 : }
  258. 258 :
  259. 259 : }
  260. 260 :
  261. 261 : FuzzyModule.DEFUZ_TYPE = Object.freeze( {
  262. 262 : MAXAV: 0,
  263. 263 : CENTROID: 1
  264. 264 : } );
  265. 265 :
  266. 266 : export { FuzzyModule };