mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-25 05:41:12 +00:00
8b4ee660aa
MozReview-Commit-ID: CmwdMx2xBbn
117 lines
3.7 KiB
JavaScript
117 lines
3.7 KiB
JavaScript
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
"use strict";
|
|
|
|
const {classes: Cc, interfaces: Ci, utils: Cu, results: Cr} = Components;
|
|
|
|
let { getChromeWindow } = Cu.import("resource:///modules/syncedtabs/util.js", {});
|
|
|
|
let log = Cu.import("resource://gre/modules/Log.jsm", {})
|
|
.Log.repository.getLogger("Sync.RemoteTabs");
|
|
|
|
this.EXPORTED_SYMBOLS = [
|
|
"SyncedTabsDeckView"
|
|
];
|
|
|
|
/**
|
|
* SyncedTabsDeckView
|
|
*
|
|
* Instances of SyncedTabsDeckView render DOM nodes from a given state.
|
|
* No state is kept internaly and the DOM will completely
|
|
* rerender unless the state flags `isUpdatable`, which helps
|
|
* make small changes without the overhead of a full rerender.
|
|
*/
|
|
const SyncedTabsDeckView = function (window, tabListComponent, props) {
|
|
this.props = props;
|
|
|
|
this._window = window;
|
|
this._doc = window.document;
|
|
|
|
this._tabListComponent = tabListComponent;
|
|
this._deckTemplate = this._doc.getElementById("deck-template");
|
|
this.container = this._doc.createElement("div");
|
|
};
|
|
|
|
SyncedTabsDeckView.prototype = {
|
|
render(state) {
|
|
if (state.isUpdatable) {
|
|
this.update(state);
|
|
} else {
|
|
this.create(state);
|
|
}
|
|
},
|
|
|
|
create(state) {
|
|
let deck = this._doc.importNode(this._deckTemplate.content, true).firstElementChild;
|
|
this._clearChilden();
|
|
|
|
let tabListWrapper = this._doc.createElement("div");
|
|
tabListWrapper.className = "tabs-container sync-state";
|
|
this._tabListComponent.init();
|
|
tabListWrapper.appendChild(this._tabListComponent.container);
|
|
deck.appendChild(tabListWrapper);
|
|
this.container.appendChild(deck);
|
|
|
|
this._generateDevicePromo();
|
|
|
|
this._attachListeners();
|
|
this.update(state);
|
|
},
|
|
|
|
_getBrowserBundle() {
|
|
return getChromeWindow(this._window).document.getElementById("bundle_browser");
|
|
},
|
|
|
|
_generateDevicePromo() {
|
|
let bundle = this._getBrowserBundle();
|
|
let formatArgs = ["android", "ios"].map(os => {
|
|
let link = this._doc.createElement("a");
|
|
link.textContent = bundle.getString(`appMenuRemoteTabs.mobilePromo.${os}`);
|
|
link.className = `${os}-link text-link`;
|
|
link.setAttribute("href", "#");
|
|
return link.outerHTML;
|
|
});
|
|
// Put it all together...
|
|
let contents = bundle.getFormattedString("appMenuRemoteTabs.mobilePromo.text2", formatArgs);
|
|
this.container.querySelector(".device-promo").innerHTML = contents;
|
|
},
|
|
|
|
destroy() {
|
|
this._tabListComponent.uninit();
|
|
this.container.remove();
|
|
},
|
|
|
|
update(state) {
|
|
// Note that we may also want to update elements that are outside of the
|
|
// deck, so use the document to find the class names rather than our
|
|
// container.
|
|
for (let panel of state.panels) {
|
|
if (panel.selected) {
|
|
Array.prototype.map.call(this._doc.getElementsByClassName(panel.id),
|
|
item => item.classList.add("selected"));
|
|
} else {
|
|
Array.prototype.map.call(this._doc.getElementsByClassName(panel.id),
|
|
item => item.classList.remove("selected"));
|
|
}
|
|
}
|
|
},
|
|
|
|
_clearChilden() {
|
|
while (this.container.firstChild) {
|
|
this.container.removeChild(this.container.firstChild);
|
|
}
|
|
},
|
|
|
|
_attachListeners() {
|
|
this.container.querySelector(".android-link").addEventListener("click", this.props.onAndroidClick);
|
|
this.container.querySelector(".ios-link").addEventListener("click", this.props.oniOSClick);
|
|
let syncPrefLinks = this.container.querySelectorAll(".sync-prefs");
|
|
for (let link of syncPrefLinks) {
|
|
link.addEventListener("click", this.props.onSyncPrefClick);
|
|
}
|
|
},
|
|
};
|
|
|