| OLD | NEW | 
|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be | 
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. | 
| 4 | 4 | 
| 5 'use strict'; | 5 'use strict'; | 
| 6 | 6 | 
| 7 /** | 7 /** | 
| 8  * Called from the main frame when unloading. | 8  * Called from the main frame when unloading. | 
| 9  * @return {string?} User-visible message on null if it is OK to close. | 9  * @return {string?} User-visible message on null if it is OK to close. | 
| 10  */ | 10  */ | 
| (...skipping 25 matching lines...) Expand all  Loading... | 
| 36  *      the system. | 36  *      the system. | 
| 37  * @class | 37  * @class | 
| 38  * @constructor | 38  * @constructor | 
| 39  */ | 39  */ | 
| 40 function Gallery(context, volumeManager) { | 40 function Gallery(context, volumeManager) { | 
| 41   this.container_ = document.querySelector('.gallery'); | 41   this.container_ = document.querySelector('.gallery'); | 
| 42   this.document_ = document; | 42   this.document_ = document; | 
| 43   this.context_ = context; | 43   this.context_ = context; | 
| 44   this.metadataCache_ = context.metadataCache; | 44   this.metadataCache_ = context.metadataCache; | 
| 45   this.volumeManager_ = volumeManager; | 45   this.volumeManager_ = volumeManager; | 
|  | 46   this.selectedEntry_ = null; | 
| 46 | 47 | 
| 47   this.dataModel_ = new cr.ui.ArrayDataModel([]); | 48   this.dataModel_ = new cr.ui.ArrayDataModel([]); | 
| 48   this.selectionModel_ = new cr.ui.ListSelectionModel(); | 49   this.selectionModel_ = new cr.ui.ListSelectionModel(); | 
| 49   this.displayStringFunction_ = context.displayStringFunction; | 50   this.displayStringFunction_ = context.displayStringFunction; | 
| 50 | 51 | 
| 51   this.initDom_(); | 52   this.initDom_(); | 
| 52   this.initListeners_(); | 53   this.initListeners_(); | 
| 53 } | 54 } | 
| 54 | 55 | 
| 55 /** | 56 /** | 
| 56  * Gallery extends cr.EventTarget. | 57  * Gallery extends cr.EventTarget. | 
| 57  */ | 58  */ | 
| 58 Gallery.prototype.__proto__ = cr.EventTarget.prototype; | 59 Gallery.prototype.__proto__ = cr.EventTarget.prototype; | 
| 59 | 60 | 
| 60 /** | 61 /** | 
| 61  * Create and initialize a Gallery object based on a context. | 62  * Creates and initializes a Gallery object based on a context. | 
| 62  * | 63  * | 
| 63  * @param {Object} context Gallery context. | 64  * @param {Object} context Gallery context. | 
| 64  * @param {VolumeManagerWrapper} volumeManager VolumeManager of the system. | 65  * @param {VolumeManagerWrapper} volumeManager VolumeManager of the system. | 
| 65  * @param {Array.<string>} urls Array of urls. | 66  * @param {Array.<Entry>} entries Array of entries. | 
| 66  * @param {Array.<string>} selectedUrls Array of selected urls. | 67  * @param {Array.<Entry>} selectedEntries Array of selected entries. | 
| 67  */ | 68  */ | 
| 68 Gallery.open = function(context, volumeManager, urls, selectedUrls) { | 69 Gallery.open = function(context, volumeManager, entries, selectedEntries) { | 
| 69   Gallery.instance = new Gallery(context, volumeManager); | 70   Gallery.instance = new Gallery(context, volumeManager); | 
| 70   Gallery.instance.load(urls, selectedUrls); | 71   Gallery.instance.load(entries, selectedEntries); | 
| 71 }; | 72 }; | 
| 72 | 73 | 
| 73 /** | 74 /** | 
| 74  * Tools fade-out timeout im milliseconds. | 75  * Tools fade-out timeout im milliseconds. | 
| 75  * @const | 76  * @const | 
| 76  * @type {number} | 77  * @type {number} | 
| 77  */ | 78  */ | 
| 78 Gallery.FADE_TIMEOUT = 3000; | 79 Gallery.FADE_TIMEOUT = 3000; | 
| 79 | 80 | 
| 80 /** | 81 /** | 
| (...skipping 12 matching lines...) Expand all  Loading... | 
| 93 Gallery.MOSAIC_BACKGROUND_INIT_DELAY = 1000; | 94 Gallery.MOSAIC_BACKGROUND_INIT_DELAY = 1000; | 
| 94 | 95 | 
| 95 /** | 96 /** | 
| 96  * Types of metadata Gallery uses (to query the metadata cache). | 97  * Types of metadata Gallery uses (to query the metadata cache). | 
| 97  * @const | 98  * @const | 
| 98  * @type {string} | 99  * @type {string} | 
| 99  */ | 100  */ | 
| 100 Gallery.METADATA_TYPE = 'thumbnail|filesystem|media|streaming|drive'; | 101 Gallery.METADATA_TYPE = 'thumbnail|filesystem|media|streaming|drive'; | 
| 101 | 102 | 
| 102 /** | 103 /** | 
| 103  * Initialize listeners. | 104  * Initializes listeners. | 
| 104  * @private | 105  * @private | 
| 105  */ | 106  */ | 
| 106 Gallery.prototype.initListeners_ = function() { | 107 Gallery.prototype.initListeners_ = function() { | 
| 107   this.document_.oncontextmenu = function(e) { e.preventDefault(); }; | 108   this.document_.oncontextmenu = function(e) { e.preventDefault(); }; | 
| 108   this.keyDownBound_ = this.onKeyDown_.bind(this); | 109   this.keyDownBound_ = this.onKeyDown_.bind(this); | 
| 109   this.document_.body.addEventListener('keydown', this.keyDownBound_); | 110   this.document_.body.addEventListener('keydown', this.keyDownBound_); | 
| 110 | 111 | 
| 111   this.inactivityWatcher_ = new MouseInactivityWatcher( | 112   this.inactivityWatcher_ = new MouseInactivityWatcher( | 
| 112       this.container_, Gallery.FADE_TIMEOUT, this.hasActiveTool.bind(this)); | 113       this.container_, Gallery.FADE_TIMEOUT, this.hasActiveTool.bind(this)); | 
| 113 | 114 | 
| (...skipping 10 matching lines...) Expand all  Loading... | 
| 124   this.volumeManager_.addEventListener('externally-unmounted', | 125   this.volumeManager_.addEventListener('externally-unmounted', | 
| 125       this.onExternallyUnmounted_.bind(this)); | 126       this.onExternallyUnmounted_.bind(this)); | 
| 126 }; | 127 }; | 
| 127 | 128 | 
| 128 /** | 129 /** | 
| 129  * Closes gallery when a volume containing the selected item is unmounted. | 130  * Closes gallery when a volume containing the selected item is unmounted. | 
| 130  * @param {Event} event The unmount event. | 131  * @param {Event} event The unmount event. | 
| 131  * @private | 132  * @private | 
| 132  */ | 133  */ | 
| 133 Gallery.prototype.onExternallyUnmounted_ = function(event) { | 134 Gallery.prototype.onExternallyUnmounted_ = function(event) { | 
| 134   if (!this.selectedItemFilesystemPath_) | 135   if (!this.selectedEntry_) | 
| 135     return; | 136     return; | 
| 136   if (this.selectedItemFilesystemPath_.indexOf(event.mountPath) == 0) | 137 | 
|  | 138   if (this.volumeManager_.getVolumeInfo(this.selectedEntry_) === | 
|  | 139       event.volumeInfo) { | 
| 137     this.onBack_(); | 140     this.onBack_(); | 
|  | 141   } | 
| 138 }; | 142 }; | 
| 139 | 143 | 
| 140 /** | 144 /** | 
| 141  * Beforeunload handler. | 145  * Beforeunload handler. | 
| 142  * @return {string?} User-visible message on null if it is OK to close. | 146  * @return {string?} User-visible message on null if it is OK to close. | 
| 143  */ | 147  */ | 
| 144 Gallery.prototype.onBeforeUnload = function() { | 148 Gallery.prototype.onBeforeUnload = function() { | 
| 145   return this.slideMode_.onBeforeUnload(); | 149   return this.slideMode_.onBeforeUnload(); | 
| 146 }; | 150 }; | 
| 147 | 151 | 
| 148 /** | 152 /** | 
| 149  * Unload the Gallery. | 153  * Unloads the Gallery. | 
| 150  * @param {boolean} exiting True if the app is exiting. | 154  * @param {boolean} exiting True if the app is exiting. | 
| 151  */ | 155  */ | 
| 152 Gallery.prototype.onUnload = function(exiting) { | 156 Gallery.prototype.onUnload = function(exiting) { | 
| 153   if (!this.context_.searchResults) { | 157   if (!this.context_.searchResults) { | 
| 154     this.metadataCache_.removeObserver(this.thumbnailObserverId_); | 158     this.metadataCache_.removeObserver(this.thumbnailObserverId_); | 
| 155   } | 159   } | 
| 156   this.slideMode_.onUnload(exiting); | 160   this.slideMode_.onUnload(exiting); | 
| 157 }; | 161 }; | 
| 158 | 162 | 
| 159 /** | 163 /** | 
| (...skipping 100 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 260  * @return {HTMLElement} Newly created button. | 264  * @return {HTMLElement} Newly created button. | 
| 261  * @private | 265  * @private | 
| 262  */ | 266  */ | 
| 263 Gallery.prototype.createToolbarButton_ = function(className, title) { | 267 Gallery.prototype.createToolbarButton_ = function(className, title) { | 
| 264   var button = util.createChild(this.toolbar_, className, 'button'); | 268   var button = util.createChild(this.toolbar_, className, 'button'); | 
| 265   button.title = this.displayStringFunction_(title); | 269   button.title = this.displayStringFunction_(title); | 
| 266   return button; | 270   return button; | 
| 267 }; | 271 }; | 
| 268 | 272 | 
| 269 /** | 273 /** | 
| 270  * Load the content. | 274  * Loads the content. | 
| 271  * | 275  * | 
| 272  * @param {Array.<string>} urls Array of urls. | 276  * @param {Array.<Entry>} entries Array of entries. | 
| 273  * @param {Array.<string>} selectedUrls Array of selected urls. | 277  * @param {Array.<Entry>} selectedEntries Array of selected entries. Must be a | 
|  | 278  *     subset of {@code entries}. | 
| 274  */ | 279  */ | 
| 275 Gallery.prototype.load = function(urls, selectedUrls) { | 280 Gallery.prototype.load = function(entries, selectedEntries) { | 
| 276   var items = []; | 281   var items = []; | 
| 277   for (var index = 0; index < urls.length; ++index) { | 282   for (var index = 0; index < entries.length; ++index) { | 
| 278     items.push(new Gallery.Item(urls[index])); | 283     items.push(new Gallery.Item(entries[index])); | 
| 279   } | 284   } | 
| 280   this.dataModel_.push.apply(this.dataModel_, items); | 285   this.dataModel_.push.apply(this.dataModel_, items); | 
| 281 | 286 | 
| 282   this.selectionModel_.adjustLength(this.dataModel_.length); | 287   this.selectionModel_.adjustLength(this.dataModel_.length); | 
| 283 | 288 | 
| 284   for (var i = 0; i != selectedUrls.length; i++) { | 289   for (var i = 0; i !== selectedEntries.length; i++) { | 
| 285     var selectedIndex = urls.indexOf(selectedUrls[i]); | 290     var selectedIndex = entries.indexOf(selectedEntries[i]); | 
| 286     if (selectedIndex >= 0) | 291     if (selectedIndex >= 0) | 
| 287       this.selectionModel_.setIndexSelected(selectedIndex, true); | 292       this.selectionModel_.setIndexSelected(selectedIndex, true); | 
| 288     else | 293     else | 
| 289       console.error('Cannot select ' + selectedUrls[i]); | 294       console.error('Cannot select ' + selectedEntries[i]); | 
| 290   } | 295   } | 
| 291 | 296 | 
| 292   if (this.selectionModel_.selectedIndexes.length == 0) | 297   if (this.selectionModel_.selectedIndexes.length === 0) | 
| 293     this.onSelection_(); | 298     this.onSelection_(); | 
| 294 | 299 | 
| 295   var mosaic = this.mosaicMode_ && this.mosaicMode_.getMosaic(); | 300   var mosaic = this.mosaicMode_ && this.mosaicMode_.getMosaic(); | 
| 296 | 301 | 
| 297   // Mosaic view should show up if most of the selected files are images. | 302   // Mosaic view should show up if most of the selected files are images. | 
| 298   var imagesCount = 0; | 303   var imagesCount = 0; | 
| 299   for (var i = 0; i != selectedUrls.length; i++) { | 304   for (var i = 0; i !== selectedEntries.length; i++) { | 
| 300     if (FileType.getMediaType(selectedUrls[i]) == 'image') | 305     if (FileType.getMediaType(selectedEntries[i]) === 'image') | 
| 301       imagesCount++; | 306       imagesCount++; | 
| 302   } | 307   } | 
| 303   var mostlyImages = imagesCount > (selectedUrls.length / 2.0); | 308   var mostlyImages = imagesCount > (selectedEntries.length / 2.0); | 
| 304 | 309 | 
| 305   var forcedMosaic = (this.context_.pageState && | 310   var forcedMosaic = (this.context_.pageState && | 
| 306        this.context_.pageState.gallery == 'mosaic'); | 311        this.context_.pageState.gallery === 'mosaic'); | 
| 307 | 312 | 
| 308   var showMosaic = (mostlyImages && selectedUrls.length > 1) || forcedMosaic; | 313   var showMosaic = (mostlyImages && selectedEntries.length > 1) || forcedMosaic; | 
| 309   if (mosaic && showMosaic) { | 314   if (mosaic && showMosaic) { | 
| 310     this.setCurrentMode_(this.mosaicMode_); | 315     this.setCurrentMode_(this.mosaicMode_); | 
| 311     mosaic.init(); | 316     mosaic.init(); | 
| 312     mosaic.show(); | 317     mosaic.show(); | 
| 313     this.inactivityWatcher_.check();  // Show the toolbar. | 318     this.inactivityWatcher_.check();  // Show the toolbar. | 
| 314     cr.dispatchSimpleEvent(this, 'loaded'); | 319     cr.dispatchSimpleEvent(this, 'loaded'); | 
| 315   } else { | 320   } else { | 
| 316     this.setCurrentMode_(this.slideMode_); | 321     this.setCurrentMode_(this.slideMode_); | 
| 317     var maybeLoadMosaic = function() { | 322     var maybeLoadMosaic = function() { | 
| 318       if (mosaic) | 323       if (mosaic) | 
| 319         mosaic.init(); | 324         mosaic.init(); | 
| 320       cr.dispatchSimpleEvent(this, 'loaded'); | 325       cr.dispatchSimpleEvent(this, 'loaded'); | 
| 321     }.bind(this); | 326     }.bind(this); | 
| 322     /* TODO: consider nice blow-up animation for the first image */ | 327     /* TODO: consider nice blow-up animation for the first image */ | 
| 323     this.slideMode_.enter(null, function() { | 328     this.slideMode_.enter(null, function() { | 
| 324         // Flash the toolbar briefly to show it is there. | 329         // Flash the toolbar briefly to show it is there. | 
| 325         this.inactivityWatcher_.kick(Gallery.FIRST_FADE_TIMEOUT); | 330         this.inactivityWatcher_.kick(Gallery.FIRST_FADE_TIMEOUT); | 
| 326       }.bind(this), | 331       }.bind(this), | 
| 327       maybeLoadMosaic); | 332       maybeLoadMosaic); | 
| 328   } | 333   } | 
| 329 }; | 334 }; | 
| 330 | 335 | 
| 331 /** | 336 /** | 
| 332  * Close the Gallery and go to Files.app. | 337  * Closes the Gallery and go to Files.app. | 
| 333  * @private | 338  * @private | 
| 334  */ | 339  */ | 
| 335 Gallery.prototype.back_ = function() { | 340 Gallery.prototype.back_ = function() { | 
| 336   if (util.isFullScreen(this.context_.appWindow)) { | 341   if (util.isFullScreen(this.context_.appWindow)) { | 
| 337     util.toggleFullScreen(this.context_.appWindow, | 342     util.toggleFullScreen(this.context_.appWindow, | 
| 338                           false);  // Leave the full screen mode. | 343                           false);  // Leave the full screen mode. | 
| 339   } | 344   } | 
| 340   this.context_.onBack(this.getSelectedUrls()); | 345   this.context_.onBack(this.getSelectedEntries()); | 
| 341 }; | 346 }; | 
| 342 | 347 | 
| 343 /** | 348 /** | 
| 344  * Handle user's 'Back' action (Escape or a click on the X icon). | 349  * Handles user's 'Back' action (Escape or a click on the X icon). | 
| 345  * @private | 350  * @private | 
| 346  */ | 351  */ | 
| 347 Gallery.prototype.onBack_ = function() { | 352 Gallery.prototype.onBack_ = function() { | 
| 348   this.executeWhenReady(this.back_.bind(this)); | 353   this.executeWhenReady(this.back_.bind(this)); | 
| 349 }; | 354 }; | 
| 350 | 355 | 
| 351 /** | 356 /** | 
| 352  * Handle user's 'Close' action. | 357  * Handles user's 'Close' action. | 
| 353  * @private | 358  * @private | 
| 354  */ | 359  */ | 
| 355 Gallery.prototype.onClose_ = function() { | 360 Gallery.prototype.onClose_ = function() { | 
| 356   this.executeWhenReady(this.context_.onClose); | 361   this.executeWhenReady(this.context_.onClose); | 
| 357 }; | 362 }; | 
| 358 | 363 | 
| 359 /** | 364 /** | 
| 360  * Handle user's 'Maximize' action (Escape or a click on the X icon). | 365  * Handles user's 'Maximize' action (Escape or a click on the X icon). | 
| 361  * @private | 366  * @private | 
| 362  */ | 367  */ | 
| 363 Gallery.prototype.onMaximize_ = function() { | 368 Gallery.prototype.onMaximize_ = function() { | 
| 364   this.executeWhenReady(this.context_.onMaximize); | 369   this.executeWhenReady(this.context_.onMaximize); | 
| 365 }; | 370 }; | 
| 366 | 371 | 
| 367 /** | 372 /** | 
| 368  * Execute a function when the editor is done with the modifications. | 373  * Executes a function when the editor is done with the modifications. | 
| 369  * @param {function} callback Function to execute. | 374  * @param {function} callback Function to execute. | 
| 370  */ | 375  */ | 
| 371 Gallery.prototype.executeWhenReady = function(callback) { | 376 Gallery.prototype.executeWhenReady = function(callback) { | 
| 372   this.currentMode_.executeWhenReady(callback); | 377   this.currentMode_.executeWhenReady(callback); | 
| 373 }; | 378 }; | 
| 374 | 379 | 
| 375 /** | 380 /** | 
| 376  * @return {Object} File browser private API. | 381  * @return {Object} File browser private API. | 
| 377  */ | 382  */ | 
| 378 Gallery.getFileBrowserPrivate = function() { | 383 Gallery.getFileBrowserPrivate = function() { | 
| (...skipping 12 matching lines...) Expand all  Loading... | 
| 391 * External user action event handler. | 396 * External user action event handler. | 
| 392 * @private | 397 * @private | 
| 393 */ | 398 */ | 
| 394 Gallery.prototype.onUserAction_ = function() { | 399 Gallery.prototype.onUserAction_ = function() { | 
| 395   this.closeShareMenu_(); | 400   this.closeShareMenu_(); | 
| 396   // Show the toolbar and hide it after the default timeout. | 401   // Show the toolbar and hide it after the default timeout. | 
| 397   this.inactivityWatcher_.kick(); | 402   this.inactivityWatcher_.kick(); | 
| 398 }; | 403 }; | 
| 399 | 404 | 
| 400 /** | 405 /** | 
| 401  * Set the current mode, update the UI. | 406  * Sets the current mode, update the UI. | 
| 402  * @param {Object} mode Current mode. | 407  * @param {Object} mode Current mode. | 
| 403  * @private | 408  * @private | 
| 404  */ | 409  */ | 
| 405 Gallery.prototype.setCurrentMode_ = function(mode) { | 410 Gallery.prototype.setCurrentMode_ = function(mode) { | 
| 406   if (mode != this.slideMode_ && mode != this.mosaicMode_) | 411   if (mode !== this.slideMode_ && mode !== this.mosaicMode_) | 
| 407     console.error('Invalid Gallery mode'); | 412     console.error('Invalid Gallery mode'); | 
| 408 | 413 | 
| 409   this.currentMode_ = mode; | 414   this.currentMode_ = mode; | 
| 410   this.container_.setAttribute('mode', this.currentMode_.getName()); | 415   this.container_.setAttribute('mode', this.currentMode_.getName()); | 
| 411   this.updateSelectionAndState_(); | 416   this.updateSelectionAndState_(); | 
| 412   this.updateButtons_(); | 417   this.updateButtons_(); | 
| 413 }; | 418 }; | 
| 414 | 419 | 
| 415 /** | 420 /** | 
| 416  * Mode toggle event handler. | 421  * Mode toggle event handler. | 
| (...skipping 16 matching lines...) Expand all  Loading... | 
| 433   var onModeChanged = function() { | 438   var onModeChanged = function() { | 
| 434     this.changingMode_ = false; | 439     this.changingMode_ = false; | 
| 435     if (opt_callback) opt_callback(); | 440     if (opt_callback) opt_callback(); | 
| 436   }.bind(this); | 441   }.bind(this); | 
| 437 | 442 | 
| 438   var tileIndex = Math.max(0, this.selectionModel_.selectedIndex); | 443   var tileIndex = Math.max(0, this.selectionModel_.selectedIndex); | 
| 439 | 444 | 
| 440   var mosaic = this.mosaicMode_.getMosaic(); | 445   var mosaic = this.mosaicMode_.getMosaic(); | 
| 441   var tileRect = mosaic.getTileRect(tileIndex); | 446   var tileRect = mosaic.getTileRect(tileIndex); | 
| 442 | 447 | 
| 443   if (this.currentMode_ == this.slideMode_) { | 448   if (this.currentMode_ === this.slideMode_) { | 
| 444     this.setCurrentMode_(this.mosaicMode_); | 449     this.setCurrentMode_(this.mosaicMode_); | 
| 445     mosaic.transform( | 450     mosaic.transform( | 
| 446         tileRect, this.slideMode_.getSelectedImageRect(), true /* instant */); | 451         tileRect, this.slideMode_.getSelectedImageRect(), true /* instant */); | 
| 447     this.slideMode_.leave(tileRect, | 452     this.slideMode_.leave(tileRect, | 
| 448         function() { | 453         function() { | 
| 449           // Animate back to normal position. | 454           // Animate back to normal position. | 
| 450           mosaic.transform(); | 455           mosaic.transform(); | 
| 451           mosaic.show(); | 456           mosaic.show(); | 
| 452           onModeChanged(); | 457           onModeChanged(); | 
| 453         }.bind(this)); | 458         }.bind(this)); | 
| (...skipping 24 matching lines...) Expand all  Loading... | 
| 478   /* TODO(dgozman): Implement Undo delete, Remove the confirmation dialog. */ | 483   /* TODO(dgozman): Implement Undo delete, Remove the confirmation dialog. */ | 
| 479 | 484 | 
| 480   var itemsToRemove = this.getSelectedItems(); | 485   var itemsToRemove = this.getSelectedItems(); | 
| 481   var plural = itemsToRemove.length > 1; | 486   var plural = itemsToRemove.length > 1; | 
| 482   var param = plural ? itemsToRemove.length : itemsToRemove[0].getFileName(); | 487   var param = plural ? itemsToRemove.length : itemsToRemove[0].getFileName(); | 
| 483 | 488 | 
| 484   function deleteNext() { | 489   function deleteNext() { | 
| 485     if (!itemsToRemove.length) | 490     if (!itemsToRemove.length) | 
| 486       return;  // All deleted. | 491       return;  // All deleted. | 
| 487 | 492 | 
| 488     var url = itemsToRemove.pop().getUrl(); | 493     // TODO(hirono): Use fileOperationManager. | 
| 489     webkitResolveLocalFileSystemURL(url, | 494     var entry = itemsToRemove.pop().getEntry(); | 
| 490         function(entry) { | 495     entry.remove(deleteNext, function() { | 
| 491           entry.remove(deleteNext, | 496       util.flog('Error deleting: ' + entry.fullPath, deleteNext); | 
| 492               util.flog('Error deleting ' + url, deleteNext)); | 497     }); | 
| 493         }, |  | 
| 494         util.flog('Error resolving ' + url, deleteNext)); |  | 
| 495   } | 498   } | 
| 496 | 499 | 
| 497   // Prevent the Gallery from handling Esc and Enter. | 500   // Prevent the Gallery from handling Esc and Enter. | 
| 498   this.document_.body.removeEventListener('keydown', this.keyDownBound_); | 501   this.document_.body.removeEventListener('keydown', this.keyDownBound_); | 
| 499   var restoreListener = function() { | 502   var restoreListener = function() { | 
| 500     this.document_.body.addEventListener('keydown', this.keyDownBound_); | 503     this.document_.body.addEventListener('keydown', this.keyDownBound_); | 
| 501   }.bind(this); | 504   }.bind(this); | 
| 502 | 505 | 
| 503   cr.ui.dialogs.BaseDialog.OK_LABEL = this.displayStringFunction_( | 506   cr.ui.dialogs.BaseDialog.OK_LABEL = this.displayStringFunction_( | 
| 504       'GALLERY_OK_LABEL'); | 507       'GALLERY_OK_LABEL'); | 
| (...skipping 21 matching lines...) Expand all  Loading... | 
| 526 | 529 | 
| 527 /** | 530 /** | 
| 528  * @return {Array.<Gallery.Item>} Current selection. | 531  * @return {Array.<Gallery.Item>} Current selection. | 
| 529  */ | 532  */ | 
| 530 Gallery.prototype.getSelectedItems = function() { | 533 Gallery.prototype.getSelectedItems = function() { | 
| 531   return this.selectionModel_.selectedIndexes.map( | 534   return this.selectionModel_.selectedIndexes.map( | 
| 532       this.dataModel_.item.bind(this.dataModel_)); | 535       this.dataModel_.item.bind(this.dataModel_)); | 
| 533 }; | 536 }; | 
| 534 | 537 | 
| 535 /** | 538 /** | 
| 536  * @return {Array.<string>} Array of currently selected urls. | 539  * @return {Array.<Entry>} Array of currently selected entries. | 
| 537  */ | 540  */ | 
| 538 Gallery.prototype.getSelectedUrls = function() { | 541 Gallery.prototype.getSelectedEntries = function() { | 
| 539   return this.selectionModel_.selectedIndexes.map(function(index) { | 542   return this.selectionModel_.selectedIndexes.map(function(index) { | 
| 540     return this.dataModel_.item(index).getUrl(); | 543     return this.dataModel_.item(index).getEntry(); | 
| 541   }.bind(this)); | 544   }.bind(this)); | 
| 542 }; | 545 }; | 
| 543 | 546 | 
| 544 /** | 547 /** | 
| 545  * @return {Gallery.Item} Current single selection. | 548  * @return {Gallery.Item} Current single selection. | 
| 546  */ | 549  */ | 
| 547 Gallery.prototype.getSingleSelectedItem = function() { | 550 Gallery.prototype.getSingleSelectedItem = function() { | 
| 548   var items = this.getSelectedItems(); | 551   var items = this.getSelectedItems(); | 
| 549   if (items.length > 1) | 552   if (items.length > 1) | 
| 550     throw new Error('Unexpected multiple selection'); | 553     throw new Error('Unexpected multiple selection'); | 
| (...skipping 17 matching lines...) Expand all  Loading... | 
| 568   this.selectionModel_.adjustLength(this.dataModel_.length); | 571   this.selectionModel_.adjustLength(this.dataModel_.length); | 
| 569 }; | 572 }; | 
| 570 | 573 | 
| 571 /** | 574 /** | 
| 572  * Content change event handler. | 575  * Content change event handler. | 
| 573  * @param {Event} event Event. | 576  * @param {Event} event Event. | 
| 574  * @private | 577  * @private | 
| 575 */ | 578 */ | 
| 576 Gallery.prototype.onContentChange_ = function(event) { | 579 Gallery.prototype.onContentChange_ = function(event) { | 
| 577   var index = this.dataModel_.indexOf(event.item); | 580   var index = this.dataModel_.indexOf(event.item); | 
| 578   if (index != this.selectionModel_.selectedIndex) | 581   if (index !== this.selectionModel_.selectedIndex) | 
| 579     console.error('Content changed for unselected item'); | 582     console.error('Content changed for unselected item'); | 
| 580   this.updateSelectionAndState_(); | 583   this.updateSelectionAndState_(); | 
| 581 }; | 584 }; | 
| 582 | 585 | 
| 583 /** | 586 /** | 
| 584  * Keydown handler. | 587  * Keydown handler. | 
| 585  * | 588  * | 
| 586  * @param {Event} event Event. | 589  * @param {Event} event Event. | 
| 587  * @private | 590  * @private | 
| 588  */ | 591  */ | 
| (...skipping 27 matching lines...) Expand all  Loading... | 
| 616     case 'U+007F':  // Delete | 619     case 'U+007F':  // Delete | 
| 617     case 'Shift-U+0033':  // Shift+'3' (Delete key might be missing). | 620     case 'Shift-U+0033':  // Shift+'3' (Delete key might be missing). | 
| 618       this.delete_(); | 621       this.delete_(); | 
| 619       break; | 622       break; | 
| 620   } | 623   } | 
| 621 }; | 624 }; | 
| 622 | 625 | 
| 623 // Name box and rename support. | 626 // Name box and rename support. | 
| 624 | 627 | 
| 625 /** | 628 /** | 
| 626  * Update the UI related to the selected item and the persistent state. | 629  * Updates the UI related to the selected item and the persistent state. | 
| 627  * | 630  * | 
| 628  * @private | 631  * @private | 
| 629  */ | 632  */ | 
| 630 Gallery.prototype.updateSelectionAndState_ = function() { | 633 Gallery.prototype.updateSelectionAndState_ = function() { | 
| 631   var path; | 634   var path; | 
| 632   var displayName = ''; | 635   var displayName = ''; | 
| 633 | 636 | 
| 634   var selectedItems = this.getSelectedItems(); | 637   var selectedItems = this.getSelectedItems(); | 
| 635   if (selectedItems.length == 1) { | 638   if (selectedItems.length === 1) { | 
| 636     var item = selectedItems[0]; | 639     var item = selectedItems[0]; | 
| 637     path = util.extractFilePath(item.getUrl()); | 640     var entry = item.getEntry(); | 
| 638     var fullName = item.getFileName(); | 641     window.top.document.title = entry.name; | 
| 639     window.top.document.title = fullName; | 642     displayName = ImageUtil.getDisplayNameFromName(entry.name); | 
| 640     displayName = ImageUtil.getFileNameFromFullName(fullName); |  | 
| 641   } else if (selectedItems.length > 1 && this.context_.curDirEntry) { | 643   } else if (selectedItems.length > 1 && this.context_.curDirEntry) { | 
| 642     // If the Gallery was opened on search results the search query will not be | 644     // If the Gallery was opened on search results the search query will not be | 
| 643     // recorded in the app state and the relaunch will just open the gallery | 645     // recorded in the app state and the relaunch will just open the gallery | 
| 644     // in the curDirEntry directory. | 646     // in the curDirEntry directory. | 
| 645     path = this.context_.curDirEntry.fullPath; | 647     path = this.context_.curDirEntry.fullPath; | 
| 646     window.top.document.title = this.context_.curDirEntry.name; | 648     window.top.document.title = this.context_.curDirEntry.name; | 
| 647     displayName = | 649     displayName = | 
| 648         this.displayStringFunction_('GALLERY_ITEMS_SELECTED', | 650         this.displayStringFunction_('GALLERY_ITEMS_SELECTED', | 
| 649                                     selectedItems.length); | 651                                     selectedItems.length); | 
| 650   } | 652   } | 
| 651 | 653 | 
| 652   window.top.util.updateAppState(path, | 654   window.top.util.updateAppState(path, | 
| 653       {gallery: (this.currentMode_ == this.mosaicMode_ ? 'mosaic' : 'slide')}); | 655       {gallery: (this.currentMode_ === this.mosaicMode_ ? 'mosaic' : 'slide')}); | 
| 654 | 656 | 
| 655   // We can't rename files in readonly directory. | 657   // We can't rename files in readonly directory. | 
| 656   // We can only rename a single file. | 658   // We can only rename a single file. | 
| 657   this.filenameEdit_.disabled = selectedItems.length != 1 || | 659   this.filenameEdit_.disabled = selectedItems.length !== 1 || | 
| 658                                 this.context_.readonlyDirName; | 660                                 this.context_.readonlyDirName; | 
| 659 | 661 | 
| 660   this.filenameEdit_.value = displayName; | 662   this.filenameEdit_.value = displayName; | 
| 661 | 663 | 
| 662   // Resolve real filesystem path of the current file. | 664   // Resolve real filesystem path of the current file. | 
| 663   if (this.selectionModel_.selectedIndexes.length) { | 665   if (this.selectionModel_.selectedIndexes.length) { | 
| 664     var selectedIndex = this.selectionModel_.selectedIndex; | 666     var selectedIndex = this.selectionModel_.selectedIndex; | 
| 665     var selectedItem = | 667     var selectedItem = | 
| 666         this.dataModel_.item(this.selectionModel_.selectedIndex); | 668         this.dataModel_.item(this.selectionModel_.selectedIndex); | 
| 667 | 669     this.selectedEntry_ = selectedItem.getEntry(); | 
| 668     this.selectedItemFilesystemPath_ = null; |  | 
| 669     webkitResolveLocalFileSystemURL(selectedItem.getUrl(), |  | 
| 670       function(entry) { |  | 
| 671         if (this.selectionModel_.selectedIndex != selectedIndex) |  | 
| 672           return; |  | 
| 673         this.selectedItemFilesystemPath_ = entry.fullPath; |  | 
| 674       }.bind(this)); |  | 
| 675   } | 670   } | 
| 676 }; | 671 }; | 
| 677 | 672 | 
| 678 /** | 673 /** | 
| 679  * Click event handler on filename edit box | 674  * Click event handler on filename edit box | 
| 680  * @private | 675  * @private | 
| 681  */ | 676  */ | 
| 682 Gallery.prototype.onFilenameFocus_ = function() { | 677 Gallery.prototype.onFilenameFocus_ = function() { | 
| 683   ImageUtil.setAttribute(this.filenameSpacer_, 'renaming', true); | 678   ImageUtil.setAttribute(this.filenameSpacer_, 'renaming', true); | 
| 684   this.filenameEdit_.originalValue = this.filenameEdit_.value; | 679   this.filenameEdit_.originalValue = this.filenameEdit_.value; | 
| 685   setTimeout(this.filenameEdit_.select.bind(this.filenameEdit_), 0); | 680   setTimeout(this.filenameEdit_.select.bind(this.filenameEdit_), 0); | 
| 686   this.onUserAction_(); | 681   this.onUserAction_(); | 
| 687 }; | 682 }; | 
| 688 | 683 | 
| 689 /** | 684 /** | 
| 690  * Blur event handler on filename edit box. | 685  * Blur event handler on filename edit box. | 
| 691  * | 686  * | 
| 692  * @param {Event} event Blur event. | 687  * @param {Event} event Blur event. | 
| 693  * @return {boolean} if default action should be prevented. | 688  * @return {boolean} if default action should be prevented. | 
| 694  * @private | 689  * @private | 
| 695  */ | 690  */ | 
| 696 Gallery.prototype.onFilenameEditBlur_ = function(event) { | 691 Gallery.prototype.onFilenameEditBlur_ = function(event) { | 
| 697   if (this.filenameEdit_.value && this.filenameEdit_.value[0] == '.') { | 692   if (this.filenameEdit_.value && this.filenameEdit_.value[0] === '.') { | 
| 698     this.prompt_.show('GALLERY_FILE_HIDDEN_NAME', 5000); | 693     this.prompt_.show('GALLERY_FILE_HIDDEN_NAME', 5000); | 
| 699     this.filenameEdit_.focus(); | 694     this.filenameEdit_.focus(); | 
| 700     event.stopPropagation(); | 695     event.stopPropagation(); | 
| 701     event.preventDefault(); | 696     event.preventDefault(); | 
| 702     return false; | 697     return false; | 
| 703   } | 698   } | 
| 704 | 699 | 
| 705   var item = this.getSingleSelectedItem(); | 700   var item = this.getSingleSelectedItem(); | 
| 706   var oldUrl = item.getUrl(); | 701   var oldEntry = item.getEntry(); | 
| 707 | 702 | 
| 708   var onFileExists = function() { | 703   var onFileExists = function() { | 
| 709     this.prompt_.show('GALLERY_FILE_EXISTS', 3000); | 704     this.prompt_.show('GALLERY_FILE_EXISTS', 3000); | 
| 710     this.filenameEdit_.value = name; | 705     this.filenameEdit_.value = name; | 
| 711     this.filenameEdit_.focus(); | 706     this.filenameEdit_.focus(); | 
| 712   }.bind(this); | 707   }.bind(this); | 
| 713 | 708 | 
| 714   var onSuccess = function() { | 709   var onSuccess = function() { | 
| 715     var e = new Event('content'); | 710     var event = new Event('content'); | 
| 716     e.item = item; | 711     event.item = item; | 
| 717     e.oldUrl = oldUrl; | 712     event.oldEntry = oldEntry; | 
| 718     e.metadata = null;  // Metadata unchanged. | 713     event.metadata = null;  // Metadata unchanged. | 
| 719     this.dataModel_.dispatchEvent(e); | 714     this.dataModel_.dispatchEvent(event); | 
| 720   }.bind(this); | 715   }.bind(this); | 
| 721 | 716 | 
| 722   if (this.filenameEdit_.value) { | 717   if (this.filenameEdit_.value) { | 
| 723     this.getSingleSelectedItem().rename( | 718     this.getSingleSelectedItem().rename( | 
| 724         this.filenameEdit_.value, onSuccess, onFileExists); | 719         this.filenameEdit_.value, onSuccess, onFileExists); | 
| 725   } | 720   } | 
| 726 | 721 | 
| 727   ImageUtil.setAttribute(this.filenameSpacer_, 'renaming', false); | 722   ImageUtil.setAttribute(this.filenameSpacer_, 'renaming', false); | 
| 728   this.onUserAction_(); | 723   this.onUserAction_(); | 
| 729 }; | 724 }; | 
| (...skipping 60 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 790   if (!this.shareButton_.hasAttribute('disabled')) | 785   if (!this.shareButton_.hasAttribute('disabled')) | 
| 791     this.shareMenu_.hidden = !this.shareMenu_.hidden; | 786     this.shareMenu_.hidden = !this.shareMenu_.hidden; | 
| 792   this.inactivityWatcher_.check(); | 787   this.inactivityWatcher_.check(); | 
| 793 }; | 788 }; | 
| 794 | 789 | 
| 795 /** | 790 /** | 
| 796  * Updates available actions list based on the currently selected urls. | 791  * Updates available actions list based on the currently selected urls. | 
| 797  * @private. | 792  * @private. | 
| 798  */ | 793  */ | 
| 799 Gallery.prototype.updateShareMenu_ = function() { | 794 Gallery.prototype.updateShareMenu_ = function() { | 
| 800   var urls = this.getSelectedUrls(); | 795   var entries = this.getSelectedEntries(); | 
| 801 | 796 | 
| 802   function isShareAction(task) { | 797   function isShareAction(task) { | 
| 803     var taskParts = task.taskId.split('|'); | 798     var taskParts = task.taskId.split('|'); | 
| 804     return taskParts[0] != chrome.runtime.id; | 799     return taskParts[0] !== chrome.runtime.id; | 
| 805   } | 800   } | 
| 806 | 801 | 
| 807   var api = Gallery.getFileBrowserPrivate(); | 802   var api = Gallery.getFileBrowserPrivate(); | 
| 808   var mimeTypes = [];  // TODO(kaznacheev) Collect mime types properly. | 803   var mimeTypes = [];  // TODO(kaznacheev) Collect mime types properly. | 
| 809 | 804 | 
| 810   var createShareMenu = function(tasks) { | 805   var createShareMenu = function(tasks) { | 
| 811     var wasHidden = this.shareMenu_.hidden; | 806     var wasHidden = this.shareMenu_.hidden; | 
| 812     this.shareMenu_.hidden = true; | 807     this.shareMenu_.hidden = true; | 
| 813     var items = this.shareMenu_.querySelectorAll('.item'); | 808     var items = this.shareMenu_.querySelectorAll('.item'); | 
| 814     for (var i = 0; i != items.length; i++) { | 809     for (var i = 0; i !== items.length; i++) { | 
| 815       items[i].parentNode.removeChild(items[i]); | 810       items[i].parentNode.removeChild(items[i]); | 
| 816     } | 811     } | 
| 817 | 812 | 
| 818     for (var t = 0; t != tasks.length; t++) { | 813     for (var t = 0; t !== tasks.length; t++) { | 
| 819       var task = tasks[t]; | 814       var task = tasks[t]; | 
| 820       if (!isShareAction(task)) continue; | 815       if (!isShareAction(task)) continue; | 
| 821 | 816 | 
| 822       var item = util.createChild(this.shareMenu_, 'item'); | 817       var item = util.createChild(this.shareMenu_, 'item'); | 
| 823       item.textContent = task.title; | 818       item.textContent = task.title; | 
| 824       item.style.backgroundImage = 'url(' + task.iconUrl + ')'; | 819       item.style.backgroundImage = 'url(' + task.iconUrl + ')'; | 
| 825       item.addEventListener('click', function(taskId) { | 820       item.addEventListener('click', function(taskId) { | 
| 826         this.toggleShare_();  // Hide the menu. | 821         this.toggleShare_();  // Hide the menu. | 
| 827         this.executeWhenReady(api.executeTask.bind(api, taskId, urls)); | 822         this.executeWhenReady(api.executeTask.bind(api, taskId, entries)); | 
| 828       }.bind(this, task.taskId)); | 823       }.bind(this, task.taskId)); | 
| 829     } | 824     } | 
| 830 | 825 | 
| 831     var empty = this.shareMenu_.querySelector('.item') == null; | 826     var empty = this.shareMenu_.querySelector('.item') === null; | 
| 832     ImageUtil.setAttribute(this.shareButton_, 'disabled', empty); | 827     ImageUtil.setAttribute(this.shareButton_, 'disabled', empty); | 
| 833     this.shareMenu_.hidden = wasHidden || empty; | 828     this.shareMenu_.hidden = wasHidden || empty; | 
| 834   }.bind(this); | 829   }.bind(this); | 
| 835 | 830 | 
| 836   // Create or update the share menu with a list of sharing tasks and show | 831   // Create or update the share menu with a list of sharing tasks and show | 
| 837   // or hide the share button. | 832   // or hide the share button. | 
| 838   if (!urls.length) | 833   // TODO(mtomasz): Pass Entries directly, instead of URLs. | 
|  | 834   if (!entries.length) | 
| 839     createShareMenu([]);  // Empty list of tasks, since there is no selection. | 835     createShareMenu([]);  // Empty list of tasks, since there is no selection. | 
| 840   else | 836   else | 
| 841     api.getFileTasks(urls, mimeTypes, createShareMenu); | 837     api.getFileTasks(util.entriesToURLs(entries), mimeTypes, createShareMenu); | 
| 842 }; | 838 }; | 
| 843 | 839 | 
| 844 /** | 840 /** | 
| 845  * Updates thumbnails. | 841  * Updates thumbnails. | 
| 846  * @private | 842  * @private | 
| 847  */ | 843  */ | 
| 848 Gallery.prototype.updateThumbnails_ = function() { | 844 Gallery.prototype.updateThumbnails_ = function() { | 
| 849   if (this.currentMode_ == this.slideMode_) | 845   if (this.currentMode_ === this.slideMode_) | 
| 850     this.slideMode_.updateThumbnails(); | 846     this.slideMode_.updateThumbnails(); | 
| 851 | 847 | 
| 852   if (this.mosaicMode_) { | 848   if (this.mosaicMode_) { | 
| 853     var mosaic = this.mosaicMode_.getMosaic(); | 849     var mosaic = this.mosaicMode_.getMosaic(); | 
| 854     if (mosaic.isInitialized()) | 850     if (mosaic.isInitialized()) | 
| 855       mosaic.reload(); | 851       mosaic.reload(); | 
| 856   } | 852   } | 
| 857 }; | 853 }; | 
| 858 | 854 | 
| 859 /** | 855 /** | 
| 860  * Updates buttons. | 856  * Updates buttons. | 
| 861  * @private | 857  * @private | 
| 862  */ | 858  */ | 
| 863 Gallery.prototype.updateButtons_ = function() { | 859 Gallery.prototype.updateButtons_ = function() { | 
| 864   if (this.modeButton_) { | 860   if (this.modeButton_) { | 
| 865     var oppositeMode = | 861     var oppositeMode = | 
| 866         this.currentMode_ == this.slideMode_ ? this.mosaicMode_ : | 862         this.currentMode_ === this.slideMode_ ? this.mosaicMode_ : | 
| 867                                                this.slideMode_; | 863                                                 this.slideMode_; | 
| 868     this.modeButton_.title = | 864     this.modeButton_.title = | 
| 869         this.displayStringFunction_(oppositeMode.getTitle()); | 865         this.displayStringFunction_(oppositeMode.getTitle()); | 
| 870   } | 866   } | 
| 871 }; | 867 }; | 
| OLD | NEW | 
|---|