browser.ts 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  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 BrowserName = threema.BrowserName;
  18. export class BrowserService {
  19. private logTag: string = '[BrowserService]';
  20. private browser: threema.BrowserInfo;
  21. private $log: ng.ILogService;
  22. private $window: ng.IWindowService;
  23. private isPageVisible = true;
  24. private supportsExtendedLocaleCompareCache: boolean;
  25. public static $inject = ['$log', '$window'];
  26. constructor($log: ng.ILogService, $window: ng.IWindowService) {
  27. // Angular services
  28. this.$log = $log;
  29. this.$window = $window;
  30. this.initializePageVisibility();
  31. }
  32. private initializePageVisibility() {
  33. const onChange = (isVisible: any) => {
  34. if (this.isPageVisible !== isVisible) {
  35. this.isPageVisible = isVisible;
  36. }
  37. };
  38. let pageHiddenKey = 'hidden';
  39. // add default visibility change listener
  40. let defaultListener;
  41. if (pageHiddenKey in document) {
  42. defaultListener = 'visibilitychange';
  43. } else if ('mozHidden' in document) {
  44. pageHiddenKey = 'mozHidden';
  45. defaultListener = 'mozvisibilitychange';
  46. } else if ('webkitHidden' in document) {
  47. pageHiddenKey = 'webkitHidden';
  48. defaultListener = 'webkitvisibilitychange';
  49. } else if ('msHidden' in document) {
  50. pageHiddenKey = 'msHidden';
  51. defaultListener = 'msvisibilitychange';
  52. }
  53. document.addEventListener(defaultListener, function() {
  54. onChange(!this[pageHiddenKey]);
  55. });
  56. // configure other document and window events
  57. const map = {
  58. focus: true,
  59. blur: false,
  60. };
  61. for (const event in map) {
  62. if (map[event] !== undefined) {
  63. document.addEventListener(event, () => {
  64. onChange(map[event]);
  65. }, false);
  66. window.addEventListener(event, () => {
  67. onChange(map[event]);
  68. }, false);
  69. }
  70. }
  71. // initial visible state set
  72. if (document[pageHiddenKey] !== undefined ) {
  73. onChange(!document[pageHiddenKey]);
  74. }
  75. }
  76. public getBrowser(): threema.BrowserInfo {
  77. if (this.browser === undefined) {
  78. this.browser = {
  79. chrome: false,
  80. firefox: false,
  81. fxios: false,
  82. ie: false,
  83. edge: false,
  84. opera: false,
  85. safari: false,
  86. } as threema.BrowserInfo;
  87. const uagent = this.$window.navigator.userAgent.toLowerCase();
  88. this.browser.chrome = /webkit/.test(uagent) && /chrome/.test(uagent) && !/edge/.test(uagent);
  89. this.browser.firefox = /mozilla/.test(uagent) && /firefox/.test(uagent);
  90. this.browser.fxios = /mozilla/.test(uagent) && /fxios/.test(uagent);
  91. this.browser.ie = (/msie/.test(uagent) || /trident/.test(uagent)) && !/edge/.test(uagent);
  92. this.browser.edge = /edge/.test(uagent);
  93. this.browser.safari = /safari/.test(uagent) && /applewebkit/.test(uagent)
  94. && !/chrome/.test(uagent) && !/fxios/.test(uagent);
  95. this.browser.opera = /mozilla/.test(uagent) && /applewebkit/.test(uagent)
  96. && /chrome/.test(uagent) && /safari/.test(uagent) && /opr/.test(uagent);
  97. if (this.browser.opera && this.browser.chrome) {
  98. this.browser.chrome = false;
  99. }
  100. // Mobile detection
  101. this.browser.mobile = false;
  102. if (this.browser.safari && /mobile/.test(uagent)) {
  103. this.browser.mobile = true;
  104. } else if (this.browser.fxios) {
  105. this.browser.mobile = true;
  106. }
  107. for (const x in this.browser) {
  108. if (this.browser[x]) {
  109. let b;
  110. if (x === 'ie') {
  111. b = 'msie';
  112. } else if (x === 'edge') {
  113. b = 'edge';
  114. } else if (x === 'opera') {
  115. b = 'opr';
  116. } else if (x === 'safari') {
  117. b = 'version';
  118. } else {
  119. b = x;
  120. }
  121. let match = uagent.match(new RegExp('(' + b + ')( |\/)([0-9]+)'));
  122. let version;
  123. if (match) {
  124. version = match[3];
  125. } else {
  126. match = uagent.match(new RegExp('rv:([0-9]+)'));
  127. version = match ? match[1] : '';
  128. }
  129. const versionInt: number = parseInt(match[3], 10);
  130. this.browser.version = isNaN(versionInt) ? undefined : versionInt;
  131. break;
  132. }
  133. }
  134. if (this.browser.chrome) {
  135. this.browser.name = BrowserName.Chrome;
  136. this.browser.textInfo = 'Chrome ' + this.browser.version;
  137. }
  138. if (this.browser.firefox) {
  139. this.browser.name = BrowserName.Firefox;
  140. this.browser.textInfo = 'Firefox ' + this.browser.version;
  141. }
  142. if (this.browser.fxios) {
  143. this.browser.name = BrowserName.FirefoxIos;
  144. this.browser.textInfo = 'Firefox (iOS) ' + this.browser.version;
  145. }
  146. if (this.browser.ie) {
  147. this.browser.name = BrowserName.InternetExplorer;
  148. this.browser.textInfo = 'Internet Explorer ' + this.browser.version;
  149. }
  150. if (this.browser.edge) {
  151. this.browser.name = BrowserName.Edge;
  152. this.browser.textInfo = 'Edge ' + this.browser.version;
  153. }
  154. if (this.browser.safari) {
  155. this.browser.name = BrowserName.Safari;
  156. this.browser.textInfo = 'Safari ' + this.browser.version;
  157. }
  158. if (this.browser.opera) {
  159. this.browser.name = BrowserName.Opera;
  160. this.browser.textInfo = 'Opera ' + this.browser.version;
  161. }
  162. if (this.browser.textInfo && this.browser.mobile) {
  163. this.browser.textInfo += ' [Mobile]';
  164. }
  165. }
  166. return this.browser;
  167. }
  168. public isVisible() {
  169. return this.isPageVisible;
  170. }
  171. /**
  172. * Return whether the current browser supports the WebRTC task or not.
  173. */
  174. public supportsWebrtcTask() {
  175. if (this.browser === undefined) {
  176. this.getBrowser();
  177. }
  178. if (this.browser.safari || this.browser.fxios) {
  179. return false;
  180. }
  181. return true;
  182. }
  183. /**
  184. * Return whether the browser supports extended `string.localeCompare` options.
  185. */
  186. public supportsExtendedLocaleCompare() {
  187. if (this.supportsExtendedLocaleCompareCache !== undefined) {
  188. return this.supportsExtendedLocaleCompareCache;
  189. }
  190. function getSupport(): boolean {
  191. try {
  192. 'foo'.localeCompare('bar', 'i');
  193. } catch (e) {
  194. return e.name === 'RangeError';
  195. }
  196. return false;
  197. }
  198. const support = getSupport();
  199. this.supportsExtendedLocaleCompareCache = support;
  200. this.$log.debug(this.logTag, 'Browser',
  201. support ? 'supports' : 'does not support',
  202. 'extended locale compare options');
  203. return support;
  204. }
  205. }