]>
git.wh0rd.org - chrome-ext/tabs-backup.git/blob - background.js
5aff375df3e747389f1a761198e7dd80d6ba882d
1 // Create a backup on first install (or if storage is wiped for some reason.
2 chrome
.storage
.local
.get(function(items
) {
4 if (items
.prefs_max_backup_items
=== undefined) {
5 chrome
.storage
.local
.set({prefs_max_backup_items
: 10});
7 if (items
.prefs_backup_timer
=== undefined) {
8 chrome
.storage
.local
.set({prefs_backup_timer
: 30});
11 // If a backup exists already, nothing to do.
12 if (items
.backups_list
) {
16 // Create a backup now
18 var formattedDate
= date_format (d
);
20 backupNow(true, formattedDate
, function({success
, backupName
, backupObj
}) {
25 function initAlarm () {
26 //console.log("initAlarm");
28 var BACKUP_ALARM_NAME
= "backup_alarm";
30 // Clear any previous alarm
31 chrome
.alarms
.clearAll();
33 chrome
.storage
.local
.get(function(items
) {
34 const timerMinutes
= items
.prefs_backup_timer
;
35 chrome
.alarms
.create(BACKUP_ALARM_NAME
, {periodInMinutes
: timerMinutes
});
41 function onAlarm (alarm
) {
43 var formattedDate
= date_format (d
);
45 console
.log("Alarm {" + alarm
+ "} fired up: " + formattedDate
);
47 // if last backup time != lastTabsEdit
48 // perform automatic backup
49 backupNow(true, formattedDate
, function({success
, backupName
, backupObj
}) {
50 // automatic backup completed
51 chrome
.runtime
.sendMessage({
52 action
: 'insertBackupItem',
53 args
: [backupName
, backupObj
, true /*insertAtBeginning*/, true /*doAnimation*/],
58 chrome
.alarms
.onAlarm
.addListener(onAlarm
);
60 function date_prependZero (val
) {
61 return val
< 10 ? "0" + val
: "" + val
;
65 function date_format (d
) {
66 var monthOneOffset
= d
.getMonth() + 1; // convert from 0-11 to 1-12
68 var formattedDate
= d
.getFullYear() + "-" + date_prependZero(monthOneOffset
) + "-" + date_prependZero(d
.getDate())
69 + " " + date_prependZero(d
.getHours()) + ":" + date_prependZero(d
.getMinutes()) + ":" + date_prependZero(d
.getSeconds());
75 function backupNowManual (callbackDone
) {
77 var formattedDate
= date_format (d
);
79 backupNow(false, formattedDate
, callbackDone
);
84 function deleteOldestBackup () {
85 chrome
.storage
.local
.get(function(items
) {
86 if(!items
.backups_list
) {
90 var backupsList
= items
.backups_list
;
91 var numItemsToDelete
= backupsList
.length
- items
.prefs_max_backup_items
;
92 if (numItemsToDelete
> 0) {
94 var loopFunc = function () {
97 var deletedBackupName
= backupsList
[i
-1];
98 chrome
.runtime
.sendMessage({
99 action
: 'removeBackupItemDiv',
100 args
: [deletedBackupName
],
105 if (i
>= numItemsToDelete
) {
109 deleteBackup(backupsList
[i
]).then(loopFunc
);
118 //var isCreatingBackup = false;
120 function backupNow(isAutomatic
, backupName
, callbackDone
) {
121 console
.log("backupNow - isAutomatic: " + isAutomatic
+ " name: " + backupName
);
122 /*if (isCreatingBackup === true) {
123 console.log("backupNow - already running..skipping..");
127 //isCreatingBackup = true;
129 /*if (!confirm("Perform a full backup? All windows and their tabs will be saved!")) {
135 isAutomatic
: isAutomatic
,
139 chrome
.windows
.getAll({populate
: true}, function (window_list
) {
142 for(var i
=0;i
<window_list
.length
;i
++) {
143 var window
= window_list
[i
];
145 //console.log ("Window " + i);
152 height
: window
.height
,
156 var windowTabs
= window
.tabs
;
158 // If it's a single window sittig at the new tab page, don't bother
159 // saving it. This is a nice shortcut when things crash as it will
160 // only show a single window.
161 if (windowTabs
.length
== 1) {
162 const tab
= windowTabs
[0];
163 if (tab
.url
== 'chrome://newtab/')
167 for (var j
= 0; j
< windowTabs
.length
; j
++) {
168 var tab
= windowTabs
[j
];
170 //console.log("==> Tab " + j + " (" + tab.index + "): " + tabUrl);
172 // Ignore windows that we can't/shouldn't backup.
173 if (tab
.url
.startsWith('chrome-untrusted://')) {
180 highlighted
: tab
.highlighted
,
184 // Add tab to tabs arrays
185 bkpWindow
.tabs
.push(bkpTab
);
188 if (bkpWindow
.tabs
.length
) {
189 totNumTabs
+= bkpWindow
.tabs
.length
;
191 fullBackup
.windows
.push(bkpWindow
);
198 fullBackup
.totNumTabs
= totNumTabs
;
200 var storageSetValues
= {};
201 storageSetValues
[backupName
] = fullBackup
;
204 chrome
.storage
.local
.set(storageSetValues
, function () {
205 if (chrome
.runtime
.lastError
) {
206 //isCreatingBackup = false;
207 // TODO change icon to error..
208 //alert ("Error: " + chrome.runtime.lastError.message);
209 updateBrowserActionIcon (1);
211 callbackDone({success
: false});
213 console
.log("backup saved");
214 //alert("Backup saved successfully!");
216 chrome
.storage
.local
.get(function(items
) {
217 var backupsList
= [];
218 if(items
.backups_list
) {
219 backupsList
= items
.backups_list
;
222 console
.log("Updating 'backups_list' - cur. size: " + backupsList
.length
);
224 backupsList
.push(backupName
);
226 chrome
.storage
.local
.set({"backups_list": backupsList
}, function () {
227 //isCreatingBackup = false;
229 if (chrome
.runtime
.lastError
) {
230 console
.log ("Error saving backups_list: " + chrome
.runtime
.lastError
.message
);
231 updateBrowserActionIcon (1);
232 callbackDone({success
: false});
234 console
.log("Backups list saved successfully");
236 updateBrowserActionIcon (0);
240 backupObj
: fullBackup
,
243 if (backupsList
.length
> items
.prefs_max_backup_items
) {
244 deleteOldestBackup();
258 function updateBrowserActionIcon (status
) {
262 icon
= "icon_ok.png";
265 icon
= "icon_error.png";
269 chrome
.action
.setIcon({path
: icon
});
272 async
function deleteBackup(backupName
) {
273 console
.log("Deleting backup " + backupName
);
275 await chrome
.storage
.local
.remove(backupName
);
276 //console.log("=> Deleted backup " + backupName);
278 const items
= await chrome
.storage
.local
.get("backups_list");
279 //console.log("==> got backups_list " + backupName);
281 if (!items
.backups_list
) {
285 var backupsList
= items
.backups_list
;
287 var index
= backupsList
.indexOf(backupName
);
289 backupsList
.splice(index
, 1);
292 //console.log("===> Updating backups_list (removing " + backupName + ")");
294 await chrome
.storage
.local
.set({"backups_list": backupsList
});
295 //console.log("===> Updated backups_list (removed " + backupName + ")");
298 function restoreNow(backupName
) {
299 console
.log("restoreNow");
301 chrome
.storage
.local
.get(backupName
, function(items
) {
302 if(!items
[backupName
]) {
303 alert("No Backup found");
307 /*if (!confirm("Restore full backup?")) {
311 if (confirm("Would you like to close all existing windows first?")) {
312 chrome.windows.getAll({populate : false}, function (window_list) {
313 for(var i=0;i<window_list.length;i++) {
314 var window = window_list[i];
315 chrome.windows.remove(window.id);
321 var fullBackup
= items
[backupName
];
323 for(var i
=0;i
<fullBackup
.windows
.length
;i
++) {
324 const window
= fullBackup
.windows
[i
];
326 //console.log ("Window " + i);
330 const windowTabs
= window
.tabs
;
331 for (let j
= 0; j
< windowTabs
.length
; j
++) {
332 const tab
= windowTabs
[j
];
333 const tabUrl
= tab
.url
;
334 urlsToOpen
.push(tabUrl
);
337 const windowProperties
= {
343 height
: window
.height
,
346 // Create a new Window
347 chrome
.windows
.create(windowProperties
, function(createdWindow
) {
348 // Chrome errors if the dimensions are set on non-normal windows.
349 // So we create the window first with the right settings, then
350 // update the window state.
351 if (window
.state
!= 'normal') {
352 chrome
.windows
.update(createdWindow
.id
, {state
: window
.state
});
355 chrome
.windows
.get(createdWindow
.id
, {populate
: true}, ({tabs
}) => {
356 for (let tabi
= 0; tabi
< windowTabs
.length
; ++tabi
) {
357 const oldtab
= windowTabs
[tabi
];
358 const newtab
= tabs
[tabi
];
359 chrome
.tabs
.update(newtab
.id
, {
360 highlighted
: oldtab
.highlighted
,
361 pinned
: oldtab
.pinned
,
363 if (!oldtab
.highlighted
) {
364 // If we discard a tab too fast, Chrome will completely
365 // throw it away. Wait until it's in a stable enough
366 // state for us to discard it.
368 const checktab
= (id
) => {
369 if (retryCount
-- < 0)
371 chrome
.tabs
.get(id
, (tab
) => {
373 setTimeout(() => checktab(id
), 500);
375 chrome
.tabs
.discard(id
);
389 * Callback from other pages (like the popup).
391 chrome
.runtime
.onMessage
.addListener(function(request
, sender
, sendResponse
) {
392 console
.log(`Got message from ${sender.id}: action=${request.action}`, request
);
394 let asyncResponse
= false;
395 switch (request
?.action
) {
401 restoreNow(...request
.args
);
405 deleteBackup(...request
.args
).then(sendResponse
);
406 asyncResponse
= true;
409 case 'backupNowManual':
410 backupNowManual(sendResponse
);
411 asyncResponse
= true;
414 return asyncResponse
;