What our clients say…

 

IMBED

"use strict"; if (!Element.prototype.matches) { Element.prototype.matches = Element.prototype.msMatchesSelector || Element.prototype.webkitMatchesSelector; } (function () { let script = document.currentScript; let container; if (typeof script === 'undefined') { let scriptTags = document.getElementsByTagName('script'); let scriptFound = false; for (let i = scriptTags.length - 1; i >= 0 && !scriptFound; i--) { if (scriptTags[i].src.indexOf('widgetAdv.min.js') >= 0) { script = scriptTags[i]; scriptFound = true; } } } container = script.parentElement; let config = { layout: '', cssPrefix: 'revwid-', mobileBreakpoint: 1024, carouselSwipeBuffer: 75, readMoreLimit: 455, cssPath: getCssPath(), }; let stylesheet; let baseUrl = container.dataset.url; let businessId = parseInt(container.dataset.bid); let businessIds = container.dataset.bid; // multi location widget let totalPages = 0; let socialLogos = {}; let widget = { cssLoaded: false, unique: config.cssPrefix + Math.random().toString(36).substr(2, 9), additionalId: container.hasAttribute('data-aid') ? container.getAttribute('data-aid') : false, preview: container.hasAttribute('data-preview'), masonryTimeout: null, jsonLd: container.dataset.hasOwnProperty('jsonld') ? container.dataset.jsonld : 1 }; container.setAttribute('data-unique', widget.unique); function getCssPath() { return container.hasAttribute('data-css') ? container.getAttribute('data-css') : 'https://widget.reviewability.com/css/widgetAdv.min.css'; } function initWidget(divContainer, json) { if (!widget.cssLoaded) { setTimeout(function () { initWidget(divContainer, json); }, 100); return; } if (json.content.search('data-widget-layout="dataOnly"') !== -1) { stylesheet.parentNode.removeChild(stylesheet); } divContainer.innerHTML += json.content; // Set the container for this widget to our container property widget.container = getContainerElement(); // Store the layout type for this widget (retrieved via data value on the container). // Some functionality is unique to the layout type. Valid options: full, vertical, horizontal config.layout = widget.container.dataset.widgetLayout; if (config.layout === 'horizontal') { config.readMoreLimit = 200; } widget.reviewContainerWidth = 0; widget.touchStartX = 0; widget.touchStartY = 0; widget.touchEndX = 0; widget.touchEndY = 0; widget.slideToLast = false; widget.reviews = ""; widget.ui = { reviewContainer: "", filterButton: "", filterSelects: "", boxes: "", painationArrowNext: "", painationArrowPrevious: "", }; /** * Settings for the carousel. Only the 'horizontal' layout uses the carousel. */ widget.carousel = { reviewsPerSlide: 3, currentIndex: 0, previousIndex: 0, ui: { arrowNext: "", arrowPrevious: "", scrollableArea: "", } }; /** * Initialize the core widget by setting values and binding listeners */ widget.init = function () { // Attach event listers to the widget's UI elements attachListeners(); // Set the reviews widget.reviews = widget.container.querySelectorAll('.js-review'); // In order for our flexbox masonry effect to properly wrap two columns on 'full' layout we need to give the container an // explicit height. This is called multiple times, so scoping it to the 'full' layout only occurs in the function. widget.setReviewContainerProperties(); if (config.layout == 'horizontal') { widget.initCarousel(); // Set the box height explicilty on 'horizontal' view so that all the boxes are the same height widget.setBoxHeight(); } }; /** * Initialize the carousel. This should only be called if the widget uses the carousel (currently only "horizontal" layout does) */ widget.initCarousel = function () { widget.setReviewContainerWidthValue(); setReviewsPerSlide(); widget.carousel.ui.arrowNext = widget.container.querySelector('.js-carousel-next-arrow'); widget.carousel.ui.arrowPrevious = widget.container.querySelector('.js-carousel-previous-arrow'); widget.carousel.ui.scrollableArea = widget.container.querySelector('.js-scrollable-area'); widget.carousel.ui.arrowNext.addEventListener('click', function (e) { widget.carousel.goToNextSlide(); }); widget.carousel.ui.arrowPrevious.addEventListener('click', function (e) { widget.carousel.goToPreviousSlide(); }); widget.carousel.ui.scrollableArea.addEventListener('touchstart', function (event) { widget.touchStartX = event.changedTouches[0].screenX; widget.touchStartY = event.changedTouches[0].screenY; }, false); widget.carousel.ui.scrollableArea.addEventListener('touchend', function (event) { widget.touchEndX = event.changedTouches[0].screenX; widget.touchEndY = event.changedTouches[0].screenY; handleSwipe(); }, false); }; /** * Get the containing element for this review widget. We'll assume the script tag is nested within the container. */ function getContainerElement() { if (typeof container !== 'undefined') { return container.querySelector('.js-widget-container'); } else { let scriptTags = document.getElementsByTagName('script'); let containerTag = scriptTags[scriptTags.length - 1].parentNode; return containerTag.querySelector('.js-widget-container'); } } /** * Attach listeners to widget DOM elements and window events */ function attachListeners() { // Bind main UI elements to the elements within this review widget's context widget.ui.reviewContainer = widget.container.querySelector('.js-review-container'); widget.ui.boxes = widget.container.querySelectorAll('.js-box'); // only used if we are forcing review boxes to be same height. Per update it's not needed at this time. widget.ui.filterContainer = widget.container.querySelector('.js-filter-container'); widget.ui.filterButton = widget.container.querySelector('.js-filter-button'); widget.ui.painationArrowNext = widget.container.querySelector('.js-next-arrow'); widget.ui.painationArrowPrevious = widget.container.querySelector('.js-previous-arrow'); // Adjust widget element widths & heights as needed when the window is resized window.addEventListener('resize', function (event) { if (config.layout === 'full') { widget.setReviewContainerProperties(); } if (config.layout === 'horizontal') { widget.setBoxHeight(); widget.setReviewContainerWidthValue(); setReviewsPerSlide(); } }); document.addEventListener('click', function (event) { if (isShowingFilterMenu() && !widget.ui.filterContainer.contains(event.target)) { widget.ui.filterContainer.classList.remove(config.cssPrefix + 'show-invisibles'); widget.ui.filterButton.classList.remove(config.cssPrefix + 'is-active'); } }); // This is a bit of a hack. By listening to touch events we can ensure the "Filter" link closes on mobile when a user clicks away from it widget.container.addEventListener('touchstart', function (event) { }, false); widget.ui.filterButton.addEventListener('click', function (event) { widget.ui.filterContainer.classList.toggle(config.cssPrefix + 'show-invisibles'); widget.ui.filterButton.classList.toggle(config.cssPrefix + 'is-active'); }); } /** * This is used for carousel only - we need to know the width of a slide so that we can transition as a user navigates the carousel */ widget.setReviewContainerWidthValue = function () { widget.reviewContainerWidth = widget.ui.reviewContainer.clientWidth; }; /** * This is used on 'full' layouts only. We need to set an explicit height for the container so that our flexbox will wrap */ widget.setReviewContainerProperties = function () { // Only set the container height on non-mobile views. // For mobile we will just list everything as one column and can have an auto height if (config.layout === 'full') { doMasonry(); if (widget.masonryTimeout) { window.clearTimeout(widget.masonryTimeout); } widget.masonryTimeout = setTimeout(doMasonry, 50); } // Add an 'is-small-container' class to the widget if the viewport isn't mobile, but it is within a smaller container if (isSmallContainer() && (config.layout == 'full' || config.layout == 'horizontal')) { widget.container.classList.add(config.cssPrefix + 'is-small-container'); } else { widget.container.classList.remove(config.cssPrefix + 'is-small-container'); } }; /** * Not currently used. This would ensure each review box is the same height */ widget.setBoxHeight = function () { /* if (config.layout == 'horizontal') { setBoxHeightValues(getMaxBoxHeightValue()); } */ return false; }; /** * Widget Carousel: Go to a specific slide */ widget.carousel.goToIndex = function (index) { if (index == widget.carousel.previousIndex) { return; } widget.carousel.previousIndex = index; var offset = -(index * (widget.reviewContainerWidth - (isMobile() ? 32 : 0))); widget.ui.reviewContainer.style.setProperty('transform', 'translateX(' + offset + 'px)', 'important'); }; /** * Widget Carousel: Return whether the user is viewing the last slide */ widget.carousel.isOnLastSlide = function () { return (widget.carousel.currentIndex + 1) >= Math.ceil(widget.reviews.length / widget.carousel.reviewsPerSlide); }; /** * Widget Carousel: Return whether the user is viewing the first slide */ widget.carousel.isOnFirstSlide = function () { return widget.carousel.currentIndex === 0; }; widget.getPageNumber = function () { return parseInt(widget.container.querySelector('.revwid-pagination-link.revwid-is-active').dataset.page); }; widget.isOnFirstPage = function () { return widget.getPageNumber() === 1; }; widget.isOnLastPage = function () { return widget.getPageNumber() === totalPages; }; /** * Widget Carousel: Advance the user to the next slide if able */ widget.carousel.goToNextSlide = function () { widget.slideToLast = false; if (!widget.carousel.isOnLastSlide()) { widget.carousel.currentIndex++; widget.carousel.goToIndex(widget.carousel.currentIndex); } else { widget.container.querySelectorAll(".revwid-pagination-list .revwid-is-right-arrow")[0].click(); } setCarouselDisabledStatus(); }; /** * Widget Carousel: Move the user to the previous slide if able */ widget.carousel.goToPreviousSlide = function () { if (!widget.carousel.isOnFirstSlide()) { widget.slideToLast = false; widget.carousel.currentIndex--; widget.carousel.goToIndex(widget.carousel.currentIndex); } else { widget.slideToLast = true; widget.container.querySelectorAll(".revwid-pagination-list .revwid-is-left-arrow")[0].click(); } setCarouselDisabledStatus(); }; /** * Widget Carousel: Disable / Enable carousel previous & next arrows links as necessary */ function setCarouselDisabledStatus() { if (widget.carousel.isOnLastSlide() && widget.isOnLastPage()) { widget.carousel.ui.arrowNext.classList.add(config.cssPrefix + 'is-disabled'); } else { widget.carousel.ui.arrowNext.classList.remove(config.cssPrefix + 'is-disabled'); } if (widget.carousel.isOnFirstSlide() && widget.isOnFirstPage()) { widget.carousel.ui.arrowPrevious.classList.add(config.cssPrefix + 'is-disabled'); } else { widget.carousel.ui.arrowPrevious.classList.remove(config.cssPrefix + 'is-disabled'); } } /** * Widget Carousel: Allow swiping left / right to traverse the carousel on mobile views */ function handleSwipe() { if ( (Math.abs(widget.touchEndY - widget.touchStartY) < 20 || widget.touchStartY == 0) && Math.abs(widget.touchEndX - widget.touchStartX) > 10 ) { if (widget.touchEndX + config.carouselSwipeBuffer < widget.touchStartX) { widget.carousel.goToNextSlide(); } if (widget.touchEndX - config.carouselSwipeBuffer > widget.touchStartX) { widget.carousel.goToPreviousSlide(); } } } function setReviewsPerSlide() { widget.carousel.reviewsPerSlide = isMobile() ? 1 : 3; } function isMobile() { return window.innerWidth < config.mobileBreakpoint; } function isSmallContainer() { return !isMobile() && widget.container.clientWidth < config.mobileBreakpoint; } function isShowingFilterMenu() { return widget.ui.filterContainer.classList.contains(config.cssPrefix + 'show-invisibles'); } function doMasonry() { let reviewsCont = widget.container.querySelector('.js-review-container'); // Set the container's height on non-mobile if (!isMobile()) { // set height to big number because if container is smaller than review then review clientHeight will be // equal to container height, not real value reviewsCont.style.height = 1000; let els1 = widget.container.querySelectorAll('.js-is-order-1'); let els2 = widget.container.querySelectorAll('.js-is-order-2'); let column1Height = 0, column2Height = 0; if (els1.length > 0) { if (els1.length === 1) { column1Height = els1[0].clientHeight; } else { forEach(els1, function (a) { column1Height += a.clientHeight; }); } } else { column1Height = widget.container.querySelector('.revwid-reviews .revwid-has-text-centered').clientHeight; } if (els2.length > 0) { if (els2.length === 1) { column2Height = els2[0].clientHeight; } else { forEach(els2, function (a) { column2Height += a.clientHeight; }); } } reviewsCont.style.height = ((column1Height > column2Height ? column1Height : column2Height) + 10) + 'px'; } else { reviewsCont.style.height = 'auto'; } widget.masonryTimeout = null; } renderReviews(json.reviews.reviews); renderPagination(json.reviews.pages, 1); widget.init(); } function ajax(data) { let $ajax = new XMLHttpRequest(); let url = data.url; let onSuccess = typeof data.success === 'function' ? data.success : function (t) { }; let onError = typeof data.error === 'function' ? data.error : function (t) { }; let type = typeof data.type === 'string' ? data.type : "GET"; $ajax.onreadystatechange = function () { if (this.readyState === 4 && this.status === 200) { onSuccess(this.responseText); } else if (this.readyState === 4 && this.status >= 300) { onError(this.responseText); } }; $ajax.open(type, url, true); $ajax.setRequestHeader('X-Requested-With', 'XMLHttpRequest'); $ajax.send(); } function getRndInteger(min, max) { return Math.floor(Math.random() * (max - min)) + min; } function encode(string) { try { string = encodeURI(string.trim()); } catch (e) { string = escape(string.trim()); } return string; } function removeEmojis(str) { return str.replace(/[\u2700-\u27bf]|(?:\ud83c[\udde6-\uddff]){2}|[\ud800-\udbff][\udc00-\udfff]|[\u0023-\u0039]\ufe0f?\u20e3|\u3299|\u3297|\u303d|\u3030|\u24c2|\ud83c[\udd70-\udd71]|\ud83c[\udd7e-\udd7f]|\ud83c\udd8e|\ud83c[\udd91-\udd9a]|\ud83c[\udde6-\uddff]|[\ud83c[\ude01-\ude02]|\ud83c\ude1a|\ud83c\ude2f|[\ud83c[\ude32-\ude3a]|[\ud83c[\ude50-\ude51]|\u203c|\u2049|[\u25aa-\u25ab]|\u25b6|\u25c0|[\u25fb-\u25fe]|\u00a9|\u00ae|\u2122|\u2139|\ud83c\udc04|[\u2600-\u26FF]|\u2b05|\u2b06|\u2b07|\u2b1b|\u2b1c|\u2b50|\u2b55|\u231a|\u231b|\u2328|\u23cf|[\u23e9-\u23f3]|[\u23f8-\u23fa]|\ud83c\udccf|\u2934|\u2935|[\u2190-\u21ff]?/g, ''); } function getShareableUrls(review) { let cleanContent = removeEmojis(review.review); let desc = (cleanContent.length >= 240) ? cleanContent.slice(0, 240) + '...' : cleanContent; let url = baseUrl + '/share/feedback/' + review.id + '/' + getRndInteger(100000, 999999); let caption = ''; if (review.type === 'facebookRecommend') { url += '/2'; } else if (review.type !== 'gfs') { url += '/1'; } else { url += '/0'; } for (let $i = 1; $i <= review.rating; $i++) { caption += '★'; } caption += ' ' + review.author; url = encode(url); desc = encode(desc); caption = encode(caption); return { facebook: 'https://www.facebook.com/sharer.php?display=popup&caption=' + caption + '&u=' + url, twitter: 'https://twitter.com/share?url=' + url + '&text=' + desc + '&hashtags=Review', linkedin: 'https://www.linkedin.com/shareArticle?mini=true&title=' + caption + '&summary=' + desc + '&source=' + baseUrl + '&url=' + url, buffer: 'https://buffer.com/add?text=' + desc + '&url=' + url, } } function renderReview(order, review) { let template = container.querySelector('.revwid-review-template').innerHTML; let charLimit = config.readMoreLimit; let responsesHtml = ''; let responseClass = config.layout === 'horizontal' ? 'revwid-review-full-text' : ''; let text = container.querySelector(".revwid-text"); forEach(review.responses, function (r) { responsesHtml += '
' + '' + r.responder + ' - ' + r.date + '

' + r.content + '

'; }); let socialLogo = review.type in socialLogos ? socialLogos[review.type] : false; let starsHtml = '', googleAttributes = ''; if (review.type === 'facebookRecommend') { template = template.replace(/
.*<\/div>/, ""); socialLogo = 'facebook' in socialLogos ? socialLogos['facebook'] : false; starsHtml = '
' + '' + (review.rating > 0 ? text.getAttribute("data-translation-yes") : text.getAttribute("data-translation-no")) + ' ' + text.getAttribute("data-translation-recommends") + '
' } else { for (let $i = 1; $i <= review.rating; $i++) { starsHtml += ' '; } if (Math.round(review.rating) !== review.rating) { starsHtml += ' '; } for (let $i = review.rating + 1; $i <= 5; $i++) { starsHtml += ' '; } } if (!!review.attributes && !!review.attributesPositive) { googleAttributes = '

' + (review.attributesPositive === "1" ? text.getAttribute("data-translation-positive") : text.getAttribute("data-translation-critical")) + ': ' + review.attributes.split(',').map(function(i){ let trans = text.getAttribute("data-translation-"+i.toLowerCase()); return !!trans ? trans : i.charAt(0).toUpperCase() + i.slice(1);; }).join(', ') + '

'; } let socialHtml = socialLogo ? '' : (review.type === 'gfs' ? '' : ''); let shareUrl = getShareableUrls(review); let params = { order: order, id: review.id, stars: starsHtml, rating: review.rating, social_logo: socialHtml, author: review.author, date: review.reviewReceived, review: review.review, responses: responsesHtml, remove_read_mode: "remove-element", share_fb: shareUrl.facebook, share_tt: shareUrl.twitter, share_lk: shareUrl.linkedin, share_bf: shareUrl.buffer, google_attributes: googleAttributes }; let isLongReviewText = params.review.length > charLimit; if (isLongReviewText || (config.layout === 'horizontal' && review.responses.length > 0)) { params.review_full = params.review; params.remove_read_mode = ""; if (isLongReviewText) { params.review = params.review.substr(0, charLimit - 50) + '...'; } } for (let key in params) { if (params.hasOwnProperty(key)) { template = template.split('{*' + key + '*}').join(params[key]); } } return template; } function renderPagination(total, current) { current = parseInt(current); total = parseInt(total); let holder = container.querySelector('.revwid-pagination-list'); let lis = [], available = []; lis.push('
  • '); let limit = 4; let limitHalf = parseInt(limit / 2); if (total <= 7) { for (let $i = 1; $i <= total; $i++) { available.push($i); } } else { if (current < limit) { for (let $i = 1; $i <= limit; $i++) { available.push($i); } available.push('...'); available.push(total); } else if (current > total - limit) { available.push(1); available.push('...'); for (let $i = total - limit; $i <= total; $i++) { available.push($i); } } else { available.push(1); available.push('...'); for (let $i = current - limitHalf; $i <= current + limitHalf; $i++) { available.push($i); } available.push('...'); available.push(total); } } forEach(available, function (e) { if (e === '...') { lis.push('
  • ...
  • '); } else { lis.push('
  • ' + e + '
  • '); } }); lis.push('
  • '); holder.innerHTML = lis.join(''); } function getNoResultsHtml() { let text = container.querySelector(".revwid-text"); let html = "
    "; html += text.getAttribute("data-translation-message-no-testimonials") + "

    "; html += "" + text.getAttribute("data-translation-message-no-firstReview") + "
    "; return html; } function renderReviews(data) { let $order = 1; let pieces = []; forEach(data, function (r) { $order++; pieces.push(renderReview(($order % 2) + 1, r)); }); let reviewsCont = container.querySelector('.revwid-reviews'); reviewsCont.innerHTML = pieces.length === 0 ? getNoResultsHtml() : pieces.join(""); forEach(reviewsCont.querySelectorAll('.remove-element'), function (e) { e.parentNode.removeChild(e); }); if (widget.container) { widget.reviews = widget.container.querySelectorAll('.js-review'); } if (config.layout === 'horizontal') { if (widget.slideToLast) { widget.slideToLast = false; widget.carousel.currentIndex = (widget.reviews.length / widget.carousel.reviewsPerSlide) - 1; } else { widget.carousel.currentIndex = 0; } widget.carousel.goToIndex(widget.carousel.currentIndex); } // Review source logos - after an image asset is loaded let's recalculate the review container height to ensure our grid displays properly let logos = reviewsCont.querySelectorAll('img'); forEach(Array.prototype.slice.call(logos), function (element) { if (config.layout === 'full') { element.addEventListener('load', function () { widget.setReviewContainerProperties(); }); } if (config.layout === 'horizontal') { equalizeReviewHeaderHeights(); } if (element.complete && config.layout === 'full') { widget.setReviewContainerProperties(); } }); } function equalizeReviewHeaderHeights() { if (widget.carousel.reviewsPerSlide > 1) { const chunks = []; const boxesArray = Array.prototype.slice.call(widget.ui.boxes); while (boxesArray.length) { chunks.push(boxesArray.splice(0, widget.carousel.reviewsPerSlide)); } for (let i = 0; i < chunks.length; i++) { const chunk = chunks[i]; const headers = []; let maxHeight = -1; for (let j = 0; j < chunk.length; j++) { const header = chunk[j].querySelector('.' + config.cssPrefix + 'review-header'); maxHeight = Math.max(maxHeight, header.clientHeight); headers.push(header); } for (let j = 0; j < headers.length; j++) { headers[j].style.height = maxHeight; } } } } function appendStylesheet() { stylesheet = document.createElement('link'); stylesheet.setAttribute('rel', 'stylesheet'); stylesheet.setAttribute('type', 'text/css'); stylesheet.onload = function () { widget.cssLoaded = true; }; stylesheet.setAttribute('href', config.cssPath); document.getElementsByTagName('head')[0].appendChild(stylesheet); } // IE 11 fix for forEach function forEach(elements, callable) { for (let i = 0; i < elements.length; i++) { callable(elements[i]); } } function live(eventType, elementQuerySelector, cb) { document.addEventListener(eventType, function (event) { let $selector = '[data-unique="' + widget.unique + '"] ' + elementQuerySelector; let qs = document.querySelectorAll($selector); if (qs) { let el = event.target, index = -1; while (el && ((index = Array.prototype.indexOf.call(qs, el)) === -1)) { el = el.parentElement; } if (index > -1) { cb.call(el, event); } } }); } function apllyMainParametersToUrl(parameterBag) { if (widget.additionalId) { parameterBag.push('aid=' + widget.additionalId); } if (widget.preview) { parameterBag.push('preview=' + widget.preview); } return parameterBag; } function getMainUrl() { let widgetMainUrlParts = apllyMainParametersToUrl([]); return baseUrl + '/widget/b-' + businessIds + '?' + widgetMainUrlParts.join('&'); } var getNextSibling = function (elem, selector) { // Get the next sibling element var sibling = elem.nextElementSibling; var attempt = 0; // If there's no selector, return the first sibling if (!selector) { return sibling; } // If the sibling matches our selector, use it // If not, jump to the next sibling and continue the loop while (sibling && attempt < 10) { if (sibling.matches(selector)) { return sibling; } sibling = sibling.nextElementSibling; attempt++; } }; function loadAjaxPage(page, getPages) { let orderParts = widget.container.querySelector('[name=revwid-selected-order]').value.split('-'), socials = widget.container.querySelector('[name=revwid-selected-social]'); socials = socials ? socials.value : null; let urlParts = []; urlParts.push('order=' + orderParts[0]); urlParts.push('dir=' + orderParts[1]); if (socials !== 'none' && socials !== null && socials !== '') { urlParts.push('socials=' + socials); } if (getPages) { urlParts.push('getPages=' + 1); } urlParts = apllyMainParametersToUrl(urlParts); ajax({ url: baseUrl + '/widget/reviews/b-' + businessIds + '/p-' + page + '?' + urlParts.join('&'), success: function (data) { let json = JSON.parse(data); if (getPages) { totalPages = json.reviews.pages; } renderReviews(json.reviews.reviews); renderPagination(totalPages, page); widget.setReviewContainerProperties(); } }); } appendStylesheet(); live('click', '.revwid-share-buttons a', function (e) { e.preventDefault(); if (this.classList.contains('js-share-icon-close')) { return; } let intWidth = '600', intHeight = '400'; let dualScreenLeft = window.screenLeft ? window.screenLeft : screen.left; let dualScreenTop = window.screenTop ? window.screenTop : screen.top; let width = window.innerWidth ? window.innerWidth : document.documentElement.clientWidth ? document.documentElement.clientWidth : screen.width; let height = window.innerHeight ? window.innerHeight : document.documentElement.clientHeight ? document.documentElement.clientHeight : screen.height; let left = ((width / 2) - (intWidth / 2)) + dualScreenLeft; let top = ((height / 2) - (intHeight / 2)) + dualScreenTop; let strTitle = ((typeof this.getAttribute('title') !== 'undefined') ? this.getAttribute('title') : 'Social Share'), strParam = 'width=' + intWidth + ',height=' + intHeight + ',resizable=yes, top=' + top + ', left=' + left; window.open(this.getAttribute('href'), strTitle, strParam).focus(); }); live('click', '.revwid-pagination-link[data-page]', function (e) { if (this.classList.contains('revwid-is-active')) { return; } if (config.layout !== 'horizontal') { document.querySelector('div[data-unique="' + widget.unique + '"]').scrollIntoView(true); } loadAjaxPage(this.dataset.page, false); }); live('change', '[name="revwid-selected-social"], [name="revwid-selected-order"]', function (e) { loadAjaxPage(1, true); }); live('click', '.revwid-pagination .revwid-pagination-arrow', function (e) { let page = parseInt(widget.container.querySelector('.revwid-pagination-link.revwid-is-active').dataset.page); if (this.classList.contains('revwid-is-disabled')) { return; } if (this.classList.contains('revwid-is-left-arrow')) { page--; } if (this.classList.contains('revwid-is-right-arrow')) { page++; } if (config.layout !== 'horizontal') { document.querySelector('div[data-unique="' + widget.unique + '"]').scrollIntoView(true); } loadAjaxPage(page, false); }); live('click', '[data-share-tools-for]', function (e) { this.classList.toggle(config.cssPrefix + 'is-active'); e.preventDefault(); }); live('click', '.js-read-more', function (e) { let id = this.getAttribute('data-review-id'); let selector = '.revwid-review[data-review-id="' + id + '"]'; widget.container.querySelector(selector).classList.toggle(config.cssPrefix + 'is-expanded'); widget.setReviewContainerProperties(); e.preventDefault(); }); function createJsonLdScriptTag(json) { let script = document.createElement("script"); script.className = "json-ld-content"; script.type = "application/ld+json"; script.innerText = json.jsonLd; container.parentNode.appendChild(script); } ajax({ url: getMainUrl(), success: function (data) { let jsonLd = getNextSibling(container, '.json-ld-content'); let json = JSON.parse(data); if (json.baseUrl) { baseUrl = json.baseUrl; } if (json.legacy) { // legacy widget layouts mode: container.setAttribute('id', 'e2wget5widget'); let script = document.createElement("script"); script.id = 'e2wWidgetScript'; script.async = true; script.type = "text/javascript"; script.src = 'https://widget.reviewability.com/js/widgetAjax.min.js'; script.setAttribute('data-src', baseUrl + '/reviews-mobile.js/' + businessIds + '.0'); script.setAttribute('data-jsonld', widget.jsonLd); if (widget.preview) { script.setAttribute('data-preview', 1); } if (widget.additionalId > 0) { script.setAttribute('data-additionalId', widget.additionalId); } container.innerHTML = ""; container.appendChild(script); } else { socialLogos = json.socials; try { if (widget.jsonLd == 1 && jsonLd.matches('.json-ld-content')) { jsonLd.innerHTML = json.jsonLd; } } catch (e) { createJsonLdScriptTag(json); } initWidget(container, json); totalPages = json.reviews.pages; } try { if (widget.jsonLd == 0 && jsonLd.matches('.json-ld-content')) { jsonLd.parentNode.removeChild(jsonLd); } } catch (e) { // json ld element not found } } }); })();