I made an app that loads images from urls and display them in coverflow view using CarouselWidget
When i run my app in Genymotion for the first time, it downloads the db, got results from query, but doesn't build coverflow. When i run it second time, it builds the view. Why it doesn't build it for the first time? I have included coverflow building in onPostExecute method.
When i run it on a real device, i always see the white screen. When i run it on a device in debug mode and do step out, it runs fine. How can i debug this on a device, if i don't get any errors?
How it's possible to load the images dynamically? Not all at the same time, but during swipe.
Here is my code.
public class MainActivity extends AppCompatActivity {
private static final String TAG = "qwe";
public static final int DIALOG_DOWNLOAD_PROGRESS = 0;
private static final String DB1_NAME = "SuperupDB.sqlite";
private static final String DB2_NAME = "InventoryDB.sqlite";
private static final String DB_URL = "https://storage.googleapis.com/shared_data/MKGn299XpJ6mn9c6/clientsync/db/SuperupDB.zip";
private String dbQuery;
private ProgressDialog mProgressDialog;
private static String DB_PATH;
private CoverFlowAdapter mAdapter;
private ArrayList<Bitmap> mData = new ArrayList<>();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
if (android.os.Build.VERSION.SDK_INT >= 17) {
DB_PATH = this.getApplicationInfo().dataDir + "/databases/";
} else {
DB_PATH = "/data/data/" + this.getPackageName() + "/databases/";
}
File dbFile1 = new File(DB_PATH + DB1_NAME);
File dbFile2 = new File(DB_PATH + DB2_NAME);
Log.d(TAG, dbFile1.getPath() + " " + dbFile1.exists());
Log.d(TAG, dbFile2.getPath() + " " + dbFile2.exists());
dbQuery = "Select p.PrdImgURL, cp.PrdID " +
"from Products as p " +
"inner join CategoryProduct_MM as cp " +
"on p.PrdID = cp.PrdID " +
"inner join InventoryDB.Facing as f " +
"on p.PrdID = f.PrdID " +
"where cp.CategoryID == 1 " +
"and p.PrdImgURL is not null " +
"order by f.`Order`";
if (dbFile1.exists() && dbFile2.exists()) {
loadProducts();
} else {
new DownloadDB().execute(DB_URL);
}
}
private void loadProducts() {
dbHandler dbh = dbHandler.getInstance(this);
SQLiteDatabase db = dbh.getWritableDatabase();
// attach second db to get products order from it
db.execSQL("attach database ? as InventoryDB", new String[]{DB_PATH + DB2_NAME});
db.beginTransaction();
Cursor c = db.rawQuery(dbQuery, null);
int count = c.getCount();
String url[] = new String[count];
int i = 0;
while (c.moveToNext()) {
url[i] = c.getString(c.getColumnIndex("PrdImgURL"));
i++;
}
for (i = 0; i < url.length; i++) {
loadBitmap(url[i]);
Log.d(TAG, url[i]);
}
mAdapter = new CoverFlowAdapter(getApplicationContext());
mAdapter.setData(mData);
CoverFlowCarousel mCoverFlow = (CoverFlowCarousel) findViewById(R.id.coverflow);
mCoverFlow.setAdapter(mAdapter);
}
private void loadBitmap(final String url) {
RequestQueue queue = Volley.newRequestQueue(this);
ImageRequest imageRequest = new ImageRequest(url,
new Response.Listener<Bitmap>() {
@Override
public void onResponse(Bitmap bitmap) {
//mData.add(bitmap);
mData.add(addShadow(bitmap, bitmap.getHeight(), bitmap.getWidth(), Color.BLACK, 7, 3, 7));
}
}, 0, 0, null,
new Response.ErrorListener() {
public void onErrorResponse(VolleyError error) {
Log.d(TAG, "loadBitmap: " + error.toString());
}
});
queue.add(imageRequest);
}
public Bitmap addShadow(final Bitmap bm, final int dstHeight, final int dstWidth, int color, int size, float dx, float dy) {
final Bitmap mask = Bitmap.createBitmap(dstWidth, dstHeight, Bitmap.Config.ALPHA_8);
final Matrix scaleToFit = new Matrix();
final RectF src = new RectF(0, 0, bm.getWidth(), bm.getHeight());
final RectF dst = new RectF(0, 0, dstWidth - dx, dstHeight - dy);
scaleToFit.setRectToRect(src, dst, Matrix.ScaleToFit.CENTER);
final Matrix dropShadow = new Matrix(scaleToFit);
dropShadow.postTranslate(dx, dy);
final Canvas maskCanvas = new Canvas(mask);
final Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
maskCanvas.drawBitmap(bm, scaleToFit, paint);
paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.MULTIPLY));
maskCanvas.drawBitmap(bm, dropShadow, paint);
final BlurMaskFilter filter = new BlurMaskFilter(size, BlurMaskFilter.Blur.NORMAL);
paint.reset();
paint.setAntiAlias(true);
paint.setColor(color);
paint.setMaskFilter(filter);
paint.setFilterBitmap(true);
final Bitmap ret = Bitmap.createBitmap(dstWidth + size, dstHeight + size, Bitmap.Config.ARGB_8888);
final Canvas retCanvas = new Canvas(ret);
retCanvas.drawBitmap(mask, 0, 0, paint);
retCanvas.drawBitmap(bm, scaleToFit, null);
mask.recycle();
return ret;
}
@Override
protected Dialog onCreateDialog(int id) {
switch (id) {
case DIALOG_DOWNLOAD_PROGRESS:
mProgressDialog = new ProgressDialog(this);
mProgressDialog.setMessage("Loading database..");
mProgressDialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
mProgressDialog.setCancelable(false);
mProgressDialog.show();
return mProgressDialog;
default:
return null;
}
}
class DownloadDB extends AsyncTask<String, String, String> {
@Override
protected void onPreExecute() {
super.onPreExecute();
showDialog(DIALOG_DOWNLOAD_PROGRESS);
}
@Override
protected String doInBackground(String... aurl) {
int count;
try {
URL url = new URL(aurl[0]);
URLConnection conn = url.openConnection();
conn.connect();
long lengthOfFile = conn.getContentLength();
Log.d(TAG, "Length of file: " + lengthOfFile);
InputStream input = new BufferedInputStream(url.openStream());
ZipInputStream zis = new ZipInputStream(input);
//default database directory
File dbDir = new File(DB_PATH);
if (!dbDir.exists())
dbDir.mkdir();
try {
ZipEntry ze;
while ((ze = zis.getNextEntry()) != null) {
if (ze.getName().equals(DB1_NAME) || ze.getName().equals(DB2_NAME)) {
lengthOfFile = ze.getSize();
FileOutputStream fout = new FileOutputStream(new File(dbDir, ze.getName()));
byte[] data = new byte[1024];
long total = 0;
while ((count = zis.read(data)) != -1) {
total += count;
publishProgress("" + (int) ((total * 100) / lengthOfFile));
fout.write(data, 0, count);
}
zis.closeEntry();
fout.close();
}
}
} catch (Exception e) {
Log.d(TAG, e.toString());
} finally {
zis.close();
}
} catch (Exception e) {
Log.d(TAG, e.toString());
}
return null;
}
protected void onProgressUpdate(String... progress) {
Log.d(TAG, progress[0]);
mProgressDialog.setProgress(Integer.parseInt(progress[0]));
}
@Override
protected void onPostExecute(String unused) {
dismissDialog(DIALOG_DOWNLOAD_PROGRESS);
loadProducts();
}
}
}
public class dbHandler extends SQLiteOpenHelper {
private static final int VERSION = 1;
private static final String DATABASE_NAME = "SuperupDB.sqlite";
private static final String TAG = "qwe";
private static File DATABASE_FILE;
// This is an indicator if we need to copy the
// database file.
private boolean mInvalidDatabaseFile = false;
private boolean mIsUpgraded = false;
private Context mContext;
/**
* number of users of the database connection.
*/
private int mOpenConnections = 0;
private static dbHandler mInstance;
synchronized static public dbHandler getInstance(Context context) {
if (mInstance == null) {
mInstance = new dbHandler(context.getApplicationContext());
}
return mInstance;
}
private dbHandler(Context context) {
super(context, DATABASE_NAME, null, VERSION);
this.mContext = context;
SQLiteDatabase db = null;
try {
db = getReadableDatabase();
if (db != null) {
db.close();
}
DATABASE_FILE = context.getDatabasePath(DATABASE_NAME);
/* if (mInvalidDatabaseFile) {
//copyDatabase();
Log.d(TAG, "Where is db?");
}*/
if (mIsUpgraded) {
doUpgrade();
}
} catch (SQLiteException e) {
} finally {
if (db != null && db.isOpen()) {
db.close();
}
}
}
@Override
public void onCreate(SQLiteDatabase db) {
//mInvalidDatabaseFile = true;
}
@Override
public void onUpgrade(SQLiteDatabase database,
int old_version, int new_version) {
//mInvalidDatabaseFile = true;
mIsUpgraded = true;
}
/**
* called if a database upgrade is needed
*/
private void doUpgrade() {
// implement the database upgrade here.
}
@Override
public synchronized void onOpen(SQLiteDatabase db) {
super.onOpen(db);
// increment the number of users of the database connection.
mOpenConnections++;
if (!db.isReadOnly()) {
// Enable foreign key constraints
db.execSQL("PRAGMA foreign_keys=ON;");
}
}
/**
* implementation to avoid closing the database connection while it is in
* use by others.
*/
@Override
public synchronized void close() {
mOpenConnections--;
if (mOpenConnections == 0) {
super.close();
}
}
private void setDatabaseVersion() {
SQLiteDatabase db = null;
try {
db = SQLiteDatabase.openDatabase(DATABASE_FILE.getAbsolutePath(), null,
SQLiteDatabase.OPEN_READWRITE);
db.execSQL("PRAGMA user_version = " + VERSION);
} catch (SQLiteException e) {
} finally {
if (db != null && db.isOpen()) {
db.close();
}
}
}
}
Solved it with lazy image loading by Universal Image Loader.
Here is the example