Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Testing #158

Draft
wants to merge 58 commits into
base: webview
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 21 commits
Commits
Show all changes
58 commits
Select commit Hold shift + click to select a range
ee10b26
setup
grcoleman Jul 14, 2020
6973b6b
adds night theme color scheme, adds setForceDarkStrategy, adds media …
grcoleman Jul 14, 2020
efcf673
setup
grcoleman Jul 14, 2020
d6836d6
adds night theme color scheme, adds setForceDarkStrategy, adds media …
grcoleman Jul 14, 2020
c2de65f
fix conflicts
grcoleman Jul 14, 2020
761ecf0
changes path, adds icon choosing logic
grcoleman Jul 15, 2020
6db9c65
changes initial image path, adds symbos values
grcoleman Jul 15, 2020
55a9608
adds meta tag, corrects media query, adds setDefaultNightMode
grcoleman Jul 15, 2020
3e2c009
adds media query
grcoleman Jul 15, 2020
d70a861
changes path, adds icon choosing logic
grcoleman Jul 15, 2020
47f53af
changes initial image path, adds symbos values
grcoleman Jul 15, 2020
f429a26
changes fetch to correct path
grcoleman Jul 16, 2020
bc9ae5f
changes fetch path
grcoleman Jul 16, 2020
f0b47b9
changes background color, adds clarification on ForceDarkStrategy
grcoleman Jul 20, 2020
5dc103e
add clarifying comments on location of data, change path to ensure co…
grcoleman Jul 20, 2020
2c773f8
adds testing outline and dependencies
grcoleman Jul 21, 2020
403446f
changes res path, fixes comment in main.js
grcoleman Jul 21, 2020
77ff7e0
adds tests for createJsObject and drop down menu
grcoleman Jul 21, 2020
7185787
add description to tests,add ids to menu
grcoleman Jul 21, 2020
61c895e
splits light and dark themes into seperate files, adds logic to check…
grcoleman Jul 22, 2020
66b3c49
removes static text color setting from style.css
grcoleman Jul 22, 2020
da570e4
Merge pull request #157 from gcoleman799/change-data-source
gcoleman799 Jul 22, 2020
4323afe
adds testing outline and dependencies
grcoleman Jul 21, 2020
aa3df09
adds tests for createJsObject and drop down menu
grcoleman Jul 21, 2020
6d218b7
add description to tests,add ids to menu
grcoleman Jul 21, 2020
f263800
adds set up for unit tests
grcoleman Jul 22, 2020
42111ec
setup
grcoleman Jul 14, 2020
d42ded6
adds night theme color scheme, adds setForceDarkStrategy, adds media …
grcoleman Jul 14, 2020
0d4fdf8
setup
grcoleman Jul 14, 2020
85d21b8
adds night theme color scheme, adds setForceDarkStrategy, adds media …
grcoleman Jul 14, 2020
637fec3
adds meta tag, corrects media query, adds setDefaultNightMode
grcoleman Jul 15, 2020
2c83302
adds media query
grcoleman Jul 15, 2020
3a9d8fa
changes background color, adds clarification on ForceDarkStrategy
grcoleman Jul 20, 2020
f8b0c11
splits light and dark themes into seperate files, adds logic to check…
grcoleman Jul 22, 2020
be2899c
removes static text color setting from style.css
grcoleman Jul 22, 2020
7c3b2c1
add comment about default reloading
grcoleman Jul 22, 2020
60105b6
fixes conflicts
grcoleman Jul 22, 2020
e9e7a72
thread test and callback test outline
grcoleman Jul 23, 2020
4934ed2
fixes merge conflicts
grcoleman Jul 23, 2020
d7806ed
revert to one css file
grcoleman Jul 24, 2020
13d0b17
puts createJsObject in seperate file, adds handlers and runnables to …
grcoleman Jul 24, 2020
ce63069
converts runable to lambda, fixes message val, fixes loadDataWithBase…
grcoleman Jul 27, 2020
fa60356
adds coroutines to tests
grcoleman Jul 27, 2020
b4c6e58
changes dark theme css formatting
grcoleman Jul 28, 2020
35b902d
Merge pull request #155 from gcoleman799/custom-dark-theme
gcoleman799 Jul 28, 2020
77350b7
changes initial image path, gets rid of unncessary logging, fixes all…
grcoleman Jul 29, 2020
00b047b
Merge pull request #160 from gcoleman799/share-feature
gcoleman799 Jul 29, 2020
048851d
adds testing outline and dependencies
grcoleman Jul 21, 2020
7f7152e
adds tests for createJsObject and drop down menu
grcoleman Jul 21, 2020
bbc334b
add description to tests,add ids to menu
grcoleman Jul 21, 2020
4528470
adds set up for unit tests
grcoleman Jul 22, 2020
fa580f4
thread test and callback test outline
grcoleman Jul 23, 2020
684f2ef
adds testing outline and dependencies
grcoleman Jul 21, 2020
016ac01
adds tests for createJsObject and drop down menu
grcoleman Jul 21, 2020
c920025
puts createJsObject in seperate file, adds handlers and runnables to …
grcoleman Jul 24, 2020
e7bb953
uses CompletableDeffered in tests
grcoleman Jul 28, 2020
84cea6b
uses Completable Deferredand run blocking
grcoleman Jul 29, 2020
5c447de
Fixes merge conflicts
grcoleman Jul 29, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 11 additions & 1 deletion WebView/app/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,16 @@ dependencies {
implementation "com.google.android.material:material:$material_components_version"

testImplementation 'junit:junit:4.13'
testImplementation 'androidx.test:core:1.0.0'
testImplementation 'org.mockito:mockito-core:1.10.19'

androidTestImplementation 'androidx.test.ext:junit:1.1.1'
androidTestImplementation 'androidx.test.espresso:espresso-core:3.2.0'
}
androidTestImplementation 'androidx.test.espresso:espresso-web:3.1.0'
androidTestImplementation 'com.android.support.test.espresso:espresso-contrib:3.0.2'
androidTestImplementation 'com.android.support.test.espresso:espresso-intents:3.0.2'
androidTestImplementation 'androidx.test:rules:1.1.1'
androidTestImplementation 'androidx.test:runner:1.1.0'
androidTestImplementation "org.jetbrains.kotlinx:kotlinx-coroutines-core:1.3.0"

}

This file was deleted.

Original file line number Diff line number Diff line change
@@ -0,0 +1,170 @@
/*
* Copyright (C) 2020 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package com.android.samples.webviewdemo

import android.content.Context
import android.os.Handler
import android.os.Looper
import android.util.Log
import android.webkit.WebView
import androidx.test.core.app.ApplicationProvider
import androidx.test.espresso.web.assertion.WebViewAssertions.webMatches
import androidx.test.espresso.web.model.Atoms.castOrDie
import androidx.test.espresso.web.model.Atoms.script
import androidx.test.espresso.web.sugar.Web.onWebView
import androidx.test.espresso.web.webdriver.DriverAtoms.findElement
import androidx.test.espresso.web.webdriver.DriverAtoms.getText
import androidx.test.espresso.web.webdriver.DriverAtoms.webClick
import androidx.test.espresso.web.webdriver.Locator
import androidx.test.ext.junit.runners.AndroidJUnit4
import androidx.test.rule.ActivityTestRule
import junit.framework.Assert.assertEquals
import junit.framework.Assert.assertTrue
import kotlinx.coroutines.GlobalScope
import kotlinx.coroutines.async
import kotlinx.coroutines.delay
import kotlinx.coroutines.launch
import kotlinx.coroutines.runBlocking
import org.hamcrest.CoreMatchers.`is`
import org.hamcrest.CoreMatchers.containsString
import org.junit.Rule
import org.junit.Test
import org.junit.runner.RunWith


/**
* Launch, interact, and verify conditions in an activity that has a WebView instance.
*/
@RunWith(AndroidJUnit4::class)


class MainActivityTest {

val context = ApplicationProvider.getApplicationContext<Context>()

@Rule
@JvmField
val mainActivityRule = ActivityTestRule(MainActivity::class.java)
fun afterActivityLaunched() {
// Technically we do not need to do this - MainActivity has javascript turned on.
// Other WebViews in your app may have javascript turned off, however since the only way
// to automate WebViews is through javascript, it must be enabled.
onWebView().forceJavascriptEnabled()
}

// Test to check that the drop down menu behaves as expected
@Test
fun dropDownMenu_SanFran() {
mainActivityRule.getActivity()
onWebView()
.withElement(findElement(Locator.ID, "location"))
.perform(webClick()) // Similar to perform(click())
.withElement(findElement(Locator.ID, "SF"))
.perform(webClick()) // Similar to perform(click())
.withElement(findElement(Locator.ID, "title"))
.check(webMatches(getText(), containsString("San Francisco")))
}

// Test for checking createJsObject
@Test
fun jsObjectIsInjectedAndContainsPostMessage() {
mainActivityRule.getActivity()
onWebView()
.check(
webMatches(
script(
"return jsObject && jsObject.postMessage != null;",
castOrDie(Boolean::class.javaObjectType)
),
`is`(true)
)
)
}

@Test
fun valueInCallback_compareValueInput_returnsTrue() = runBlocking() {
mainActivityRule.activity
// Setup
val jsObjName = "jsObject"
val allowedOriginRules = setOf<String>("https://example.com")
val expectedMessage = "hello"
val callback = async { return@async "hello" }
// Get a handler that can be used to post to the main thread
val mainHandler = Handler(Looper.getMainLooper());
mainHandler.post {
run {
val webView = WebView(context)
// Create JsObject
createJsObject(
webView,
jsObjName,
allowedOriginRules
) { message -> callback
}
//Inject JsObject into Html
webView.loadDataWithBaseURL(
"https://example.com", "<html></html>",
"text/html", null, null
)
//Call js code to invoke callback
webView.evaluateJavascript("${jsObjName}.postMessage(`hello`)", null)
}
}
// evaluate what comes out -> it should be hello
assertEquals(expectedMessage, callback.await())
}

@Test
// Checks that postMessage runs on the UI thread
fun checkingThreadCallbackRunsOn() = runBlocking {
mainActivityRule.activity
// Setup
val jsObjName = "jsObject"
val allowedOriginRules = setOf<String>("https://example.com")
val callback = async { isUiThread() }
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looking at this again: how does this control when the coroutine executes? I wonder if the coroutine is executing on its own before the callback happens, which might explain test failures (the code is running on a non-UI thread, so the assertion would fail).

// Get a handler that can be used to post to the main thread
val mainHandler = Handler(Looper.getMainLooper())
// Start Interacting with webView on UI thread
mainHandler.post {
run {
val webView = WebView(context)
// Create JsObject
createJsObject(
webView,
jsObjName,
allowedOriginRules
) { message -> callback }
//Inject JsObject into Html
webView.loadDataWithBaseURL(
"https://example.com", "<html></html>",
"text/html", "UTF-8", null
)
//Call js code to invoke callback
webView.evaluateJavascript("${jsObjName}.postMessage(`hello`)", null)
}
}
assertTrue(callback.await())
}

/**
* Returns true if the current thread is the UI thread based on the
* Looper.
*/
private fun isUiThread(): Boolean {
return Looper.myLooper() == Looper.getMainLooper()
}
}
8 changes: 4 additions & 4 deletions WebView/app/src/main/assets/index.html
Original file line number Diff line number Diff line change
Expand Up @@ -22,12 +22,12 @@
<body onload="getData()">
<label for="location">Choose a location:</label>
<select name="location" id="location" onchange="getData()">
<option value="newYork">New York</option>
<option value="sanFrancisco">San Francisco</option>
<option value="london">London</option>
<option id= "NYC" value="newYork">New York</option>
<option id= "SF" value="sanFrancisco">San Francisco</option>
gcoleman799 marked this conversation as resolved.
Show resolved Hide resolved
<option id= "LDN" value="london">London</option>
</select>
<h1 id="title">Location</h1>
<img alt="weather" class="icon" id="icon" src="https://gcoleman799.github.io/res/drawable/sunny.png" />
<img alt="weather" class="icon" id="icon" src="https://raw.githubusercontent.com/res/drawable/sunny.png" />
<h2 class="currentTemp" id="currentTemp">Current Temp</h2>
<h2 class="shortDescription" id="shortDescription">Short Description</h2>
<p class="longDescription" id="longDescription">Long Description</p>
Expand Down
30 changes: 22 additions & 8 deletions WebView/app/src/main/assets/main.js
Original file line number Diff line number Diff line change
Expand Up @@ -18,26 +18,40 @@ function sendAndroidMessage() {
* in the WebViewCompat reference doc, the second parameter, MessagePorts, is optional.
* Also note that onmessage, addEventListener and removeEventListener are not supported.
*/
// TODO: Change message to account for changes in data
jsObject.postMessage("The weather in " + `${document.getElementById("title").innerText}` + " today is " +
`${document.getElementById("shortDescription").innerText} `);
}


function getData() {
// TODO: Change the path to grab data from new location; Change longDescription and currentTemp to work with changes in data
fetch("https://gcoleman799.github.io/Asset-Loader/weather.json").then(function(resp) {
// This JSON files is hosted over the web
fetch("https://raw.githubusercontent.com/android/views-widgets-samples/webview/WebView/sampleData/weather.json").then(function(resp) {
return resp.json();
}).then(function(data) {
var form = document.getElementById("location");
var currentLocation = form.options[form.selectedIndex].value;
console.log(data[currentLocation].description);
document.getElementById("title").innerText = form.options[form.selectedIndex].text;
document.getElementById("currentTemp").innerText = data[currentLocation].currentTemp;
document.getElementById("currentTemp").innerText = `${data[currentLocation].currentTemp}`+ "\xB0 F";
document.getElementById("shortDescription").innerText = data[currentLocation].description;
document.getElementById("longDescription").innerText = "Today in " + `${form.options[form.selectedIndex].text}`
+ " there is a " + `${data[currentLocation].chancePrecip}` + " chance of precipitation and the humidity is "
+ `${data[currentLocation].humidity}.`;
document.getElementById("icon").src = data[currentLocation].icon;
+ " there is a " + `${data[currentLocation].chancePrecip}` + "% chance of precipitation and the humidity is "
+ `${data[currentLocation].humidity}` + "%.";
document.getElementById("icon").src = getIcon(data[currentLocation].description);
})
}

// TODO: Create getIcon() function to decide which icon to render.
/* These icons are hosted locally, in the res/drawable folder. However, we can call them using
* http(s):// URLs because we have configured AssetLoader in MainActivity. It is desirable to
* access the files in this way because it is compatible with the Same-Origin policy.
*/
function getIcon(description){
switch(description) {
case "Rainy":
return "https://raw.githubusercontent.com/views-widgets-samples/res/drawable/rain.png";
case "Clear Sky":
return "https://raw.githubusercontent.com/views-widgets-samples/res/drawable/sunny.png";
default:
return "https://raw.githubusercontent.com/views-widgets-samples/res/drawable/partly_cloudy.png";
}
}
Loading