| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389 |
- /* globals related_posts_js_options */
- /**
- * Load related posts
- */
- ( function () {
- 'use strict';
- var jprp = {
- response: null,
- /**
- * Utility get related posts JSON endpoint from URLs
- *
- * @param {string} URL (optional)
- * @return {string} Endpoint URL
- */
- getEndpointURL: function ( URL ) {
- var locationObject,
- is_customizer =
- 'undefined' !== typeof wp &&
- wp.customize &&
- wp.customize.settings &&
- wp.customize.settings.url &&
- wp.customize.settings.url.self;
- // If we're in Customizer, write the correct URL.
- if ( is_customizer ) {
- locationObject = document.createElement( 'a' );
- locationObject.href = wp.customize.settings.url.self;
- } else {
- locationObject = document.location;
- }
- if ( 'string' === typeof URL && URL.match( /^https?:\/\// ) ) {
- locationObject = document.createElement( 'a' );
- locationObject.href = URL;
- }
- var args = 'relatedposts=1';
- var relatedPosts = document.querySelector( '#jp-relatedposts' );
- if ( ! relatedPosts ) {
- return false;
- }
- if ( relatedPosts.hasAttribute( 'data-exclude' ) ) {
- args += '&relatedposts_exclude=' + relatedPosts.getAttribute( 'data-exclude' );
- }
- if ( is_customizer ) {
- args += '&jetpackrpcustomize=1';
- }
- var pathname = locationObject.pathname;
- if ( '/' !== pathname[ 0 ] ) {
- pathname = '/' + pathname;
- }
- if ( '' === locationObject.search ) {
- return pathname + '?' + args;
- } else {
- return pathname + locationObject.search + '&' + args;
- }
- },
- getAnchor: function ( post, classNames ) {
- var anchorTitle = post.title;
- var anchor = document.createElement( 'a' );
- anchor.setAttribute( 'class', classNames );
- anchor.setAttribute( 'href', post.url );
- anchor.setAttribute( 'title', anchorTitle );
- anchor.setAttribute( 'data-origin', post.url_meta.origin );
- anchor.setAttribute( 'data-position', post.url_meta.position );
- if ( '' !== post.rel ) {
- anchor.setAttribute( 'rel', post.rel );
- }
- var div = document.createElement( 'div' );
- div.appendChild( anchor );
- var anchorHTML = div.innerHTML;
- return [ anchorHTML.substring( 0, anchorHTML.length - 4 ), '</a>' ];
- },
- generateMinimalHtml: function ( posts, options ) {
- var self = this;
- var html = '';
- posts.forEach( function ( post, index ) {
- var anchor = self.getAnchor( post, 'jp-relatedposts-post-a' );
- var classes = 'jp-relatedposts-post jp-relatedposts-post' + index;
- if ( post.classes.length > 0 ) {
- classes += ' ' + post.classes.join( ' ' );
- }
- html +=
- '<p class="' +
- classes +
- '" data-post-id="' +
- post.id +
- '" data-post-format="' +
- post.format +
- '">';
- html +=
- '<span class="jp-relatedposts-post-title">' +
- anchor[ 0 ] +
- post.title +
- anchor[ 1 ] +
- '</span>';
- if ( options.showDate ) {
- html +=
- '<time class="jp-relatedposts-post-date" datetime="' +
- post.date +
- '">' +
- post.date +
- '</time>';
- }
- if ( options.showContext ) {
- html += '<span class="jp-relatedposts-post-context">' + post.context + '</span>';
- }
- html += '</p>';
- } );
- return (
- '<div class="jp-relatedposts-items jp-relatedposts-items-minimal jp-relatedposts-' +
- options.layout +
- ' ">' +
- html +
- '</div>'
- );
- },
- generateVisualHtml: function ( posts, options ) {
- var self = this;
- var html = '';
- posts.forEach( function ( post, index ) {
- var anchor = self.getAnchor( post, 'jp-relatedposts-post-a' );
- var classes = 'jp-relatedposts-post jp-relatedposts-post' + index;
- if ( post.classes.length > 0 ) {
- classes += ' ' + post.classes.join( ' ' );
- }
- if ( ! post.img.src ) {
- classes += ' jp-relatedposts-post-nothumbs';
- } else {
- classes += ' jp-relatedposts-post-thumbs';
- }
- var dummyContainer = document.createElement( 'p' );
- dummyContainer.innerHTML = post.excerpt;
- var excerpt = dummyContainer.textContent;
- html +=
- '<div class="' +
- classes +
- '" data-post-id="' +
- post.id +
- '" data-post-format="' +
- post.format +
- '">';
- if ( post.img.src ) {
- html +=
- anchor[ 0 ] +
- '<img class="jp-relatedposts-post-img" loading="lazy" src="' +
- post.img.src +
- '" width="' +
- post.img.width +
- '" height="' +
- post.img.height +
- '" alt="' +
- post.img.alt_text +
- '" />' +
- anchor[ 1 ];
- } else {
- var anchor_overlay = self.getAnchor(
- post,
- 'jp-relatedposts-post-a jp-relatedposts-post-aoverlay'
- );
- html += anchor_overlay[ 0 ] + anchor_overlay[ 1 ];
- }
- html +=
- '<' +
- related_posts_js_options.post_heading +
- ' class="jp-relatedposts-post-title">' +
- anchor[ 0 ] +
- post.title +
- anchor[ 1 ] +
- '</' +
- related_posts_js_options.post_heading +
- '>';
- html += '<p class="jp-relatedposts-post-excerpt">' + excerpt + '</p>';
- if ( options.showDate ) {
- html +=
- '<time class="jp-relatedposts-post-date" datetime="' +
- post.date +
- '">' +
- post.date +
- '</time>';
- }
- if ( options.showContext ) {
- html += '<p class="jp-relatedposts-post-context">' + post.context + '</p>';
- }
- html += '</div>';
- } );
- return (
- '<div class="jp-relatedposts-items jp-relatedposts-items-visual jp-relatedposts-' +
- options.layout +
- ' ">' +
- html +
- '</div>'
- );
- },
- /**
- * We want to set a max height on the excerpt however we want to set
- * this according to the natual pacing of the page as we never want to
- * cut off a line of text in the middle so we need to do some detective
- * work.
- */
- setVisualExcerptHeights: function () {
- var elements = document.querySelectorAll(
- '#jp-relatedposts .jp-relatedposts-post-nothumbs .jp-relatedposts-post-excerpt'
- );
- if ( ! elements.length ) {
- return;
- }
- var firstElementStyles = getComputedStyle( elements[ 0 ] );
- var fontSize = parseInt( firstElementStyles.fontSize, 10 );
- var lineHeight = parseInt( firstElementStyles.lineHeight, 10 );
- // Show 5 lines of text
- for ( var i = 0; i < elements.length; i++ ) {
- elements[ i ].style.maxHeight = ( 5 * lineHeight ) / fontSize + 'em';
- }
- },
- getTrackedUrl: function ( anchor ) {
- var args = 'relatedposts_hit=1';
- args += '&relatedposts_origin=' + anchor.getAttribute( 'data-origin' );
- args += '&relatedposts_position=' + anchor.getAttribute( 'data-position' );
- var pathname = anchor.pathname;
- if ( '/' !== pathname[ 0 ] ) {
- pathname = '/' + pathname;
- }
- if ( '' === anchor.search ) {
- return pathname + '?' + args;
- } else {
- return pathname + anchor.search + '&' + args;
- }
- },
- cleanupTrackedUrl: function () {
- if ( 'function' !== typeof history.replaceState ) {
- return;
- }
- var cleaned_search = document.location.search.replace(
- /\brelatedposts_[a-z]+=[0-9]*&?\b/gi,
- ''
- );
- if ( '?' === cleaned_search ) {
- cleaned_search = '';
- }
- if ( document.location.search !== cleaned_search ) {
- history.replaceState( {}, document.title, document.location.pathname + cleaned_search );
- }
- },
- };
- function afterPostsHaveLoaded() {
- jprp.setVisualExcerptHeights();
- var posts = document.querySelectorAll( '#jp-relatedposts a.jp-relatedposts-post-a' );
- Array.prototype.forEach.call( posts, function ( post ) {
- document.addEventListener( 'click', function () {
- post.href = jprp.getTrackedUrl( post );
- } );
- } );
- }
- /**
- * Initialize Related Posts.
- */
- function startRelatedPosts() {
- jprp.cleanupTrackedUrl();
- var endpointURL = jprp.getEndpointURL();
- var relatedPosts = document.querySelector( '#jp-relatedposts' );
- if ( ! endpointURL ) {
- return;
- }
- if ( document.querySelectorAll( '#jp-relatedposts .jp-relatedposts-post' ).length ) {
- afterPostsHaveLoaded();
- return;
- }
- var request = new XMLHttpRequest();
- request.open( 'GET', endpointURL, true );
- request.setRequestHeader( 'x-requested-with', 'XMLHttpRequest' );
- request.onreadystatechange = function () {
- if ( this.readyState === XMLHttpRequest.DONE && this.status === 200 ) {
- try {
- var response = JSON.parse( request.responseText );
- if ( 0 === response.items.length || 0 === relatedPosts.length ) {
- return;
- }
- jprp.response = response;
- var html,
- showThumbnails,
- options = {};
- if ( 'undefined' !== typeof wp && wp.customize ) {
- showThumbnails = wp.customize.instance( 'jetpack_relatedposts[show_thumbnails]' ).get();
- options.showDate = wp.customize.instance( 'jetpack_relatedposts[show_date]' ).get();
- options.showContext = wp.customize
- .instance( 'jetpack_relatedposts[show_context]' )
- .get();
- options.layout = wp.customize.instance( 'jetpack_relatedposts[layout]' ).get();
- } else {
- showThumbnails = response.show_thumbnails;
- options.showDate = response.show_date;
- options.showContext = response.show_context;
- options.layout = response.layout;
- }
- html = ! showThumbnails
- ? jprp.generateMinimalHtml( response.items, options )
- : jprp.generateVisualHtml( response.items, options );
- var div = document.createElement( 'div' );
- relatedPosts.appendChild( div );
- div.outerHTML = html;
- if ( options.showDate ) {
- var dates = relatedPosts.querySelectorAll( '.jp-relatedposts-post-date' );
- Array.prototype.forEach.call( dates, function ( date ) {
- date.style.display = 'block';
- } );
- }
- relatedPosts.style.display = 'block';
- afterPostsHaveLoaded();
- } catch ( error ) {
- // Do nothing
- }
- }
- };
- request.send();
- }
- function init() {
- if ( 'undefined' !== typeof wp && wp.customize ) {
- if ( wp.customize.selectiveRefresh ) {
- wp.customize.selectiveRefresh.bind( 'partial-content-rendered', function ( placement ) {
- if ( 'jetpack_relatedposts' === placement.partial.id ) {
- startRelatedPosts();
- }
- } );
- }
- wp.customize.bind( 'preview-ready', startRelatedPosts );
- } else {
- startRelatedPosts();
- }
- }
- if ( document.readyState !== 'loading' ) {
- init();
- } else {
- document.addEventListener( 'DOMContentLoaded', init );
- }
- } )();
|