Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1011)

Unified Diff: chrome/browser/resources/shared/js/parse_html_subset.js

Issue 1759007: Refactor parts of the NTP to split things into more managable chunks.... (Closed) Base URL: svn://chrome-svn/chrome/trunk/src/
Patch Set: Revert class/tag name changes in html file Created 10 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: chrome/browser/resources/shared/js/parse_html_subset.js
===================================================================
--- chrome/browser/resources/shared/js/parse_html_subset.js (revision 0)
+++ chrome/browser/resources/shared/js/parse_html_subset.js (working copy)
@@ -1,1500 +1,4 @@
-
-// 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;
-
-function mostVisitedPages(data, firstRun) {
- logEvent('received most visited pages');
-
- // We append the class name with the "filler" so that we can style fillers
- // differently.
- var maxItems = 8;
- data.length = Math.min(maxItems, data.length);
- var len = data.length;
- for (var i = len; i < maxItems; i++) {
- data[i] = {filler: true};
- }
-
- mostVisitedData = data;
- renderMostVisited(data);
-
- gotMostVisited = true;
- onDataLoaded();
-
- // Only show the first run notification if first run.
- if (firstRun) {
- 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;
- for (var i = 0; i < data.length; i++) {
- var d = data[i];
- var t = children[i];
-
- // If we have a filler continue
- var oldClassName = t.className;
- var newClassName = getThumbnailClassName(d);
- if (oldClassName != newClassName) {
- t.className = newClassName;
- }
-
- // No need to continue if this is a filler.
- if (newClassName == 'thumbnail-container filler') {
- // Make sure the user cannot tab to the filler.
- t.tabIndex = -1;
- continue;
- }
- // Allow focus.
- t.tabIndex = 1;
-
- t.href = d.url;
- t.querySelector('.pin').title = localStrings.getString(d.pinned ?
- 'unpinthumbnailtooltip' : 'pinthumbnailtooltip');
- t.querySelector('.remove').title =
- localStrings.getString('removethumbnailtooltip');
-
- // There was some concern that a malformed malicious URL could cause an XSS
- // attack but setting style.backgroundImage = 'url(javascript:...)' does
- // not execute the JavaScript in WebKit.
-
- var thumbnailUrl = d.thumbnailUrl || 'chrome://thumb/' + d.url;
- t.querySelector('.thumbnail-wrapper').style.backgroundImage =
- url(thumbnailUrl);
- var titleDiv = t.querySelector('.title > div');
- titleDiv.xtitle = titleDiv.textContent = d.title;
- var faviconUrl = d.faviconUrl || 'chrome://favicon/' + d.url;
- titleDiv.style.backgroundImage = url(faviconUrl);
- titleDiv.dir = d.direction;
- }
-}
-
-/**
- * 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 || '',
- data.thumbnailUrl || '', String(index)]);
- },
- getItem: function(el) {
- return findAncestorByClass(el, 'thumbnail-container');
- },
-
- getHref: function(el) {
- return el.href;
- },
-
- togglePinned: function(el) {
- var index = this.getThumbnailIndex(el);
- var data = mostVisitedData[index];
- data.pinned = !data.pinned;
- if (data.pinned) {
- this.addPinnedUrl_(data, index);
- } else {
- chrome.send('removePinnedURL', [data.url]);
- }
- this.updatePinnedDom_(el, data.pinned);
- },
-
- updatePinnedDom_: function(el, pinned) {
- el.querySelector('.pin').title = localStrings.getString(pinned ?
- 'unpinthumbnailtooltip' : 'pinthumbnailtooltip');
- if (pinned) {
- addClass(el, 'pinned');
- } else {
- removeClass(el, 'pinned');
- }
- },
-
- getThumbnailIndex: function(el) {
- var nodes = el.parentNode.querySelectorAll('.thumbnail-container');
- return Array.prototype.indexOf.call(nodes, el);
- },
-
- swapPosition: function(source, destination) {
- var nodes = source.parentNode.querySelectorAll('.thumbnail-container');
- var sourceIndex = this.getThumbnailIndex(source);
- var destinationIndex = this.getThumbnailIndex(destination);
- swapDomNodes(source, destination);
-
- var sourceData = mostVisitedData[sourceIndex];
- this.addPinnedUrl_(sourceData, destinationIndex);
- sourceData.pinned = true;
- this.updatePinnedDom_(source, true);
-
- var destinationData = mostVisitedData[destinationIndex];
- // Only update the destination if it was pinned before.
- if (destinationData.pinned) {
- this.addPinnedUrl_(destinationData, sourceIndex);
- }
- mostVisitedData[destinationIndex] = sourceData;
- mostVisitedData[sourceIndex] = destinationData;
- },
-
- blacklist: function(el) {
- var self = this;
- var url = this.getHref(el);
- chrome.send('blacklistURLFromMostVisited', [url]);
-
- addClass(el, 'hide');
-
- // Find the old item.
- var oldUrls = {};
- var oldIndex = -1;
- var oldItem;
- for (var i = 0; i < mostVisitedData.length; i++) {
- if (mostVisitedData[i].url == url) {
- oldItem = mostVisitedData[i];
- oldIndex = i;
- }
- oldUrls[mostVisitedData[i].url] = true;
- }
-
- // Send 'getMostVisitedPages' with a callback since we want to find the new
- // page and add that in the place of the removed page.
- chromeSend('getMostVisited', [], 'mostVisitedPages', function(data) {
- // Find new item.
- var newItem;
- for (var i = 0; i < data.length; i++) {
- if (!(data[i].url in oldUrls)) {
- newItem = data[i];
- break;
- }
- }
-
- if (!newItem) {
- // If no other page is available to replace the blacklisted item,
- // we need to reorder items s.t. all filler items are in the rightmost
- // indices.
- mostVisitedPages(data);
-
- // Replace old item with new item in the mostVisitedData array.
- } else if (oldIndex != -1) {
- mostVisitedData.splice(oldIndex, 1, newItem);
- mostVisitedPages(mostVisitedData);
- addClass(el, 'fade-in');
- }
-
- // We wrap the title in a <span class=blacklisted-title>. We pass an empty
- // string to the notifier function and use DOM to insert the real string.
- var actionText = localStrings.getString('undothumbnailremove');
-
- // Show notification and add undo callback function.
- var wasPinned = oldItem.pinned;
- showNotification('', actionText, function() {
- self.removeFromBlackList(url);
- if (wasPinned) {
- self.addPinnedUrl_(oldItem, oldIndex);
- }
- chrome.send('getMostVisited');
- });
-
- // Now change the DOM.
- var removeText = localStrings.getString('thumbnailremovednotification');
- var notifySpan = document.querySelector('#notification > span');
- notifySpan.textContent = removeText;
-
- // Focus the undo link.
- var undoLink = document.querySelector(
- '#notification > .link > [tabindex]');
- undoLink.focus();
- });
- },
-
- removeFromBlackList: function(url) {
- chrome.send('removeURLsFromMostVisitedBlacklist', [url]);
- },
-
- clearAllBlacklisted: function() {
- chrome.send('clearMostVisitedURLsBlacklist', []);
- hideNotification();
- },
-
- updateDisplayMode: function() {
- if (!this.dirty_) {
- return;
- }
- updateSimpleSection('most-visited-section', Section.THUMB);
- },
-
- dirty_: false,
-
- invalidate: function() {
- this.dirty_ = true;
- },
-
- layout: function() {
- if (!this.dirty_) {
- return;
- }
- var d0 = Date.now();
-
- var mostVisitedElement = $('most-visited');
- var thumbnails = mostVisitedElement.children;
- var hidden = !(shownSections & Section.THUMB);
-
-
- // We set overflow to hidden so that the most visited element does not
- // "leak" when we hide and show it.
- if (hidden) {
- mostVisitedElement.style.overflow = 'hidden';
- }
-
- applyMostVisitedRects();
-
- // Only set overflow to visible if the element is shown.
- if (!hidden) {
- afterTransition(function() {
- mostVisitedElement.style.overflow = '';
- });
- }
-
- this.dirty_ = false;
-
- logEvent('mostVisited.layout: ' + (Date.now() - d0));
- },
-
- getRectByIndex: function(index) {
- return getMostVisitedLayoutRects()[index];
- }
-};
-
-// 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')) {
- mostVisited.togglePinned(mostVisited.getItem(target));
- e.preventDefault();
- } else if (hasClass(target, 'remove')) {
- mostVisited.blacklist(mostVisited.getItem(target));
- e.preventDefault();
- }
-});
-
-// Allow blacklisting most visited site using the keyboard.
-$('most-visited').addEventListener('keydown', function(e) {
- if (!IS_MAC && e.keyCode == 46 || // Del
- IS_MAC && e.metaKey && e.keyCode == 8) { // Cmd + Backspace
- mostVisited.blacklist(e.target);
- }
-});
-
-$('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()) {
- applyMostVisitedRects();
- }
-}
-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 = {
- currentOverItem_: null,
- get currentOverItem() {
- return this.currentOverItem_;
- },
- set currentOverItem(item) {
- var style;
- if (item != this.currentOverItem_) {
- if (this.currentOverItem_) {
- style = this.currentOverItem_.firstElementChild.style;
- style.left = style.top = '';
- }
- this.currentOverItem_ = item;
-
- if (item) {
- // Make the drag over item move 15px towards the source. The movement is
- // done by only moving the edit-mode-border (as in the mocks) and it is
- // done with relative positioning so that the movement does not change
- // the drop target.
- var dragIndex = mostVisited.getThumbnailIndex(this.dragItem);
- var overIndex = mostVisited.getThumbnailIndex(item);
- if (dragIndex == -1 || overIndex == -1) {
- return;
- }
-
- var dragRect = mostVisited.getRectByIndex(dragIndex);
- var overRect = mostVisited.getRectByIndex(overIndex);
-
- var x = dragRect.left - overRect.left;
- var y = dragRect.top - overRect.top;
- var z = Math.sqrt(x * x + y * y);
- var z2 = 15;
- var x2 = x * z2 / z;
- var y2 = y * z2 / z;
-
- style = this.currentOverItem_.firstElementChild.style;
- style.left = x2 + 'px';
- style.top = y2 + 'px';
- }
- }
- },
- dragItem: null,
- startX: 0,
- startY: 0,
- startScreenX: 0,
- startScreenY: 0,
- dragEndTimer: null,
-
- handleDragStart: function(e) {
- var thumbnail = mostVisited.getItem(e.target);
- if (thumbnail) {
- // Don't set data since HTML5 does not allow setting the name for
- // url-list. Instead, we just rely on the dragging of link behavior.
- this.dragItem = thumbnail;
- addClass(this.dragItem, 'dragging');
- this.dragItem.style.zIndex = 2;
- e.dataTransfer.effectAllowed = 'copyLinkMove';
- }
- },
-
- handleDragEnter: function(e) {
- if (this.canDropOnElement(this.currentOverItem)) {
- e.preventDefault();
- }
- },
-
- handleDragOver: function(e) {
- var item = mostVisited.getItem(e.target);
- this.currentOverItem = item;
- if (this.canDropOnElement(item)) {
- e.preventDefault();
- e.dataTransfer.dropEffect = 'move';
- }
- },
-
- handleDragLeave: function(e) {
- var item = mostVisited.getItem(e.target);
- if (item) {
- e.preventDefault();
- }
-
- this.currentOverItem = null;
- },
-
- handleDrop: function(e) {
- var dropTarget = mostVisited.getItem(e.target);
- if (this.canDropOnElement(dropTarget)) {
- dropTarget.style.zIndex = 1;
- mostVisited.swapPosition(this.dragItem, dropTarget);
- // The timeout below is to allow WebKit to see that we turned off
- // pointer-event before moving the thumbnails so that we can get out of
- // hover mode.
- window.setTimeout(function() {
- mostVisited.invalidate();
- mostVisited.layout();
- }, 10);
- e.preventDefault();
- if (this.dragEndTimer) {
- window.clearTimeout(this.dragEndTimer);
- this.dragEndTimer = null;
- }
- afterTransition(function() {
- dropTarget.style.zIndex = '';
- });
- }
- },
-
- handleDragEnd: function(e) {
- var dragItem = this.dragItem;
- if (dragItem) {
- dragItem.style.pointerEvents = '';
- removeClass(dragItem, 'dragging');
-
- afterTransition(function() {
- // Delay resetting zIndex to let the animation finish.
- dragItem.style.zIndex = '';
- // Same for overflow.
- dragItem.parentNode.style.overflow = '';
- });
-
- mostVisited.invalidate();
- mostVisited.layout();
- this.dragItem = null;
- }
- },
-
- handleDrag: function(e) {
- // Moves the drag item making sure that it is not displayed outside the
- // browser viewport.
- var item = mostVisited.getItem(e.target);
- var rect = document.querySelector('#most-visited').getBoundingClientRect();
- item.style.pointerEvents = 'none';
-
- var x = this.startX + e.screenX - this.startScreenX;
- var y = this.startY + e.screenY - this.startScreenY;
-
- // The position of the item is relative to #most-visited so we need to
- // subtract that when calculating the allowed position.
- x = Math.max(x, -rect.left);
- x = Math.min(x, document.body.clientWidth - rect.left - item.offsetWidth -
- 2);
- // The shadow is 2px
- y = Math.max(-rect.top, y);
- y = Math.min(y, document.body.clientHeight - rect.top - item.offsetHeight -
- 2);
-
- // Override right in case of RTL.
- item.style.right = 'auto';
- item.style.left = x + 'px';
- item.style.top = y + 'px';
- item.style.zIndex = 2;
- },
-
- // We listen to mousedown to get the relative position of the cursor for dnd.
- handleMouseDown: function(e) {
- var item = mostVisited.getItem(e.target);
- if (item) {
- this.startX = item.offsetLeft;
- this.startY = item.offsetTop;
- this.startScreenX = e.screenX;
- this.startScreenY = e.screenY;
-
- // We don't want to focus the item on mousedown. However, to prevent focus
- // one has to call preventDefault but this also prevents the drag and drop
- // (sigh) so we only prevent it when the user is not doing a left mouse
- // button drag.
- if (e.button != 0) // LEFT
- e.preventDefault();
- }
- },
-
- canDropOnElement: function(el) {
- return this.dragItem && el && hasClass(el, 'thumbnail-container') &&
- !hasClass(el, 'filler');
- },
-
- init: function() {
- var el = $('most-visited');
- el.addEventListener('dragstart', bind(this.handleDragStart, this));
- el.addEventListener('dragenter', bind(this.handleDragEnter, this));
- el.addEventListener('dragover', bind(this.handleDragOver, this));
- el.addEventListener('dragleave', bind(this.handleDragLeave, this));
- el.addEventListener('drop', bind(this.handleDrop, this));
- el.addEventListener('dragend', bind(this.handleDragEnd, this));
- el.addEventListener('drag', bind(this.handleDrag, this));
- el.addEventListener('mousedown', bind(this.handleMouseDown, this));
- }
-};
-
-dnd.init();
-
-/**
* Whitelist of tag names allowed in parseHtmlSubset.
* @type {[string]}
*/
@@ -1573,43 +77,3 @@
});
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'));
-}

Powered by Google App Engine
This is Rietveld 408576698