// Copyright (c) 2011 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. cr.define('ntp4', function() { 'use strict'; var TilePage = ntp4.TilePage; /** * Creates a new Most Visited object for tiling. * @constructor * @extends {HTMLAnchorElement} */ function MostVisited() { var el = cr.doc.createElement('a'); el.__proto__ = MostVisited.prototype; el.initialize(); return el; } MostVisited.prototype = { __proto__: HTMLAnchorElement.prototype, initialize: function() { this.reset(); this.addEventListener('click', this.handleClick_.bind(this)); this.addEventListener('keydown', this.handleKeyDown_.bind(this)); }, /** * Clears the DOM hierarchy for this node, setting it back to the default * for a blank thumbnail. */ reset: function() { this.className = 'most-visited filler'; // TODO(estade): why do we need edit-mode-border? this.innerHTML = '
' + '
' + '
' + '
' + '
' + '
' + '
' + '
' + '' + '' + // thumbnail-shield provides a gradient fade effect. '
' + '
' + '' + '
' + '
'; this.tabIndex = -1; }, /** * Update the appearance of this tile according to |data|. * @param {Object} data A dictionary of relevant data for the page. */ updateForData: function(data) { if (data.filler) { this.reset(); return; } this.data_ = data; this.tabIndex = 0; this.classList.remove('filler'); var title = this.querySelector('.title'); title.textContent = data.title; var faviconUrl = data.faviconUrl || 'chrome://favicon/' + data.url; title.style.backgroundImage = url(faviconUrl); title.dir = data.direction; var thumbnailUrl = data.thumbnailUrl || 'chrome://thumb/' + data.url; this.querySelector('.thumbnail').style.backgroundImage = url(thumbnailUrl); this.href = data.url; this.updatePinnedState_(); }, /** * Handles a click on the tile. * @param {Event} e The click event. */ handleClick_: function(e) { var target = e.target; if (target.classList.contains('pin')) { this.togglePinned_(); e.preventDefault(); } else if (target.classList.contains('remove')) { this.blacklist_(); e.preventDefault(); } else { var index = Array.prototype.indexOf.call(this.parentNode.children, this); if (index != -1) chrome.send('metrics', ['NTP_MostVisited' + index]); } }, /** * Allow blacklisting most visited site using the keyboard. */ handleKeyDown_: function(e) { if (!IS_MAC && e.keyCode == 46 || // Del IS_MAC && e.metaKey && e.keyCode == 8) { // Cmd + Backspace this.blacklist_(); } }, /** * Changes the visual state of the page and updates the model. */ togglePinned_: function() { var data = this.data_; data.pinned = !data.pinned; if (data.pinned) { chrome.send('addPinnedURL', [ data.url, data.title, data.faviconUrl || '', data.thumbnailUrl || '', // TODO(estade): should not need to convert index to string. String(data.index) ]); } else { chrome.send('removePinnedURL', [data.url]); } this.updatePinnedState_(); }, /** * Updates the DOM for the current pinned state. */ updatePinnedState_: function() { if (this.data_.pinned) { this.classList.add('pinned'); this.querySelector('.pin').title = templateData.unpinthumbnailtooltip; } else { this.classList.remove('pinned'); this.querySelector('.pin').title = templateData.pinthumbnailtooltip; } }, /** * Permanently removes a page from Most Visited. */ blacklist_: function() { chrome.send('blacklistURLFromMostVisited', [url]); this.reset(); // TODO(estade): request a replacement site. }, /** * Set the size and position of the most visited tile. * @param {number} size The total size of |this|. * @param {number} x The x-position. * @param {number} y The y-position. * animate. */ setBounds: function(size, x, y) { this.style.width = size + 'px'; this.style.height = heightForWidth(size) + 'px'; this.style.left = x + 'px'; this.style.top = y + 'px'; }, }; var mostVisitedPageGridValues = { // The fewest tiles we will show in a row. minColCount: 2, // The most tiles we will show in a row. maxColCount: 4, // TODO(estade): Change these to real values. // The smallest a tile can be. minTileWidth: 200, // The biggest a tile can be. maxTileWidth: 240, }; TilePage.initGridValues(mostVisitedPageGridValues); /** * Calculates the height for a Most Visited tile for a given width. The size * is based on the thumbnail, which should have a 212:132 ratio (the rest of * the arithmetic accounts for padding). * @return {number} The height. */ function heightForWidth(width) { return (width - 6) * 132 / 212 + 29; } var THUMBNAIL_COUNT = 8; /** * Creates a new MostVisitedPage object. * @param {string} name The display name for the page. * @constructor * @extends {TilePage} */ function MostVisitedPage(name) { var el = new TilePage(name, mostVisitedPageGridValues); el.__proto__ = MostVisitedPage.prototype; el.initialize(); return el; } MostVisitedPage.prototype = { __proto__: TilePage.prototype, initialize: function() { this.classList.add('most-visited-page'); this.data_ = null; this.mostVisitedTiles_ = this.getElementsByClassName('most-visited'); }, /** * Create blank (filler) tiles. * @private */ createTiles_: function() { for (var i = 0; i < THUMBNAIL_COUNT; i++) { this.appendTile(new MostVisited()); } }, /** * Update the tiles after a change to |data_|. */ updateTiles_: function() { for (var i = 0; i < THUMBNAIL_COUNT; i++) { var page = this.data_[i]; page.index = i; var tile = this.mostVisitedTiles_[i]; if (i >= this.data_.length) tile.reset(); else tile.updateForData(page); } }, /** * Array of most visited data objects. * @type {Array} */ get data() { return this.data_; }, set data(data) { // The first time data is set, create the tiles. if (!this.data_) this.createTiles_(); // We append the class name with the "filler" so that we can style fillers // differently. this.data_ = data.slice(0, THUMBNAIL_COUNT); this.updateTiles_(); }, /** @inheritDoc */ heightForWidth: heightForWidth, }; return { MostVisitedPage: MostVisitedPage, }; });