updated DropDown workaround: load it as soon as possible and load also for iframes where possibile (same origin)

This commit is contained in:
Lorenzo Pichilli 2019-12-12 16:28:05 +01:00
parent 9b85ae9c01
commit c05e0e3963
6 changed files with 491 additions and 486 deletions

View File

@ -35,7 +35,7 @@
</value> </value>
</option> </option>
</component> </component>
<component name="ProjectRootManager" version="2" project-jdk-name="Android API 24 Platform" project-jdk-type="Android SDK" /> <component name="ProjectRootManager" version="2" project-jdk-name="Android API 28 Platform" project-jdk-type="Android SDK" />
<component name="ProjectType"> <component name="ProjectType">
<option name="id" value="io.flutter" /> <option name="id" value="io.flutter" />
</component> </component>

View File

@ -16,9 +16,13 @@
<component name="ChangeListManager"> <component name="ChangeListManager">
<list default="true" id="9b41f7a2-a71e-4923-91fb-249d7815b3e7" name="Default" comment=""> <list default="true" id="9b41f7a2-a71e-4923-91fb-249d7815b3e7" name="Default" comment="">
<change beforePath="$PROJECT_DIR$/.idea/workspace.xml" beforeDir="false" afterPath="$PROJECT_DIR$/.idea/workspace.xml" afterDir="false" /> <change beforePath="$PROJECT_DIR$/.idea/workspace.xml" beforeDir="false" afterPath="$PROJECT_DIR$/.idea/workspace.xml" afterDir="false" />
<change beforePath="$PROJECT_DIR$/README.md" beforeDir="false" afterPath="$PROJECT_DIR$/README.md" afterDir="false" />
<change beforePath="$PROJECT_DIR$/android/src/main/java/com/pichillilorenzo/flutter_inappwebview/InAppWebView/InAppWebView.java" beforeDir="false" afterPath="$PROJECT_DIR$/android/src/main/java/com/pichillilorenzo/flutter_inappwebview/InAppWebView/InAppWebView.java" afterDir="false" /> <change beforePath="$PROJECT_DIR$/android/src/main/java/com/pichillilorenzo/flutter_inappwebview/InAppWebView/InAppWebView.java" beforeDir="false" afterPath="$PROJECT_DIR$/android/src/main/java/com/pichillilorenzo/flutter_inappwebview/InAppWebView/InAppWebView.java" afterDir="false" />
<change beforePath="$PROJECT_DIR$/android/src/main/java/com/pichillilorenzo/flutter_inappwebview/InAppWebView/InAppWebViewClient.java" beforeDir="false" afterPath="$PROJECT_DIR$/android/src/main/java/com/pichillilorenzo/flutter_inappwebview/InAppWebView/InAppWebViewClient.java" afterDir="false" />
<change beforePath="$PROJECT_DIR$/example/ios/.symlinks/plugins/flutter_inappwebview/.idea/workspace.xml" beforeDir="false" afterPath="$PROJECT_DIR$/example/ios/.symlinks/plugins/flutter_inappwebview/.idea/workspace.xml" afterDir="false" /> <change beforePath="$PROJECT_DIR$/example/ios/.symlinks/plugins/flutter_inappwebview/.idea/workspace.xml" beforeDir="false" afterPath="$PROJECT_DIR$/example/ios/.symlinks/plugins/flutter_inappwebview/.idea/workspace.xml" afterDir="false" />
<change beforePath="$PROJECT_DIR$/example/ios/.symlinks/plugins/flutter_inappwebview/example/lib/main.dart" beforeDir="false" afterPath="$PROJECT_DIR$/example/ios/.symlinks/plugins/flutter_inappwebview/example/lib/main.dart" afterDir="false" /> <change beforePath="$PROJECT_DIR$/example/ios/.symlinks/plugins/flutter_inappwebview/example/lib/main.dart" beforeDir="false" afterPath="$PROJECT_DIR$/example/ios/.symlinks/plugins/flutter_inappwebview/example/lib/main.dart" afterDir="false" />
<change beforePath="$PROJECT_DIR$/example/lib/main.dart" beforeDir="false" afterPath="$PROJECT_DIR$/example/lib/main.dart" afterDir="false" />
<change beforePath="$PROJECT_DIR$/lib/src/webview_options.dart" beforeDir="false" afterPath="$PROJECT_DIR$/lib/src/webview_options.dart" afterDir="false" />
</list> </list>
<ignored path="$PROJECT_DIR$/.dart_tool/" /> <ignored path="$PROJECT_DIR$/.dart_tool/" />
<ignored path="$PROJECT_DIR$/.idea/" /> <ignored path="$PROJECT_DIR$/.idea/" />
@ -35,14 +39,14 @@
<component name="DefaultGradleProjectSettings"> <component name="DefaultGradleProjectSettings">
<option name="isMigrated" value="true" /> <option name="isMigrated" value="true" />
</component> </component>
<component name="ExecutionTargetManager" SELECTED_TARGET="Pixel_3_XL_API_24" /> <component name="ExecutionTargetManager" SELECTED_TARGET="LGH870776aa626" />
<component name="FileEditorManager"> <component name="FileEditorManager">
<leaf SIDE_TABS_SIZE_LIMIT_KEY="300"> <leaf SIDE_TABS_SIZE_LIMIT_KEY="300">
<file pinned="false" current-in-tab="true"> <file pinned="false" current-in-tab="true">
<entry file="file://$PROJECT_DIR$/example/lib/main.dart"> <entry file="file://$PROJECT_DIR$/example/lib/main.dart">
<provider selected="true" editor-type-id="text-editor"> <provider selected="true" editor-type-id="text-editor">
<state relative-caret-position="481"> <state relative-caret-position="224">
<caret line="78" column="1" selection-start-line="78" selection-start-column="1" selection-end-line="78" selection-end-column="1" /> <caret line="64" column="27" lean-forward="true" selection-start-line="64" selection-start-column="27" selection-end-line="64" selection-end-column="27" />
<folding> <folding>
<element signature="e#0#20#0" expanded="true" /> <element signature="e#0#20#0" expanded="true" />
</folding> </folding>
@ -65,8 +69,8 @@
<file pinned="false" current-in-tab="false"> <file pinned="false" current-in-tab="false">
<entry file="file://$PROJECT_DIR$/lib/src/webview_options.dart"> <entry file="file://$PROJECT_DIR$/lib/src/webview_options.dart">
<provider selected="true" editor-type-id="text-editor"> <provider selected="true" editor-type-id="text-editor">
<state relative-caret-position="373"> <state relative-caret-position="328">
<caret line="81" column="46" selection-start-line="81" selection-start-column="46" selection-end-line="81" selection-end-column="46" /> <caret line="412" column="102" lean-forward="true" selection-start-line="412" selection-start-column="5" selection-end-line="412" selection-end-column="102" />
<folding> <folding>
<element signature="e#0#17#0" expanded="true" /> <element signature="e#0#17#0" expanded="true" />
</folding> </folding>
@ -98,7 +102,7 @@
<file pinned="false" current-in-tab="false"> <file pinned="false" current-in-tab="false">
<entry file="file://$PROJECT_DIR$/lib/src/in_app_browser.dart"> <entry file="file://$PROJECT_DIR$/lib/src/in_app_browser.dart">
<provider selected="true" editor-type-id="text-editor"> <provider selected="true" editor-type-id="text-editor">
<state relative-caret-position="4575"> <state relative-caret-position="535">
<caret line="315" column="50" selection-start-line="315" selection-start-column="50" selection-end-line="315" selection-end-column="50" /> <caret line="315" column="50" selection-start-line="315" selection-start-column="50" selection-end-line="315" selection-end-column="50" />
<folding> <folding>
<element signature="e#0#20#0" expanded="true" /> <element signature="e#0#20#0" expanded="true" />
@ -110,8 +114,13 @@
<file pinned="false" current-in-tab="false"> <file pinned="false" current-in-tab="false">
<entry file="file://$PROJECT_DIR$/README.md"> <entry file="file://$PROJECT_DIR$/README.md">
<provider selected="true" editor-type-id="text-editor"> <provider selected="true" editor-type-id="text-editor">
<state relative-caret-position="6180"> <state relative-caret-position="48">
<caret line="412" column="3" selection-start-line="412" selection-start-column="3" selection-end-line="412" selection-end-column="21" /> <caret line="412" column="372" selection-start-line="412" selection-start-column="372" selection-end-line="412" selection-end-column="372" />
<folding>
<element signature="e#0#20#0" expanded="true" />
<element signature="e#0#39#0" expanded="true" />
<element signature="e#0#39#0" expanded="true" />
</folding>
</state> </state>
</provider> </provider>
</entry> </entry>
@ -137,7 +146,6 @@
</component> </component>
<component name="FindInProjectRecents"> <component name="FindInProjectRecents">
<findStrings> <findStrings>
<find>_staticChannel</find>
<find>defa</find> <find>defa</find>
<find>InAppWebViewStatic</find> <find>InAppWebViewStatic</find>
<find>removeFromSuperview</find> <find>removeFromSuperview</find>
@ -167,6 +175,7 @@
<find>\n</find> <find>\n</find>
<find>dropDownWorkaroundEnabled</find> <find>dropDownWorkaroundEnabled</find>
<find>debuggingEnabled</find> <find>debuggingEnabled</find>
<find>\n&quot; +</find>
</findStrings> </findStrings>
<replaceStrings> <replaceStrings>
<replace>activity.getPreferences(0)</replace> <replace>activity.getPreferences(0)</replace>
@ -176,6 +185,7 @@
<replace>Event fired</replace> <replace>Event fired</replace>
<replace>flutterInAppWebViewPlatformReady</replace> <replace>flutterInAppWebViewPlatformReady</replace>
<replace>flutter_inappwebview</replace> <replace>flutter_inappwebview</replace>
<replace>&quot; +</replace>
</replaceStrings> </replaceStrings>
<dirStrings> <dirStrings>
<dir>$PROJECT_DIR$/example/android</dir> <dir>$PROJECT_DIR$/example/android</dir>
@ -254,39 +264,6 @@
<foldersAlwaysOnTop value="true" /> <foldersAlwaysOnTop value="true" />
</navigator> </navigator>
<panes> <panes>
<pane id="ProjectPane">
<subPane>
<expand>
<path>
<item name="flutter_inappwebview" type="b2602c69:ProjectViewProjectNode" />
<item name="flutter_inappwebview" type="462c0819:PsiDirectoryNode" />
</path>
<path>
<item name="flutter_inappwebview" type="b2602c69:ProjectViewProjectNode" />
<item name="flutter_inappwebview" type="462c0819:PsiDirectoryNode" />
<item name="lib" type="462c0819:PsiDirectoryNode" />
</path>
<path>
<item name="flutter_inappwebview" type="b2602c69:ProjectViewProjectNode" />
<item name="flutter_inappwebview" type="462c0819:PsiDirectoryNode" />
<item name="lib" type="462c0819:PsiDirectoryNode" />
<item name="src" type="462c0819:PsiDirectoryNode" />
</path>
</expand>
<select />
</subPane>
</pane>
<pane id="Scope">
<subPane subId="Project Files">
<expand>
<path>
<item name="Root" type="cbb8eebc:String" user="Root" />
<item name="flutter_inappbrowser" type="cbb8eebc:String" user="flutter_inappbrowser" />
</path>
</expand>
<select />
</subPane>
</pane>
<pane id="PackagesPane" /> <pane id="PackagesPane" />
<pane id="AndroidView"> <pane id="AndroidView">
<subPane> <subPane>
@ -317,6 +294,39 @@
<select /> <select />
</subPane> </subPane>
</pane> </pane>
<pane id="Scope">
<subPane subId="Project Files">
<expand>
<path>
<item name="Root" type="cbb8eebc:String" user="Root" />
<item name="flutter_inappbrowser" type="cbb8eebc:String" user="flutter_inappbrowser" />
</path>
</expand>
<select />
</subPane>
</pane>
<pane id="ProjectPane">
<subPane>
<expand>
<path>
<item name="flutter_inappwebview" type="b2602c69:ProjectViewProjectNode" />
<item name="flutter_inappwebview" type="462c0819:PsiDirectoryNode" />
</path>
<path>
<item name="flutter_inappwebview" type="b2602c69:ProjectViewProjectNode" />
<item name="flutter_inappwebview" type="462c0819:PsiDirectoryNode" />
<item name="lib" type="462c0819:PsiDirectoryNode" />
</path>
<path>
<item name="flutter_inappwebview" type="b2602c69:ProjectViewProjectNode" />
<item name="flutter_inappwebview" type="462c0819:PsiDirectoryNode" />
<item name="lib" type="462c0819:PsiDirectoryNode" />
<item name="src" type="462c0819:PsiDirectoryNode" />
</path>
</expand>
<select />
</subPane>
</pane>
</panes> </panes>
</component> </component>
<component name="PropertiesComponent"> <component name="PropertiesComponent">
@ -466,7 +476,7 @@
</todo-panel> </todo-panel>
</component> </component>
<component name="ToolWindowManager"> <component name="ToolWindowManager">
<frame x="0" y="23" width="1920" height="1057" extended-state="6" /> <frame x="0" y="23" width="1920" height="1057" extended-state="0" />
<editor active="true" /> <editor active="true" />
<layout> <layout>
<window_info content_ui="combo" id="Project" order="0" sideWeight="0.6177474" visible="true" weight="0.1368477" /> <window_info content_ui="combo" id="Project" order="0" sideWeight="0.6177474" visible="true" weight="0.1368477" />
@ -480,7 +490,7 @@
<window_info id="Resources Explorer" order="8" /> <window_info id="Resources Explorer" order="8" />
<window_info anchor="bottom" id="Message" order="0" /> <window_info anchor="bottom" id="Message" order="0" />
<window_info anchor="bottom" id="Find" order="1" weight="0.32745314" /> <window_info anchor="bottom" id="Find" order="1" weight="0.32745314" />
<window_info active="true" anchor="bottom" id="Run" order="2" sideWeight="0.49478307" visible="true" weight="0.44455957" /> <window_info active="true" anchor="bottom" id="Run" order="2" sideWeight="0.49478307" visible="true" weight="0.38445595" />
<window_info anchor="bottom" id="Debug" order="3" weight="0.34196892" /> <window_info anchor="bottom" id="Debug" order="3" weight="0.34196892" />
<window_info anchor="bottom" id="Cvs" order="4" weight="0.25" /> <window_info anchor="bottom" id="Cvs" order="4" weight="0.25" />
<window_info anchor="bottom" id="Inspection" order="5" weight="0.4" /> <window_info anchor="bottom" id="Inspection" order="5" weight="0.4" />
@ -855,9 +865,16 @@
</state> </state>
</provider> </provider>
</entry> </entry>
<entry file="file://$PROJECT_DIR$/CHANGELOG.md">
<provider selected="true" editor-type-id="text-editor">
<state relative-caret-position="180">
<caret line="12" column="34" selection-start-line="12" selection-start-column="34" selection-end-line="12" selection-end-column="34" />
</state>
</provider>
</entry>
<entry file="file://$PROJECT_DIR$/lib/src/in_app_browser.dart"> <entry file="file://$PROJECT_DIR$/lib/src/in_app_browser.dart">
<provider selected="true" editor-type-id="text-editor"> <provider selected="true" editor-type-id="text-editor">
<state relative-caret-position="4575"> <state relative-caret-position="535">
<caret line="315" column="50" selection-start-line="315" selection-start-column="50" selection-end-line="315" selection-end-column="50" /> <caret line="315" column="50" selection-start-line="315" selection-start-column="50" selection-end-line="315" selection-end-column="50" />
<folding> <folding>
<element signature="e#0#20#0" expanded="true" /> <element signature="e#0#20#0" expanded="true" />
@ -867,22 +884,20 @@
</entry> </entry>
<entry file="file://$PROJECT_DIR$/README.md"> <entry file="file://$PROJECT_DIR$/README.md">
<provider selected="true" editor-type-id="text-editor"> <provider selected="true" editor-type-id="text-editor">
<state relative-caret-position="6180"> <state relative-caret-position="48">
<caret line="412" column="3" selection-start-line="412" selection-start-column="3" selection-end-line="412" selection-end-column="21" /> <caret line="412" column="372" selection-start-line="412" selection-start-column="372" selection-end-line="412" selection-end-column="372" />
</state> <folding>
</provider> <element signature="e#0#20#0" expanded="true" />
</entry> <element signature="e#0#39#0" expanded="true" />
<entry file="file://$PROJECT_DIR$/CHANGELOG.md"> <element signature="e#0#39#0" expanded="true" />
<provider selected="true" editor-type-id="text-editor"> </folding>
<state relative-caret-position="180">
<caret line="12" column="34" selection-start-line="12" selection-start-column="34" selection-end-line="12" selection-end-column="34" />
</state> </state>
</provider> </provider>
</entry> </entry>
<entry file="file://$PROJECT_DIR$/lib/src/webview_options.dart"> <entry file="file://$PROJECT_DIR$/lib/src/webview_options.dart">
<provider selected="true" editor-type-id="text-editor"> <provider selected="true" editor-type-id="text-editor">
<state relative-caret-position="373"> <state relative-caret-position="328">
<caret line="81" column="46" selection-start-line="81" selection-start-column="46" selection-end-line="81" selection-end-column="46" /> <caret line="412" column="102" lean-forward="true" selection-start-line="412" selection-start-column="5" selection-end-line="412" selection-end-column="102" />
<folding> <folding>
<element signature="e#0#17#0" expanded="true" /> <element signature="e#0#17#0" expanded="true" />
</folding> </folding>
@ -891,8 +906,8 @@
</entry> </entry>
<entry file="file://$PROJECT_DIR$/example/lib/main.dart"> <entry file="file://$PROJECT_DIR$/example/lib/main.dart">
<provider selected="true" editor-type-id="text-editor"> <provider selected="true" editor-type-id="text-editor">
<state relative-caret-position="481"> <state relative-caret-position="224">
<caret line="78" column="1" selection-start-line="78" selection-start-column="1" selection-end-line="78" selection-end-column="1" /> <caret line="64" column="27" lean-forward="true" selection-start-line="64" selection-start-column="27" selection-end-line="64" selection-end-column="27" />
<folding> <folding>
<element signature="e#0#20#0" expanded="true" /> <element signature="e#0#20#0" expanded="true" />
</folding> </folding>

View File

@ -410,7 +410,7 @@ Instead, on the `onLoadStop` WebView event, you can use `callHandler` directly:
* `hardwareAcceleration`: Boolean value to enable Hardware Acceleration in the WebView. * `hardwareAcceleration`: Boolean value to enable Hardware Acceleration in the WebView.
* `supportMultipleWindows`: Sets whether the WebView whether supports multiple windows. * `supportMultipleWindows`: Sets whether the WebView whether supports multiple windows.
* `regexToCancelSubFramesLoading`: Regular expression used by `shouldOverrideUrlLoading` event to cancel navigation for frames that are not the main frame. If the url request of a subframe matches the regular expression, then the request of that subframe is canceled. * `regexToCancelSubFramesLoading`: Regular expression used by `shouldOverrideUrlLoading` event to cancel navigation for frames that are not the main frame. If the url request of a subframe matches the regular expression, then the request of that subframe is canceled.
* `dropDownWorkaroundEnabled`: Enable a temporary workaround for html dropdowns (`<select>` tags) (available on Android 19+). It requires **JavaScript enabled**. It attempts to block click events for the dropdowns creating a custom `<div>` layer over the dropdown to intercept user's clicks. The default value is `false`. * `dropDownWorkaroundEnabled`: Enable a temporary workaround for html dropdowns (`<select>` tags) (available on Android 19+). It requires **JavaScript enabled**. It attempts to block click events for the dropdowns creating a custom `<div>` layer over the dropdown to intercept user's clicks. This workaround is applied as soon as the web page fires the `DOMContentLoaded` JavaScript event. The default value is `false`.
##### `InAppWebView` iOS-specific options ##### `InAppWebView` iOS-specific options

View File

@ -515,18 +515,20 @@ final public class InAppWebView extends InputAwareWebView {
"})(window.fetch);"; "})(window.fetch);";
// Android API 19+ // Android API 19+
// ResizeSensor: https://github.com/marcj/css-element-queries/blob/master/src/ResizeSensor.js
static final String dropDownWorkaroundJS = "(function() {" + static final String dropDownWorkaroundJS = "(function() {" +
" function dropDownWorkaround(window, document) {" +
" var uuid = 0;" + " var uuid = 0;" +
" function prepare() {" + " function prepare() {" +
" if (typeof window === \"undefined\") {" + " if (typeof window === 'undefined') {" +
" return null;" + " return null;" +
" }" + " }" +
" var globalWindow =" + " var globalWindow =" +
" typeof window != \"undefined\" && window.Math == Math" + " typeof window != 'undefined' && window.Math == Math" +
" ? window" + " ? window" +
" : typeof self != \"undefined\" && self.Math == Math" + " : typeof self != 'undefined' && self.Math == Math" +
" ? self" + " ? self" +
" : Function(\"return this\")();" + " : Function('return this')();" +
" var requestAnimationFrame =" + " var requestAnimationFrame =" +
" globalWindow.requestAnimationFrame ||" + " globalWindow.requestAnimationFrame ||" +
" globalWindow.mozRequestAnimationFrame ||" + " globalWindow.mozRequestAnimationFrame ||" +
@ -537,12 +539,12 @@ final public class InAppWebView extends InputAwareWebView {
" function forEachElement(elements, callback) {" + " function forEachElement(elements, callback) {" +
" var elementsType = Object.prototype.toString.call(elements);" + " var elementsType = Object.prototype.toString.call(elements);" +
" var isCollectionTyped =" + " var isCollectionTyped =" +
" \"[object Array]\" === elementsType ||" + " '[object Array]' === elementsType ||" +
" \"[object NodeList]\" === elementsType ||" + " '[object NodeList]' === elementsType ||" +
" \"[object HTMLCollection]\" === elementsType ||" + " '[object HTMLCollection]' === elementsType ||" +
" \"[object Object]\" === elementsType ||" + " '[object Object]' === elementsType ||" +
" (\"undefined\" !== typeof jQuery && elements instanceof jQuery) ||" + " ('undefined' !== typeof jQuery && elements instanceof jQuery) ||" +
" (\"undefined\" !== typeof Elements && elements instanceof Elements);" + " ('undefined' !== typeof Elements && elements instanceof Elements);" +
" var i = 0," + " var i = 0," +
" j = elements.length;" + " j = elements.length;" +
" if (isCollectionTyped) {" + " if (isCollectionTyped) {" +
@ -560,7 +562,6 @@ final public class InAppWebView extends InputAwareWebView {
" height: element.offsetHeight" + " height: element.offsetHeight" +
" };" + " };" +
" }" + " }" +
"" +
" var rect = element.getBoundingClientRect();" + " var rect = element.getBoundingClientRect();" +
" return {" + " return {" +
" width: Math.round(rect.width)," + " width: Math.round(rect.width)," +
@ -579,14 +580,12 @@ final public class InAppWebView extends InputAwareWebView {
" this.add = function(ev) {" + " this.add = function(ev) {" +
" q.push(ev);" + " q.push(ev);" +
" };" + " };" +
"" +
" var i, j;" + " var i, j;" +
" this.call = function(sizeInfo) {" + " this.call = function(sizeInfo) {" +
" for (i = 0, j = q.length; i < j; i++) {" + " for (i = 0, j = q.length; i < j; i++) {" +
" q[i].call(this, sizeInfo);" + " q[i].call(this, sizeInfo);" +
" }" + " }" +
" };" + " };" +
"" +
" this.remove = function(ev) {" + " this.remove = function(ev) {" +
" var newQueue = [];" + " var newQueue = [];" +
" for (i = 0, j = q.length; i < j; i++) {" + " for (i = 0, j = q.length; i < j; i++) {" +
@ -594,7 +593,6 @@ final public class InAppWebView extends InputAwareWebView {
" }" + " }" +
" q = newQueue;" + " q = newQueue;" +
" };" + " };" +
"" +
" this.length = function() {" + " this.length = function() {" +
" return q.length;" + " return q.length;" +
" };" + " };" +
@ -605,90 +603,74 @@ final public class InAppWebView extends InputAwareWebView {
" element.resizedAttached.add(resized);" + " element.resizedAttached.add(resized);" +
" return;" + " return;" +
" }" + " }" +
"" +
" element.resizedAttached = new EventQueue();" + " element.resizedAttached = new EventQueue();" +
" element.resizedAttached.add(resized);" + " element.resizedAttached.add(resized);" +
"" + " element.resizeSensor = document.createElement('div');" +
" element.resizeSensor = document.createElement(\"div\");" + " element.resizeSensor.dir = 'ltr';" +
" element.resizeSensor.dir = \"ltr\";" + " element.resizeSensor.className = 'resize-sensor';" +
" element.resizeSensor.className = \"resize-sensor\";" +
"" +
" var style = {" + " var style = {" +
" pointerEvents: \"none\"," + " pointerEvents: 'none'," +
" position: \"absolute\"," + " position: 'absolute'," +
" left: \"0px\"," + " left: '0px'," +
" top: \"0px\"," + " top: '0px'," +
" right: \"0px\"," + " right: '0px'," +
" bottom: \"0px\"," + " bottom: '0px'," +
" overflow: \"hidden\"," + " overflow: 'hidden'," +
" zIndex: \"-1\"," + " zIndex: '-1'," +
" visibility: \"hidden\"," + " visibility: 'hidden'," +
" maxWidth: \"100%\"" + " maxWidth: '100%'" +
" };" + " };" +
" var styleChild = {" + " var styleChild = {" +
" position: \"absolute\"," + " position: 'absolute'," +
" left: \"0px\"," + " left: '0px'," +
" top: \"0px\"," + " top: '0px'," +
" transition: \"0s\"" + " transition: '0s'" +
" };" + " };" +
"" +
" setStyle(element.resizeSensor, style);" + " setStyle(element.resizeSensor, style);" +
"" + " var expand = document.createElement('div');" +
" var expand = document.createElement(\"div\");" + " expand.className = 'resize-sensor-expand';" +
" expand.className = \"resize-sensor-expand\";" +
" setStyle(expand, style);" + " setStyle(expand, style);" +
"" + " var expandChild = document.createElement('div');" +
" var expandChild = document.createElement(\"div\");" +
" setStyle(expandChild, styleChild);" + " setStyle(expandChild, styleChild);" +
" expand.appendChild(expandChild);" + " expand.appendChild(expandChild);" +
"" + " var shrink = document.createElement('div');" +
" var shrink = document.createElement(\"div\");" + " shrink.className = 'resize-sensor-shrink';" +
" shrink.className = \"resize-sensor-shrink\";" +
" setStyle(shrink, style);" + " setStyle(shrink, style);" +
"" + " var shrinkChild = document.createElement('div');" +
" var shrinkChild = document.createElement(\"div\");" +
" setStyle(shrinkChild, styleChild);" + " setStyle(shrinkChild, styleChild);" +
" setStyle(shrinkChild, { width: \"200%\", height: \"200%\" });" + " setStyle(shrinkChild, { width: '200%', height: '200%' });" +
" shrink.appendChild(shrinkChild);" + " shrink.appendChild(shrinkChild);" +
"" +
" element.resizeSensor.appendChild(expand);" + " element.resizeSensor.appendChild(expand);" +
" element.resizeSensor.appendChild(shrink);" + " element.resizeSensor.appendChild(shrink);" +
" element.appendChild(element.resizeSensor);" + " element.appendChild(element.resizeSensor);" +
"" +
" var computedStyle = window.getComputedStyle(element);" + " var computedStyle = window.getComputedStyle(element);" +
" var position = computedStyle" + " var position = computedStyle" +
" ? computedStyle.getPropertyValue(\"position\")" + " ? computedStyle.getPropertyValue('position')" +
" : null;" + " : null;" +
" if (" + " if (" +
" \"absolute\" !== position &&" + " 'absolute' !== position &&" +
" \"relative\" !== position &&" + " 'relative' !== position &&" +
" \"fixed\" !== position &&" + " 'fixed' !== position &&" +
" \"sticky\" !== position" + " 'sticky' !== position" +
" ) {" + " ) {" +
" element.style.position = \"relative\";" + " element.style.position = 'relative';" +
" }" + " }" +
"" +
" var dirty, rafId;" + " var dirty, rafId;" +
" var size = getElementSize(element);" + " var size = getElementSize(element);" +
" var lastWidth = 0;" + " var lastWidth = 0;" +
" var lastHeight = 0;" + " var lastHeight = 0;" +
" var initialHiddenCheck = true;" + " var initialHiddenCheck = true;" +
" lastAnimationFrame = 0;" + " lastAnimationFrame = 0;" +
"" +
" var resetExpandShrink = function() {" + " var resetExpandShrink = function() {" +
" var width = element.offsetWidth;" + " var width = element.offsetWidth;" +
" var height = element.offsetHeight;" + " var height = element.offsetHeight;" +
"" + " expandChild.style.width = width + 10 + 'px';" +
" expandChild.style.width = width + 10 + \"px\";" + " expandChild.style.height = height + 10 + 'px';" +
" expandChild.style.height = height + 10 + \"px\";" +
"" +
" expand.scrollLeft = width + 10;" + " expand.scrollLeft = width + 10;" +
" expand.scrollTop = height + 10;" + " expand.scrollTop = height + 10;" +
"" +
" shrink.scrollLeft = width + 10;" + " shrink.scrollLeft = width + 10;" +
" shrink.scrollTop = height + 10;" + " shrink.scrollTop = height + 10;" +
" };" + " };" +
"" +
" var reset = function() {" + " var reset = function() {" +
" if (initialHiddenCheck) {" + " if (initialHiddenCheck) {" +
" var invisible =" + " var invisible =" +
@ -697,63 +679,48 @@ final public class InAppWebView extends InputAwareWebView {
" if (!lastAnimationFrame) {" + " if (!lastAnimationFrame) {" +
" lastAnimationFrame = requestAnimationFrame(function() {" + " lastAnimationFrame = requestAnimationFrame(function() {" +
" lastAnimationFrame = 0;" + " lastAnimationFrame = 0;" +
"" +
" reset();" + " reset();" +
" });" + " });" +
" }" + " }" +
"" +
" return;" + " return;" +
" } else {" + " } else {" +
" initialHiddenCheck = false;" + " initialHiddenCheck = false;" +
" }" + " }" +
" }" + " }" +
"" +
" resetExpandShrink();" + " resetExpandShrink();" +
" };" + " };" +
" element.resizeSensor.resetSensor = reset;" + " element.resizeSensor.resetSensor = reset;" +
"" +
" var onResized = function() {" + " var onResized = function() {" +
" rafId = 0;" + " rafId = 0;" +
"" +
" if (!dirty) return;" + " if (!dirty) return;" +
"" +
" lastWidth = size.width;" + " lastWidth = size.width;" +
" lastHeight = size.height;" + " lastHeight = size.height;" +
"" +
" if (element.resizedAttached) {" + " if (element.resizedAttached) {" +
" element.resizedAttached.call(size);" + " element.resizedAttached.call(size);" +
" }" + " }" +
" };" + " };" +
"" +
" var onScroll = function() {" + " var onScroll = function() {" +
" size = getElementSize(element);" + " size = getElementSize(element);" +
" dirty = size.width !== lastWidth || size.height !== lastHeight;" + " dirty = size.width !== lastWidth || size.height !== lastHeight;" +
"" +
" if (dirty && !rafId) {" + " if (dirty && !rafId) {" +
" rafId = requestAnimationFrame(onResized);" + " rafId = requestAnimationFrame(onResized);" +
" }" + " }" +
"" +
" reset();" + " reset();" +
" };" + " };" +
"" +
" var addEvent = function(el, name, cb) {" + " var addEvent = function(el, name, cb) {" +
" if (el.attachEvent) {" + " if (el.attachEvent) {" +
" el.attachEvent(\"on\" + name, cb);" + " el.attachEvent('on' + name, cb);" +
" } else {" + " } else {" +
" el.addEventListener(name, cb);" + " el.addEventListener(name, cb);" +
" }" + " }" +
" };" + " };" +
"" + " addEvent(expand, 'scroll', onScroll);" +
" addEvent(expand, \"scroll\", onScroll);" + " addEvent(shrink, 'scroll', onScroll);" +
" addEvent(shrink, \"scroll\", onScroll);" +
"" +
" lastAnimationFrame = requestAnimationFrame(reset);" + " lastAnimationFrame = requestAnimationFrame(reset);" +
" }" + " }" +
"" +
" forEachElement(element, function(elem) {" + " forEachElement(element, function(elem) {" +
" attachResizeEvent(elem, callback);" + " attachResizeEvent(elem, callback);" +
" });" + " });" +
"" +
" this.detach = function(ev) {" + " this.detach = function(ev) {" +
" if (!lastAnimationFrame) {" + " if (!lastAnimationFrame) {" +
" window.cancelAnimationFrame(lastAnimationFrame);" + " window.cancelAnimationFrame(lastAnimationFrame);" +
@ -761,22 +728,19 @@ final public class InAppWebView extends InputAwareWebView {
" }" + " }" +
" ResizeSensor.detach(element, ev);" + " ResizeSensor.detach(element, ev);" +
" };" + " };" +
"" +
" this.reset = function() {" + " this.reset = function() {" +
" element.resizeSensor.resetSensor();" + " element.resizeSensor.resetSensor();" +
" };" + " };" +
" };" + " };" +
"" +
" ResizeSensor.reset = function(element) {" + " ResizeSensor.reset = function(element) {" +
" forEachElement(element, function(elem) {" + " forEachElement(element, function(elem) {" +
" elem.resizeSensor.resetSensor();" + " elem.resizeSensor.resetSensor();" +
" });" + " });" +
" };" + " };" +
"" +
" ResizeSensor.detach = function(element, ev) {" + " ResizeSensor.detach = function(element, ev) {" +
" forEachElement(element, function(elem) {" + " forEachElement(element, function(elem) {" +
" if (!elem) return;" + " if (!elem) return;" +
" if (elem.resizedAttached && typeof ev === \"function\") {" + " if (elem.resizedAttached && typeof ev === 'function') {" +
" elem.resizedAttached.remove(ev);" + " elem.resizedAttached.remove(ev);" +
" if (elem.resizedAttached.length()) return;" + " if (elem.resizedAttached.length()) return;" +
" }" + " }" +
@ -789,8 +753,7 @@ final public class InAppWebView extends InputAwareWebView {
" }" + " }" +
" });" + " });" +
" };" + " };" +
"" + " if (typeof MutationObserver !== 'undefined') {" +
" if (typeof MutationObserver !== \"undefined\") {" +
" var observer = new MutationObserver(function(mutations) {" + " var observer = new MutationObserver(function(mutations) {" +
" for (var i in mutations) {" + " for (var i in mutations) {" +
" if (mutations.hasOwnProperty(i)) {" + " if (mutations.hasOwnProperty(i)) {" +
@ -803,33 +766,31 @@ final public class InAppWebView extends InputAwareWebView {
" }" + " }" +
" }" + " }" +
" });" + " });" +
"" + " document.addEventListener('DOMContentLoaded', function(event) {" +
" document.addEventListener(\"DOMContentLoaded\", function(event) {" +
" observer.observe(document.body, {" + " observer.observe(document.body, {" +
" childList: true," + " childList: true," +
" subtree: true" + " subtree: true" +
" });" + " });" +
" });" + " });" +
" }" + " }" +
"" +
" return ResizeSensor;" + " return ResizeSensor;" +
" }" + " }" +
" var ResizeSensor = prepare();" + " var ResizeSensor = prepare();" +
" function pseudoStyle(htmlElement, pseudoElement, styles) {" + " function pseudoStyle(htmlElement, pseudoElement, styles) {" +
" var _sheetId = \"pseudoStyles\";" + " var _sheetId = 'pseudoStyles';" +
" var _head = document.head || document.getElementsByTagName(\"head\")[0];" + " var _head = document.head || document.getElementsByTagName('head')[0];" +
" var _sheet =" + " var _sheet =" +
" document.getElementById(_sheetId) || document.createElement(\"style\");" + " document.getElementById(_sheetId) || document.createElement('style');" +
" _sheet.id = _sheetId;" + " _sheet.id = _sheetId;" +
" var styleClass = htmlElement.className.split(\" \").join(\".\");" + " var styleClass = htmlElement.className.split(' ').join('.');" +
" if (styleClass.charAt(0) != \".\") {" + " if (styleClass.charAt(0) != '.') {" +
" styleClass = \".\" + styleClass;" + " styleClass = '.' + styleClass;" +
" }" + " }" +
" _sheet.innerHTML += \"\n\" + styleClass + \":\" + pseudoElement + \"{\";" + " _sheet.innerHTML += '\\n' + styleClass + ':' + pseudoElement + '{';" +
" for (var key in styles) {" + " for (var key in styles) {" +
" _sheet.innerHTML += key + \":\" + styles[key] + \";\n\";" + " _sheet.innerHTML += key + ':' + styles[key] + ';\\n';" +
" }" + " }" +
" _sheet.innerHTML += \"}\";" + " _sheet.innerHTML += '}';" +
" _head.appendChild(_sheet);" + " _head.appendChild(_sheet);" +
" }" + " }" +
" function getIndexSelectValues(select) {" + " function getIndexSelectValues(select) {" +
@ -851,30 +812,35 @@ final public class InAppWebView extends InputAwareWebView {
" }" + " }" +
" }" + " }" +
" function addSpanWrapper(selectElement) {" + " function addSpanWrapper(selectElement) {" +
" var spanElement = document.createElement(\"span\");" + " var spanElement = document.createElement('span');" +
" spanElement.flutterInAppWebViewSelect = selectElement;" + " spanElement.flutterInAppWebViewSelect = selectElement;" +
" new ResizeSensor(spanElement, function() {" + " new ResizeSensor(spanElement, function() {" +
" var style = selectElement.currentStyle || window.getComputedStyle(selectElement);" + " var style =" +
" var xMargin = parseFloat(style.marginLeft) + parseFloat(style.marginRight);" + " selectElement.currentStyle || window.getComputedStyle(selectElement);" +
" var yMargin = parseFloat(style.marginTop) + parseFloat(style.marginBottom);" + " var xMargin =" +
" selectElement.style.width = (spanElement.getBoundingClientRect().width - xMargin) + \"px\";" + " parseFloat(style.marginLeft) + parseFloat(style.marginRight);" +
" selectElement.style.height = (spanElement.getBoundingClientRect().height - yMargin) + \"px\";" + " var yMargin =" +
" parseFloat(style.marginTop) + parseFloat(style.marginBottom);" +
" selectElement.style.width =" +
" spanElement.getBoundingClientRect().width - xMargin + 'px';" +
" selectElement.style.height =" +
" spanElement.getBoundingClientRect().height - yMargin + 'px';" +
" });" + " });" +
" spanElement.className = \"flutterInAppWebViewSelect-\" + uuid;" + " spanElement.className = 'flutterInAppWebViewSelect-' + uuid;" +
" spanElement.style.padding = \"0px\";" + " spanElement.style.padding = '0px';" +
" spanElement.style.margin = \"0px\";" + " spanElement.style.margin = '0px';" +
" spanElement.style.border = \"0px\";" + " spanElement.style.border = '0px';" +
" spanElement.style.display = \"inline-block\";" + " spanElement.style.display = 'inline-block';" +
" spanElement.style.backgroundColor = \"transparent\";" + " spanElement.style.backgroundColor = 'transparent';" +
" spanElement.style.position = \"relative\";" + " spanElement.style.position = 'relative';" +
" pseudoStyle(spanElement, \"before\", {" + " pseudoStyle(spanElement, 'before', {" +
" content: \"' '\"," + " content: '\\' \\''," +
" width: \"100%\"," + " width: '100%'," +
" height: \"100%\"," + " height: '100%'," +
" position: \"absolute\"," + " position: 'absolute'," +
" top: \"0\"," + " top: '0'," +
" left: \"0\"," + " left: '0'," +
" \"z-index\": 99999999" + " 'z-index': 99999999" +
" });" + " });" +
" selectElement.flutterInAppWebViewSpanWrapper = spanElement;" + " selectElement.flutterInAppWebViewSpanWrapper = spanElement;" +
" var clickEventListener = function(event) {" + " var clickEventListener = function(event) {" +
@ -887,7 +853,7 @@ final public class InAppWebView extends InputAwareWebView {
" this.flutterInAppWebViewSelect.focus();" + " this.flutterInAppWebViewSelect.focus();" +
" var options = [];" + " var options = [];" +
" var optionElements = this.flutterInAppWebViewSelect.querySelectorAll(" + " var optionElements = this.flutterInAppWebViewSelect.querySelectorAll(" +
" \"option\"" + " 'option'" +
" );" + " );" +
" for (var i = 0; i < optionElements.length; i++) {" + " for (var i = 0; i < optionElements.length; i++) {" +
" var optionElement = optionElements[i];" + " var optionElement = optionElements[i];" +
@ -897,8 +863,9 @@ final public class InAppWebView extends InputAwareWebView {
" });" + " });" +
" }" + " }" +
" var isMultiple = !!this.flutterInAppWebViewSelect.multiple;" + " var isMultiple = !!this.flutterInAppWebViewSelect.multiple;" +
" window." + JavaScriptBridgeInterface.name + ".callHandler(" + " window.flutter_inappwebview" +
" \"flutterInAppWebViewDropDownWorkaround\"," + " .callHandler(" +
" 'flutterInAppWebViewDropDownWorkaround'," +
" getIndexSelectValues(this.flutterInAppWebViewSelect)," + " getIndexSelectValues(this.flutterInAppWebViewSelect)," +
" isMultiple," + " isMultiple," +
" options" + " options" +
@ -910,7 +877,10 @@ final public class InAppWebView extends InputAwareWebView {
" self.flutterInAppWebViewSelect.value = result.values[0];" + " self.flutterInAppWebViewSelect.value = result.values[0];" +
" }" + " }" +
" } else {" + " } else {" +
" setMultipleValues(self.flutterInAppWebViewSelect, result.values);" + " setMultipleValues(" +
" self.flutterInAppWebViewSelect," +
" result.values" +
" );" +
" }" + " }" +
" }" + " }" +
" var evt = document.createEvent('HTMLEvents');" + " var evt = document.createEvent('HTMLEvents');" +
@ -965,6 +935,26 @@ final public class InAppWebView extends InputAwareWebView {
" childList: true," + " childList: true," +
" subtree: true" + " subtree: true" +
" });" + " });" +
" }" +
" document.addEventListener('DOMContentLoaded', function() {" +
" dropDownWorkaround(window, document);" +
" var iframes = document.querySelectorAll('iframe');" +
" for (var i = 0; i < iframes.length; i++) {" +
" var iframe = iframes[i];" +
" var frameDocument;" +
" try {" +
" frameDocument = iframe.contentDocument;" +
" if (!frameDocument && iframe.contentWindow) {" +
" frameDocument = iframe.contentWindow.document;" +
" }" +
" if (frameDocument != null) {" +
" dropDownWorkaround(window, frameDocument);" +
" }" +
" } catch (error) {" +
" console.error(error);" +
" }" +
" }" +
" });" +
"})();"; "})();";

View File

@ -180,6 +180,9 @@ public class InAppWebViewClient extends WebViewClient {
if (webView.options.useOnLoadResource) { if (webView.options.useOnLoadResource) {
js += InAppWebView.resourceObserverJS.replaceAll("[\r\n]+", ""); js += InAppWebView.resourceObserverJS.replaceAll("[\r\n]+", "");
} }
if (webView.options.dropDownWorkaroundEnabled && Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
js += InAppWebView.dropDownWorkaroundJS.replaceAll("[\r\n]+", "");
}
js += InAppWebView.printJS.replaceAll("[\r\n]+", ""); js += InAppWebView.printJS.replaceAll("[\r\n]+", "");
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
@ -226,10 +229,6 @@ public class InAppWebViewClient extends WebViewClient {
String js = InAppWebView.platformReadyJS.replaceAll("[\r\n]+", ""); String js = InAppWebView.platformReadyJS.replaceAll("[\r\n]+", "");
if (webView.options.dropDownWorkaroundEnabled) {
js += InAppWebView.dropDownWorkaroundJS.replaceAll("[\r\n]+", "");
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
webView.evaluateJavascript(js, (ValueCallback<String>) null); webView.evaluateJavascript(js, (ValueCallback<String>) null);
} else { } else {

View File

@ -410,6 +410,7 @@ class AndroidInAppWebViewOptions
///Enable a temporary workaround for html dropdowns (`<select>` tags). It requires **JavaScript enabled**. ///Enable a temporary workaround for html dropdowns (`<select>` tags). It requires **JavaScript enabled**.
///It attempts to block click events for the dropdowns creating a custom `<div>` layer over the dropdown to intercept user's clicks. ///It attempts to block click events for the dropdowns creating a custom `<div>` layer over the dropdown to intercept user's clicks.
///This workaround is applied as soon as the web page fires the `DOMContentLoaded` JavaScript event.
///The default value is `false`. ///The default value is `false`.
/// ///
///**NOTE**: available on Android 19+. ///**NOTE**: available on Android 19+.