Index: chrome/browser/resources/ntp/most_visited.js |
=================================================================== |
--- chrome/browser/resources/ntp/most_visited.js (revision 0) |
+++ chrome/browser/resources/ntp/most_visited.js (working copy) |
@@ -1,52 +1,7 @@ |
+// Copyright (c) 2010 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
-// Helpers |
- |
-function findAncestorByClass(el, className) { |
- return findAncestor(el, function(el) { |
- return hasClass(el, className); |
- }); |
-} |
- |
-/** |
- * Return the first ancestor for which the {@code predicate} returns true. |
- * @param {Node} node The node to check. |
- * @param {function(Node) : boolean} predicate The function that tests the |
- * nodes. |
- * @return {Node} The found ancestor or null if not found. |
- */ |
-function findAncestor(node, predicate) { |
- var last = false; |
- while (node != null && !(last = predicate(node))) { |
- node = node.parentNode; |
- } |
- return last ? node : null; |
-} |
- |
-// WebKit does not have Node.prototype.swapNode |
-// https://bugs.webkit.org/show_bug.cgi?id=26525 |
-function swapDomNodes(a, b) { |
- var afterA = a.nextSibling; |
- if (afterA == b) { |
- swapDomNodes(b, a); |
- return; |
- } |
- var aParent = a.parentNode; |
- b.parentNode.replaceChild(a, b); |
- aParent.insertBefore(b, afterA); |
-} |
- |
-function bind(fn, selfObj, var_args) { |
- var boundArgs = Array.prototype.slice.call(arguments, 2); |
- return function() { |
- var args = Array.prototype.slice.call(arguments); |
- args.unshift.apply(args, boundArgs); |
- return fn.apply(selfObj, args); |
- } |
-} |
- |
-const IS_MAC = /$Mac/.test(navigator.platform); |
- |
-var loading = true; |
var mostVisitedData = []; |
var gotMostVisited = false; |
@@ -73,173 +28,12 @@ |
showFirstRunNotification(); |
} |
} |
- |
-function getAppsCallback(data) { |
- var appsSection = $('apps-section'); |
- appsSection.innerHTML = ''; |
- appsSection.style.display = data.length ? 'block' : ''; |
- |
- data.forEach(function(app) { |
- appsSection.appendChild(apps.createElement(app)); |
- }); |
-} |
- |
-var apps = { |
- /** |
- * @this {!HTMLAnchorElement} |
- */ |
- handleClick_: function() { |
- chrome.send('launchApp', [this.id]); |
- return false; |
- }, |
- |
- createElement: function(app) { |
- var a = document.createElement('a'); |
- a.xtitle = a.textContent = app['name']; |
- a.href = app['launch_url']; |
- a.id = app['id']; |
- a.onclick = apps.handleClick_; |
- a.style.backgroundImage = url(app['icon']); |
- return a; |
- } |
-}; |
- |
-var tipCache = {}; |
- |
-function tips(data) { |
- logEvent('received tips'); |
- tipCache = data; |
- renderTip(); |
-} |
- |
-function createTip(data) { |
- if (data.length) { |
- if (data[0].set_homepage_tip) { |
- var homepageButton = document.createElement('button'); |
- homepageButton.className = 'link'; |
- homepageButton.textContent = data[0].set_homepage_tip; |
- homepageButton.addEventListener('click', setAsHomePageLinkClicked); |
- return homepageButton; |
- } else { |
- try { |
- return parseHtmlSubset(data[0].tip_html_text); |
- } catch (parseErr) { |
- console.error('Error parsing tips: ' + parseErr.message); |
- } |
- } |
- } |
- // Return an empty DF in case of failure. |
- return document.createDocumentFragment(); |
-} |
- |
-function clearTipLine() { |
- var tipElement = $('tip-line'); |
- // There should always be only one tip. |
- tipElement.textContent = ''; |
- tipElement.removeEventListener('click', setAsHomePageLinkClicked); |
-} |
- |
-function renderTip() { |
- clearTipLine(); |
- var tipElement = $('tip-line'); |
- tipElement.appendChild(createTip(tipCache)); |
- fixLinkUnderlines(tipElement); |
-} |
- |
-function recentlyClosedTabs(data) { |
- logEvent('received recently closed tabs'); |
- // We need to store the recent items so we can update the layout on a resize. |
- recentItems = data; |
- renderRecentlyClosed(); |
-} |
- |
-var recentItems = []; |
- |
-function renderRecentlyClosed() { |
- // We remove all items but the header and the nav |
- var recentlyClosedElement = $('recently-closed'); |
- var headerEl = recentlyClosedElement.firstElementChild; |
- var navEl = recentlyClosedElement.lastElementChild.lastElementChild; |
- var parentEl = navEl.parentNode; |
- |
- for (var el = navEl.previousElementSibling; el; |
- el = navEl.previousElementSibling) { |
- parentEl.removeChild(el); |
- } |
- |
- // Create new items |
- recentItems.forEach(function(item) { |
- var el = createRecentItem(item); |
- parentEl.insertBefore(el, navEl); |
- }); |
- |
- layoutRecentlyClosed(); |
-} |
- |
-function createRecentItem(data) { |
- var isWindow = data.type == 'window'; |
- var el; |
- if (isWindow) { |
- el = document.createElement('span'); |
- el.className = 'item link window'; |
- el.tabItems = data.tabs; |
- el.tabIndex = 0; |
- el.textContent = formatTabsText(data.tabs.length); |
- } else { |
- el = document.createElement('a'); |
- el.className = 'item'; |
- el.href = data.url; |
- el.style.backgroundImage = url('chrome://favicon/' + data.url); |
- el.dir = data.direction; |
- el.textContent = data.title; |
- } |
- el.sessionId = data.sessionId; |
- el.xtitle = data.title; |
- var wrapperEl = document.createElement('span'); |
- wrapperEl.appendChild(el); |
- return wrapperEl; |
-} |
- |
-function onShownSections(mask) { |
- logEvent('received shown sections'); |
- if (mask != shownSections) { |
- var oldShownSections = shownSections; |
- shownSections = mask; |
- |
- // Only invalidate most visited if needed. |
- if ((mask & Section.THUMB) != (oldShownSections & Section.THUMB)) { |
- mostVisited.invalidate(); |
- } |
- |
- mostVisited.updateDisplayMode(); |
- renderRecentlyClosed(); |
- } |
-} |
- |
-function saveShownSections() { |
- chrome.send('setShownSections', [String(shownSections)]); |
-} |
- |
function getThumbnailClassName(data) { |
return 'thumbnail-container' + |
(data.pinned ? ' pinned' : '') + |
(data.filler ? ' filler' : ''); |
} |
-function url(s) { |
- // http://www.w3.org/TR/css3-values/#uris |
- // Parentheses, commas, whitespace characters, single quotes (') and double |
- // quotes (") appearing in a URI must be escaped with a backslash |
- var s2 = s.replace(/(\(|\)|\,|\s|\'|\"|\\)/g, '\\$1'); |
- // WebKit has a bug when it comes to URLs that end with \ |
- // https://bugs.webkit.org/show_bug.cgi?id=28885 |
- if (/\\\\$/.test(s2)) { |
- // Add a space to work around the WebKit bug. |
- s2 += ' '; |
- } |
- return 'url("' + s2 + '")'; |
-} |
- |
function renderMostVisited(data) { |
var parent = $('most-visited'); |
var children = parent.children; |
@@ -284,84 +78,6 @@ |
} |
} |
-/** |
- * Calls chrome.send with a callback and restores the original afterwards. |
- */ |
-function chromeSend(name, params, callbackName, callback) { |
- var old = global[callbackName]; |
- global[callbackName] = function() { |
- // restore |
- global[callbackName] = old; |
- |
- var args = Array.prototype.slice.call(arguments); |
- return callback.apply(global, args); |
- }; |
- chrome.send(name, params); |
-} |
- |
-var LayoutMode = { |
- SMALL: 1, |
- NORMAL: 2 |
-}; |
- |
-var layoutMode = useSmallGrid() ? LayoutMode.SMALL : LayoutMode.NORMAL; |
- |
-function handleWindowResize() { |
- if (window.innerWidth < 10) { |
- // We're probably a background tab, so don't do anything. |
- return; |
- } |
- |
- var oldLayoutMode = layoutMode; |
- layoutMode = useSmallGrid() ? LayoutMode.SMALL : LayoutMode.NORMAL |
- |
- if (layoutMode != oldLayoutMode){ |
- mostVisited.invalidate(); |
- mostVisited.layout(); |
- renderRecentlyClosed(); |
- } |
-} |
- |
-function showSection(section) { |
- if (!(section & shownSections)) { |
- shownSections |= section; |
- |
- switch (section) { |
- case Section.THUMB: |
- mostVisited.invalidate(); |
- mostVisited.updateDisplayMode(); |
- mostVisited.layout(); |
- break; |
- case Section.RECENT: |
- renderRecentlyClosed(); |
- break; |
- case Section.TIPS: |
- removeClass($('tip-line'), 'hidden'); |
- break; |
- } |
- } |
-} |
- |
-function hideSection(section) { |
- if (section & shownSections) { |
- shownSections &= ~section; |
- |
- switch (section) { |
- case Section.THUMB: |
- mostVisited.invalidate(); |
- mostVisited.updateDisplayMode(); |
- mostVisited.layout(); |
- break; |
- case Section.RECENT: |
- renderRecentlyClosed(); |
- break; |
- case Section.TIPS: |
- addClass($('tip-line'), 'hidden'); |
- break; |
- } |
- } |
-} |
- |
var mostVisited = { |
addPinnedUrl_: function(data, index) { |
chrome.send('addPinnedURL', [data.url, data.title, data.faviconUrl || '', |
@@ -550,484 +266,6 @@ |
} |
}; |
-// Recently closed |
- |
-function layoutRecentlyClosed() { |
- var recentShown = shownSections & Section.RECENT; |
- updateSimpleSection('recently-closed', Section.RECENT); |
- |
- if (recentShown) { |
- var recentElement = $('recently-closed'); |
- var style = recentElement.style; |
- // We cannot use clientWidth here since the width has a transition. |
- var spacing = 20; |
- var headerEl = recentElement.firstElementChild; |
- var navEl = recentElement.lastElementChild.lastElementChild; |
- var navWidth = navEl.offsetWidth; |
- // Subtract 10 for the padding |
- var availWidth = (useSmallGrid() ? 690 : 918) - navWidth - 10; |
- |
- // Now go backwards and hide as many elements as needed. |
- var elementsToHide = []; |
- for (var el = navEl.previousElementSibling; el; |
- el = el.previousElementSibling) { |
- if (el.offsetLeft + el.offsetWidth + spacing > availWidth) { |
- elementsToHide.push(el); |
- } |
- } |
- |
- elementsToHide.forEach(function(el) { |
- el.parentNode.removeChild(el); |
- }); |
- } |
-} |
- |
-/** |
- * This function is called by the backend whenever the sync status section |
- * needs to be updated to reflect recent sync state changes. The backend passes |
- * the new status information in the newMessage parameter. The state includes |
- * the following: |
- * |
- * syncsectionisvisible: true if the sync section needs to show up on the new |
- * tab page and false otherwise. |
- * title: the header for the sync status section. |
- * msg: the actual message (e.g. "Synced to foo@gmail.com"). |
- * linkisvisible: true if the link element should be visible within the sync |
- * section and false otherwise. |
- * linktext: the text to display as the link in the sync status (only used if |
- * linkisvisible is true). |
- * linkurlisset: true if an URL should be set as the href for the link and false |
- * otherwise. If this field is false, then clicking on the link |
- * will result in sending a message to the backend (see |
- * 'SyncLinkClicked'). |
- * linkurl: the URL to use as the element's href (only used if linkurlisset is |
- * true). |
- */ |
-function syncMessageChanged(newMessage) { |
- var syncStatusElement = $('sync-status'); |
- var style = syncStatusElement.style; |
- |
- // Hide the section if the message is emtpy. |
- if (!newMessage['syncsectionisvisible']) { |
- style.display = 'none'; |
- return; |
- } |
- style.display = 'block'; |
- |
- // Set the sync section background color based on the state. |
- if (newMessage.msgtype == 'error') { |
- style.backgroundColor = 'tomato'; |
- } else { |
- style.backgroundColor = ''; |
- } |
- |
- // Set the text for the header and sync message. |
- var titleElement = syncStatusElement.firstElementChild; |
- titleElement.textContent = newMessage.title; |
- var messageElement = titleElement.nextElementSibling; |
- messageElement.textContent = newMessage.msg; |
- |
- // Remove what comes after the message |
- while (messageElement.nextSibling) { |
- syncStatusElement.removeChild(messageElement.nextSibling); |
- } |
- |
- if (newMessage.linkisvisible) { |
- var el; |
- if (newMessage.linkurlisset) { |
- // Use a link |
- el = document.createElement('a'); |
- el.href = newMessage.linkurl; |
- } else { |
- el = document.createElement('button'); |
- el.className = 'link'; |
- el.addEventListener('click', syncSectionLinkClicked); |
- } |
- el.textContent = newMessage.linktext; |
- syncStatusElement.appendChild(el); |
- fixLinkUnderline(el); |
- } |
-} |
- |
-/** |
- * Invoked when the link in the sync status section is clicked. |
- */ |
-function syncSectionLinkClicked(e) { |
- chrome.send('SyncLinkClicked'); |
- e.preventDefault(); |
-} |
- |
-/** |
- * Invoked when link to start sync in the promo message is clicked, and Chrome |
- * has already been synced to an account. |
- */ |
-function syncAlreadyEnabled(message) { |
- showNotification(message.syncEnabledMessage, |
- localStrings.getString('close')); |
-} |
- |
-/** |
- * Returns the text used for a recently closed window. |
- * @param {number} numTabs Number of tabs in the window. |
- * @return {string} The text to use. |
- */ |
-function formatTabsText(numTabs) { |
- if (numTabs == 1) |
- return localStrings.getString('closedwindowsingle'); |
- return localStrings.getStringF('closedwindowmultiple', numTabs); |
-} |
- |
-/** |
- * We need both most visited and the shown sections to be considered loaded. |
- * @return {boolean} |
- */ |
-function onDataLoaded() { |
- if (gotMostVisited) { |
- mostVisited.layout(); |
- loading = false; |
- // Remove class name in a timeout so that changes done in this JS thread are |
- // not animated. |
- window.setTimeout(function() { |
- ensureSmallGridCorrect(); |
- removeClass(document.body, 'loading'); |
- }, 1); |
- } |
-} |
- |
-// Theme related |
- |
-function themeChanged() { |
- $('themecss').href = 'chrome://theme/css/newtab.css?' + Date.now(); |
- updateAttribution(); |
-} |
- |
-function updateAttribution() { |
- $('attribution-img').src = 'chrome://theme/theme_ntp_attribution?' + |
- Date.now(); |
-} |
- |
-function bookmarkBarAttached() { |
- document.documentElement.setAttribute('bookmarkbarattached', 'true'); |
-} |
- |
-function bookmarkBarDetached() { |
- document.documentElement.setAttribute('bookmarkbarattached', 'false'); |
-} |
- |
-function viewLog() { |
- var lines = []; |
- var start = log[0][1]; |
- |
- for (var i = 0; i < log.length; i++) { |
- lines.push((log[i][1] - start) + ': ' + log[i][0]); |
- } |
- |
- console.log(lines.join('\n')); |
-} |
- |
-// Updates the visibility of the menu items. |
-function updateOptionMenu() { |
- var menuItems = $('option-menu').children; |
- for (var i = 0; i < menuItems.length; i++) { |
- var item = menuItems[i]; |
- var command = item.getAttribute('command'); |
- if (command == 'show' || command == 'hide') { |
- var section = Section[item.getAttribute('section')]; |
- var visible = shownSections & section; |
- item.setAttribute('command', visible ? 'hide' : 'show'); |
- } |
- } |
-} |
- |
-// We apply the size class here so that we don't trigger layout animations |
-// onload. |
- |
-handleWindowResize(); |
- |
-var localStrings = new LocalStrings(); |
- |
-/////////////////////////////////////////////////////////////////////////////// |
-// Things we know are not needed at startup go below here |
- |
-function afterTransition(f) { |
- if (loading) { |
- // Make sure we do not use a timer during load since it slows down the UI. |
- f(); |
- } else { |
- // The duration of all transitions are .15s |
- window.setTimeout(f, 150); |
- } |
-} |
- |
-// Notification |
- |
- |
-var notificationTimeout; |
- |
-function showNotification(text, actionText, opt_f, opt_delay) { |
- var notificationElement = $('notification'); |
- var f = opt_f || function() {}; |
- var delay = opt_delay || 10000; |
- |
- function show() { |
- window.clearTimeout(notificationTimeout); |
- addClass(notificationElement, 'show'); |
- addClass(document.body, 'notification-shown'); |
- } |
- |
- function delayedHide() { |
- notificationTimeout = window.setTimeout(hideNotification, delay); |
- } |
- |
- function doAction() { |
- f(); |
- hideNotification(); |
- } |
- |
- // Remove any possible first-run trails. |
- removeClass(notification, 'first-run'); |
- |
- var actionLink = notificationElement.querySelector('.link-color'); |
- notificationElement.firstElementChild.textContent = text; |
- actionLink.textContent = actionText; |
- |
- actionLink.onclick = doAction; |
- actionLink.onkeydown = handleIfEnterKey(doAction); |
- notificationElement.onmouseover = show; |
- notificationElement.onmouseout = delayedHide; |
- actionLink.onfocus = show; |
- actionLink.onblur = delayedHide; |
- // Enable tabbing to the link now that it is shown. |
- actionLink.tabIndex = 0; |
- |
- show(); |
- delayedHide(); |
-} |
- |
-/** |
- * Hides the notifier. |
- */ |
-function hideNotification() { |
- var notificationElement = $('notification'); |
- removeClass(notificationElement, 'show'); |
- removeClass(document.body, 'notification-shown'); |
- var actionLink = notificationElement.querySelector('.link-color'); |
- // Prevent tabbing to the hidden link. |
- actionLink.tabIndex = -1; |
- // Setting tabIndex to -1 only prevents future tabbing to it. If, however, the |
- // user switches window or a tab and then moves back to this tab the element |
- // may gain focus. We therefore make sure that we blur the element so that the |
- // element focus is not restored when coming back to this window. |
- actionLink.blur(); |
-} |
- |
-function showFirstRunNotification() { |
- showNotification(localStrings.getString('firstrunnotification'), |
- localStrings.getString('closefirstrunnotification'), |
- null, 30000); |
- var notificationElement = $('notification'); |
- addClass(notification, 'first-run'); |
-} |
- |
- |
-/** |
- * This handles the option menu. |
- * @param {Element} button The button element. |
- * @param {Element} menu The menu element. |
- * @constructor |
- */ |
-function OptionMenu(button, menu) { |
- this.button = button; |
- this.menu = menu; |
- this.button.onmousedown = bind(this.handleMouseDown, this); |
- this.button.onkeydown = bind(this.handleKeyDown, this); |
- this.boundHideMenu_ = bind(this.hide, this); |
- this.boundMaybeHide_ = bind(this.maybeHide_, this); |
- this.menu.onmouseover = bind(this.handleMouseOver, this); |
- this.menu.onmouseout = bind(this.handleMouseOut, this); |
- this.menu.onmouseup = bind(this.handleMouseUp, this); |
-} |
- |
-OptionMenu.prototype = { |
- show: function() { |
- updateOptionMenu(); |
- this.positionMenu_(); |
- this.menu.style.display = 'block'; |
- addClass(this.button, 'open'); |
- this.button.focus(); |
- |
- // Listen to document and window events so that we hide the menu when the |
- // user clicks outside the menu or tabs away or the whole window is blurred. |
- document.addEventListener('focus', this.boundMaybeHide_, true); |
- document.addEventListener('mousedown', this.boundMaybeHide_, true); |
- }, |
- |
- positionMenu_: function() { |
- this.menu.style.top = this.button.getBoundingClientRect().bottom + 'px'; |
- }, |
- |
- hide: function() { |
- this.menu.style.display = 'none'; |
- removeClass(this.button, 'open'); |
- this.setSelectedIndex(-1); |
- |
- document.removeEventListener('focus', this.boundMaybeHide_, true); |
- document.removeEventListener('mousedown', this.boundMaybeHide_, true); |
- }, |
- |
- isShown: function() { |
- return this.menu.style.display == 'block'; |
- }, |
- |
- /** |
- * Callback for document mousedown and focus. It checks if the user tried to |
- * navigate to a different element on the page and if so hides the menu. |
- * @param {Event} e The mouse or focus event. |
- * @private |
- */ |
- maybeHide_: function(e) { |
- if (!this.menu.contains(e.target) && !this.button.contains(e.target)) { |
- this.hide(); |
- } |
- }, |
- |
- handleMouseDown: function(e) { |
- if (this.isShown()) { |
- this.hide(); |
- } else { |
- this.show(); |
- } |
- }, |
- |
- handleMouseOver: function(e) { |
- var el = e.target; |
- if (!el.hasAttribute('command')) { |
- this.setSelectedIndex(-1); |
- } else { |
- var index = Array.prototype.indexOf.call(this.menu.children, el); |
- this.setSelectedIndex(index); |
- } |
- }, |
- |
- handleMouseOut: function(e) { |
- this.setSelectedIndex(-1); |
- }, |
- |
- handleMouseUp: function(e) { |
- var item = this.getSelectedItem(); |
- if (item) { |
- this.executeItem(item); |
- } |
- }, |
- |
- handleKeyDown: function(e) { |
- var item = this.getSelectedItem(); |
- |
- var self = this; |
- function selectNextVisible(m) { |
- var children = self.menu.children; |
- var len = children.length; |
- var i = self.selectedIndex_; |
- if (i == -1 && m == -1) { |
- // Edge case when we need to go the last item fisrt. |
- i = 0; |
- } |
- while (true) { |
- i = (i + m + len) % len; |
- item = children[i]; |
- if (item && item.hasAttribute('command') && |
- item.style.display != 'none') { |
- break; |
- } |
- } |
- if (item) { |
- self.setSelectedIndex(i); |
- } |
- } |
- |
- switch (e.keyIdentifier) { |
- case 'Down': |
- if (!this.isShown()) { |
- this.show(); |
- } |
- selectNextVisible(1); |
- e.preventDefault(); |
- break; |
- case 'Up': |
- if (!this.isShown()) { |
- this.show(); |
- } |
- selectNextVisible(-1); |
- e.preventDefault(); |
- break; |
- case 'Esc': |
- case 'U+001B': // Maybe this is remote desktop playing a prank? |
- this.hide(); |
- break; |
- case 'Enter': |
- case 'U+0020': // Space |
- if (this.isShown()) { |
- if (item) { |
- this.executeItem(item); |
- } else { |
- this.hide(); |
- } |
- } else { |
- this.show(); |
- } |
- e.preventDefault(); |
- break; |
- } |
- }, |
- |
- selectedIndex_: -1, |
- setSelectedIndex: function(i) { |
- if (i != this.selectedIndex_) { |
- var items = this.menu.children; |
- var oldItem = items[this.selectedIndex_]; |
- if (oldItem) { |
- oldItem.removeAttribute('selected'); |
- } |
- var newItem = items[i]; |
- if (newItem) { |
- newItem.setAttribute('selected', 'selected'); |
- } |
- this.selectedIndex_ = i; |
- } |
- }, |
- |
- getSelectedItem: function() { |
- return this.menu.children[this.selectedIndex_] || null; |
- }, |
- |
- executeItem: function(item) { |
- var command = item.getAttribute('command'); |
- if (command in this.commands) { |
- this.commands[command].call(this, item); |
- } |
- |
- this.hide(); |
- } |
-}; |
- |
-var optionMenu = new OptionMenu($('option-button'), $('option-menu')); |
-optionMenu.commands = { |
- 'clear-all-blacklisted' : function() { |
- mostVisited.clearAllBlacklisted(); |
- chrome.send('getMostVisited'); |
- }, |
- 'show': function(item) { |
- var section = Section[item.getAttribute('section')]; |
- showSection(section); |
- saveShownSections(); |
- }, |
- 'hide': function(item) { |
- var section = Section[item.getAttribute('section')]; |
- hideSection(section); |
- saveShownSections(); |
- } |
-}; |
- |
$('most-visited').addEventListener('click', function(e) { |
var target = e.target; |
if (hasClass(target, 'pin')) { |
@@ -1047,199 +285,10 @@ |
} |
}); |
-$('main').addEventListener('click', function(e) { |
- if (e.target.tagName == 'H2') { |
- var p = e.target.parentNode; |
- var section = p.getAttribute('section'); |
- if (section) { |
- if (shownSections & Section[section]) |
- hideSection(Section[section]); |
- else |
- showSection(Section[section]); |
- saveShownSections(); |
- } |
- } |
-}); |
- |
-function handleIfEnterKey(f) { |
- return function(e) { |
- if (e.keyIdentifier == 'Enter') { |
- f(e); |
- } |
- }; |
-} |
- |
-function maybeReopenTab(e) { |
- var el = findAncestor(e.target, function(el) { |
- return el.sessionId !== undefined; |
- }); |
- if (el) { |
- chrome.send('reopenTab', [String(el.sessionId)]); |
- e.preventDefault(); |
- |
- // HACK(arv): After the window onblur event happens we get a mouseover event |
- // on the next item and we want to make sure that we do not show a tooltip |
- // for that. |
- window.setTimeout(function() { |
- windowTooltip.hide(); |
- }, 2 * WindowTooltip.DELAY); |
- } |
-} |
- |
-function maybeShowWindowTooltip(e) { |
- var f = function(el) { |
- return el.tabItems !== undefined; |
- }; |
- var el = findAncestor(e.target, f); |
- var relatedEl = findAncestor(e.relatedTarget, f); |
- if (el && el != relatedEl) { |
- windowTooltip.handleMouseOver(e, el, el.tabItems); |
- } |
-} |
- |
- |
-var recentlyClosedElement = $('recently-closed'); |
- |
-recentlyClosedElement.addEventListener('click', maybeReopenTab); |
-recentlyClosedElement.addEventListener('keydown', |
- handleIfEnterKey(maybeReopenTab)); |
- |
-recentlyClosedElement.addEventListener('mouseover', maybeShowWindowTooltip); |
-recentlyClosedElement.addEventListener('focus', maybeShowWindowTooltip, true); |
- |
-/** |
- * This object represents a tooltip representing a closed window. It is |
- * shown when hovering over a closed window item or when the item is focused. It |
- * gets hidden when blurred or when mousing out of the menu or the item. |
- * @param {Element} tooltipEl The element to use as the tooltip. |
- * @constructor |
- */ |
-function WindowTooltip(tooltipEl) { |
- this.tooltipEl = tooltipEl; |
- this.boundHide_ = bind(this.hide, this); |
- this.boundHandleMouseOut_ = bind(this.handleMouseOut, this); |
-} |
- |
-WindowTooltip.trackMouseMove_ = function(e) { |
- WindowTooltip.clientX = e.clientX; |
- WindowTooltip.clientY = e.clientY; |
-}; |
- |
-/** |
- * Time in ms to delay before the tooltip is shown. |
- * @type {number} |
- */ |
-WindowTooltip.DELAY = 300; |
- |
-WindowTooltip.prototype = { |
- timer: 0, |
- handleMouseOver: function(e, linkEl, tabs) { |
- this.linkEl_ = linkEl; |
- if (e.type == 'mouseover') { |
- this.linkEl_.addEventListener('mousemove', WindowTooltip.trackMouseMove_); |
- this.linkEl_.addEventListener('mouseout', this.boundHandleMouseOut_); |
- } else { // focus |
- this.linkEl_.addEventListener('blur', this.boundHide_); |
- } |
- this.timer = window.setTimeout(bind(this.show, this, e.type, linkEl, tabs), |
- WindowTooltip.DELAY); |
- }, |
- show: function(type, linkEl, tabs) { |
- window.addEventListener('blur', this.boundHide_); |
- this.linkEl_.removeEventListener('mousemove', |
- WindowTooltip.trackMouseMove_); |
- window.clearTimeout(this.timer); |
- |
- this.renderItems(tabs); |
- var rect = linkEl.getBoundingClientRect(); |
- var bodyRect = document.body.getBoundingClientRect(); |
- var rtl = document.documentElement.dir == 'rtl'; |
- |
- this.tooltipEl.style.display = 'block'; |
- var tooltipRect = this.tooltipEl.getBoundingClientRect(); |
- var x, y; |
- |
- // When focused show below, like a drop down menu. |
- if (type == 'focus') { |
- x = rtl ? |
- rect.left + bodyRect.left + rect.width - this.tooltipEl.offsetWidth : |
- rect.left + bodyRect.left; |
- y = rect.top + bodyRect.top + rect.height; |
- } else { |
- x = bodyRect.left + (rtl ? |
- WindowTooltip.clientX - this.tooltipEl.offsetWidth : |
- WindowTooltip.clientX); |
- // Offset like a tooltip |
- y = 20 + WindowTooltip.clientY + bodyRect.top; |
- } |
- |
- // We need to ensure that the tooltip is inside the window viewport. |
- x = Math.min(x, bodyRect.width - tooltipRect.width); |
- x = Math.max(x, 0); |
- y = Math.min(y, bodyRect.height - tooltipRect.height); |
- y = Math.max(y, 0); |
- |
- this.tooltipEl.style.left = x + 'px'; |
- this.tooltipEl.style.top = y + 'px'; |
- }, |
- handleMouseOut: function(e) { |
- // Don't hide when move to another item in the link. |
- var f = function(el) { |
- return el.tabItems !== undefined; |
- }; |
- var el = findAncestor(e.target, f); |
- var relatedEl = findAncestor(e.relatedTarget, f); |
- if (el && el != relatedEl) { |
- this.hide(); |
- } |
- }, |
- hide: function() { |
- window.clearTimeout(this.timer); |
- window.removeEventListener('blur', this.boundHide_); |
- this.linkEl_.removeEventListener('mousemove', |
- WindowTooltip.trackMouseMove_); |
- this.linkEl_.removeEventListener('mouseout', this.boundHandleMouseOut_); |
- this.linkEl_.removeEventListener('blur', this.boundHide_); |
- this.linkEl_ = null; |
- |
- this.tooltipEl.style.display = 'none'; |
- }, |
- renderItems: function(tabs) { |
- var tooltip = this.tooltipEl; |
- tooltip.textContent = ''; |
- |
- tabs.forEach(function(tab) { |
- var span = document.createElement('span'); |
- span.className = 'item'; |
- span.style.backgroundImage = url('chrome://favicon/' + tab.url); |
- span.dir = tab.direction; |
- span.textContent = tab.title; |
- tooltip.appendChild(span); |
- }); |
- } |
-}; |
- |
-var windowTooltip = new WindowTooltip($('window-tooltip')); |
- |
-window.addEventListener('load', bind(logEvent, global, 'Tab.NewTabOnload', |
- true)); |
window.addEventListener('load', onDataLoaded); |
window.addEventListener('resize', handleWindowResize); |
-document.addEventListener('DOMContentLoaded', |
- bind(logEvent, global, 'Tab.NewTabDOMContentLoaded', true)); |
-// Whether or not we should send the initial 'GetSyncMessage' to the backend |
-// depends on the value of the attribue 'syncispresent' which the backend sets |
-// to indicate if there is code in the backend which is capable of processing |
-// this message. This attribute is loaded by the JSTemplate and therefore we |
-// must make sure we check the attribute after the DOM is loaded. |
-document.addEventListener('DOMContentLoaded', |
- callGetSyncMessageIfSyncIsPresent); |
- |
-// Set up links and text-decoration for promotional message. |
-document.addEventListener('DOMContentLoaded', setUpPromoMessage); |
- |
// Work around for http://crbug.com/25329 |
function ensureSmallGridCorrect() { |
if (wasSmallGrid != useSmallGrid()) { |
@@ -1248,59 +297,6 @@ |
} |
document.addEventListener('DOMContentLoaded', ensureSmallGridCorrect); |
-/** |
- * The sync code is not yet built by default on all platforms so we have to |
- * make sure we don't send the initial sync message to the backend unless the |
- * backend told us that the sync code is present. |
- */ |
-function callGetSyncMessageIfSyncIsPresent() { |
- if (document.documentElement.getAttribute('syncispresent') == 'true') { |
- chrome.send('GetSyncMessage'); |
- } |
-} |
- |
-function setAsHomePageLinkClicked(e) { |
- chrome.send('setHomePage'); |
- e.preventDefault(); |
-} |
- |
-function onHomePageSet(data) { |
- showNotification(data[0], data[1]); |
- // Removes the "make this my home page" tip. |
- clearTipLine(); |
-} |
- |
-function hideAllMenus() { |
- optionMenu.hide(); |
-} |
- |
-window.addEventListener('blur', hideAllMenus); |
-window.addEventListener('keydown', function(e) { |
- if (e.keyIdentifier == 'Alt' || e.keyIdentifier == 'Meta') { |
- hideAllMenus(); |
- } |
-}, true); |
- |
-// Tooltip for elements that have text that overflows. |
-document.addEventListener('mouseover', function(e) { |
- // We don't want to do this while we are dragging because it makes things very |
- // janky |
- if (dnd.dragItem) { |
- return; |
- } |
- |
- var el = findAncestor(e.target, function(el) { |
- return el.xtitle; |
- }); |
- if (el && el.xtitle != el.title) { |
- if (el.scrollWidth > el.clientWidth) { |
- el.title = el.xtitle; |
- } else { |
- el.title = ''; |
- } |
- } |
-}); |
- |
// DnD |
var dnd = { |
@@ -1494,122 +490,3 @@ |
dnd.init(); |
-/** |
- * Whitelist of tag names allowed in parseHtmlSubset. |
- * @type {[string]} |
- */ |
-var allowedTags = ['A', 'B', 'STRONG']; |
- |
-/** |
- * Parse a very small subset of HTML. |
- * @param {string} s The string to parse. |
- * @throws {Error} In case of non supported markup. |
- * @return {DocumentFragment} A document fragment containing the DOM tree. |
- */ |
-var allowedAttributes = { |
- 'href': function(node, value) { |
- // Only allow a[href] starting with http:// and https:// |
- return node.tagName == 'A' && (value.indexOf('http://') == 0 || |
- value.indexOf('https://') == 0); |
- }, |
- 'target': function(node, value) { |
- // Allow a[target] but reset the value to "". |
- if (node.tagName != 'A') |
- return false; |
- node.setAttribute('target', ''); |
- return true; |
- } |
-} |
- |
-/** |
- * Parse a very small subset of HTML. This ensures that insecure HTML / |
- * javascript cannot be injected into the new tab page. |
- * @param {string} s The string to parse. |
- * @throws {Error} In case of non supported markup. |
- * @return {DocumentFragment} A document fragment containing the DOM tree. |
- */ |
-function parseHtmlSubset(s) { |
- function walk(n, f) { |
- f(n); |
- for (var i = 0; i < n.childNodes.length; i++) { |
- walk(n.childNodes[i], f); |
- } |
- } |
- |
- function assertElement(node) { |
- if (allowedTags.indexOf(node.tagName) == -1) |
- throw Error(node.tagName + ' is not supported'); |
- } |
- |
- function assertAttribute(attrNode, node) { |
- var n = attrNode.nodeName; |
- var v = attrNode.nodeValue; |
- if (!allowedAttributes.hasOwnProperty(n) || !allowedAttributes[n](node, v)) |
- throw Error(node.tagName + '[' + n + '="' + v + '"] is not supported'); |
- } |
- |
- var r = document.createRange(); |
- r.selectNode(document.body); |
- // This does not execute any scripts. |
- var df = r.createContextualFragment(s); |
- walk(df, function(node) { |
- switch (node.nodeType) { |
- case Node.ELEMENT_NODE: |
- assertElement(node); |
- var attrs = node.attributes; |
- for (var i = 0; i < attrs.length; i++) { |
- assertAttribute(attrs[i], node); |
- } |
- break; |
- |
- case Node.COMMENT_NODE: |
- case Node.DOCUMENT_FRAGMENT_NODE: |
- case Node.TEXT_NODE: |
- break; |
- |
- default: |
- throw Error('Node type ' + node.nodeType + ' is not supported'); |
- } |
- }); |
- return df; |
-} |
- |
-/** |
- * Makes links and buttons support a different underline color. |
- * @param {Node} node The node to search for links and buttons in. |
- */ |
-function fixLinkUnderlines(node) { |
- var elements = node.querySelectorAll('a,button'); |
- Array.prototype.forEach.call(elements, fixLinkUnderline); |
-} |
- |
-/** |
- * Wraps the content of an element in a a link-color span. |
- * @param {Element} el The element to wrap. |
- */ |
-function fixLinkUnderline(el) { |
- var span = document.createElement('span'); |
- span.className = 'link-color'; |
- while (el.hasChildNodes()) { |
- span.appendChild(el.firstChild); |
- } |
- el.appendChild(span); |
-} |
- |
-updateAttribution(); |
- |
-// Closes the promo line when close button is clicked. |
-$('promo-close').onclick = function (e) { |
- addClass($('promo-line'), 'hidden'); |
- chrome.send('stopPromoLineMessage'); |
- e.preventDefault(); |
-}; |
- |
-// Set bookmark sync button to start bookmark sync process on click; also set |
-// link underline colors correctly. |
-function setUpPromoMessage() { |
- var syncButton = document.querySelector('#promo-message button'); |
- syncButton.className = 'sync-button link'; |
- syncButton.onclick = syncSectionLinkClicked; |
- fixLinkUnderlines($('promo-message')); |
-} |