A JavaScript library that is meant to simplify things a bit.
Here is the documentation of EnderLibrary:
You can use the SettingsObject()
Object to change the current framework settings safely. You are only allowed to change the settings using this object so you can lock them whenever you want!
var defaultSettings = new SettingsObject(); //The SettingsObject always returns a fresh object with the default settings of the library.
defaultSettings.settings.message.all = false; //This means that the library will not show any messages in the console
You, of course, need to pass the new object to the library to change the settings.
The output object is the following:
{
settings: {
message: {
all: true,
error: true,
warn: true,
normal: true,
},
resources: false,
resourceRequest: true
}
}
You can use this object to communicate with the library when you need to make changes to the way it works.
{
lock: function(callback) { ... }, //callback(isSet, didFire), `isSet` is a boolean that tells you if the environment settings were locked properly or not. and `didFire` is a boolean that tells you if there was an attempt to change the environment settings while it's locked!
changeSettings: function(settingsObject) { ... }, //You can pass the SettingsObject with the changes you did to the environment using this function
isPageSecure: function(callback) { ... } //callback(result, score), this function is not ready yet, it'll be used to measure how secure the page is
}
You can use this object to change the loading behaviour of the page and the website! Dynamic loading is basically making it so the website never has to load the same resources all over again whenever you change the page. (You might know this kind of loading as Ajax loading)
There are three protocols for the dynamic loading object:
"not-dynamic"
, "dynamic"
, and "fully-dynamic"
!
The "not-dynamic"
protocol is the default protocol used whenever you load the library! It means that dynamic loading is disabled by default.
In the "dynamic"
protocol, the page will load dynamically whenever the pathname changes - unless it contains search arguments in the URL, e.g. https://example.com/?name=c -. By default, it will follow the filtering process. You can prevent that by listening to the 'content-loaded' event and prevent the next step (It's not recommended that you do that, as EnderLibrary has a good enough automatic filtering system for most use cases):
window.location.dynamic.on('content-loaded', function(e){ //This event is fired when the content of the targeted page is loaded (with no modifications)
e.preventInsert(); //This will prevent the library from inserting the content to the page
e.content; //You can do whatever you wanna do with the content of the page!
})
The "fully-dynamic"
protocol doesn't care about the search arguments, it will always load dynamically! And links in the page itself will be automatically loaded dynamiclly when pressed, so you don't have to deal with them manually.
However, you should always use the refresh function whenever you modify the DOM content of the page:
window.location.dynamic.links.refresh(); //This function is applied automatically when the page content is inserted by the default loading process. But if you change the DOM content by yourself, new elements aren't effected by the current protocol unless you run this command.
removeEventListener('click', window.location.dynamic.links._eventFunction); //If you wish to modify an element, you can use the _eventFunction variable to remove the eventListner that was set my the library!
You can change the protocol whenever you wish to do so, you can just change the value of the protocol variable:
window.location.dynamic.protocol = "not-dynamic"; //Disable dynamic loading
window.location.dynamic.protocol = "dynamic"; //Enable dynamic loading
window.location.dynamic.protocol = "fully-dynamic"; //Enable full dynamic loading
You can change the page URL when the dynamic protocol is enabled just like this:
window.location.dynamic.assign(url);
/*or*/
window.location.dynamic.href = url;
window.location.dynamic.reload(); //You could also reload the page using this
If you wish to retrieve the value of the last url before switching to the dynamic protocol, you can use the _href
variable:
window.location.dynamic._href;
window.location.dynamic._pathname;
window.location.dynamic._search;
window.location.dynamic._hash;
The search function only works with the "fully-dynamic"
protocol:
window.location.dynamic.search = "?name=value";
window.location.hash = "hmm"; //There is no hash variable for the dynamic object, as the normal hash object is still 100% functional!
There are many events for the dynamic object:
window.location.dynamic.on('protocol-change', function(){ //Fired when the dynamic protocol changes
//Your Code
});
window.location.dynamic.on('loading-start', function(e){ //Fired when a loading attempt is made
e.url; //The target URL
e.isReplacement; //Is this URL a replacement for the current URL history
});
window.location.dynamic.on('loading-end', function(){ //Fired when a loading process is finished successfully
//Your Code
});
window.location.dynamic.on('loading-failed', function(stageCode, error){ //Fired when a loading process fails
if(stageCode == -1){
//Failed at the targeting stage, probably because of a cross-domain or a security policy!
}else if(stageCode == 0){
//Failed when trying to load the page content, could be a problem with the connection!
}else if(stageCode == 1){
//Failed while trying to filter the page content!
}else if(stageCode == 2){
//Failed while trying to insert the page content!
}
});
window.location.dynamic.on('content-loaded', function(e){ //Fired when the content of the targeted page is loaded, but not yet filtered or inserted into the page!
e.content; //The content of hte page
e.preventInsert(); //A function to prevent the content from being inserted into the page
e.status; //The request status code
e.url; //The URL object of the page
e.refresh(); //The links refresh function (window.location.dynamic.links.refresh)
});
Warning: never use the dynamic protocol with pages that are too different from the current page! That would be so inefficient. You can, for example, do this to prevent such a thing from happening when you have the dynamic protocol enabled:
window.location.dynamic.on('content-loaded', function(e) {
if (e.url.pathname.indexOf("/different") == 0) { //In this example, the way the page `/different` and its sub-pages are built is completely different from the way the current page is built and structured!
e.preventInsert();
window.location.href = e.url.href;
}
...
});
The _dynamic
attribute is used in the default filtering process for the dynamic protocol. It can tell the library where each element belongs in the page!
It will find already-existing elements in the page with the same value of the _dynamic
attribute and replace them with the new element!
<div _dynamic="topPage">...</div> <!-- The dynamic element with the value of topPage will be replaced with this element! -->
You can't have the _
operator at the start of the value inside the _dynamic
attribute! You can't also add a double @
operator (@@
) -unless you wanna specify a command-. These are the only limitations of this value. (You can add spaces, special charecters, etc.)
<div _dynamic="_topPage">...</div> <!-- This is wrong, this may result in an error being thrown while inserting the page content! -->
<div _dynamic="topPage_">...</div> <!-- Nothing bad will happen here, you're good to go! -->
<div _dynamic="top Page">...</div> <!-- Don't worry, spaces are allowed too! -->
Of course, websites might need to do more things other than replacing an already existing element into the page!
Dynamic elements that have not been assigned a type will be inserted at the bottom of the <body>
element!
The code used inside the _dynamic
attribute follows this syntax system:
<div _dynamic="@@(<method_name>)(<value>)(<additional_value>)... [@@(<...>)...]"></div>
The string between the opening and closing Parentheses is whitespace-sensitive, so be careful with it!
<div _dynamic="@@(! selector)(.name)"></div> <!-- Wrong! -->
<div _dynamic="@@( selector)(.name)"></div> <!-- Wrong! -->
<div _dynamic="@@(!selector )(.name)"></div> <!-- Wrong! -->
<div _dynamic="@@(!selector)(.name)"></div> <!-- Right! -->
<div _dynamic="@@(selector)(.name)"></div> <!-- Right! -->
The selector method works mostly like the document.querySelectorAll()
method. But there is an extra thing in there. You can use the @
operator to tell the library where to put your element!
<div _dynamic="@@(selector)(body @top)"></div> <!-- "Put this element in the top section of the body element!" -->
<div _dynamic="@@(selector)(body @bottom)"></div> <!-- "Put this element in the bottom section of the body element!" -->
<div _dynamic="@@(selector)(body @0)"></div> <!-- "Put this element before the 1st element in the body element (index 0, meaning that it should be the first element after it's inserted, this is the same as `@top`)" -->
<div _dynamic="@@(selector)(body @1)"></div> <!-- "Put this element before the 2nd element in the body element" -->
<div _dynamic="@@(selector)(body @#myElement)"></div> <!-- "Put this element before the element with the id myElement" -->
Note that everything written after the @
operator will be used to select where you want to insert the element!
<div _dynamic="@@(selector)(body @.myElement .mine)"></div> <!-- "Put this element before the element with the class 'myElement', that has a child with the class 'mine'!" If no element -one that has the class 'myElement' and has a child with the class 'mine'- is present, this selector will fail! -->
<div _dynamic="topPage">...</div> <!-- The dynamic element with the value of topPage will be replaced with this element! -->
<div _dynamic="@@(defineder)(topPage) [...]">...</div> <!-- The dynamic element with the value of topPage will be replaced with this element! -->
In case you wanna use a method, and keep the element defineded, you can use the defineder
method!
The dynamic attribute can also use selectors to tell the library where to insert this element. In case a selector returns an empty result, it will be inserted at the bottom of the document. (Unless it's a strict one!)
<div _dynamic="@@(selector)(#myElement)">...</div>
This element will be inserted in the element with the ID "myElement". If no such element is found, it will be inserted at the bottom of the document.
If you use a selector and a defineder in the same attribute, priority will be given to the defineder method. If you want to prioritise the selector method, you need to use a strict selector:
<div _dynamic="@@(defineder)(topPage) @@(selector)(#myElement)">...</div> <!-- If no `topPage` dynamic element is present, the selector will be followed. If not, the selector will be ignored! -->
<div _dynamic="@@(defineder)(topPage) @@(!selector)(#myElement)">...</div> <!-- The selector will be followed first here! If there is no match, the defineder method will be followed after! -->
<div _dynamic="@@(!selector)(#myElement)">...</div>
This element will be inserted in the element with the ID "myElement". However, this is a strict selector, strict selector mean that this element will not be inserted into the page if not match is found! (Unless a defineder method is present)
<div _dynamic="@@(selector)(.myElement)">...</div>
This element will be inserted in the element with the class name "myElement". If no such element is found, it will be inserted at the bottom of the document. And if there is more than one matching result, it will be inserted into the bottom of the document. And if an index is specified, it will be inserted in that index (if it was present). (NOTE: if this was a strict selector, and no index was present/the specified index was out of range, the element will not be inserted!)
To specify an index, you could do this:
<div _dynamic="@@(selector)(.myElement)(0)">...</div>
You can also use the resource method!
<div _dynamic="@@(resource)(<type>)">...</div>
resource will be inserted to the page according to the type!
Style
resources will be inserted in the <head>
element, and Scripts will
be inserted at the bottom of <body>
.
If a dynamic resource loads an already existing resource in the page, it will not be inserted!
<element _dynamic="@@(!resource)(<type>)" />
If the resource is strict, it will replace the already loaded resource.
Meta resources will be inserted at the top of the page! (Remember, meta data is not important in most of the use cases, so don't just add Meta tags randomly)
<div _dynamic="@@(selector)(...) @@(resource)(...)">...</div> <!-- This is wrong! -->
Please note that you can't use the resource method with a selector method on the same element. Only the first method menthioned in the _dynamic
attribute will be followed!
<div _dynamic="... @@(rule)(unload)">...</div>
Rules are, well, rules that the library will not break -no matter the condition-. The unload
rule means that the specified dynamic element will be unloaded/removed once the page content changes.
<div _dynamic="... @@(rule)(constant)">...</div>
The constant
rule means that once this dynamic element is loaded, it will never be removed! Even when a strict selector is trying to remove it!
<div _dynamic="... @@(rule)(flex)">...</div>
The flex
rule means that once this dynamic element is loaded, it will be recognised as a flexible element! It will mostly behave like a normal element. But, when it gets replaced by another element, it will be saved for use later. When another load attempt happens, the element that replaced this element will be replaced again by the flexible element!
in addition to the normal forEach, you can use this function for all types of objects:
forEach(Object, function(item){
//Do stuff
});
You can get, set, and remove cookies using these functions without the need to parse the cookies string:
document.cookies.set(cookieName); // This will set a cookie with the value of an empty string (""), and it will expire in 30 days
document.cookies.set(cookieName, cookieValue); //This cookie will expire in 30 days too
document.cookies.set(cookieName, cookieValue, expirationInDays);//You can, of course, change the time before this cookie expires
document.cookies.get(cookieName); //This will return the value of this cookie
document.cookies.remove(cookieName); //This will remove the specified cookie
You can get the values of the URL parameters without the need to parse the search string:
document.parameters.get(parameterName); //This will return the value of the parameter
document.parameters.get("*"); //This will return a JSON object with all the values of all URL parameters
You can create element templates for repetitive elements:
var myTemplate = document.createElementTemplate({ //This function will return an ElementTemplate object
tag: "div"
class: "myClass1 myClass2",
name: "rep",
_content: "Some Random Bullshit"
});
var element1 = myTemplate.createElement(), //You can use the `createElement()` function on the ElementTemplate object to generate the element itself
element2 = myTemplate.createElement(); //You are not limited in numbers at all
This is still an experimental feature, it's not coded yet, and it might be canceled!
document.resources; //Not functional yet! This is a JSON object that will contain all the resources loaded using the <document-resource> element
This object is not coded yet, as it might actually be a really bad idea to enourge the user to use this feature!
document.contentSecurityPolicy; //Not functional yet!
This object is not coded yet, it will be used to make requests for resources in the page. The reason to introduce this object is so the user can easily block requests using the environment settings.
var resourceRequest = new ResourceRequest(url, settings); //Not functional yet!
You can use this object to make classList changes on a global scale to an element:
var posts = document.getElementById("posts-container").getGlobalClassList(); //You can use the `getGlobalClassList()` function on an element to select all elements inside it!
posts.remove(class1, class2, ...);
posts.add(class1, class2, ...);
posts.replace(oldClass, newClass);
This is simply the same as doing <element>.outerHTML = ""
!
You can use this function to insert elements inside the selected parent element:
var myNewElement1 = document.createElement("div"), myNewElement2 = document.createElement("div");
document.body.insertElement(myNewElement1); //This element will be inserted at the bottom of the parent element (In this case, it's the <body> element)
document.body.insertElement(myNewElement2, true); //This element will also be inserted to the parent element, but at the top of it! The second argument is used to tell the insert function where to insert the element (false ==> bottom, true ==> top). The default value is false.
This will insert your CSS code in a style tag inside the specified element!
This will insert your JS code in a script tag inside the specified element!
This will set the pointer-events
and user-select
values to "none", and will set the draggable attribute to "false". This means that the user can't interact with this element at all!
This will set the pointer-events
and user-select
values to "auto", and will set the draggable attribute to "true".
This will return an array of all elements inside the specified parent element!
This will return an array of all elements that are in the top level of the specified element! Top level elements are elements that are assigned the specified element as a parent.
It has been generalised to include more types of objects!
This will empty the specified array!
This will replace strings on a global space in the Array! (It works like the normal replace function, but it applys to all the members of the array)
This will convert the specified string to a normal HTML element and return it.