browser.ts 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. /**
  2. * This file is part of Threema Web.
  3. *
  4. * Threema Web is free software: you can redistribute it and/or modify it
  5. * under the terms of the GNU Affero General Public License as published by
  6. * the Free Software Foundation, either version 3 of the License, or (at
  7. * your option) any later version.
  8. *
  9. * This program is distributed in the hope that it will be useful, but
  10. * WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
  12. * General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU Affero General Public License
  15. * along with Threema Web. If not, see <http://www.gnu.org/licenses/>.
  16. */
  17. import {BrowserInfo} from '../helpers/browser_info';
  18. import BrowserName = threema.BrowserName;
  19. export class BrowserService {
  20. private logTag: string = '[BrowserService]';
  21. private browser: BrowserInfo;
  22. private $log: ng.ILogService;
  23. private $window: ng.IWindowService;
  24. private isPageVisible = true;
  25. private supportsExtendedLocaleCompareCache: boolean;
  26. public static $inject = ['$log', '$window'];
  27. constructor($log: ng.ILogService, $window: ng.IWindowService) {
  28. // Angular services
  29. this.$log = $log;
  30. this.$window = $window;
  31. this.initializePageVisibility();
  32. }
  33. private initializePageVisibility() {
  34. const onChange = (isVisible: any) => {
  35. if (this.isPageVisible !== isVisible) {
  36. this.isPageVisible = isVisible;
  37. }
  38. };
  39. let pageHiddenKey = 'hidden';
  40. // add default visibility change listener
  41. let defaultListener;
  42. if (pageHiddenKey in document) {
  43. defaultListener = 'visibilitychange';
  44. } else if ('mozHidden' in document) {
  45. pageHiddenKey = 'mozHidden';
  46. defaultListener = 'mozvisibilitychange';
  47. } else if ('webkitHidden' in document) {
  48. pageHiddenKey = 'webkitHidden';
  49. defaultListener = 'webkitvisibilitychange';
  50. } else if ('msHidden' in document) {
  51. pageHiddenKey = 'msHidden';
  52. defaultListener = 'msvisibilitychange';
  53. }
  54. document.addEventListener(defaultListener, function() {
  55. onChange(!this[pageHiddenKey]);
  56. });
  57. // configure other document and window events
  58. const map = {
  59. focus: true,
  60. blur: false,
  61. };
  62. for (const event in map) {
  63. if (map[event] !== undefined) {
  64. document.addEventListener(event, () => {
  65. onChange(map[event]);
  66. }, false);
  67. window.addEventListener(event, () => {
  68. onChange(map[event]);
  69. }, false);
  70. }
  71. }
  72. // initial visible state set
  73. if (document[pageHiddenKey] !== undefined ) {
  74. onChange(!document[pageHiddenKey]);
  75. }
  76. }
  77. public getBrowser(): BrowserInfo {
  78. if (this.browser === undefined) {
  79. const browser = {
  80. chrome: false,
  81. chromeIos: false,
  82. firefox: false,
  83. firefoxIos: false,
  84. ie: false,
  85. edge: false,
  86. opera: false,
  87. safari: false,
  88. };
  89. const uagent = this.$window.navigator.userAgent.toLowerCase();
  90. browser.chrome = /webkit/.test(uagent) && /chrome/.test(uagent) && !/edge/.test(uagent);
  91. browser.chromeIos = /mozilla/.test(uagent) && /crios/.test(uagent);
  92. browser.firefox = /mozilla/.test(uagent) && /firefox/.test(uagent);
  93. browser.firefoxIos = /mozilla/.test(uagent) && /fxios/.test(uagent);
  94. browser.ie = (/msie/.test(uagent) || /trident/.test(uagent)) && !/edge/.test(uagent);
  95. browser.edge = /edge/.test(uagent);
  96. browser.safari = /safari/.test(uagent) && /applewebkit/.test(uagent)
  97. && !/chrome/.test(uagent) && !/fxios/.test(uagent) && !/crios/.test(uagent);
  98. browser.opera = /mozilla/.test(uagent) && /applewebkit/.test(uagent)
  99. && /chrome/.test(uagent) && /safari/.test(uagent) && /opr/.test(uagent);
  100. if (browser.opera && browser.chrome) {
  101. browser.chrome = false;
  102. }
  103. let version = null;
  104. for (const x in browser) {
  105. if (browser[x]) {
  106. let b;
  107. if (x === 'ie') {
  108. b = 'msie';
  109. } else if (x === 'edge') {
  110. b = 'edge';
  111. } else if (x === 'opera') {
  112. b = 'opr';
  113. } else if (x === 'firefoxIos') {
  114. b = 'fxios';
  115. } else if (x === 'chromeIos') {
  116. b = 'crios';
  117. } else if (x === 'safari') {
  118. b = 'version';
  119. } else {
  120. b = x;
  121. }
  122. let match = uagent.match(new RegExp('(' + b + ')( |\/)([0-9]+)'));
  123. let versionString;
  124. if (match) {
  125. versionString = match[3];
  126. } else {
  127. match = uagent.match(new RegExp('rv:([0-9]+)'));
  128. versionString = match ? match[1] : '';
  129. }
  130. const versionInt: number = parseInt(versionString, 10);
  131. version = isNaN(versionInt) ? undefined : versionInt;
  132. break;
  133. }
  134. }
  135. if (browser.chrome) {
  136. this.browser = new BrowserInfo(uagent, BrowserName.Chrome, version);
  137. }
  138. if (browser.chromeIos) {
  139. this.browser = new BrowserInfo(uagent, BrowserName.ChromeIos, version, true);
  140. }
  141. if (browser.firefox) {
  142. this.browser = new BrowserInfo(uagent, BrowserName.Firefox, version);
  143. }
  144. if (browser.firefoxIos) {
  145. this.browser = new BrowserInfo(uagent, BrowserName.FirefoxIos, version, true);
  146. }
  147. if (browser.ie) {
  148. this.browser = new BrowserInfo(uagent, BrowserName.InternetExplorer, version);
  149. }
  150. if (browser.edge) {
  151. this.browser = new BrowserInfo(uagent, BrowserName.Edge, version);
  152. }
  153. if (browser.safari) {
  154. const mobile = /mobile/.test(uagent);
  155. this.browser = new BrowserInfo(uagent, BrowserName.Safari, version, mobile);
  156. }
  157. if (browser.opera) {
  158. this.browser = new BrowserInfo(uagent, BrowserName.Opera, version);
  159. }
  160. }
  161. return this.browser;
  162. }
  163. public isVisible() {
  164. return this.isPageVisible;
  165. }
  166. /**
  167. * Return whether the current browser supports the WebRTC task or not.
  168. */
  169. public supportsWebrtcTask() {
  170. if (this.browser === undefined) {
  171. this.getBrowser();
  172. }
  173. return this.browser.supportsWebrtcTask();
  174. }
  175. /**
  176. * Return whether the browser supports extended `string.localeCompare` options.
  177. */
  178. public supportsExtendedLocaleCompare() {
  179. if (this.supportsExtendedLocaleCompareCache !== undefined) {
  180. return this.supportsExtendedLocaleCompareCache;
  181. }
  182. function getSupport(): boolean {
  183. try {
  184. 'foo'.localeCompare('bar', 'i');
  185. } catch (e) {
  186. return e.name === 'RangeError';
  187. }
  188. return false;
  189. }
  190. const support = getSupport();
  191. this.supportsExtendedLocaleCompareCache = support;
  192. this.$log.debug(this.logTag, 'Browser',
  193. support ? 'supports' : 'does not support',
  194. 'extended locale compare options');
  195. return support;
  196. }
  197. }