Added create folder to select folder dialog

pull/210/head
M66B 2 years ago
parent 2668f19acd
commit 828b6ee115

@ -1626,6 +1626,13 @@ public class AdapterFolder extends RecyclerView.Adapter<AdapterFolder.ViewHolder
return selected.get(position).id; return selected.get(position).id;
} }
int getPositionForKey(long key) {
for (int pos = 0; pos < selected.size(); pos++)
if (selected.get(pos).id.equals(key))
return pos;
return RecyclerView.NO_POSITION;
}
public TupleFolderEx getItemAtPosition(int pos) { public TupleFolderEx getItemAtPosition(int pos) {
if (pos >= 0 && pos < selected.size()) if (pos >= 0 && pos < selected.size())
return selected.get(pos); return selected.get(pos);

@ -2761,19 +2761,8 @@ class Core {
folder.setProperties(); folder.setProperties();
folder.setSpecials(account); folder.setSpecials(account);
if (selectable && parent != null && EntityFolder.USER.equals(parent.type)) { if (selectable)
folder.synchronize = parent.synchronize; folder.inheritFrom(parent);
folder.poll = parent.poll;
folder.poll_factor = parent.poll_factor;
folder.download = parent.download;
folder.auto_classify_source = parent.auto_classify_source;
folder.auto_classify_target = parent.auto_classify_target;
folder.sync_days = parent.sync_days;
folder.keep_days = parent.keep_days;
folder.unified = parent.unified;
folder.navigation = parent.navigation;
folder.notify = parent.notify;
}
folder.id = db.folder().insertFolder(folder); folder.id = db.folder().insertFolder(folder);
Log.i(folder.name + " added type=" + folder.type + " sync=" + folder.synchronize); Log.i(folder.name + " added type=" + folder.type + " sync=" + folder.synchronize);

@ -314,6 +314,25 @@ public class EntityFolder extends EntityOrder implements Serializable {
} }
} }
void inheritFrom(EntityFolder parent) {
if (parent == null)
return;
if (!EntityFolder.USER.equals(parent.type))
return;
this.synchronize = parent.synchronize;
this.poll = parent.poll;
this.poll_factor = parent.poll_factor;
this.download = parent.download;
this.auto_classify_source = parent.auto_classify_source;
this.auto_classify_target = parent.auto_classify_target;
this.sync_days = parent.sync_days;
this.keep_days = parent.keep_days;
this.unified = parent.unified;
this.navigation = parent.navigation;
this.notify = parent.notify;
}
static boolean shouldPoll(String type) { static boolean shouldPoll(String type) {
int sync = EntityFolder.SYSTEM_FOLDER_SYNC.indexOf(type); int sync = EntityFolder.SYSTEM_FOLDER_SYNC.indexOf(type);
return (sync < 0 || EntityFolder.SYSTEM_FOLDER_POLL.get(sync)); return (sync < 0 || EntityFolder.SYSTEM_FOLDER_POLL.get(sync));

@ -19,22 +19,30 @@ package eu.faircode.email;
Copyright 2018-2022 by Marcel Bokhorst (M66B) Copyright 2018-2022 by Marcel Bokhorst (M66B)
*/ */
import static android.app.Activity.RESULT_CANCELED;
import static android.app.Activity.RESULT_OK; import static android.app.Activity.RESULT_OK;
import android.app.Dialog; import android.app.Dialog;
import android.content.Context; import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.SharedPreferences; import android.content.SharedPreferences;
import android.os.Bundle; import android.os.Bundle;
import android.text.Editable; import android.text.Editable;
import android.text.TextUtils;
import android.text.TextWatcher; import android.text.TextWatcher;
import android.view.KeyEvent;
import android.view.LayoutInflater; import android.view.LayoutInflater;
import android.view.View; import android.view.View;
import android.view.inputmethod.EditorInfo;
import android.view.inputmethod.InputMethodManager; import android.view.inputmethod.InputMethodManager;
import android.widget.AdapterView; import android.widget.AdapterView;
import android.widget.ArrayAdapter; import android.widget.ArrayAdapter;
import android.widget.AutoCompleteTextView; import android.widget.AutoCompleteTextView;
import android.widget.Button; import android.widget.Button;
import android.widget.EditText;
import android.widget.ImageButton; import android.widget.ImageButton;
import android.widget.Spinner;
import android.widget.TextView; import android.widget.TextView;
import androidx.annotation.NonNull; import androidx.annotation.NonNull;
@ -45,18 +53,27 @@ import androidx.preference.PreferenceManager;
import androidx.recyclerview.widget.LinearLayoutManager; import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView; import androidx.recyclerview.widget.RecyclerView;
import com.google.android.material.floatingactionbutton.FloatingActionButton;
import org.json.JSONArray; import org.json.JSONArray;
import org.json.JSONException; import org.json.JSONException;
import java.text.Collator;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.Date; import java.util.Date;
import java.util.List; import java.util.List;
import java.util.Locale;
import java.util.concurrent.ExecutorService; import java.util.concurrent.ExecutorService;
public class FragmentDialogFolder extends FragmentDialogBase { public class FragmentDialogFolder extends FragmentDialogBase {
private int result = 0; private int result = 0;
private AdapterFolder adapter;
private LinearLayoutManager llm;
private static final int MAX_SELECTED_FOLDERS = 5; private static final int MAX_SELECTED_FOLDERS = 5;
private static final int REQUEST_FOLDER_NAME = 1;
private static final ExecutorService executor = private static final ExecutorService executor =
Helper.getBackgroundExecutor(1, "folder"); Helper.getBackgroundExecutor(1, "folder");
@ -94,6 +111,7 @@ public class FragmentDialogFolder extends FragmentDialogBase {
final Button btnFavorite3 = dview.findViewById(R.id.btnFavorite3); final Button btnFavorite3 = dview.findViewById(R.id.btnFavorite3);
final ImageButton ibResetFavorites = dview.findViewById(R.id.ibResetFavorites); final ImageButton ibResetFavorites = dview.findViewById(R.id.ibResetFavorites);
final RecyclerView rvFolder = dview.findViewById(R.id.rvFolder); final RecyclerView rvFolder = dview.findViewById(R.id.rvFolder);
final FloatingActionButton fabAdd = dview.findViewById(R.id.fabAdd);
final ContentLoadingProgressBar pbWait = dview.findViewById(R.id.pbWait); final ContentLoadingProgressBar pbWait = dview.findViewById(R.id.pbWait);
final Group grpReady = dview.findViewById(R.id.grpReady); final Group grpReady = dview.findViewById(R.id.grpReady);
@ -130,10 +148,10 @@ public class FragmentDialogFolder extends FragmentDialogBase {
etSearch.setAdapter(frequent); etSearch.setAdapter(frequent);
rvFolder.setHasFixedSize(false); rvFolder.setHasFixedSize(false);
final LinearLayoutManager llm = new LinearLayoutManager(context); llm = new LinearLayoutManager(context);
rvFolder.setLayoutManager(llm); rvFolder.setLayoutManager(llm);
final AdapterFolder adapter = new AdapterFolder(context, getViewLifecycleOwner(), adapter = new AdapterFolder(context, getViewLifecycleOwner(),
account, false, false, false, false, false, new AdapterFolder.IFolderSelectedListener() { account, false, false, false, false, false, new AdapterFolder.IFolderSelectedListener() {
@Override @Override
public void onFolderSelected(@NonNull TupleFolderEx folder) { public void onFolderSelected(@NonNull TupleFolderEx folder) {
@ -272,6 +290,20 @@ public class FragmentDialogFolder extends FragmentDialogBase {
} }
}); });
fabAdd.hide();
fabAdd.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Bundle args = new Bundle();
args.putLong("account", account);
FragmentDialogEditName fragment = new FragmentDialogEditName();
fragment.setArguments(args);
fragment.setTargetFragment(FragmentDialogFolder.this, REQUEST_FOLDER_NAME);
fragment.show(getParentFragmentManager(), "folder:name");
}
});
Bundle args = new Bundle(); Bundle args = new Bundle();
args.putLong("account", account); args.putLong("account", account);
args.putLongArray("disabled", disabled); args.putLongArray("disabled", disabled);
@ -297,6 +329,7 @@ public class FragmentDialogFolder extends FragmentDialogBase {
DB db = DB.getInstance(context); DB db = DB.getInstance(context);
Data data = new Data(); Data data = new Data();
data.account = db.account().getAccount(account);
data.folders = db.folder().getFoldersEx(account); data.folders = db.folder().getFoldersEx(account);
data.favorites = db.folder().getFavoriteFolders(account, 3, disabled); data.favorites = db.folder().getFavoriteFolders(account, 3, disabled);
@ -325,6 +358,11 @@ public class FragmentDialogFolder extends FragmentDialogBase {
adapter.setDisabled(Helper.fromLongArray(disabled)); adapter.setDisabled(Helper.fromLongArray(disabled));
adapter.set(data.folders); adapter.set(data.folders);
if (data.account.protocol == EntityAccount.TYPE_IMAP)
fabAdd.show();
else
fabAdd.hide();
grpReady.setVisibility(View.VISIBLE); grpReady.setVisibility(View.VISIBLE);
} }
} }
@ -361,7 +399,190 @@ public class FragmentDialogFolder extends FragmentDialogBase {
}); });
} }
@Override
public void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
super.onActivityResult(requestCode, resultCode, data);
try {
switch (requestCode) {
case REQUEST_FOLDER_NAME:
if (resultCode == RESULT_OK && data != null)
onFolderName(data.getBundleExtra("args"));
break;
}
} catch (Throwable ex) {
Log.e(ex);
}
}
private void onFolderName(Bundle args) {
new SimpleTask<List<TupleFolderEx>>() {
@Override
protected List<TupleFolderEx> onExecute(Context context, Bundle args) throws Throwable {
long aid = args.getLong("account");
String name = args.getString("name");
String pid = args.getString("parent");
DB db = DB.getInstance(context);
try {
db.beginTransaction();
EntityAccount account = db.account().getAccount(aid);
if (account == null)
return null;
EntityFolder parent = (TextUtils.isEmpty(pid) ? null : db.folder().getFolderByName(account.id, pid));
if (parent != null)
name = parent.name + parent.separator + name;
EntityFolder folder = new EntityFolder();
folder.tbc = true;
folder.account = account.id;
folder.name = name;
folder.type = EntityFolder.USER;
folder.parent = (parent == null ? null : parent.id);
folder.setProperties();
folder.inheritFrom(parent);
folder.id = db.folder().insertFolder(folder);
args.putLong("folder", folder.id);
db.setTransactionSuccessful();
} finally {
db.endTransaction();
}
ServiceSynchronize.reload(context, aid, true, "create folder");
return db.folder().getFoldersEx(aid);
}
@Override
protected void onExecuted(Bundle args, List<TupleFolderEx> folders) {
if (folders == null)
return;
adapter.set(folders);
long fid = args.getLong("folder");
int pos = adapter.getPositionForKey(fid);
if (pos == RecyclerView.NO_POSITION)
return;
llm.scrollToPositionWithOffset(pos, 0);
}
@Override
protected void onException(Bundle args, Throwable ex) {
Log.unexpectedError(getParentFragmentManager(), ex);
}
}.execute(this, args, "folder:add");
}
public static class FragmentDialogEditName extends FragmentDialogBase {
@NonNull
@Override
public Dialog onCreateDialog(@Nullable Bundle savedInstanceState) {
final Context context = getContext();
View view = LayoutInflater.from(context).inflate(R.layout.dialog_folder_add, null);
final EditText etName = view.findViewById(R.id.etName);
final Spinner spParent = view.findViewById(R.id.spParent);
etName.setOnEditorActionListener(new TextView.OnEditorActionListener() {
@Override
public boolean onEditorAction(TextView v, int actionId, KeyEvent event) {
if (actionId != EditorInfo.IME_ACTION_DONE)
return false;
AlertDialog dialog = (AlertDialog) getDialog();
if (dialog == null)
return false;
Button btnOk = dialog.getButton(AlertDialog.BUTTON_POSITIVE);
if (btnOk == null)
return false;
btnOk.performClick();
return true;
}
});
ArrayAdapter<String> adapter = new ArrayAdapter<>(context, R.layout.spinner_item1, android.R.id.text1);
adapter.setDropDownViewResource(R.layout.spinner_item1_dropdown);
spParent.setAdapter(adapter);
etName.setText(null);
view.post(new Runnable() {
@Override
public void run() {
etName.requestFocus();
Helper.showKeyboard(etName);
}
});
new SimpleTask<List<String>>() {
@Override
protected List<String> onExecute(Context context, Bundle args) throws Throwable {
long account = args.getLong("account");
DB db = DB.getInstance(context);
List<EntityFolder> folders = db.folder().getFolders(account, false, false);
if (folders == null)
return null;
Collator collator = Collator.getInstance(Locale.getDefault());
collator.setStrength(Collator.SECONDARY); // Case insensitive, process accents etc
Collections.sort(folders, new Comparator<EntityFolder>() {
@Override
public int compare(EntityFolder f1, EntityFolder f2) {
return collator.compare(f1.name, f2.name);
}
});
List<String> result = new ArrayList<>();
result.add("");
for (EntityFolder folder : folders)
result.add(folder.name);
return result;
}
@Override
protected void onExecuted(Bundle args, List<String> parents) {
adapter.clear();
adapter.addAll(parents);
}
@Override
protected void onException(Bundle args, Throwable ex) {
Log.unexpectedError(getParentFragmentManager(), ex);
}
}.execute(this, getArguments(), "folder:parents");
return new AlertDialog.Builder(context)
.setView(view)
.setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
String parent = (String) spParent.getSelectedItem();
String name = etName.getText().toString().trim();
if (TextUtils.isEmpty(name))
sendResult(RESULT_CANCELED);
else {
getArguments().putString("parent", parent);
getArguments().putString("name", name);
sendResult(RESULT_OK);
}
}
})
.setNegativeButton(android.R.string.cancel, null)
.create();
}
}
private static class Data { private static class Data {
private EntityAccount account;
private List<TupleFolderEx> folders; private List<TupleFolderEx> folders;
private List<EntityFolder> favorites; private List<EntityFolder> favorites;
} }

@ -0,0 +1,60 @@
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:padding="24dp">
<TextView
android:id="@+id/tvTitle"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:drawableStart="@drawable/twotone_add_24"
android:drawablePadding="6dp"
android:text="@string/title_create_folder"
android:textAppearance="@style/TextAppearance.AppCompat.Large"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<TextView
android:id="@+id/tvName"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="24dp"
android:labelFor="@+id/etName"
android:text="@string/title_create_folder_name"
android:textAppearance="@style/TextAppearance.AppCompat.Medium"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/tvTitle" />
<eu.faircode.email.EditTextPlain
android:id="@+id/etName"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:imeOptions="actionDone"
android:inputType="textPersonName|textCapWords"
android:text="Name"
android:textAppearance="@style/TextAppearance.AppCompat.Medium"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/tvName" />
<TextView
android:id="@+id/tvParent"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="12dp"
android:labelFor="@+id/spParent"
android:text="@string/title_create_folder_parent"
android:textAppearance="@style/TextAppearance.AppCompat.Medium"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/etName" />
<Spinner
android:id="@+id/spParent"
android:layout_width="0dp"
android:layout_height="wrap_content"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/tvParent" />
</androidx.constraintlayout.widget.ConstraintLayout>

@ -110,6 +110,8 @@
android:layout_width="0dp" android:layout_width="0dp"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:layout_marginTop="6dp" android:layout_marginTop="6dp"
android:clipToPadding="false"
android:paddingBottom="48dp"
android:scrollbarStyle="outsideOverlay" android:scrollbarStyle="outsideOverlay"
android:scrollbars="none" android:scrollbars="none"
app:fastScrollEnabled="false" app:fastScrollEnabled="false"
@ -123,6 +125,20 @@
app:layout_constraintStart_toStartOf="parent" app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/btnFavorite1" /> app:layout_constraintTop_toBottomOf="@+id/btnFavorite1" />
<com.google.android.material.floatingactionbutton.FloatingActionButton
android:id="@+id/fabAdd"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginEnd="6dp"
android:contentDescription="@string/title_add"
app:backgroundTint="?attr/colorFabBackground"
app:elevation="0dp"
app:fabSize="mini"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:srcCompat="@drawable/twotone_add_24"
app:tint="?attr/colorFabForeground" />
<eu.faircode.email.ContentLoadingProgressBar <eu.faircode.email.ContentLoadingProgressBar
android:id="@+id/pbWait" android:id="@+id/pbWait"
style="@style/Base.Widget.AppCompat.ProgressBar" style="@style/Base.Widget.AppCompat.ProgressBar"

@ -959,6 +959,10 @@
<string name="title_advanced_never_favorite">Never favorite</string> <string name="title_advanced_never_favorite">Never favorite</string>
<string name="title_advanced_edit_name">Edit name</string> <string name="title_advanced_edit_name">Edit name</string>
<string name="title_create_folder">Create folder</string>
<string name="title_create_folder_name">Folder name</string>
<string name="title_create_folder_parent">Parent folder</string>
<string name="title_advanced_swipe_actions">Set swipe actions</string> <string name="title_advanced_swipe_actions">Set swipe actions</string>
<string name="title_advanced_swipe_actions_hint">This will set the swipe left and right action for all IMAP accounts</string> <string name="title_advanced_swipe_actions_hint">This will set the swipe left and right action for all IMAP accounts</string>
<string name="title_advanced_swipe_sensitivity">Left/right swipe sensitivity</string> <string name="title_advanced_swipe_sensitivity">Left/right swipe sensitivity</string>

Loading…
Cancel
Save