/** @jsx React.DOM */ /* 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/. */ /* jshint newcap:false, esnext:true */ /* global loop:true, React */ var loop = loop || {}; loop.conversation = (function(OT, mozL10n) { "use strict"; var sharedViews = loop.shared.views, // aliasing translation function as __ for concision __ = mozL10n.get; /** * App router. * @type {loop.desktopRouter.DesktopConversationRouter} */ var router; var IncomingCallView = React.createClass({displayName: 'IncomingCallView', propTypes: { model: React.PropTypes.func.isRequired }, getInitialState: function() { return {showDeclineMenu: false}; }, componentDidMount: function() { window.addEventListener('click', this.clickHandler); window.addEventListener('blur', this._hideDeclineMenu); }, componentWillUnmount: function() { window.removeEventListener('click', this.clickHandler); window.removeEventListener('blur', this._hideDeclineMenu); }, clickHandler: function(e) { var target = e.target; if (!target.classList.contains('btn-chevron')) { this._hideDeclineMenu(); } }, /** * Used for adding different styles to the panel * @returns {String} Corresponds to the client platform * */ _getTargetPlatform: function() { var platform="unknown_platform"; if (navigator.platform.indexOf("Win") !== -1) { platform = "windows"; } if (navigator.platform.indexOf("Mac") !== -1) { platform = "mac"; } if (navigator.platform.indexOf("Linux") !== -1) { platform = "linux"; } return platform; }, _handleAccept: function() { this.props.model.trigger("accept"); }, _handleDecline: function() { this.props.model.trigger("decline"); }, _handleDeclineBlock: function(e) { this.props.model.trigger("declineAndBlock"); /* Prevent event propagation * stop the click from reaching parent element */ return false; }, _toggleDeclineMenu: function() { var currentState = this.state.showDeclineMenu; this.setState({showDeclineMenu: !currentState}); }, _hideDeclineMenu: function() { this.setState({showDeclineMenu: false}); }, render: function() { /* jshint ignore:start */ var btnClassAccept = "btn btn-success btn-accept"; var btnClassBlock = "btn btn-error btn-block"; var btnClassDecline = "btn btn-error btn-decline"; var conversationPanelClass = "incoming-call " + this._getTargetPlatform(); var cx = React.addons.classSet; var declineDropdownMenuClasses = cx({ "native-dropdown-menu": true, "decline-block-menu": true, "visually-hidden": !this.state.showDeclineMenu }); return ( React.DOM.div({className: conversationPanelClass}, React.DOM.h2(null, __("incoming_call")), React.DOM.div({className: "button-group incoming-call-action-group"}, React.DOM.div({className: "button-chevron-menu-group"}, React.DOM.div({className: "button-group-chevron"}, React.DOM.div({className: "button-group"}, React.DOM.button({className: btnClassDecline, onClick: this._handleDecline}, __("incoming_call_decline_button") ), React.DOM.div({className: "btn-chevron", onClick: this._toggleDeclineMenu} ) ), React.DOM.ul({className: declineDropdownMenuClasses}, React.DOM.li({className: "btn-block", onClick: this._handleDeclineBlock}, __("incoming_call_decline_and_block_button") ) ) ) ), React.DOM.button({className: btnClassAccept, onClick: this._handleAccept}, __("incoming_call_answer_button") ) ) ) ); /* jshint ignore:end */ } }); /** * Call ended view. * @type {loop.shared.views.BaseView} */ var EndedCallView = sharedViews.BaseView.extend({ template: _.template([ '
', ' ', '
' ].join("")), className: "call-ended", events: { "click button": "closeWindow" }, closeWindow: function(event) { event.preventDefault(); // XXX For now, we just close the window. window.close(); } }); /** * Conversation router. * * Required options: * - {loop.shared.models.ConversationModel} conversation Conversation model. * - {loop.shared.components.Notifier} notifier Notifier component. * * @type {loop.shared.router.BaseConversationRouter} */ var ConversationRouter = loop.desktopRouter.DesktopConversationRouter.extend({ routes: { "incoming/:version": "incoming", "call/accept": "accept", "call/decline": "decline", "call/ongoing": "conversation", "call/ended": "ended", "call/declineAndBlock": "declineAndBlock" }, /** * @override {loop.shared.router.BaseConversationRouter.startCall} */ startCall: function() { this.navigate("call/ongoing", {trigger: true}); }, /** * @override {loop.shared.router.BaseConversationRouter.endCall} */ endCall: function() { this.navigate("call/ended", {trigger: true}); }, /** * Incoming call route. * * @param {String} loopVersion The version from the push notification, set * by the router from the URL. */ incoming: function(loopVersion) { window.navigator.mozLoop.startAlerting(); this._conversation.set({loopVersion: loopVersion}); this._conversation.once("accept", function() { this.navigate("call/accept", {trigger: true}); }.bind(this)); this._conversation.once("decline", function() { this.navigate("call/decline", {trigger: true}); }.bind(this)); this._conversation.once("declineAndBlock", function() { this.navigate("call/declineAndBlock", {trigger: true}); }.bind(this)); this.loadReactComponent(loop.conversation.IncomingCallView({ model: this._conversation })); }, /** * Accepts an incoming call. */ accept: function() { window.navigator.mozLoop.stopAlerting(); this._conversation.initiate({ client: new loop.Client(), outgoing: false }); }, /** * Declines an incoming call. */ decline: function() { window.navigator.mozLoop.stopAlerting(); // XXX For now, we just close the window window.close(); }, /** * Decline and block an incoming call * @note: * - loopToken is the callUrl identifier. It gets set in the panel * after a callUrl is received */ declineAndBlock: function() { window.navigator.mozLoop.stopAlerting(); var token = navigator.mozLoop.getLoopCharPref('loopToken'); var client = new loop.Client(); client.deleteCallUrl(token, function(error) { // XXX The conversation window will be closed when this cb is triggered // figure out if there is a better way to report the error to the user console.log(error); }); window.close(); }, /** * conversation is the route when the conversation is active. The start * route should be navigated to first. */ conversation: function() { if (!this._conversation.isSessionReady()) { console.error("Error: navigated to conversation route without " + "the start route to initialise the call first"); this._notifier.errorL10n("cannot_start_call_session_not_ready"); return; } /*jshint newcap:false*/ this.loadReactComponent(sharedViews.ConversationView({ sdk: OT, model: this._conversation })); }, /** * XXX: load a view with a close button for now? */ ended: function() { this.loadView(new EndedCallView()); } }); /** * Panel initialisation. */ function init() { // Do the initial L10n setup, we do this before anything // else to ensure the L10n environment is setup correctly. mozL10n.initialize(window.navigator.mozLoop); document.title = mozL10n.get("incoming_call_title"); router = new ConversationRouter({ conversation: new loop.shared.models.ConversationModel({}, {sdk: OT}), notifier: new sharedViews.NotificationListView({el: "#messages"}) }); Backbone.history.start(); } return { ConversationRouter: ConversationRouter, EndedCallView: EndedCallView, IncomingCallView: IncomingCallView, init: init }; })(window.OT, document.mozL10n);