1662 lines
43 KiB
JavaScript
1662 lines
43 KiB
JavaScript
/**
|
||
* @file Interact with k-fet API.
|
||
* @copyright 2017 cof-geek
|
||
* @license MIT
|
||
*/
|
||
|
||
|
||
/**
|
||
* Virtual namespace for models.
|
||
* <br><br>
|
||
*
|
||
* A model subclasses {@link Models.ModelObject}.<br>
|
||
* A model whose instances can be got from API subclasses
|
||
* {@link Models.APIModelObject}.<br>
|
||
* A model to manage ModelObject forests
|
||
* {@link Models.ModelForest}.<br>
|
||
* A ModelObject that can be fetched through API
|
||
* {@link Models.APIModelForest}.<br>
|
||
* These classes should not be used directly.
|
||
* <br><br>
|
||
*
|
||
* Models with API support:
|
||
* {@link Models.Account} (partial),
|
||
* {@link Models.Checkout} (partial).
|
||
* <br>
|
||
* Models without API support:
|
||
* {@link Models.Statement},
|
||
* {@link Models.ArticleCategory},
|
||
* {@link Models.Article},
|
||
* {@link Models.HistoryGroup},
|
||
* {@link Models.OperationGroup},
|
||
* {@link Models.TransferGroup},
|
||
* {@link Models.Operation},
|
||
* {@link Models.Purchase},
|
||
* {@link Models.SpecialOperation},
|
||
* {@link Models.Transfer}.
|
||
* <br>
|
||
* Implementations of ModelForest:
|
||
* {@link Models.ArticleList}
|
||
* {@link Models.OperationList}
|
||
*
|
||
* @namespace Models
|
||
*/
|
||
|
||
|
||
/**
|
||
* Extended object
|
||
* @memberof Models
|
||
*/
|
||
class ModelObject {
|
||
|
||
/**
|
||
* These properties always exist on instances of this class.
|
||
* @abstract
|
||
* @type {string[]}
|
||
*/
|
||
static get props() { return []; }
|
||
|
||
/**
|
||
* Default values for properties given by <tt>props</tt> static member.
|
||
* @abstract
|
||
* @type {Object.<string, *>}
|
||
*/
|
||
static get default_data() { return {}; }
|
||
|
||
/**
|
||
* Create new instance from data or default values.
|
||
* @param {Object} [data={}] - data to store in instance
|
||
*/
|
||
constructor(data) {
|
||
this.from(data || {});
|
||
}
|
||
|
||
/**
|
||
* Check if an instance is empty.
|
||
* @return {boolean}
|
||
*/
|
||
is_empty() {
|
||
return ( this.id === undefined ) ? false : this.id == 0;
|
||
}
|
||
|
||
/**
|
||
* Returns a {@link Formatters|Formatter} used for rendering.
|
||
* @default {@link Formatters.Formatter}
|
||
* @type {Formatter}
|
||
*/
|
||
formatter() { return Formatter; }
|
||
|
||
/**
|
||
* Set properties of this instance from data object ones. If a property is
|
||
* not present in data object, its value fallback to
|
||
* {@link Models.ModelObject.default_data} one.
|
||
* @param {Object} data
|
||
* @todo Restrict to props
|
||
*/
|
||
from(data) {
|
||
// TODO: add restrict
|
||
this.clear();
|
||
this.update(data);
|
||
}
|
||
|
||
/**
|
||
* Update properties of this instance from data ones.
|
||
* @param {Object} data
|
||
*/
|
||
update(data) {
|
||
$.extend(this, data);
|
||
}
|
||
|
||
/**
|
||
* Clear properties to {@link Models.ModelObject.default_data|default_data}.
|
||
*/
|
||
clear() {
|
||
$.extend(this, this.constructor.default_data);
|
||
}
|
||
|
||
|
||
/**
|
||
* Display stored data in container.
|
||
* @param {jQuery} $container
|
||
* @param {Object} [options] Options for formatter render method.
|
||
* @param {Formatters.Formatter}
|
||
* [formatter={@link Models.ModelObject#formatter|this.formatter()}]
|
||
* Formatter class to use.
|
||
* @return {jQuery} The DOM element $container, allowing methods chaining.
|
||
*/
|
||
display($container, options, formatter) {
|
||
formatter = formatter || this.formatter();
|
||
formatter.render(this, $container, options);
|
||
return $container;
|
||
}
|
||
|
||
/**
|
||
* Compares two ModelObject instances.
|
||
*/
|
||
static compare(a, b) {
|
||
return a.id - b.id;
|
||
}
|
||
}
|
||
|
||
|
||
/**
|
||
* Describes a model whose instances can be obtained through API.
|
||
* @extends Models.ModelObject
|
||
* @memberof Models
|
||
*/
|
||
class APIModelObject extends ModelObject {
|
||
|
||
/**
|
||
* Request url to get array of model instances data.
|
||
* @abstract
|
||
* @type {string}
|
||
*/
|
||
static get url_model() {}
|
||
|
||
/**
|
||
* Request url to create an instance.
|
||
* @abstract
|
||
* @type {string}
|
||
*/
|
||
static url_create() {}
|
||
|
||
/**
|
||
* Request url to edit an instance of this model.
|
||
* @abstract
|
||
* @param {*} api_pk - Identifier of a model instance.
|
||
* @type {string}
|
||
*/
|
||
static url_update_for(api_pk) {}
|
||
|
||
/**
|
||
* Request url to get a single model instance data.
|
||
* @abstract
|
||
* @param {*} api_pk - Identifier of a model instance.
|
||
* @return {string}
|
||
*/
|
||
static url_read_for(api_pk) {}
|
||
|
||
/**
|
||
* See {@link Models.ModelObject|new ModelObject(data)}.
|
||
* @param {Object} [data={}] - data to store in instance
|
||
*/
|
||
constructor(data) {
|
||
super(data);
|
||
|
||
if (this.id === undefined)
|
||
this.id = 0;
|
||
}
|
||
|
||
/**
|
||
* Identifier of the model instance in api requests.
|
||
* @default <tt>this.id</tt>
|
||
*/
|
||
get api_pk() { return this.id; }
|
||
|
||
/**
|
||
* Request url used to get current instance data.
|
||
*/
|
||
get url_read() {
|
||
if (this._url_object === undefined)
|
||
return this.is_empty() ? '' : this.constructor.url_read_for(this.api_pk);
|
||
return this._url_object;
|
||
}
|
||
|
||
set url_read(v) { this._url_object = v; }
|
||
|
||
/**
|
||
* Get data of a distant model instance. It sends a GET HTTP request to
|
||
* {@link Models.APIModelObject#url_read}.
|
||
* @param {object} [api_options] Additional data appended to the request.
|
||
*/
|
||
fromAPI(api_options) {
|
||
api_options = api_options || {};
|
||
|
||
api_options['format'] = 'json';
|
||
|
||
return $.getJSON(this.url_read, api_options)
|
||
.done( (json) => this.from(json) );
|
||
}
|
||
|
||
/**
|
||
* Get data of a distant model instance via its id.
|
||
* @param {*} api_pk
|
||
* @param {object} [api_options]
|
||
* @param {jQueryAjaxSuccess} [on_success]
|
||
* @param {jQueryAjaxError} [on_error]
|
||
* @see {@link Models.APIModelObject#fromAPI|fromAPI}
|
||
*/
|
||
get_by_apipk(api_pk, api_options) {
|
||
this.url_read = this.constructor.url_read_for(api_pk);
|
||
return this.fromAPI(api_options);
|
||
}
|
||
|
||
/**
|
||
* Add <tt>empty_props</tt> and <tt>empty_attrs</tt> options to
|
||
* <tt>render</tt> call in {@link Models.ModelObject#display} if instance
|
||
* is empty.
|
||
* @see {@link Models.ModelObject#display}
|
||
* @see {@link Models.APIModelObject#is_empty}
|
||
*/
|
||
display($container, options, formatter) {
|
||
if (this.is_empty()) {
|
||
options.empty_props = true;
|
||
options.empty_attrs = true;
|
||
}
|
||
return ModelObject.prototype.display.call(this, $container, options, formatter);
|
||
}
|
||
|
||
}
|
||
|
||
|
||
/**
|
||
* Account model. Can be accessed through API.
|
||
* @extends Models.APIModelObject
|
||
* @memberof Models
|
||
*/
|
||
class Account extends APIModelObject {
|
||
|
||
/**
|
||
* Properties retrieved through API.
|
||
* @default <tt>['id', 'trigramme', 'name', 'nickname', 'email', 'is_cof',
|
||
* 'promo', 'balance', 'is_frozen', 'departement']</tt>
|
||
* @see {@link Models.ModelObject.props|ModelObject.props}
|
||
*/
|
||
static get props() {
|
||
return ['id', 'trigramme', 'name', 'nickname', 'email', 'is_cof',
|
||
'promo', 'balance', 'is_frozen', 'departement'];
|
||
}
|
||
|
||
/**
|
||
* Default values for Account model instances.
|
||
* @default <tt>{ 'id': 0, 'trigramme': '', 'name': '', 'nickname': '',
|
||
* 'email': '', ''is_cof': false, 'promo': '', 'balance': 0,
|
||
* 'is_frozen': false, 'departement': '' }</tt>
|
||
* @see {@link Models.ModelObject.default_data|ModelObject.default_data}
|
||
*/
|
||
static get default_data() {
|
||
return {
|
||
'id': 0, 'trigramme': '', 'name': '', 'nickname': '', 'email': '',
|
||
'is_cof' : false, 'promo': '', 'balance': 0, 'is_frozen': false,
|
||
'departement': '',
|
||
};
|
||
};
|
||
|
||
/**
|
||
* @default <tt>django-js-reverse('kfet.account')</tt>
|
||
* @see {@link Models.APIModelObject.url_model|APIModelObject.url_model}
|
||
*/
|
||
static get url_model() { return Urls['kfet.account'](); }
|
||
|
||
static url_create(trigramme) {
|
||
var url = Urls['kfet.account.create']();
|
||
if (trigramme) {
|
||
var trigramme_url = encodeURIComponent(trigramme);
|
||
url += `?trigramme=${trigramme_url}`
|
||
}
|
||
return url;
|
||
}
|
||
|
||
/**
|
||
* @default <tt>django-js-reverse('kfet.account.read')(trigramme)</tt>
|
||
* @param {string} trigramme
|
||
* @see {@link Models.APIModelObject.url_read_for|APIModelObject.url_read_for}
|
||
*/
|
||
static url_read_for(trigramme) {
|
||
var trigramme_url = encodeURIComponent(trigramme);
|
||
return Urls['kfet.account.read'](trigramme_url);
|
||
}
|
||
|
||
static url_update_for(trigramme) {
|
||
var trigramme_url = encodeURIComponent(trigramme);
|
||
return Urls['kfet.account.update'](trigramme_url);
|
||
}
|
||
|
||
/**
|
||
* @default <tt>this.trigramme</tt>
|
||
*/
|
||
get api_pk() { return this.trigramme; }
|
||
|
||
/**
|
||
* @default {@link Formatters.AccountFormatter}
|
||
*/
|
||
formatter() {
|
||
return (this.trigramme == 'LIQ') ? LIQFormatter : AccountFormatter;
|
||
}
|
||
|
||
// take care of "balance" type
|
||
// API currently returns a string object (serialization of Decimal type within Django)
|
||
get balance() { return this._balance; }
|
||
set balance(v) { return this._balance = floatCheck(v); }
|
||
|
||
/**
|
||
* Balance converted to UKF according to cof status.
|
||
*/
|
||
get balance_ukf() { return amountToUKF(this.balance, this.is_cof, true); }
|
||
|
||
}
|
||
|
||
|
||
/**
|
||
* Checkout model. Can be accessed through API.
|
||
* @extends Models.APIModelObject
|
||
* @memberof Models
|
||
*/
|
||
class Checkout extends APIModelObject {
|
||
|
||
/**
|
||
* Properties retrieved through API.
|
||
* @default <tt>['id', 'name', 'balance', 'valid_from', 'valid_to']</tt>
|
||
* @see {@link Models.ModelObject.props|ModelObject.props}
|
||
*/
|
||
static get props() {
|
||
return ['id', 'name', 'balance', 'valid_from', 'valid_to'];
|
||
}
|
||
|
||
/**
|
||
* Default values for Account model instances.
|
||
* @default <tt>{ 'id': 0, 'name': '', 'balance': 0, 'valid_from': '',
|
||
* 'valid_to': '' }</tt>
|
||
* @see {@link Models.ModelObject.default_data|ModelObject.default_data}
|
||
*/
|
||
static get default_data() {
|
||
return {
|
||
'id': 0, 'name': '', 'balance': 0, 'valid_from': '', 'valid_to': '',
|
||
};
|
||
}
|
||
|
||
/**
|
||
* Not supported by API yet.
|
||
* @see {@link Models.APIModelObject.url_model|APIModelObject.url_model}
|
||
*/
|
||
static get url_model() { return ''; }
|
||
|
||
/**
|
||
* @default <tt>django-js-reverse('kfet.kpsul.checkout_data.read')(pk)</tt>
|
||
* @param {string} api_pk - a checkout id
|
||
* @see {@link Models.APIModelObject.url_read_for|APIModelObject.url_read_for}
|
||
*/
|
||
static url_read_for(api_pk) {
|
||
return Urls['kfet.checkout.read'](api_pk);
|
||
}
|
||
|
||
static url_update_for(api_pk) {
|
||
return Urls['kfet.checkout.update'](api_pk);
|
||
}
|
||
|
||
/**
|
||
* @default {@link Formatters.CheckoutFormatter}
|
||
*/
|
||
formatter() {
|
||
return CheckoutFormatter;
|
||
}
|
||
|
||
// take care of "balance" type
|
||
// API currently returns a string object (serialization of Decimal type within Django)
|
||
get balance() { return this._balance; }
|
||
set balance(v) { this._balance = floatCheck(v); }
|
||
|
||
}
|
||
|
||
|
||
/**
|
||
* Statement model. Cannot be accessed through API.
|
||
* @extends Models.ModelObject
|
||
* @memberof Models
|
||
*/
|
||
class Statement extends ModelObject {
|
||
|
||
/**
|
||
* Properties associated to a statement.
|
||
* @default <tt>['id', 'at', 'balance_old', 'balance_new', 'by']</tt>
|
||
* @see {@link Models.ModelObject.props|ModelObject.props}
|
||
*/
|
||
static get props() {
|
||
return ['id', 'at', 'balance_old', 'balance_new', 'by'];
|
||
}
|
||
|
||
/**
|
||
* Default values for Statement model instances.
|
||
* @default <tt>{ 'id': 0, 'at': '', 'balance_old': 0, 'balance_new': 0,
|
||
* 'by': '' }</tt>
|
||
* @see {@link Models.ModelObject.default_data|ModelObject.default_data}
|
||
*/
|
||
static get default_data() {
|
||
return {
|
||
'id': 0, 'at': '', 'balance_old': 0, 'balance_new': 0, 'by': '',
|
||
};
|
||
}
|
||
|
||
static url_create(checkout_pk) {
|
||
return Urls['kfet.checkoutstatement.create'](checkout_pk);
|
||
}
|
||
|
||
/**
|
||
* @default {@link Formatters.StatementFormatter}
|
||
*/
|
||
formatter() {
|
||
return StatementFormatter;
|
||
}
|
||
|
||
// take care of "balance" type
|
||
// API currently returns a string object (serialization of Decimal type within Django)
|
||
get balance_old() { return this._balance_old; }
|
||
set balance_old(v) { this._balance_old = floatCheck(v); }
|
||
|
||
get balance_new() { return this._balance_new; }
|
||
set balance_new(v) { this._balance_new = floatCheck(v); }
|
||
|
||
get at() { return this._at; }
|
||
set at(v) { this._at = moment.isMoment(v) ? v : moment.tz(v, 'UTC'); }
|
||
|
||
}
|
||
|
||
/**
|
||
* ArticleCategory model. Cannot be accessed through API.
|
||
* @extends Models.ModelObject
|
||
* @memberof Models
|
||
*/
|
||
class ArticleCategory extends ModelObject {
|
||
|
||
/**
|
||
* Properties associated to a category
|
||
* @default <tt>['id', 'name']</tt>
|
||
* @see {@link Models.ModelObject.props|ModelObject.props}
|
||
*/
|
||
static get props() {
|
||
return ['id', 'name'];
|
||
}
|
||
|
||
/**
|
||
* Default values for ArticleCategory model instances.
|
||
* @default <tt>{ 'id': 0, 'name': '' }</tt>
|
||
* @see {@link Models.ModelObject.default_data|ModelObject.default_data}
|
||
*/
|
||
static get default_data() {
|
||
return {'id': 0, 'name': ''};
|
||
}
|
||
|
||
/**
|
||
* @default {@link Formatters.ArticleCategoryFormatter}
|
||
*/
|
||
formatter() {
|
||
return ArticleCategoryFormatter;
|
||
}
|
||
|
||
/**
|
||
* Comparison function between ArticleCategory model instances.
|
||
* @see {@link Models.ModelObject.compare|ModelObject.compare}
|
||
*/
|
||
static compare(a, b) {
|
||
return a.name.localeCompare(b.name) ;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Article model. Cannot be accessed through API.
|
||
* @extends Models.ModelObject
|
||
* @memberof Models
|
||
*/
|
||
class Article extends ModelObject {
|
||
/**
|
||
* Properties associated to an article
|
||
* @default <tt>['id', 'name']</tt>
|
||
* @see {@link Models.ModelObject.props|ModelObject.props}
|
||
*/
|
||
static get props() {
|
||
return ['id', 'name', 'price', 'stock', 'category'];
|
||
}
|
||
|
||
/**
|
||
* Default values for Article model instances.
|
||
* @default <tt>{ 'id': 0, 'name': '', 'price': 0, 'stock': 0 }</tt>
|
||
* @see {@link Models.ModelObject.default_data|ModelObject.default_data}
|
||
*/
|
||
static get default_data() {
|
||
return { 'id': 0, 'name': '', 'price': 0, 'stock': 0 };
|
||
}
|
||
|
||
/**
|
||
* @default {@link Formatters.ArticleFormatter}
|
||
*/
|
||
formatter() {
|
||
return ArticleFormatter;
|
||
}
|
||
|
||
/**
|
||
* Comparison function between Article model instances.
|
||
* @see {@link Models.ModelObject.compare|ModelObject.compare}
|
||
*/
|
||
static compare(a, b) {
|
||
return a.name.localeCompare(b.name);
|
||
}
|
||
|
||
// Take care of 'price' type
|
||
// API currently returns a string object (serialization of Decimal type within Django)
|
||
get price() { return this._price; }
|
||
set price(v) { this._price = floatCheck(v); }
|
||
|
||
is_low_stock(nb) {
|
||
return (-5 <= this.stock - nb && this.stock - nb <= 5);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Day model. Cannot be accessed through API.
|
||
* @extends Model.ModelObject
|
||
* @memberof Models
|
||
*/
|
||
class Day extends ModelObject {
|
||
|
||
/**
|
||
* Properties associated to a day.
|
||
* @default <tt>['id', 'date']</tt>
|
||
* @see {@link Models.ModelObject.props|ModelObject.props}
|
||
*/
|
||
static get props() { return ['id', 'date'] }
|
||
|
||
/**
|
||
* Default values for Day model instances
|
||
* @default <tt>{'id': '', 'date': moment()}</tt>
|
||
* @see {@link Models.ModelObject.default_data|ModelObject.default_data}
|
||
*/
|
||
static get default_data() { return {'id': '', 'date': moment()}; }
|
||
|
||
/**
|
||
* @default {@link Formatters.DayFormatter}
|
||
*/
|
||
formatter() {
|
||
return DayFormatter;
|
||
}
|
||
|
||
/**
|
||
* Comparison function between Day model instances.
|
||
* @see {@link Models.ModelObject.compare|ModelObject.compare}
|
||
*/
|
||
static compare (a, b) {
|
||
//Days are sorted by most recent first
|
||
if (a.date < b.date) return 1;
|
||
else if (a.date > b.date) return -1;
|
||
else return 0;
|
||
}
|
||
|
||
//Parse date and round it
|
||
get date() { return this._date; }
|
||
set date(v) { this._date = dateUTCToParis(v).startOf('date'); }
|
||
}
|
||
|
||
/**
|
||
* HistoryGroup model. Should not be used directly.
|
||
* @extends Models.ModelObject
|
||
* @memberof Models
|
||
*/
|
||
class HistoryGroup extends ModelObject {
|
||
|
||
/**
|
||
* Properties assowiated to HistoryGroup instances.
|
||
* @default <tt>['id', 'at', 'comment', 'valid_by']</tt>
|
||
* @see {@link Models.ModelObject.props|ModelObject.props}
|
||
*/
|
||
static get props() {
|
||
return ['id', 'at', 'comment', 'valid_by'];
|
||
}
|
||
|
||
/**
|
||
* Default values for HistoryGroup model instances
|
||
* @default <tt>{ 'id': 0, 'at': moment(), 'comment': '',
|
||
'valid_by': '' }</tt>
|
||
* @see {@link Models.ModelObject.default_data|ModelObject.default_data}
|
||
*/
|
||
static get default_data() {
|
||
return {'id': 0, 'at': moment(), 'comment': '',
|
||
'valid_by': '',};
|
||
}
|
||
|
||
/**
|
||
* Comparison function between HistoryGroup model instances.
|
||
* @see {@link Models.ModelObject.compare|ModelObject.compare}
|
||
*/
|
||
static compare(a, b) {
|
||
//Groups are sorted by most recent first
|
||
if (a.at.isBefore(b.at)) return 1;
|
||
else if (a.at.isAfter(b.at)) return -1;
|
||
else return 0;
|
||
}
|
||
|
||
// Parse the date to a moment() object
|
||
get at() { return this._at; }
|
||
set at(v) { this._at = dateUTCToParis(v); }
|
||
}
|
||
|
||
/**
|
||
* OperationGroup model. Cannot be accessed through API.
|
||
* @extends Models.HistoryGroup
|
||
* @memberof Models
|
||
*/
|
||
class OperationGroup extends HistoryGroup {
|
||
|
||
/**
|
||
* Properties associated with an opegroup.
|
||
* @default <tt>{@link Models.HistoryGroup.props|HistoryGroup.props} +
|
||
* ['amount', 'is_cof', 'trigramme']</tt>
|
||
* @see {@link Models.ModelObject.props|ModelObject.props}
|
||
*/
|
||
static get props() {
|
||
return HistoryGroup.props.concat(['amount', 'is_cof', 'trigramme']);
|
||
}
|
||
|
||
/**
|
||
* Default values for OperationGroup instances.
|
||
* @default <tt>{@link Models.HistoryGroup.default_data|HistoryGroup.default_data} +
|
||
* {'amount': 0, 'is_cof': false, 'trigramme': ''}</tt>
|
||
* @see {@link Models.ModelObject.default_data|ModelObject.default_data}
|
||
*/
|
||
static get default_data() {
|
||
return $.extend({}, HistoryGroup.default_data,
|
||
{'amount': 0, 'is_cof': false, 'trigramme': ''});
|
||
}
|
||
|
||
/**
|
||
* @default {@link Formatters.OpegroupFormatter}
|
||
*/
|
||
formatter() {
|
||
return OpegroupFormatter;
|
||
}
|
||
|
||
get amount() { return this._amount; }
|
||
set amount(v) { this._amount = floatCheck(v); }
|
||
}
|
||
|
||
/**
|
||
* TransferGroup model. Cannot be accessed through API.
|
||
* @extends Models.HistoryGroup
|
||
* @memberof Models
|
||
*/
|
||
class TransferGroup extends HistoryGroup {
|
||
|
||
/**
|
||
* @default {@link Formatters.TransferGroupFormatter}
|
||
*/
|
||
formatter() {
|
||
return TransferGroupFormatter;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Operation model. Should not be used directly.
|
||
* @extends Models.ModelObject
|
||
* @memberof Models
|
||
*/
|
||
class Operation extends ModelObject {
|
||
|
||
/**
|
||
* Properties associated to an operation.
|
||
* @default <tt>['id', 'amount', 'canceled_at', 'canceled_by']</tt>
|
||
* @see {@link Models.ModelObject.props|ModelObject.props}
|
||
*/
|
||
static get props() {
|
||
return ['id', 'amount', 'canceled_at', 'canceled_by'];
|
||
}
|
||
|
||
/**
|
||
* Default values for Operation model instances
|
||
* @default <tt>{'id': '', 'amount': 0, 'canceled_at': undefined, 'canceled_by': '' }</tt>
|
||
* @see {@link Models.ModelObject.default_data|ModelObject.default_data}
|
||
*/
|
||
static get default_data() {
|
||
return {'id': '', 'amount': 0, 'canceled_at': undefined, 'canceled_by': '' };
|
||
}
|
||
|
||
get amount() { return this._amount; }
|
||
set amount(v) { this._amount = floatCheck(v); }
|
||
|
||
get canceled_at() { return this._canceled_at; }
|
||
set canceled_at(v) {
|
||
if (v)
|
||
this._canceled_at = dateUTCToParis(v);
|
||
else
|
||
this._canceled_at = undefined;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Purchase model. Cannot be accessed through API.
|
||
* @extends Models.Operation
|
||
* @memberof Models
|
||
*/
|
||
class Purchase extends Operation {
|
||
|
||
/**
|
||
* Additional properties for purchases.
|
||
* @default <tt>{@link Models.Operation.props|Operation.props} + ['article_name', 'article_nb',
|
||
* 'addcost_amount', 'addcost_for']</tt>
|
||
* @see {@link Models.ModelObject.props|ModelObject.props}
|
||
*/
|
||
static get props() {
|
||
return Operation.props.concat(
|
||
['article_name', 'article_nb', 'addcost_amount', 'addcost_for']
|
||
);
|
||
}
|
||
|
||
/**
|
||
* Default values for Operation model instances
|
||
* @default <tt>{@link Models.Operation.default_data|Operation.default_data} + {
|
||
* 'article_name': '', 'article_nb': 0,
|
||
* 'addcost_amount': 0, 'addcost_for': ''
|
||
* }
|
||
* @see {@link Models.ModelObject.default_data|ModelObject.default_data}
|
||
*/
|
||
static get default_data() {
|
||
return $.extend({}, Operation.default_data, {
|
||
'article_name': '', 'article_nb': 0,
|
||
'addcost_amount': 0, 'addcost_for': ''
|
||
});
|
||
}
|
||
|
||
/**
|
||
* @default {@link Formatters.PurchaseFormatter}
|
||
*/
|
||
formatter() {
|
||
return PurchaseFormatter;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Special operation model. Cannot be accessed through API.
|
||
* @extends Models.Operation
|
||
* @memberof Models
|
||
*/
|
||
class SpecialOperation extends Operation {
|
||
|
||
/**
|
||
* Additional properties for special operations.
|
||
* @default <tt>{@link Models.Operation.props|Operation.props} + ['type', 'is_checkout']</tt>
|
||
* @see {@link Models.ModelObject.props|ModelObject.props}
|
||
*/
|
||
static get props() {
|
||
return Operation.props.concat(['type', 'is_checkout']);
|
||
}
|
||
|
||
/**
|
||
* Verbose names for operation types
|
||
* @type {Object}
|
||
*/
|
||
static get verbose_types() {
|
||
return {
|
||
'deposit': 'Charge',
|
||
'withdraw': 'Retrait',
|
||
'edit': 'Édition',
|
||
'initial': 'Initial',
|
||
};
|
||
}
|
||
|
||
/**
|
||
* Default values for SpecialOperation model instances
|
||
* @default <tt>{@link Models.Operation.default_data|Operation.default_data} + {'type': '', 'is_checkout': false}
|
||
* @see {@link Models.ModelObject.default_data|ModelObject.default_data}
|
||
*/
|
||
static get default_data() {
|
||
return $.extend({}, Operation.default_data, {'type': '', 'is_checkout': false});
|
||
}
|
||
|
||
/**
|
||
* @default {@link Formatters.SpecialOpeFormatter}
|
||
*/
|
||
formatter() {
|
||
return SpecialOpeFormatter;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Transfer model. Cannot be accessed through API.
|
||
* @exetends Models.Operation
|
||
* @memberof Models
|
||
*/
|
||
class Transfer extends Operation {
|
||
|
||
/**
|
||
* Additional properties for transfers.
|
||
* @default <tt>{@link Models.Operation.props|Operation.props} + ['from_acc', 'to_acc']</tt>
|
||
* @see {@link Models.ModelObject.props|ModelObject.props}
|
||
*/
|
||
static get props() {
|
||
return Operation.props.concat(['from_acc', 'to_acc']);
|
||
}
|
||
|
||
/**
|
||
* Default values for Transfer model instances
|
||
* @default <tt>{@link Models.Operation.default_data|Operation.default_data} + {'from_acc': '', 'to_acc': ''}
|
||
* @see {@link Models.ModelObject.default_data|ModelObject.default_data}
|
||
*/
|
||
static get default_data() {
|
||
return $.extend({}, Operation.default_data, {'from_acc': '', 'to_acc': ''});
|
||
}
|
||
|
||
/**
|
||
* @default {@link Formatters.TransferFormatter}
|
||
*/
|
||
formatter() {
|
||
return TransferFormatter;
|
||
}
|
||
}
|
||
|
||
|
||
/**
|
||
* Node for ModelForest object
|
||
* @memberof Models
|
||
*/
|
||
class TreeNode {
|
||
|
||
constructor(type, content) {
|
||
this.modelname = type;
|
||
this.content = content;
|
||
this.parent = null;
|
||
this.children = [];
|
||
}
|
||
}
|
||
|
||
|
||
/**
|
||
* Simple {@link Models.ModelObject} forest.
|
||
* @memberof Models
|
||
*/
|
||
class ModelForest {
|
||
|
||
/**
|
||
* Dictionary associating types to classes
|
||
* @abstract
|
||
* @type {Object}
|
||
*/
|
||
static get models() { return {}; }
|
||
|
||
/**
|
||
* Comparison function for nodes
|
||
* @abstract
|
||
* @param {class} model Model to use for comparison
|
||
* @param {Models.TreeNode} a
|
||
* @param {Models.TreeNode} b
|
||
* @see {@link Models.ModelObject.compare|ModelObject.compare}
|
||
*/
|
||
static compare(model, a, b) {
|
||
return model.compare(a.content, b.content);
|
||
}
|
||
|
||
|
||
/**
|
||
* Creates empty instance and populates it with data if given
|
||
* @param {Object[]} [datalist=[]]
|
||
*/
|
||
constructor(datalist) {
|
||
this.from(datalist || []);
|
||
}
|
||
|
||
/**
|
||
* Fetches an object from the instance data, or creates it if
|
||
* it does not exist yet.<br>
|
||
* If direction >= 0, parent objects are created recursively.
|
||
* If direction <= 0, child objects are created recursively.
|
||
* @param {Object} data
|
||
* @param {number} direction
|
||
*/
|
||
get_or_create(data, direction) {
|
||
var model = this.constructor.models[data.modelname];
|
||
|
||
var existing = this.find(data.modelname, data.content.id);
|
||
if (existing) {
|
||
return existing;
|
||
}
|
||
|
||
var content = new this.constructor.models[data.modelname](data.content);
|
||
var node = new TreeNode(data.modelname, content);
|
||
|
||
if (data.child_sort)
|
||
node.child_sort = this.constructor.models[data.child_sort];
|
||
else
|
||
node.child_sort = ModelObject;
|
||
|
||
if (direction <= 0) {
|
||
if (data.parent) {
|
||
var parent = this.get_or_create(data.parent, -1);
|
||
node.parent = parent;
|
||
parent.children.push(node);
|
||
} else {
|
||
this.roots.push(node);
|
||
}
|
||
}
|
||
|
||
if (direction >= 0 && data.children) {
|
||
for (let child_data of data.children) {
|
||
var child = this.get_or_create(child_data, 1);
|
||
child.parent = node;
|
||
node.children.push(child);
|
||
}
|
||
}
|
||
|
||
return node ;
|
||
}
|
||
|
||
/**
|
||
* Resets then populates the instance with the given data.
|
||
* @param {Object[]} datalist
|
||
*/
|
||
from(datalist) {
|
||
this.roots = [];
|
||
for (let data of datalist) {
|
||
this.get_or_create(data, 0);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Removes all Models.TreeNode from the tree.
|
||
*/
|
||
clear() {
|
||
this.from([]);
|
||
}
|
||
|
||
/**
|
||
* Renders a node (and all its offspring) and returns the
|
||
* corresponding jQuery object.
|
||
* @param {Models.TreeNode} node
|
||
* @param {Object} templates Templates to render each model
|
||
* @param {Object} [options] Options for element render method
|
||
*/
|
||
render_element(node, templates, options) {
|
||
var template = templates[node.modelname];
|
||
var options = options || {} ;
|
||
|
||
var $container = $('<div></div>');
|
||
$container.attr('id', node.modelname+'-'+node.content.id);
|
||
|
||
var $rendered = node.content.display($(template), options);
|
||
$container.append($rendered);
|
||
|
||
//dirty
|
||
node.children.sort(this.constructor.compare.bind(null, node.child_sort));
|
||
|
||
for (let child of node.children) {
|
||
var $child = this.render_element(child, templates, options);
|
||
$container.append($child);
|
||
}
|
||
|
||
return $container;
|
||
}
|
||
|
||
/**
|
||
* Renders node and adds it to given container.<br>
|
||
* Assumes that the inserted node is the 'youngest'.
|
||
* @param {jQuery} $container
|
||
* @param {Models.TreeNode} node
|
||
* @param {Object} templates Templates to render each model
|
||
* @param {Object} [options] Options for element render method
|
||
*/
|
||
add_to_container($container, node, templates, options) {
|
||
var existing = node.parent ;
|
||
var first_missing = node;
|
||
|
||
while (existing && !($container.find('#'+existing.modelname+'-'+existing.id).length)) {
|
||
first_missing = existing ;
|
||
existing = existing.parent;
|
||
}
|
||
|
||
var $to_insert = this.render_element(first_missing, templates, options);
|
||
if (existing) {
|
||
$container.find('#'+existing.modelname+'-'+existing.content.id+'>:first-child').after($to_insert);
|
||
} else {
|
||
$container.prepend($to_insert);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Display stored data in container.
|
||
* @param {jQuery} $container
|
||
* @param {Object} templates Templates to render each model
|
||
* @param {Object} [options] Options for element render method
|
||
*/
|
||
display($container, templates, options) {
|
||
this.roots.sort(this.constructor.compare.bind(null, this.root_sort));
|
||
for (let root of this.roots) {
|
||
$container.append(this.render_element(root, templates, options));
|
||
}
|
||
|
||
return $container;
|
||
}
|
||
|
||
traverse(callback) {
|
||
function recurse(node) {
|
||
callback(node) ;
|
||
|
||
for (let child of node.children)
|
||
recurse(child);
|
||
}
|
||
|
||
for (let root of this.roots)
|
||
recurse(root);
|
||
}
|
||
|
||
/**
|
||
* Find instance in tree with given type and id
|
||
* @param {string} type
|
||
* @param {number} id
|
||
*/
|
||
find(type, id) {
|
||
var result = null;
|
||
function callback(node) {
|
||
if (node.modelname === type && node.content.id == id)
|
||
result = node ;
|
||
}
|
||
|
||
this.traverse(callback);
|
||
|
||
return result ;
|
||
|
||
}
|
||
}
|
||
|
||
|
||
/**
|
||
* Describes a model list that can be filled through API.
|
||
* @extends Models.ModelForest
|
||
* @memberof Models
|
||
*/
|
||
class APIModelForest extends ModelForest {
|
||
|
||
/**
|
||
* Request url to fill the model.
|
||
* @abstract
|
||
* @type {string}
|
||
*/
|
||
static get url_model() {}
|
||
|
||
/**
|
||
* Fills the instance with distant data. It sends a GET HTTP request to
|
||
* {@link Models.APIModelForest#url_model}.
|
||
* @param {object} [api_options] Additional data appended to the request.
|
||
*/
|
||
fromAPI(api_options) {
|
||
api_options = api_options || {};
|
||
|
||
api_options['format'] = 'json';
|
||
|
||
return $.getJSON(this.constructor.url_model, api_options)
|
||
.done( (json) => this.from(json) );
|
||
}
|
||
}
|
||
|
||
|
||
/**
|
||
* ArticleList model. Can be accessed through API.
|
||
* @extends Models.APIModelForest
|
||
* @memberof Models
|
||
*/
|
||
class ArticleList extends APIModelForest {
|
||
|
||
/**
|
||
* Default structure for ArticleList instances
|
||
* @abstract
|
||
* @default <tt>{'article': Article,
|
||
'category': ArticleCategory}</tt>
|
||
*/
|
||
static get models() {
|
||
return {'article': Article,
|
||
'category': ArticleCategory};
|
||
}
|
||
|
||
/**
|
||
* Default url to get ArticleList data
|
||
* @default <tt>django-js-reverse('kfet.kpsul.articles_data')</tt>
|
||
* @see {@link Models.APIModelForest.url_model|APIModelForest.url_model}
|
||
*/
|
||
static get url_model() {
|
||
return Urls['kfet.kpsul.articles_data']();
|
||
}
|
||
|
||
/**
|
||
* Provides model to sort root objects
|
||
* {@see Models.ModelForest.constructor|ModelForest.constructor}
|
||
*/
|
||
constructor() {
|
||
super();
|
||
this.root_sort = ArticleCategory;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* OperationList model. Can be accessed through API.
|
||
* @extends Models.APIModelList
|
||
* @memberof Models
|
||
*/
|
||
class OperationList extends APIModelForest {
|
||
/**
|
||
* Default structure for OperationList instances.
|
||
* @default <tt>[Operation, OperationGroup, Day]</tt>
|
||
* @see {@link Models.ModelList.models|ModelList.models}
|
||
*/
|
||
static get models() {
|
||
return {
|
||
'day': Day,
|
||
'opegroup': OperationGroup,
|
||
'transfergroup': TransferGroup,
|
||
'purchase': Purchase,
|
||
'specialope': SpecialOperation,
|
||
'transfer': Transfer,
|
||
};
|
||
}
|
||
|
||
/**
|
||
* Default url to get OperationList data.
|
||
* @default <tt>django-js-reverse('kfet.history.json')</tt>
|
||
* @see {@link Models.APIModelList.url_model|APIModelList.url_model}
|
||
*/
|
||
static get url_model() {
|
||
return Urls['kfet.history.json']();
|
||
}
|
||
|
||
constructor() {
|
||
super();
|
||
this.root_sort = Day;
|
||
}
|
||
}
|
||
|
||
|
||
/* ---------- ---------- */
|
||
|
||
/**
|
||
* Virtual namespace for formatters.
|
||
* <br><br>
|
||
*
|
||
* A formatter subclasses {@link Formatters.Formatter}.<br>
|
||
* Formatters should be accessed statically only.
|
||
*
|
||
* @namespace Formatters
|
||
*/
|
||
|
||
|
||
/**
|
||
* @memberof Formatters
|
||
*/
|
||
class Formatter {
|
||
|
||
static get props() { return []; }
|
||
static get attrs() { return []; }
|
||
|
||
/**
|
||
* Get value of a property through current formatter if defined, object itself otherwise.
|
||
*
|
||
* @param {*} object - Object used for formatting
|
||
* @param {string} name - Name of property to render
|
||
* @param {string} [prefix=''] - Prefix used for formatter method search
|
||
* @return {*} Formatted property value
|
||
*/
|
||
static get(object, name, prefix) {
|
||
prefix = prefix || '';
|
||
var method = prefix + name;
|
||
if (this[method] !== undefined)
|
||
return this[method](object);
|
||
return object[name];
|
||
}
|
||
|
||
/**
|
||
* Get formatted value of a property using prefix 'prop_'.
|
||
* @see Formatter.get
|
||
*/
|
||
static get_prop(object, name) {
|
||
return this.get(object, name, 'prop_');
|
||
}
|
||
|
||
/**
|
||
* Get formatted value of a property using prefix 'attr_'.
|
||
* @see Formatter.get
|
||
*/
|
||
static get_attr(obj, attr) {
|
||
return this.get(obj, attr, 'attr_');
|
||
}
|
||
|
||
/**
|
||
* Render container.
|
||
* @param {*} object - Object used for formatting
|
||
* @param {jQueryNode} $container
|
||
* @param {object} [options]
|
||
* <tt>options['props']</tt> should be an array of
|
||
* properties (default: <tt>class.props</tt>).<br>
|
||
* <tt>options['prefix_prop']</tt> is added in front of
|
||
* properties names in selector (default: <tt>'.'</tt>).<br>
|
||
* <tt>options['prefix_attr']</tt> is added in front of
|
||
* attributes names for container (default: <tt>''</tt>).
|
||
*/
|
||
static render(object, $container, options) {
|
||
options.props = options.props || [];
|
||
options.attrs = options.attrs || [];
|
||
|
||
var props = options.override_props ? options.props : this.props.concat(options.props);
|
||
var attrs = options.override_attrs ? options.attrs : this.attrs.concat(options.attrs);
|
||
|
||
props = options.remove_props ? props.diff(options.remove_props) : props;
|
||
attrs = options.remove_attrs ? props.diff(options.remove_attrs) : attrs;
|
||
|
||
var prefix_prop = options.prefix_prop !== undefined ? options.prefix_prop : '.';
|
||
var prefix_attr = options.prefix_attr !== undefined ? options.prefix_attr : '';
|
||
|
||
for (let prop of props) {
|
||
var selector = prefix_prop + prop;
|
||
var html = options.empty_props ? '' : this.get_prop(object, prop);
|
||
$container.find( selector ).html( html );
|
||
}
|
||
|
||
for (let attr of attrs) {
|
||
var name = prefix_attr + attr;
|
||
var value = options.empty_attrs ? '' : this.get_attr(object, attr);
|
||
$container.attr( name, value );
|
||
}
|
||
|
||
return $container;
|
||
}
|
||
|
||
}
|
||
|
||
|
||
/**
|
||
* @memberof Formatters
|
||
* @extends Formatters.Formatter
|
||
*/
|
||
class AccountFormatter extends Formatter {
|
||
|
||
/**
|
||
* Properties renderable in html.
|
||
* @default {@link Models.Account.props}<tt> + ['balance_ukf']</tt>
|
||
*/
|
||
static get props() {
|
||
return Account.props.concat(['balance_ukf']);
|
||
}
|
||
|
||
/**
|
||
* Added attributes to $container element.
|
||
* @default <tt>['data-balance']</tt>
|
||
*/
|
||
static get attrs() {
|
||
return ['data_balance'];
|
||
}
|
||
|
||
static get _data_balance() {
|
||
return {
|
||
'default': '', 'frozen': 'frozen',
|
||
'ok': 'ok', 'low': 'low', 'neg': 'neg',
|
||
};
|
||
}
|
||
|
||
/**
|
||
* <tt>a.balance</tt> with two decimals.
|
||
*/
|
||
static prop_balance(a) {
|
||
return a.balance.toFixed(2);
|
||
}
|
||
|
||
/**
|
||
* <tt>'COF' if account is cof or 'Non-COF'
|
||
*/
|
||
static prop_is_cof(a) {
|
||
return a.is_cof ? 'COF' : 'Non-COF';
|
||
}
|
||
|
||
/**
|
||
* Value of data_attribute according to is_frozen status and balance of
|
||
* account <tt>a</tt>.
|
||
*/
|
||
static attr_data_balance(a) {
|
||
if (a.is_frozen) { return this._data_balance.frozen; }
|
||
else if (a.balance >= 5) { return this._data_balance.ok; }
|
||
else if (a.balance >= 0) { return this._data_balance.low; }
|
||
else /* a.balance < 0 */ { return this._data_balance.neg; }
|
||
}
|
||
|
||
}
|
||
|
||
|
||
/**
|
||
* @memberof Formatters
|
||
* @extends Formatters.AccountFormatter
|
||
*/
|
||
class LIQFormatter extends AccountFormatter {
|
||
|
||
/**
|
||
* Rendering a property always returns the empty string
|
||
* @default <tt>''</tt>
|
||
*/
|
||
static get_prop() {
|
||
return '';
|
||
}
|
||
|
||
/**
|
||
* Attribute <tt>data_balance</tt> is always <tt>ok</tt>.
|
||
*/
|
||
static attr_data_balance(a) { return this._data_balance.ok; }
|
||
|
||
}
|
||
|
||
|
||
/**
|
||
* @memberof Formatters
|
||
* @extends Formatters.Formatter
|
||
*/
|
||
class CheckoutFormatter extends Formatter {
|
||
|
||
/**
|
||
* Properties renderable to html.
|
||
* @default {@link Models.Checkout.props}
|
||
*/
|
||
static get props() {
|
||
return Checkout.props;
|
||
}
|
||
|
||
/**
|
||
* <tt>c.balance</tt> with two decimals.
|
||
*/
|
||
static prop_balance(c) {
|
||
return c.balance.toFixed(2);
|
||
}
|
||
|
||
}
|
||
|
||
|
||
/**
|
||
* @memberof Formatters
|
||
* @extends Formatters.Formatter
|
||
*/
|
||
|
||
class StatementFormatter extends Formatter {
|
||
|
||
/**
|
||
* Properties renderable to html.
|
||
* @default {@link Models.Statement.props}
|
||
*/
|
||
static get props() {
|
||
return Statement.props;
|
||
}
|
||
|
||
/**
|
||
* <tt>s.balance_old</tt> with two decimals.
|
||
*/
|
||
static prop_balance_old(s) {
|
||
return s.balance_old.toFixed(2);
|
||
}
|
||
|
||
/**
|
||
* <tt>s.balance_new</tt> with two decimals.
|
||
*/
|
||
static prop_balance_new(s) {
|
||
return s.balance_new.toFixed(2);
|
||
}
|
||
|
||
/**
|
||
* <tt>s.at</tt> formatted as <tt>DD/MM/YY à HH:MM</tt>.
|
||
*/
|
||
static prop_at(s) {
|
||
return moment.isMoment(s.at) ? s.at.tz('Europe/Paris').format('DD/MM/YY à HH:mm') : s.at;
|
||
}
|
||
|
||
}
|
||
|
||
|
||
/**
|
||
* @memberof Formatters
|
||
* @extends Formatters.Formatter
|
||
*/
|
||
|
||
class ArticleCategoryFormatter extends Formatter {
|
||
|
||
/**
|
||
* Properties renderable to html.
|
||
* @default {@link Models.ArticleCategory.props}
|
||
*/
|
||
static get props() {
|
||
return ArticleCategory.props;
|
||
}
|
||
}
|
||
|
||
|
||
/**
|
||
* @memberof Formatters
|
||
* @extends Formatters.Formatter
|
||
*/
|
||
class ArticleFormatter extends Formatter {
|
||
|
||
/**
|
||
* Properties renderable to html.
|
||
* @default {@link Models.Article.props}
|
||
*/
|
||
static get props() {
|
||
return Article.props;
|
||
}
|
||
|
||
static get attrs() {
|
||
return ['data_stock'];
|
||
}
|
||
|
||
/**
|
||
* <tt>s.price</tt> converted to UKF.
|
||
*/
|
||
static prop_price(s) {
|
||
return amountToUKF(s.price, true);
|
||
}
|
||
|
||
static get _data_stock() {
|
||
return {
|
||
'default': '', 'low': 'low',
|
||
'ok': 'ok', 'neg': 'neg',
|
||
};
|
||
}
|
||
|
||
/**
|
||
* Value of data_stock attribute based on s.stock.
|
||
*/
|
||
static attr_data_stock(s) {
|
||
if (s.stock > 5) { return this._data_stock.ok; }
|
||
else if (s.stock >= -5) { return this._data_stock.low; }
|
||
else /* s.stock < -5 */{ return this._data_stock.neg; }
|
||
}
|
||
}
|
||
|
||
|
||
/**
|
||
* @extends Formatters.Formatter
|
||
* @memberof Formatters
|
||
* @todo don't display trigramme in account_read
|
||
*/
|
||
class HistoryGroupFormatter extends Formatter {
|
||
|
||
/**
|
||
* <tt>a.time<tt> formatted as <tt>HH:mm:ss</tt>
|
||
*/
|
||
static prop_time(a) {
|
||
return a.at.format('HH:mm:ss');
|
||
}
|
||
|
||
/**
|
||
* <tt>valid_by</tt> property is displayed only if <tt>a.valid_by</tt> is nonempty.
|
||
*/
|
||
static prop_valid_by(a) {
|
||
if (a.valid_by)
|
||
return 'Par '+a.valid_by;
|
||
else
|
||
return '';
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @memberof Formatters
|
||
* @extends Formatters.HistoryGroupFormatter
|
||
*/
|
||
class TransferGroupFormatter extends HistoryGroupFormatter {
|
||
|
||
/**
|
||
* Properties renderable to html.
|
||
* @default {@link Models.TransferGroup.props} <tt> + ['infos', 'time']</tt>
|
||
*/
|
||
static get props() {
|
||
return TransferGroup.props.concat(['infos', 'time']);
|
||
}
|
||
|
||
/**
|
||
* Generic info for transfergroups.
|
||
*/
|
||
static prop_infos(a) {
|
||
//TODO find better thing to put here
|
||
return 'Transferts';
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @memberof Formatters
|
||
* @extends Formatters.HistoryGroupFormatter
|
||
*/
|
||
class OpegroupFormatter extends HistoryGroupFormatter {
|
||
|
||
/**
|
||
* Properties renderable to html.
|
||
* @default {@link Models.OperationGroup.props} <tt> + ['time']</tt>
|
||
*/
|
||
static get props() {
|
||
return OperationGroup.props.concat(['time']);
|
||
}
|
||
|
||
/**
|
||
* <tt>a.amount</tt> displayed according to <tt>a.is_cof</tt> and <tt>a.trigramme</tt> values.
|
||
*/
|
||
static prop_amount(a) {
|
||
return amountDisplay(a.amount, a.is_cof, a.trigramme);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @extends Formatters.Formatter
|
||
* @memberof Formatters
|
||
*/
|
||
class DayFormatter extends Formatter {
|
||
|
||
/**
|
||
* Properties renderable to html.
|
||
* @default {@link Models.Day.props}
|
||
*/
|
||
static get props() {
|
||
return Day.props;
|
||
}
|
||
|
||
/**
|
||
* <tt>a.date</tt> formatted as <tt>D MMMM</tt>
|
||
*/
|
||
static prop_date(a) {
|
||
return a.date.format('D MMMM');
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @extends Formatters.Formatter
|
||
* @memberof Formatters
|
||
*/
|
||
class OperationFormatter extends Formatter {
|
||
|
||
/**
|
||
* Properties renderable to html.
|
||
* @default <tt>['amount', 'infos1', 'infos2', 'addcost', 'canceled']</tt>
|
||
*/
|
||
static get props() {
|
||
return ['amount', 'infos1', 'infos2', 'addcost', 'canceled'];
|
||
}
|
||
|
||
static get attrs() {
|
||
return ['canceled'];
|
||
}
|
||
|
||
/**
|
||
* <tt>a.amount</tt> displayed according to <tt>a.is_cof</tt> and <tt>a.trigramme</tt> values.
|
||
*/
|
||
static prop_amount(a) {
|
||
return amountDisplay(a.amount, a.is_cof, a.trigramme);
|
||
}
|
||
|
||
/**
|
||
* <tt>addcost</tt> property is displayed iff <tt>a.addcost_for</tt> is nonempty.
|
||
*/
|
||
static prop_addcost(a) {
|
||
if (a.addcost_for) {
|
||
return '('+amountDisplay(a.addcost_amount, a.is_cof)
|
||
+'UKF pour '+a.addcost_for+')';
|
||
} else {
|
||
return '';
|
||
}
|
||
}
|
||
|
||
/**
|
||
* <tt>canceled</tt> property is displayed iff <tt>a.canceled_at</tt> is defined.
|
||
*/
|
||
static prop_canceled(a) {
|
||
if (a.canceled_at) {
|
||
var cancel = 'Annulé';
|
||
if (a.canceled_by)
|
||
cancel += ' par '+a.canceled_by;
|
||
|
||
cancel += ' le '+a.canceled_at.format('DD/MM/YY à HH:mm:ss');
|
||
return cancel ;
|
||
} else {
|
||
return '';
|
||
}
|
||
}
|
||
|
||
static attr_canceled(a) {
|
||
return a.canceled_at ? 'true' : 'false' ;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @extends Formatters.OperationFormatter
|
||
* @memberof Formatters
|
||
*/
|
||
class PurchaseFormatter extends OperationFormatter {
|
||
|
||
static prop_infos1(a) {
|
||
return a.article_nb;
|
||
}
|
||
|
||
static prop_infos2(a) {
|
||
return a.article_name;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @extends Formatters.OperationFormatter
|
||
* @memberof Formatters
|
||
*/
|
||
class SpecialOpeFormatter extends OperationFormatter {
|
||
|
||
/**
|
||
* <tt>a.amount</tt> with two decimal places.
|
||
*/
|
||
static prop_infos1(a) {
|
||
return a.amount.toFixed(2)+'€';
|
||
}
|
||
|
||
static prop_infos2(a) {
|
||
return SpecialOperation.verbose_types[a.type] || '' ;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @extends Formatters.OperationFormatter
|
||
* @memberof Formatters
|
||
*/
|
||
class TransferFormatter extends OperationFormatter {
|
||
|
||
/**
|
||
* <tt>a.amount</tt> with two decimal places.
|
||
*/
|
||
static prop_amount(a) {
|
||
return a.amount.toFixed(2)+'€';
|
||
}
|
||
|
||
static prop_infos1(a) {
|
||
return a.from_acc;
|
||
}
|
||
|
||
static prop_infos2(a) {
|
||
return a.to_acc;
|
||
}
|
||
}
|