OLD | NEW |
1 /* | 1 /* |
2 * This file is part of Adblock Plus <https://adblockplus.org/>, | 2 * This file is part of Adblock Plus <https://adblockplus.org/>, |
3 * Copyright (C) 2006-present eyeo GmbH | 3 * Copyright (C) 2006-present eyeo GmbH |
4 * | 4 * |
5 * Adblock Plus is free software: you can redistribute it and/or modify | 5 * Adblock Plus is free software: you can redistribute it and/or modify |
6 * it under the terms of the GNU General Public License version 3 as | 6 * it under the terms of the GNU General Public License version 3 as |
7 * published by the Free Software Foundation. | 7 * published by the Free Software Foundation. |
8 * | 8 * |
9 * Adblock Plus is distributed in the hope that it will be useful, | 9 * Adblock Plus is distributed in the hope that it will be useful, |
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of | 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
(...skipping 13 matching lines...) Expand all Loading... |
24 const {EventEmitter} = require("./events"); | 24 const {EventEmitter} = require("./events"); |
25 const {filterNotifier} = require("./filterNotifier"); | 25 const {filterNotifier} = require("./filterNotifier"); |
26 | 26 |
27 /** | 27 /** |
28 * Lookup table, lists of element hiding exceptions by selector | 28 * Lookup table, lists of element hiding exceptions by selector |
29 * @type {Map.<string,ElemHideException[]>} | 29 * @type {Map.<string,ElemHideException[]>} |
30 */ | 30 */ |
31 let exceptions = new Map(); | 31 let exceptions = new Map(); |
32 | 32 |
33 /** | 33 /** |
| 34 * Number of element hiding exceptions per domain. |
| 35 * @type {Map.<string,number>} |
| 36 */ |
| 37 let exceptionCountByDomain = new Map(); |
| 38 |
| 39 /** |
34 * Set containing known element exceptions | 40 * Set containing known element exceptions |
35 * @type {Set.<ElemHideException>} | 41 * @type {Set.<ElemHideException>} |
36 */ | 42 */ |
37 let knownExceptions = new Set(); | 43 let knownExceptions = new Set(); |
38 | 44 |
39 /** | 45 /** |
40 * Container for element hiding exceptions | 46 * Container for element hiding exceptions |
41 * @class | 47 * @class |
42 */ | 48 */ |
43 exports.ElemHideExceptions = Object.assign(Object.create(new EventEmitter()), { | 49 exports.ElemHideExceptions = Object.assign(Object.create(new EventEmitter()), { |
44 /** | 50 /** |
45 * Removes all known exceptions | 51 * Removes all known exceptions |
46 */ | 52 */ |
47 clear() | 53 clear() |
48 { | 54 { |
49 exceptions.clear(); | 55 exceptions.clear(); |
| 56 exceptionCountByDomain.clear(); |
50 knownExceptions.clear(); | 57 knownExceptions.clear(); |
51 | 58 |
52 filterNotifier.emit("elemhideupdate"); | 59 filterNotifier.emit("elemhideupdate"); |
53 }, | 60 }, |
54 | 61 |
55 /** | 62 /** |
56 * Add a new element hiding exception | 63 * Add a new element hiding exception |
57 * @param {ElemHideException} exception | 64 * @param {ElemHideException} exception |
58 */ | 65 */ |
59 add(exception) | 66 add(exception) |
60 { | 67 { |
61 if (knownExceptions.has(exception)) | 68 if (knownExceptions.has(exception)) |
62 return; | 69 return; |
63 | 70 |
64 let {selector} = exception; | 71 let {selector} = exception; |
65 let list = exceptions.get(selector); | 72 let list = exceptions.get(selector); |
66 if (list) | 73 if (list) |
67 list.push(exception); | 74 list.push(exception); |
68 else | 75 else |
69 exceptions.set(selector, [exception]); | 76 exceptions.set(selector, [exception]); |
70 | 77 |
| 78 for (let [domain] of exception.domains || []) |
| 79 { |
| 80 if (domain == "") |
| 81 continue; |
| 82 |
| 83 let count = exceptionCountByDomain.get(domain) || 0; |
| 84 exceptionCountByDomain.set(domain, count + 1); |
| 85 } |
| 86 |
71 knownExceptions.add(exception); | 87 knownExceptions.add(exception); |
72 | 88 |
73 this.emit("added", exception); | 89 this.emit("added", exception); |
74 | 90 |
75 filterNotifier.emit("elemhideupdate"); | 91 filterNotifier.emit("elemhideupdate"); |
76 }, | 92 }, |
77 | 93 |
78 /** | 94 /** |
79 * Removes an element hiding exception | 95 * Removes an element hiding exception |
80 * @param {ElemHideException} exception | 96 * @param {ElemHideException} exception |
81 */ | 97 */ |
82 remove(exception) | 98 remove(exception) |
83 { | 99 { |
84 if (!knownExceptions.has(exception)) | 100 if (!knownExceptions.has(exception)) |
85 return; | 101 return; |
86 | 102 |
87 let list = exceptions.get(exception.selector); | 103 let list = exceptions.get(exception.selector); |
88 let index = list.indexOf(exception); | 104 let index = list.indexOf(exception); |
89 if (index >= 0) | 105 if (index >= 0) |
90 list.splice(index, 1); | 106 list.splice(index, 1); |
91 | 107 |
| 108 for (let [domain] of exception.domains || []) |
| 109 { |
| 110 if (domain == "") |
| 111 continue; |
| 112 |
| 113 let count = exceptionCountByDomain.get(domain) || 0; |
| 114 if (count == 1) |
| 115 exceptionCountByDomain.delete(domain); |
| 116 else if (count > 1) |
| 117 exceptionCountByDomain.set(domain, count - 1); |
| 118 } |
| 119 |
92 knownExceptions.delete(exception); | 120 knownExceptions.delete(exception); |
93 | 121 |
94 this.emit("removed", exception); | 122 this.emit("removed", exception); |
95 | 123 |
96 filterNotifier.emit("elemhideupdate"); | 124 filterNotifier.emit("elemhideupdate"); |
97 }, | 125 }, |
98 | 126 |
99 /** | 127 /** |
100 * Checks whether any exception rules are registered for a selector | 128 * Checks whether any exception rules are registered for a selector |
101 * @param {string} selector | 129 * @param {string} selector |
102 * @returns {boolean} | 130 * @returns {boolean} |
103 */ | 131 */ |
104 hasExceptions(selector) | 132 hasExceptions(selector) |
105 { | 133 { |
106 return exceptions.has(selector); | 134 return exceptions.has(selector); |
107 }, | 135 }, |
108 | 136 |
109 /** | 137 /** |
| 138 * Checks whether any exception rules are registered for a domain |
| 139 * @param {string} domain |
| 140 * @returns {boolean} |
| 141 */ |
| 142 domainHasExceptions(domain) |
| 143 { |
| 144 return exceptionCountByDomain.has(domain); |
| 145 }, |
| 146 |
| 147 /** |
110 * Checks whether an exception rule is registered for a selector on a | 148 * Checks whether an exception rule is registered for a selector on a |
111 * particular domain. | 149 * particular domain. |
112 * @param {string} selector | 150 * @param {string} selector |
113 * @param {?string} docDomain | 151 * @param {?string} docDomain |
114 * @return {?ElemHideException} | 152 * @return {?ElemHideException} |
115 */ | 153 */ |
116 getException(selector, docDomain) | 154 getException(selector, docDomain) |
117 { | 155 { |
118 let list = exceptions.get(selector); | 156 let list = exceptions.get(selector); |
119 if (!list) | 157 if (!list) |
120 return null; | 158 return null; |
121 | 159 |
122 for (let i = list.length - 1; i >= 0; i--) | 160 for (let i = list.length - 1; i >= 0; i--) |
123 { | 161 { |
124 if (list[i].isActiveOnDomain(docDomain)) | 162 if (list[i].isActiveOnDomain(docDomain)) |
125 return list[i]; | 163 return list[i]; |
126 } | 164 } |
127 | 165 |
128 return null; | 166 return null; |
129 } | 167 } |
130 }); | 168 }); |
OLD | NEW |