eduinaf-ajax-filter-posts/assets/js/ajax-filter-posts.js
Robbert de Kuiper e1f0aa5ed8 First commit
2017-03-30 16:28:54 +02:00

264 lines
7.4 KiB
JavaScript
Executable File

(function (){
var container = document.querySelector('.js-container-async');
var filterTogglers = container.getElementsByClassName('js-toggle-filters');
var queryParams = {
'page' : null,
'tax' : {},
'quantity' : 0,
'postType': 'post',
}
function init() {
if (container) {
// Set the amoun of post per page
queryParams.quantity = parseInt(container.dataset.quantity, 10);
queryParams.postType = container.dataset.postType;
// Add event listeners with event propagination
on(container,'click', 'a[data-filter]', function(event) {
handleFilterEvent(event.target);
event.preventDefault();
return true;
});
on(container, 'click', '.js-load-more', function(event) {
handleLoadMoreEvent(event.target);
event.preventDefault();
return true;
});
on(container, 'click', '.js-reset-filters', function() {
resetFilters();
event.preventDefault();
return true;
});
// Add event listner on all filter toggle buttons
[].slice.call(filterTogglers).forEach(function(button){
button.addEventListener('click', toggleFilters)
});
}
}
function handleFilterEvent(filter) {
if (!filter.classList.contains('is-active')) {
filter.classList.add('is-active');
updateQueryParams({
page: 1,
filter: filter.dataset.filter,
term: filter.dataset.term,
});
} else {
filter.classList.remove('is-active');
removeQueryParam(filter.dataset.filter, filter.dataset.term);
}
getAJAXPosts({reset: true});
}
function handleLoadMoreEvent(button){
updateQueryParams({ page: parseInt(button.dataset.page, 10) })
getAJAXPosts({reset: false});
}
function resetFilters() {
// Convert nodeList to array to prevent fail on for each
var activeFilters = Array.prototype.slice.call(container.querySelectorAll('a[data-filter].is-active'));
// Remove active classes
activeFilters.forEach(function(filter){
filter.classList.remove('is-active');
});
// Empty taxonomy from query
queryParams.tax = {};
queryParams.page = 1;
getAJAXPosts({reset: true});
}
function toggleFilters() {
console.log(container);
container.classList.toggle('is-expanded-filters');
}
function updateQueryParams(params) {
queryParams.page = params.page;
// If we're also updating the taxonomy
if (params.filter) {
if (queryParams.tax.hasOwnProperty(params.filter)) {
queryParams.tax[params.filter].push(params.term);
} else {
queryParams.tax[params.filter] = [params.term];
}
}
}
/**
* Remove a term from the set of query params
*
* @param string tax taxonomy of the term to remove
* @param {tring term term to remove
*/
function removeQueryParam(tax, term) {
if (queryParams.tax.hasOwnProperty(tax)) {
if (queryParams.tax[tax].indexOf(term) > -1) {
queryParams.tax[tax].splice( queryParams.tax[tax].indexOf(term) , 1 );
}
}
}
/**
* Get new posts via Ajax
*
* Retrieve a new set of posts based on the created query
*
* @return string server side generated HTML
*/
function getAJAXPosts(args) {
var content = container.querySelector('.ajax-posts__posts');
var status = container.querySelector('.ajax-posts__status');
// Set status to querying
container.classList.add('is-waiting');
var request = new XMLHttpRequest();
request.open('POST', filterPosts.ajaxUrl, true);
request.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
request.timeout = 2000; // time in milliseconds
request.onload = function() {
//remove load more button
var loadMoreButton = content.querySelector('.js-load-more');
if (loadMoreButton) {
content.removeChild(loadMoreButton);
}
var response = JSON.parse(this.response).data;
if (this.status === 200) {
if (args.reset){
content.innerHTML = response.content;
} else {
content.innerHTML += response.content;
}
}
else {
status.innerHTML = response.message;
}
// Resolve status
container.classList.remove('is-waiting');
};
request.ontimeout = function() {
status.innerHTML = filterPosts.timeoutMessage;
}
request.send(objectToQueryString({
action: 'process_filter_change',
nonce: filterPosts.nonce,
params: queryParams,
}));
}
/**
* Helper function for event delegation
*
* To add event listeners on dynamic content, you can add a listener
* on thewrapping container, find the dom-node that triggered
* the event and check if that node mach our
*
* @param NodeElement el wrapping element for the dynamic content
* @param string eventName type of event, e.g. click, mouseenter, etc
* @param string selector selector criteria of the element where the action should be on
* @param Function fn callback funciton
* @return Function The callback
*/
function on(el, eventName, selector, fn) {
var element = el;
element.addEventListener(eventName, function(event) {
var possibleTargets = element.querySelectorAll(selector);
var target = event.target;
for (var i = 0, l = possibleTargets.length; i < l; i++) {
var el = target;
var p = possibleTargets[i];
while(el && el !== element) {
if (el === p) {
return fn.call(p, event);
}
el = el.parentNode;
}
}
});
}
/**
* Convert an deep object to a url parameter list
*
* Boiled down from jQuery
*
* WordPress Ajax post request doesn't accept JSON only form-urlencoded!
* Took me a while to get...
* Although seems not to be totally true:
* http://wordpress.stackexchange.com/questions/177554/allowing-admin-ajax-php-to-receive-application-json-instead-of-x-www-form-url
*
*/
function objectToQueryString(a) {
var prefix, s, add, name, r20, output;
s = [];
r20 = /%20/g;
add = function (key, value) {
// If value is a function, invoke it and return its value
value = ( typeof value == 'function' ) ? value() : ( value == null ? "" : value );
s[ s.length ] = encodeURIComponent(key) + "=" + encodeURIComponent(value);
};
if (a instanceof Array) {
for (name in a) {
add(name, a[name]);
}
} else {
for (prefix in a) {
buildParams(prefix, a[ prefix ], add);
}
}
output = s.join("&").replace(r20, "+");
return output;
};
/**
* Helper function to create URL parameters of deep object
*
* Boiled down from jQuery
*/
function buildParams(prefix, obj, add) {
var name, i, l, rbracket;
rbracket = /\[\]$/;
if (obj instanceof Array) {
for (i = 0, l = obj.length; i < l; i++) {
if (rbracket.test(prefix)) {
add(prefix, obj[i]);
} else {
buildParams(prefix + "[" + ( typeof obj[i] === "object" ? i : "" ) + "]", obj[i], add);
}
}
} else if (typeof obj == "object") {
// Serialize object item.
for (name in obj) {
buildParams(prefix + "[" + name + "]", obj[ name ], add);
}
} else {
// Serialize scalar item.
add(prefix, obj);
}
}
init();
}());