| /* |
| * Copyright (C) 2011 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.test.tilebenchmark; |
| |
| import android.content.Context; |
| import android.os.CountDownTimer; |
| import android.util.AttributeSet; |
| import android.util.Log; |
| import android.webkit.WebView; |
| |
| import com.test.tilebenchmark.ProfileActivity.ProfileCallback; |
| import com.test.tilebenchmark.RunData.TileData; |
| |
| public class ProfiledWebView extends WebView { |
| private int mSpeed; |
| |
| private boolean mIsTesting = false; |
| private boolean mIsScrolling = false; |
| private ProfileCallback mCallback; |
| private long mContentInvalMillis; |
| private boolean mHadToBeForced = false; |
| private int mTestCount = 0; |
| private static final int LOAD_STALL_MILLIS = 5000; // nr of millis after load, |
| // before test is forced |
| |
| public ProfiledWebView(Context context) { |
| super(context); |
| } |
| |
| public ProfiledWebView(Context context, AttributeSet attrs) { |
| super(context, attrs); |
| } |
| |
| public ProfiledWebView(Context context, AttributeSet attrs, int defStyle) { |
| super(context, attrs, defStyle); |
| } |
| |
| public ProfiledWebView(Context context, AttributeSet attrs, int defStyle, |
| boolean privateBrowsing) { |
| super(context, attrs, defStyle, privateBrowsing); |
| } |
| |
| @Override |
| protected void onDraw(android.graphics.Canvas canvas) { |
| if (mIsTesting && mIsScrolling) { |
| if (canScrollVertically(1)) { |
| scrollBy(0, mSpeed); |
| } else { |
| stopScrollTest(); |
| mIsScrolling = false; |
| } |
| } |
| super.onDraw(canvas); |
| } |
| |
| /* |
| * Called once the page is loaded to start scrolling for evaluating tiles. |
| * If autoScrolling isn't set, stop must be called manually. Before |
| * scrolling, invalidate all content and redraw it, measuring time taken. |
| */ |
| public void startScrollTest(ProfileCallback callback, boolean autoScrolling) { |
| mIsScrolling = autoScrolling; |
| mCallback = callback; |
| mIsTesting = false; |
| mContentInvalMillis = System.currentTimeMillis(); |
| registerPageSwapCallback(); |
| contentInvalidateAll(); |
| invalidate(); |
| |
| mTestCount++; |
| final int testCount = mTestCount; |
| |
| if (autoScrolling) { |
| // after a while, force it to start even if the pages haven't swapped |
| new CountDownTimer(LOAD_STALL_MILLIS, LOAD_STALL_MILLIS) { |
| @Override |
| public void onTick(long millisUntilFinished) { |
| } |
| |
| @Override |
| public void onFinish() { |
| if (testCount == mTestCount && !mIsTesting) { |
| mHadToBeForced = true; |
| Log.d("ProfiledWebView", "num " + testCount |
| + " forcing a page swap with a scroll..."); |
| scrollBy(0, 1); |
| invalidate(); // ensure a redraw so that auto-scrolling can occur |
| } |
| } |
| }.start(); |
| } |
| } |
| |
| /* |
| * Called after the manual contentInvalidateAll, after the tiles have all |
| * been redrawn. |
| */ |
| @Override |
| protected void pageSwapCallback(boolean startAnim) { |
| mContentInvalMillis = System.currentTimeMillis() - mContentInvalMillis; |
| super.pageSwapCallback(startAnim); |
| Log.d("ProfiledWebView", "REDRAW TOOK " + mContentInvalMillis |
| + "millis"); |
| mIsTesting = true; |
| invalidate(); // ensure a redraw so that auto-scrolling can occur |
| tileProfilingStart(); |
| } |
| |
| /* |
| * Called once the page has stopped scrolling |
| */ |
| public void stopScrollTest() { |
| tileProfilingStop(); |
| mIsTesting = false; |
| |
| if (mCallback == null) { |
| tileProfilingClear(); |
| return; |
| } |
| |
| RunData data = new RunData(super.tileProfilingNumFrames()); |
| // record the time spent (before scrolling) rendering the page |
| data.singleStats.put(getResources().getString(R.string.render_millis), |
| (double)mContentInvalMillis); |
| // record if the page render timed out |
| Log.d("ProfiledWebView", "hadtobeforced = " + mHadToBeForced); |
| data.singleStats.put(getResources().getString(R.string.render_stalls), |
| mHadToBeForced ? 1.0 : 0.0); |
| mHadToBeForced = false; |
| |
| for (int frame = 0; frame < data.frames.length; frame++) { |
| data.frames[frame] = new TileData[ |
| tileProfilingNumTilesInFrame(frame)]; |
| for (int tile = 0; tile < data.frames[frame].length; tile++) { |
| int left = tileProfilingGetInt(frame, tile, "left"); |
| int top = tileProfilingGetInt(frame, tile, "top"); |
| int right = tileProfilingGetInt(frame, tile, "right"); |
| int bottom = tileProfilingGetInt(frame, tile, "bottom"); |
| |
| boolean isReady = super.tileProfilingGetInt( |
| frame, tile, "isReady") == 1; |
| int level = tileProfilingGetInt(frame, tile, "level"); |
| |
| float scale = tileProfilingGetFloat(frame, tile, "scale"); |
| |
| data.frames[frame][tile] = data.new TileData(left, top, right, bottom, |
| isReady, level, scale); |
| } |
| } |
| tileProfilingClear(); |
| |
| mCallback.profileCallback(data); |
| } |
| |
| @Override |
| public void loadUrl(String url) { |
| if (!url.startsWith("http://") && !url.startsWith("file://")) { |
| url = "http://" + url; |
| } |
| super.loadUrl(url); |
| } |
| |
| public void setAutoScrollSpeed(int speedInt) { |
| mSpeed = speedInt; |
| } |
| } |