Super Enum Factory: SEnum(list)
Enums are great, for reducing a list of values, keys, labels, etc
In computer programming, an enumerated type (also called enumeration or enum, or factor in the R programming language, and a categorical variable in statistics) is a data type consisting of a set of named values called elements, members or enumerators of the type.
https://en.wikipedia.org/wiki/Enumerated_type
You should always use enums when a variable (especially a method parameter) can only take one out of a small set of possible values. Examples would be things like type constants (contract status: "permanent", "temp", "apprentice"), or flags ("execute now", "defer execution").
http://stackoverflow.com/a/4709224/194105
You end up with values
which are smaller, so less goes into your database, and
less "over the wire", from server to client and back.
The translation from values
to labels
is all in very small application code,
cached on the client.
Install
npm install --save s-enum
Terms
key
should be a camelCase string selector for any node.- if you provide one, we don't do anything to it
- if you do not provide one, we attempt to translate
label
to camelCase value
can be anything... usually a number or string or function- if you don't provide one, we attempt to create a numerical value incrementing
from
0
label
should be a human readable string- if you don't provide one, we use key
Usage Example: Days of the Week (basics)
First you create your enum object... this can happen anywhere
var Days = SEnum([
"Sunday",
"Monday",
"Tuesday",
"Wednesday",
"Thursday",
"Friday",
"Saturday"
]);
Days.values() === [0, 1, 2, 3, 4, 5, 6];
Days.keys() === ["sunday", "monday", "tuesday", "wednesday", "thursday", "friday", "saturday"];
Days.labels() === ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"];
Days.options() === {0: "Sunday", 1: "Monday", 2: "Tuesday", 3: "Wednesday", 4: "Thursday", 5: "Friday", 6: "Saturday"};
Days.0.label === "Sunday";
Days.0.key === "sunday";
Days.0.value === 0;
Days.sunday.label === "Sunday"
Days.sunday.key === "sunday"
Days.sunday.value === 0;
Days.get("sunday", "value") === 0;
Days.get(0, "label") === "Sunday";
Days.get("not-found", "value", "defaultValue") === "defaultValue";
Usage Example: Status
First you create your enum object... this can happen anywhere, but in this
example we are going to think about attaching this to an Object
(a data model / Collection object).
Tasks.Statuses = SEnum([
{ key: "submitted", value: 1, label: "Submitted",
icon: "fa fa-check" },
{ key: "accepted", value: 2, label: "Accepted",
icon: "fa fa-check-circle" },
{ key: "completed", value: 31, label: "All Done Yo",
icon: "fa fa-check-square", finished: true }
]);
Tasks.Statuses.get("submitted", "icon") === "fa fa-check";
Tasks.Statuses.get(1, "icon") === "fa fa-check";
Tasks.Statuses.get(31, "finished") === true;
Tasks.Statuses.get(1, "finished") === undefined;
Tasks.Statuses.get(1, "finished", "defaultValue") === "defaultValue";
Tasks.Statuses.1.icon === "fa fa-check";
Tasks.Statuses.submitted.icon === "fa fa-check";
Integrating into Meteor (optional)
When this was originally written, it was for a MeteorJS application.
The following logic is how you can automatically translate/transform inputs
into SEnum values before writing to the database. You can also validate to
only allow these values.
If you want, you can setup a
simple-schema
to only allow values from the SEnum.
allowedValues: Tasks.Statuses.values(),
allowedValues: Tasks.Statuses.values().concat(Tasks.Statuses.keys()),
You can setup a transform, translating all keys
& labels
into values
If you want to, you can setup a
Mongo.Collection transform
This will allow a user to pass in any value
or key
, and verify that it's got
a valid, value
; if not it changes to 0
.
Tasks = new Mongo.Collection('tasks', {
transform: function(doc) {
if (_.has(doc, "status")) {
doc.status = Tasks.Statuses.value(doc.status, 0);
}
return doc;
}
});
If you're using autoform/schemas you can setup autoValue
something like the
following.
This will allow a user to pass in any value
or key
, and verify that it's got
a valid, value
; if not it unsets... (you could choose not to unset, and
instead get validation errors based on allowedValues
).
status: {
type: Number,
optional: true,
autoValue: function() {
var autoFormField = this.field("status");
if (!autoFormField.isSet) {
this.unset();
return;
}
var status = Tasks.Statuses.value(autoFormField.value);
if (typeof status === "undefined") {
this.unset();
return;
}
return status;
}
}
And for any template, you can easily translate
value
-> label
or any value
-> any other field...
This allows some fancy and simple comparable translations.
Template.Example.helpers({
status: function() {
return Tasks.Statuses.label(this.status);
},
statusIcon: function() {
var icon = Tasks.Statuses.get(this.status, "icon");
if (!icon) {
icon = "fa fa-question";
}
return '<i class="' + icon '"></i>';
},
statusLabelAndIcon: function() {
var label = Tasks.Statuses.label(this.status, "Unknown");
var icon = Tasks.Statuses.get(this.status, "icon");
if (!icon) {
icon = "fa fa-question";
}
var finished = Tasks.Statuses.get(this.status, "finished");
var className = finished ? 'success' : 'default';
return '<span class="label label-' + className + '">' +
'<i class="' + icon '"></i> ' + label +
'</span>';
}
});
TODO / Roadmap