├── app ├── .gitignore ├── src │ ├── main │ │ ├── res │ │ │ ├── menu │ │ │ │ └── menu_main.xml │ │ │ ├── mipmap-hdpi │ │ │ │ └── ic_launcher.png │ │ │ ├── mipmap-mdpi │ │ │ │ └── ic_launcher.png │ │ │ ├── mipmap-xhdpi │ │ │ │ └── ic_launcher.png │ │ │ ├── mipmap-xxhdpi │ │ │ │ └── ic_launcher.png │ │ │ ├── values │ │ │ │ ├── styles.xml │ │ │ │ ├── dimens.xml │ │ │ │ └── strings.xml │ │ │ ├── values-w820dp │ │ │ │ └── dimens.xml │ │ │ └── layout │ │ │ │ ├── activity_real_library.xml │ │ │ │ ├── activity_stress_test.xml │ │ │ │ ├── activity_line_endings.xml │ │ │ │ ├── activity_evaluate_js_string.xml │ │ │ │ ├── activity_destroy.xml │ │ │ │ ├── activity_percentage.xml │ │ │ │ ├── activity_call_js_function.xml │ │ │ │ ├── activity_character_escape.xml │ │ │ │ └── activity_main.xml │ │ ├── java │ │ │ └── com │ │ │ │ └── evgenii │ │ │ │ └── jsevaluatortests │ │ │ │ ├── JSUtils.java │ │ │ │ ├── LineEndingsActivity.java │ │ │ │ ├── DestroyActivity.java │ │ │ │ ├── PercentageActivity.java │ │ │ │ ├── EvaluateJsStringActivity.java │ │ │ │ ├── MainActivity.java │ │ │ │ ├── CallJsFunctionActivity.java │ │ │ │ ├── CharacterEscape.java │ │ │ │ ├── StressTestActivity.java │ │ │ │ └── RealLibrary.java │ │ ├── AndroidManifest.xml │ │ └── assets │ │ │ └── real_library │ │ │ └── cryptojs.js │ └── androidTest │ │ └── java │ │ └── com │ │ └── evgenii │ │ └── jsevaluatortests │ │ ├── mocks │ │ ├── HandlerWrapperMock.java │ │ ├── CallJavaResultInterfaceMock.java │ │ ├── JsCallbackMock.java │ │ └── WebViewWrapperMock.java │ │ ├── JavaScriptInterfaceTest.java │ │ ├── RealLibraryTests.java │ │ ├── StressTestActivityTests.java │ │ ├── LineEndingsActivityTests.java │ │ ├── JsFunctionCallFormatterTests.java │ │ ├── CharacterEscapeTests.java │ │ ├── DestroyActivityTests.java │ │ ├── EvaluateJsActivityTests.java │ │ ├── PercentageActivityTest.java │ │ ├── WebViewWrapperTest.java │ │ ├── CallJsFunctionActivityTests.java │ │ └── JsEvaluatorTests.java ├── proguard-rules.pro └── build.gradle ├── settings.gradle ├── jsevaluator ├── src │ └── main │ │ ├── AndroidManifest.xml │ │ ├── res │ │ └── values │ │ │ └── strings.xml │ │ └── java │ │ └── com │ │ └── evgenii │ │ └── jsevaluator │ │ ├── interfaces │ │ ├── HandlerWrapperInterface.java │ │ ├── CallJavaResultInterface.java │ │ ├── JsCallback.java │ │ ├── WebViewWrapperInterface.java │ │ └── JsEvaluatorInterface.java │ │ ├── HandlerWrapper.java │ │ ├── JavaScriptInterface.java │ │ ├── JsFunctionCallFormatter.java │ │ ├── WebViewWrapper.java │ │ └── JsEvaluator.java ├── consumer-proguard-rules.pro ├── .gitignore ├── proguard-rules.pro └── build.gradle ├── .gitignore ├── js_evaluator_screenshot_2.png ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── js_evaluator_screenshot_1_2014_08_30.png ├── CHANGELOG.md ├── LICENSE ├── gradle.properties ├── gradlew.bat ├── gradlew └── README.md /app/.gitignore: -------------------------------------------------------------------------------- 1 | /build 2 | -------------------------------------------------------------------------------- /settings.gradle: -------------------------------------------------------------------------------- 1 | include ':app', ':jsevaluator' 2 | -------------------------------------------------------------------------------- /jsevaluator/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .gradle 2 | /local.properties 3 | /.idea 4 | .DS_Store 5 | /build 6 | *.iml 7 | -------------------------------------------------------------------------------- /app/src/main/res/menu/menu_main.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | -------------------------------------------------------------------------------- /js_evaluator_screenshot_2.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/evgenyneu/js-evaluator-for-android/HEAD/js_evaluator_screenshot_2.png -------------------------------------------------------------------------------- /jsevaluator/src/main/res/values/strings.xml: -------------------------------------------------------------------------------- 1 | 2 | JSEvaluator 3 | 4 | -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/evgenyneu/js-evaluator-for-android/HEAD/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /js_evaluator_screenshot_1_2014_08_30.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/evgenyneu/js-evaluator-for-android/HEAD/js_evaluator_screenshot_1_2014_08_30.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-hdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/evgenyneu/js-evaluator-for-android/HEAD/app/src/main/res/mipmap-hdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-mdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/evgenyneu/js-evaluator-for-android/HEAD/app/src/main/res/mipmap-mdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/evgenyneu/js-evaluator-for-android/HEAD/app/src/main/res/mipmap-xhdpi/ic_launcher.png -------------------------------------------------------------------------------- /app/src/main/res/mipmap-xxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/evgenyneu/js-evaluator-for-android/HEAD/app/src/main/res/mipmap-xxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /jsevaluator/consumer-proguard-rules.pro: -------------------------------------------------------------------------------- 1 | -keepattributes JavascriptInterface 2 | -keepclassmembers class * { 3 | @android.webkit.JavascriptInterface ; 4 | } -------------------------------------------------------------------------------- /jsevaluator/.gitignore: -------------------------------------------------------------------------------- 1 | # Exclude everything from build directory except the files in build/outputs/aar/ 2 | 3 | build/* 4 | !build/outputs/ 5 | 6 | build/outputs/* 7 | !build/outputs/aar/ 8 | *.iml -------------------------------------------------------------------------------- /jsevaluator/src/main/java/com/evgenii/jsevaluator/interfaces/HandlerWrapperInterface.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluator.interfaces; 2 | 3 | public interface HandlerWrapperInterface { 4 | public void post(Runnable r); 5 | } 6 | -------------------------------------------------------------------------------- /app/src/main/res/values/styles.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /app/src/main/res/values/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 16dp 4 | 16dp 5 | 6 | -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | distributionBase=GRADLE_USER_HOME 2 | distributionPath=wrapper/dists 3 | distributionUrl=https\://services.gradle.org/distributions/gradle-7.6-bin.zip 4 | networkTimeout=10000 5 | zipStoreBase=GRADLE_USER_HOME 6 | zipStorePath=wrapper/dists 7 | -------------------------------------------------------------------------------- /jsevaluator/src/main/java/com/evgenii/jsevaluator/interfaces/CallJavaResultInterface.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluator.interfaces; 2 | 3 | /** 4 | * Used in JavaScriptInterface to interact with JsRunner 5 | */ 6 | public interface CallJavaResultInterface { 7 | public void jsCallFinished(String value); 8 | } 9 | -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | # JsEvaluator version history 2 | 3 | ## v2.0.0 (2016-12-17) 4 | 5 | Added ability to catch JavaScript runtime errors ([codebymikey](https://github.com/codebymikey)). 6 | 7 | ## v1.0.5 (2016-05-22) 8 | 9 | Added destroy() and getWebView() methods. 10 | 11 | ## v1.0.4 (2016-04-20) 12 | 13 | Changed minSdkVersion to 14 (from 15). 14 | 15 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/mocks/HandlerWrapperMock.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests.mocks; 2 | 3 | import com.evgenii.jsevaluator.interfaces.HandlerWrapperInterface; 4 | 5 | public class HandlerWrapperMock implements HandlerWrapperInterface { 6 | @Override 7 | public void post(Runnable runnable) { 8 | runnable.run(); 9 | } 10 | } 11 | -------------------------------------------------------------------------------- /jsevaluator/src/main/java/com/evgenii/jsevaluator/interfaces/JsCallback.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluator.interfaces; 2 | 3 | /** 4 | * Interface for passing code that will be executed after the JS has finished 5 | */ 6 | 7 | public interface JsCallback { 8 | public abstract void onResult(String value); 9 | public abstract void onError(String errorMessage); 10 | } 11 | -------------------------------------------------------------------------------- /app/src/main/res/values-w820dp/dimens.xml: -------------------------------------------------------------------------------- 1 | 2 | 5 | 64dp 6 | 7 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/mocks/CallJavaResultInterfaceMock.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests.mocks; 2 | 3 | import com.evgenii.jsevaluator.interfaces.CallJavaResultInterface; 4 | 5 | public class CallJavaResultInterfaceMock implements CallJavaResultInterface { 6 | public String jsCallFinished_paramValue; 7 | 8 | @Override 9 | public void jsCallFinished(String value) { 10 | jsCallFinished_paramValue = value; 11 | } 12 | } 13 | -------------------------------------------------------------------------------- /jsevaluator/src/main/java/com/evgenii/jsevaluator/HandlerWrapper.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluator; 2 | 3 | import android.os.Handler; 4 | 5 | import com.evgenii.jsevaluator.interfaces.HandlerWrapperInterface; 6 | 7 | public class HandlerWrapper implements HandlerWrapperInterface { 8 | private final Handler mHandler; 9 | 10 | public HandlerWrapper() { 11 | mHandler = new Handler(); 12 | } 13 | 14 | @Override 15 | public void post(Runnable r) { 16 | mHandler.post(r); 17 | } 18 | } 19 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/mocks/JsCallbackMock.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests.mocks; 2 | 3 | import com.evgenii.jsevaluator.interfaces.JsCallback; 4 | 5 | public class JsCallbackMock implements JsCallback { 6 | public String resultValue; 7 | public String errorMessage; 8 | 9 | @Override 10 | public void onResult(String value) { 11 | resultValue = value; 12 | } 13 | 14 | @Override 15 | public void onError(String message) { 16 | errorMessage = message; 17 | } 18 | } 19 | -------------------------------------------------------------------------------- /jsevaluator/src/main/java/com/evgenii/jsevaluator/interfaces/WebViewWrapperInterface.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluator.interfaces; 2 | 3 | import android.webkit.WebView; 4 | 5 | public interface WebViewWrapperInterface { 6 | public void loadJavaScript(String javascript); 7 | 8 | // Destroys the web view in order to free the memory. 9 | // The web view can not be accessed after is has been destroyed. 10 | public void destroy(); 11 | 12 | // Returns the WebView object 13 | public WebView getWebView(); 14 | } 15 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/mocks/WebViewWrapperMock.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests.mocks; 2 | 3 | import java.util.ArrayList; 4 | 5 | import com.evgenii.jsevaluator.interfaces.WebViewWrapperInterface; 6 | import android.webkit.WebView; 7 | 8 | 9 | public class WebViewWrapperMock implements WebViewWrapperInterface { 10 | public ArrayList mLoadedJavaScript = new ArrayList(); 11 | 12 | @Override 13 | public void loadJavaScript(String javascript) { 14 | mLoadedJavaScript.add(javascript); 15 | } 16 | 17 | @Override 18 | public void destroy() { } 19 | 20 | @Override 21 | public WebView getWebView() { return null; } 22 | } 23 | -------------------------------------------------------------------------------- /jsevaluator/src/main/java/com/evgenii/jsevaluator/interfaces/JsEvaluatorInterface.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluator.interfaces; 2 | 3 | import android.webkit.WebView; 4 | 5 | public interface JsEvaluatorInterface { 6 | public void callFunction(String jsCode, JsCallback resultCallback, String name, Object... args); 7 | 8 | public void evaluate(String jsCode); 9 | 10 | public void evaluate(String jsCode, JsCallback resultCallback); 11 | 12 | // Destroys the web view in order to free the memory. 13 | // The web view can not be accessed after is has been destroyed. 14 | public void destroy(); 15 | 16 | // Returns the WebView object 17 | public WebView getWebView(); 18 | } 19 | -------------------------------------------------------------------------------- /app/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # By default, the flags in this file are appended to flags specified 3 | # in /Users/evgenii/Library/Android/sdk/tools/proguard/proguard-android.txt 4 | # You can edit the include path and order by changing the proguardFiles 5 | # directive in build.gradle. 6 | # 7 | # For more details, see 8 | # http://developer.android.com/guide/developing/tools/proguard.html 9 | 10 | # Add any project specific keep options here: 11 | 12 | # If your project uses WebView with JS, uncomment the following 13 | # and specify the fully qualified class name to the JavaScript interface 14 | # class: 15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 16 | # public *; 17 | #} 18 | -------------------------------------------------------------------------------- /jsevaluator/src/main/java/com/evgenii/jsevaluator/JavaScriptInterface.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluator; 2 | 3 | import android.webkit.JavascriptInterface; 4 | 5 | import com.evgenii.jsevaluator.interfaces.CallJavaResultInterface; 6 | 7 | /** 8 | * Passed in addJavascriptInterface of WebView to allow web views's JS execute 9 | * Java code 10 | */ 11 | public class JavaScriptInterface { 12 | private final CallJavaResultInterface mCallJavaResultInterface; 13 | 14 | public JavaScriptInterface(CallJavaResultInterface callJavaResult) { 15 | mCallJavaResultInterface = callJavaResult; 16 | } 17 | 18 | @JavascriptInterface 19 | public void returnResultToJava(String value) { 20 | mCallJavaResultInterface.jsCallFinished(value); 21 | } 22 | } -------------------------------------------------------------------------------- /app/src/main/java/com/evgenii/jsevaluatortests/JSUtils.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests; 2 | 3 | import android.graphics.Color; 4 | import android.graphics.Typeface; 5 | import android.text.Spannable; 6 | import android.text.SpannableString; 7 | import android.text.Spanned; 8 | import android.text.style.ForegroundColorSpan; 9 | import android.text.style.StyleSpan; 10 | 11 | class JSUtils { 12 | 13 | static SpannableString getErrorSpan(String errorMessage){ 14 | SpannableString span = new SpannableString("Error: " + errorMessage); 15 | span.setSpan(new StyleSpan(Typeface.BOLD), 0, 6, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE); 16 | span.setSpan(new ForegroundColorSpan(Color.RED), 0, span.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE); 17 | return span; 18 | } 19 | } 20 | -------------------------------------------------------------------------------- /app/build.gradle: -------------------------------------------------------------------------------- 1 | apply plugin: 'com.android.application' 2 | 3 | android { 4 | compileSdkVersion 33 5 | 6 | defaultConfig { 7 | applicationId "com.evgenii.jsevaluatortests" 8 | namespace "com.evgenii.jsevaluatortests" 9 | minSdkVersion 14 10 | targetSdkVersion 33 11 | versionCode 1 12 | versionName "1.0" 13 | } 14 | buildTypes { 15 | release { 16 | minifyEnabled false 17 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' 18 | } 19 | } 20 | 21 | useLibrary 'android.test.runner' 22 | useLibrary 'android.test.base' 23 | } 24 | 25 | dependencies { 26 | implementation fileTree(include: ['*.jar'], dir: 'libs') 27 | implementation 'androidx.appcompat:appcompat:1.5.1' 28 | implementation project(':jsevaluator') 29 | } 30 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/JavaScriptInterfaceTest.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests; 2 | 3 | import android.test.AndroidTestCase; 4 | 5 | import com.evgenii.jsevaluator.JavaScriptInterface; 6 | import com.evgenii.jsevaluatortests.mocks.CallJavaResultInterfaceMock; 7 | 8 | public class JavaScriptInterfaceTest extends AndroidTestCase { 9 | protected JavaScriptInterface mJavaScriptInterface; 10 | protected CallJavaResultInterfaceMock mCallJavaResultInterfaceMock; 11 | 12 | public void testReturnResultToJava() { 13 | mCallJavaResultInterfaceMock = new CallJavaResultInterfaceMock(); 14 | mJavaScriptInterface = new JavaScriptInterface( 15 | mCallJavaResultInterfaceMock); 16 | 17 | mJavaScriptInterface.returnResultToJava("test value"); 18 | assertEquals("test value", 19 | mCallJavaResultInterfaceMock.jsCallFinished_paramValue); 20 | } 21 | } -------------------------------------------------------------------------------- /jsevaluator/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # Add project specific ProGuard rules here. 2 | # By default, the flags in this file are appended to flags specified 3 | # in /Users/evgenii/Library/Android/sdk/tools/proguard/proguard-android.txt 4 | # You can edit the include path and order by changing the proguardFiles 5 | # directive in build.gradle. 6 | # 7 | # For more details, see 8 | # http://developer.android.com/guide/developing/tools/proguard.html 9 | 10 | # Add any project specific keep options here: 11 | 12 | # If your project uses WebView with JS, uncomment the following 13 | # and specify the fully qualified class name to the JavaScript interface 14 | # class: 15 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 16 | # public *; 17 | #} 18 | 19 | # js-evaluator-for-android 20 | -keepattributes JavascriptInterface 21 | -keepclassmembers class * { 22 | @android.webkit.JavascriptInterface ; 23 | } 24 | -------------------------------------------------------------------------------- /jsevaluator/build.gradle: -------------------------------------------------------------------------------- 1 | apply plugin: 'com.android.library' 2 | 3 | android { 4 | compileSdkVersion 33 5 | 6 | defaultConfig { 7 | namespace "com.evgenii.jsevaluator" 8 | minSdkVersion 14 9 | targetSdkVersion 33 10 | versionCode 1 11 | versionName "1.0" 12 | version="1.0" 13 | archivesBaseName = "jsevaluator" 14 | consumerProguardFiles 'consumer-proguard-rules.pro' 15 | } 16 | buildTypes { 17 | release { 18 | minifyEnabled false 19 | proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' 20 | } 21 | } 22 | 23 | libraryVariants.all { variant -> 24 | variant.outputs.all { output -> 25 | if (outputFileName.endsWith('.aar')) { 26 | outputFileName = "${archivesBaseName}-${version}.aar" 27 | } 28 | } 29 | } 30 | } 31 | 32 | dependencies { 33 | implementation fileTree(dir: 'libs', include: ['*.jar']) 34 | implementation 'androidx.appcompat:appcompat:1.5.1' 35 | } 36 | -------------------------------------------------------------------------------- /jsevaluator/src/main/java/com/evgenii/jsevaluator/JsFunctionCallFormatter.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluator; 2 | 3 | public class JsFunctionCallFormatter { 4 | public static String paramToString(Object param) { 5 | String str = ""; 6 | if (param instanceof String) { 7 | str = (String) param; 8 | str = str.replace("\\", "\\\\"); 9 | str = str.replace("\"", "\\\""); 10 | str = str.replace("\n", "\\n"); 11 | str = String.format("\"%s\"", str); 12 | } else { 13 | try { 14 | @SuppressWarnings("unused") 15 | final double d = Double.parseDouble(param.toString()); 16 | str = param.toString(); 17 | } catch (final NumberFormatException nfe) { 18 | } 19 | } 20 | 21 | return str; 22 | } 23 | 24 | public static String toString(String functionName, Object... args) { 25 | final StringBuilder paramsStr = new StringBuilder(); 26 | 27 | for (final Object param : args) { 28 | if (paramsStr.length() > 0) { 29 | paramsStr.append(", "); 30 | } 31 | 32 | paramsStr.append(paramToString(param)); 33 | } 34 | 35 | return String.format("%s(%s)", functionName, paramsStr); 36 | } 37 | } 38 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2014 Evgenii Neumerzhitckii 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining 6 | a copy of this software and associated documentation files (the 7 | "Software"), to deal in the Software without restriction, including 8 | without limitation the rights to use, copy, modify, merge, publish, 9 | distribute, sublicense, and/or sell copies of the Software, and to 10 | permit persons to whom the Software is furnished to do so, subject to 11 | the following conditions: 12 | 13 | The above copyright notice and this permission notice shall be 14 | included in all copies or substantial portions of the Software. 15 | 16 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 17 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 18 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 19 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE 20 | LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION 21 | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION 22 | WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_real_library.xml: -------------------------------------------------------------------------------- 1 | 10 | 11 | 16 | 17 | 25 | 26 | 27 | -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_stress_test.xml: -------------------------------------------------------------------------------- 1 | 10 | 11 | 16 | 17 | 25 | 26 | 27 | 28 | 29 | -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_line_endings.xml: -------------------------------------------------------------------------------- 1 | 10 | 11 | 12 | 17 | 18 | 26 | 27 | 28 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/RealLibraryTests.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests; 2 | 3 | import android.test.ActivityInstrumentationTestCase2; 4 | import android.test.suitebuilder.annotation.MediumTest; 5 | import android.widget.TextView; 6 | 7 | public class RealLibraryTests extends ActivityInstrumentationTestCase2 { 8 | 9 | private RealLibrary mActivity; 10 | 11 | public RealLibraryTests() { 12 | super(RealLibrary.class); 13 | } 14 | 15 | @Override 16 | protected void setUp() throws Exception { 17 | super.setUp(); 18 | mActivity = getActivity(); 19 | } 20 | 21 | public void testPreconditions() { 22 | assertNotNull("mActivity is null", mActivity); 23 | } 24 | 25 | public void testTestRealLibraries() throws InterruptedException { 26 | final TextView resultTextView = (TextView) mActivity.findViewById(R.id.realLibraryResult); 27 | 28 | final String expectedResult = "Result: jQuery is working! CryptoJs is working!"; 29 | 30 | for (int i = 0; i < 100; i++) { 31 | Thread.sleep(100); 32 | if (resultTextView.getText().toString().equals(expectedResult)) { 33 | break; 34 | } 35 | } 36 | assertEquals(expectedResult, resultTextView.getText().toString()); 37 | } 38 | } -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/StressTestActivityTests.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests; 2 | 3 | import android.test.ActivityInstrumentationTestCase2; 4 | import android.test.suitebuilder.annotation.MediumTest; 5 | import android.view.View; 6 | import android.widget.TextView; 7 | 8 | public class StressTestActivityTests extends ActivityInstrumentationTestCase2 { 9 | 10 | private StressTestActivity mActivity; 11 | 12 | public StressTestActivityTests() { 13 | super(StressTestActivity.class); 14 | } 15 | 16 | @Override 17 | protected void setUp() throws Exception { 18 | super.setUp(); 19 | mActivity = getActivity(); 20 | } 21 | 22 | @MediumTest 23 | public void test_runsJSAndShowsResult() throws InterruptedException { 24 | final TextView resultTextView = (TextView) mActivity 25 | .findViewById(R.id.js_stress_result_text_view); 26 | 27 | assertTrue(View.VISIBLE == resultTextView.getVisibility()); 28 | 29 | final String expectedResult = "Result: 44000"; 30 | 31 | for (int i = 0; i < 100; i++) { 32 | Thread.sleep(100); 33 | if (resultTextView.getText().toString().equals(expectedResult)) { 34 | break; 35 | } 36 | } 37 | assertEquals(expectedResult, resultTextView.getText().toString()); 38 | } 39 | } 40 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/LineEndingsActivityTests.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests; 2 | 3 | import android.test.ActivityInstrumentationTestCase2; 4 | import android.test.suitebuilder.annotation.MediumTest; 5 | import android.widget.TextView; 6 | 7 | public class LineEndingsActivityTests extends ActivityInstrumentationTestCase2 { 8 | 9 | private LineEndingsActivity mActivity; 10 | 11 | public LineEndingsActivityTests() { 12 | super(LineEndingsActivity.class); 13 | } 14 | 15 | @Override 16 | protected void setUp() throws Exception { 17 | super.setUp(); 18 | mActivity = getActivity(); 19 | } 20 | 21 | public void testPreconditions() { 22 | assertNotNull("mActivity is null", mActivity); 23 | } 24 | 25 | @MediumTest 26 | public void testTestLineEndings() throws InterruptedException { 27 | final TextView resultTextView = (TextView) mActivity 28 | .findViewById(R.id.lineEndingsViewResult); 29 | 30 | final String expectedResult = "Result: success!"; 31 | 32 | for (int i = 0; i < 100; i++) { 33 | Thread.sleep(100); 34 | if (resultTextView.getText().toString().equals(expectedResult)) { 35 | break; 36 | } 37 | } 38 | 39 | assertEquals(expectedResult, resultTextView.getText().toString()); 40 | } 41 | } 42 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/JsFunctionCallFormatterTests.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests; 2 | 3 | import android.test.AndroidTestCase; 4 | 5 | import com.evgenii.jsevaluator.JsFunctionCallFormatter; 6 | 7 | public class JsFunctionCallFormatterTests extends AndroidTestCase { 8 | public void testParamToString_float() { 9 | assertEquals("123.22", JsFunctionCallFormatter.paramToString(123.22)); 10 | } 11 | 12 | public void testParamToString_integer() { 13 | assertEquals("123", JsFunctionCallFormatter.paramToString(123)); 14 | } 15 | 16 | public void testParamToString_string() { 17 | assertEquals("\"Boy's bike\"", JsFunctionCallFormatter.paramToString("Boy's bike")); 18 | } 19 | 20 | public void testParamToString_stringBackslash() { 21 | assertEquals("\"\\\\\"", JsFunctionCallFormatter.paramToString("\\")); 22 | } 23 | 24 | public void testParamToString_stringEscapesDoubleQuotes() { 25 | assertEquals("\"\\\"\"", JsFunctionCallFormatter.paramToString("\"")); 26 | } 27 | 28 | public void testParamToString_stringEscapesNewLine() { 29 | assertEquals("\"\\n\"", JsFunctionCallFormatter.paramToString("\n")); 30 | } 31 | 32 | public void testToString() { 33 | final String js = JsFunctionCallFormatter.toString("drink", "Cow's milk", 5); 34 | 35 | assertEquals("drink(\"Cow's milk\", 5)", js); 36 | 37 | } 38 | } 39 | -------------------------------------------------------------------------------- /app/src/main/java/com/evgenii/jsevaluatortests/LineEndingsActivity.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests; 2 | 3 | import android.app.Activity; 4 | import android.os.Bundle; 5 | import android.widget.TextView; 6 | 7 | import com.evgenii.jsevaluator.JsEvaluator; 8 | import com.evgenii.jsevaluator.interfaces.JsCallback; 9 | 10 | public class LineEndingsActivity extends Activity { 11 | JsEvaluator mJsEvaluator; 12 | 13 | protected void evaluate() { 14 | // Use CR, LF characters and their combinations 15 | mJsEvaluator.evaluate("1;\r\n2;\n3;\r4;\n\r5;", new JsCallback() { 16 | @Override 17 | public void onResult(final String resultValue) { 18 | final TextView jsResultTextView = (TextView) findViewById(R.id.lineEndingsViewResult); 19 | String resultStr = "failed"; 20 | if (resultValue.equals("5")) { 21 | resultStr = "success!"; 22 | } 23 | 24 | jsResultTextView.setText(String.format("Result: %s", resultStr)); 25 | } 26 | 27 | @Override 28 | public void onError(String errorMessage) { 29 | final TextView jsResultTextView = (TextView) findViewById(R.id.lineEndingsViewResult); 30 | jsResultTextView.setText(JSUtils.getErrorSpan(errorMessage)); 31 | } 32 | }); 33 | } 34 | 35 | @Override 36 | protected void onCreate(Bundle savedInstanceState) { 37 | super.onCreate(savedInstanceState); 38 | setContentView(R.layout.activity_line_endings); 39 | 40 | // Show the Up button in the action bar. 41 | getActionBar().setDisplayHomeAsUpEnabled(true); 42 | 43 | mJsEvaluator = new JsEvaluator(this); 44 | 45 | evaluate(); 46 | } 47 | } -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/CharacterEscapeTests.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests; 2 | 3 | import android.test.ActivityInstrumentationTestCase2; 4 | import android.test.TouchUtils; 5 | import android.test.suitebuilder.annotation.MediumTest; 6 | import android.widget.Button; 7 | import android.widget.TextView; 8 | 9 | public class CharacterEscapeTests extends ActivityInstrumentationTestCase2 { 10 | 11 | private CharacterEscape mActivity; 12 | 13 | public CharacterEscapeTests() { 14 | super(CharacterEscape.class); 15 | } 16 | 17 | @Override 18 | protected void setUp() throws Exception { 19 | super.setUp(); 20 | mActivity = getActivity(); 21 | } 22 | 23 | @MediumTest 24 | public void testCallFunctionButton_clickButtonAndShowResult() { 25 | final Button callFunctionButton = (Button) mActivity 26 | .findViewById(R.id.buttonCharacterEscapeCallFunction); 27 | 28 | final TextView resultTextView = (TextView) mActivity 29 | .findViewById(R.id.textViewCharacterEscapeResult); 30 | 31 | TouchUtils.clickView(this, callFunctionButton); 32 | assertEquals( 33 | "Result: \nmultiline: newline: \n apostrophe: \' quote: \" slashes: /\\/ closing tag: " 34 | + "unicode: 日本 characters: !@#$%^&*()-=_+`~<>? \n#####\n" 35 | + "multiline: \n newline: \\n apostrophe: \' quote: \" slashes: \\/\\\\/ closing tag: " 36 | + "unicode: 日本 characters: !@#$%^&*()-=_+`~<>?", resultTextView.getText() 37 | .toString()); 38 | } 39 | 40 | public void testPreconditions() { 41 | assertNotNull("mActivity is null", mActivity); 42 | } 43 | } 44 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/DestroyActivityTests.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests; 2 | 3 | import android.test.ActivityInstrumentationTestCase2; 4 | import android.test.TouchUtils; 5 | import android.test.suitebuilder.annotation.MediumTest; 6 | import android.view.View; 7 | import android.widget.Button; 8 | import android.widget.TextView; 9 | 10 | 11 | public class DestroyActivityTests extends 12 | ActivityInstrumentationTestCase2 { 13 | 14 | private DestroyActivity mActivity; 15 | 16 | public DestroyActivityTests() { 17 | super(DestroyActivity.class); 18 | } 19 | 20 | @Override 21 | protected void setUp() throws Exception { 22 | super.setUp(); 23 | mActivity = getActivity(); 24 | } 25 | 26 | @MediumTest 27 | public void testEvaluateButton_clickButtonAndShowResult() { 28 | final Button evaluateButton = (Button) mActivity.findViewById(R.id.button_evaluate_on_destroy); 29 | 30 | final TextView resultTextView = (TextView) mActivity.findViewById(R.id.js_result_destroy_text_view); 31 | 32 | TouchUtils.clickView(this, evaluateButton); 33 | assertTrue(View.VISIBLE == resultTextView.getVisibility()); 34 | assertEquals("Result: 4", resultTextView.getText().toString()); 35 | 36 | final Button destroyButton = (Button) mActivity.findViewById(R.id.button_destroy); 37 | TouchUtils.clickView(this, destroyButton); 38 | assertEquals("Web view destroyed", resultTextView.getText().toString()); 39 | } 40 | 41 | public void testPreconditions() { 42 | assertNotNull("mActivity is null", mActivity); 43 | } 44 | } 45 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/EvaluateJsActivityTests.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests; 2 | 3 | import android.test.ActivityInstrumentationTestCase2; 4 | import android.test.TouchUtils; 5 | import android.test.suitebuilder.annotation.MediumTest; 6 | import android.view.View; 7 | import android.widget.Button; 8 | import android.widget.TextView; 9 | 10 | public class EvaluateJsActivityTests extends 11 | ActivityInstrumentationTestCase2 { 12 | 13 | private EvaluateJsStringActivity mActivity; 14 | 15 | public EvaluateJsActivityTests() { 16 | super(EvaluateJsStringActivity.class); 17 | } 18 | 19 | @Override 20 | protected void setUp() throws Exception { 21 | super.setUp(); 22 | mActivity = getActivity(); 23 | } 24 | 25 | @MediumTest 26 | public void testEvaluateButton_clickButtonAndShowResult() { 27 | final Button evaluateButton = (Button) mActivity.findViewById(R.id.button_evaluate); 28 | 29 | final TextView resultTextView = (TextView) mActivity.findViewById(R.id.js_result_text_view); 30 | 31 | TouchUtils.clickView(this, evaluateButton); 32 | assertTrue(View.VISIBLE == resultTextView.getVisibility()); 33 | assertEquals("Result: 4", resultTextView.getText().toString()); 34 | } 35 | 36 | @MediumTest 37 | public void testJsTextView_shouldHaveJsText() { 38 | final TextView editJsTextView = (TextView) mActivity.findViewById(R.id.edit_java_script); 39 | 40 | final String expected = mActivity.getString(R.string.edit_js_text); 41 | final String actual = editJsTextView.getText().toString(); 42 | assertEquals(expected, actual); 43 | } 44 | 45 | public void testPreconditions() { 46 | assertNotNull("mActivity is null", mActivity); 47 | } 48 | } 49 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/PercentageActivityTest.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests; 2 | 3 | import android.test.ActivityInstrumentationTestCase2; 4 | import android.test.TouchUtils; 5 | import android.test.suitebuilder.annotation.MediumTest; 6 | import android.view.View; 7 | import android.widget.Button; 8 | import android.widget.TextView; 9 | 10 | public class PercentageActivityTest extends ActivityInstrumentationTestCase2 { 11 | private PercentageActivity mActivity; 12 | 13 | public PercentageActivityTest() { 14 | super(PercentageActivity.class); 15 | } 16 | 17 | @Override 18 | protected void setUp() throws Exception { 19 | super.setUp(); 20 | mActivity = getActivity(); 21 | } 22 | 23 | @MediumTest 24 | public void testEvaluateButton_clickButtonAndShowResult() { 25 | final Button evaluateButton = (Button) mActivity 26 | .findViewById(R.id.button_evaluate_percentage); 27 | 28 | final TextView resultTextView = (TextView) mActivity 29 | .findViewById(R.id.percentageResultLabel); 30 | 31 | TouchUtils.clickView(this, evaluateButton); 32 | assertTrue(View.VISIBLE == resultTextView.getVisibility()); 33 | assertEquals("Result: 5%7E 12.5% growth", resultTextView.getText().toString()); 34 | } 35 | 36 | @MediumTest 37 | public void testJsTextView_shouldHaveJsText() { 38 | final TextView editJsTextView = (TextView) mActivity.findViewById(R.id.percentageCodeLabel); 39 | 40 | final String expected = mActivity.getString(R.string.percetage_js_code); 41 | final String actual = editJsTextView.getText().toString(); 42 | assertEquals(expected, actual); 43 | } 44 | 45 | public void testPreconditions() { 46 | assertNotNull("mActivity is null", mActivity); 47 | } 48 | 49 | } 50 | -------------------------------------------------------------------------------- /app/src/main/java/com/evgenii/jsevaluatortests/DestroyActivity.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests; 2 | 3 | import android.app.Activity; 4 | import android.os.Bundle; 5 | import android.view.View; 6 | import android.widget.TextView; 7 | 8 | import com.evgenii.jsevaluator.JsEvaluator; 9 | import com.evgenii.jsevaluator.interfaces.JsCallback; 10 | 11 | public class DestroyActivity extends Activity { 12 | JsEvaluator mJsEvaluator; 13 | 14 | @Override 15 | protected void onCreate(Bundle savedInstanceState) { 16 | super.onCreate(savedInstanceState); 17 | setContentView(R.layout.activity_destroy); 18 | 19 | mJsEvaluator = new JsEvaluator(this); 20 | } 21 | 22 | public void onEvaluateClicked(View view) { 23 | mJsEvaluator.evaluate("2 + 2", new JsCallback() { 24 | @Override 25 | public void onResult(final String resultValue) { 26 | final TextView jsResultTextView = (TextView) findViewById(R.id.js_result_destroy_text_view); 27 | jsResultTextView.setText(String.format("Result: %s", 28 | resultValue)); 29 | } 30 | 31 | @Override 32 | public void onError(String errorMessage) { 33 | final TextView jsResultTextView = (TextView) findViewById(R.id.js_result_destroy_text_view); 34 | jsResultTextView.setText(JSUtils.getErrorSpan(errorMessage)); 35 | } 36 | }); 37 | } 38 | 39 | public void onDestroyClicked(View view) { 40 | mJsEvaluator.destroy(); 41 | 42 | if (mJsEvaluator.getWebView() == null) { 43 | final TextView jsResultTextView = (TextView) findViewById(R.id.js_result_destroy_text_view); 44 | jsResultTextView.setText("Web view destroyed"); 45 | } 46 | } 47 | } 48 | -------------------------------------------------------------------------------- /app/src/androidTest/java/com/evgenii/jsevaluatortests/WebViewWrapperTest.java: -------------------------------------------------------------------------------- 1 | package com.evgenii.jsevaluatortests; 2 | 3 | import android.app.Instrumentation; 4 | import android.test.ActivityInstrumentationTestCase2; 5 | import android.util.Base64; 6 | 7 | import com.evgenii.jsevaluator.WebViewWrapper; 8 | 9 | import java.io.UnsupportedEncodingException; 10 | 11 | public class WebViewWrapperTest extends ActivityInstrumentationTestCase2 { 12 | 13 | private MainActivity mActivity; 14 | 15 | public WebViewWrapperTest() { 16 | super(MainActivity.class); 17 | } 18 | 19 | @Override 20 | protected void setUp() throws Exception { 21 | super.setUp(); 22 | mActivity = getActivity(); 23 | } 24 | 25 | public void testLoadJavaScript() { 26 | final Instrumentation mInstrumentation = getInstrumentation(); 27 | mActivity.runOnUiThread(new Runnable() { 28 | @Override 29 | public void run() { 30 | final WebViewWrapper wrapper = new WebViewWrapper(mActivity, null); 31 | wrapper.loadJavaScript("2 + 3"); 32 | } 33 | }); 34 | 35 | mInstrumentation.waitForIdleSync(); 36 | } 37 | 38 | public void testDestroyWebView() { 39 | final Instrumentation mInstrumentation = getInstrumentation(); 40 | mActivity.runOnUiThread(new Runnable() { 41 | @Override 42 | public void run() { 43 | final WebViewWrapper wrapper = new WebViewWrapper(mActivity, null); 44 | wrapper.loadJavaScript("2 + 3"); 45 | 46 | assertNotNull(wrapper.getWebView()); 47 | wrapper.destroy(); 48 | Boolean exceptionRaised = false; 49 | 50 | try { 51 | wrapper.loadJavaScript("2 + 3"); 52 | // Must fail because the web view has been destroyed. 53 | } catch (final NullPointerException e) { 54 | exceptionRaised = true; 55 | } 56 | 57 | assertTrue(exceptionRaised); 58 | assertNull(wrapper.getWebView()); 59 | } 60 | }); 61 | 62 | mInstrumentation.waitForIdleSync(); 63 | } 64 | } 65 | -------------------------------------------------------------------------------- /app/src/main/res/layout/activity_evaluate_js_string.xml: -------------------------------------------------------------------------------- 1 | 10 | 11 | 23 | 24 | 25 | 26 |