Merge pull request #1997 from yurydelendik/font-refact-1
Rectoring font loading concurency
This commit is contained in:
		
						commit
						d77bafa678
					
				
							
								
								
									
										126
									
								
								src/fonts.js
									
									
									
									
									
								
							
							
						
						
									
										126
									
								
								src/fonts.js
									
									
									
									
									
								
							| @ -404,28 +404,15 @@ function mapPrivateUseChars(code) { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| var FontLoader = { | var FontLoader = { | ||||||
|   listeningForFontLoad: false, |   loadingContext: { | ||||||
|  |     requests: [], | ||||||
|  |     nextRequestId: 0 | ||||||
|  |   }, | ||||||
| 
 | 
 | ||||||
|   bind: function fontLoaderBind(fonts, callback) { |   bind: function fontLoaderBind(fonts, callback) { | ||||||
|     function checkFontsLoaded() { |     assert(!isWorker, 'bind() shall be called from main thread'); | ||||||
|       for (var i = 0, ii = fonts.length; i < ii; i++) { |  | ||||||
|         var fontObj = fonts[i]; |  | ||||||
|         if (fontObj.loading) { |  | ||||||
|           return false; |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       document.documentElement.removeEventListener( |  | ||||||
|         'pdfjsFontLoad', checkFontsLoaded, false); |  | ||||||
| 
 |  | ||||||
|       // Use timeout to fix chrome intermittent failures on font loading.
 |  | ||||||
|       setTimeout(callback, 0); |  | ||||||
|       return true; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     var rules = [], names = [], fontsToLoad = []; |  | ||||||
|     var fontCreateTimer = 0; |  | ||||||
| 
 | 
 | ||||||
|  |     var rules = [], fontsToLoad = []; | ||||||
|     for (var i = 0, ii = fonts.length; i < ii; i++) { |     for (var i = 0, ii = fonts.length; i < ii; i++) { | ||||||
|       var font = fonts[i]; |       var font = fonts[i]; | ||||||
| 
 | 
 | ||||||
| @ -436,8 +423,6 @@ var FontLoader = { | |||||||
|       } |       } | ||||||
|       font.attached = true; |       font.attached = true; | ||||||
| 
 | 
 | ||||||
|       fontsToLoad.push(font); |  | ||||||
| 
 |  | ||||||
|       var str = ''; |       var str = ''; | ||||||
|       var data = font.data; |       var data = font.data; | ||||||
|       if (data) { |       if (data) { | ||||||
| @ -448,28 +433,51 @@ var FontLoader = { | |||||||
|         var rule = font.bindDOM(str); |         var rule = font.bindDOM(str); | ||||||
|         if (rule) { |         if (rule) { | ||||||
|           rules.push(rule); |           rules.push(rule); | ||||||
|           names.push(font.loadedName); |           fontsToLoad.push(font); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     this.listeningForFontLoad = false; |     var request = FontLoader.queueLoadingCallback(callback); | ||||||
|     if (!isWorker && rules.length) { |     if (rules.length > 0) { | ||||||
|       FontLoader.prepareFontLoadEvent(rules, names, fontsToLoad); |       FontLoader.prepareFontLoadEvent(rules, fontsToLoad, request); | ||||||
|     } |     } else { | ||||||
| 
 |       request.complete(); | ||||||
|     if (!checkFontsLoaded()) { |  | ||||||
|       document.documentElement.addEventListener( |  | ||||||
|         'pdfjsFontLoad', checkFontsLoaded, false); |  | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|  | 
 | ||||||
|  |   queueLoadingCallback: function FontLoader_queueLoadingCallback(callback) { | ||||||
|  |     function LoadLoader_completeRequest() { | ||||||
|  |       assert(!request.end, 'completeRequest() cannot be called twice'); | ||||||
|  |       request.end = Date.now(); | ||||||
|  | 
 | ||||||
|  |       // sending all completed requests in order how they were queued
 | ||||||
|  |       while (context.requests.length > 0 && context.requests[0].end) { | ||||||
|  |         var otherRequest = context.requests.shift(); | ||||||
|  |         setTimeout(otherRequest.callback, 0); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     var context = FontLoader.loadingContext; | ||||||
|  |     var requestId = 'pdfjs-font-loading-' + (context.nextRequestId++); | ||||||
|  |     var request = { | ||||||
|  |       id: requestId, | ||||||
|  |       complete: LoadLoader_completeRequest, | ||||||
|  |       callback: callback, | ||||||
|  |       started: Date.now() | ||||||
|  |     }; | ||||||
|  |     context.requests.push(request); | ||||||
|  |     return request; | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|   // Set things up so that at least one pdfjsFontLoad event is
 |   // Set things up so that at least one pdfjsFontLoad event is
 | ||||||
|   // dispatched when all the @font-face |rules| for |names| have been
 |   // dispatched when all the @font-face |rules| for |fonts| have been
 | ||||||
|   // loaded in a subdocument.  It's expected that the load of |rules|
 |   // loaded in a subdocument.  It's expected that the load of |rules|
 | ||||||
|   // has already started in this (outer) document, so that they should
 |   // has already started in this (outer) document, so that they should
 | ||||||
|   // be ordered before the load in the subdocument.
 |   // be ordered before the load in the subdocument.
 | ||||||
|   prepareFontLoadEvent: function fontLoaderPrepareFontLoadEvent(rules, names, |   prepareFontLoadEvent: function fontLoaderPrepareFontLoadEvent(rules, | ||||||
|                                                                 fonts) { |                                                                 fonts, | ||||||
|  |                                                                 request) { | ||||||
|       /** Hack begin */ |       /** Hack begin */ | ||||||
|       // There's no event when a font has finished downloading so the
 |       // There's no event when a font has finished downloading so the
 | ||||||
|       // following code is a dirty hack to 'guess' when a font is
 |       // following code is a dirty hack to 'guess' when a font is
 | ||||||
| @ -493,6 +501,20 @@ var FontLoader = { | |||||||
|       // The postMessage() hackery was added to work around chrome bug
 |       // The postMessage() hackery was added to work around chrome bug
 | ||||||
|       // 82402.
 |       // 82402.
 | ||||||
| 
 | 
 | ||||||
|  |       var requestId = request.id; | ||||||
|  |       // Validate the requestId parameter -- the value used to construct HTML.
 | ||||||
|  |       if (!/^[\w\-]+$/.test(requestId)) { | ||||||
|  |         error('Invalid request id: ' + requestId); | ||||||
|  | 
 | ||||||
|  |         // Normally the error-function throws. But if a malicious code
 | ||||||
|  |         // intercepts the function call then the return is needed.
 | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       var names = []; | ||||||
|  |       for (var i = 0, ii = fonts.length; i < ii; i++) | ||||||
|  |         names.push(fonts[i].loadedName); | ||||||
|  | 
 | ||||||
|       // Validate the names parameter -- the values can used to construct HTML.
 |       // Validate the names parameter -- the values can used to construct HTML.
 | ||||||
|       if (!/^\w+$/.test(names.join(''))) { |       if (!/^\w+$/.test(names.join(''))) { | ||||||
|         error('Invalid font name(s): ' + names.join()); |         error('Invalid font name(s): ' + names.join()); | ||||||
| @ -514,22 +536,21 @@ var FontLoader = { | |||||||
|       div.innerHTML = html; |       div.innerHTML = html; | ||||||
|       document.body.appendChild(div); |       document.body.appendChild(div); | ||||||
| 
 | 
 | ||||||
|       if (!this.listeningForFontLoad) { |       window.addEventListener( | ||||||
|         window.addEventListener( |         'message', | ||||||
|           'message', |         function fontLoaderMessage(e) { | ||||||
|           function fontLoaderMessage(e) { |           if (e.data !== requestId) | ||||||
|             var fontNames = JSON.parse(e.data); |             return; | ||||||
|             for (var i = 0, ii = fonts.length; i < ii; ++i) { |           for (var i = 0, ii = fonts.length; i < ii; ++i) { | ||||||
|               var font = fonts[i]; |             var font = fonts[i]; | ||||||
|               font.loading = false; |             font.loading = false; | ||||||
|             } |           } | ||||||
|             var evt = document.createEvent('Events'); |           request.complete(); | ||||||
|             evt.initEvent('pdfjsFontLoad', true, false); |           // cleanup
 | ||||||
|             document.documentElement.dispatchEvent(evt); |           document.body.removeChild(frame); | ||||||
|           }, |           window.removeEventListener('message', fontLoaderMessage, false); | ||||||
|           false); |         }, | ||||||
|         this.listeningForFontLoad = true; |         false); | ||||||
|       } |  | ||||||
| 
 | 
 | ||||||
|       // XXX we should have a time-out here too, and maybe fire
 |       // XXX we should have a time-out here too, and maybe fire
 | ||||||
|       // pdfjsFontLoadFailed?
 |       // pdfjsFontLoadFailed?
 | ||||||
| @ -540,13 +561,8 @@ var FontLoader = { | |||||||
|       } |       } | ||||||
|       src += '</style>'; |       src += '</style>'; | ||||||
|       src += '<script type="application/javascript">'; |       src += '<script type="application/javascript">'; | ||||||
|       var fontNamesArray = ''; |  | ||||||
|       for (var i = 0, ii = names.length; i < ii; ++i) { |  | ||||||
|         fontNamesArray += '"' + names[i] + '", '; |  | ||||||
|       } |  | ||||||
|       src += '  var fontNames=[' + fontNamesArray + '];\n'; |  | ||||||
|       src += '  window.onload = function fontLoaderOnload() {\n'; |       src += '  window.onload = function fontLoaderOnload() {\n'; | ||||||
|       src += '    parent.postMessage(JSON.stringify(fontNames), "*");\n'; |       src += '    parent.postMessage("' + requestId + '", "*");\n'; | ||||||
|       src += '  }'; |       src += '  }'; | ||||||
|       // Hack so the end script tag isn't counted if this is inline JS.
 |       // Hack so the end script tag isn't counted if this is inline JS.
 | ||||||
|       src += '</scr' + 'ipt></head><body>'; |       src += '</scr' + 'ipt></head><body>'; | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user