Working with Arrays in Javascript requires manual iteration, is error-prone, difficult to read, repetitive and time-consuming.
PowerArray simplifies that bads by expanding the Array prototype, adding simple but powerfull capabilities to any Array
.
Worried about changes on the Array Prototype?
This documentation is under construction!
Examples using using the Where, Sort, Take, Remove and RunEach prototype functions and different PowerArray global auxiliary functions.
//participants = [ {name : 'Alex', age: 20, categories: [ {categoriId: 10, score: 3.1}, {categoryId: 11, score: 5} ], {...}, ...];
participants
.Where({ age : Between(20, 30), categories: {categoryId : 1}}) // Where => to filter. 1 condition-object with 2 conditions
.Sort({ categories : { score: Sort.Ascending }) // Sort => Sort ascending by category score from categories (an array of objects)
.Take(10) // Take only the first 10 results
.RunEach(function(participant, i) {
console.debug(i + ' ' + participant.name);
});
//programmatically generated Conditions object:
var conditions = {};
if(have_to_check_age)
conditions.age = Between(20, 30);
if(have_to_check_fav_colors) // colors_to_check = ["red", "green","yellow"]
conditions.favoriteColor = In(colors_to_check);
results = participants.Where(coditions);
Removing data:
peopleArray.Remove({ lastName: IsEmpty()}); // IsEmpty covers "", null, undefined, []
peopleArray.Remove({ address: IsDefined()});
It also adds global auxiliary functions (like 'Between', 'IsEmpty' from the previous examples), that are ready to be embedded on Conditions-Object.
PowerArray have different types of functions. Each function type is designed for a different scenario and have a different usage. There are mainly three functions types/use cases:
This type of functions are those that can be directly invoked on your own Array. It is not necessary to make anything with your array before using this function. The most used prototyped functions are: Where, First, Last, Sort, RunEach, Take, Remove, etc. This kind of functions returns always a new array, making all this functions types chainable which each other: After filtering an array by using the Where functions, the resulting array cann immediately be used for another prototyped function like for example Sort.
someArray
.Where({...conditions-object...}) // Prototyped function Where used
.Sort({...sort criteria....}) // Prototyped function Sort used
.RunEach(function(item, i, arr) { ...do stuff...}); // Prototyped function RunEach used
This kind of functions are those that can be used as values on an Object-Condition. We call this functions "Auxiliar" functions, like In, Contains, Between, GreaterThan, etc. complete list here
someArray
.Where({
someProp: GreaterThan(24) //Auxiliar global function GreaterThan used
otherProp: Between(10,20) //Auxiliar global function Between used
});
The functions that are stored on the pa object can be invoked as usual:
pa.utils.GetTypeOf([1,2,3]); //Utility function pa.utils.GetTypeOf used
pa.utils.GetTypeOf("hello");
var r = pa.utils.parseBoolean("true"); //Utility function pa.utils.parseBoolean used
pa.utils.isNullEmptyOrUndefined(null); //Utility function pa.utils.isNullEmptyOrUndefined used
var args = pa.utils.ArgumentsToArray(arguments); //Utility function pa.utils.ArgumentsToArray used
PowerArray adds also some auxiliary functions to avoid writing the same snippets over and over again. All they are accesible after loading the library, click here for a complete list. #AuxiliarFunctionsDescription
To simplify filtering tasks, PowerArray relies mainly on the Where function, which offers a standard mechanism to formulate filtering conditions.
The return value of a where function call, is ALWAYS a new array, in which each position is a reference to an item on your original array that fulfilled the conditions. The Where function doesn't change anything on the original array, it just creates a new array of references to all matching elements. Because it returns an array, it's also chainable with other PowerArray functions like Sort or RunEach.
You can pass conditions to the Where function (first argument in all signatures) as functions or by using Conditions-objects. It is necessary to understand Conditions-objects before you start using the Where function.
anyArray.Where(conditionsObject ,keepOrder)
- conditionsObject => type
Object
(a Conditions-Object) - keepOrder => type
boolean
, indicating if the original order should be kept or not. Optional, default false.
Examples:
//given an Array of objects called 'peopleArray'
//Filtering by first level properties 'name' and 'lastname'
var result = peopleArray.Where({ name: 'John', lastName: 'Lennon' });
//Filtering by first level property 'name' and by second level property 'partner'
var result = peopleArray.Where({
name: 'John',
partner: { name: 'Yoko'} // 'partner' is an object, and it's expected to have a property name
});
//Multiple conditions and auxiliar functions
var result = peopleArray.Where({
age: Between( 40, 50) ,
hobbies: Contains('Swimming'), // 'hobbies' is an array of strings. Auxiliar function 'Contains' used
favoriteColor: In('Black','Blue') // 'favoriteColor' is a string. Auxiliar function 'In' used
});
.Where([conditionsObjects] ,keepOrder)
- conditionsObject => type
Array
(of Conditions-Objects) - keepOrder => type
boolean
, indicating if the original order should be kept or not. Optional, default false.
This signature accepts an array of Conditions-Objects. If an item fulfill all conditions of at least one
Condition-Object, it will be included in the results (and other Condition-Objects will not be evaluated for that item).
This Signature is ideal to build (a kind of) OR
statements that cannot be expressed in a single Conditions-Object.
Examples:
// Given an Array of objects representing persons, called 'peopleArray'
// Task: find all single men between 20 and 30 years old and all married woman between 25 and 35 years old
var result = peopleArray.Where([
{ gender : 'M', age : Between(20,30), status: 'Single' }, // Auxiliar function 'Between' used
{ gender : 'W', age : Between(25,35), status: 'Married' }
]);
// Given an array of orders called 'ordersArray'
// Task: find all invoices "from location "Sydney" having an amount <= 1000" or "having amount > 50000, regardless location"
var result = ordersArray.Where(
{ amount : SmallerOrEqualThan(1000), location : 'Sydney' }, // Auxiliar function 'SmallerOrEqualThan' used
{ amount : GreaterThan(50000) } // Auxiliar function 'GreaterThan' used
);
// In this two exmaples, we have multiple criterions that overlap each oder (age on the first example, amount on the second).
// This overlap makes impossible to formulate both criteria on a single Conditions-Object, and that's why it's neccesary to use this signature in such cases.
.Where(func ,keepOrder)
- func type Function.
- keepOrder type boolean, indicating if the original order should be kept or not. Optional, default false.
This signature can be used when filtering arrays of any type. Pass a function which will receive the item to evaluate, the index of that item on the original array, and the array self as parameters in that order.
Examples:
//Filtering arrays of primitives
[1, 2, 3, 4, 5, 6, 7].Where(GreaterThan(4)));
// returns [5,6,7]
["Red","Green", "Black", "Blue"].Where(Like('e')); //using auxiliary function Like
//returns ["Blue", "Green", "Red"]
// Given an Array of objects representing persons, called 'peopleArray'
var somePeople = peopleArray.Where(function (person, index, array) {
//parameter person => an item of people array
//parameter Index => the index of the current person item in the original array
//array => a reference to the array we are filtering (array[index] is always equal to person)
if(/*conditions*/)
return true; //returning true (or any truthy value), the item will be included on the result.
return false; //return false to exclude it.
});
It's a plain Javscript object containing N filtering criteria, that can be used to filter objects arrays. Each property present on a given Conditions-Object, represents a filtering criteria for a property that should be present on each array item in which the filtering is applied.
In the following code:
var result = peopleArray.Where({favoriteColor: 'Red'});
the Conditions-Object is {favoriteColor: 'Red'}
. It's expected that peopleArray be an array of objects, each having a property called 'favoriteColor'.
Within a single Conditions-Object, you can can also target items sub-properties if they have (mostly on objects-arrays), for Example:
var result = peopleArray.Where({ //peopleArray is an array of objects representing persons
favoriteColor: 'Red', //each person has a first-level String property called "FavoriteColor". We pass the value we are searching for as explicit primitive
Childs: { // to another first-level property called "childs", which contains an array of objects, we pass a new Contdition-object.
name: In('John','Juan', 'Joan') //on the new condition object, we can filter by property name. In this case by using the auxiliar function 'In'
}
});
It is known that changing prototypes or working with global functions is not really wanted, because horrible things could happen. I agree with that.
Because it can strongly improve how you write and organize your code, opening up really beautiful possibilities. Arrays are a fundamental part of any Javascript solution, they are everywhere! Extending them by modifying your prototype allows programmers to do things that are normally not possible to build in such a simple way (without using transpilers or just writing more code). Here some of them:
-
Sql-like access for array filtering (deep):
var germanClients = clientsArray.Where({city: In('Berlin','Hamburg'), });
-
Chaining functions The PowerArray prototyped functions Where, Sort, Take and RunEach returns always an array, making them chainable, allowing us to filter, sort and execute in a single line way:
clientsArray .Where({city: In('Berlin','Hamburg')}) .Sort({companyName: Sort.Ascendig}) .RunEach((item){ ...do stuff... });
The vast majority of times, we (programmers, architects, etc.) can control which libraries and modules will live in our javascript ecosystem and which not, strongly reducing the collision risks. Until now, there are not known issues, and the advantages far outweigh the potential disadvantages: write less code that is much easier to understand: simply save time and money ;)
No, it just adds new things, and only if the desired names are not already taken. If you add this library to a project, your existing code will definitively not be influenced. The extra functionalities that PowerArray adds to Javascript have to be explicitly called to work.
Basically, PowerArray loads everything he needs to work on his own global object called "pa", as many frameworks do. The "pa" object is a container, in which there are multiple functions. Some of this functions, are designed to work with any object with Array prototype (or array like objects), and others designed to operate globally (internal pa.auxiliaryFunctions).
During the initialization process, each of such functions is is evaluated, to check if the name is already in use before modifying anything. Only if they are free, a pointer to the corresponding pa functions is set on the prototype array, or the global scope.
If a name that a PowerArray function wants to use is already occupated by other framework, library or whatever, PowerArray don't change anything, just sends a warning to the console.
The 'only consequence' will be a change on the syntax you have to use to run the specific PowerArray function:
-
For global functions (as all auxiliary functions GreaterThan, SmallerThan, In, InIgnoreCase, Like, NotLike, etc.) you have to use the prefix "pa.".
For example by using the pa standard function SmallerThan:
use var result = someArray.Where({ priority: pa.SmallerThan(5)});
instead of var result = someArray.Where({ priority: SmallerThan(5)});
<li>
For prototype functions (Where, Sort, findByProperty, findIndexByProperty, etc.), you have to envolve the array in which you want to make an operation with a function call. For Example:<br>
<table>
<tr>
<td>use</td>
<td><code>var result = <font color=red>pa(</font>someArray<font color=red>)</font>.Where({ categoryId : 33});</code></td>
</tr>
<tr>
<td>instead of</td>
<td><code>var result = someArray.Where({ categoryId : 33});</code></td>
</tr>
</table>
</li>
Considering all that, if you load the PowerArray library as last in your html, the the potential problems are minimal. If there is any problem the solution will be always the same: add "pa." as prefix, or surround your array with function "pa()" before use it.
*Under Construction (the readme file, the library works fine :)*n
The MIT License (MIT)
Copyright (c) 2015 Sebastian Menendez (sebastian.menendez[at]gmail.com)
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.* *