Skip to content
This repository was archived by the owner on Apr 12, 2024. It is now read-only.

Interpolations in CJK and Cyrillic languages broken in IE11 #14924

Closed
jshoudy11 opened this issue Jul 15, 2016 · 22 comments
Closed

Interpolations in CJK and Cyrillic languages broken in IE11 #14924

jshoudy11 opened this issue Jul 15, 2016 · 22 comments

Comments

@jshoudy11
Copy link

Bug
Certain interpolations are being broken into multiple spans. For example in a template we have
{{ctrl.variableName}}.

In Chrome, Firefox, Safari, Edge this always becomes Text associated with variable

but in IE11 in CJK languages this is becoming {{ctrl.variableName}} which displays on our page as the variable.

There are several places this bug is displayed in production for our site but I will include steps to reproduce for just one of them.

To reproduce:

  1. Go to https://console.cloud.google.com in IE11
  2. Click the hamburger menu in the top left -> click API Manager
  3. Click Credentials in the left side nav (if you're prompted to create a project click "Create")
  4. Click "Create credentials" -> "API Key" -> "Server Key" -> "Create"
  5. Go back to the Credentials Page and click on the name of your new API Key
  6. Click the vertical "..." in the top right -> Preferences -> Language and regional formats
  7. Change the language to a CJK language (Japanese for example) and click Save
  8. The page will reload. Observe the {{ctrl.key.currentKey}} and other variables displayed in the page.

Here is a screenshot of the issue:
angularbug

** Details **
This is a bug specific to IE11 (and earlier IE). I'm using IE version 11.0.10240.16603 on Windows 10 Enterprise. We are running angular 1.5 but this also occurred in 1.4

@gkalpak
Copy link
Member

gkalpak commented Jul 16, 2016

I can reproduce this. A very similar quirk was fixed with #11796. Ooc, are you using MutationObservers?

@gkalpak gkalpak self-assigned this Jul 16, 2016
@gkalpak
Copy link
Member

gkalpak commented Jul 16, 2016

You don't happen to have a more minimal reproduction (or at least one with unminified sources), do you? 😛

@gkalpak gkalpak removed their assignment Jul 16, 2016
@gkalpak
Copy link
Member

gkalpak commented Jul 16, 2016

I tried my best to reproduce this, but failed miserably. And I have been equally unsuccessful at debugging this directly on https://console.cloud.google.com in IE11. 😟

If someone is able to provide a more "debuggable" reproduction or has access to investigate on the dev environment of https://console.cloud.google.com, that would help a lot 😄

@jshoudy11
Copy link
Author

Hey Georgios, thanks for looking into this. I have also not had much luck with getting a minimal reproduction but I do have access to the dev environment. Do you have any ideas for places where I can start to debug?

@gkalpak
Copy link
Member

gkalpak commented Jul 19, 2016

@jshoudy11 , I was finally able to reproduce it 🎉 It seems to be a combination of ngTransclude with the known IE11 bug wrt to MutationObservers and {{ characters following "special characters", such as — (or CJK characters).

It is interesting that there is different behavior between 1.5.7 and the current master - probably as a result of our stopping to wrap transcluded text-content in spans (as of 7617c08). Just to be clear, the behavior is problematic on both versions, but different: 1.5.7 demo, snapshot demo

(Now, all we need to find out is whether/how we can work around this 😄)

@gkalpak
Copy link
Member

gkalpak commented Jul 19, 2016

Btw, the simplest work-around for application code, seems to be wrapping your transcluded text nodes in spans. E.g.:

<!-- BEFORE: -->
<directive-with-transclusion>
  My text content
</directive-with-transclusion>

<!-- AFTER: -->
<directive-with-transclusion>
  <span>My text content</span>
</directive-with-transclusion>

@lgalfaso
Copy link
Contributor

lgalfaso commented Jul 19, 2016

This is a wild guess on what the issue is (as I do not have access to IE). First IE by using MutationObservers splits the text into two nodes. This is common for 1.5 and 1.6.

In 1.5 here
https://github.com/angular/angular.js/blob/v1.5.x/src/ng/compile.js#L1758
we do not look for two consecutive text nodes, then these are wrapped in span element, so the interpolation fails.

In 1.6 here
https://github.com/angular/angular.js/blob/master/src/ng/compile.js#L2047
given that the content is transcluded, there is no parent and the nodes are not merged.

@jshoudy11
Copy link
Author

Thank you for the help @gkalpak and @lgalfaso. I actually do not believe we are using MutationObservers in the areas where this error is occurring. But I will try to wrap the interpolations in question in span tags and see if it works. I don't have access to IE today but I will tomorrow. I'll let you know if it works.

@Narretz
Copy link
Contributor

Narretz commented Jul 20, 2016

@gkalpak your snapshot demo is broken in IE11 for me even without mutation observers.

@gkalpak
Copy link
Member

gkalpak commented Jul 20, 2016

@Narretz , IE apparently has this "feature" that if you run it once with the MutationObserver, removing it and clicking the "Run" button doesn't fix the problem (it is as if it is running with the MutationObserver).

You have to save it (e.g. fork the pen) without the MutationObserver. Only when reloading the whole page do changes in MutationObserver take effect (or something along these lines).

@Narretz
Copy link
Contributor

Narretz commented Jul 20, 2016

Ah, okay, thanks for the explanation. IE strikes again.

@jshoudy11
Copy link
Author

@gkalpak Confirmed that wrapping in span tags is a functioning workaround for us. Thank you for the help. I'm sure you guys will keep working on a more permanent workaround within angular and I'll stay tuned.

@jshoudy11
Copy link
Author

jshoudy11 commented Jul 21, 2016

@gkalpak is there an ETA on a solution to this in the angular code?

@gkalpak
Copy link
Member

gkalpak commented Jul 25, 2016

@jshoudy11, I am planning to look into this today some time this week soon. So, (if all goes well) the fix will be in 1.5.9.

@jshoudy11
Copy link
Author

@gkalpak what is the status of this?

gkalpak added a commit to gkalpak/angular.js that referenced this issue Aug 13, 2016
@gkalpak
Copy link
Member

gkalpak commented Aug 13, 2016

@jshoudy11, I have a fix for this on my fork (for 1.6.x), but I am not very happy with it. And it's kind of a breaking change, so I guess it won't be backported to 1.5.x. Here is the fix: gkalpak@3ad0acd

Your safest bet is still to always wrap transcuded (top-level) text nodes. I'll open a PR soon.

gkalpak added a commit to gkalpak/angular.js that referenced this issue Aug 15, 2016
As explained in angular#11781 and angular#14924, IE11 can (under certain circumstances) break up a text node into
multiple consecutive ones, breaking interpolated expressions (e.g. `{{'foo'}}` would become
`{{` + `'foo'}}`). To work-around this IE11 bug, angular#11796 introduced extra logic to merge consecutive
text nodes (on IE11 only), which relies on the text nodes' having the same `parentNode`.

This approach works fine in the common case, where `compileNodes` is called with a live NodeList
object, because removing a text node from its parent will automatically update the latter's
`.childNodes` NodeList. It falls short though, when calling `compileNodes` with either a
jqLite/jQuery collection or an Array. In fails in two ways:

1. If the text nodes do not have a parent at the moment of compiling, there will be no merging.
   (This happens for example on directives with `$transclude: {...}`.)
2. If the text nodes do have a parent, just removing a text node from its parent does **not** remove
   it from the collection/array, which means that the merged text nodes will still get compiled and
   linked (and possibly be displayed in the view). E.g. `['{{text1}}', '{{text2}}', '{{text3}}']`
   will become `['{{text1}}{{text2}}{{text3}}', '{{text2}}', '{{text3}}']`.

--
This commit works around the above problems by:

1. Merging consecutive text nodes in the provided list, even if they have no parent.
2. When merging a txt node, explicitly remove it from the list (unless it is a live, auto-updating
   list).

This can nonetheless have undesirable (albeit rare) side effects by overzealously merging text
nodes that are not meant to be merged (see "BREAKING CHANGE" section below).

Fixes angular#14924

BREAKING CHANGE:

**Note:** Everything described below affects **IE11 only**.

Previously, consecutive text nodes would not get merged if they had no parent. They will now, which
might have unexpectd side effects in the following cases:

1. Passing an array or jqLite/jQuery collection of parent-less text nodes to `$compile` directly:

    ```js
    // Assuming:
    var textNodes = [
      document.createTextNode('{{'),
      document.createTextNode('"foo"'),
      document.createTextNode('}}')
    ];
    var compiledNodes = $compile(textNodes)($rootScope);

    // Before:
    console.log(compiledNodes.length);   // 3
    console.log(compiledNodes.text());   // {{'foo'}}

    // After:
    console.log(compiledNodes.length);   // 1
    console.log(compiledNodes.text());   // foo

    // To get the old behavior, compile each node separately:
    var textNodes = [
      document.createTextNode('{{'),
      document.createTextNode('"foo"'),
      document.createTextNode('}}')
    ];
    var compiledNodes = angular.element(textNodes.map(function (node) {
      return $compile(node)($rootScope)[0];
    }));
    ```

2. Using multi-slot transclusion with non-consecutive, default-content text nodes (that form
   interpolated expressions when merged):

   ```js
   // Assuming the following component:
   .compoent('someThing', {
     template: '<ng-transclude><!-- Default content goes here --></ng-transclude>'
     transclude: {
       ignored: 'veryImportantContent'
     }
   })
   ```

   ```html
   <!-- And assuming the following view:
   <some-thing>
     {{
     <very-important-content>Nooot</very-important-content>
     'foo'}}
   </some-thing>

   <!-- Before: -->
   <some-thing>
     <ng-transclude>
       {{       <-- Two separate
       'foo'}}  <-- text nodes
     </ng-transclude>
   </some-thing>

   <!-- After: -->
   <some-thing>
     <ng-transclude>
       foo  <-- The text nodes were merged into `{{'foo'}}`, which was then interpolated
     </ng-transclude>
   </some-thing>

   <!-- To (visually) get the old behavior, wrap top-level text-nodes on
   <!-- multi-slot transclusion directives into `<span>`; e.g.:
   <some-thing>
     <span>{{</span>
     <very-important-content>Nooot</very-important-content>
     <span>'foo'}}</span>
   </some-thing>

   <!-- Result: -->
   <some-thing>
     <ng-transclude>
       <span>{{</span>       <-- Two separate
       <span>'foo'}}</span>  <-- nodes
     </ng-transclude>
   </some-thing>
   ```
gkalpak added a commit to gkalpak/angular.js that referenced this issue Aug 15, 2016
As explained in angular#11781 and angular#14924, IE11 can (under certain circumstances) break up a text node into
multiple consecutive ones, breaking interpolated expressions (e.g. `{{'foo'}}` would become
`{{` + `'foo'}}`). To work-around this IE11 bug, angular#11796 introduced extra logic to merge consecutive
text nodes (on IE11 only), which relies on the text nodes' having the same `parentNode`.

This approach works fine in the common case, where `compileNodes` is called with a live NodeList
object, because removing a text node from its parent will automatically update the latter's
`.childNodes` NodeList. It falls short though, when calling `compileNodes` with either a
jqLite/jQuery collection or an Array. In fails in two ways:

1. If the text nodes do not have a parent at the moment of compiling, there will be no merging.
   (This happens for example on directives with `$transclude: {...}`.)
2. If the text nodes do have a parent, just removing a text node from its parent does **not** remove
   it from the collection/array, which means that the merged text nodes will still get compiled and
   linked (and possibly be displayed in the view). E.g. `['{{text1}}', '{{text2}}', '{{text3}}']`
   will become `['{{text1}}{{text2}}{{text3}}', '{{text2}}', '{{text3}}']`.

--
This commit works around the above problems by:

1. Merging consecutive text nodes in the provided list, even if they have no parent.
2. When merging a txt node, explicitly remove it from the list (unless it is a live, auto-updating
   list).

This can nonetheless have undesirable (albeit rare) side effects by overzealously merging text
nodes that are not meant to be merged (see "BREAKING CHANGE" section below).

Fixes angular#14924

BREAKING CHANGE:

**Note:** Everything described below affects **IE11 only**.

Previously, consecutive text nodes would not get merged if they had no parent. They will now, which
might have unexpectd side effects in the following cases:

1. Passing an array or jqLite/jQuery collection of parent-less text nodes to `$compile` directly:

    ```js
    // Assuming:
    var textNodes = [
      document.createTextNode('{{'),
      document.createTextNode('"foo"'),
      document.createTextNode('}}')
    ];
    var compiledNodes = $compile(textNodes)($rootScope);

    // Before:
    console.log(compiledNodes.length);   // 3
    console.log(compiledNodes.text());   // {{'foo'}}

    // After:
    console.log(compiledNodes.length);   // 1
    console.log(compiledNodes.text());   // foo

    // To get the old behavior, compile each node separately:
    var textNodes = [
      document.createTextNode('{{'),
      document.createTextNode('"foo"'),
      document.createTextNode('}}')
    ];
    var compiledNodes = angular.element(textNodes.map(function (node) {
      return $compile(node)($rootScope)[0];
    }));
    ```

2. Using multi-slot transclusion with non-consecutive, default-content text nodes (that form
   interpolated expressions when merged):

   ```js
   // Assuming the following component:
   .compoent('someThing', {
     template: '<ng-transclude><!-- Default content goes here --></ng-transclude>'
     transclude: {
       ignored: 'veryImportantContent'
     }
   })
   ```

   ```html
   <!-- And assuming the following view: -->
   <some-thing>
     {{
     <very-important-content>Nooot</very-important-content>
     'foo'}}
   </some-thing>

   <!-- Before: -->
   <some-thing>
     <ng-transclude>
       {{       <-- Two separate
       'foo'}}  <-- text nodes
     </ng-transclude>
   </some-thing>

   <!-- After: -->
   <some-thing>
     <ng-transclude>
       foo  <-- The text nodes were merged into `{{'foo'}}`, which was then interpolated
     </ng-transclude>
   </some-thing>

   <!-- To (visually) get the old behavior, wrap top-level text-nodes on -->
   <!-- multi-slot transclusion directives into `<span>`; e.g.:          -->
   <some-thing>
     <span>{{</span>
     <very-important-content>Nooot</very-important-content>
     <span>'foo'}}</span>
   </some-thing>

   <!-- Result: -->
   <some-thing>
     <ng-transclude>
       <span>{{</span>       <-- Two separate
       <span>'foo'}}</span>  <-- nodes
     </ng-transclude>
   </some-thing>
   ```
@gkalpak
Copy link
Member

gkalpak commented Aug 25, 2016

@jshoudy11, #15025 has been merged. Feel free to try it out and report back 😄

@jshoudy11
Copy link
Author

@gkalpak is it only in 1.6 (as you originally suspected) or is it backported to 1.5.x as well?

@gkalpak
Copy link
Member

gkalpak commented Aug 26, 2016

1.6.x only.

@jshoudy11
Copy link
Author

@gkalpak will this ever be backported to 1.5.x? Our project is currently on 1.5 with no immediate plans to upgrade to 1.6. We have to continue to use the span work around unless this is backported

@jshoudy11
Copy link
Author

Other question: when will 1.6 be available in google3? Currently 1.5 is the latest version

@gkalpak
Copy link
Member

gkalpak commented Sep 6, 2016

@gkalpak will this ever be backported to 1.5.x?

Nope, because it is a breaking change.

Our project is currently on 1.5 with no immediate plans to upgrade to 1.6. We have to continue to use the span work around unless this is backported

Iirc you were using custom snapshot build, no? If that is true, you are already using 1.6 pre-alpha 😃

Other question: when will 1.6 be available in google3? Currently 1.5 is the latest version

It has to be released first, I guess (which will hopefully happen soon(-ish)).

Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.