Backup Search Engines
-
This modification backs up and restores your search engines, by adding a backup button and a restore input field in
vivaldi://settings/search
. Caution: Works only when you check โOpen Settings in a Tabโ invivaldi://settings/appearance/
.Meanwhile Vivaldi has added search engine sync. You might still want to use this mod, because it
- acts as a backup when you lose your sync data
- gives you the ability to migrate search engines to other sync accounts, profiles without sync
- lets you change/fix search engine favicons from the backup (be careful while editing the JSON data)
- lets you share engines with other users
- Backup: Click the Backup button. This will copy the data to your clipboard. From there paste it to your notes panel, or a dedicated file.
- Restore: Copy the backup data and paste or drag&drop it into the Restore Backup input field. This will restore your backup, but you will lose all search engines you have added since your last backup.
Create a Javascript (
.js
) file and copy/paste following code. See pinned topics on this forum board for instructions.// Backup Search Engines // version 2022.4.0 // https://forum.vivaldi.net/post/277594 // Adds functionality to backup and restore search engines in // vivaldi://settings/search. (function backupSearchEngines() { function msg(print) { clearTimeout(msgTimeout); if (print === "backup") { info.innerText = "Backup copied to clipboard"; } else if (print === "restore") { info.innerText = "Search engines restored"; } else { info.innerText = "Code error, aborted"; } msgTimeout = setTimeout(() => (info.innerText = ""), 5000); } function bringingItAllBackHome(remains, defaultsArray) { vivaldi.searchEngines.getTemplateUrls((engines) => { const getKeys = engines.templateUrls.map((e) => e.keyword); for (let i = 0; i < defaultsArray.length; i++) { const index = getKeys.lastIndexOf(defaultsArray[i][0]); const id = engines.templateUrls[index].id.toString(); const ds = defaultsArray[i][1]; vivaldi.searchEngines.setDefault(ds, id); } remains.forEach((remove) => { vivaldi.searchEngines.removeTemplateUrl(remove); }); msg("restore"); }); } function exec(collection) { vivaldi.searchEngines.getTemplateUrls((engines) => { const oldDefaults = [ engines.defaultImage, engines.defaultPrivate, engines.defaultSearch, ]; const newDefaults = [ collection.defaultImage, collection.defaultPrivate, collection.defaultSearch, collection.defaultSearchField, collection.defaultSearchFieldPrivate, collection.defaultSpeeddials, collection.defaultSpeeddialsPrivate, ]; engines.templateUrls.forEach((engine) => { if (oldDefaults.indexOf(engine.id) === -1) { vivaldi.searchEngines.removeTemplateUrl(engine.id); } }); console.info("restoring search engines..."); const defaultsArray = []; collection.templateUrls.forEach((collect) => { vivaldi.searchEngines.addTemplateUrl(collect, () => { console.info(` \u2022 ${collect.name}`); if (newDefaults.indexOf(collect.id) > -1) { const indeces = newDefaults .map((e, i) => (e === collect.id ? i : "")) .filter(String); indeces.forEach((index) => { let ds; if (index === 0) { ds = vivaldi.searchEngines.DefaultType.DEFAULT_IMAGE; } else if (index === 1) { ds = vivaldi.searchEngines.DefaultType.DEFAULT_PRIVATE; } else if (index === 2) { ds = vivaldi.searchEngines.DefaultType.DEFAULT_SEARCH; } else if (index === 3) { ds = vivaldi.searchEngines.DefaultType.DEFAULT_SEARCH_FIELD; } else if (index === 4) { ds = vivaldi.searchEngines.DefaultType.DEFAULT_SEARCH_FIELD_PRIVATE; } else if (index === 5) { ds = vivaldi.searchEngines.DefaultType.DEFAULT_SPEEDDIALS; } else { ds = vivaldi.searchEngines.DefaultType.DEFAULT_SPEEDDIALS_PRIVATE; } const tunnel = [collect.keyword, ds]; defaultsArray.push(tunnel); }); } }); }); const remains = [...new Set(oldDefaults)]; bringingItAllBackHome(remains, defaultsArray); }); } function restore(e) { e.preventDefault(); e.stopPropagation(); let backupCode; let collection; if (e.type === "paste") { const clipboardData = e.clipboardData; backupCode = clipboardData.getData("text"); } else { backupCode = e.dataTransfer.getData("text"); } try { collection = JSON.parse(backupCode); } catch (err) { msg("error"); return; } if ( "defaultImage" in collection && "defaultPrivate" in collection && "defaultSearch" in collection ) { exec(collection); } else { msg("error"); } } function backup() { vivaldi.searchEngines.getTemplateUrls((engines) => { const backupCode = JSON.stringify(engines, null, 2); navigator.clipboard.writeText(backupCode); msg("backup"); }); } function ui() { const check = document.getElementById("vm-bse-backup"); if (!check) { const place = document.querySelector( ".setting-section > div > .setting-group.unlimited > .setting-single" ); const btn = document.createElement("input"); btn.setAttribute("type", "button"); btn.setAttribute("value", "Backup"); btn.classList.add("vm-bse-backup"); place.insertBefore(btn, place.lastChild); btn.addEventListener("click", backup); const input = document.createElement("input"); input.setAttribute("type", "text"); input.setAttribute("placeholder", "Restore Backup"); input.classList.add("vm-bse-restore"); place.insertBefore(input, place.lastChild); input.addEventListener("paste", restore); input.addEventListener("drop", restore); info = document.createElement("span"); info.classList.add("vm-bse-msg"); place.insertBefore(info, place.lastChild); } } const css = ` .vm-bse-restore { width: 130px; margin-left: 6px; margin-top: 6px; } .vm-bse-restore::-webkit-input-placeholder { opacity: 1; color: var(--colorHighlightBg); text-align: center; } .vm-bse-msg { margin-left: 12px; } `; let msgTimeout; const settingsUrl = "chrome-extension://mpognobbkildjkofajifpdfhcoklimli/components/settings/settings.html?path="; chrome.tabs.onUpdated.addListener((tabId, changeInfo, tab) => { if (changeInfo.url === `${settingsUrl}search`) { setTimeout(ui, 100); const check = document.getElementById("vm-bse-css"); if (!check) { const style = document.createElement("style"); style.id = "vm-bse-css"; style.innerHTML = css; document.getElementsByTagName("head")[0].appendChild(style); } } }); })();
History
- 2021.9.0 Clear removed engines, reset history IDs
- 2022.3.3 Complete rewrite for new backend
- 2022.3.4 Better handling of default search restoration
- 2022.4.0 Restore all defaults (search field, speed dial, private sf/sd), clean up code
Bonus
-
Backup search engines from a profile without the modification: Open devtools for the user interface, visit the console tab and execute following code, copy the output and save to file.
vivaldi.searchEngines.getTemplateUrls((engines) => { console.info(JSON.stringify(engines, null, 2)); });
-
Use an emoji as search engine favicon: Visit https://favicon.io/emoji-favicons/ and pick an emoji. Download the zip archive and extract. Visit https://dopiaza.org/tools/datauri/index.php, upload either the 32px or the 16px version of the image and generate data uri. Copy code and paste as faviconURL of the target engine in the search engines backup. Restore backup.
-
While testing this mod I noticed problems with Vivaldi's native/existing option to restore defaults. If you restore defaults and tick the option to keep custom searches, default searches you have edited will be doubled now (default + edited custom). When selected they will share the same name and the same keyword and in case one of them is the default for private search, this will be doubled too. It's quite a mess really, is this a known problem or will someone have to report this?
-
Ok, not bad, your idea. But I have something simpler.
I use 'ContextSearch web-ext' from the Google Web Store:https://chrome.google.com/webstore/detail/contextsearch-web-ext/ddippghibegbgpjcaaijbacfhjjeafjh
With this extension, you can easily sync your search engine with all browsers and make a local backup on your hard drive.
-
I always find it quite odd that vivaldi decided to re-implement their own storage of search engines.
If they just extended
chrome://settings/searchEngines
then not only would that be (I imagine) less duplication of work, but it would by default get synced through the normal chrome sync interface. -
Fantastic! To test after backup (copied on notes), I've deleted all but DDG, and then restored everything from the backup perfectly, even setting Google as default for normal searches and DDG for private searches.
Remember to push Save button too after restoring, I've noticed that the search field list of engines isn't refreshed if you don't. -
@luetage said in Backup Search Engines:
A companion mod to Import and Export Themes. Makes a backup of all your search engines, which is something that still hasn't been implemented to native Vivaldi Sync. Just recently I reset my profile and was surprised that I had to input all my custom searches again. Quite annoying, so here we goโฆ
The mod adds a backup button and a restore input field to
vivaldi://settings/search
.
- Backup: Click the Backup button. This will copy the data to your clipboard. From there paste it to your notes panel, or anywhere really, but Vivaldi Notes have native Sync, so that's a plus.
- Restore: Copy the backup data and paste or drag&drop it into the Restore Backup input field. This will restore your backup, but you will lose all search engines you have added since your last backup.
Copy and paste following code into a
custom.js
file. See pinned topics on this forum board for instructions./* Backup Search Engines */ function _msgSearch(pnt) { clearTimeout(_msgTimeout); if (pnt === 'backup') { _infoSearch.innerText = 'Search engines backup copied to clipboard.'; } else if (pnt === 'restore') { _infoSearch.innerText = 'Search engines restored.'; } else { _infoSearch.innerText = 'Search engines code error.' } _msgTimeout = setTimeout(function() { _infoSearch.innerText = ''; }, 50000); }; function _restoreSearch() { event.preventDefault(); event.stopPropagation(); if (_eventSearch === 'paste') { var clipboardData = event.clipboardData || window.clipboardData; var engineCode = clipboardData.getData('text'); } else { var engineCode = event.dataTransfer.getData('text'); } try { engines = JSON.parse(engineCode); } catch(err) { _msgSearch('error'); return; } if (Object.keys(engines)[0] === 'context') { chrome.storage.local.set({'SEARCH_ENGINE_COLLECTION': engines}, function() { _msgSearch('restore'); }); } else { _msgSearch('error'); } }; function _backupSearch() { chrome.storage.local.get({'SEARCH_ENGINE_COLLECTION': ''}, function(back) { const engines = back.SEARCH_ENGINE_COLLECTION; const engineCode = JSON.stringify(engines); navigator.clipboard.writeText(engineCode); _msgSearch('backup'); }); }; function searchEngines() { const search = document.querySelector('.setting-search-engine'); const check = document.getElementById('backupSearch'); if (search && !check) { const place = document.querySelector('.setting-section .setting-group.unlimited .setting-single'); const style = document.createElement('style'); style.type = 'text/css'; style.innerHTML = '#backupSearch, #restoreSearch {margin-left: var(--padding);}#restoreSearch{width: 130px;margin-top: var(--padding);}#restoreSearch::-webkit-input-placeholder {opacity: 1;color: var(--colorHighlightBg);text-align: center;}'; document.getElementsByTagName('head')[0].appendChild(style); const backupBtn = document.createElement('input'); backupBtn.setAttribute('type', 'button'); backupBtn.setAttribute('value', 'Backup'); backupBtn.id = 'backupSearch'; place.insertBefore(backupBtn, place.lastChild); const restoreInput = document.createElement('input'); restoreInput.setAttribute('type', 'text'); restoreInput.setAttribute('placeholder', 'Restore Backup'); restoreInput.id = 'restoreSearch'; place.insertBefore(restoreInput, place.lastChild); _infoSearch = document.createElement('span'); _infoSearch.id = 'msgConfirm'; place.insertBefore(_infoSearch, place.lastChild); document.getElementById('backupSearch').addEventListener('click', _backupSearch); const restoreSearch = document.getElementById('restoreSearch'); restoreSearch.addEventListener('paste', function() { _eventSearch = 'paste'; _restoreSearch(event); }); restoreSearch.addEventListener('drop', function() { _eventSearch = 'drop'; _restoreSearch(event); }); _msgTimeout = {}; } }; // Loop waiting for the browser to load the UI. You can call all functions from just one instance. setTimeout(function wait() { const browser = document.getElementById('browser'); if (browser) { document.body.addEventListener('click', function() { setTimeout(searchEngines, 50); }); } else { setTimeout(wait, 300); } }, 300);
Funktioniert bei mir nicht.
-
@LonM Yeah, for example the exception list for cookies is synced โ that's classic chromium storage. But on the other hand I don't see the issue to just sync all custom engines while retaining the defaults per installation. Someone just has to find the time to do it.
-
@iAN-CooG Good point, I didn't notice. Seems like the address bar switches automatically, but the list and the active search engine in the search field stays on the old values. I just updated the mod to trigger the click on the save button automatically after a backup has been restored (don't want anyone to get a cramp in their fingers :P). There was also an issue that messages were displayed for too long, fixed that too.
-
@stardepp We are not on the German forum board, please write English. And it's hard to tell what goes wrong for you. Have you created a file called custom.js and did you edit browser.html to refer to this file? Does any other modification work? Did you restart Vivaldi after you made the change to the application, or did you at least open a new window?
-
@luetage ok, sorry.
-
@luetage Amazing! Thanks to you, it will be more hasslefree to refresh a profile.
-
I tested it and it works well, but i had no doubt.
@luetage you're great, you did two very useful js.
-
@luetage said in Backup Search Engines:
... I just updated the mod to trigger the click on the save button automatically after a backup has been restored (don't want anyone to get a cramp in their fingers :P). There was also an issue that messages were displayed for too long, fixed that too.
Thank you so much, very helpfull mod indeed!
Works ok with Vivaldi 2.4.1488.35 on both Win10-1809_x64, Lmint19.1_x64 and Lmint19.1_i686 (tested also with snapshot 2.5.1497.4 on Lmint19.1_x64).
-
I can't seem to get the restore backup to work. Every time I try to paste/drag&drop the data into the input field, I receive the message "Search engines code error".
I tried it both inside the stable and the snapshot versions.
-
@AltCode This means your backup is faulty. The input field throws an error when the json file is invalid. I just tested the code resetting search engines to default and restoring from backup on snapshot and it works.
Well, give me your search engines code and I'll take a look at it, maybe it's salvageable.
-
Code
{"customOrder":false,"default":"cj3trn97f00003o5l5z6o68tl","defaultPrivate":"cj3trn97g00033o5lj3x9hg2j","engines":[{"custom":false,"faviconUrl":"","historyId":1,"id":"cj3trn97f00003o5l5z6o68tl","keyword":"b","name":"Bing","post":false,"removed":false,"suggestUrl":"https://www.bing.com/osjson.aspx?query=%s","url":"https://www.bing.com/search?FORM=INCOH2&PC=IFJ1&PTAG=ICO-c9d0fc87&q=%s"},{"custom":true,"faviconUrl":"","historyId":2,"id":"cj3trn97g00013o5ljema6csh","keyword":"y","name":"Yahoo!","post":false,"removed":false,"suggestUrl":"","url":"https://us.search.yahoo.com/yhs/search?hspart=iry&hsimp=yhs-fullyhosted_009&type=dpp_vvldnu_00_00¶m1=1¶m2=pa%3Ddowncoll&p=%s"},{"custom":false,"faviconUrl":"","historyId":3,"id":"cj3trn97g00023o5ls9d8wvgb","keyword":"s","name":"Startpage.com","post":false,"removed":false,"suggestUrl":"https://www.startpage.com/cgi-bin/csuggest?query=%s&limit=10&lang=english&format=json","url":"https://www.startpage.com/rvd/search?query=%s&language=auto"},{"custom":false,"faviconUrl":"","historyId":4,"id":"cj3trn97g00033o5lj3x9hg2j","keyword":"d","name":"DuckDuckGo","post":false,"removed":false,"suggestUrl":"https://ac.duckduckgo.com/ac/?q=%s&type=list","url":"https://duckduckgo.com/?q=%s&t=vivaldi"},{"custom":false,"faviconUrl":"","historyId":5,"id":"cj3trn97g00043o5l8swtsbjo","keyword":"e","name":"Ecosia","post":false,"removed":false,"suggestUrl":"","url":"https://www.ecosia.org/search?tt=vivaldi&q=%s"},{"custom":false,"faviconUrl":"","historyId":6,"id":"cj3trn97g00053o5lye6a3670","keyword":"w","name":"Wikipedia","post":false,"removed":false,"suggestUrl":"https://en.wikipedia.org/w/api.php?action=opensearch&search=%s","url":"https://en.wikipedia.org/wiki/Special:Search?search=%s"},{"custom":true,"faviconUrl":"","historyId":7,"id":"cj3trn97g00063o5lbhkdv3av","keyword":"g","name":"Google","post":false,"removed":false,"suggestUrl":"https://www.google.com/complete/search?q=%s","url":"https://www.google.com/search?q=%s"},{"custom":false,"faviconUrl":"","historyId":8,"id":"cj4trn97g0006c0c35c091be5","keyword":"q","name":"Qwant","post":false,"removed":false,"suggestUrl":"https://api.qwant.com/api/suggest/?q=%s&client=opensearch&lang=en_gb","url":"https://www.qwant.com/?client=brz-vivaldi&q=%s"},{"custom":true,"faviconUrl":"","historyId":9,"id":"cjtyovrku001t3y5zkxj0umxf","keyword":"r","name":"Reddit","post":false,"removed":false,"suggestUrl":"","url":"https://www.reddit.com/r/%s"},{"custom":true,"faviconUrl":"","historyId":10,"id":"cjtyowbme001u3y5zuka8l8l5","keyword":"l","name":"LEO.org","post":false,"removed":false,"suggestUrl":"","url":"https://dict.leo.org/german-english/?search=%s"},{"custom":true,"faviconUrl":"","historyId":11,"id":"cjtyowypf001v3y5za3u427i4","keyword":"ig","name":"IGDB","post":false,"removed":false,"suggestUrl":"","url":"https://www.igdb.com/search?utf8=%E2%9C%93&type=1&q=%s"},{"custom":true,"faviconUrl":"","historyId":12,"id":"cjtyoxkkp001w3y5zwu7eluic","keyword":"im","name":"IMDb","post":false,"removed":false,"suggestUrl":"","url":"https://www.imdb.com/find?ref_=nv_sr_fn&q=%s&s=all"},{"custom":true,"faviconUrl":"","historyId":13,"id":"cjtyoy8yr001x3y5zlwc32yrq","keyword":"m","name":"Metacritic","post":false,"removed":false,"suggestUrl":"","url":"http://www.metacritic.com/search/game/%s/results"}],"nextHistoryId":14,"version":"3"}
-
@AltCode I think it is due to this test line
if (Object.keys(engines)[0] === 'context') {
can change to
if (Object.keys(engines)[0] === 'customOrder') {
or
if (Array.isArray(engines.engines)) {
-
@AltCode The json is valid, we have another problem here.
@tam710562 NIce ninja, just wanted to post. Yeah, that's the problem, I thought every backup starts with the word "context" but apparently not. When changing context to customOrder it works.
-
The array is array thing doesn't work for me and I don't think it would make much sense, because I want to make sure no one pastes a theme code into the search import. I thought every search backup starts with context, but apparently not โ don't know what we could do here. The logical thing would be to include "customOrder" and make it an additional possibility, but then again I don't if that's enough, could be there are other arrays out there that start with something else entirelyโฆ hmm
-
@luetage @tam710562 That seems to have worked, thanks. As an added bonus, it allowed me to add a favicon to a search engine that refused to load one.