browser/metro/base/content/bindings/notification.xml
author Sam Foster <sfoster@mozilla.com>
Thu, 06 Mar 2014 09:45:15 -0500
changeset 183179 f081a9d022d66ad09716899b08d3818e5878bbf9
parent 159584 35a772328e246abec98b50226213514ce9d5fe43
permissions -rw-r--r--
Bug 967793 - Download notification disappears after a download link opens in a new tab (Aurora roll-up). r=mbrubeck, a=sledru

<?xml version="1.0"?>

<!-- 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/. -->

<!DOCTYPE bindings [
<!ENTITY % notificationDTD SYSTEM "chrome://global/locale/notification.dtd">
%notificationDTD;
]>

<bindings id="notificationBindings"
          xmlns="http://www.mozilla.org/xbl"
          xmlns:xbl="http://www.mozilla.org/xbl"
          xmlns:html="http://www.w3.org/1999/xhtml"
          xmlns:xul="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul">

  <binding id="notificationbox" extends="chrome://global/content/bindings/notification.xml#notificationbox">
    <content>
      <xul:stack xbl:inherits="hidden=notificationshidden"
                 class="notificationbox-stack">
        <xul:spacer/>
        <children includes="notification"/>
      </xul:stack>
      <html:div anonid="layer1" class="notification-layer"></html:div>
      <html:div anonid="layer2" class="notification-layer"></html:div>
      <children/>
    </content>

    <implementation>
      <constructor>
        <![CDATA[
          this.addEventListener("AlertActive", this.handleEvent, true);
          this.addEventListener("AlertClose", this.handleEvent, true);
          this.setAttribute("count", 0);
        ]]>
      </constructor>
      <destructor>
        <![CDATA[
          this.removeEventListener("AlertActive", this.handleEvent, true);
          this.removeEventListener("AlertClose", this.handleEvent, true);
        ]]>
      </destructor>
      <method name="adoptNotification">
        <parameter name="aItem"/>
        <body>
          <![CDATA[
            // insert an existing notification element
            // XXX: borrows code from appendNotification in toolkit/content/widgets/notification.xml
            // if this sticks around, we'll want to refactor both to eliminate duplication

            let priority = aItem.priority;
            // check for where the notification should be inserted according to
            // priority. If two are equal, the existing one appears on top.
            let notifications = this.allNotifications;
            let insertPos = null;
            for (let n = notifications.length - 1; n >= 0; n--) {
              if (notifications[n].priority < priority)
                break;
              insertPos = notifications[n];
            }
            if (!insertPos) {
              aItem.style.position = "fixed";
              aItem.style.top = "100%";
              aItem.style.marginTop = "-15px";
              aItem.style.opacity = "0";
            }
            let label = aItem.label;
            this.insertBefore(aItem, insertPos);
            aItem.label = label;

            if (!insertPos)
              this._showNotification(aItem, true, true);

            // Fire event for accessibility APIs
            var event = document.createEvent("Events");
            event.initEvent("AlertActive", true, true);
            aItem.dispatchEvent(event);

            return aItem;
          ]]>
        </body>
      </method>
      <method name="removeNotification">
        <parameter name="aItem"/>
        <parameter name="aSkipAnimation"/>
        <body>
          <![CDATA[
            if (aItem == this.currentNotification)
              this.removeCurrentNotification(aSkipAnimation);
            else if (aItem != this._closedNotification)
              this._removeNotificationElement(aItem);

            // Fire notification closed event.
            let event = new Event('AlertClose');
            event.notification = aItem;
            this.dispatchEvent(event);

            return aItem;
          ]]>
        </body>
      </method>
      <method name="handleEvent">
        <parameter name="aEvent"/>
        <body>
          <![CDATA[
            switch (aEvent.type) {
              case "AlertActive":
              case "AlertClose":
                this.setAttribute("count", this.allNotifications.length);
                break;
            }
          ]]>
        </body>
      </method>
    </implementation>
  </binding>

  <binding id="notification" role="xul:alert" extends="chrome://global/content/bindings/notification.xml#notification">
    <implementation>
      <property name="_messageContainer" onget="return document.getAnonymousElementByAttribute(this, 'anonid', 'messageText');"/>
      <property name="label">
        <getter><![CDATA[
          if (this._messageContainer.childElementCount) {
            // return a document fragment when our label is a complex value containing elements
            // by cloning childNodes into a document fragment, the returned value
            // is *not* live and will survive unbinding of this notification
            let frag = this.ownerDocument.createDocumentFragment();
            let containerNode = this._messageContainer;
            for(let cnode of containerNode.childNodes) {
              frag.appendChild(cnode.cloneNode(true));
            }
            return frag;
          } else {
            return String.trim(this._messageContainer.textContent) ||
                   this.getAttribute("label");
          }
        ]]></getter>
        <setter><![CDATA[
            // accept a string or node (e.g. document fragment, element or text node) as label value
            if (val && "object" == typeof val && ('nodeType' in val)) {
              let containerNode = this._messageContainer;
              let cnode;
              while((cnode = containerNode.firstChild)) {
                cnode.remove();
              }
              if (val.ownerDocument !== this.ownerDocument) {
                val = this.ownerDocument.importNode(val, true);
              }
              return containerNode.appendChild(val);
            } else {
              return (this._messageContainer.textContent = val);
            }
        ]]></setter>
      </property>
    </implementation>
  </binding>
</bindings>