Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Release 2.1.0 #37

Merged
merged 72 commits into from
Dec 17, 2024
Merged
Changes from 1 commit
Commits
Show all changes
72 commits
Select commit Hold shift + click to select a range
c28ad8d
added link prefetch options component, LinkPrefetch class and js for …
geckod22 Oct 23, 2024
03256c3
added defer attribute to script tag
geckod22 Oct 25, 2024
4145534
phpcs on new class
geckod22 Oct 25, 2024
ba1198a
added copy text and changed toggle style
geckod22 Nov 4, 2024
3f90470
added restapi to manage the settings, adding bottom margin to fields
geckod22 Nov 4, 2024
f7801ae
New branch Jetpack Boost
Nov 4, 2024
fbf0991
Fix: phpcs
Nov 5, 2024
fe39f82
removed console.log
geckod22 Nov 7, 2024
3236585
Tweak: aligned toggle input next to labels
Nov 8, 2024
61fa41f
fixed check for mobile
geckod22 Nov 11, 2024
8374c5f
added fix for minified js
geckod22 Nov 11, 2024
1764da6
Tweak: replace "console.log" with "console.error"
Nov 13, 2024
94f7456
Tweak: groups imports by type
AleTorrisi Nov 18, 2024
d4c4074
Tweak: component and variable name in "Advanced Settings" section
AleTorrisi Nov 18, 2024
49feb01
Update includes/RestApi/LinkPrefetchController.php
geckod22 Nov 18, 2024
24dff12
Update includes/RestApi/LinkPrefetchController.php
geckod22 Nov 18, 2024
ecbdb47
updated LinkPrefetchController with more fix
geckod22 Nov 18, 2024
f743bfe
adding suggested improvements to LinkPrefetchController.php
geckod22 Nov 18, 2024
b7f0abd
removed unused Permissions class and removed LinkPrefetch new and con…
geckod22 Nov 18, 2024
1427769
defaultText.js fix
geckod22 Nov 18, 2024
62999b1
Update assets/js/linkPrefetch.js
geckod22 Nov 18, 2024
b5a5284
Update components/linkPrefetch/index.js
geckod22 Nov 18, 2024
c9c5775
moved non-react scripts to scripts folder
geckod22 Nov 18, 2024
2202638
added RestApi.php
geckod22 Nov 18, 2024
5b5c22d
applied suggestion changes to component index.js
geckod22 Nov 18, 2024
319e8c9
Tweak: better handling of Rest API
AleTorrisi Nov 18, 2024
5f18dca
other fixies for RestApi and LinkPrefetch Starting
geckod22 Nov 18, 2024
23ed28b
Tweak: handling of failure during saving options
AleTorrisi Nov 18, 2024
6bf3faf
Tweak: remove unused constant
AleTorrisi Nov 18, 2024
a569d2a
Tweak: parameter name
AleTorrisi Nov 18, 2024
3e39933
Tweak: import 'useState' and ''useEffect' from WordPress library inst…
AleTorrisi Nov 18, 2024
af2d843
moved inline styles to file and load it
geckod22 Nov 18, 2024
d29a258
some phpcs fixiews
geckod22 Nov 18, 2024
e010497
phpcs
geckod22 Nov 18, 2024
be0bf02
Tweak: fetch options function
AleTorrisi Nov 18, 2024
d1c6e62
Fix remove unused text
AleTorrisi Nov 18, 2024
b71cd5f
Tweak: store Jetpack Boost options in NewFold Runtime object avoiding…
AleTorrisi Nov 18, 2024
b58793d
Tweak: handling of style script
AleTorrisi Nov 18, 2024
e2170d6
linkPrefetch.js lint
geckod22 Nov 18, 2024
717557b
Tweak: replaced "src/css" folder with "styles" folder
AleTorrisi Nov 18, 2024
70b1e1f
JS Lint
AleTorrisi Nov 18, 2024
37799c1
Fix phpcs
AleTorrisi Nov 18, 2024
a113d03
phpcs fix
AleTorrisi Nov 18, 2024
ba9e987
show the caching admin bar item and dynamically set the settings link…
geckod22 Nov 19, 2024
96f9b80
Lint js files
arunshenoy99 Nov 20, 2024
736cf48
New: upsell option "Optimize Critical CSS Loading"
AleTorrisi Nov 25, 2024
3bbc47a
Removed useless comment
AleTorrisi Nov 25, 2024
2f0e7d7
Fix: hide premium option "Critical CSS" when module is disabled
AleTorrisi Nov 26, 2024
0e5d09a
Test commit for cy test
AleTorrisi Nov 26, 2024
c748863
Fix: Jetpack Boost upsell link
AleTorrisi Nov 26, 2024
a78673c
fix for mousedown event
geckod22 Nov 27, 2024
74361d5
Fix: get correctly the site url in case of subdomain
AleTorrisi Nov 27, 2024
cd5d1d8
adding minified js
geckod22 Nov 27, 2024
e7e7c10
Testing cypress fix
arunshenoy99 Dec 12, 2024
64bf1bd
Fix Web and Mojo fatal error
arunshenoy99 Dec 12, 2024
150f392
Merge pull request #25 from newfold-labs/enhance/PRESS7-60-jetpack-boost
arunshenoy99 Dec 14, 2024
9fc34ae
Merge branch 'release/v2.1.0' of https://github.com/newfold-labs/wp-m…
arunshenoy99 Dec 14, 2024
e8537e7
Merge pull request #26 from newfold-labs/enhance/PRESS7-64-link-prefetch
arunshenoy99 Dec 14, 2024
d2577fe
Merge branch 'release/v2.1.0' of https://github.com/newfold-labs/wp-m…
arunshenoy99 Dec 14, 2024
e082ccc
Merge pull request #29 from newfold-labs/fix/PRESS7-78-clear-cache-ba…
arunshenoy99 Dec 14, 2024
f94146f
Fix: set default value for exclusion settings
AleTorrisi Dec 16, 2024
dc9112c
fixed issue when exclude keywords is empty
geckod22 Dec 16, 2024
6d00bd3
added timeout for ignorekeywords field
geckod22 Dec 16, 2024
4e7e7ef
sanitize ignorekeywords input field
geckod22 Dec 16, 2024
1ef2bfa
little change in jetpack option for a string translation
geckod22 Dec 16, 2024
ea3bf5c
Update lint workflow with path and debug
arunshenoy99 Dec 17, 2024
fae8475
Minor formatting fixes
arunshenoy99 Dec 17, 2024
ebe6485
Fix formatting in PHP files
arunshenoy99 Dec 17, 2024
9e0ab82
Remove debug statement from workflow
arunshenoy99 Dec 17, 2024
2407609
Merge pull request #38 from newfold-labs/release/v2.1.0-bugfix
arunshenoy99 Dec 17, 2024
d9e62a5
Improve lint workflow
arunshenoy99 Dec 17, 2024
5083aeb
Update workflow to wait for 1 second
arunshenoy99 Dec 17, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Update assets/js/linkPrefetch.js
Co-authored-by: Arun Shenoy <devarunshenoy99@gmail.com>
  • Loading branch information
geckod22 and arunshenoy99 authored Nov 18, 2024

Verified

This commit was created on GitHub.com and signed with GitHub’s verified signature.
commit 62999b1b3152e965deabd15c33df4471a9b0bdc6
290 changes: 87 additions & 203 deletions assets/js/linkPrefetch.js
Original file line number Diff line number Diff line change
@@ -1,219 +1,103 @@
window.addEventListener( 'load', () => {
/* check if the browser supports Prefetch */
const testlink = document.createElement("link"),
supportsPrefetchCheck = testlink.relList && testlink.relList.supports && testlink.relList.supports("prefetch"),
/* check if the user has set a reduced data usage option on the user agent or if the current connection effective type is 2g */
navigatorConnectionCheck = navigator.connection && (navigator.connection.saveData || (navigator.connection.effectiveType || "").includes("2g")),
intersectionObserverCheck = window.IntersectionObserver && "isIntersecting" in IntersectionObserverEntry.prototype;

if ( ! supportsPrefetchCheck || navigatorConnectionCheck ) {
return;
} else {
class LP_APP {
constructor(config) {
this.config = config;
this.activeOnDesktop = config.activeOnDesktop;
this.behavior = config.behavior;
this.hoverDelay = config.hoverDelay;
this.ignoreKeywords = config.ignoreKeywords.split(',');
this.instantClick = config.instantClick;
this.mobileActive = config.activeOnMobile;
this.isMobile = config.isMobile;
this.mobileBehavior = config.mobileBehavior;
this.prefetchedUrls = new Set();
this.timerIdentifier;
this.eventListenerOptions = { capture: !0, passive: !0 };
}
/**
* Init
* @returns {void}
*/
init() {
const isChrome = navigator.userAgent.indexOf("Chrome/") > -1,
chromeVersion = isChrome && parseInt(navigator.userAgent.substring(navigator.userAgent.indexOf("Chrome/") + "Chrome/".length));

if ( isChrome && chromeVersion < 110 ) {return;}
if ( this.isMobile && ! this.mobileActive ) {return;}
if ( ! this.isMobile && ! this.activeOnDesktop ) {return;}

if ( ! this.isMobile ) {
if ( 'mouseHover' === this.behavior ) {
let hoverDelay = parseInt(this.hoverDelay);
hoverDelay = isNaN(hoverDelay) ? 60 : hoverDelay;
document.addEventListener("mouseover", this.mouseHover.bind(this), this.eventListenerOptions);
} else if ( 'mouseDown' === this.behavior ) {
if ( this.instantClick ) {
document.addEventListener("mousedown", this.mouseDownToClick.bind(this), this.eventListenerOptions);
} else {
document.addEventListener("mousedown", this.mouseDown.bind(this), this.eventListenerOptions)
}
}
}
document.addEventListener('DOMContentLoaded', () => {
const testLink = document.createElement('link');
const supportsPrefetch = testLink.relList?.supports?.('prefetch');
const isDataSaver = navigator.connection?.saveData || navigator.connection?.effectiveType?.includes('2g');
const supportsIntersectionObserver = 'IntersectionObserver' in window && 'isIntersecting' in IntersectionObserverEntry.prototype;

if ( this.mobileActive && this.isMobile ) {
if ( 'touchstart' === this.mobileBehavior ) {
document.addEventListener("touchstart", this.touchstart.bind(this), this.eventListenerOptions);
} else if ( 'viewport' && intersectionObserverCheck ) {
this.viewport();
}
}
}
/**
* Viewport handler
* @returns {void}
*/
viewport() {
const io = new IntersectionObserver((e) => {
e.forEach((e) => {
if (e.isIntersecting) {
const n = e.target;
io.unobserve(n);
this.canPrefetch(n) && this.prefetchIt(n.href);
}
});
});
let requestIdleCallback = window.requestIdleCallback ||
function (cb) {
var start = Date.now();
return setTimeout(function () {
cb({
didTimeout: false,
timeRemaining: function () {
return Math.max(0, 50 - (Date.now() - start));
}
});
}, 1);
};
requestIdleCallback( () => {
return setTimeout(function () {
return document.querySelectorAll("a").forEach(function (a) {
return io.observe(a);
});
}, 1000);
}, { timeout: 1000 });
}
/**
* Mouse Down handler
* @param {Event} e - listener event
* @returns {void}
*/
mouseDown(e) {
const el = e.target.closest("a");
this.canPrefetch(el) && this.prefetchIt(el.href);
}
if (!supportsPrefetch || isDataSaver) return;

/**
* Mouse Down handler for instant click
* @param {Event} e - listener event
* @returns {void}
*/
mouseDownToClick(e) {
//if (performance.now() - o < r) return;
const el = e.target.closest("a");
if (e.which > 1 || e.metaKey || e.ctrlKey) return;
if (!el) return;
el.addEventListener(
"click",
function (t) {
'lpappinstantclick' != t.detail && t.preventDefault();
},
{ capture: !0, passive: !1, once: !0 }
);
const n = new MouseEvent("click", { view: window, bubbles: !0, cancelable: !1, detail: 'lpappinstantclick' });
el.dispatchEvent(n);
}
class LinkPrefetcher {
constructor(config) {
Object.assign(this, {
activeOnDesktop: config.activeOnDesktop,
behavior: config.behavior,
hoverDelay: parseInt(config.hoverDelay) || 60,
ignoreKeywords: config.ignoreKeywords.split(','),
instantClick: config.instantClick,
mobileActive: config.activeOnMobile,
isMobile: config.isMobile,
mobileBehavior: config.mobileBehavior,
prefetchedUrls: new Set(),
eventOptions: { capture: true, passive: true },
timerId: null,
});
}

touchstart(e) {
const el = e.target.closest("a");
this.canPrefetch(el) && this.prefetchIt(el.href);
}
init() {
const isChrome = navigator.userAgent.includes('Chrome/');
const chromeVersion = isChrome && parseInt(navigator.userAgent.split('Chrome/')[1]);

/**
* Clean Timers
* @param {Event} t - listener event
* @returns {void}
*/
clean(t) {
if ( t.relatedTarget && t.target.closest("a") == t.relatedTarget.closest("a") || this.timerIdentifier ) {
clearTimeout( this.timerIdentifier );
this.timerIdentifier = void(0);
}
if (isChrome && chromeVersion < 110) return;
if ((this.isMobile && !this.mobileActive) || (!this.isMobile && !this.activeOnDesktop)) return;

if (!this.isMobile) {
this.behavior === 'mouseHover' && document.addEventListener('mouseover', this.handleHover.bind(this), this.eventOptions);
this.behavior === 'mouseDown' && document.addEventListener('mousedown', this.instantClick ? this.handleInstantClick.bind(this) : this.handleMouseDown.bind(this), this.eventOptions);
}

/**
* Mouse hover function
* @param {Event} e - listener event
* @returns {void}
*/
mouseHover(e) {
if ( !("closest" in e.target) ) return;
const link = e.target.closest("a");
if ( this.canPrefetch( link ) ) {
link.addEventListener("mouseout", this.clean.bind(this), { passive: !0 });
this.timerIdentifier = setTimeout(()=> {
this.prefetchIt( link.href );
this.timerIdentifier = void(0);
}, this.hoverDelay);
}
if (this.isMobile && this.mobileBehavior === 'viewport' && supportsIntersectionObserver) {
this.setupViewportObserver();
}

/**
* Can the url be prefetched or not
* @param {Element} el - link element
* @returns {boolean} - if it can be prefetched
*/
canPrefetch( el ) {
if ( el && el.href ) {
/* it has been just prefetched before */
if (this.prefetchedUrls.has(el.href)) {
return false;
}
}

/* avoid if it is the same url as the actual location */
if ( el.href.replace(/\/$/, "") !== location.origin.replace(/\/$/, "") && el.href.replace(/\/$/, "") !== location.href.replace(/\/$/, "") ) {
return true;
setupViewportObserver() {
const observer = new IntersectionObserver((entries) => {
entries.forEach((entry) => {
if (entry.isIntersecting) {
observer.unobserve(entry.target);
this.prefetchIfEligible(entry.target.href);
}
});
});

/* checking exclusions */
const exclude = this.ignoreKeywords.filter( k => {
if ( el.href.indexOf (k) > -1) {
return k;
}
})
if ( exclude.length > 0 ) { return false; }

}

return false;
}
const idleCallback = window.requestIdleCallback || ((cb) => setTimeout(cb, 1));
idleCallback(() => setTimeout(() => document.querySelectorAll('a').forEach((a) => observer.observe(a)), 1000));
}

/**
* Append link rel=prefetch to the head
* @param {string} url - url to prefetch
* @returns {void}
*/
prefetchIt(url) {
const toPrefechLink = document.createElement("link");
handleMouseDown(event) {
const el = event.target.closest('a');
this.prefetchIfEligible(el?.href);
}

toPrefechLink.rel = "prefetch";
toPrefechLink.href = url;
toPrefechLink.as = "document";
handleInstantClick(event) {
const el = event.target.closest('a');
if (!el || event.which > 1 || event.metaKey || event.ctrlKey) return;
el.addEventListener('click', (e) => e.detail !== 'instantClick' && e.preventDefault(), { capture: true, passive: false, once: true });
el.dispatchEvent(new MouseEvent('click', { bubbles: true, cancelable: false, detail: 'instantClick' }));
}

document.head.appendChild(toPrefechLink);
this.prefetchedUrls.add(url);
}
handleHover(event) {
const link = event.target.closest('a');
if (!link || !this.prefetchIfEligible(link.href)) return;
link.addEventListener('mouseout', () => clearTimeout(this.timerId), this.eventOptions);
this.timerId = setTimeout(() => this.prefetchIt(link.href), this.hoverDelay);
}

prefetchIfEligible(url) {
if (!url || this.prefetchedUrls.has(url)) return false;
if (url.replace(/\/$/, '') === location.href.replace(/\/$/, '') || this.ignoreKeywords.some((k) => url.includes(k))) return false;
return true;
}

prefetchIt(url) {
const link = document.createElement('link');
link.rel = 'prefetch';
link.href = url;
document.head.appendChild(link);
this.prefetchedUrls.add(url);
}
/*
default config:
'activeOnDesktop' => true,
'behavior' =>'mouseHover',
'hoverDelay' => 60,
'instantClick' => true ,
'activeOnMobile' => true ,
'mobileBehavior' => 'viewport',
'ignoreKeywords' =>'wp-admin,#,?',
*/
const lpapp = new LP_APP( window.LP_CONFIG );
lpapp.init();
}

const config = window.LP_CONFIG || {
activeOnDesktop: true,
behavior: 'mouseHover',
hoverDelay: 60,
instantClick: true,
activeOnMobile: true,
mobileBehavior: 'viewport',
ignoreKeywords: 'wp-admin,#,?',
};

new LinkPrefetcher(config).init();
});