summaryrefslogtreecommitdiffstats
path: root/toolkit/jetpack/sdk/stylesheet/style.js
blob: 7ec0787e11377f677c61d9b6bcd836b777cbcdd4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
/* 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";

module.metadata = {
  "stability": "experimental"
};

const { Cc, Ci } = require("chrome");
const { Class } = require("../core/heritage");
const { URL, isLocalURL } = require('../url');
const events = require("../system/events");
const { loadSheet, removeSheet, isTypeValid } = require("./utils");
const { isString } = require("../lang/type");
const { attachTo, detachFrom } = require("../content/mod");
const { data } = require('../self');

const { freeze, create } = Object;

function Style({ source, uri, type }) {
  source = source == null ? null : freeze([].concat(source));
  uri = uri == null ? null : freeze([].concat(uri));
  type = type == null ? "author" : type;

  if (source && !source.every(isString))
    throw new Error('Style.source must be a string or an array of strings.');

  if (uri && !uri.every(isLocalURL))
    throw new Error('Style.uri must be a local URL or an array of local URLs');

  if (type && !isTypeValid(type))
    throw new Error('Style.type must be "agent", "user" or "author"');

  return freeze(create(Style.prototype, {
    "source": { value: source, enumerable: true },
    "uri": { value: uri, enumerable: true },
    "type": { value: type, enumerable: true }
  }));
};

exports.Style = Style;

attachTo.define(Style, function (style, window) {
  if (style.uri) {
    for (let uri of style.uri)
      loadSheet(window, data.url(uri), style.type);
  }

  if (style.source) {
    let uri = "data:text/css;charset=utf-8,";

    uri += encodeURIComponent(style.source.join(""));

    loadSheet(window, uri, style.type);
  }
});

detachFrom.define(Style, function (style, window) {
  if (style.uri)
    for (let uri of style.uri)
      removeSheet(window, data.url(uri));

  if (style.source) {
    let uri = "data:text/css;charset=utf-8,";

    uri += encodeURIComponent(style.source.join(""));

    removeSheet(window, uri, style.type);
  }
});