Brak opisu

oneof.js 5.6KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. "use strict";
  2. module.exports = OneOf;
  3. // extends ReflectionObject
  4. var ReflectionObject = require("./object");
  5. ((OneOf.prototype = Object.create(ReflectionObject.prototype)).constructor = OneOf).className = "OneOf";
  6. var Field = require("./field"),
  7. util = require("./util");
  8. /**
  9. * Constructs a new oneof instance.
  10. * @classdesc Reflected oneof.
  11. * @extends ReflectionObject
  12. * @constructor
  13. * @param {string} name Oneof name
  14. * @param {string[]|Object.<string,*>} [fieldNames] Field names
  15. * @param {Object.<string,*>} [options] Declared options
  16. * @param {string} [comment] Comment associated with this field
  17. */
  18. function OneOf(name, fieldNames, options, comment) {
  19. if (!Array.isArray(fieldNames)) {
  20. options = fieldNames;
  21. fieldNames = undefined;
  22. }
  23. ReflectionObject.call(this, name, options);
  24. /* istanbul ignore if */
  25. if (!(fieldNames === undefined || Array.isArray(fieldNames)))
  26. throw TypeError("fieldNames must be an Array");
  27. /**
  28. * Field names that belong to this oneof.
  29. * @type {string[]}
  30. */
  31. this.oneof = fieldNames || []; // toJSON, marker
  32. /**
  33. * Fields that belong to this oneof as an array for iteration.
  34. * @type {Field[]}
  35. * @readonly
  36. */
  37. this.fieldsArray = []; // declared readonly for conformance, possibly not yet added to parent
  38. /**
  39. * Comment for this field.
  40. * @type {string|null}
  41. */
  42. this.comment = comment;
  43. }
  44. /**
  45. * Oneof descriptor.
  46. * @interface IOneOf
  47. * @property {Array.<string>} oneof Oneof field names
  48. * @property {Object.<string,*>} [options] Oneof options
  49. */
  50. /**
  51. * Constructs a oneof from a oneof descriptor.
  52. * @param {string} name Oneof name
  53. * @param {IOneOf} json Oneof descriptor
  54. * @returns {OneOf} Created oneof
  55. * @throws {TypeError} If arguments are invalid
  56. */
  57. OneOf.fromJSON = function fromJSON(name, json) {
  58. return new OneOf(name, json.oneof, json.options, json.comment);
  59. };
  60. /**
  61. * Converts this oneof to a oneof descriptor.
  62. * @param {IToJSONOptions} [toJSONOptions] JSON conversion options
  63. * @returns {IOneOf} Oneof descriptor
  64. */
  65. OneOf.prototype.toJSON = function toJSON(toJSONOptions) {
  66. var keepComments = toJSONOptions ? Boolean(toJSONOptions.keepComments) : false;
  67. return util.toObject([
  68. "options" , this.options,
  69. "oneof" , this.oneof,
  70. "comment" , keepComments ? this.comment : undefined
  71. ]);
  72. };
  73. /**
  74. * Adds the fields of the specified oneof to the parent if not already done so.
  75. * @param {OneOf} oneof The oneof
  76. * @returns {undefined}
  77. * @inner
  78. * @ignore
  79. */
  80. function addFieldsToParent(oneof) {
  81. if (oneof.parent)
  82. for (var i = 0; i < oneof.fieldsArray.length; ++i)
  83. if (!oneof.fieldsArray[i].parent)
  84. oneof.parent.add(oneof.fieldsArray[i]);
  85. }
  86. /**
  87. * Adds a field to this oneof and removes it from its current parent, if any.
  88. * @param {Field} field Field to add
  89. * @returns {OneOf} `this`
  90. */
  91. OneOf.prototype.add = function add(field) {
  92. /* istanbul ignore if */
  93. if (!(field instanceof Field))
  94. throw TypeError("field must be a Field");
  95. if (field.parent && field.parent !== this.parent)
  96. field.parent.remove(field);
  97. this.oneof.push(field.name);
  98. this.fieldsArray.push(field);
  99. field.partOf = this; // field.parent remains null
  100. addFieldsToParent(this);
  101. return this;
  102. };
  103. /**
  104. * Removes a field from this oneof and puts it back to the oneof's parent.
  105. * @param {Field} field Field to remove
  106. * @returns {OneOf} `this`
  107. */
  108. OneOf.prototype.remove = function remove(field) {
  109. /* istanbul ignore if */
  110. if (!(field instanceof Field))
  111. throw TypeError("field must be a Field");
  112. var index = this.fieldsArray.indexOf(field);
  113. /* istanbul ignore if */
  114. if (index < 0)
  115. throw Error(field + " is not a member of " + this);
  116. this.fieldsArray.splice(index, 1);
  117. index = this.oneof.indexOf(field.name);
  118. /* istanbul ignore else */
  119. if (index > -1) // theoretical
  120. this.oneof.splice(index, 1);
  121. field.partOf = null;
  122. return this;
  123. };
  124. /**
  125. * @override
  126. */
  127. OneOf.prototype.onAdd = function onAdd(parent) {
  128. ReflectionObject.prototype.onAdd.call(this, parent);
  129. var self = this;
  130. // Collect present fields
  131. for (var i = 0; i < this.oneof.length; ++i) {
  132. var field = parent.get(this.oneof[i]);
  133. if (field && !field.partOf) {
  134. field.partOf = self;
  135. self.fieldsArray.push(field);
  136. }
  137. }
  138. // Add not yet present fields
  139. addFieldsToParent(this);
  140. };
  141. /**
  142. * @override
  143. */
  144. OneOf.prototype.onRemove = function onRemove(parent) {
  145. for (var i = 0, field; i < this.fieldsArray.length; ++i)
  146. if ((field = this.fieldsArray[i]).parent)
  147. field.parent.remove(field);
  148. ReflectionObject.prototype.onRemove.call(this, parent);
  149. };
  150. /**
  151. * Decorator function as returned by {@link OneOf.d} (TypeScript).
  152. * @typedef OneOfDecorator
  153. * @type {function}
  154. * @param {Object} prototype Target prototype
  155. * @param {string} oneofName OneOf name
  156. * @returns {undefined}
  157. */
  158. /**
  159. * OneOf decorator (TypeScript).
  160. * @function
  161. * @param {...string} fieldNames Field names
  162. * @returns {OneOfDecorator} Decorator function
  163. * @template T extends string
  164. */
  165. OneOf.d = function decorateOneOf() {
  166. var fieldNames = new Array(arguments.length),
  167. index = 0;
  168. while (index < arguments.length)
  169. fieldNames[index] = arguments[index++];
  170. return function oneOfDecorator(prototype, oneofName) {
  171. util.decorateType(prototype.constructor)
  172. .add(new OneOf(oneofName, fieldNames));
  173. Object.defineProperty(prototype, oneofName, {
  174. get: util.oneOfGetter(fieldNames),
  175. set: util.oneOfSetter(fieldNames)
  176. });
  177. };
  178. };