Current File : /home/pacjaorg/public_html/wp-content/plugins/js_composer/modules/seo/assets/js/utils.js
/* =========================================================
 * Copyright 2023 Wpbakery
 *
 * WPBakery Page Builder util functions for the SEO Analysis panel in the navbar
 *
 * ========================================================= */
/* global vc */
if (!window.vc) {
	window.vc = {};
}

(function ($) {

	window.vc.seo_utils = {
		getTextContent: function (data) {
			data = data
				.replace(/\s*\bdata-vcv-[^"<>]+"[^"<>]+"+/g, '')
				.replace(/<!--\[vcvSourceHtml]/g, '')
				.replace(/\[\/vcvSourceHtml]-->/g, '')
				.replace(/<\//g, ' </');
			var range = document.createRange();
			var documentFragment = range.createContextualFragment(data);

			var helper = documentFragment.querySelector('style, script, noscript, meta, title, #vc_no-content-helper, .vc_controls');

			while (helper) {
				var parentNode = helper.parentNode;
				parentNode.removeChild(helper);
				helper = documentFragment.querySelector('style, script, noscript, meta, title, #vc_no-content-helper, .vc_controls');
			}

			return documentFragment && documentFragment.textContent && documentFragment.textContent.trim();
		},
		/**
		 * Creates a hidden element with the purpose to calculate the sizes of elements and adds these elements to the body.
		 *
		 * @returns {HTMLElement} The created hidden element.
		 */
		createMeasurementElement: function () {
			var hiddenElement = document.createElement('div');
			hiddenElement.id = 'vc-measurement-element';

			// Styles to prevent unintended scrolling in Gutenberg.
			hiddenElement.style.position = 'absolute';
			hiddenElement.style.left = '-9999em';
			hiddenElement.style.top = 0;
			hiddenElement.style.height = 0;
			hiddenElement.style.overflow = 'hidden';
			hiddenElement.style.fontFamily = 'arial, sans-serif';
			hiddenElement.style.fontSize = '20px';
			hiddenElement.style.fontWeight = '400';

			document.body.appendChild(hiddenElement);
			return hiddenElement;
		},
		/**
		 * Measures the width of the text using a hidden element.
		 *
		 * @param {string} text The text to measure the width for.
		 * @returns {number} The width in pixels.
		 */
		measureTextWidth: function (text) {
			var element = document.getElementById('vc-measurement-element');
			if (!element) {
				element = this.createMeasurementElement();
			}
			element.innerHTML = text;
			return element.offsetWidth;
		},
		/**
		 * Finds a keyphrase in a provided text string
		 *
		 * @param {string} text The text check for keyphrase.
		 * @param {string} keyphrase The keyphrase.
		 * @returns {object} The object about found keyphrase.
		 */
		findKeyphrase: function (text, keyphrase) {
			text = text.toLowerCase();
			keyphrase = keyphrase.trim().toLowerCase();
			// Escape special characters in the keyphrase
			var escapedKeyphrase = keyphrase.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');

			// Create a regular expression with word boundaries and case-insensitivity
			var regex = new RegExp('\\b' + escapedKeyphrase + '\\b', 'gi');

			// Use the regular expression to find matches in the text
			var matches = Array.from(text.matchAll(regex));

			if (matches.length) {
				return {
					found: true,
					count: matches.length,
					positions: matches.map(function (match) {
						return match.index;
					}),
				};
			} else {
				return {
					found: false,
					count: 0,
					positions: [],
				};
			}
		},
		/**
		 * Slugify a string
		 *
		 * @param {string} text The text string to slugify.
		 * @returns {tring} Slugified string.
		 */
		slugify: function (text) {
			return text.toString().toLowerCase()
				.replace(/\s+/g, '-') // Replace spaces with -
				.replace(/[^\w\-]+/g, '') // Remove all non-word characters
				.replace(/\-\-+/g, '-') // Replace multiple - with single -
				.replace(/^-+/, '') // Trim - from start of text
				.replace(/-+$/, ''); // Trim - from end of text
		},
		/**
		 * Finds a keyphrase in a provided url slug string
		 *
		 * @param {string} slug The url slug string.
		 * @param {string} keyphrase The keyphrase.
		 * @returns {object} The object about found keyphrase.
		 */
		findKeyphraseInSlug: function (slug, keyphrase) {
			// Slugify the keyphrase and slug (users can use spaces in slug)
			var slugifiedKeyphrase = this.slugify(keyphrase);
			var slugifiedSlug = this.slugify(slug);

			// Escape special characters in the slugified keyphrase and slug
			var escapedKeyphrase = slugifiedKeyphrase.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
			var escapedSlug = slugifiedSlug.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');

			// Create a regular expression with hyphens and word boundaries
			var regex = new RegExp('\\b' + escapedKeyphrase.split('-').join('\\b-\\b') + '\\b', 'gi');

			// Use the regular expression to find matches in the slug
			var matches = Array.from(escapedSlug.matchAll(regex));

			if (matches.length > 0) {
				return {
					found: true,
					count: matches.length,
					positions: matches.map(function (match) {
						return match.index;
					}),
				};
			} else {
				return {
					found: false,
					count: 0,
					positions: [],
				};
			}
		},
		/**
		 * Finds a keyphrase in a provided list of images
		 *
		 * @param {object} $images The jQuery object of all the images in the content.
		 * @param {string} keyphrase The keyphrase.
		 * @returns {object} The object about images that contain keyphrase.
		 */
		findKeyphraseInAltTag: function ($images, keyphrase) {
			var totalImages = $images.length;
			var imagesWithKeyphrase = 0;
			keyphrase = keyphrase.trim().toLowerCase();

			// Loop through each image in the jQuery object
			$images.each(function () {
				// Get the alt attribute of the current image
				var altText = $(this).attr('alt');

				// Check if the alt attribute contains the keyphrase
				if (altText && altText.toLowerCase().includes(keyphrase)) {
					imagesWithKeyphrase++;
				}
			});
			// Calculate the percentage of images with the keyphrase
			var percentage = (imagesWithKeyphrase / totalImages) * 100;

			return {
				percentage: percentage,
				imagesWithKeyphrase: imagesWithKeyphrase
			};
		},
		/**
		 * Finds a keyphrase density in a provided text string
		 *
		 * @param {string} text The text check for keyphrase.
		 * @param {string} keyphrase The keyphrase.
		 * @returns {object} The object about the text that contain keyphrase.
		 */
		findKeyphraseDensity: function (text, keyphrase) {
			keyphrase = keyphrase.trim().toLowerCase();
			var totalWords = text.trim().split(/\s+/).length; // Count words in the text
			var keyphraseRegExp = new RegExp('\\b' + keyphrase + '\\b', 'gi');
			var keyphraseOccurrences = text.match(keyphraseRegExp) || [];

			var advisedMinOccurrences = Math.ceil(0.005 * totalWords);
			var advisedMaxOccurrences = Math.ceil(0.03 * totalWords);

			return {
				keyphraseOccurrences: keyphraseOccurrences,
				advisedMinOccurrences: advisedMinOccurrences,
				advisedMaxOccurrences: advisedMaxOccurrences,
			};
		},
		getParagraphs: function (data) {
			var paragraphs = data.find('p');
			// Remove paragraphs that have any of the target classes or ids
			var preventedSelectors = ['.vc_ui-help-block'];
			paragraphs = paragraphs.filter(function (index, element) {
				var $paragraph = $(element);
				// Check if the paragraph contains any of the target selectors
				var containsPreventedSelectors = preventedSelectors.some(function (selector) {
					return $paragraph.is(selector);
				});
				return !containsPreventedSelectors;
			});
			// Remove paragraphs without text content
			paragraphs = paragraphs.filter(function (index, element) {
				return $(element).text().trim().length > 0;
			});
			// Remove paragraphs that only contain <a> tags
			paragraphs = paragraphs.filter(function (index, element) {
				return !(1 === $(element).contents().length && 1 === $(element).children('a').length);
			});
			// Return the text content of the first paragraph
			return paragraphs;
		},
		getSentences: function (text) {
			var sentences = text.match(/\(?[^\.\?\!]+[\.!\?]\)?/g);
			return sentences || [];
		},
		/**
		 * Checks whether the text contains three or more sentences in a row all starting with the same word.
		 *
		 * @param {string} text The text to check for consecutive sentences.
		 * @returns {Object} An object containing the analysis results.
		 *   - consecutiveCount {number} - The number of consecutive sentences with the same starting word.
		 *   - state {boolean} - True if three or more consecutive sentences start with the same word, otherwise false.
		 */
		hasConsecutiveSentences: function (text) {
			// Split the text into sentences using a simple regular expression
			var sentences = this.getSentences(text);

			// Check for consecutive sentences with the same start word
			var consecutiveCount = 1;
			for (var i = 1; i < sentences.length; i++) {
				var currentStartWord = sentences[i].split(' ')[0];
				var previousStartWord = sentences[i - 1].split(' ')[0];

				if (currentStartWord === previousStartWord) {
					consecutiveCount++;
					if (consecutiveCount >= 3) {
						return {
							consecutiveCount: consecutiveCount,
							state: true
						};
					}
				} else {
					consecutiveCount = 1;
				}
			}

			return {
				consecutiveCount: consecutiveCount,
				state: false
			};
		},
		getPassiveVoicePercentage: function (paragraphs) {
			// Initialize counters
			var totalSentences = 0;
			var passiveVoiceSentences = 0;

			if (paragraphs.length) {
				var _this = this;
				// Iterate through paragraphs
				paragraphs.each(function (index, element) {
					// Split paragraph into sentences
					var sentences = _this.getSentences($(element).text());

					// Update total sentence count
					totalSentences += sentences.length;

					// Check for passive voice in each sentence
					sentences.forEach(function (sentence) {
						if (_this.hasPassiveVoice(sentence)) {
							passiveVoiceSentences++;
						}
					});
				});
			}

			// Calculate percentage
			var percentage = totalSentences ? (passiveVoiceSentences / totalSentences) * 100 : 0;

			return percentage.toFixed(2);
		},
		hasPassiveVoice: function (text) {
			// Regular expression to identify passive voice patterns
			var passiveVoiceRegex = /\b(am|are|is|was|were|been|being)\s+[^.!?]*\b(by)\b/;
			// Check if the text contains passive voice
			return passiveVoiceRegex.test(text);
		},
		getWordsCount: function (text) {
			var punctuationRegexString = "\\–\\-\\(\\)_\\[\\]’‘“”〝〞〟‟„\"'.?!:;,¿¡«»‹›\u2014\u00d7\u002b\u0026\u06d4\u061f\u060C\u061B\u3002\uff61" +
				"\uff01\u203c\uff1f\u2047\u2049\u2048\u2025\u2026\u30fb\u30fc\u3001\u3003\u3004\u3006\u3007\u3008\u3009\u300a\u300b\u300c\u300d\u300e" +
				"\u300f\u3010\u3011\u3012\u3013\u3014\u3015\u3016\u3017\u3018\u3019\u301a\u301b\u301c\u301d\u301e\u301f\u3020\u3036\u303c\u303d\uff5b" +
				"\uff5d\uff5c\uff5e\uff5f\uff60\uff62\uff63\uff64\uff3b\uff3d\uff65\uffe5\uff04\uff05\uff20\uff06\uff07\uff08\uff09\uff0a\uff0f\uff1a" +
				"\uff1b\uff1c\uff1e\uff3c\\<>";
			var interJectionRegexString = "([ " + punctuationRegexString + "])";
			// Punctuation marks are tokenized as if they were words.
			var words = text.split(/\s/g);
			// Punctuation marks are tokenized as if they were words.
			words = words.reduce(function (result, word) {
				var newWord = word.replace(new RegExp(interJectionRegexString, 'g'), " $1 ");
				return result.concat(newWord.split(" "));
			}, []);

			words = words.filter(function (word) {
				return "" !== word.trim();
			});

			return words.length;
		},
		/**
		 * Checks the subheading distribution in an HTML string.
		 *
		 * @param {string} htmlString - The input HTML string to analyze.
		 * @returns {Array} An array of objects, each representing a section of text with its analysis results.
		 *   - wordCount {number} - The number of words in the section.
		 *   - subheadingCount {number} - The number of subheadings in the section.
		 */
		getTextSectionCount: function (htmlString) {
			var parser = new DOMParser();
			var doc = parser.parseFromString(htmlString, 'text/html');

			// Helper function to count words in a text node
			function countWords (text) {
				return text.trim().split(/\s+/).length;
			}

			// Helper function to check if a node contains subheadings
			function countSubheadings (node) {
				var subheadingElements = node.querySelectorAll('h1, h2, h3, h4, h5, h6');
				return subheadingElements.length;
			}

			// Helper function to analyze a section and return the results
			function analyzeSection (node) {
				var paragraphElements = node.querySelectorAll('p');
				var wordCount = 0;
				var subheadingCount = countSubheadings(node);

				// Count words only within paragraph elements
				paragraphElements.forEach(function (paragraph) {
					return wordCount += countWords(paragraph.textContent || '');
				});

				return {
					wordCount: wordCount,
					subheadingCount: subheadingCount
				};
			}

			// Traverse the DOM to analyze each section
			var nodes = doc.body.childNodes;
			var results = [];

			nodes.forEach(function (node) {
				if (node.nodeType === Node.ELEMENT_NODE) {
					// Analyze the section and push results to the array
					var sectionResults = analyzeSection(node);
					results.push(sectionResults);
				}
			});

			return results;
		}
	};

})(window.jQuery);
Site is undergoing maintenance

PACJA Events

Maintenance mode is on

Site will be available soon. Thank you for your patience!