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.


    0_1552486216104_Screenshot 2019-03-13 15.09.58.png


    • 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 = '';
        }, 5000);
    };
    
    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 {
            var engines = JSON.parse(engineCode);
        }
        catch(err) {
            _msgSearch('error');
            return;
        }
        if ('engines' in engines && 'default' in engines && 'defaultPrivate' in engines) {
            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 styleCheck = document.getElementById('searchEngines');
            if (!styleCheck) {
                const style = document.createElement('style');
                style.type = 'text/css';
                style.id = 'searchEngines';
                style.innerHTML = '#backupSearch, #restoreSearch {margin-left: 6px;}#restoreSearch{width: 130px;margin-top: 6px;}#restoreSearch::-webkit-input-placeholder {opacity: 1;color: var(--colorHighlightBg);text-align: center;}';
                document.getElementsByTagName('head')[0].appendChild(style);
            }
            const place = document.querySelector('.setting-section .setting-group.unlimited .setting-single');
            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 = {};
        }
    };
    
    setTimeout(function wait() {
        const browser = document.getElementById('browser');
        if (browser) {
            document.body.addEventListener('click', function() {
                setTimeout(searchEngines, 50);
            });
        }
        else {
            setTimeout(wait, 300);
        }
    }, 300);
    
    


  • 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.

    0_1552487160982_2019-03-13 15_20_49-Window.jpg


  • Moderator

    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.


    0_1552486216104_Screenshot 2019-03-13 15.09.58.png


    • 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.


  • Moderator

    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.



  • @luetage

    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&param1=1&param2=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.



Looks like your connection to Vivaldi Forum was lost, please wait while we try to reconnect.