Created
October 22, 2024 17:47
-
-
Save mimoklef/361e9191ead9bb4936019fb50e3eda49 to your computer and use it in GitHub Desktop.
It's a user script that restores Google Maps features recently removed by Google in Europe, such as the tabs "Maps" button, "Open in Maps" button and the ability to click on mini-maps in Google search results. With this script, you can enjoy the simplicity and seamless navigation between Google search results and Google Maps.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// ==UserScript== | |
// @name Search Google Maps Back | |
// @name:fr Retour de Google Maps Dans Recherche | |
// @namespace http://tampermonkey.net/ | |
// @version 2024-10-22 | |
// @description This script bring google maps button back, makes search maps (big/mini and micro one) clickable and adds Open in Maps button back. It might not work anymore in a few months and need an update so feel free to update it whenever you want, I'll try to do it whenever I can. | |
// @description:fr Ce script remet le bouton google maps, rend les cartes de recherche (grandes/mini et micro) cliquables et ajoute le bouton Open in Maps (Ouvrir sur Maps) aux grandes cartes. Il se peut qu'il ne fonctionne plus dans quelques mois et qu'une mise à jour soit nécessaire, alors n'hésitez pas à le mettre à jour quand vous le souhaitez, pour ma part je le ferai dès que possible. | |
// @author Mimouy | Mimo (Mohamed) Bouyakhlef : https://github.com/mimouy | |
// @match https://www.google.com/search* | |
// @include https://www.google.tld/search* | |
// @icon https://i.ibb.co/RcMNxV3/gmback.jpg | |
// @grant none | |
// @license MIT | |
// @downloadURL https://update.greasyfork.org/scripts/513482/Search%20Google%20Maps%20Back.user.js | |
// @updateURL https://update.greasyfork.org/scripts/513482/Search%20Google%20Maps%20Back.meta.js | |
// ==/UserScript== | |
//Link to the git repo : https://github.com/mimouy/Search-Google-Maps-Back | |
(function() { | |
'use strict'; | |
function addMapsButton() { | |
// Find the list container of existing tabs | |
const tabsContainer = document.querySelector('.crJ18e'); | |
//Get the search query | |
const searchQuery = new URLSearchParams(window.location.search).get('q'); | |
//Use the search query as a link | |
const mapsLink = `//maps.google.com/maps?q=${searchQuery}`; | |
// Adding the Maps button to the tab if not already in | |
if (tabsContainer) { | |
// Check if already has a Maps button | |
let hasGoogleMapsLink = false; | |
tabsContainer.querySelectorAll('a').forEach(link => { | |
if (link.href.includes('google.com/maps')) { | |
hasGoogleMapsLink = true; | |
} | |
}); | |
if (hasGoogleMapsLink) { | |
//Already has a Maps button | |
} else { | |
// Create the Maps button elements (updated) | |
const mapsListItem = document.createElement('div'); | |
mapsListItem.jsname = 'VIftV'; | |
// mapsListItem.classList.add('Ap1Qsc'); | |
mapsListItem.setAttribute('role', 'listitem'); | |
// Replace this entire section with the provided <a> element | |
const mapsButton = document.createElement('a'); | |
mapsButton.href = mapsLink; | |
mapsButton.jsname = "ONH4Gc"; | |
mapsButton.classList.add("LatpMc"); | |
mapsButton.classList.add("nPDzT"); | |
mapsButton.classList.add("T3FoJb"); | |
mapsButton.dataset.navigation = "server"; | |
mapsButton.dataset.hveid = "CAEQCA"; | |
//mapsButton.textContent = "Maps"; // Set the inner text if need to | |
const mapsButtonText = document.createElement('div'); | |
mapsButtonText.jsname = "bVqjv"; | |
mapsButtonText.classList.add("YmvwI"); | |
mapsButtonText.textContent = "Maps"; | |
mapsButton.appendChild(mapsButtonText); | |
// Append the mapsButton to the list item | |
mapsListItem.appendChild(mapsButton); | |
// Insert the Maps button after the All button | |
const firstTabsChild = tabsContainer.firstElementChild; | |
firstTabsChild.insertBefore(mapsButton, firstTabsChild.children[1]); | |
} | |
} | |
// Find the small map element if any, to map it clickable | |
const smallMapElement = document.querySelector('.KY6ERe'); | |
if (smallMapElement) { | |
// Look for the links (or div) parent | |
const targetElement = smallMapElement.querySelector('.ZqGZZ.xP81Pd'); | |
//Look for the micro map image | |
if (targetElement) { | |
//Is the mini map image a link ? | |
if (targetElement.tagName.toLowerCase() === 'a') { | |
//Mini map is already a link, nothing to do | |
}else{ | |
//Create a new Map image which will be a link | |
let newMapImage = document.createElement('a'); | |
let parent = targetElement.parentNode; | |
let children = targetElement.childNodes; | |
// Copy all attributes and childs to the newMapImage | |
Array.prototype.forEach.call(targetElement.attributes, function (attr) { | |
newMapImage.setAttribute(attr.name, attr.value); | |
}); | |
Array.prototype.forEach.call(children, function (elem) { | |
newMapImage.appendChild(elem); | |
}); | |
newMapImage.href = mapsLink; | |
//Replace old image by the new (link) one | |
parent.replaceChild(newMapImage, targetElement); | |
} | |
} else { | |
// | |
} | |
} else { | |
//No small map | |
} | |
// Find the micro map element if any, to make it clickable | |
const microMapElement = document.querySelector('.Ggdpnf.kno-fb-ctx'); | |
if (microMapElement) { | |
// Look for the link's (which is a div if none) parent | |
const microTargetElement = microMapElement.querySelector('* > * > a'); | |
if (microTargetElement) { | |
if (microTargetElement.tagName.toLowerCase() === 'a' && microTargetElement.hasAttribute('href')) { | |
//Micro map is already a link, nothing to do | |
}else{ | |
//Micro map is not a link (do not have a href) | |
microTargetElement.href = mapsLink; | |
// Create the Zoom icon div | |
const newDiv = document.createElement('div'); | |
newDiv.setAttribute('jscontroller', 'hnlzI'); | |
newDiv.setAttribute('class', 'sEtYzd duf-h TUOsUe BSRXQc sxd9Pc'); | |
newDiv.setAttribute('jsaction', 'KQB0gd;rcuQ6b:npT2md'); | |
newDiv.setAttribute('data-ved', '2ahUKEwjo44_p66CJAxXtFFkFHcHlDIUQkNEBegQIUxAJ'); | |
// Create img element (zoom icon) inside the div | |
const img = document.createElement('img'); | |
img.setAttribute('class', 'kf0xcf oYQBg FIfWIe Tbiej u60jwe'); | |
img.setAttribute('src', ''); | |
img.setAttribute('alt', ''); | |
img.setAttribute('height', '24'); | |
img.setAttribute('width', '24'); | |
img.setAttribute('data-csiid', 'nQAXZ6ihIe2p5NoPwcuzqAg_9'); | |
img.setAttribute('data-atf', '0'); | |
newDiv.style.cssText = ` | |
background-color: rgba(48, 49, 52, 0.8); | |
box-shadow: 0px 1px 2px rgba(0, 0, 0, 0.2); | |
height: 32px; | |
width: 32px; | |
display: block; | |
outline: 0; | |
position: absolute; | |
top: 4px; | |
vertical-align: middle; | |
right: 4px; | |
`; | |
// Append the img to the Zoom div | |
newDiv.appendChild(img); | |
// Find the gImg element | |
const gImgElement = microTargetElement.querySelector('g-img'); | |
// Insert the zoom icon right after the gImh element so it shows on it (top right) | |
if (gImgElement) { | |
gImgElement.parentNode.insertBefore(newDiv, gImgElement.nextSibling); | |
} else { | |
//no gImg found !? | |
} | |
} | |
} else { | |
//Didn't find the link !? | |
} | |
} else { | |
//No small map | |
} | |
//lu map section (when yout type an exact address) | |
const addressMapElement = document.querySelector('.lu_map_section'); | |
if (addressMapElement) { | |
// Searching for the link containing "maps/dir/" | |
const adirElement = document.querySelector('a[href*="maps/dir/"]'); | |
if (adirElement) { | |
// Clone link | |
const clonedAElement = adirElement.cloneNode(true); | |
clonedAElement.href = mapsLink; | |
// Insert the clone which will become a Map button | |
adirElement.parentNode.insertBefore(clonedAElement, adirElement.nextSibling); | |
// Searching the "Direction" text | |
const targetDiv = clonedAElement.querySelector('.QuU3Wb.sjVJQd'); | |
if (targetDiv) { | |
// Changing "Direction" to "Map" | |
const newDiv = document.createElement('div'); | |
newDiv.textContent = 'Map'; | |
targetDiv.innerHTML = ''; | |
targetDiv.appendChild(newDiv); | |
// Creating Map SVG element | |
const svgElement = document.createElementNS('http://www.w3.org/2000/svg', 'svg'); | |
svgElement.setAttribute('focusable', 'false'); | |
svgElement.setAttribute('xmlns', 'http://www.w3.org/2000/svg'); | |
svgElement.setAttribute('viewBox', '0 0 24 24'); | |
svgElement.style.width = '60%'; | |
svgElement.style.fill = '#8ab4f8'; | |
const pathElement = document.createElementNS('http://www.w3.org/2000/svg', 'path'); | |
pathElement.setAttribute('d', 'M20.5 3l-.16.03L15 5.1 9 3 3.36 4.9c-.21.07-.36.25-.36.48V20.5c0 .28.22.5.5.5l.16-.03L9 18.9l6 2.1 5.64-1.9c.21-.07.36-.25.36-.48V3.5c0-.28-.22-.5-.5-.5zM15 19l-6-2.11V5l6 2.11V19z'); | |
svgElement.appendChild(pathElement); | |
//Searching the icon div | |
const svgDiv = clonedAElement.querySelector('.kHtcsd'); | |
if(svgDiv){ | |
svgDiv.innerHTML = ''; | |
svgDiv.appendChild(svgElement); | |
} | |
} | |
} else { | |
//No direction link found ? | |
} | |
}else{ | |
//No lu_map_section found | |
} | |
//Big expandable map change direction to open in maps | |
// Wait for the page to load entirely | |
window.addEventListener('load', () => { | |
// Find the big map's buttons div class="EeWPwe", which contains Direction and Open in maps | |
const bigMapButtonsElement = document.querySelector('.EeWPwe'); | |
if (bigMapButtonsElement) { | |
// Find all bigMapButtonsElement a childs (which are Direction and Open in Maps buttons) | |
const aElements = bigMapButtonsElement.querySelectorAll('a'); | |
// If there is only one <a> = No Open in maps button, only Direction one | |
if (aElements.length === 1) { | |
// Clone it | |
const clonedAElement = aElements[0].cloneNode(true); | |
// Change the link for Direction to Maps one | |
if (clonedAElement.href.includes('maps/dir/')) { | |
clonedAElement.href = mapsLink; | |
} | |
// Add the clone | |
aElements[0].parentNode.insertBefore(clonedAElement, aElements[0].nextSibling); | |
// Find the element with "m0MNmc" which contains text "Direction" and change it to Open in Maps (Sorry for ppl who have their google in other langages) | |
const m0MNmcSpan = clonedAElement.querySelector('.m0MNmc'); | |
if (m0MNmcSpan) { | |
m0MNmcSpan.textContent = 'Open in Maps'; //You can put whatever you want here, if you want it to show in another langage | |
} | |
// Find the "POUQwd WN4Zxc" span in the clone, which is the icon one, and change it to Maps icon | |
const pouqwdElement = clonedAElement.querySelector('.POUQwd.WN4Zxc'); | |
if (pouqwdElement) { | |
// Create maps icon | |
const newDiv = document.createElement('div'); | |
newDiv.className = 'POUQwd WN4Zxc'; | |
newDiv.innerHTML = ` | |
<span> | |
<span style="height:20px;line-height:20px;width:20px" class="z1asCe Y5lOv"> | |
<svg focusable="false" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"> | |
<path d="M20.5 3l-.16.03L15 5.1 9 3 3.36 4.9c-.21.07-.36.25-.36.48V20.5c0 .28.22.5.5.5l.16-.03L9 18.9l6 2.1 5.64-1.9c.21-.07.36-.25.36-.48V3.5c0-.28-.22-.5-.5-.5zM15 19l-6-2.11V5l6 2.11V19z"></path> | |
</svg> | |
</span> | |
</span> | |
`; | |
// Replace it | |
pouqwdElement.parentNode.replaceChild(newDiv, pouqwdElement); | |
} | |
} else if (aElements.length > 1) { | |
//There are two elements so I think there's no need to do anything, as the second one must be "Open in Maps" button | |
} else { | |
//No <a> found ? | |
} | |
}else { | |
//No "EeWPwe" found ? | |
} | |
}); | |
} | |
addMapsButton(); | |
})(); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment