use of com.ichi2.libanki.Media in project AnkiChinaAndroid by ankichinateam.
the class DeckPicker method showStartupScreensAndDialogs.
public void showStartupScreensAndDialogs(SharedPreferences preferences, int skip) {
if (!BackupManager.enoughDiscSpace(CollectionHelper.getCurrentAnkiDroidDirectory(this))) {
Timber.i("Not enough space to do backup");
showDialogFragment(DeckPickerNoSpaceLeftDialog.newInstance());
} else if (preferences.getBoolean("noSpaceLeft", false)) {
Timber.i("No space left");
showDialogFragment(DeckPickerBackupNoSpaceLeftDialog.newInstance());
preferences.edit().remove("noSpaceLeft").apply();
} else if ("".equals(preferences.getString("lastVersion", ""))) {
Timber.i("Fresh install");
preferences.edit().putString("lastVersion", VersionUtils.getPkgVersionName()).apply();
onFinishedStartup();
} else if (skip < 2 && !preferences.getString("lastVersion", "").equals(VersionUtils.getPkgVersionName())) {
Timber.i("AnkiDroid is being updated and a collection already exists.");
// The user might appreciate us now, see if they will help us get better?
if (!preferences.contains(UsageAnalytics.ANALYTICS_OPTIN_KEY)) {
showDialogFragment(DeckPickerAnalyticsOptInDialog.newInstance());
}
// For upgrades, we check if we are upgrading
// to a version that contains additions to the database integrity check routine that we would
// like to run on all collections. A missing version number is assumed to be a fresh
// installation of AnkiDroid and we don't run the check.
long current = VersionUtils.getPkgVersionCode();
Timber.i("Current AnkiDroid version: %s", current);
long previous;
if (preferences.contains(UPGRADE_VERSION_KEY)) {
// Upgrading currently installed app
previous = getPreviousVersion(preferences, current);
} else {
// Fresh install
previous = current;
}
preferences.edit().putLong(UPGRADE_VERSION_KEY, current).apply();
// It is rebuilt on the next sync or media check
if (previous < 20300200) {
Timber.i("Deleting media database");
File mediaDb = new File(CollectionHelper.getCurrentAnkiDroidDirectory(this), "collection.media.ad.db2");
if (mediaDb.exists()) {
mediaDb.delete();
}
}
// Recommend the user to do a full-sync if they're upgrading from before 2.3.1beta8
if (previous < 20301208) {
Timber.i("Recommend the user to do a full-sync");
mRecommendFullSync = true;
}
// Fix "font-family" definition in templates created by AnkiDroid before 2.6alhpa23
if (previous < 20600123) {
Timber.i("Fixing font-family definition in templates");
try {
Models models = getCol().getModels();
for (Model m : models.all()) {
String css = m.getString("css");
if (css.contains("font-familiy")) {
m.put("css", css.replace("font-familiy", "font-family"));
models.save(m);
}
}
models.flush();
} catch (JSONException e) {
Timber.e(e, "Failed to upgrade css definitions.");
}
}
// Check if preference upgrade or database check required, otherwise go to new feature screen
int upgradePrefsVersion = AnkiDroidApp.CHECK_PREFERENCES_AT_VERSION;
int upgradeDbVersion = AnkiDroidApp.CHECK_DB_AT_VERSION;
// Specifying a checkpoint in the future is not supported, please don't do it!
if (current < upgradePrefsVersion) {
Timber.e("Checkpoint in future produced.");
UIUtils.showSimpleSnackbar(this, "Invalid value for CHECK_PREFERENCES_AT_VERSION", false);
onFinishedStartup();
return;
}
if (current < upgradeDbVersion) {
Timber.e("Invalid value for CHECK_DB_AT_VERSION");
UIUtils.showSimpleSnackbar(this, "Invalid value for CHECK_DB_AT_VERSION", false);
onFinishedStartup();
return;
}
// Skip full DB check if the basic check is OK
// TODO: remove this variable if we really want to do the full db check on every user
boolean skipDbCheck = false;
// noinspection ConstantConditions
if ((!skipDbCheck && previous < upgradeDbVersion) || previous < upgradePrefsVersion) {
if (previous < upgradePrefsVersion) {
Timber.i("showStartupScreensAndDialogs() running upgradePreferences()");
upgradePreferences(previous);
}
// noinspection ConstantConditions
if (!skipDbCheck && previous < upgradeDbVersion) {
Timber.i("showStartupScreensAndDialogs() running integrityCheck()");
// #5852 - since we may have a warning about disk space, we don't want to force a check database
// and show a warning before the user knows what is happening.
new MaterialDialog.Builder(this).title(R.string.integrity_check_startup_title).content(R.string.integrity_check_startup_content).positiveText(R.string.integrity_check_positive).negativeText(R.string.close).onPositive((materialDialog, dialogAction) -> integrityCheck()).onNeutral((materialDialog, dialogAction) -> this.restartActivity()).onNegative((materialDialog, dialogAction) -> this.restartActivity()).canceledOnTouchOutside(false).cancelable(false).build().show();
} else if (previous < upgradePrefsVersion) {
Timber.i("Updated preferences with no integrity check - restarting activity");
// If integrityCheck() doesn't occur, but we did update preferences we should restart DeckPicker to
// proceed
this.restartActivity();
}
} else {
// If no changes are required we go to the new features activity
// There the "lastVersion" is set, so that this code is not reached again
// if (VersionUtils.isReleaseVersion()) {
// Timber.i("Displaying new features");
// Intent infoIntent = new Intent(this, Info.class);
// infoIntent.putExtra(Info.TYPE_EXTRA, Info.TYPE_NEW_VERSION);
//
// if (skip != 0) {
// startActivityForResultWithAnimation(infoIntent, SHOW_INFO_NEW_VERSION,
// ActivityTransitionAnimation.LEFT);
// } else {
// startActivityForResultWithoutAnimation(infoIntent, SHOW_INFO_NEW_VERSION);
// }
// } else {
Timber.i("Dev Build - not showing 'new features'");
// Don't show new features dialog for development builds
preferences.edit().putString("lastVersion", VersionUtils.getPkgVersionName()).apply();
String ver = getResources().getString(R.string.updated_version, VersionUtils.getPkgVersionName());
UIUtils.showSnackbar(this, ver, true, -1, null, findViewById(R.id.root_layout), null);
showStartupScreensAndDialogs(preferences, 2);
// }
}
} else {
// this is the main call when there is nothing special required
Timber.i("No startup screens required");
onFinishedStartup();
}
}
use of com.ichi2.libanki.Media in project AnkiChinaAndroid by ankichinateam.
the class DialogHandler method handleMessage.
@Override
public void handleMessage(Message msg) {
Bundle msgData = msg.getData();
String messageName = sMessageNameList[msg.what];
UsageAnalytics.sendAnalyticsScreenView(messageName);
Timber.i("Handling Message: %s", messageName);
if (msg.what == MSG_SHOW_COLLECTION_LOADING_ERROR_DIALOG) {
// Collection could not be opened
mActivity.get().showDatabaseErrorDialog(DatabaseErrorDialog.DIALOG_LOAD_FAILED);
} else if (msg.what == MSG_SHOW_COLLECTION_IMPORT_REPLACE_DIALOG) {
// Handle import of collection package APKG
mActivity.get().showImportDialog(ImportDialog.DIALOG_IMPORT_REPLACE_CONFIRM, msgData.getString("importPath"));
} else if (msg.what == MSG_SHOW_COLLECTION_IMPORT_ADD_DIALOG) {
// Handle import of deck package APKG
mActivity.get().showImportDialog(ImportDialog.DIALOG_IMPORT_ADD_CONFIRM, msgData.getString("importPath"));
} else if (msg.what == MSG_SHOW_SYNC_ERROR_DIALOG) {
if (mActivity.get() instanceof DeckPicker) {
int id = msgData.getInt("dialogType");
String message = msgData.getString("dialogMessage");
((DeckPicker) mActivity.get()).showSyncErrorDialog(id, message);
}
} else if (msg.what == MSG_SHOW_EXPORT_COMPLETE_DIALOG) {
// Export complete
AsyncDialogFragment f = DeckPickerExportCompleteDialog.newInstance(msgData.getString("exportPath"));
mActivity.get().showAsyncDialogFragment(f);
} else if (msg.what == MSG_SHOW_MEDIA_CHECK_COMPLETE_DIALOG) {
if (mActivity.get() instanceof DeckPicker) {
// Media check results
int id = msgData.getInt("dialogType");
if (id != MediaCheckDialog.DIALOG_CONFIRM_MEDIA_CHECK) {
List<List<String>> checkList = new ArrayList<>();
checkList.add(msgData.getStringArrayList("nohave"));
checkList.add(msgData.getStringArrayList("unused"));
checkList.add(msgData.getStringArrayList("invalid"));
((DeckPicker) mActivity.get()).showMediaCheckDialog(id, checkList);
}
}
} else if (msg.what == MSG_SHOW_DATABASE_ERROR_DIALOG) {
// Database error dialog
mActivity.get().showDatabaseErrorDialog(msgData.getInt("dialogType"));
} else if (msg.what == MSG_SHOW_FORCE_FULL_SYNC_DIALOG) {
// Confirmation dialog for forcing full sync
ConfirmationDialog dialog = new ConfirmationDialog();
Runnable confirm = new Runnable() {
@Override
public void run() {
// Bypass the check once the user confirms
CollectionHelper.getInstance().getCol(AnkiDroidApp.getInstance()).modSchemaNoCheck();
}
};
dialog.setConfirm(confirm);
dialog.setArgs(msgData.getString("message"));
(mActivity.get()).showDialogFragment(dialog);
} else if (msg.what == MSG_DO_SYNC) {
if (mActivity.get() instanceof DeckPicker) {
SharedPreferences preferences = AnkiDroidApp.getSharedPrefs(mActivity.get());
Resources res = mActivity.get().getResources();
Collection col = mActivity.get().getCol();
String hkey = preferences.getString("hkey", "");
long millisecondsSinceLastSync = col.getTime().intTimeMS() - preferences.getLong("lastSyncTime", 0);
boolean limited = millisecondsSinceLastSync < INTENT_SYNC_MIN_INTERVAL;
if (!limited && hkey.length() > 0 && Connection.isOnline()) {
((DeckPicker) mActivity.get()).sync();
} else {
String err = res.getString(R.string.sync_error);
if (limited) {
long remainingTimeInSeconds = Math.max((INTENT_SYNC_MIN_INTERVAL - millisecondsSinceLastSync) / 1000, 1);
// getQuantityString needs an int
int remaining = (int) Math.min(Integer.MAX_VALUE, remainingTimeInSeconds);
String message = res.getQuantityString(R.plurals.sync_automatic_sync_needs_more_time, remaining, remaining);
mActivity.get().showSimpleNotification(err, message, NotificationChannels.Channel.SYNC);
} else {
mActivity.get().showSimpleNotification(err, res.getString(R.string.youre_offline), NotificationChannels.Channel.SYNC);
}
}
mActivity.get().finishWithoutAnimation();
}
}
}
use of com.ichi2.libanki.Media in project AnkiChinaAndroid by ankichinateam.
the class Connection method doInBackgroundSync.
private Payload doInBackgroundSync(Payload data) {
sIsCancellable = true;
Timber.d("doInBackgroundSync()");
// Block execution until any previous background task finishes, or timeout after 5s
boolean ok = CollectionTask.waitToFinish(5);
String hkey = (String) data.data[0];
boolean media = (Boolean) data.data[1];
String conflictResolution = (String) data.data[2];
HostNum hostNum = (HostNum) data.data[3];
long restServerSpace = (long) data.data[4];
// Use safe version that catches exceptions so that full sync is still possible
Collection col = CollectionHelper.getInstance().getColSafe(AnkiDroidApp.getInstance());
boolean colCorruptFullSync = false;
if (!CollectionHelper.getInstance().colIsOpen() || !ok) {
if (conflictResolution != null && "download".equals(conflictResolution)) {
colCorruptFullSync = true;
} else {
data.success = false;
data.result = new Object[] { "genericError" };
return data;
}
}
try {
CollectionHelper.getInstance().lockCollection();
HttpSyncer server = new RemoteServer(this, hkey, hostNum);
Syncer client = new Syncer(col, server, hostNum);
// run sync and check state
boolean noChanges = false;
if (conflictResolution == null) {
Timber.i("Sync - starting sync");
publishProgress(R.string.sync_prepare_syncing);
Object[] ret = client.sync(this, restServerSpace);
data.message = client.getSyncMsg();
if (ret == null) {
data.success = false;
data.result = new Object[] { "genericError" };
return data;
}
String retCode = (String) ret[0];
if (!"noChanges".equals(retCode) && !"success".equals(retCode)) {
data.success = false;
data.result = ret;
// Check if there was a sanity check error
if ("sanityCheckError".equals(retCode)) {
// Force full sync next time
col.modSchemaNoCheck();
col.save();
}
return data;
}
// save and note success state
if ("noChanges".equals(retCode)) {
// publishProgress(R.string.sync_no_changes_message);
noChanges = true;
}
restServerSpace = client.getRestSpace();
} else {
try {
// Disable sync cancellation for full-sync
sIsCancellable = false;
server = new FullSyncer(col, hkey, this, hostNum);
if ("upload".equals(conflictResolution)) {
Timber.i("Sync - fullsync - upload collection");
publishProgress(R.string.sync_preparing_full_sync_message);
Object[] ret = server.upload(restServerSpace);
col.reopen();
if (ret == null) {
data.success = false;
data.result = new Object[] { "genericError" };
return data;
}
if (!ret[0].equals(HttpSyncer.ANKIWEB_STATUS_OK)) {
data.success = false;
data.result = ret;
return data;
}
} else if ("download".equals(conflictResolution)) {
Timber.i("Sync - fullsync - download collection");
publishProgress(R.string.sync_downloading_message);
Object[] ret = server.download();
if (ret == null) {
Timber.w("Sync - fullsync - unknown error");
data.success = false;
data.result = new Object[] { "genericError" };
return data;
}
if ("success".equals(ret[0])) {
data.success = true;
col.reopen();
}
if (!"success".equals(ret[0])) {
Timber.w("Sync - fullsync - download failed");
data.success = false;
data.result = ret;
if (!colCorruptFullSync) {
col.reopen();
}
return data;
}
}
} catch (OutOfMemoryError e) {
AnkiDroidApp.sendExceptionReport(e, "doInBackgroundSync-fullSync");
data.success = false;
data.result = new Object[] { "OutOfMemoryError" };
return data;
} catch (RuntimeException e) {
if (timeoutOccurred(e)) {
data.result = new Object[] { "connectionError", e };
} else if ("UserAbortedSync".equals(e.getMessage())) {
data.result = new Object[] { "UserAbortedSync", e };
} else {
AnkiDroidApp.sendExceptionReport(e, "doInBackgroundSync-fullSync");
data.result = new Object[] { "IOException", e };
}
data.success = false;
return data;
}
}
// clear undo to avoid non syncing orphans (because undo resets usn too
if (!noChanges) {
col.clearUndo();
}
// then move on to media sync
sIsCancellable = true;
boolean noMediaChanges = false;
boolean enoughServerSpace = true;
String mediaError = null;
if (media) {
server = new RemoteMediaServer(col, hkey, this, hostNum);
MediaSyncer mediaClient = new MediaSyncer(col, (RemoteMediaServer) server, this);
String ret;
try {
// ������ܻ���Ϊ�ϴ��ļ������ռ��С���׳��쳣���ڴ�֮ǰ��Ҫ����ʣ��ռ��С
ret = mediaClient.sync(restServerSpace);
Timber.e("sync media ret is null");
if (ret == null) {
mediaError = AnkiDroidApp.getAppResources().getString(R.string.sync_media_error);
} else {
if ("corruptMediaDB".equals(ret)) {
mediaError = AnkiDroidApp.getAppResources().getString(R.string.sync_media_db_error);
noMediaChanges = true;
}
if ("noChanges".equals(ret)) {
publishProgress(R.string.sync_media_no_changes);
noMediaChanges = true;
}
if ("sanityFailed".equals(ret)) {
mediaError = AnkiDroidApp.getAppResources().getString(R.string.sync_media_sanity_failed);
} else {
publishProgress(R.string.sync_media_success);
}
}
} catch (RuntimeException e) {
if (timeoutOccurred(e)) {
data.result = new Object[] { "connectionError", e };
} else if ("UserAbortedSync".equals(e.getMessage())) {
data.result = new Object[] { "UserAbortedSync", e };
}
mediaError = AnkiDroidApp.getAppResources().getString(R.string.sync_media_error) + "\n\n" + e.getLocalizedMessage();
}
}
if (noChanges && (!media || noMediaChanges)) {
data.success = false;
data.result = new Object[] { "noChanges" };
return data;
} else {
data.success = true;
data.data = new Object[] { conflictResolution, col, mediaError };
return data;
}
} catch (MediaSyncException e) {
Timber.e("Media sync rejected by server");
data.success = false;
data.result = new Object[] { "mediaSyncServerError", e };
AnkiDroidApp.sendExceptionReport(e, "doInBackgroundSync");
return data;
} catch (UnknownHttpResponseException e) {
Timber.e(e, "doInBackgroundSync -- unknown response code error");
data.success = false;
int code = e.getResponseCode();
String msg = e.getLocalizedMessage();
data.result = new Object[] { "error", code, msg };
return data;
} catch (NoEnoughServerSpaceException e) {
Timber.e("NoEnoughServerSpaceException ");
e.printStackTrace();
data.success = false;
data.result = new Object[] { "noServerSpace", e.rest, e.need };
return data;
} catch (Exception e) {
// Global error catcher.
// Try to give a human readable error, otherwise print the raw error message
Timber.e(e, "doInBackgroundSync error");
data.success = false;
if (timeoutOccurred(e)) {
data.result = new Object[] { "connectionError", e };
} else if ("UserAbortedSync".equals(e.getMessage())) {
data.result = new Object[] { "UserAbortedSync", e };
} else {
AnkiDroidApp.sendExceptionReport(e, "doInBackgroundSync");
data.result = new Object[] { e.getLocalizedMessage(), e };
}
return data;
} finally {
Timber.i("Sync Finished - Closing Collection");
// don't bump mod time unless we explicitly save
if (col != null) {
col.close(false);
}
CollectionHelper.getInstance().unlockCollection();
}
}
use of com.ichi2.libanki.Media in project AnkiChinaAndroid by ankichinateam.
the class SdCardReceiver method onReceive.
@Override
public void onReceive(Context context, Intent intent) {
if (intent.getAction().equals(Intent.ACTION_MEDIA_EJECT)) {
Timber.i("media eject detected - closing collection and sending broadcast");
Intent i = new Intent();
i.setAction(MEDIA_EJECT);
context.sendBroadcast(i);
try {
Collection col = CollectionHelper.getInstance().getCol(context);
if (col != null) {
col.close();
}
} catch (Exception e) {
Timber.w(e, "Exception while trying to close collection likely because it was already unmounted");
}
} else if (intent.getAction().equals(Intent.ACTION_MEDIA_MOUNTED)) {
Timber.i("media mount detected - sending broadcast");
Intent i = new Intent();
i.setAction(MEDIA_MOUNT);
context.sendBroadcast(i);
}
}
use of com.ichi2.libanki.Media in project AnkiChinaAndroid by ankichinateam.
the class ImageFieldTest method testNoImagePathIsNothing.
@Test
public void testNoImagePathIsNothing() {
String knownBadImage = "<br />";
Collection col = collectionWithMediaFolder("media");
String imageSrc = ImageField.getImageFullPath(col, knownBadImage);
assertThat("no media should return no paths", imageSrc, equalTo(""));
}
Aggregations