- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
会的,我有一个包含 MainFragment 的 MainActivity,MainFragment 实现了一个 RecipeDataFetcher 回调接口(interface),将通过 Volley/network 获取的数据带到 RecipeDataFetcher 类中。MainFragment 有一个 recyclerview,它应该显示几个食谱卡片
-我需要对 RecyclerView 项目执行测试
这是我的主要 fragment `公共(public)类
MainFragment extends Fragment implements RecipesRecyclerAdapter.OnRecipeItemSelected,
RecipesDataFetcher.RecipesFetcherDataListener {
private View v;
private RecipesDataFetcher recipesDataFetcher;
private OnFragmentInteractionListener mListener;
private RecyclerView recyclerView;
private RecipesRecyclerAdapter recipesRecyclerAdapter;
private Recipes mRecipes;
private RecyclerView.LayoutManager layoutManager;
private SimpleIdlingResource mIdlingResource;
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
v = inflater.inflate(R.layout.fragment_main, container, false);
mRecipes = new Recipes();
recyclerView = (RecyclerView) v.findViewById(R.id.rv_recipes);
if (mTwoPanel) {
layoutManager = new GridLayoutManager(getActivity(), 2);
} else {
layoutManager = new LinearLayoutManager(getActivity());
}
recipesRecyclerAdapter = new RecipesRecyclerAdapter(this, mRecipes);
recyclerView.setLayoutManager(layoutManager);
recyclerView.setAdapter(recipesRecyclerAdapter);
return v;
}
@VisibleForTesting
@NonNull
public SimpleIdlingResource getIdlingResource() {
if (mIdlingResource == null) {
mIdlingResource = new SimpleIdlingResource();
}
return mIdlingResource;
}
@Override
public void onSaveInstanceState(Bundle outState) {
// super.onSaveInstanceState(outState);
}
@Override
public void onAttach(Context context) {
super.onAttach(context);
if (context instanceof OnFragmentInteractionListener) {
mListener = (OnFragmentInteractionListener) context;
} else {
throw new RuntimeException(context.toString()
+ " must implement OnFragmentInteractionListener");
}
}
@Override
public void onDetach() {
super.onDetach();
mListener = null;
}
@Override
public void onItemSelected(int itemPosition) {
mListener.onFragmentInteraction(mRecipes.get(itemPosition), mRecipes.size());
}
@Override
public void onResume() {
super.onResume();
if (mRecipes.isEmpty()) {
getData();
}
}
@Override
public void onConnectionFailure() {
if (!checkOnlineState(getActivity())) {
Logging.shortToast(getActivity(), getString(R.string.internet_error));
} else {
Logging.shortToast(getActivity(), getString(R.string.server_error));
}
}
private void getData() {
recipesDataFetcher = new RecipesDataFetcher(getActivity(), this);
recipesDataFetcher.getRecipes(mIdlingResource);
}
@Override
public void onConnectionDone(Recipes recipes) {
mRecipes.addAll(recipes);
recipesRecyclerAdapter.notifyDataSetChanged();
//TODO: Do bulk inset in a background thread and for once
ContentValues[] contentValues = new ContentValues[recipes.size()];
for (int i = 0; i < recipes.size(); i++) {
ContentValues contentValue = new ContentValues();
Gson gson = new Gson();
String ingredients = gson.toJson(recipes.get(i).getIngredients());
String steps = gson.toJson(recipes.get(i).getSteps());
contentValue.put(COLUMN_RECIPE_INGREDIENTS, ingredients);
contentValue.put(COLUMN_RECIPE_STEPS, steps);
contentValue.put(COLUMN_RECIPE_SERVINGS, recipes.get(i).getServings());
contentValue.put(COLUMN_RECIPE_IMAGE, recipes.get(i).getImage());
contentValue.put(COLUMN_RECIPE_NAME, recipes.get(i).getName());
contentValues[i] = contentValue;
}
getContext().getContentResolver().bulkInsert(CONTENT_URI, contentValues);
if(mIdlingResource!=null){
mIdlingResource.setIdleState(true);
}
}
public interface OnFragmentInteractionListener {
void onFragmentInteraction(Recipe recipe, int size);
}`
这是我的 dataFetching 类
public class RecipesDataFetcher extends BaseDataFetcher {
public RecipesDataFetcher(Context context, BaseDataFetcherListener mListener) {
super(context, mListener);
}
public void getRecipes(SimpleIdlingResource simpleIdlingResource) {
if(simpleIdlingResource!=null){
simpleIdlingResource.setIdleState(false);}
String URL = BaseURL;
Logging.log("getCountries: " + URL);
JsonArrayRequest jsonObjReq = new JsonArrayRequest(URL, (JSONArray jsonArr) -> {
Logging.log("getCountries response: " + jsonArr.toString());
Recipes recipes = new Recipes(jsonArr);
((RecipesFetcherDataListener) mListener).onConnectionDone(recipes);
if(simpleIdlingResource!=null){
simpleIdlingResource.setIdleState(true);
}
}, this.errorListener);
retryPolicy(jsonObjReq);
getReQ().add(jsonObjReq);
}
public interface RecipesFetcherDataListener extends BaseDataFetcherListener {
void onConnectionDone(Recipes recipes);
}
这是我的测试类
@RunWith(AndroidJUnit4.class)
public class MyFragmentTest {
private SimpleIdlingResource mIdlingResource;
@Rule
public FragmentTestRule<MainFragment> mFragmentTestRule = new FragmentTestRule<>(MainFragment.class);
@Before
public void registerIdlingResource() {
// Launch the activity to make the fragment visible
mFragmentTestRule.launchActivity(null);
mIdlingResource = mFragmentTestRule.getFragment().getIdlingResource();
Espresso.registerIdlingResources(mIdlingResource);
}
@Test
public void fragment_can_be_instantiated() {
// Then to test item in position 0
onView(withId(R.id.rv_recipes))
.perform(RecyclerViewActions.actionOnItemAtPosition(0, click()));
}
// unregister resources
@After
public void unregisterIdlingResource() {
if (mIdlingResource != null) {
Espresso.unregisterIdlingResources(mIdlingResource);
}
}
}
最后 here是整个代码项目,请让我知道我到底做错了什么 :D,非常感谢您的帮助。
最佳答案
错误解释
参观完您的存储库并运行 MyFragmentTest 测试类
我遇到了两个错误
发生这种情况是因为在 FragmentTestRule 类中,您扩展了 ActivityTestRule <TestActivity>,其中 TestActivity 未实现 OnFragmentInteractionListener,因此解决方案是您拥有扩展已经实现此接口(interface)的 MainActivity,因此此类应如下所示:
public class FragmentTestRule<F extends Fragment> extends ActivityTestRule<MainActivity> {
private final Class<F> mFragmentClass;
private F mFragment;
public FragmentTestRule(final Class<F> fragmentClass) {
super(MainActivity.class, true, false);
mFragmentClass = fragmentClass;
}
.....
发生这种情况是因为在您的 MyFragmentTest注释 @Before 在任何开始之前被调用,这导致您的 fragment 尚未创建,这使得这一行没有意义
mIdlingResource = mFragmentTestRule.getFragment().getIdlingResource();
所以在我看来:不要使用 fragment 。 Activity 更容易测试。您可以单独测试每个 Activity 。在大多数情况下,Fragments 与 Activity 相比没有任何优势。 fragment 只会使实现和测试更加困难。
使用 volley 运行空闲资源的替代解决方案
(a) 在MainActivity.java
添加这些行
@Nullable
public SimpleIdlingResource mIdlingResource;
@VisibleForTesting
@NonNull
public SimpleIdlingResource getIdlingResource() {
if (mIdlingResource == null) {
mIdlingResource = new SimpleIdlingResource();
}
return mIdlingResource;
}
并在您的 onCreate 中调用此方法 getIdlingResource();从 idlingResouce 中获取实例。
(b) 在 MainFragment.java 中
(c) 最后这是新的 MainActivityTest
@RunWith(AndroidJUnit4.class)
public class MainActivityTest {
@Rule
public ActivityTestRule<MainActivity> mActivityTestRule =
new ActivityTestRule<>(MainActivity.class);
private IdlingResource mIdlingResource;
// Registers any resource that needs to be synchronized with Espresso before the test is run.
@Before
public void registerIdlingResource() {
mIdlingResource = mActivityTestRule.getActivity().getIdlingResource();
// To prove that the test fails, omit this call:
Espresso.registerIdlingResources(mIdlingResource);
}
@Test
public void idlingResourceTest() {
onView(withId(R.id.rv_recipes)).check(matches(isDisplayed()));
onView(withId(R.id.rv_recipes))
.perform(RecyclerViewActions.actionOnItemAtPosition(0, click()));
}
// Remember to unregister resources when not needed to avoid malfunction.
@After
public void unregisterIdlingResource() {
if (mIdlingResource != null) {
Espresso.unregisterIdlingResources(mIdlingResource);
}
}
}
抱歉,回答太长了希望我能帮上忙 :) :)
关于android - 如何在使用 IdlingResource 获取数据的 fragment 中使用 Espresso 测试 Recycler View ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44318782/
我需要告诉 Espresso 等到我的 Activity 空闲。我正在使用界面IdlingResource为达到这个。这是我用于测试的代码: public class MyActivity e
我注意到 Espresso 每五秒轮询一个特定的 IdlingResource 的 isIdleNow() 方法。 是否可以调整这个粒度? 提前致谢! 最佳答案 简答: 不,从 3.0.1 版开始,它
我想使用空闲资源,因为我在我的应用程序中使用 RxJava 和 EventBus,有时我的测试会失败(我认为这是因为同步)。 依赖关系: androidTestCompile 'com.android
我在使用 Espresso 进行测试时遇到了空闲资源的问题。 这是行不通的。它只被调用两次,仅此而已,即使返回 false。 public class MyIdlingResource impleme
我正在尝试测试 MyActivity 在传递不正确的 intent extras 时显示一个警告对话框。这是一个 url,所以我将 url 传递给内部 webView 以加载 url 并在发生任何错误
我有以下 Espresso 代码: @Test fun backgroundWorkDisplaysTextAfterLoading() { onView(withId(R.id.btn_ne
我正在尝试找到一种方法来很好地实现一个 IdlingResource,它将轮询 CoroutineDispatcher 的 isActive 属性。但是,从调试来看,检查此属性时似乎从来没有 Acti
我是 Espresso 的新手,正在尝试在 fragment 上编写测试,在 fragment 实例化时进行 Retrofit 调用。接到电话后,我想检查 fragment 以查看 View 是否存在
我最近将我的应用程序从使用异步任务转换为 rxjava。现在,由于 espresso 没有用于 rxjava 的空闲资源,我的 espresso 测试不再等待我的数据调用完成。我注意到你可以制作自定义
我正在尝试测试 AutoCompleteTextView 是否会在键入某些单词后显示项目。但是在键入和显示弹出窗口之间存在延迟。首先,我使用的是 Thread.sleep(),它工作得很好。但我知道这
我有一个 Activity 。它包含 fragment ,这个 fragment 有 RecyclerView。从服务器加载的一些数据并显示在 RecyclerView 的第一项中。 我需要为此项目设
我希望能够让 Espresso 将 Picasso 作为 IdlingResource 进行监控,以便我可以运行 ViewMatcher s 一旦图像已成功加载。 通过浏览 Picasso 源代码,我
我正在尝试编写 Espresso 单元测试,该测试依赖于与外部应用程序建立 TCP/IP 网络连接以便成功通过的组件。 测试失败是因为 TCP/IP 网络花费的时间比 Espresso 允许的时间长.
我正在使用 Espresso 编写 UI 测试。应用程序与服务器紧密合作,因此在很多情况下,我需要等待计算值,或者获取并显示数据等。Espresso 建议为此使用 IdlingResource。我的
我有一个托管 fragment F1 的 Activity 。单击按钮后,F1 将替换为另一个 fragment F2。按下后退按钮时,应用会通过退出过渡动画从 F2 返回到 F1。 我的 Espre
我基本上有一个自定义的 IdlingResource,它接受一个 View 构造函数参数。我找不到任何真正谈论如何实现它的地方。 我正在尝试使用这个答案:https://stackoverflow.c
在这里,我想要一个自定义的查找 View 函数:如果使用给定的 Matcher 在 10 秒内找不到 View ,则返回 bool 值 false,否则返回 View 。注意,我不想使用 Idling
在我的测试中,我有一个阶段,在按下按钮后,应用程序会执行大量异步计算并向云服务发出请求,之后它会显示特定 View 。 是否可以使用 Espresso 的 IdlingResource 实现来等待某个
我正在尝试将我的一些 Robotium 单元测试转换为使用 Espresso,但在通过测试更新 UI 时遇到了问题。该测试针对的是一个 fragment ,该 fragment 是一种显示来自对象的数
我正在编写集成测试,在 UI 中执行操作,使用 Retrofit 启动网络调用。 我知道我需要实现 CountingIdlingResource ,但我想以正确的方式进行(如果已经完成,则不要重新发明
我是一名优秀的程序员,十分优秀!