Home Reference Source

src/jg/character.js

  1. import _ from 'lodash';
  2. import qualities from "./qualities/index";
  3.  
  4. const _prioritySort = ({priority}) => -priority || 0;
  5. const _groupOmitKeys = ['id', 'name', 'priority', 'hidden'];
  6.  
  7. /**
  8. * This class is created from the character object you specify in your game.
  9. */
  10. export default class Character {
  11. /**
  12. *
  13. * @param {object} args
  14. * @param {string} args.id
  15. * @param {string} args.name
  16. * @param {number} args.priority
  17. * @param {Boolean} args.showInSidebar
  18. * @param {string|function(): string} args.description Currently unused
  19. * @param {quality[]} args.qualities
  20. * @param {*} args.state Initial value of {@link state}
  21. */
  22. constructor({qualities, id, name, priority = 0, showInSidebar = true, description = '', state = {}}) {
  23. /**
  24. * The ID you specified for this character. Must be unique across all characters.
  25. * @type {string}
  26. */
  27. this.id = id;
  28.  
  29. /**
  30. * The name you specified for this character.
  31. * @type {string}
  32. */
  33. this.name = name;
  34.  
  35. /**
  36. * Arbitrary, JSON-safe data about this character. You may update it any time you want.
  37. * @type {*}
  38. */
  39. this.state = _.cloneDeep(state);
  40.  
  41. Object.assign(this, {qualities, description, showInSidebar, priority});
  42.  
  43. this.updateQualities()
  44. }
  45.  
  46. /** @ignore */
  47. updateQualities() {
  48. this._shallowQualities = {};
  49. Object.keys(this.qualities).forEach((k) => {
  50. const group = this.qualities[k];
  51. group.id = k;
  52. for (const k2 of _.keys(_.omit(group, _groupOmitKeys))) {
  53. if (this._shallowQualities[k2]) {
  54. throw Error("You have two qualities with the same ID. Please don't do that.")
  55. }
  56. group[k2].id = k2;
  57. this._shallowQualities[k2] = group[k2];
  58. if (group[k2].value === undefined) group[k2].value = group[k2].initialValue;
  59. }
  60. });
  61. /** @ignore */
  62. this.sortedQualityGroups = _.sortBy(Object.values(this.qualities), _prioritySort);
  63. }
  64.  
  65. /** @ignore */
  66. toSave() {
  67. return _.pick(this, ['id', 'qualities', 'name', 'showInSidebar', 'description', 'state']);
  68. }
  69.  
  70. /** @ignore */
  71. loadSave(obj) {
  72. _.assign(this, obj);
  73. this.updateQualities();
  74. }
  75.  
  76. /** @ignore */
  77. getDescription() {
  78. if (_.isFunction(this.description)) {
  79. return this.description.apply(this, arguments);
  80. } else {
  81. return this.description || this.id;
  82. }
  83. }
  84.  
  85. /** @ignore */
  86. sortedQualities(groupName) {
  87. return _.sortBy(Object.values(_.omit(this.qualities[groupName], _groupOmitKeys)), _prioritySort);
  88. }
  89.  
  90. /**
  91. * Return the current value of this quality (without formatting it)
  92. *
  93. * @param {string} id ID of the quality whose value you want
  94. * @returns {*}
  95. */
  96. getQuality(id) {
  97. return this._shallowQualities[id].value;
  98. }
  99.  
  100. /**
  101. * Return the _initial_ value of this quality from the start of the game(without formatting it).
  102. * You can use this to see how the value has changed since the start of the game.
  103. *
  104. * @param {string} id ID of the quality whose value you want
  105. * @returns {*}
  106. */
  107. getQualityInitial(id) {
  108. return this._shallowQualities[id].initialValue;
  109. }
  110.  
  111. /**
  112. * Returns the formatted value of the given quality.
  113. *
  114. * @param {string} id ID of the quality whose value you want
  115. * @returns {string}
  116. */
  117. formatQuality(id) {
  118. const quality = this._shallowQualities[id];
  119. if (!qualities[quality.type]) {
  120. console.error("Undefined quality type:", quality.type);
  121. return '';
  122. }
  123. return qualities[quality.type].format(this, quality, quality.value);
  124. }
  125.  
  126. /**
  127. * Return the human-readable name for the given quality ID.
  128. *
  129. * @param {string} id ID of the quality whose value you want
  130. * @returns {string}
  131. */
  132. formatQualityName(id) {
  133. const quality = this._shallowQualities[id];
  134. return quality.name;
  135. }
  136.  
  137. /**
  138. * Qualities **must** be modified using this method.
  139. *
  140. * @param {string} id ID of the quality to modify on this character
  141. * @param {*} value New value for the quality
  142. */
  143. setQuality(id, value) {
  144. this._shallowQualities[id].value = value;
  145. }
  146.  
  147. /**
  148. * Add an integer value to the given quality.
  149. *
  150. * @param {string} id ID of the quality to modify on this character
  151. * @param {number} delta Amount to add
  152. * @returns {number} The new value
  153. */
  154. addToQuality(id, delta) {
  155. return this._shallowQualities[id].value += delta;
  156. }
  157. }