gestioCOF/kfet/static/kfet/js/kfet.api.js

1662 lines
43 KiB
JavaScript
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/**
* @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;
}
}