diff --git a/app/build.gradle b/app/build.gradle index bfc791ad8..680c97c0d 100644 --- a/app/build.gradle +++ b/app/build.gradle @@ -14,8 +14,8 @@ android { targetSdkVersion rootProject.ext.targetSdkVersion compileSdk rootProject.ext.compileSdkVersion - versionCode 156 - versionName "3.0.5" + versionCode 157 + versionName "3.0.6" multiDexEnabled true resValue "string", "authorities", defaultConfig.applicationId + '.debug.cameraupload.provider' diff --git a/app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/BottomSheetMenuAdapter.java b/app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/BottomSheetMenuAdapter.java deleted file mode 100644 index ae88e6671..000000000 --- a/app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/BottomSheetMenuAdapter.java +++ /dev/null @@ -1,71 +0,0 @@ -package com.seafile.seadroid2.bottomsheetmenu; - -import android.view.LayoutInflater; -import android.view.MenuItem; -import android.view.View; -import android.view.ViewGroup; -import android.widget.ImageView; -import android.widget.TextView; - -import androidx.annotation.NonNull; -import androidx.recyclerview.widget.RecyclerView; - -import com.seafile.seadroid2.R; - -import java.util.ArrayList; -import java.util.List; - -public class BottomSheetMenuAdapter extends RecyclerView.Adapter { - private List dataList = new ArrayList<>(); - private int columnCount = 1; - private OnMenuClickListener onMenuClickListener; - - public BottomSheetMenuAdapter(List dataList, int columnCount) { - this.dataList = dataList; - this.columnCount = columnCount; - } - - public void setOnMenuClickListener(OnMenuClickListener onMenuClickListener) { - this.onMenuClickListener = onMenuClickListener; - } - - @NonNull - @Override - public BottomSheetViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { - int layoutId = columnCount == 1 ? R.layout.bottom_sheet_item_list : R.layout.bottom_sheet_item_grid; - View view = LayoutInflater.from(parent.getContext()).inflate(layoutId, parent, false); - return new BottomSheetViewHolder(view); - } - - @Override - public void onBindViewHolder(@NonNull BottomSheetViewHolder holder, int position) { - final int p = position; - holder.icon.setImageDrawable(dataList.get(p).getIcon()); - holder.name.setText(dataList.get(p).getTitle()); - if (onMenuClickListener != null) { - holder.itemView.setOnClickListener(new View.OnClickListener() { - @Override - public void onClick(View v) { - onMenuClickListener.onMenuClick(dataList.get(p)); - } - }); - } - } - - @Override - public int getItemCount() { - return dataList.size(); - } - - public static class BottomSheetViewHolder extends RecyclerView.ViewHolder { - public ImageView icon; - public TextView name; - - public BottomSheetViewHolder(@NonNull View itemView) { - super(itemView); - icon = itemView.findViewById(R.id.icon); - name = itemView.findViewById(R.id.name); - } - } - -} diff --git a/app/src/main/java/com/seafile/seadroid2/config/AbsLayoutItemType.java b/app/src/main/java/com/seafile/seadroid2/config/AbsLayoutItemType.java index d922b5eea..43aa784d4 100644 --- a/app/src/main/java/com/seafile/seadroid2/config/AbsLayoutItemType.java +++ b/app/src/main/java/com/seafile/seadroid2/config/AbsLayoutItemType.java @@ -11,6 +11,8 @@ public class AbsLayoutItemType { public static final int DIRENT_GRID = 12; public static final int DIRENT_GALLERY = 13; + public static final int SEARCH = 50; + public static final int ACTIVITY = 20; public static final int GROUP_ITEM = 30; diff --git a/app/src/main/java/com/seafile/seadroid2/context/NavContext.java b/app/src/main/java/com/seafile/seadroid2/context/NavContext.java index 3018fd620..a0cab4654 100644 --- a/app/src/main/java/com/seafile/seadroid2/context/NavContext.java +++ b/app/src/main/java/com/seafile/seadroid2/context/NavContext.java @@ -54,7 +54,7 @@ public void pop() { navStack.pop(); } - public void navToPath(RepoModel repoModel, String full_path) { + public void switchToPath(RepoModel repoModel, String full_path) { navStack.clear(); navStack.push(repoModel); diff --git a/app/src/main/java/com/seafile/seadroid2/enums/RepoSelectType.java b/app/src/main/java/com/seafile/seadroid2/enums/RepoSelectType.java index 72254c1c4..8d8421a74 100644 --- a/app/src/main/java/com/seafile/seadroid2/enums/RepoSelectType.java +++ b/app/src/main/java/com/seafile/seadroid2/enums/RepoSelectType.java @@ -11,7 +11,7 @@ public enum RepoSelectType { NOT_SELECTABLE(-1), ONLY_ACCOUNT(0), ONLY_REPO(1), - DIRENT(2); + FOLDER(2); RepoSelectType(int i) { diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/db/AppDatabase.java b/app/src/main/java/com/seafile/seadroid2/framework/data/db/AppDatabase.java index db8a5fa0c..f406282f7 100644 --- a/app/src/main/java/com/seafile/seadroid2/framework/data/db/AppDatabase.java +++ b/app/src/main/java/com/seafile/seadroid2/framework/data/db/AppDatabase.java @@ -1,7 +1,6 @@ package com.seafile.seadroid2.framework.data.db; import androidx.annotation.NonNull; -import androidx.room.Dao; import androidx.room.Database; import androidx.room.Room; import androidx.room.RoomDatabase; @@ -9,34 +8,34 @@ import androidx.sqlite.db.SupportSQLiteDatabase; import com.seafile.seadroid2.SeadroidApplication; -import com.seafile.seadroid2.framework.data.db.dao.CertCacheDAO; import com.seafile.seadroid2.framework.data.db.dao.DirentDAO; import com.seafile.seadroid2.framework.data.db.dao.EncKeyCacheDAO; import com.seafile.seadroid2.framework.data.db.dao.FileTransferDAO; import com.seafile.seadroid2.framework.data.db.dao.FolderBackupMonitorDAO; +import com.seafile.seadroid2.framework.data.db.dao.PermissionDAO; import com.seafile.seadroid2.framework.data.db.dao.RepoDAO; import com.seafile.seadroid2.framework.data.db.dao.StarredDirentDAO; -import com.seafile.seadroid2.framework.data.db.entities.CertEntity; import com.seafile.seadroid2.framework.data.db.entities.DirentModel; import com.seafile.seadroid2.framework.data.db.entities.EncKeyCacheEntity; import com.seafile.seadroid2.framework.data.db.entities.FolderBackupMonitorEntity; +import com.seafile.seadroid2.framework.data.db.entities.PermissionEntity; import com.seafile.seadroid2.framework.data.db.entities.RepoModel; import com.seafile.seadroid2.framework.data.db.entities.FileTransferEntity; import com.seafile.seadroid2.framework.data.db.entities.StarredModel; +import com.seafile.seadroid2.framework.util.SLogs; @Database(entities = { RepoModel.class, DirentModel.class, FolderBackupMonitorEntity.class, EncKeyCacheEntity.class, - CertEntity.class, FileTransferEntity.class, - StarredModel.class -}, version = 2, exportSchema = false) + StarredModel.class, + PermissionEntity.class, +}, version = 3, exportSchema = false) public abstract class AppDatabase extends RoomDatabase { private static final String DATABASE_NAME = "seafile_room.db"; private static volatile AppDatabase _instance; - public static AppDatabase getInstance() { if (_instance == null) { synchronized (AppDatabase.class) { @@ -44,6 +43,7 @@ public static AppDatabase getInstance() { _instance = Room .databaseBuilder(SeadroidApplication.getAppContext(), AppDatabase.class, DATABASE_NAME) .addMigrations(MIGRATION_1_2) + .addMigrations(MIGRATION_2_3) .build(); } } @@ -63,6 +63,33 @@ public void migrate(@NonNull SupportSQLiteDatabase database) { }; + static final Migration MIGRATION_2_3 = new Migration(2, 3) { + + @Override + public void migrate(@NonNull SupportSQLiteDatabase database) { + database.execSQL("DROP TABLE IF EXISTS cert_cache"); + + database.execSQL("DROP TABLE IF EXISTS permissions"); + database.execSQL("CREATE TABLE permissions (" + + "id INTEGER NOT NULL DEFAULT 0, " + + "repo_id TEXT NOT NULL DEFAULT '', " + + "description TEXT, " + + "name TEXT DEFAULT '', " + + "'create' INTEGER NOT NULL DEFAULT 0, " + + "upload INTEGER NOT NULL DEFAULT 0, " + + "download INTEGER NOT NULL DEFAULT 0, " + + "preview INTEGER NOT NULL DEFAULT 0, " + + "copy INTEGER NOT NULL DEFAULT 0, " + + "'delete' INTEGER NOT NULL DEFAULT 0, " + + "modify INTEGER NOT NULL DEFAULT 0, " + + "download_external_link INTEGER NOT NULL DEFAULT 0, " + + "'v' INTEGER NOT NULL DEFAULT 1, " + + "data_status INTEGER NOT NULL DEFAULT 0, " + + "PRIMARY KEY(repo_id, id))"); + } + }; + + public abstract RepoDAO repoDao(); public abstract DirentDAO direntDao(); @@ -71,11 +98,9 @@ public void migrate(@NonNull SupportSQLiteDatabase database) { public abstract EncKeyCacheDAO encKeyCacheDAO(); - @Deprecated - public abstract CertCacheDAO certDAO(); - public abstract FolderBackupMonitorDAO folderBackupMonitorDAO(); public abstract FileTransferDAO fileTransferDAO(); + public abstract PermissionDAO permissionDAO(); } diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/db/dao/CertCacheDAO.java b/app/src/main/java/com/seafile/seadroid2/framework/data/db/dao/CertCacheDAO.java deleted file mode 100644 index e7e8fe1ae..000000000 --- a/app/src/main/java/com/seafile/seadroid2/framework/data/db/dao/CertCacheDAO.java +++ /dev/null @@ -1,34 +0,0 @@ -package com.seafile.seadroid2.framework.data.db.dao; - -import androidx.room.Dao; -import androidx.room.Insert; -import androidx.room.OnConflictStrategy; -import androidx.room.Query; - -import com.seafile.seadroid2.framework.data.db.entities.CertEntity; - -import java.util.List; - -import io.reactivex.Completable; - -@Deprecated -@Dao -public interface CertCacheDAO { - @Insert(onConflict = OnConflictStrategy.REPLACE) - Completable insert(CertEntity entity); - - @Insert(onConflict = OnConflictStrategy.REPLACE) - Completable insertAllAsync(List entities); - - @Insert(onConflict = OnConflictStrategy.REPLACE) - void insertAll(List entities); - - @Query("DELETE FROM cert_cache where url = :u") - void deleteByUrl(String u); - - @Query("select * from cert_cache where url = :u limit 1") - List getListByUrl(String u); - -// @Query("select * from repo_config_cache where repo_id = :repoId limit 1") -// Single> getByRepoId(String repoId); -} diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/db/dao/DirentDAO.java b/app/src/main/java/com/seafile/seadroid2/framework/data/db/dao/DirentDAO.java index 0653c7bb2..faf9800dc 100644 --- a/app/src/main/java/com/seafile/seadroid2/framework/data/db/dao/DirentDAO.java +++ b/app/src/main/java/com/seafile/seadroid2/framework/data/db/dao/DirentDAO.java @@ -38,6 +38,9 @@ public interface DirentDAO { @Query("select * from dirents where uid in ( :uids )") List getListByIdsSync(List uids); + @Query("select * from dirents where uid in ( :uids )") + Single> getListByIdsAsync(List uids); + @Insert(onConflict = OnConflictStrategy.REPLACE) void insert(DirentModel model); diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/db/dao/PermissionDAO.java b/app/src/main/java/com/seafile/seadroid2/framework/data/db/dao/PermissionDAO.java new file mode 100644 index 000000000..5cbab7c71 --- /dev/null +++ b/app/src/main/java/com/seafile/seadroid2/framework/data/db/dao/PermissionDAO.java @@ -0,0 +1,44 @@ +package com.seafile.seadroid2.framework.data.db.dao; + +import androidx.room.Dao; +import androidx.room.Insert; +import androidx.room.OnConflictStrategy; +import androidx.room.Query; +import androidx.room.Update; + +import com.seafile.seadroid2.framework.data.db.entities.FileTransferEntity; +import com.seafile.seadroid2.framework.data.db.entities.PermissionEntity; +import com.seafile.seadroid2.framework.data.db.entities.PermissionEntity; + +import java.util.List; + +import io.reactivex.Completable; +import io.reactivex.Single; + +@Dao +public interface PermissionDAO { + + @Query("select * from permissions where repo_id = :repoId and id = :id limit 1") + List getByIdSync(String repoId, int id); + + @Query("select * from permissions where repo_id = :repoId and id = :id limit 1") + Single> getByIdAsync(String repoId, int id); + + @Query("select * from permissions where repo_id = :repoId") + Single> getByRepoIdAsync(String repoId); + + + @Query("DELETE FROM permissions") + void deleteAll(); + + @Insert(onConflict = OnConflictStrategy.REPLACE) + void insert(PermissionEntity PermissionEntity); + + @Insert(onConflict = OnConflictStrategy.REPLACE) + void insertAllSync(List list); + + @Insert(onConflict = OnConflictStrategy.REPLACE) + Completable insertAllAsync(List list); + + +} diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/CertEntity.java b/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/CertEntity.java deleted file mode 100644 index 92177c948..000000000 --- a/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/CertEntity.java +++ /dev/null @@ -1,22 +0,0 @@ -package com.seafile.seadroid2.framework.data.db.entities; - -import androidx.room.Entity; -import androidx.room.PrimaryKey; - -@Deprecated -@Entity(tableName = "cert_cache") -public class CertEntity { - @PrimaryKey(autoGenerate = true) - public long id; - - public String url; - public String cert; - - @Override - public String toString() { - return "CertEntity{" + - "url='" + url + '\'' + - ", cert='" + cert + '\'' + - '}'; - } -} diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/DirentModel.java b/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/DirentModel.java index 4b29a29f0..b0f7ff205 100644 --- a/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/DirentModel.java +++ b/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/DirentModel.java @@ -5,6 +5,7 @@ import android.text.TextUtils; import androidx.annotation.NonNull; +import androidx.annotation.Nullable; import androidx.room.Entity; import androidx.room.Ignore; import androidx.room.PrimaryKey; @@ -19,6 +20,8 @@ import com.seafile.seadroid2.framework.util.Times; import com.seafile.seadroid2.framework.util.Utils; +import org.apache.commons.lang3.StringUtils; + @Entity(tableName = "dirents") public class DirentModel extends BaseModel implements Parcelable { @@ -102,7 +105,9 @@ public String getSubtitle() { public int getIcon() { if (isDir()) { - if (!hasWritePermission()) { + if (isCustomPermission()) { + return R.drawable.baseline_folder_24; + } else if (!hasWritePermission()) { return R.drawable.baseline_folder_read_only_24; } else { return R.drawable.baseline_folder_24; @@ -111,6 +116,9 @@ public int getIcon() { return Icons.getFileIcon(name); } + /** + * You'll also need to check if it's a custom permission + */ public boolean hasWritePermission() { if (TextUtils.isEmpty(permission)) { return false; @@ -127,6 +135,9 @@ public boolean hasWritePermission() { return permission.contains("w"); } + /** + * You'll also need to check if it's a custom permission + */ public boolean hasDownloadPermission() { if (TextUtils.isEmpty(permission)) { return false; @@ -143,6 +154,15 @@ public boolean hasDownloadPermission() { return true; } + public boolean isCustomPermission() { + return !TextUtils.isEmpty(permission) && permission.startsWith("custom-"); + } + + public int getCustomPermissionNum() { + String[] ss = StringUtils.split(permission, "-"); + return Integer.parseInt(ss[1]); + } + public static DirentModel convertStarredModelToThis(StarredModel starredModel) { if (starredModel == null) { return null; diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/PermissionEntity.java b/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/PermissionEntity.java new file mode 100644 index 000000000..365037da8 --- /dev/null +++ b/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/PermissionEntity.java @@ -0,0 +1,28 @@ +package com.seafile.seadroid2.framework.data.db.entities; + +import androidx.annotation.NonNull; +import androidx.room.Entity; + +import com.seafile.seadroid2.framework.data.model.BaseModel; + +@Entity(tableName = "permissions", primaryKeys = {"repo_id", "id"}) +public class PermissionEntity extends BaseModel { + + @NonNull + public String repo_id = ""; + + public int id; + public String description; + public String name; + + //permissions + public boolean create; + public boolean upload; + public boolean download; + public boolean preview; + public boolean copy; + public boolean delete; + public boolean modify; + public boolean download_external_link; + //1001010 +} diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/RepoModel.java b/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/RepoModel.java index 5b73beed7..bc9784ebf 100644 --- a/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/RepoModel.java +++ b/app/src/main/java/com/seafile/seadroid2/framework/data/db/entities/RepoModel.java @@ -11,6 +11,8 @@ import com.seafile.seadroid2.framework.data.model.adapter.EncryptFieldJsonAdapter; import com.seafile.seadroid2.framework.util.Utils; +import org.apache.commons.lang3.StringUtils; + @Entity(tableName = "repos", primaryKeys = {"repo_id", "group_id"}) public class RepoModel extends BaseModel { @@ -78,14 +80,20 @@ public String getSubtitle() { } public int getIcon() { - if (encrypted) + if (encrypted) { return R.drawable.baseline_repo_encrypted_24; - if (!hasWritePermission()) + } else if (isCustomPermission()) { + return R.drawable.baseline_repo_24; + } else if (!hasWritePermission()) { return R.drawable.baseline_repo_readonly_24; + } return R.drawable.baseline_repo_24; } + /** + * You'll also need to check if it's a custom permission + */ public boolean hasWritePermission() { if (TextUtils.isEmpty(permission)) { return false; @@ -99,9 +107,27 @@ public boolean hasWritePermission() { return false; } - return !TextUtils.isEmpty(permission) && permission.contains("w"); + return permission.contains("w"); } + /** + * if start with "custom-" + */ + public boolean isCustomPermission() { + return !TextUtils.isEmpty(permission) && permission.startsWith("custom-"); + } + + /** + * please check {@link #isCustomPermission()} first + */ + public int getCustomPermissionNum() { + if (!isCustomPermission()) { + throw new IllegalArgumentException("please check isCustomPermission() first"); + } + + String[] ss = StringUtils.split(permission, "-"); + return Integer.parseInt(ss[1]); + } // /** // * If the result is true, and the decryption was successful, diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/model/BaseModel.java b/app/src/main/java/com/seafile/seadroid2/framework/data/model/BaseModel.java index ba76eaac6..fdcf7d675 100644 --- a/app/src/main/java/com/seafile/seadroid2/framework/data/model/BaseModel.java +++ b/app/src/main/java/com/seafile/seadroid2/framework/data/model/BaseModel.java @@ -2,6 +2,7 @@ import androidx.room.Ignore; +import com.seafile.seadroid2.config.Constants; import com.seafile.seadroid2.framework.data.db.entities.FileTransferEntity; import com.seafile.seadroid2.ui.data_migrate.DataMigrationActivity; @@ -28,7 +29,7 @@ public class BaseModel { /** * @see com.seafile.seadroid2.config.Constants.DataStatus */ - public int data_status = 0; + public int data_status = Constants.DataStatus.NORMAL; @Ignore public boolean is_checked = false; diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/model/ResultModel.java b/app/src/main/java/com/seafile/seadroid2/framework/data/model/ResultModel.java index 3a3bef276..512468939 100644 --- a/app/src/main/java/com/seafile/seadroid2/framework/data/model/ResultModel.java +++ b/app/src/main/java/com/seafile/seadroid2/framework/data/model/ResultModel.java @@ -3,4 +3,12 @@ public class ResultModel { public boolean success; public String error_msg; + + @Override + public String toString() { + return "ResultModel{" + + "success=" + success + + ", error_msg='" + error_msg + '\'' + + '}'; + } } diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/model/permission/PermissionListWrapperModel.java b/app/src/main/java/com/seafile/seadroid2/framework/data/model/permission/PermissionListWrapperModel.java new file mode 100644 index 000000000..8694d5045 --- /dev/null +++ b/app/src/main/java/com/seafile/seadroid2/framework/data/model/permission/PermissionListWrapperModel.java @@ -0,0 +1,9 @@ +package com.seafile.seadroid2.framework.data.model.permission; + +import com.seafile.seadroid2.framework.data.db.entities.PermissionEntity; + +import java.util.List; + +public class PermissionListWrapperModel { + public List permission_list; +} diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/model/permission/PermissionModel.java b/app/src/main/java/com/seafile/seadroid2/framework/data/model/permission/PermissionModel.java new file mode 100644 index 000000000..6c39febde --- /dev/null +++ b/app/src/main/java/com/seafile/seadroid2/framework/data/model/permission/PermissionModel.java @@ -0,0 +1,12 @@ +package com.seafile.seadroid2.framework.data.model.permission; + +public class PermissionModel { + public boolean create; + public boolean upload; + public boolean download; + public boolean preview; + public boolean copy; + public boolean delete; + public boolean modify; + public boolean download_external_link; +} diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/model/permission/PermissionWrapperModel.java b/app/src/main/java/com/seafile/seadroid2/framework/data/model/permission/PermissionWrapperModel.java new file mode 100644 index 000000000..46f14fdb5 --- /dev/null +++ b/app/src/main/java/com/seafile/seadroid2/framework/data/model/permission/PermissionWrapperModel.java @@ -0,0 +1,8 @@ +package com.seafile.seadroid2.framework.data.model.permission; + +public class PermissionWrapperModel { + public int id; + public String description; + public String name; + public PermissionModel permission; +} diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/model/repo/Dirent2Model.java b/app/src/main/java/com/seafile/seadroid2/framework/data/model/repo/Dirent2Model.java index 7fda920cf..47d42e265 100644 --- a/app/src/main/java/com/seafile/seadroid2/framework/data/model/repo/Dirent2Model.java +++ b/app/src/main/java/com/seafile/seadroid2/framework/data/model/repo/Dirent2Model.java @@ -13,4 +13,20 @@ public class Dirent2Model extends BaseModel { public String user_contact_email; public boolean repo_encrypted; public boolean is_dir; + + @Override + public String toString() { + return "Dirent2Model{" + + "repo_id='" + repo_id + '\'' + + ", repo_name='" + repo_name + '\'' + + ", path='" + path + '\'' + + ", obj_name='" + obj_name + '\'' + + ", mtime='" + mtime + '\'' + + ", user_email='" + user_email + '\'' + + ", user_name='" + user_name + '\'' + + ", user_contact_email='" + user_contact_email + '\'' + + ", repo_encrypted=" + repo_encrypted + + ", is_dir=" + is_dir + + '}'; + } } diff --git a/app/src/main/java/com/seafile/seadroid2/framework/data/model/search/SearchModel.java b/app/src/main/java/com/seafile/seadroid2/framework/data/model/search/SearchModel.java index 7be2f540c..51d5cb61d 100644 --- a/app/src/main/java/com/seafile/seadroid2/framework/data/model/search/SearchModel.java +++ b/app/src/main/java/com/seafile/seadroid2/framework/data/model/search/SearchModel.java @@ -30,6 +30,10 @@ public class SearchModel extends BaseModel implements Parcelable { public long last_modified; public long size; // size of file, 0 if type is dir + public boolean isDir() { + return is_dir; + } + public String getTitle() { String formatName = StringUtils.substringAfterLast(fullpath, '/'); if (TextUtils.isEmpty(formatName)) { @@ -46,7 +50,8 @@ public String getTitle() { // } public String getSubtitle() { - return repo_name; + String p = Utils.getPathFromFullPath(fullpath); + return repo_name + p; } @@ -69,6 +74,7 @@ public static DirentModel converterThis2DirentModel(SearchModel model) { d.name = model.name; d.repo_id = model.repo_id; d.repo_name = model.repo_name; + d.last_modified_at = model.last_modified; d.size = model.size; return d; } diff --git a/app/src/main/java/com/seafile/seadroid2/framework/datastore/sp/SettingsManager.java b/app/src/main/java/com/seafile/seadroid2/framework/datastore/sp/SettingsManager.java index 7a0cd636e..47ebf8b2d 100644 --- a/app/src/main/java/com/seafile/seadroid2/framework/datastore/sp/SettingsManager.java +++ b/app/src/main/java/com/seafile/seadroid2/framework/datastore/sp/SettingsManager.java @@ -4,6 +4,7 @@ import com.seafile.seadroid2.R; import com.seafile.seadroid2.SeadroidApplication; +import com.seafile.seadroid2.preferences.Settings; /** * Access the app settings @@ -98,4 +99,16 @@ public final class SettingsManager { public static final long DECRYPTION_EXPIRATION_TIME = 1000 * 60 * 60 * 24;//1 days public static final int REPO_ENC_VERSION = 2; + + + //force refresh starred list state + public static final String ON_FORCE_REFRESH_STARRED_LIST_KEY = "on_force_refresh_starred_list"; + + public static void setForceRefreshStarredListState() { + Settings.getCommonPreferences().edit().putBoolean(ON_FORCE_REFRESH_STARRED_LIST_KEY, true).apply(); + } + + public static boolean getForceRefreshStarredListState() { + return Settings.getCommonPreferences().getBoolean(ON_FORCE_REFRESH_STARRED_LIST_KEY, false); + } } diff --git a/app/src/main/java/com/seafile/seadroid2/framework/http/BaseOkHttpClient.java b/app/src/main/java/com/seafile/seadroid2/framework/http/BaseOkHttpClient.java index cf31fd645..1e3f73424 100644 --- a/app/src/main/java/com/seafile/seadroid2/framework/http/BaseOkHttpClient.java +++ b/app/src/main/java/com/seafile/seadroid2/framework/http/BaseOkHttpClient.java @@ -3,6 +3,7 @@ import android.text.TextUtils; import com.blankj.utilcode.util.NetworkUtils; +import com.seafile.seadroid2.BuildConfig; import com.seafile.seadroid2.SeadroidApplication; import com.seafile.seadroid2.account.Account; import com.seafile.seadroid2.framework.http.interceptor.HeaderInterceptor; @@ -47,7 +48,7 @@ protected List getInterceptors() { //print log HttpLoggingInterceptor loggingInterceptor = new HttpLoggingInterceptor(); loggingInterceptor.setLevel(HttpLoggingInterceptor.Level.BASIC); -// loggingInterceptor.setLevel(BuildConfig.DEBUG ? HttpLoggingInterceptor.Level.BODY : HttpLoggingInterceptor.Level.BASIC); + loggingInterceptor.setLevel(BuildConfig.DEBUG ? HttpLoggingInterceptor.Level.BODY : HttpLoggingInterceptor.Level.BASIC); interceptors.add(loggingInterceptor); if (account != null && !TextUtils.isEmpty(account.token)) { diff --git a/app/src/main/java/com/seafile/seadroid2/framework/util/Objs.java b/app/src/main/java/com/seafile/seadroid2/framework/util/Objs.java index 334ba299a..275987c7f 100644 --- a/app/src/main/java/com/seafile/seadroid2/framework/util/Objs.java +++ b/app/src/main/java/com/seafile/seadroid2/framework/util/Objs.java @@ -391,6 +391,7 @@ private static TreeMap> groupRepos(List repos //////dirent //////////////////////////// public static Single> getDirentsSingleFromServer(Account account, String repoId, String repoName, String parentDir) { + Single netSingle = HttpIO.getInstanceByAccount(account).execute(RepoService.class).getDirents(repoId, parentDir); return netSingle.flatMap(new Function>>() { @Override diff --git a/app/src/main/java/com/seafile/seadroid2/framework/util/Utils.java b/app/src/main/java/com/seafile/seadroid2/framework/util/Utils.java index c4b6341a8..886356dbf 100644 --- a/app/src/main/java/com/seafile/seadroid2/framework/util/Utils.java +++ b/app/src/main/java/com/seafile/seadroid2/framework/util/Utils.java @@ -75,7 +75,27 @@ public static JSONObject parseJsonObject(String json) { } } + public static String getPathFromFullPath(String path) { + if (path == null) { + // the caller should not give null + Log.w(DEBUG_TAG, "path is null"); + return null; + } + if (!path.contains("/")) { + return "/"; + } + + if (path.endsWith("/")) { + return path; + } + + String parent = path.substring(0, path.lastIndexOf("/")); + if (parent.isEmpty()) { + return "/"; + } else + return parent; + } public static String getParentPath(String path) { if (path == null) { // the caller should not give null diff --git a/app/src/main/java/com/seafile/seadroid2/ssl/CertsDBHelper.java b/app/src/main/java/com/seafile/seadroid2/ssl/CertsDBHelper.java index 3ffceb286..dc20576a3 100644 --- a/app/src/main/java/com/seafile/seadroid2/ssl/CertsDBHelper.java +++ b/app/src/main/java/com/seafile/seadroid2/ssl/CertsDBHelper.java @@ -1,23 +1,10 @@ package com.seafile.seadroid2.ssl; -import java.io.ByteArrayInputStream; -import java.io.ByteArrayOutputStream; -import java.io.IOException; -import java.io.ObjectInputStream; -import java.io.ObjectOutputStream; -import java.security.cert.X509Certificate; -import java.util.List; - import android.content.Context; -import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.database.sqlite.SQLiteOpenHelper; -import android.util.Base64; -import com.blankj.utilcode.util.CollectionUtils; import com.seafile.seadroid2.SeadroidApplication; -import com.seafile.seadroid2.framework.data.db.AppDatabase; -import com.seafile.seadroid2.framework.data.db.entities.CertEntity; public class CertsDBHelper extends SQLiteOpenHelper { // If you change the database schema, you must increment the database version. diff --git a/app/src/main/java/com/seafile/seadroid2/ssl/CertsManager.java b/app/src/main/java/com/seafile/seadroid2/ssl/CertsManager.java index 5cc91bb45..d39f018d5 100644 --- a/app/src/main/java/com/seafile/seadroid2/ssl/CertsManager.java +++ b/app/src/main/java/com/seafile/seadroid2/ssl/CertsManager.java @@ -5,21 +5,12 @@ import java.util.Map; import android.text.TextUtils; -import android.util.Log; import com.blankj.utilcode.util.EncryptUtils; import com.google.common.collect.Maps; -import com.seafile.seadroid2.R; -import com.seafile.seadroid2.framework.data.db.AppDatabase; -import com.seafile.seadroid2.framework.data.db.entities.CertEntity; -import com.seafile.seadroid2.framework.datastore.DataManager; import com.seafile.seadroid2.framework.datastore.DataStoreKeys; -import com.seafile.seadroid2.framework.datastore.DataStoreManager; -import com.seafile.seadroid2.framework.datastore.sp.SettingsManager; -import com.seafile.seadroid2.framework.util.ConcurrentAsyncTask; import com.seafile.seadroid2.account.Account; import com.seafile.seadroid2.preferences.Settings; -import com.seafile.seadroid2.preferences.SharedPreferencesHelper; /** * Save the ssl certificates the user has confirmed to trust diff --git a/app/src/main/java/com/seafile/seadroid2/ui/account/AccountDetailActivity.java b/app/src/main/java/com/seafile/seadroid2/ui/account/AccountDetailActivity.java index c6595be0c..b96d0ce27 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/account/AccountDetailActivity.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/account/AccountDetailActivity.java @@ -436,26 +436,4 @@ private void login() { getViewModel().login(tempAccount, passwd, authToken, rememberDevice); } - - private Dialog dialog; - - private void showProgressDialog() { - - if (dialog == null) { - MaterialAlertDialogBuilder builder = new MaterialAlertDialogBuilder(this); - builder.setView(R.layout.layout_dialog_progress_bar); - dialog = builder.create(); - } - - if (dialog.isShowing()) { - dialog.dismiss(); - } - dialog.show(); - } - - private void dismissProgressDialog() { - if (dialog != null) { - dialog.dismiss(); - } - } } diff --git a/app/src/main/java/com/seafile/seadroid2/ui/account/AccountsActivity.java b/app/src/main/java/com/seafile/seadroid2/ui/account/AccountsActivity.java index 5581f2764..97ee1984d 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/account/AccountsActivity.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/account/AccountsActivity.java @@ -203,28 +203,6 @@ private void onListItemClick(int position) { } } - private Dialog dialog; - - private void showProgressDialog() { - - if (dialog == null) { - MaterialAlertDialogBuilder builder = new MaterialAlertDialogBuilder(this); - builder.setView(R.layout.layout_dialog_progress_bar); - dialog = builder.create(); - } - - if (dialog.isShowing()) { - dialog.dismiss(); - } - dialog.show(); - } - - private void dismissProgressDialog() { - if (dialog != null) { - dialog.dismiss(); - } - } - @Override protected void onDestroy() { Log.d(DEBUG_TAG, "onDestroy"); diff --git a/app/src/main/java/com/seafile/seadroid2/ui/base/BaseActivity.java b/app/src/main/java/com/seafile/seadroid2/ui/base/BaseActivity.java index 380cc8381..db0bf401e 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/base/BaseActivity.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/base/BaseActivity.java @@ -1,5 +1,6 @@ package com.seafile.seadroid2.ui.base; +import android.app.Dialog; import android.os.Bundle; import androidx.annotation.NonNull; @@ -8,6 +9,7 @@ import androidx.appcompat.app.AppCompatDelegate; import androidx.appcompat.widget.Toolbar; +import com.google.android.material.dialog.MaterialAlertDialogBuilder; import com.seafile.seadroid2.R; import com.seafile.seadroid2.framework.helper.NightModeHelper; @@ -47,4 +49,32 @@ public void setContentView(int layoutResID) { super.setContentView(layoutResID); getActionBarToolbar(); } + + private Dialog dialog; + + public void showProgressDialog(boolean isShow) { + if (isShow) { + showProgressDialog(); + } else { + dismissProgressDialog(); + } + } + + public void showProgressDialog() { + if (dialog == null) { + MaterialAlertDialogBuilder builder = new MaterialAlertDialogBuilder(this); + builder.setView(R.layout.layout_dialog_progress_bar); + dialog = builder.create(); + } + + if (!dialog.isShowing()) { + dialog.show(); + } + } + + public void dismissProgressDialog() { + if (dialog != null && dialog.isShowing()) { + dialog.dismiss(); + } + } } diff --git a/app/src/main/java/com/seafile/seadroid2/ui/base/fragment/CustomDialogFragment.java b/app/src/main/java/com/seafile/seadroid2/ui/base/fragment/CustomDialogFragment.java index a0337386b..185c75638 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/base/fragment/CustomDialogFragment.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/base/fragment/CustomDialogFragment.java @@ -77,11 +77,11 @@ public void showLoading(boolean isShow) { @NonNull @Override public Dialog onCreateDialog(@Nullable Bundle savedInstanceState) { - rootView = LayoutInflater.from(requireContext()).inflate(R.layout.layout_dialog_container, null); + rootView = getLayoutInflater().inflate(R.layout.layout_dialog_container, null); LinearLayout containerView = rootView.findViewById(R.id.container); // - LayoutInflater.from(requireContext()).inflate(getLayoutId(), containerView); + getLayoutInflater().inflate(getLayoutId(), containerView); //action bar positiveView = rootView.findViewById(R.id.text_view_positive); diff --git a/app/src/main/java/com/seafile/seadroid2/ui/base/viewmodel/BaseViewModel.java b/app/src/main/java/com/seafile/seadroid2/ui/base/viewmodel/BaseViewModel.java index b06e1fbb5..8cbc4d260 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/base/viewmodel/BaseViewModel.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/base/viewmodel/BaseViewModel.java @@ -52,7 +52,11 @@ public class BaseViewModel extends ViewModel { private final MutableLiveData RefreshLiveData = new MutableLiveData<>(false); private final MutableLiveData> ExceptionLiveData = new MutableLiveData<>(); private final MutableLiveData SeafExceptionLiveData = new MutableLiveData<>(); + private final MutableLiveData ShowLoadingDialogLiveData = new MutableLiveData<>(false); + public MutableLiveData getShowLoadingDialogLiveData() { + return ShowLoadingDialogLiveData; + } public MutableLiveData getRefreshLiveData() { return RefreshLiveData; diff --git a/app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/ActionMenu.java b/app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/ActionMenu.java similarity index 98% rename from app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/ActionMenu.java rename to app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/ActionMenu.java index eea46dd88..72b8826cf 100644 --- a/app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/ActionMenu.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/ActionMenu.java @@ -14,7 +14,7 @@ * limitations under the License. */ -package com.seafile.seadroid2.bottomsheetmenu; +package com.seafile.seadroid2.ui.bottomsheetmenu; import android.content.ComponentName; import android.content.Context; @@ -29,7 +29,7 @@ import java.util.Iterator; import java.util.List; -class ActionMenu implements android.view.Menu { +public class ActionMenu implements android.view.Menu { private static final int[] sCategoryToOrder = new int[]{ 1, /* No category */ 4, /* CONTAINER */ @@ -58,7 +58,7 @@ class ActionMenu implements android.view.Menu { private boolean mIsQwerty; private ArrayList mItems; - ActionMenu(Context context) { + public ActionMenu(Context context) { mContext = context; mItems = new ArrayList<>(); } diff --git a/app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/ActionMenuItem.java b/app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/ActionMenuItem.java similarity index 99% rename from app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/ActionMenuItem.java rename to app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/ActionMenuItem.java index 47f28f68c..0c6399341 100644 --- a/app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/ActionMenuItem.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/ActionMenuItem.java @@ -1,4 +1,4 @@ -package com.seafile.seadroid2.bottomsheetmenu; +package com.seafile.seadroid2.ui.bottomsheetmenu; /* * Copyright (C) 2010 The Android Open Source Project diff --git a/app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/BottomSheetHelper.java b/app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/BottomSheetHelper.java similarity index 96% rename from app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/BottomSheetHelper.java rename to app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/BottomSheetHelper.java index 77ca716dd..b1415e039 100644 --- a/app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/BottomSheetHelper.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/BottomSheetHelper.java @@ -1,4 +1,4 @@ -package com.seafile.seadroid2.bottomsheetmenu; +package com.seafile.seadroid2.ui.bottomsheetmenu; import android.view.MenuItem; diff --git a/app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/BottomSheetMenuAdapter.java b/app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/BottomSheetMenuAdapter.java new file mode 100644 index 000000000..53c286cd5 --- /dev/null +++ b/app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/BottomSheetMenuAdapter.java @@ -0,0 +1,67 @@ +package com.seafile.seadroid2.ui.bottomsheetmenu; + +import android.content.Context; +import android.content.res.ColorStateList; +import android.view.LayoutInflater; +import android.view.MenuItem; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ImageView; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.annotation.Nullable; +import androidx.core.content.ContextCompat; + +import com.seafile.seadroid2.R; +import com.seafile.seadroid2.ui.base.adapter.BaseAdapter; +import com.seafile.seadroid2.ui.base.viewholder.BaseViewHolder; + +public class BottomSheetMenuAdapter extends BaseAdapter { + private final int columnCount; + + public BottomSheetMenuAdapter(int columnCount) { + this.columnCount = columnCount; + } + + @Override + protected void onBindViewHolder(@NonNull BottomSheetMenuAdapter.BottomSheetViewHolder holder, int i, @Nullable MenuItem menuItem) { + if (menuItem == null) { + return; + } + + holder.icon.setImageDrawable(menuItem.getIcon()); + holder.name.setText(menuItem.getTitle()); + + holder.name.setEnabled(menuItem.isEnabled()); + holder.itemView.setClickable(menuItem.isEnabled()); + int color; + if (menuItem.isEnabled()) { + color = ContextCompat.getColor(getContext(), R.color.material_grey_600); + } else { + color = ContextCompat.getColor(getContext(), R.color.material_grey_400); + } + holder.icon.setImageTintList(ColorStateList.valueOf(color)); + + } + + @NonNull + @Override + protected BottomSheetMenuAdapter.BottomSheetViewHolder onCreateViewHolder(@NonNull Context context, @NonNull ViewGroup viewGroup, int i) { + int layoutId = columnCount == 1 ? R.layout.bottom_sheet_item_list : R.layout.bottom_sheet_item_grid; + View view = LayoutInflater.from(getContext()).inflate(layoutId, viewGroup, false); + return new BottomSheetMenuAdapter.BottomSheetViewHolder(view); + } + + public static class BottomSheetViewHolder extends BaseViewHolder { + public ImageView icon; + public TextView name; + + public BottomSheetViewHolder(@NonNull View itemView) { + super(itemView); + icon = itemView.findViewById(R.id.icon); + name = itemView.findViewById(R.id.text); + } + } + +} diff --git a/app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/BottomSheetMenuFragment.java b/app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/BottomSheetMenuFragment.java similarity index 80% rename from app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/BottomSheetMenuFragment.java rename to app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/BottomSheetMenuFragment.java index af28dff79..e561e1620 100644 --- a/app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/BottomSheetMenuFragment.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/BottomSheetMenuFragment.java @@ -1,4 +1,4 @@ -package com.seafile.seadroid2.bottomsheetmenu; +package com.seafile.seadroid2.ui.bottomsheetmenu; import android.app.Dialog; import android.content.Context; @@ -19,6 +19,7 @@ import androidx.recyclerview.widget.RecyclerView; import com.blankj.utilcode.util.CollectionUtils; +import com.chad.library.adapter4.BaseQuickAdapter; import com.google.android.material.bottomsheet.BottomSheetDialog; import com.google.android.material.bottomsheet.BottomSheetDialogFragment; import com.seafile.seadroid2.R; @@ -49,7 +50,7 @@ public void onCreate(@Nullable Bundle savedInstanceState) { @Override public Dialog onCreateDialog(Bundle savedInstanceState) { BottomSheetDialog bottomSheetDialog = new BottomSheetDialog(requireContext(), R.style.ThemeOverlay_Catalog_BottomSheetDialog_Scrollable); - bottomSheetDialog.setContentView(R.layout.layout_bottom_sheet_recycler_menu); + bottomSheetDialog.setContentView(R.layout.layout_bottom_sheet_menu_dialog); bottomSheetDialog.setDismissWithAnimation(true); View parentView = bottomSheetDialog.findViewById(R.id.bottom_sheet_container); @@ -64,29 +65,37 @@ public void setView(View rootView) { TextView title = rootView.findViewById(R.id.title); title.setText(builder.title); } - - if (!CollectionUtils.isEmpty(builder.menuItems)) { - RecyclerView rv = rootView.findViewById(R.id.rv); - if (builder.columnCount == 1) { - rv.setLayoutManager(new LinearLayoutManager(requireContext())); - } else { - rv.setLayoutManager(new GridLayoutManager(requireContext(), builder.columnCount)); - } - BottomSheetMenuAdapter adapter = new BottomSheetMenuAdapter(builder.menuItems, builder.columnCount); - adapter.setOnMenuClickListener(new OnMenuClickListener() { - @Override - public void onMenuClick(MenuItem menuItem) { - dismiss(); - if (builder.onMenuClickListener != null) { - builder.onMenuClickListener.onMenuClick(menuItem); - } - } - }); + if (CollectionUtils.isEmpty(builder.menuItems)) { + return; + } - rv.setAdapter(adapter); + RecyclerView rv = rootView.findViewById(R.id.rv); + if (builder.columnCount == 1) { + rv.setLayoutManager(new LinearLayoutManager(requireContext())); + } else { + rv.setLayoutManager(new GridLayoutManager(requireContext(), builder.columnCount)); } + BottomSheetMenuAdapter adapter = getBottomSheetMenuAdapter(); + adapter.submitList(builder.menuItems); + + rv.setAdapter(adapter); + + } + + private @NonNull BottomSheetMenuAdapter getBottomSheetMenuAdapter() { + BottomSheetMenuAdapter adapter = new BottomSheetMenuAdapter(builder.columnCount); + adapter.setOnItemClickListener(new BaseQuickAdapter.OnItemClickListener() { + @Override + public void onClick(@NonNull BaseQuickAdapter baseQuickAdapter, @NonNull View view, int i) { + dismiss(); + if (builder.onMenuClickListener != null) { + builder.onMenuClickListener.onMenuClick(adapter.getItem(i)); + } + } + }); + return adapter; } @@ -122,7 +131,6 @@ public Builder setMenuSheetId(int menuSheetId) { inflater.inflate(menuSheetId, menu); List items = new ArrayList(menu.size()); - for (int i = 0; i < menu.size(); i++) { items.add(menu.getItem(i)); } diff --git a/app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/OnMenuClickListener.java b/app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/OnMenuClickListener.java similarity index 69% rename from app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/OnMenuClickListener.java rename to app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/OnMenuClickListener.java index e6bae6e4f..f76e4ca3e 100644 --- a/app/src/main/java/com/seafile/seadroid2/bottomsheetmenu/OnMenuClickListener.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/bottomsheetmenu/OnMenuClickListener.java @@ -1,4 +1,4 @@ -package com.seafile.seadroid2.bottomsheetmenu; +package com.seafile.seadroid2.ui.bottomsheetmenu; import android.view.MenuItem; diff --git a/app/src/main/java/com/seafile/seadroid2/ui/data_migrate/DataMigrationActivity.java b/app/src/main/java/com/seafile/seadroid2/ui/data_migrate/DataMigrationActivity.java index e5c71c37c..b7a26bb75 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/data_migrate/DataMigrationActivity.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/data_migrate/DataMigrationActivity.java @@ -26,7 +26,6 @@ import com.seafile.seadroid2.databinding.ActivityDataMigrationBinding; import com.seafile.seadroid2.framework.data.DatabaseHelper; import com.seafile.seadroid2.framework.data.db.AppDatabase; -import com.seafile.seadroid2.framework.data.db.entities.CertEntity; import com.seafile.seadroid2.framework.data.db.entities.EncKeyCacheEntity; import com.seafile.seadroid2.framework.data.db.entities.FileTransferEntity; import com.seafile.seadroid2.framework.data.db.entities.FolderBackupMonitorEntity; @@ -986,34 +985,25 @@ private void queryCertsDB() { null // The sort order ); - List list = CollectionUtils.newArrayList(); try { c.moveToFirst(); while (!c.isAfterLast()) { - CertEntity item = new CertEntity(); int urlIndex = c.getColumnIndexOrThrow("url"); int certIndex = c.getColumnIndexOrThrow("cert"); - item.url = c.getString(urlIndex); - item.cert = c.getString(certIndex); + String url = c.getString(urlIndex); + String cert = c.getString(certIndex); c.moveToNext(); - list.add(item); + String keyPrefix = EncryptUtils.encryptMD5ToString(url); + Settings.getCommonPreferences().edit().putString(DataStoreKeys.KEY_SERVER_CERT_INFO + "_" + keyPrefix, cert).apply(); } } finally { c.close(); } - SLogs.d("--------------------" + table); - for (CertEntity entity : list) { - SLogs.d(entity.toString()); - - String keyPrefix = EncryptUtils.encryptMD5ToString(entity.url); - Settings.getCommonPreferences().edit().putString(DataStoreKeys.KEY_SERVER_CERT_INFO + "_" + keyPrefix, entity.cert).apply(); - } - //Not stored in the database // AppDatabase.getInstance().certDAO().insertAll(list); SLogs.d("--------------------" + table + " -> 完成"); diff --git a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/CopyMoveDialogFragment.java b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/CopyMoveDialogFragment.java index 46e3d8156..0c7ac4a39 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/CopyMoveDialogFragment.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/CopyMoveDialogFragment.java @@ -37,11 +37,14 @@ protected int getLayoutId() { @Override public int getDialogTitleRes() { if (ctx != null) { - if (ctx.isdir) { - return ctx.isCopy() ? R.string.copy_folder_ing : R.string.move_folder_ing; - } else { - return ctx.isCopy() ? R.string.copy_file_ing : R.string.move_file_ing; - } +// if (ctx.isdir) { +// return ctx.isCopy() ? R.string.copy_folder_ing : R.string.move_folder_ing; +// } else { +// return ctx.isCopy() ? R.string.copy_file_ing : R.string.move_file_ing; +// } + + return ctx.isCopy() ? R.string.copy_file_ing : R.string.move_file_ing; + } return super.getDialogTitleRes(); @@ -52,11 +55,13 @@ protected void initView(LinearLayout containerView) { super.initView(containerView); int strMsgId; - if (ctx.isdir) { - strMsgId = ctx.isCopy() ? R.string.copy_file_from : R.string.move_file_from; - } else { - strMsgId = ctx.isCopy() ? R.string.copy_file_from : R.string.move_file_from; - } +// if (ctx.isdir) { +// strMsgId = ctx.isCopy() ? R.string.copy_file_from : R.string.move_file_from; +// } else { +// strMsgId = ctx.isCopy() ? R.string.copy_file_from : R.string.move_file_from; +// } + + strMsgId = ctx.isCopy() ? R.string.copy_file_from : R.string.move_file_from; String strMsg = getString(strMsgId); diff --git a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/DeleteFileDialogFragment.java b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/DeleteFileDialogFragment.java index 653273e47..e47c448e8 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/DeleteFileDialogFragment.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/DeleteFileDialogFragment.java @@ -1,8 +1,11 @@ package com.seafile.seadroid2.ui.dialog_fragment; +import android.os.Bundle; +import android.os.Parcelable; import android.widget.LinearLayout; import android.widget.TextView; +import androidx.annotation.Nullable; import androidx.lifecycle.Observer; import com.blankj.utilcode.util.CollectionUtils; @@ -17,22 +20,29 @@ import java.util.List; public class DeleteFileDialogFragment extends RequestCustomDialogFragmentWithVM { - private List dirents; - private boolean isDir; - - public static DeleteFileDialogFragment newInstance() { - return new DeleteFileDialogFragment(); + private List dirents; + private boolean isDir = false; + + public static DeleteFileDialogFragment newInstance(List direntIds) { + DeleteFileDialogFragment fragment = new DeleteFileDialogFragment(); + Bundle bundle = new Bundle(); + bundle.putStringArrayList("dirent_ids", new ArrayList<>(direntIds)); + fragment.setArguments(bundle); + return fragment; } - public void initData(DirentModel dirent) { - this.dirents = new ArrayList<>(); - dirents.add(dirent); - isDir = dirent.isDir(); - } + @Override + public void onCreate(@Nullable Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + + Bundle bundle = getArguments(); + + if (bundle == null || !bundle.containsKey("dirent_ids")) { + throw new RuntimeException("need a dirent_ids param"); + } + + dirents = bundle.getStringArrayList("dirent_ids"); - public void initData(List dirents) { - this.dirents = dirents; - isDir = false; } @Override @@ -47,8 +57,7 @@ protected void onPositiveClick() { return; } - Account account = SupportAccountManager.getInstance().getCurrentAccount(); - getViewModel().deleteDirents(account.getSignature(), dirents, false); + getViewModel().delete(dirents, false); } @Override diff --git a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/DeleteRepoDialogFragment.java b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/DeleteRepoDialogFragment.java index b52185acc..119ba31d9 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/DeleteRepoDialogFragment.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/DeleteRepoDialogFragment.java @@ -8,18 +8,22 @@ import androidx.annotation.Nullable; import androidx.lifecycle.Observer; +import com.blankj.utilcode.util.CollectionUtils; import com.blankj.utilcode.util.ToastUtils; import com.seafile.seadroid2.R; import com.seafile.seadroid2.ui.base.fragment.RequestCustomDialogFragmentWithVM; import com.seafile.seadroid2.framework.data.model.ResultModel; import com.seafile.seadroid2.ui.dialog_fragment.viewmodel.DeleteRepoViewModel; +import java.util.ArrayList; +import java.util.List; + public class DeleteRepoDialogFragment extends RequestCustomDialogFragmentWithVM { - private String repoId; + private List repoIds; - public static DeleteRepoDialogFragment newInstance(String repoId) { + public static DeleteRepoDialogFragment newInstance(List repoIds) { Bundle args = new Bundle(); - args.putString("repoId", repoId); + args.putStringArrayList("repo_ids", new ArrayList<>(repoIds)); DeleteRepoDialogFragment fragment = new DeleteRepoDialogFragment(); fragment.setArguments(args); return fragment; @@ -28,18 +32,14 @@ public static DeleteRepoDialogFragment newInstance(String repoId) { @Override public void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); - if (getArguments() == null) { - throw new IllegalArgumentException("this dialogFragment need Arguments"); - } - if (!getArguments().containsKey("repoId")) { - throw new IllegalArgumentException("this dialogFragment need repoId param"); + if (getArguments() == null || !getArguments().containsKey("repo_ids")) { + throw new IllegalArgumentException("this dialogFragment need Arguments"); } - repoId = getArguments().getString("repoId", null); - - if (TextUtils.isEmpty(repoId)) { - throw new IllegalArgumentException("this dialogFragment need repoId param"); + repoIds = getArguments().getStringArrayList("repo_ids"); + if (CollectionUtils.isEmpty(repoIds)) { + throw new IllegalArgumentException("need repoIds param"); } } @@ -55,7 +55,7 @@ public int getDialogTitleRes() { @Override protected void onPositiveClick() { - getViewModel().deleteRepo(repoId); + getViewModel().deleteRepo(repoIds); } @Override diff --git a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/DialogService.java b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/DialogService.java index 8eff6cbfe..7a9a88ed0 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/DialogService.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/DialogService.java @@ -9,6 +9,7 @@ import java.util.List; import java.util.Map; +import io.reactivex.Flowable; import io.reactivex.Single; import okhttp3.RequestBody; import retrofit2.http.Body; diff --git a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/RenameDialogFragment.java b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/RenameDialogFragment.java index 76d7216b3..596a220ce 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/RenameDialogFragment.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/RenameDialogFragment.java @@ -1,35 +1,61 @@ package com.seafile.seadroid2.ui.dialog_fragment; +import android.os.Bundle; import android.text.Editable; import android.text.TextUtils; import android.widget.EditText; import android.widget.LinearLayout; +import androidx.annotation.Nullable; import androidx.lifecycle.Observer; import com.blankj.utilcode.util.ToastUtils; import com.seafile.seadroid2.R; +import com.seafile.seadroid2.framework.data.db.entities.DirentModel; import com.seafile.seadroid2.ui.base.fragment.RequestCustomDialogFragmentWithVM; import com.seafile.seadroid2.framework.data.model.dirents.FileCreateModel; import com.seafile.seadroid2.ui.dialog_fragment.viewmodel.RenameRepoViewModel; -public class RenameDialogFragment extends RequestCustomDialogFragmentWithVM { - private String curName, repoId, curPath, type; +import java.util.ArrayList; +import java.util.List; - public static RenameDialogFragment newInstance() { - return new RenameDialogFragment(); - } +public class RenameDialogFragment extends RequestCustomDialogFragmentWithVM { /** - * @param type "repo" or "dir" or "file" + * "repo" or "dir" or "file" */ - public void initData(String curName, String curPath, String repoId, String type) { - this.curName = curName; - this.curPath = curPath; - this.repoId = repoId; - this.type = type; + private String type; + private String curName, repoId, curPath; + + public static RenameDialogFragment newInstance(String curName, String curPath, String repoId, String type) { + + RenameDialogFragment fragment = new RenameDialogFragment(); + Bundle bundle = new Bundle(); + bundle.putString("name", curName); + bundle.putString("path", curPath); + bundle.putString("repoId", repoId); + bundle.putString("type", type); + fragment.setArguments(bundle); + return fragment; } + @Override + public void onCreate(@Nullable Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + + Bundle bundle = getArguments(); + + if (bundle == null || !bundle.containsKey("repoId")) { + throw new RuntimeException("need a dirent param"); + } + + curName = bundle.getString("name"); + curPath = bundle.getString("path"); + repoId = bundle.getString("repoId"); + type = bundle.getString("type"); + } + + @Override protected int getLayoutId() { return R.layout.view_dialog_new_file; diff --git a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/viewmodel/DeleteDirsViewModel.java b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/viewmodel/DeleteDirsViewModel.java index 93bd12646..045bcc476 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/viewmodel/DeleteDirsViewModel.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/viewmodel/DeleteDirsViewModel.java @@ -33,8 +33,27 @@ public MutableLiveData getActionLiveData() { return ActionLiveData; } + public void delete(List direntIds, boolean isDeleteLocalFile) { + Single> dSingle = AppDatabase.getInstance().direntDao().getListByIdsAsync(direntIds); + addSingleDisposable(dSingle, new Consumer>() { + @Override + public void accept(List dirents) throws Exception { + deleteDirents(dirents, isDeleteLocalFile); + } + }, new Consumer() { + @Override + public void accept(Throwable throwable) throws Exception { + getRefreshLiveData().setValue(false); + getActionLiveData().setValue(true); + + String string = getErrorMsgByThrowable(throwable); + DeleteDirentModel d = new DeleteDirentModel(); + d.error_msg = string; + } + }); + } - public void deleteDirents(String related_account, List dirents, boolean isDeleteLocalFile) { + private void deleteDirents(List dirents, boolean isDeleteLocalFile) { getRefreshLiveData().setValue(true); Flowable flowable; @@ -43,8 +62,7 @@ public void deleteDirents(String related_account, List dirents, boo .flatMapSingle(new Function>() { @Override public SingleSource apply(DirentModel dirent) throws Exception { - String obj = dirent.isDir() ? "dir" : "file"; - return HttpIO.getCurrentInstance().execute(DialogService.class).deleteDirent(dirent.repo_id, obj, dirent.full_path); + return HttpIO.getCurrentInstance().execute(DialogService.class).deleteDirent(dirent.repo_id, dirent.type, dirent.full_path); } }); } else { @@ -89,8 +107,7 @@ public SingleSource apply(String filePath) throws Exception { } } - String obj = dirent.isDir() ? "dir" : "file"; - return HttpIO.getCurrentInstance().execute(DialogService.class).deleteDirent(dirent.repo_id, obj, dirent.full_path); + return HttpIO.getCurrentInstance().execute(DialogService.class).deleteDirent(dirent.repo_id, dirent.type, dirent.full_path); } }); } diff --git a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/viewmodel/DeleteRepoViewModel.java b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/viewmodel/DeleteRepoViewModel.java index 1ffb31cbe..bf55d0333 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/viewmodel/DeleteRepoViewModel.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/dialog_fragment/viewmodel/DeleteRepoViewModel.java @@ -4,8 +4,10 @@ import androidx.lifecycle.MutableLiveData; +import com.blankj.utilcode.util.CollectionUtils; import com.seafile.seadroid2.framework.datastore.sp_livedata.AlbumBackupSharePreferenceHelper; import com.seafile.seadroid2.framework.datastore.sp_livedata.FolderBackupSharePreferenceHelper; +import com.seafile.seadroid2.framework.util.SLogs; import com.seafile.seadroid2.preferences.Settings; import com.seafile.seadroid2.ui.base.viewmodel.BaseViewModel; import com.seafile.seadroid2.framework.data.model.ResultModel; @@ -13,8 +15,19 @@ import com.seafile.seadroid2.ui.dialog_fragment.DialogService; import com.seafile.seadroid2.ui.folder_backup.RepoConfig; +import org.reactivestreams.Publisher; + +import java.util.ArrayList; +import java.util.List; +import java.util.concurrent.Flow; + +import io.reactivex.Flowable; import io.reactivex.Single; +import io.reactivex.SingleSource; +import io.reactivex.functions.Action; import io.reactivex.functions.Consumer; +import io.reactivex.functions.Function; +import kotlin.Pair; public class DeleteRepoViewModel extends BaseViewModel { private final MutableLiveData ActionLiveData = new MutableLiveData<>(); @@ -23,20 +36,33 @@ public MutableLiveData getActionLiveData() { return ActionLiveData; } - public void deleteRepo(String repoId) { + public void deleteRepo(List repoIds) { + if (CollectionUtils.isEmpty(repoIds)) { + return; + } getRefreshLiveData().setValue(true); - Single single = HttpIO.getCurrentInstance().execute(DialogService.class).deleteRepo(repoId); - addSingleDisposable(single, new Consumer() { + List>> flowableList = new ArrayList<>(); + for (String repoId : repoIds) { + flowableList.add(getDeleteFlowable(repoId)); + } + + Flowable> mergedFlowable = Flowable.mergeDelayError(flowableList, 5, Flowable.bufferSize()); + addFlowableDisposable(mergedFlowable, new Consumer>() { @Override - public void accept(String resultModel) throws Exception { - getRefreshLiveData().setValue(false); + public void accept(Pair pair) throws Exception { + + SLogs.d("DeleteRepoViewModel deleteRepo result: repoId = " + pair.getFirst() + ", result = " + pair.getSecond()); + + + String repoId = pair.getFirst(); + String o = pair.getSecond(); ResultModel resultModel1 = new ResultModel(); - if (TextUtils.equals("success", resultModel)) { + if (TextUtils.equals("success", o)) { resultModel1.success = true; } else { - resultModel1.error_msg = resultModel; + resultModel1.error_msg = o; } if (resultModel1.success) { @@ -52,8 +78,6 @@ public void accept(String resultModel) throws Exception { FolderBackupSharePreferenceHelper.writeBackupSwitch(false); } } - - getActionLiveData().setValue(resultModel1); } }, new Consumer() { @Override @@ -64,8 +88,25 @@ public void accept(Throwable throwable) throws Exception { resultModel.error_msg = getErrorMsgByThrowable(throwable); getActionLiveData().setValue(resultModel); } + }, new Action() { + @Override + public void run() throws Exception { + //todo 检查一下getActionLiveData + getRefreshLiveData().setValue(false); + getActionLiveData().setValue(new ResultModel()); + } }); } + private Flowable> getDeleteFlowable(String repoId) { + Single single = HttpIO.getCurrentInstance().execute(DialogService.class).deleteRepo(repoId); + return single.flatMap(new Function>>() { + @Override + public SingleSource> apply(String s) throws Exception { + return Single.just(new Pair<>(repoId, s)); + } + }).toFlowable(); + } + } diff --git a/app/src/main/java/com/seafile/seadroid2/ui/editor/EditorActivity.java b/app/src/main/java/com/seafile/seadroid2/ui/editor/EditorActivity.java index 4e4fa1f7c..89bec8bf6 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/editor/EditorActivity.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/editor/EditorActivity.java @@ -181,25 +181,6 @@ public void afterTextChanged(Editable s) { }); } - private Dialog dialog; - - private void showProgressDialog() { - - if (dialog == null) { - MaterialAlertDialogBuilder builder = new MaterialAlertDialogBuilder(this); - builder.setView(R.layout.layout_dialog_progress_bar); - dialog = builder.create(); - } - - dialog.show(); - } - - private void dismissProgressDialog() { - if (dialog != null) { - dialog.dismiss(); - } - } - @Override public void onDetachedFromWindow() { mPerformEdit.clearHistory(); diff --git a/app/src/main/java/com/seafile/seadroid2/ui/folder_backup/FolderBackupSelectedPathActivity.java b/app/src/main/java/com/seafile/seadroid2/ui/folder_backup/FolderBackupSelectedPathActivity.java index c378eb72d..bac328156 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/folder_backup/FolderBackupSelectedPathActivity.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/folder_backup/FolderBackupSelectedPathActivity.java @@ -19,8 +19,8 @@ import com.chad.library.adapter4.QuickAdapterHelper; import com.seafile.seadroid2.R; -import com.seafile.seadroid2.bottomsheetmenu.BottomSheetHelper; -import com.seafile.seadroid2.bottomsheetmenu.BottomSheetMenuFragment; +import com.seafile.seadroid2.ui.bottomsheetmenu.BottomSheetHelper; +import com.seafile.seadroid2.ui.bottomsheetmenu.BottomSheetMenuFragment; import com.seafile.seadroid2.framework.datastore.sp_livedata.FolderBackupSharePreferenceHelper; import com.seafile.seadroid2.ui.base.BaseActivity; diff --git a/app/src/main/java/com/seafile/seadroid2/ui/main/MainActivity.java b/app/src/main/java/com/seafile/seadroid2/ui/main/MainActivity.java index d6aef4156..e0e86fc00 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/main/MainActivity.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/main/MainActivity.java @@ -2,7 +2,6 @@ import android.Manifest; import android.app.Activity; -import android.app.Dialog; import android.content.ComponentName; import android.content.Context; import android.content.DialogInterface; @@ -50,6 +49,7 @@ import com.seafile.seadroid2.framework.data.ServerInfo; import com.seafile.seadroid2.framework.data.db.entities.EncKeyCacheEntity; import com.seafile.seadroid2.framework.data.db.entities.FileTransferEntity; +import com.seafile.seadroid2.framework.data.db.entities.PermissionEntity; import com.seafile.seadroid2.framework.data.db.entities.RepoModel; import com.seafile.seadroid2.framework.data.model.dirents.DirentFileModel; import com.seafile.seadroid2.framework.file_monitor.FileSyncService; @@ -69,7 +69,6 @@ import com.seafile.seadroid2.ui.dialog_fragment.PasswordDialogFragment; import com.seafile.seadroid2.ui.dialog_fragment.listener.OnRefreshDataListener; import com.seafile.seadroid2.ui.repo.RepoQuickFragment; -import com.seafile.seadroid2.ui.search.Search2Activity; import com.seafile.seadroid2.ui.transfer_list.TransferActivity; import java.io.File; @@ -287,7 +286,7 @@ public void accept(EncKeyCacheEntity encKeyCacheEntity) throws Exception { // expired showPasswordDialog(repoModel, finalPath); } else { - getNavContext().navToPath(repoModel, finalPath); + getNavContext().switchToPath(repoModel, finalPath); binding.pager.setCurrentItem(0); getReposFragment().loadData(); refreshToolbarTitle(); @@ -296,7 +295,7 @@ public void accept(EncKeyCacheEntity encKeyCacheEntity) throws Exception { }); } else { - getNavContext().navToPath(repoModel, finalPath); + getNavContext().switchToPath(repoModel, finalPath); binding.pager.setCurrentItem(0); getReposFragment().loadData(); refreshToolbarTitle(); @@ -465,6 +464,19 @@ public void onChanged(NightMode nightMode) { // } } }); + + mainViewModel.getOnActionModeLiveData().observe(this, new Observer() { + @Override + public void onChanged(Boolean aBoolean) { + + onShowRepoActionMode(aBoolean); + + } + }); + } + + private void onShowRepoActionMode(boolean show) { + binding.pager.setUserInputEnabled(!show); } private void refreshToolbarTitle() { @@ -545,7 +557,6 @@ private void finishAndStartAccountsActivity() { finish(); } - //////////////////////////// /// menu //////////////////////////// @@ -596,7 +607,6 @@ private void enableUpButton(boolean isEnable) { @Override public boolean onCreateOptionsMenu(Menu menu) { menuBinding = MenuBinding.inflate(menu, getMenuInflater()); - MenuCompat.setGroupDividerEnabled(menu, true); initSearchView(); @@ -614,9 +624,7 @@ private void initSearchView() { searchView.setOnQueryTextListener(new SearchView.OnQueryTextListener() { @Override public boolean onQueryTextSubmit(String query) { - searchView.clearFocus(); - startSearchPage(query); - return true; + return false; } @Override @@ -625,6 +633,7 @@ public boolean onQueryTextChange(String newText) { return false; } }); + menuBinding.search.collapseActionView(); menuBinding.search.setActionView(searchView); } @@ -634,13 +643,28 @@ public boolean onPrepareOptionsMenu(Menu menu) { if (binding.pager.getCurrentItem() == INDEX_LIBRARY_TAB) { if (getNavContext().inRepo()) { menuBinding.createRepo.setVisible(false); - if (getNavContext().isParentHasWritePermission()) { - menuBinding.add.setEnabled(true); - menuBinding.select.setEnabled(true); - } else { - menuBinding.add.setEnabled(false); - menuBinding.select.setEnabled(false); - } + + //check custom permission + String repoId = getNavContext().getRepoModel().repo_id; + mainViewModel.getRepoModelFromLocal(repoId, new Consumer>() { + @Override + public void accept(Pair pair) throws Exception { + if (pair.getFirst().isCustomPermission()) { + PermissionEntity permission = pair.getSecond(); + if (permission != null) { + menuBinding.add.setEnabled(permission.create); + menuBinding.select.setEnabled(permission.modify); + } + } else if (getNavContext().isParentHasWritePermission()) { + menuBinding.add.setEnabled(true); + menuBinding.select.setEnabled(true); + } else { + menuBinding.add.setEnabled(false); + menuBinding.select.setEnabled(false); + } + } + }); + } else { menuBinding.createRepo.setVisible(true); menuBinding.add.setVisible(false); @@ -662,21 +686,11 @@ public boolean onPrepareOptionsMenu(Menu menu) { @Override public boolean onOptionsItemSelected(MenuItem item) { - boolean isRet = true; if (item.getItemId() == android.R.id.home) { if (getNavContext().inRepo() && binding.pager.getCurrentItem() == INDEX_LIBRARY_TAB) { getOnBackPressedDispatcher().onBackPressed(); } - } else if (item.getItemId() == R.id.menu_action_search_go) { - Optional optional = checkServerInfo(); - if (optional.isPresent() && optional.get().isSearchEnabled()) { - SearchView searchView = (SearchView) menuBinding.search.getActionView(); - if (searchView != null) { - searchView.clearFocus(); - startSearchPage(searchView.getQuery().toString()); - } - } } else if (item.getItemId() == R.id.create_repo) { showNewRepoDialog(); } else if (item.getItemId() == R.id.add) { @@ -734,11 +748,6 @@ public boolean onMenuItemActionExpand(@NonNull MenuItem item) { menuIdState.put("select", menuBinding.select.isVisible()); menuIdState.put("transferList", menuBinding.transferList.isVisible()); - Optional optional = checkServerInfo(); - if (optional.isPresent() && optional.get().isSearchEnabled()) { - menuBinding.searchNext.setVisible(true); - } - menuBinding.sortGroup.setVisible(false); menuBinding.createRepo.setVisible(false); menuBinding.add.setVisible(false); @@ -750,7 +759,6 @@ public boolean onMenuItemActionExpand(@NonNull MenuItem item) { @Override public boolean onMenuItemActionCollapse(@NonNull MenuItem item) { - menuBinding.searchNext.setVisible(false); menuBinding.sortGroup.setVisible(Boolean.TRUE.equals(menuIdState.get("sortGroup"))); menuBinding.createRepo.setVisible(Boolean.TRUE.equals(menuIdState.get("createRepo"))); @@ -810,7 +818,6 @@ private static class MenuBinding { public MenuItem viewGallery; public MenuItem search; - public MenuItem searchNext; public MenuItem createRepo; public MenuItem add; @@ -837,7 +844,6 @@ public static MenuBinding inflate(Menu menu, MenuInflater inflater) { binding1.viewGallery = menu.findItem(R.id.menu_action_view_gallery); binding1.search = menu.findItem(R.id.menu_action_search); - binding1.searchNext = menu.findItem(R.id.menu_action_search_go); binding1.createRepo = menu.findItem(R.id.create_repo); binding1.add = menu.findItem(R.id.add); @@ -847,13 +853,6 @@ public static MenuBinding inflate(Menu menu, MenuInflater inflater) { } } - private void startSearchPage(String query) { - Optional optional = checkServerInfo(); - if (optional.isPresent() && optional.get().isSearchEnabled()) { - Search2Activity.start(this, query); - } - } - /** * @see MainActivity -> android:configChanges="uiMode|orientation|screenSize" */ @@ -914,7 +913,7 @@ private void showPasswordDialog(RepoModel repoModel, String path) { @Override public void onActionStatus(boolean isDone) { if (isDone) { - getNavContext().navToPath(repoModel, path); + getNavContext().switchToPath(repoModel, path); binding.pager.setCurrentItem(0); getReposFragment().loadData(); refreshToolbarTitle(); @@ -1154,26 +1153,6 @@ public void onActivityResult(Boolean o) { } }); - private Dialog dialog; - - private void showProgressDialog() { - if (dialog == null) { - MaterialAlertDialogBuilder builder = new MaterialAlertDialogBuilder(this); - builder.setView(R.layout.layout_dialog_progress_bar); - dialog = builder.create(); - } - - if (dialog.isShowing()) { - dialog.dismiss(); - } - dialog.show(); - } - - private void dismissProgressDialog() { - if (dialog != null) { - dialog.dismiss(); - } - } ///////////////////////////// private void doSelectedMultiFile(List uriList) { diff --git a/app/src/main/java/com/seafile/seadroid2/ui/main/MainViewModel.java b/app/src/main/java/com/seafile/seadroid2/ui/main/MainViewModel.java index 11408d1d5..c2e2849fa 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/main/MainViewModel.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/main/MainViewModel.java @@ -16,6 +16,7 @@ import com.seafile.seadroid2.account.SupportAccountManager; import com.seafile.seadroid2.framework.data.db.entities.EncKeyCacheEntity; import com.seafile.seadroid2.framework.data.db.entities.FileTransferEntity; +import com.seafile.seadroid2.framework.data.db.entities.PermissionEntity; import com.seafile.seadroid2.framework.data.model.dirents.DirentFileModel; import com.seafile.seadroid2.enums.TransferAction; import com.seafile.seadroid2.enums.TransferDataSource; @@ -56,7 +57,10 @@ import io.reactivex.Single; import io.reactivex.SingleEmitter; import io.reactivex.SingleOnSubscribe; +import io.reactivex.SingleSource; import io.reactivex.functions.Consumer; +import io.reactivex.functions.Function; +import kotlin.Pair; public class MainViewModel extends BaseViewModel { // private final MutableLiveData> OnNewFileDownloadLiveData = new MutableLiveData<>(); @@ -65,7 +69,6 @@ public class MainViewModel extends BaseViewModel { //force refresh repo/dirents private final MutableLiveData OnForceRefreshRepoListLiveData = new MutableLiveData<>(); - private final MutableLiveData OnForceRefreshStarredListLiveData = new MutableLiveData<>(); //show swipeRefresh in Repo Fragment private final MutableLiveData OnShowRefreshLoadingInRepoLiveData = new MutableLiveData<>(); @@ -74,6 +77,11 @@ public class MainViewModel extends BaseViewModel { private final MutableLiveData OnNavChangeListenerLiveData = new MutableLiveData<>(); private final MutableLiveData _searchViewExpandedLiveData = new MutableLiveData<>(false); + private final MutableLiveData _onActionModeLiveData = new MutableLiveData<>(false); + + public MutableLiveData getOnActionModeLiveData() { + return _onActionModeLiveData; + } public MutableLiveData getSearchViewExpandedLiveData() { return _searchViewExpandedLiveData; @@ -108,10 +116,6 @@ public MutableLiveData getOnNavContextChangeListenerLiveData() { return OnNavChangeListenerLiveData; } - public MutableLiveData getOnForceRefreshStarredListLiveData() { - return OnForceRefreshStarredListLiveData; - } - public MutableLiveData getServerInfoLiveData() { return _serverInfoLiveData; } @@ -173,7 +177,7 @@ public void accept(List repoModels) throws Exception { if (consumer != null) { if (CollectionUtils.isEmpty(repoModels)) { //no data in sqlite, request RepoApi again - requestRepoModelFromServer(repoId, consumer); + getRepoModelFromRemote(repoId, consumer); } else { consumer.accept(repoModels.get(0)); getOnShowRefreshLoadingInRepoLiveData().setValue(false); @@ -191,7 +195,7 @@ public void accept(Throwable throwable) throws Exception { }); } - private void requestRepoModelFromServer(String repoId, Consumer consumer) { + private void getRepoModelFromRemote(String repoId, Consumer consumer) { //from net Single singleNet = HttpIO.getCurrentInstance().execute(RepoService.class).getRepos(); addSingleDisposable(singleNet, new Consumer() { @@ -226,6 +230,58 @@ public void accept(Throwable throwable) throws Exception { }); } + public void getRepoModelFromLocal(String repoId, Consumer> consumer) { + //from db + Single> dbSingle = AppDatabase.getInstance().repoDao().getRepoById(repoId); + Single> r = dbSingle.flatMap(new Function, SingleSource>>() { + @Override + public SingleSource> apply(List repoModels) throws Exception { + if (CollectionUtils.isEmpty(repoModels)) { + return null; + } + + RepoModel repoModel = repoModels.get(0); + if (TextUtils.isEmpty(repoModel.permission)) { + return Single.just(new Pair<>(repoModel, null)); + } + + if (!repoModel.isCustomPermission()) { + return Single.just(new Pair<>(repoModel, null)); + } + + int pNum = repoModel.getCustomPermissionNum(); + + Single> pSingle = AppDatabase.getInstance().permissionDAO().getByIdAsync(repoId, pNum); + + return pSingle.flatMap(new Function, SingleSource>>() { + @Override + public SingleSource> apply(List permissionEntities) throws Exception { + if (CollectionUtils.isEmpty(permissionEntities)) { + return Single.just(new Pair<>(repoModel, null)); + } + + return Single.just(new Pair<>(repoModel, permissionEntities.get(0))); + } + }); + } + }); + + + addSingleDisposable(r, new Consumer>() { + @Override + public void accept(Pair pair) throws Exception { + if (consumer != null) { + consumer.accept(pair); + } + } + }, new Consumer() { + @Override + public void accept(Throwable throwable) throws Exception { + SLogs.e(throwable); + } + }); + } + public void getEncCacheDB(String repoId, Consumer consumer) { Single> single = AppDatabase.getInstance().encKeyCacheDAO().getListByRepoIdAsync(repoId); addSingleDisposable(single, new Consumer>() { diff --git a/app/src/main/java/com/seafile/seadroid2/ui/media/image_preview/ImagePreviewActivity.java b/app/src/main/java/com/seafile/seadroid2/ui/media/image_preview/ImagePreviewActivity.java index 48394b47b..46b08dad9 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/media/image_preview/ImagePreviewActivity.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/media/image_preview/ImagePreviewActivity.java @@ -1,43 +1,32 @@ package com.seafile.seadroid2.ui.media.image_preview; -import android.app.Dialog; import android.content.Context; import android.content.Intent; import android.os.Bundle; import android.view.View; -import android.view.WindowInsets; -import android.view.WindowInsetsController; import androidx.activity.OnBackPressedCallback; import androidx.annotation.Nullable; -import androidx.core.view.ViewCompat; -import androidx.core.view.WindowCompat; -import androidx.core.view.WindowInsetsCompat; -import androidx.core.view.WindowInsetsControllerCompat; import androidx.fragment.app.Fragment; import androidx.lifecycle.Observer; -import androidx.lifecycle.ViewModelProvider; import androidx.viewpager2.widget.ViewPager2; import com.blankj.utilcode.util.BarUtils; import com.blankj.utilcode.util.CollectionUtils; import com.blankj.utilcode.util.NetworkUtils; import com.blankj.utilcode.util.ToastUtils; -import com.google.android.material.dialog.MaterialAlertDialogBuilder; import com.seafile.seadroid2.R; -import com.seafile.seadroid2.framework.data.db.entities.DirentModel; import com.seafile.seadroid2.databinding.ActivityImagePreviewBinding; +import com.seafile.seadroid2.framework.data.db.entities.DirentModel; import com.seafile.seadroid2.framework.data.db.entities.RepoModel; +import com.seafile.seadroid2.framework.data.db.entities.StarredModel; import com.seafile.seadroid2.framework.data.model.activities.ActivityModel; import com.seafile.seadroid2.framework.data.model.search.SearchModel; -import com.seafile.seadroid2.framework.data.db.entities.StarredModel; -import com.seafile.seadroid2.ui.base.BaseActivity; +import com.seafile.seadroid2.framework.util.Objs; import com.seafile.seadroid2.ui.adapter.ViewPager2Adapter; import com.seafile.seadroid2.ui.base.BaseActivityWithVM; import com.seafile.seadroid2.ui.dialog_fragment.DeleteFileDialogFragment; import com.seafile.seadroid2.ui.dialog_fragment.listener.OnRefreshDataListener; -import com.seafile.seadroid2.framework.util.Objs; -import com.seafile.seadroid2.framework.util.Utils; import java.util.ArrayList; import java.util.List; @@ -261,10 +250,9 @@ private void deleteFile() { int position = binding.pager.getCurrentItem(); - DeleteFileDialogFragment dialogFragment = DeleteFileDialogFragment.newInstance(); - DirentModel direntModel = getSelectedDirent(); - dialogFragment.initData(direntModel); + + DeleteFileDialogFragment dialogFragment = DeleteFileDialogFragment.newInstance(CollectionUtils.newArrayList(direntModel.uid)); dialogFragment.setRefreshListener(new OnRefreshDataListener() { @Override public void onActionStatus(boolean isDone) { @@ -316,23 +304,5 @@ private void downloadFile() { } - private Dialog dialog; - private void showProgressDialog() { - if (dialog == null) { - MaterialAlertDialogBuilder builder = new MaterialAlertDialogBuilder(this); - builder.setView(R.layout.layout_dialog_progress_bar); - dialog = builder.create(); - } - - if (!dialog.isShowing()) { - dialog.show(); - } - } - - private void dismissProgressDialog() { - if (dialog != null) { - dialog.dismiss(); - } - } } diff --git a/app/src/main/java/com/seafile/seadroid2/ui/media/image_preview/ImagePreviewViewModel.java b/app/src/main/java/com/seafile/seadroid2/ui/media/image_preview/ImagePreviewViewModel.java index c49eb4b84..79db9d4b4 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/media/image_preview/ImagePreviewViewModel.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/media/image_preview/ImagePreviewViewModel.java @@ -18,6 +18,7 @@ import com.seafile.seadroid2.ui.repo.RepoService; import com.seafile.seadroid2.framework.http.HttpIO; import com.seafile.seadroid2.ui.base.viewmodel.BaseViewModel; +import com.seafile.seadroid2.ui.star.StarredService; import java.util.HashMap; import java.util.List; @@ -111,7 +112,7 @@ public void star(String repoId, String path) { requestDataMap.put("path", path); Map bodyMap = generateRequestBody(requestDataMap); - Single single = HttpIO.getCurrentInstance().execute(RepoService.class).star(bodyMap); + Single single = HttpIO.getCurrentInstance().execute(StarredService.class).star(bodyMap); addSingleDisposable(single, new Consumer() { @Override public void accept(Dirent2Model resultModel) throws Exception { @@ -134,7 +135,7 @@ public void accept(Throwable throwable) throws Exception { public void unStar(String repoId, String path) { getRefreshLiveData().setValue(true); - Single single = HttpIO.getCurrentInstance().execute(RepoService.class).unStar(repoId, path); + Single single = HttpIO.getCurrentInstance().execute(StarredService.class).unStar(repoId, path); addSingleDisposable(single, new Consumer() { @Override public void accept(ResultModel resultModel) throws Exception { diff --git a/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoQuickAdapter.java b/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoQuickAdapter.java index cbfb0bd52..96922d671 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoQuickAdapter.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoQuickAdapter.java @@ -36,16 +36,14 @@ import com.seafile.seadroid2.framework.data.db.entities.RepoModel; import com.seafile.seadroid2.framework.data.model.BaseModel; import com.seafile.seadroid2.framework.data.model.GroupItemModel; +import com.seafile.seadroid2.framework.data.model.search.SearchModel; import com.seafile.seadroid2.framework.http.HttpIO; import com.seafile.seadroid2.framework.util.GlideApp; -import com.seafile.seadroid2.framework.util.GlideRequests; import com.seafile.seadroid2.framework.util.Utils; import com.seafile.seadroid2.ui.base.adapter.BaseMultiAdapter; import com.seafile.seadroid2.ui.viewholder.GroupItemViewHolder; import com.seafile.seadroid2.widget.AnimatedStateListDrawableCompatUtils; -import org.checkerframework.checker.units.qual.A; - import java.util.ArrayList; import java.util.List; import java.util.Locale; @@ -55,7 +53,7 @@ public class RepoQuickAdapter extends BaseMultiAdapter { private final String SERVER = HttpIO.getCurrentInstance().getServerUrl(); - private boolean actionModeOn; + private boolean onActionMode; private boolean repoEncrypted = false; private FileViewType fileViewType = FileViewType.LIST; @@ -177,6 +175,18 @@ public DirentGalleryViewHolder onCreate(@NonNull Context context, @NonNull ViewG public void onBind(@NonNull DirentGalleryViewHolder viewHolder, int i, @Nullable BaseModel baseModel) { onBindDirentsGallery(viewHolder, (DirentModel) baseModel); } + }).addItemType(AbsLayoutItemType.SEARCH, new OnMultiItem() { + @NonNull + @Override + public DirentViewHolder onCreate(@NonNull Context context, @NonNull ViewGroup viewGroup, int i) { + ItemDirentBinding binding = ItemDirentBinding.inflate(LayoutInflater.from(context), viewGroup, false); + return new DirentViewHolder(binding); + } + + @Override + public void onBind(@NonNull DirentViewHolder viewHolder, int i, @Nullable BaseModel baseModel) { + onBindSearch(viewHolder, (SearchModel) baseModel); + } }).addItemType(AbsLayoutItemType.NOT_SUPPORTED, new OnMultiItem() { @NonNull @Override @@ -203,6 +213,8 @@ public int onItemViewType(int i, @NonNull List list) { } else { return AbsLayoutItemType.DIRENT_GALLERY; } + } else if (list.get(i) instanceof SearchModel) { + return AbsLayoutItemType.SEARCH; } else if (list.get(i) instanceof Account) { return AbsLayoutItemType.ACCOUNT; } @@ -261,14 +273,22 @@ private void onBindRepos(RepoViewHolder holder, RepoModel model) { holder.binding.itemSubtitle.setText(model.getSubtitle()); holder.binding.itemIcon.setImageResource(model.getIcon()); - if (selectType.ordinal() >= RepoSelectType.ONLY_REPO.ordinal()) { - holder.binding.itemSelectView.setVisibility(model.is_checked ? View.VISIBLE : View.INVISIBLE); - holder.binding.expandableToggleButton.setVisibility(View.INVISIBLE); + if (selectType.ordinal() == RepoSelectType.ONLY_REPO.ordinal() || onActionMode) { + holder.binding.getRoot().setChecked(model.is_checked); + + holder.binding.itemMultiSelect.setVisibility(View.VISIBLE); + if (model.is_checked) { + holder.binding.itemMultiSelect.setImageResource(R.drawable.ic_checkbox_checked); + } else { + holder.binding.itemMultiSelect.setImageResource(R.drawable.ic_checkbox_unchecked); + } } else { - holder.binding.expandableToggleButton.setVisibility(View.VISIBLE); - holder.binding.itemSelectView.setVisibility(View.GONE); + holder.binding.itemMultiSelect.setVisibility(View.GONE); + holder.binding.getRoot().setChecked(false); } + holder.binding.expandableToggleButton.setVisibility(View.GONE); + holder.binding.itemTitle.setCompoundDrawablePadding(Constants.DP.DP_4); if (model.starred) { holder.binding.itemTitle.setCompoundDrawables(null, null, getStarDrawable(), null); @@ -290,22 +310,20 @@ private void onBindDirents(DirentViewHolder holder, DirentModel model) { } //action mode - if (actionModeOn) { - holder.binding.itemMultiSelect.setVisibility(View.VISIBLE); - + if (onActionMode) { holder.binding.getRoot().setChecked(model.is_checked); + holder.binding.itemMultiSelect.setVisibility(View.VISIBLE); if (model.is_checked) { - holder.binding.itemMultiSelect.setImageResource(R.drawable.multi_select_item_checked); + holder.binding.itemMultiSelect.setImageResource(R.drawable.ic_checkbox_checked); } else { - holder.binding.itemMultiSelect.setImageResource(R.drawable.multi_select_item_unchecked); + holder.binding.itemMultiSelect.setImageResource(R.drawable.ic_checkbox_unchecked); } } else { holder.binding.itemMultiSelect.setVisibility(View.GONE); - holder.binding.itemMultiSelect.setImageResource(R.drawable.multi_select_item_unchecked); + holder.binding.itemMultiSelect.setImageResource(R.drawable.ic_checkbox_unchecked); holder.binding.getRoot().setChecked(false); - } holder.binding.itemDownloadStatusProgressbar.setVisibility(View.GONE); @@ -314,8 +332,11 @@ private void onBindDirents(DirentViewHolder holder, DirentModel model) { if (selectType.ordinal() < RepoSelectType.ONLY_ACCOUNT.ordinal()) { holder.binding.itemTitle.setTextColor(ContextCompat.getColor(getContext(), R.color.item_title_color)); holder.binding.itemSubtitle.setTextColor(ContextCompat.getColor(getContext(), R.color.item_subtitle_color)); - - holder.binding.expandableToggleButton.setVisibility(View.VISIBLE); +// if (onActionMode) { +// holder.binding.expandableToggleButton.setVisibility(View.GONE); +// } else { +// holder.binding.expandableToggleButton.setVisibility(View.VISIBLE); +// } } else { if (!model.isDir()) { @@ -326,8 +347,9 @@ private void onBindDirents(DirentViewHolder holder, DirentModel model) { holder.binding.itemSubtitle.setTextColor(ContextCompat.getColor(getContext(), R.color.item_subtitle_color)); } - holder.binding.expandableToggleButton.setVisibility(View.INVISIBLE); +// holder.binding.expandableToggleButton.setVisibility(View.INVISIBLE); } + holder.binding.expandableToggleButton.setVisibility(View.GONE); if (model.isDir()) { holder.binding.itemDownloadStatusProgressbar.setVisibility(View.GONE); @@ -380,18 +402,18 @@ private void onBindDirentsGrid(DirentGridViewHolder holder, DirentModel model) { } //action mode - if (actionModeOn) { + if (onActionMode) { holder.binding.itemMultiSelect.setVisibility(View.VISIBLE); holder.binding.getRoot().setChecked(model.is_checked); if (model.is_checked) { - holder.binding.itemMultiSelect.setImageResource(R.drawable.multi_select_item_checked); + holder.binding.itemMultiSelect.setImageResource(R.drawable.ic_checkbox_checked); } else { - holder.binding.itemMultiSelect.setImageResource(R.drawable.multi_select_item_unchecked); + holder.binding.itemMultiSelect.setImageResource(R.drawable.ic_checkbox_unchecked); } } else { holder.binding.itemMultiSelect.setVisibility(View.GONE); - holder.binding.itemMultiSelect.setImageResource(R.drawable.multi_select_item_unchecked); + holder.binding.itemMultiSelect.setImageResource(R.drawable.ic_checkbox_unchecked); holder.binding.getRoot().setChecked(false); } @@ -415,21 +437,45 @@ private void onBindDirentsGallery(DirentGalleryViewHolder holder, DirentModel mo } //action mode - if (actionModeOn) { + if (onActionMode) { holder.binding.itemMultiSelect.setVisibility(View.VISIBLE); holder.binding.getRoot().setChecked(model.is_checked); if (model.is_checked) { - holder.binding.itemMultiSelect.setImageResource(R.drawable.multi_select_item_checked); + holder.binding.itemMultiSelect.setImageResource(R.drawable.ic_checkbox_checked); } else { - holder.binding.itemMultiSelect.setImageResource(R.drawable.multi_select_item_unchecked); + holder.binding.itemMultiSelect.setImageResource(R.drawable.ic_checkbox_unchecked); } } else { holder.binding.getRoot().setChecked(false); holder.binding.itemMultiSelect.setVisibility(View.GONE); - holder.binding.itemMultiSelect.setImageResource(R.drawable.multi_select_item_unchecked); + holder.binding.itemMultiSelect.setImageResource(R.drawable.ic_checkbox_unchecked); + } + } + + private void onBindSearch(DirentViewHolder holder, SearchModel model) { + holder.binding.itemTitle.setText(model.name); + holder.binding.itemSubtitle.setText(model.getSubtitle()); + + holder.binding.getRoot().setBackground(AnimatedStateListDrawableCompatUtils.createDrawableCompat(getContext())); + + if (repoEncrypted || !Utils.isViewableImage(model.name)) { + holder.binding.itemIcon.setImageResource(model.getIcon()); + } else { + DirentModel direntModel = new DirentModel(); + direntModel.full_path = model.fullpath; + direntModel.repo_id = model.repo_id; + loadImage(direntModel, holder.binding.itemIcon); } + + holder.binding.expandableToggleButton.setVisibility(View.GONE); + holder.binding.itemMultiSelect.setVisibility(View.GONE); + holder.binding.itemDownloadStatusProgressbar.setVisibility(View.GONE); + holder.binding.itemDownloadStatus.setVisibility(View.GONE); + + holder.binding.itemTitle.setCompoundDrawables(null, null, null, null); + } private void loadImage(DirentModel direntModel, ImageView imageView) { @@ -459,41 +505,30 @@ private String convertThumbnailUrl(DirentModel direntModel, int size) { return String.format(Locale.ROOT, "%sapi2/repos/%s/thumbnail/?p=%s&size=%d", SERVER, direntModel.repo_id, newFilePath, size); } - public void setActionModeOn(boolean actionModeOn) { - this.actionModeOn = actionModeOn; + public void setOnActionMode(boolean on) { + this.onActionMode = on; - if (!actionModeOn) { + if (!on) { setItemSelected(false); } notifyItemRangeChanged(0, getItemCount()); } - public boolean getActionMode() { - return actionModeOn; + public boolean isOnActionMode() { + return onActionMode; } public void setItemSelected(boolean itemSelected) { for (BaseModel item : getItems()) { - if (item instanceof DirentModel) { - DirentModel model = (DirentModel) item; - model.is_checked = itemSelected; - } + item.is_checked = itemSelected; } notifyItemRangeChanged(0, getItemCount()); } - public List getSelectedList() { - List list = new ArrayList<>(); - for (BaseModel item : getItems()) { - if (item instanceof DirentModel model) { - if (model.is_checked) { - list.add(model); - } - } - } - return list; + public List getSelectedList() { + return getItems().stream().filter(f -> f.is_checked).collect(Collectors.toList()); } /** @@ -560,18 +595,16 @@ private int getSelectedPositionByMode() { return -1; } - public void notifySearchChanged(String searchContent) { + public void filterListBySearchKeyword(String searchContent) { this.searchContent = searchContent; if (CollectionUtils.isEmpty(finalList)) { - submitList(finalList); return; } List filterList; - Predicate predicate = getSearchFilter(); - if (predicate != null) { - filterList = finalList.stream().filter(predicate).collect(Collectors.toList()); + if (!TextUtils.isEmpty(searchContent)) { + filterList = finalList.stream().filter(_searchFilter).collect(Collectors.toList()); } else { filterList = finalList; } @@ -639,6 +672,12 @@ public boolean areItemsTheSame(int oldItemPosition, int newItemPosition) { return TextUtils.equals(newT.full_path, oldT.full_path); } + if (getItems().get(oldItemPosition) instanceof SearchModel) { + SearchModel newT = (SearchModel) getItems().get(oldItemPosition); + SearchModel oldT = (SearchModel) newList.get(newItemPosition); + return TextUtils.equals(newT.fullpath, oldT.fullpath); + } + return true; } @@ -685,6 +724,24 @@ public boolean areContentsTheSame(int oldItemPosition, int newItemPosition) { && TextUtils.equals(newT.server, oldT.server); } + if (getItems().get(oldItemPosition) instanceof SearchModel) { + SearchModel newT = (SearchModel) getItems().get(oldItemPosition); + SearchModel oldT = (SearchModel) newList.get(newItemPosition); + return TextUtils.equals(newT.fullpath, oldT.fullpath) + && TextUtils.equals(newT.name, oldT.name) + && newT.is_dir == oldT.is_dir + && newT.size == oldT.size + && newT.last_modified == oldT.last_modified + && TextUtils.equals(newT.repo_id, oldT.repo_id) + && TextUtils.equals(newT.repo_name, oldT.repo_name) + && TextUtils.equals(newT.repo_owner_email, oldT.repo_owner_email) + && TextUtils.equals(newT.repo_owner_name, oldT.repo_owner_name) + && TextUtils.equals(newT.repo_owner_contact_email, oldT.repo_owner_contact_email) + && TextUtils.equals(newT.thumbnail_url, oldT.thumbnail_url) + && TextUtils.equals(newT.repo_type, oldT.repo_type) + && TextUtils.equals(newT.content_highlight, oldT.content_highlight); + } + if (getItems().get(oldItemPosition) instanceof DirentModel) { DirentModel newT = (DirentModel) getItems().get(oldItemPosition); DirentModel oldT = (DirentModel) newList.get(newItemPosition); @@ -725,13 +782,6 @@ public boolean areContentsTheSame(int oldItemPosition, int newItemPosition) { diffResult.dispatchUpdatesTo(this); } - public Predicate getSearchFilter() { - if (TextUtils.isEmpty(searchContent)) { - return null; - } - return _searchFilter; - } - private String searchContent; private final Predicate _searchFilter = new Predicate<>() { @Override diff --git a/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoQuickFragment.java b/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoQuickFragment.java index f8caa48f6..ea2051def 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoQuickFragment.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoQuickFragment.java @@ -6,12 +6,14 @@ import android.content.Intent; import android.os.Bundle; import android.text.TextUtils; +import android.view.Gravity; import android.view.LayoutInflater; import android.view.Menu; import android.view.MenuInflater; import android.view.MenuItem; import android.view.View; import android.view.ViewGroup; +import android.widget.FrameLayout; import android.widget.TextView; import androidx.activity.result.ActivityResult; @@ -27,12 +29,14 @@ import androidx.lifecycle.ViewModelProvider; import androidx.media3.common.util.UnstableApi; import androidx.recyclerview.widget.GridLayoutManager; +import androidx.recyclerview.widget.RecyclerView; import androidx.work.Data; import androidx.work.WorkInfo; import com.blankj.utilcode.util.CollectionUtils; import com.blankj.utilcode.util.TimeUtils; import com.blankj.utilcode.util.ToastUtils; +import com.chad.library.adapter4.BaseQuickAdapter; import com.github.panpf.recycler.sticky.StickyItemDecoration; import com.google.android.material.dialog.MaterialAlertDialogBuilder; import com.google.common.collect.Maps; @@ -40,8 +44,9 @@ import com.seafile.seadroid2.SeafException; import com.seafile.seadroid2.account.Account; import com.seafile.seadroid2.account.SupportAccountManager; -import com.seafile.seadroid2.bottomsheetmenu.BottomSheetHelper; -import com.seafile.seadroid2.bottomsheetmenu.BottomSheetMenuFragment; +import com.seafile.seadroid2.framework.data.db.entities.PermissionEntity; +import com.seafile.seadroid2.framework.datastore.sp.SettingsManager; +import com.seafile.seadroid2.ui.bottomsheetmenu.BottomSheetMenuAdapter; import com.seafile.seadroid2.config.AbsLayoutItemType; import com.seafile.seadroid2.config.Constants; import com.seafile.seadroid2.context.CopyMoveContext; @@ -56,6 +61,7 @@ import com.seafile.seadroid2.framework.data.db.entities.RepoModel; import com.seafile.seadroid2.framework.data.model.BaseModel; import com.seafile.seadroid2.framework.data.model.GroupItemModel; +import com.seafile.seadroid2.framework.data.model.search.SearchModel; import com.seafile.seadroid2.framework.datastore.DataManager; import com.seafile.seadroid2.framework.util.Objs; import com.seafile.seadroid2.framework.util.SLogs; @@ -78,11 +84,13 @@ import com.seafile.seadroid2.ui.dialog_fragment.listener.OnRefreshDataListener; import com.seafile.seadroid2.ui.dialog_fragment.listener.OnResultListener; import com.seafile.seadroid2.ui.file.FileActivity; +import com.seafile.seadroid2.ui.main.MainActivity; import com.seafile.seadroid2.ui.main.MainViewModel; import com.seafile.seadroid2.ui.markdown.MarkdownActivity; import com.seafile.seadroid2.ui.media.image_preview.ImagePreviewActivity; import com.seafile.seadroid2.ui.media.player.exoplayer.CustomExoVideoPlayerActivity; import com.seafile.seadroid2.ui.sdoc.SDocWebViewActivity; +import com.seafile.seadroid2.ui.search.SearchViewModel; import com.seafile.seadroid2.ui.selector.ObjSelectorActivity; import com.seafile.seadroid2.view.TipsViews; @@ -90,8 +98,10 @@ import java.util.HashMap; import java.util.List; import java.util.Map; +import java.util.stream.Collectors; import io.reactivex.functions.Consumer; +import kotlin.Pair; public class RepoQuickFragment extends BaseFragmentWithVM { private static final String KEY_REPO_SCROLL_POSITION = "repo_scroll_position"; @@ -100,11 +110,14 @@ public class RepoQuickFragment extends BaseFragmentWithVM { private final int forceRefreshInterval = 1000 * 60 * 5;//5m private final HashMap mRefreshStatusExpireTimeMap = new HashMap<>(); + private final HashMap mPermissionRefreshStatusExpireTimeMap = new HashMap<>(); private LayoutFastRvBinding binding; private RepoQuickAdapter adapter; private MainViewModel mainViewModel; + private SearchViewModel searchViewModel; + private final Map scrollPositions = Maps.newHashMap(); private AppCompatActivity activity; private ActionMode actionMode; @@ -131,6 +144,7 @@ public void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); mainViewModel = new ViewModelProvider(requireActivity()).get(MainViewModel.class); + searchViewModel = new ViewModelProvider(this).get(SearchViewModel.class); } @Override @@ -226,18 +240,25 @@ private void initAdapter() { adapter.setFileViewType(lastViewType); adapter.setOnItemClickListener((baseQuickAdapter, view, i) -> { - if (adapter.getActionMode()) { + if (adapter.isOnActionMode()) { //toggle toggleAdapterItemSelectedOnLongClick(i); //update bar title updateContextualActionBar(); + + BaseModel baseModel = adapter.getItems().get(i); + if (baseModel instanceof RepoModel m) { + getViewModel().inflateRepoMenuWithPermission(requireContext(), m, getDisableMenuIds(), getWillBeRemovedMenuIds(), isPermissionForce(m.repo_id)); + } else if (baseModel instanceof DirentModel m) { + getViewModel().inflateDirentMenuWithPermission(requireContext(), m, getDisableMenuIds(), getWillBeRemovedMenuIds(), isPermissionForce(m.repo_id)); + } + return; } BaseModel baseModel = adapter.getItems().get(i); - if (baseModel instanceof RepoModel) { - RepoModel repoModel = (RepoModel) baseModel; + if (baseModel instanceof RepoModel repoModel) { if (repoModel.encrypted) { decrypt(repoModel); } else { @@ -245,66 +266,40 @@ private void initAdapter() { } } else if (baseModel instanceof DirentModel) { navTo(baseModel); + } else if (baseModel instanceof SearchModel) { + navTo(baseModel); } }); adapter.setOnItemLongClickListener((baseQuickAdapter, view, i) -> { - if (getNavContext().inRepo()) { - - //return - if (adapter.getActionMode()) { - return true; - } - - //start action mode - startContextualActionMode(); - - //toggle - toggleAdapterItemSelectedOnLongClick(i); - - //It's actually updating the title of the ActionBar - updateContextualActionBar(); - + //return + if (adapter.isOnActionMode()) { return true; - } else { - return false; } - }); -// adapter.addOnItemChildClickListener(R.id.group_container, new BaseQuickAdapter.OnItemChildClickListener() { -// @Override -// public void onItemClick(@NonNull BaseQuickAdapter baseQuickAdapter, @NonNull View view, int i) { -// GroupItemModel groupItemModel = (GroupItemModel) adapter.getItems().get(i); -// groupItemModel.is_checked = !groupItemModel.is_checked; -// Bundle bundle = new Bundle(); -// bundle.putBoolean("is_checked", groupItemModel.is_checked); -// adapter.notifyItemChanged(i, bundle); -// } -// }); + //start action mode + startContextualActionMode(); - adapter.addOnItemChildClickListener(R.id.expandable_toggle_button, (baseQuickAdapter, view, i) -> { + //toggle + toggleAdapterItemSelectedOnLongClick(i); - //when ActionMode is On, return - if (adapter.getActionMode()) { - return; - } + //It's actually updating the title of the ActionBar + updateContextualActionBar(); + + BaseModel baseModel = adapter.getItems().get(i); + if (baseModel instanceof RepoModel m) { + getViewModel().inflateRepoMenuWithPermission(requireContext(), m, getDisableMenuIds(), getWillBeRemovedMenuIds(), isPermissionForce(m.repo_id)); + } else if (baseModel instanceof DirentModel m) { + getViewModel().inflateDirentMenuWithPermission(requireContext(), m, getDisableMenuIds(), getWillBeRemovedMenuIds(), isPermissionForce(m.repo_id)); - //open bottom sheet dialog - if (getNavContext().inRepo()) { - DirentModel direntModel = (DirentModel) adapter.getItems().get(i); - if (direntModel.isDir()) { - showDirNewBottomSheet(direntModel); - } else { - showFileBottomSheet(direntModel); - } - } else { - showRepoBottomSheet((RepoModel) adapter.getItems().get(i)); } + return true; }); binding.rv.setAdapter(adapter); } + private void initViewModel() { getViewModel().getRefreshLiveData().observe(getViewLifecycleOwner(), new Observer() { @Override @@ -313,17 +308,30 @@ public void onChanged(Boolean aBoolean) { } }); + getViewModel().getShowLoadingDialogLiveData().observe(getViewLifecycleOwner(), new Observer() { + @Override + public void onChanged(Boolean aBoolean) { + + MainActivity mainActivity = (MainActivity) getActivity(); + if (mainActivity != null) { + mainActivity.showProgressDialog(aBoolean); + } + } + }); + getViewModel().getSeafExceptionLiveData().observe(getViewLifecycleOwner(), this::showErrorView); - getViewModel().getStarLiveData().observe(getViewLifecycleOwner(), aBoolean -> { + getViewModel().getStarredLiveData().observe(getViewLifecycleOwner(), aBoolean -> { if (aBoolean) { loadData(true); } - mainViewModel.getOnForceRefreshStarredListLiveData().setValue(true); + closeActionMode(); + + SettingsManager.setForceRefreshStarredListState(); }); - getViewModel().getObjsListLiveData().observe(getViewLifecycleOwner(), repoModels -> { + getViewModel().getObjListLiveData().observe(getViewLifecycleOwner(), repoModels -> { //notify navContext changed mainViewModel.getOnNavContextChangeListenerLiveData().setValue(true); @@ -333,8 +341,15 @@ public void onChanged(Boolean aBoolean) { restoreScrollPosition(); }); + getViewModel().getMenuItemListLiveData().observe(getViewLifecycleOwner(), new Observer>() { + @Override + public void onChanged(List menuItems) { + showBottomSheetWindow(menuItems); + } + }); + + mainViewModel.getOnResortListLiveData().observe(getViewLifecycleOwner(), a -> { - SLogs.d("resort: " + a); loadData(); }); @@ -345,7 +360,20 @@ public void onChanged(Boolean aBoolean) { mainViewModel.getOnSearchLiveData().observe(getViewLifecycleOwner(), new Observer() { @Override public void onChanged(String s) { - adapter.notifySearchChanged(s); + search(s); + } + }); + + mainViewModel.getOnActionModeLiveData().observe(getViewLifecycleOwner(), new Observer() { + @Override + public void onChanged(Boolean aBoolean) { + onShowActionMode(aBoolean); + } + }); + searchViewModel.getSearchListLiveData().observe(getViewLifecycleOwner(), new Observer>() { + @Override + public void onChanged(List searchModels) { + deduplicateSearchData(searchModels); } }); @@ -378,6 +406,204 @@ public void onChanged(Boolean aBoolean) { }); } + + @Override + public void onPause() { + super.onPause(); + + //close search view + } + + private View floatingView; + private BottomSheetMenuAdapter bottomSheetMenuAdapter; + + private List getDisableMenuIds() { + List selectedList = adapter.getSelectedList(); + if (selectedList == null || selectedList.isEmpty()) { + return null; + } + + if (selectedList.size() == 1) { + return null; + } + + return CollectionUtils.newArrayList(R.id.share, R.id.export, R.id.open, R.id.rename, R.id.upload); + } + + private List getWillBeRemovedMenuIds() { + List selectedList = adapter.getSelectedList(); + + if (CollectionUtils.isEmpty(selectedList)) { + return CollectionUtils.newArrayList(R.id.unstar); + } + + if (selectedList.size() == 1) { + + BaseModel baseModel = selectedList.get(0); + if (baseModel instanceof RepoModel m) { + return CollectionUtils.newArrayList(m.starred ? R.id.star : R.id.unstar); + } else if (baseModel instanceof DirentModel m) { + return CollectionUtils.newArrayList(m.starred ? R.id.star : R.id.unstar); + } + + //remove all starred menu + return CollectionUtils.newArrayList(R.id.star, R.id.unstar); + } + + boolean isAllStarred = true; + for (BaseModel baseModel : selectedList) { + if (baseModel instanceof RepoModel m) { + if (m.starred) { + continue; + } + isAllStarred = false; + break; + } else if (baseModel instanceof DirentModel m) { + if (m.starred) { + continue; + } + isAllStarred = false; + break; + } + } + + if (isAllStarred) { + return CollectionUtils.newArrayList(R.id.star); + } else { + return CollectionUtils.newArrayList(R.id.unstar); + } + } + + private void showBottomSheetWindow(List localMenuItems) { + if (CollectionUtils.isEmpty(localMenuItems)) { + removeFloatingView(); + return; + } + + if (floatingView != null && floatingView.isAttachedToWindow()) { + bottomSheetMenuAdapter.submitList(localMenuItems); + return; + } + + View decorView = requireActivity().getWindow().getDecorView(); + FrameLayout content = decorView.findViewById(android.R.id.content); + + floatingView = getLayoutInflater().inflate(R.layout.layout_bottom_sheet_menu_view, null, false); + + int columnCount = 5; + RecyclerView rv = floatingView.findViewById(R.id.rv); + rv.setLayoutManager(new GridLayoutManager(requireContext(), columnCount)); + + bottomSheetMenuAdapter = new BottomSheetMenuAdapter(columnCount); + bottomSheetMenuAdapter.setOnItemClickListener(new BaseQuickAdapter.OnItemClickListener() { + @Override + public void onClick(@NonNull BaseQuickAdapter baseQuickAdapter, @NonNull View view, int i) { + MenuItem item = bottomSheetMenuAdapter.getItem(i); + if (item == null) { + return; + } + if (!item.isEnabled()) { + return; + } + + onBottomSheetItemClick(item); + } + }); + + bottomSheetMenuAdapter.submitList(localMenuItems); + rv.setAdapter(bottomSheetMenuAdapter); + + FrameLayout.LayoutParams p = new FrameLayout.LayoutParams(-1, -2); + p.gravity = Gravity.BOTTOM; + content.addView(floatingView, p); + } + + private void onBottomSheetItemClick(MenuItem item) { + if (item == null) { + return; + } + + List selectedList = adapter.getSelectedList(); + + if (item.getItemId() == R.id.star) { + getViewModel().multiStarOrNot(selectedList, true); + } else if (item.getItemId() == R.id.unstar) { + getViewModel().multiStarOrNot(selectedList, false); + } else if (item.getItemId() == R.id.rename) { + rename(selectedList); + } else if (item.getItemId() == R.id.move) { + RepoModel repoModel = getNavContext().getRepoModel(); + String parent_dir = getNavContext().getNavPath(); + + move(repoModel.repo_id, repoModel.repo_name, parent_dir, selectedList); + } else if (item.getItemId() == R.id.copy) { + RepoModel repoModel = getNavContext().getRepoModel(); + String parent_dir = getNavContext().getNavPath(); + + copy(repoModel.repo_id, repoModel.repo_name, parent_dir, selectedList); + } else if (item.getItemId() == R.id.delete) { + if (!getNavContext().inRepo()) { + deleteRepo(selectedList); + } else { + deleteDirents(selectedList); + } + } else if (item.getItemId() == R.id.upload) { + addUploadTask(selectedList, true); + } else if (item.getItemId() == R.id.download) { + download(selectedList); + } else if (item.getItemId() == R.id.share) { + showShareDialog(selectedList); + } else if (item.getItemId() == R.id.export) { + exportFile(selectedList); + } else if (item.getItemId() == R.id.open) { + openWith(selectedList); + } + } + + private void onShowActionMode(boolean isShow) { + int p = Constants.DP.DP_32 * 4; + if (isShow) { + + } else { + p = 0; + } + + binding.rv.setPadding(0, 0, 0, p); + + if (isShow) { + if (!adapter.isOnActionMode()) { + adapter.setOnActionMode(true); + } + } else { + + removeFloatingView(); + closeActionMode(); + + } + } + + private void removeFloatingView() { + if (floatingView == null) { + return; + } + + if (!floatingView.isAttachedToWindow()) { + floatingView = null; + return; + } + + //clear permission list cache + getViewModel().clearCachePermissionMap(); + + + View decorView = requireActivity().getWindow().getDecorView(); + FrameLayout content = decorView.findViewById(android.R.id.content); + content.removeView(floatingView); + + floatingView = null; + bottomSheetMenuAdapter = null; + } + public void reloadData() { loadData(isForce()); } @@ -501,6 +727,18 @@ private boolean isForce() { return expire == null || now > expire; } + private boolean isPermissionForce(String repoId) { + long now = TimeUtils.getNowMills(); + Long expire = mPermissionRefreshStatusExpireTimeMap.get(repoId); + + boolean ss = expire == null || now > expire; + if (ss) { + mPermissionRefreshStatusExpireTimeMap.put(repoId, (now + forceRefreshInterval)); + } + return ss; + } + + public void loadData() { loadData(false); } @@ -517,6 +755,9 @@ public void loadData(boolean forceRefresh) { String k = getNavContext().getRepoModel().repo_id + getNavContext().getNavPath(); mRefreshStatusExpireTimeMap.put(k, now); } + + //force refresh permission list + mPermissionRefreshStatusExpireTimeMap.clear(); } getViewModel().loadData(getNavContext(), forceRefresh); @@ -530,6 +771,18 @@ private void notifyDataChanged(List repoModels) { } } + private void search(String keyword) { + adapter.filterListBySearchKeyword(keyword); + + // + searchViewModel.searchNext(keyword, 1, 20); + } + + private void deduplicateSearchData(List searchModels) { + //todo deduplicate + adapter.addAll(searchModels); + } + private void showEmptyTip() { FileViewType type = Settings.FILE_LIST_VIEW_TYPE.queryValue(); if (FileViewType.GALLERY == type) { @@ -566,8 +819,9 @@ private void navTo(BaseModel model) { //save saveScrollPosition(); - if (!getNavContext().inRepo()) { - getNavContext().push(model); + if (model instanceof RepoModel model1) { + getNavContext().push(model1); + loadData(isForce()); } else if (model instanceof DirentModel direntModel) { if (direntModel.isDir()) { @@ -576,13 +830,35 @@ private void navTo(BaseModel model) { } else { open(direntModel); } + } else if (model instanceof SearchModel searchModel) { + + DirentModel direntModel = SearchModel.converterThis2DirentModel(searchModel); + if (direntModel.isDir()) { + String repoId = getNavContext().getRepoModel().repo_id; + + //switch to special path in special repo + if (TextUtils.equals(repoId, direntModel.repo_id)) { + getNavContext().switchToPath(getNavContext().getRepoModel(), searchModel.fullpath); + loadData(isForce()); + } else { + getViewModel().getRepoModelAndPermissionEntity(searchModel.repo_id, isPermissionForce(searchModel.repo_id), new Consumer>() { + @Override + public void accept(Pair pair) throws Exception { + getNavContext().switchToPath(pair.getFirst(), searchModel.fullpath); + loadData(isForce()); + } + }); + } + } else { + open(direntModel); + } } } public boolean backTo() { if (getNavContext().inRepo()) { - if (adapter.getActionMode()) { - adapter.setActionModeOn(false); + if (adapter.isOnActionMode()) { + adapter.setOnActionMode(false); } else { // removeCurrentScrollPosition(); @@ -695,35 +971,39 @@ private void restoreScrollPosition() { public void onDetach() { super.onDetach(); - if (actionMode != null) { - actionMode.finish(); - } + removeFloatingView(); + closeActionMode(); + } + + @Override + public void onDestroy() { + super.onDestroy(); } private void toggleAdapterItemSelectedOnLongClick(int i) { - DirentModel direntModel = (DirentModel) adapter.getItems().get(i); - direntModel.is_checked = !direntModel.is_checked; - adapter.getItems().set(i, direntModel); - adapter.notifyItemChanged(i); + BaseModel baseModel = adapter.getItems().get(i); + if (baseModel instanceof RepoModel repoModel) { + repoModel.is_checked = !repoModel.is_checked; + adapter.set(i, repoModel); + } else if (baseModel instanceof DirentModel direntModel) { + direntModel.is_checked = !direntModel.is_checked; + adapter.set(i, direntModel); + } } public void closeActionMode() { - if (adapter.getActionMode()) { - adapter.setActionModeOn(false); + if (adapter.isOnActionMode()) { + adapter.setOnActionMode(false); + } + if (actionMode != null) { actionMode.finish(); actionMode = null; } } public void startContextualActionMode() { - if (!getNavContext().inRepo()) return; - //action mode on - if (!adapter.getActionMode()) { - adapter.setActionModeOn(true); - } - if (actionMode == null) { // start the actionMode actionMode = activity.startSupportActionMode(new ActionModeCallback()); @@ -734,8 +1014,6 @@ public void startContextualActionMode() { * update state of contextual action bar (CAB) */ public void updateContextualActionBar() { - if (!getNavContext().inRepo()) return; - if (actionMode == null) { // there are some selected items, start the actionMode actionMode = activity.startSupportActionMode(new ActionModeCallback()); @@ -760,11 +1038,13 @@ public boolean onCreateActionMode(ActionMode mode, Menu menu) { inflater.inflate(R.menu.repos_fragment_menu, menu); if (adapter == null) return true; - // to hidden "r" permissions files or folder - if (!getNavContext().isParentHasWritePermission()) { - menu.findItem(R.id.action_mode_delete).setVisible(false); - menu.findItem(R.id.action_mode_move).setVisible(false); - } + mainViewModel.getOnActionModeLiveData().setValue(true); + +// // to hidden "r" permissions files or folder +// if (!getNavContext().isParentHasWritePermission()) { +// menu.findItem(R.id.action_mode_delete).setVisible(false); +// menu.findItem(R.id.action_mode_move).setVisible(false); +// } return true; } @@ -775,8 +1055,8 @@ public boolean onPrepareActionMode(ActionMode mode, Menu menu) { * The ActionBarPolicy determines how many action button to place in the ActionBar * and the default amount is 2. */ - menu.findItem(R.id.action_mode_delete).setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS); - menu.findItem(R.id.action_mode_copy).setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS); +// menu.findItem(R.id.action_mode_delete).setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS); +// menu.findItem(R.id.action_mode_copy).setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS); menu.findItem(R.id.action_mode_select_all).setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS); // Here you can perform updates to the contextual action bar (CAB) due to @@ -787,7 +1067,7 @@ public boolean onPrepareActionMode(ActionMode mode, Menu menu) { @Override public boolean onActionItemClicked(ActionMode mode, MenuItem item) { //check data - List selectedDirents = adapter.getSelectedList(); + List selectedDirents = adapter.getSelectedList(); if (CollectionUtils.isEmpty(selectedDirents) || !getNavContext().inRepo()) { if (item.getItemId() != R.id.action_mode_select_all) { ToastUtils.showLong(R.string.action_mode_no_items_selected); @@ -801,19 +1081,20 @@ public boolean onActionItemClicked(ActionMode mode, MenuItem item) { updateContextualActionBar(); allItemsSelected = !allItemsSelected; - } else if (itemId == R.id.action_mode_delete) { - deleteDirents(selectedDirents); - } else if (itemId == R.id.action_mode_copy) { - DirentModel dirent = selectedDirents.get(0); - copyFiles(dirent.repo_id, dirent.repo_name, dirent.parent_dir, selectedDirents); - } else if (itemId == R.id.action_mode_move) { - DirentModel dirent = selectedDirents.get(0); - moveFiles(dirent.repo_id, dirent.repo_name, dirent.parent_dir, selectedDirents); - } else if (itemId == R.id.action_mode_download) { - downloadDirents(selectedDirents); - } else { - return false; } +// else if (itemId == R.id.action_mode_delete) { +// deleteDirents(selectedDirents); +// } else if (itemId == R.id.action_mode_copy) { +// DirentModel dirent = selectedDirents.get(0); +// copyFiles(dirent.repo_id, dirent.repo_name, dirent.parent_dir, selectedDirents); +// } else if (itemId == R.id.action_mode_move) { +// DirentModel dirent = selectedDirents.get(0); +// moveFiles(dirent.repo_id, dirent.repo_name, dirent.parent_dir, selectedDirents); +// } else if (itemId == R.id.action_mode_download) { +// downloadDirents(selectedDirents); +// } else { +// return false; +// } return true; } @@ -822,157 +1103,11 @@ public boolean onActionItemClicked(ActionMode mode, MenuItem item) { public void onDestroyActionMode(ActionMode mode) { if (adapter == null) return; - adapter.setActionModeOn(false); - - // Here you can make any necessary updates to the activity when - // the contextual action bar (CAB) is removed. By default, selected items are deselected/unchecked. - if (actionMode != null) { - actionMode = null; - } + mainViewModel.getOnActionModeLiveData().setValue(false); } } - public void showRepoBottomSheet(RepoModel model) { - - int rid = R.menu.bottom_sheet_op_repo; - BottomSheetMenuFragment.Builder builder = BottomSheetHelper.buildSheet(getActivity(), rid, menuItem -> { - if (menuItem.getItemId() == R.id.star || menuItem.getItemId() == R.id.unstar) { - starOrNot(model); - } else if (menuItem.getItemId() == R.id.rename_repo) { - rename(model.repo_id, null, model.repo_name, "repo"); - } else if (menuItem.getItemId() == R.id.delete_repo) { - deleteRepo(model.repo_id); - } - }); - - if (model.starred) { - builder.removeMenu(R.id.star); - } else { - builder.removeMenu(R.id.unstar); - } - - if (!model.hasWritePermission()) { - builder.removeMenu(R.id.delete_repo); - builder.removeMenu(R.id.rename_repo); - } - - builder.show(getChildFragmentManager()); - } - - public void showDirNewBottomSheet(DirentModel dirent) { - int rid = R.menu.bottom_sheet_op_dir; - BottomSheetMenuFragment.Builder builder = BottomSheetHelper.buildSheet(getActivity(), rid, menuItem -> { - int itemId = menuItem.getItemId(); - - if (itemId == R.id.star || itemId == R.id.unstar) { - starOrNot(dirent); - } else if (itemId == R.id.share) { - showShareDialog(dirent); - } else if (itemId == R.id.delete) { - deleteDirent(dirent); - } else if (itemId == R.id.copy) { - copyFiles(dirent.repo_id, dirent.repo_name, dirent.parent_dir, CollectionUtils.newArrayList(dirent)); - } else if (itemId == R.id.move) { - moveFiles(dirent.repo_id, dirent.repo_name, dirent.parent_dir, CollectionUtils.newArrayList(dirent)); - } else if (itemId == R.id.rename) { - rename(dirent.repo_id, dirent.full_path, dirent.name, "dir"); - } else if (itemId == R.id.download) { - download(dirent); - } - }); - - if (dirent.starred) { - builder.removeMenu(R.id.star); - } else { - builder.removeMenu(R.id.unstar); - } - - if (!dirent.hasWritePermission()) { - builder.removeMenu(R.id.rename); - builder.removeMenu(R.id.delete); - builder.removeMenu(R.id.move); - } - - if (!dirent.hasDownloadPermission()) { - builder.removeMenu(R.id.download); - } - - String repoId = getNavContext().getRepoModel().repo_id; - getViewModel().getRepoModelFromLocal(repoId, new Consumer() { - @Override - public void accept(RepoModel repoModel) throws Exception { - if (repoModel != null && repoModel.encrypted) { - builder.removeMenu(R.id.share); - } - builder.show(getChildFragmentManager()); - } - }); - } - - private void showFileBottomSheet(DirentModel dirent) { - int rid = R.menu.bottom_sheet_op_file; - BottomSheetMenuFragment.Builder builder = BottomSheetHelper.buildSheet(getActivity(), rid, menuItem -> { - int itemId = menuItem.getItemId(); - - if (itemId == R.id.share) { - showShareDialog(dirent); - } else if (itemId == R.id.open) { - openWith(dirent); - } else if (itemId == R.id.delete) { - deleteDirent(dirent); - } else if (itemId == R.id.copy) { - copyFiles(dirent.repo_id, dirent.repo_name, dirent.parent_dir, CollectionUtils.newArrayList(dirent)); - } else if (itemId == R.id.move) { - moveFiles(dirent.repo_id, dirent.repo_name, dirent.parent_dir, CollectionUtils.newArrayList(dirent)); - } else if (itemId == R.id.rename) { - rename(dirent.repo_id, dirent.full_path, dirent.name, "file"); - } else if (itemId == R.id.update) { - addUploadTask(dirent, true); - } else if (itemId == R.id.download) { - download(dirent); - } else if (itemId == R.id.export) { - exportFile(dirent); - } else if (itemId == R.id.star || itemId == R.id.unstar) { - starOrNot(dirent); - } - }); - - if (dirent.starred) { - builder.removeMenu(R.id.star); - } else { - builder.removeMenu(R.id.unstar); - } - - - if (!dirent.hasWritePermission()) { - builder.removeMenu(R.id.rename); - builder.removeMenu(R.id.delete); - builder.removeMenu(R.id.move); - builder.removeMenu(R.id.update); - } - - if (!dirent.hasDownloadPermission()) { - builder.removeMenu(R.id.download); - } - -// if (!Utils.isTextMimeType(dirent.name)) { -// builder.removeMenu(R.id.open); -// } - - if (getNavContext().getRepoModel().encrypted) { - builder.removeMenu(R.id.share); - } - - Account account = SupportAccountManager.getInstance().getCurrentAccount(); - - File f = DataManager.getLocalRepoFile(account, getNavContext().getRepoModel().repo_id, getNavContext().getRepoModel().repo_name, dirent.full_path); - if (!f.exists()) { - builder.removeMenu(R.id.update); - } - builder.show(getChildFragmentManager()); - } - /************ Files ************/ @@ -1026,7 +1161,7 @@ private void open(DirentModel dirent) { } //Open with another app - openWith(dirent); + openWith(CollectionUtils.newArrayList(dirent)); } private File getLocalDestinationFile(String repoId, String repoName, String fullPathInRepo) { @@ -1035,7 +1170,15 @@ private File getLocalDestinationFile(String repoId, String repoName, String full return DataManager.getLocalRepoFile(account, repoId, repoName, fullPathInRepo); } - private void openWith(DirentModel direntModel) { + private void openWith(List direntModels) { + if (CollectionUtils.isEmpty(direntModels)) { + return; + } + + closeActionMode(); + + DirentModel direntModel = (DirentModel) direntModels.get(0); + File local = getLocalDestinationFile(direntModel.repo_id, direntModel.repo_name, direntModel.full_path); if (local.exists() && local.length() == direntModel.size) { WidgetUtils.openWith(requireActivity(), local); @@ -1045,53 +1188,84 @@ private void openWith(DirentModel direntModel) { } } - public void download(DirentModel direntModel) { - BackgroundJobManagerImpl.getInstance().startDownloadChainWorker(new String[]{direntModel.uid}); + public void download(List direntModels) { + if (CollectionUtils.isEmpty(direntModels)) { + return; + } + + List direntModels1 = direntModels.stream().map(m -> (DirentModel) m).collect(Collectors.toList()); + List uids = direntModels1.stream().map(m -> m.uid).collect(Collectors.toList()); + String[] sArray = uids.toArray(new String[0]); + BackgroundJobManagerImpl.getInstance().startDownloadChainWorker(sArray); + + closeActionMode(); } - public void rename(String repoID, String curPath, String curName, String type) { - RenameDialogFragment dialogFragment = RenameDialogFragment.newInstance(); - dialogFragment.initData(curName, curPath, repoID, type); + public void rename(List models) { + if (CollectionUtils.isEmpty(models)) { + return; + } + + RenameDialogFragment dialogFragment; + + BaseModel first = models.get(0); + if (first instanceof DirentModel dirent) { + dialogFragment = RenameDialogFragment.newInstance(dirent.name, dirent.full_path, dirent.repo_id, dirent.type); + } else if (first instanceof RepoModel repo) { + dialogFragment = RenameDialogFragment.newInstance(repo.repo_name, "/", repo.repo_id, "repo"); + } else { + return; + } + dialogFragment.setRefreshListener(new OnRefreshDataListener() { @Override public void onActionStatus(boolean isDone) { if (isDone) { - mainViewModel.getOnForceRefreshRepoListLiveData().setValue(true); + ToastUtils.showLong(R.string.rename_successful); } + + loadData(true); + closeActionMode(); } }); dialogFragment.show(getChildFragmentManager(), RenameDialogFragment.class.getSimpleName()); } - public void deleteRepo(String repoID) { - DeleteRepoDialogFragment dialogFragment = DeleteRepoDialogFragment.newInstance(repoID); + public void deleteRepo(List repoModels) { + if (CollectionUtils.isEmpty(repoModels)) { + return; + } + + List repoIds = repoModels.stream().map(m -> (RepoModel) m).map(m -> m.repo_id).collect(Collectors.toList()); + DeleteRepoDialogFragment dialogFragment = DeleteRepoDialogFragment.newInstance(repoIds); dialogFragment.setRefreshListener(isDone -> { if (isDone) { ToastUtils.showLong(R.string.delete_successful); - - loadData(true); } + + closeActionMode(); + loadData(true); }); dialogFragment.show(getChildFragmentManager(), DeleteRepoDialogFragment.class.getSimpleName()); } - public void deleteDirent(DirentModel dirent) { - deleteDirents(CollectionUtils.newArrayList(dirent)); - } + public void deleteDirents(List dirents) { + if (CollectionUtils.isEmpty(dirents)) { + return; + } - public void deleteDirents(List dirents) { - DeleteFileDialogFragment dialogFragment = DeleteFileDialogFragment.newInstance(); - dialogFragment.initData(dirents); + List direntUids = dirents.stream().map(m -> (DirentModel) m).map(m -> m.uid).collect(Collectors.toList()); + DeleteFileDialogFragment dialogFragment = DeleteFileDialogFragment.newInstance(direntUids); dialogFragment.setRefreshListener(new OnRefreshDataListener() { @Override public void onActionStatus(boolean isDone) { if (isDone) { ToastUtils.showLong(R.string.delete_successful); + } - closeActionMode(); + closeActionMode(); - loadData(true); - } + loadData(true); } }); dialogFragment.show(getChildFragmentManager(), DeleteFileDialogFragment.class.getSimpleName()); @@ -1101,7 +1275,16 @@ public void onActionStatus(boolean isDone) { * Share a file. Generating a file share link and send the link or file to someone * through some app. */ - public void showShareDialog(DirentModel direntModel) { + public void showShareDialog(List dirents) { + if (CollectionUtils.isEmpty(dirents)) { + return; + } + + //close action mode firstly + closeActionMode(); + + DirentModel direntModel = (DirentModel) dirents.get(0); + MaterialAlertDialogBuilder mBuilder = new MaterialAlertDialogBuilder(requireContext()); boolean inChina = Utils.isInChina(); @@ -1129,10 +1312,29 @@ public void showShareDialog(DirentModel direntModel) { }).show(); } + + private void exportFile(List dirents) { + if (CollectionUtils.isEmpty(dirents)) { + return; + } + + DirentModel direntModel = (DirentModel) dirents.get(0); + + File destinationFile = getLocalDestinationFile(direntModel.repo_id, direntModel.repo_name, direntModel.full_path); + + if (!destinationFile.exists()) { + Intent intent = FileActivity.start(requireContext(), direntModel, "export"); + fileActivityLauncher.launch(intent); + } else { + Objs.exportFile(this, destinationFile); + } + } + + /** * Copy multiple files */ - public void copyFiles(String srcRepoId, String srcRepoName, String srcDir, List dirents) { + public void copy(String srcRepoId, String srcRepoName, String srcDir, List dirents) { chooseCopyMoveDestForMultiFiles(srcRepoId, srcRepoName, srcDir, dirents, OpType.COPY); } @@ -1140,29 +1342,25 @@ public void copyFiles(String srcRepoId, String srcRepoName, String srcDir, List< /** * Move multiple files */ - public void moveFiles(String srcRepoId, String srcRepoName, String srcDir, List dirents) { + public void move(String srcRepoId, String srcRepoName, String srcDir, List dirents) { chooseCopyMoveDestForMultiFiles(srcRepoId, srcRepoName, srcDir, dirents, OpType.MOVE); } - /** - * Move multiple files - */ - public void downloadDirents(List dirents) { - String[] uids = dirents.stream().map(m -> m.uid).toArray(String[]::new); - BackgroundJobManagerImpl.getInstance().startDownloadChainWorker(uids); - - closeActionMode(); - } - private CopyMoveContext copyMoveContext = null; /** * Choose copy/move destination for multiple files */ private void chooseCopyMoveDestForMultiFiles(String repoID, String repoName, - String dirPath, List dirents, + String dirPath, List dirents, OpType op) { - copyMoveContext = new CopyMoveContext(repoID, repoName, dirPath, dirents, op); + if (CollectionUtils.isEmpty(dirents)) { + return; + } + + List direntModels = dirents.stream().map(m -> (DirentModel) m).collect(Collectors.toList()); + + copyMoveContext = new CopyMoveContext(repoID, repoName, dirPath, direntModels, op); Intent intent = new Intent(requireContext(), ObjSelectorActivity.class); intent.putExtra(ObjSelectorActivity.DATA_ACCOUNT, SupportAccountManager.getInstance().getCurrentAccount()); @@ -1215,24 +1413,6 @@ public void onActionStatus(boolean isDone) { dialogFragment.show(getChildFragmentManager(), CopyMoveDialogFragment.class.getSimpleName()); } - private void starOrNot(BaseModel model) { - if (model instanceof RepoModel) { - RepoModel repoModel = (RepoModel) model; - if (repoModel.starred) { - getViewModel().unStar(repoModel.repo_id, "/"); - } else { - getViewModel().star(repoModel.repo_id, "/"); - } - } else if (model instanceof DirentModel) { - DirentModel direntModel = (DirentModel) model; - String path = Utils.pathJoin(getNavContext().getNavPath(), direntModel.name); - if (direntModel.starred) { - getViewModel().unStar(direntModel.repo_id, path); - } else { - getViewModel().star(direntModel.repo_id, path); - } - } - } private final ActivityResultLauncher imagePreviewActivityLauncher = registerForActivityResult(new ActivityResultContracts.StartActivityForResult(), new ActivityResultCallback() { @Override @@ -1280,8 +1460,8 @@ public void onActivityResult(ActivityResult o) { } else if ("open_with".equals(action)) { - WidgetUtils.openWith(requireActivity(), destinationFile); + } else if ("open_text_mime".equals(action)) { MarkdownActivity.start(requireActivity(), localFullPath, repoId, targetFile); @@ -1289,17 +1469,6 @@ public void onActivityResult(ActivityResult o) { } }); - private void exportFile(DirentModel dirent) { - File destinationFile = getLocalDestinationFile(dirent.repo_id, dirent.repo_name, dirent.full_path); - - if (!destinationFile.exists()) { - Intent intent = FileActivity.start(requireContext(), dirent, "export"); - fileActivityLauncher.launch(intent); - } else { - Objs.exportFile(this, destinationFile); - } - } - private void shareFile(DirentModel dirent) { if (dirent.isDir()) { Objs.shareDirToWeChat(this, dirent.repo_id, dirent.full_path); @@ -1317,7 +1486,14 @@ private void shareFile(DirentModel dirent) { ////////////////add task///////////// - private void addUploadTask(DirentModel dirent, boolean isUpdate) { + private void addUploadTask(List dirents, boolean isUpdate) { + if (CollectionUtils.isEmpty(dirents)) { + return; + } + + List direntModels = dirents.stream().map(m -> (DirentModel) m).collect(Collectors.toList()); + DirentModel dirent = direntModels.get(0); + RepoModel targetedModel = getNavContext().getRepoModel(); String targetDir = getNavContext().getNavPath(); File localFilePath = getLocalDestinationFile(dirent.repo_id, dirent.repo_name, dirent.full_path); diff --git a/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoService.java b/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoService.java index 2ffebabca..ef7678051 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoService.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoService.java @@ -1,11 +1,15 @@ package com.seafile.seadroid2.ui.repo; +import com.seafile.seadroid2.framework.data.db.entities.PermissionEntity; import com.seafile.seadroid2.framework.data.db.entities.RepoModel; import com.seafile.seadroid2.framework.data.model.ResultModel; +import com.seafile.seadroid2.framework.data.model.permission.PermissionListWrapperModel; +import com.seafile.seadroid2.framework.data.model.permission.PermissionWrapperModel; import com.seafile.seadroid2.framework.data.model.repo.Dirent2Model; import com.seafile.seadroid2.framework.data.model.repo.DirentWrapperModel; import com.seafile.seadroid2.framework.data.model.repo.RepoWrapperModel; +import java.security.Permission; import java.util.Map; import io.reactivex.Single; @@ -41,11 +45,9 @@ public interface RepoService { @GET("api/v2.1/repos/{repo_id}/dir/?with_thumbnail=true") Call getDirentsSync(@Path("repo_id") String repoId, @Query("p") String path); - @Multipart - @POST("api/v2.1/starred-items/") - Single star(@PartMap Map map); - - @DELETE("api/v2.1/starred-items/") - Single unStar(@Query("repo_id") String repoId, @Query("path") String path); + @GET("api/v2.1/repos/{repo_id}/custom-share-permissions/{permission_id}/") + Single getCustomSharePermissionById(@Path("repo_id") String repoId, @Path("permission_id") int id); + @GET("api/v2.1/repos/{repo_id}/custom-share-permissions/") + Single getCustomSharePermissions(@Path("repo_id") String repoId); } diff --git a/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoViewModel.java b/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoViewModel.java index 49fc3ddf3..2afef9171 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoViewModel.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/repo/RepoViewModel.java @@ -1,25 +1,33 @@ package com.seafile.seadroid2.ui.repo; +import android.content.Context; import android.text.TextUtils; +import android.view.MenuInflater; +import android.view.MenuItem; import androidx.lifecycle.MutableLiveData; import com.blankj.utilcode.util.CollectionUtils; import com.blankj.utilcode.util.NetworkUtils; import com.blankj.utilcode.util.ToastUtils; +import com.seafile.seadroid2.R; import com.seafile.seadroid2.SeafException; import com.seafile.seadroid2.account.Account; import com.seafile.seadroid2.account.SupportAccountManager; +import com.seafile.seadroid2.ui.bottomsheetmenu.ActionMenu; import com.seafile.seadroid2.context.NavContext; import com.seafile.seadroid2.enums.FileViewType; import com.seafile.seadroid2.framework.data.db.AppDatabase; import com.seafile.seadroid2.framework.data.db.entities.DirentModel; import com.seafile.seadroid2.framework.data.db.entities.EncKeyCacheEntity; import com.seafile.seadroid2.framework.data.db.entities.FileTransferEntity; +import com.seafile.seadroid2.framework.data.db.entities.PermissionEntity; import com.seafile.seadroid2.framework.data.db.entities.RepoModel; import com.seafile.seadroid2.framework.data.model.BaseModel; import com.seafile.seadroid2.framework.data.model.ResultModel; import com.seafile.seadroid2.enums.TransferStatus; +import com.seafile.seadroid2.framework.data.model.permission.PermissionListWrapperModel; +import com.seafile.seadroid2.framework.data.model.permission.PermissionWrapperModel; import com.seafile.seadroid2.framework.data.model.repo.Dirent2Model; import com.seafile.seadroid2.framework.http.HttpIO; import com.seafile.seadroid2.framework.util.Utils; @@ -27,29 +35,39 @@ import com.seafile.seadroid2.ui.base.viewmodel.BaseViewModel; import com.seafile.seadroid2.framework.util.Objs; import com.seafile.seadroid2.framework.util.SLogs; +import com.seafile.seadroid2.ui.star.StarredService; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; +import java.util.Objects; import java.util.Optional; +import java.util.Set; +import java.util.stream.Collectors; +import io.reactivex.Completable; +import io.reactivex.Flowable; import io.reactivex.Single; +import io.reactivex.SingleSource; +import io.reactivex.functions.Action; import io.reactivex.functions.BiFunction; import io.reactivex.functions.Consumer; +import io.reactivex.functions.Function; +import kotlin.Pair; import okhttp3.RequestBody; public class RepoViewModel extends BaseViewModel { - private final MutableLiveData> ObjsListLiveData = new MutableLiveData<>(); - private final MutableLiveData StarLiveData = new MutableLiveData<>(); + private final MutableLiveData> _objListLiveData = new MutableLiveData<>(); + private final MutableLiveData _starredLiveData = new MutableLiveData<>(); - public MutableLiveData getStarLiveData() { - return StarLiveData; + public MutableLiveData getStarredLiveData() { + return _starredLiveData; } - public MutableLiveData> getObjsListLiveData() { - return ObjsListLiveData; + public MutableLiveData> getObjListLiveData() { + return _objListLiveData; } public void getEncCacheDB(String repoId, Consumer consumer) { @@ -67,29 +85,6 @@ public void accept(List list) throws Exception { } - public void getRepoModelFromLocal(String repoId, Consumer consumer) { - //from db - Single> singleDb = AppDatabase.getInstance().repoDao().getRepoById(repoId); - addSingleDisposable(singleDb, new Consumer>() { - @Override - public void accept(List repoModels) throws Exception { - if (consumer != null) { - if (CollectionUtils.isEmpty(repoModels)) { - //no data in sqlite, request RepoApi again - consumer.accept(null); - } else { - consumer.accept(repoModels.get(0)); - } - } - } - }, new Consumer() { - @Override - public void accept(Throwable throwable) throws Exception { - SLogs.e(throwable); - } - }); - } - public void loadData(NavContext context, boolean forceRefresh) { Account account = SupportAccountManager.getInstance().getCurrentAccount(); if (account == null) { @@ -126,7 +121,7 @@ public void accept(List repoModels) { } List list = Objs.parseRepoListForAdapter(repoModels, account.getSignature(), false); - getObjsListLiveData().setValue(list); + getObjListLiveData().setValue(list); if (isForce) { loadReposFromRemote(account); @@ -149,7 +144,7 @@ private void loadReposFromRemote(Account account) { addSingleDisposable(resultSingle, new Consumer>() { @Override public void accept(List models) throws Exception { - getObjsListLiveData().setValue(models); + getObjListLiveData().setValue(models); getRefreshLiveData().setValue(false); } }, new Consumer() { @@ -188,7 +183,7 @@ public void accept(List direntModels) throws Exception { if (CollectionUtils.isEmpty(rets)) { loadDirentsFromRemote(account, context); } else { - getObjsListLiveData().setValue(Objs.parseLocalDirents(rets)); + getObjListLiveData().setValue(Objs.parseLocalDirents(rets)); getRefreshLiveData().setValue(false); } } @@ -233,7 +228,7 @@ public void accept(List direntModels) throws Exception { if (CollectionUtils.isEmpty(direntModels)) { loadDirentsFromRemote(account, context); } else { - getObjsListLiveData().setValue(Objs.parseLocalDirents(direntModels)); + getObjListLiveData().setValue(Objs.parseLocalDirents(direntModels)); getRefreshLiveData().setValue(false); } } @@ -250,6 +245,10 @@ private void loadDirentsFromRemote(Account account, NavContext context) { String repoName = context.getRepoModel().repo_name; String parentDir = context.getNavPath(); + if ("/".equals(parentDir)) { + loadPermissionFromRemote(repoId); + } + Single> resultSingle = Objs.getDirentsSingleFromServer(account, repoId, repoName, parentDir); addSingleDisposable(resultSingle, new Consumer>() { @@ -265,9 +264,9 @@ public void accept(List direntModels) throws Exception { } } - getObjsListLiveData().setValue(new ArrayList<>(rets)); + getObjListLiveData().setValue(new ArrayList<>(rets)); } else { - getObjsListLiveData().setValue(new ArrayList<>(direntModels)); + getObjListLiveData().setValue(new ArrayList<>(direntModels)); } getRefreshLiveData().setValue(false); @@ -287,48 +286,502 @@ public void accept(Throwable throwable) throws Exception { }); } - //star - public void star(String repoId, String path) { - getRefreshLiveData().setValue(true); + public List inflateMenu(Context context, int rid) { + ActionMenu menu = new ActionMenu(context); + + MenuInflater inflater = new MenuInflater(context); + inflater.inflate(rid, menu); + + List items = new ArrayList<>(menu.size()); + for (int i = 0; i < menu.size(); i++) { + items.add(menu.getItem(i)); + } + + return items; + } + + private final MutableLiveData> _menuItemListLiveData = new MutableLiveData<>(); + + public MutableLiveData> getMenuItemListLiveData() { + return _menuItemListLiveData; + } - Map requestDataMap = new HashMap<>(); - requestDataMap.put("repo_id", repoId); - requestDataMap.put("path", path); - Map bodyMap = generateRequestBody(requestDataMap); - Single single = HttpIO.getCurrentInstance().execute(RepoService.class).star(bodyMap); - addSingleDisposable(single, new Consumer() { + public void getRepoModelAndPermissionEntity(String repoId, boolean isForce, Consumer> consumer) { + Single> r = getRepoModelAndPermissionEntitySingle(repoId, isForce); + addSingleDisposable(r, new Consumer>() { @Override - public void accept(Dirent2Model resultModel) throws Exception { - getStarLiveData().setValue(true); + public void accept(Pair repoModelPermissionEntityPair) throws Exception { + if (consumer != null) { + consumer.accept(repoModelPermissionEntityPair); + } } - }, new Consumer() { + }); + } + + /** + * get repoModel and permissionEntity from local, if not exist, get from remote. + * if isForce is true, get from remote directly and save to db + */ + private Single> getRepoModelAndPermissionEntitySingle(String repoId, boolean isForce) { + Single> dbSingle = AppDatabase.getInstance().repoDao().getRepoById(repoId); + return dbSingle.flatMap(new Function, SingleSource>>() { @Override - public void accept(Throwable throwable) throws Exception { - getRefreshLiveData().setValue(false); + public SingleSource> apply(List repoModels) throws Exception { + if (CollectionUtils.isEmpty(repoModels)) { + return null; + } - String errMsg = getErrorMsgByThrowable(throwable); - ToastUtils.showLong(errMsg); + RepoModel repoModel = repoModels.get(0); + if (TextUtils.isEmpty(repoModel.permission)) { + //This issue doesn't actually happen, but it's still checked again from the remote check + return Single.just(new Pair<>(repoModel, null)); + } + + if (!repoModel.isCustomPermission()) { + + //Optionally + PermissionEntity permission = convertCustomPermission(repoModel.repo_id, repoModel.permission); + + return Single.just(new Pair<>(repoModel, permission)); + } + + if (isForce) { + + //get permission from remote + return Single.just(new Pair<>(repoModel, null)); + } + + int pNum = repoModel.getCustomPermissionNum(); + + //get special number permission from db + Single> pSingle = AppDatabase.getInstance().permissionDAO().getByIdAsync(repoId, pNum); + + return pSingle.flatMap(new Function, SingleSource>>() { + @Override + public SingleSource> apply(List permissionEntities) throws Exception { + + //no data in local db + if (CollectionUtils.isEmpty(permissionEntities)) { + return Single.just(new Pair<>(repoModel, null)); + } + + //get first permission + return Single.just(new Pair<>(repoModel, permissionEntities.get(0))); + } + }); + } + }).flatMap(new Function, SingleSource>>() { + @Override + public SingleSource> apply(Pair pair) throws Exception { + if (pair.getSecond() != null) { + return Single.just(pair); + } + + Single> permissionSingle = getLoadRepoPermissionFromRemoteSingle(repoId); + return permissionSingle.flatMap(new Function, SingleSource>>() { + @Override + public SingleSource> apply(List remotePermissionEntities) throws Exception { + if (CollectionUtils.isEmpty(remotePermissionEntities)) { + return Single.just(pair); + } + + Optional f = remotePermissionEntities.stream().filter(p -> p.id == pair.getFirst().getCustomPermissionNum()).findFirst(); + return Single.just(new Pair<>(pair.getFirst(), f.get())); + } + }); } }); } - public void unStar(String repoId, String path) { - getRefreshLiveData().setValue(true); - Single single = HttpIO.getCurrentInstance().execute(RepoService.class).unStar(repoId, path); - addSingleDisposable(single, new Consumer() { + public HashMap> getPermissionObjMap() { + return _permissionObjMap; + } + + public HashMap getPermissionStackMap() { + return _permissionMap; + } + + // + private final HashMap> _permissionObjMap = new HashMap<>(); + private final HashMap _permissionMap = new HashMap<>(); + + public void inflateRepoMenuWithPermission(Context context, RepoModel repoModel, List disableMenuIds, List removedMenuIds, boolean isForce) { + if (!repoModel.is_checked) { + //remove permission + removeCachedPermissionMapData(repoModel.permission, repoModel.repo_id); + + List list = new ArrayList<>(getPermissionStackMap().values()); + + int menuId = R.menu.bottom_sheet_op_repo; + List items = parseMenu(context, menuId, list, disableMenuIds, removedMenuIds); + + getMenuItemListLiveData().setValue(items); + + return; + } + + + Single> r = getRepoModelAndPermissionEntitySingle(repoModel.repo_id, isForce); + + addSingleDisposable(r, new Consumer>() { + @Override + public void accept(Pair pair) throws Exception { + if (null == pair) { + getSeafExceptionLiveData().setValue(SeafException.unknownException); + return; + } + + RepoModel repoModel = pair.getFirst(); + PermissionEntity entity; + if (!repoModel.isCustomPermission()) { + entity = convertCustomPermission(repoModel.repo_id, repoModel.permission); + } else { + entity = pair.getSecond(); + } + + cachePermissionMapData(repoModel.permission, repoModel.repo_id, entity); + + + List list = new ArrayList<>(getPermissionStackMap().values()); + int menuId = R.menu.bottom_sheet_op_repo; + List items = parseMenu(context, menuId, list, disableMenuIds, removedMenuIds); + + getMenuItemListLiveData().setValue(items); + } + }); + } + + + public void inflateDirentMenuWithPermission(Context context, DirentModel direntModel, List disableMenuIds, List removedMenuIds, boolean isForce) { + int menuId = R.menu.bottom_sheet_op_dirent; + + + if (!direntModel.is_checked) { + //remove permission + removeCachedPermissionMapData(direntModel.permission, direntModel.uid); + + List tempList1 = new ArrayList<>(getPermissionStackMap().values()); + List items = parseMenu(context, menuId, tempList1, disableMenuIds, removedMenuIds); + getMenuItemListLiveData().setValue(items); + return; + } + + if (!direntModel.isCustomPermission()) { + PermissionEntity entity = convertCustomPermission(direntModel.repo_id, direntModel.permission); + cachePermissionMapData(direntModel.permission, direntModel.uid, entity); + + List tempList = new ArrayList<>(getPermissionStackMap().values()); + List items = parseMenu(context, menuId, tempList, disableMenuIds, removedMenuIds); + getMenuItemListLiveData().setValue(items); + return; + } + + + // + Single> r = getRepoModelAndPermissionEntitySingle(direntModel.repo_id, isForce); + addSingleDisposable(r, new Consumer>() { @Override - public void accept(ResultModel resultModel) throws Exception { - getStarLiveData().setValue(true); + public void accept(Pair pair) throws Exception { + RepoModel repoModel = pair.getFirst(); + PermissionEntity entity; + if (!repoModel.isCustomPermission()) { + entity = convertCustomPermission(repoModel.repo_id, repoModel.permission); + } else { + entity = pair.getSecond(); + } + + cachePermissionMapData(direntModel.permission, direntModel.uid, entity); + + List list = new ArrayList<>(getPermissionStackMap().values()); + List items = parseMenu(context, menuId, list, disableMenuIds, removedMenuIds); + getMenuItemListLiveData().setValue(items); + } + }); + } + + public void clearCachePermissionMap() { + getPermissionObjMap().clear(); + getPermissionStackMap().clear(); + } + + private void removeCachedPermissionMapData(String permission, String id) { + if (!getPermissionObjMap().containsKey(permission)) { + return; + } + + Set set = getPermissionObjMap().get(permission); + if (set == null) { + return; + } + + if (!set.contains(id)) { + return; + } + + set.remove(id); + + // + if (set.isEmpty()) { + getPermissionStackMap().remove(permission); + } + } + + private void cachePermissionMapData(String permission, String id, PermissionEntity entity) { + if (!getPermissionStackMap().containsKey(permission)) { + getPermissionStackMap().put(permission, entity); + } + + if (getPermissionObjMap().containsKey(permission)) { + Objects.requireNonNull(getPermissionObjMap().get(permission)).add(id); + } else { + getPermissionObjMap().put(permission, CollectionUtils.newHashSet(id)); + } + } + + private PermissionEntity convertCustomPermission(String repoId, String p) { + + PermissionEntity permission = new PermissionEntity(); + permission.name = p; + permission.id = -1; + permission.repo_id = repoId; + + if ("cloud-edit".equals(p)) { + //用户可以通过浏览器在线查看和编辑,文件不能被下载。 + permission.create = true; + permission.upload = false; + permission.download = false; + permission.preview = true; + permission.copy = true; + permission.delete = true; + permission.modify = true; + permission.download_external_link = false; + + } else if ("preview".equals(p)) { + //用户只能通过浏览器在线查看,文件不能被下载。 + permission.create = false; + permission.upload = false; + permission.download = false; + permission.preview = true; + permission.copy = false; + permission.delete = false; + permission.modify = false; + permission.download_external_link = false; + } else if ("r".equals(p)) { + //用户可以查看、下载和同步文件 + permission.create = false; + permission.upload = false; + permission.download = true; + permission.preview = true; + permission.copy = true; + permission.delete = false; + permission.modify = false; + permission.download_external_link = false; + } else if ("rw".equals(p)) { + permission.create = true; + permission.upload = true; + permission.download = true; + permission.preview = true; + permission.copy = true; + permission.delete = true; + permission.modify = true; + permission.download_external_link = true; + } + + return permission; + } + + private List parseMenu(Context context, int menuId, List list, List disableMenuIds, List removedMenuIds) { + List items = inflateMenu(context, menuId); + + //if no permission list, disable all menu + if (CollectionUtils.isEmpty(list)) { + for (MenuItem item : items) { + item.setEnabled(false); + } + + if (!CollectionUtils.isEmpty(removedMenuIds)) { + items = items.stream().filter(item -> item.getItemId() != R.id.unstar).collect(Collectors.toList()); + } + return items; + } + + //enable firstly + for (MenuItem item : items) { + item.setEnabled(true); + } + + //to disable + for (MenuItem item : items) { + if (!item.isEnabled()) { + continue; + } + + if (item.getItemId() == R.id.rename) { + long l = list.stream().filter(f -> !f.modify).count(); + item.setEnabled(!(l > 0)); + } else if (item.getItemId() == R.id.move) { + long l = list.stream().filter(f -> !f.modify).count(); + item.setEnabled(!(l > 0)); + } else if (item.getItemId() == R.id.copy) { + long l = list.stream().filter(f -> !f.copy).count(); + item.setEnabled(!(l > 0)); + } else if (item.getItemId() == R.id.delete) { + long l = list.stream().filter(f -> !f.delete).count(); + item.setEnabled(!(l > 0)); + } else if (item.getItemId() == R.id.upload) { + long l = list.stream().filter(f -> !f.upload).count(); + item.setEnabled(!(l > 0)); + } else if (item.getItemId() == R.id.download) { + long l = list.stream().filter(f -> !f.download).count(); + item.setEnabled(!(l > 0)); + } else if (item.getItemId() == R.id.share) { + long l = list.stream().filter(f -> !f.download_external_link).count(); + item.setEnabled(!(l > 0)); + } else if (item.getItemId() == R.id.export) { + long l = list.stream().filter(f -> !f.download).count(); + item.setEnabled(!(l > 0)); + } else if (item.getItemId() == R.id.open) { + long l = list.stream().filter(f -> !f.download).count(); + item.setEnabled(!(l > 0)); + } + + if (!CollectionUtils.isEmpty(disableMenuIds)) { + if (disableMenuIds.contains(item.getItemId())) { + item.setEnabled(false); + } + } + } + + if (!CollectionUtils.isEmpty(removedMenuIds)) { + items = items.stream().filter(item -> !removedMenuIds.contains(item.getItemId())).collect(Collectors.toList()); + } + + return items; + } + + private Single> getLoadRepoPermissionFromRemoteSingle(String repoId) { + Single single = HttpIO.getCurrentInstance().execute(RepoService.class).getCustomSharePermissions(repoId); + return single.flatMap(new Function>>() { + @Override + public SingleSource> apply(PermissionListWrapperModel wrapperModel) throws Exception { + + List list = CollectionUtils.newArrayList(); + + for (PermissionWrapperModel model : wrapperModel.permission_list) { + PermissionEntity permission = new PermissionEntity(); + permission.id = model.id; + permission.name = model.name; + permission.description = model.description; + + permission.create = model.permission.create; + permission.upload = model.permission.upload; + permission.download = model.permission.download; + permission.copy = model.permission.copy; + permission.delete = model.permission.delete; + permission.modify = model.permission.modify; + permission.download_external_link = model.permission.download_external_link; + permission.preview = model.permission.preview; + permission.repo_id = repoId; + + list.add(permission); + } + + Completable insertCompletable = AppDatabase.getInstance().permissionDAO().insertAllAsync(list); + Single insertAllSingle = insertCompletable.toSingleDefault(0L); + return insertAllSingle.flatMap(new Function>>() { + @Override + public SingleSource> apply(Long aLong) throws Exception { + SLogs.d("The list has been inserted into the local database"); + return Single.just(list); + } + }); + } + }); + } + + private void loadPermissionFromRemote(String repoId) { + Single> r = getLoadRepoPermissionFromRemoteSingle(repoId); + + addSingleDisposable(r, new Consumer>() { + @Override + public void accept(List list) throws Exception { + SLogs.e("permission has been loaded"); + } + }); + } + + public void multiStarOrNot(List selectedList, boolean isStar) { + if (CollectionUtils.isEmpty(selectedList)) { + return; + } + + getShowLoadingDialogLiveData().setValue(true); + + List> singles = new ArrayList<>(); + for (BaseModel baseModel : selectedList) { + if (baseModel instanceof RepoModel m) { + + if (isStar == m.starred) { + continue; + } + + singles.add(getStarSingle(m.repo_id, "/", isStar)); + } else if (baseModel instanceof DirentModel m) { + if (isStar == m.starred) { + continue; + } + + singles.add(getStarSingle(m.repo_id, m.full_path, isStar)); + } + } + + if (CollectionUtils.isEmpty(singles)) { + getShowLoadingDialogLiveData().setValue(false); + return; + } + + Flowable flowable = Flowable.mergeDelayError(singles, 5, Flowable.bufferSize()); + addFlowableDisposable(flowable, new Consumer() { + @Override + public void accept(Object o) throws Exception { + if (o instanceof Dirent2Model dirent2Model) { + SLogs.d("dirent star success:" + dirent2Model); + } else if (o instanceof ResultModel resultModel) { + SLogs.d("repo star success:" + resultModel); + } } }, new Consumer() { @Override public void accept(Throwable throwable) throws Exception { - getRefreshLiveData().setValue(false); String errMsg = getErrorMsgByThrowable(throwable); ToastUtils.showLong(errMsg); } + }, new Action() { + @Override + public void run() throws Exception { + getShowLoadingDialogLiveData().setValue(false); + getStarredLiveData().setValue(true); + } }); + + + } + + private Flowable getStarSingle(String repoId, String path, boolean isStar) { + if (isStar) { + Map requestDataMap = new HashMap<>(); + requestDataMap.put("repo_id", repoId); + requestDataMap.put("path", path); + + Map bodyMap = generateRequestBody(requestDataMap); + Single single = HttpIO.getCurrentInstance().execute(StarredService.class).star(bodyMap); + + return single.toFlowable(); + } else { + Single single1 = HttpIO.getCurrentInstance().execute(StarredService.class).unStar(repoId, path); + return single1.toFlowable(); + } } } diff --git a/app/src/main/java/com/seafile/seadroid2/ui/search/Search2Activity.java b/app/src/main/java/com/seafile/seadroid2/ui/search/Search2Activity.java index 54ed1a95d..7e5265d6e 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/search/Search2Activity.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/search/Search2Activity.java @@ -157,7 +157,7 @@ public void onChanged(SeafException e) { } }); - getViewModel().getListLiveData().observe(this, new Observer>() { + getViewModel().getSearchListLiveData().observe(this, new Observer>() { @Override public void onChanged(List result) { if (CollectionUtils.isEmpty(result)) { @@ -348,7 +348,7 @@ private void loadNext(String query, boolean isRefresh) { page++; - getViewModel().loadNext(query, page, PAGE_SIZE); + getViewModel().searchNext(query, page, PAGE_SIZE); } public void onItemClick(SearchModel searchedFile) { diff --git a/app/src/main/java/com/seafile/seadroid2/ui/search/SearchViewModel.java b/app/src/main/java/com/seafile/seadroid2/ui/search/SearchViewModel.java index 0901259df..d36a252a5 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/search/SearchViewModel.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/search/SearchViewModel.java @@ -1,5 +1,7 @@ package com.seafile.seadroid2.ui.search; +import android.text.TextUtils; + import androidx.lifecycle.MutableLiveData; import com.blankj.utilcode.util.CollectionUtils; @@ -23,17 +25,21 @@ public class SearchViewModel extends BaseViewModel { private final MutableLiveData> mListLiveData = new MutableLiveData<>(); - public MutableLiveData> getListLiveData() { + public MutableLiveData> getSearchListLiveData() { return mListLiveData; } - public void loadNext(String q, int pageNo, int pageSize) { + public void searchNext(String q, int pageNo, int pageSize) { + if (TextUtils.isEmpty(q)){ + return; + } + getRefreshLiveData().setValue(true); Single single = HttpIO.getCurrentInstance().execute(SearchService.class).search("all", q, "all", pageNo, pageSize); addSingleDisposable(single, new Consumer() { @Override public void accept(SearchWrapperModel searchWrapperModel) throws Exception { - getListLiveData().setValue(searchWrapperModel.results); + getSearchListLiveData().setValue(searchWrapperModel.results); getRefreshLiveData().setValue(false); } }, new Consumer() { diff --git a/app/src/main/java/com/seafile/seadroid2/ui/selector/ObjSelectorActivity.java b/app/src/main/java/com/seafile/seadroid2/ui/selector/ObjSelectorActivity.java index db6e7db41..81aa7a744 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/selector/ObjSelectorActivity.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/selector/ObjSelectorActivity.java @@ -188,7 +188,7 @@ private void initRv() { binding.rv.addItemDecoration(decoration); adapter = new RepoQuickAdapter(); - adapter.setSelectType(RepoSelectType.DIRENT); + adapter.setSelectType(RepoSelectType.FOLDER); adapter.setFileViewType(FileViewType.LIST); adapter.setOnItemClickListener((baseQuickAdapter, view, i) -> { diff --git a/app/src/main/java/com/seafile/seadroid2/ui/settings/TabSettingsFragment.java b/app/src/main/java/com/seafile/seadroid2/ui/settings/TabSettingsFragment.java index b083d5b5c..a80a83bdf 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/settings/TabSettingsFragment.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/settings/TabSettingsFragment.java @@ -239,9 +239,8 @@ private void initAlbumBackupPref() { mAlbumBackupAdvanced.setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() { @Override public boolean onPreferenceClick(@NonNull Preference preference) { - BackgroundJobManagerImpl.getInstance().startMediaChainWorker(true); -// Intent intent = new Intent(requireActivity(), SettingsAlbumBackupAdvancedActivity.class); -// albumBackupAdvanceLauncher.launch(intent); + Intent intent = new Intent(requireActivity(), SettingsAlbumBackupAdvancedActivity.class); + albumBackupAdvanceLauncher.launch(intent); return true; } }); diff --git a/app/src/main/java/com/seafile/seadroid2/ui/star/StarredQuickFragment.java b/app/src/main/java/com/seafile/seadroid2/ui/star/StarredQuickFragment.java index 6b1975841..bb7c51d0b 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/star/StarredQuickFragment.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/star/StarredQuickFragment.java @@ -19,7 +19,6 @@ import androidx.annotation.OptIn; import androidx.lifecycle.Observer; import androidx.lifecycle.ViewModelProvider; -import androidx.media3.common.util.UnstableApi; import com.blankj.utilcode.util.ToastUtils; import com.chad.library.adapter4.BaseQuickAdapter; @@ -29,9 +28,10 @@ import com.seafile.seadroid2.account.Account; import com.seafile.seadroid2.account.SupportAccountManager; import com.seafile.seadroid2.annotation.Unstable; -import com.seafile.seadroid2.bottomsheetmenu.BottomSheetHelper; -import com.seafile.seadroid2.bottomsheetmenu.BottomSheetMenuFragment; -import com.seafile.seadroid2.bottomsheetmenu.OnMenuClickListener; +import com.seafile.seadroid2.framework.datastore.sp.SettingsManager; +import com.seafile.seadroid2.ui.bottomsheetmenu.BottomSheetHelper; +import com.seafile.seadroid2.ui.bottomsheetmenu.BottomSheetMenuFragment; +import com.seafile.seadroid2.ui.bottomsheetmenu.OnMenuClickListener; import com.seafile.seadroid2.config.Constants; import com.seafile.seadroid2.databinding.LayoutFrameSwipeRvBinding; import com.seafile.seadroid2.framework.data.db.entities.StarredModel; @@ -97,8 +97,6 @@ public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceStat initViewModel(); } - private boolean isForce = false; - @Override public void onFirstResume() { super.onFirstResume(); @@ -106,8 +104,10 @@ public void onFirstResume() { } @Override - public void onResume() { - super.onResume(); + public void onOtherResume() { + super.onOtherResume(); + + boolean isForce = SettingsManager.getForceRefreshStarredListState(); if (isForce) { reload(); } @@ -148,13 +148,6 @@ private void showErrorTip(SeafException seafException) { } private void initViewModel() { - mainViewModel.getOnForceRefreshStarredListLiveData().observe(getViewLifecycleOwner(), new Observer() { - @Override - public void onChanged(Boolean aBoolean) { - isForce = aBoolean; - } - }); - getViewModel().getRefreshLiveData().observe(getViewLifecycleOwner(), new Observer() { @Override public void onChanged(Boolean aBoolean) { diff --git a/app/src/main/java/com/seafile/seadroid2/ui/star/StarredService.java b/app/src/main/java/com/seafile/seadroid2/ui/star/StarredService.java index de5a85f0a..0906c241f 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/star/StarredService.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/star/StarredService.java @@ -1,18 +1,30 @@ package com.seafile.seadroid2.ui.star; import com.seafile.seadroid2.framework.data.model.ResultModel; +import com.seafile.seadroid2.framework.data.model.repo.Dirent2Model; import com.seafile.seadroid2.framework.data.model.star.StarredWrapperModel; +import java.util.Map; + import io.reactivex.Single; +import okhttp3.RequestBody; +import retrofit2.http.Body; import retrofit2.http.DELETE; import retrofit2.http.GET; +import retrofit2.http.Multipart; +import retrofit2.http.POST; +import retrofit2.http.PartMap; import retrofit2.http.Query; public interface StarredService { @GET("api/v2.1/starred-items/") Single getStarItems(); + @Multipart + @POST("api/v2.1/starred-items/") + Single star(@PartMap Map map); + @DELETE("api/v2.1/starred-items/") - Single unStarItem(@Query("repo_id") String repoId, @Query("path") String path); + Single unStar(@Query("repo_id") String repoId, @Query("path") String path); } diff --git a/app/src/main/java/com/seafile/seadroid2/ui/star/StarredViewModel.java b/app/src/main/java/com/seafile/seadroid2/ui/star/StarredViewModel.java index edbf73af9..6aafb742d 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/star/StarredViewModel.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/star/StarredViewModel.java @@ -59,7 +59,7 @@ public void accept(Throwable throwable) throws Exception { } public void unStarItem(String repoId, String path) { - Single flowable = HttpIO.getCurrentInstance().execute(StarredService.class).unStarItem(repoId, path); + Single flowable = HttpIO.getCurrentInstance().execute(StarredService.class).unStar(repoId, path); addSingleDisposable(flowable, new Consumer() { @Override public void accept(ResultModel resultModel) throws Exception { diff --git a/app/src/main/java/com/seafile/seadroid2/ui/transfer_list/TransferListFragment.java b/app/src/main/java/com/seafile/seadroid2/ui/transfer_list/TransferListFragment.java index 387e846df..c8f764aeb 100644 --- a/app/src/main/java/com/seafile/seadroid2/ui/transfer_list/TransferListFragment.java +++ b/app/src/main/java/com/seafile/seadroid2/ui/transfer_list/TransferListFragment.java @@ -25,8 +25,8 @@ import com.chad.library.adapter4.QuickAdapterHelper; import com.google.android.material.dialog.MaterialAlertDialogBuilder; import com.seafile.seadroid2.R; -import com.seafile.seadroid2.bottomsheetmenu.BottomSheetHelper; -import com.seafile.seadroid2.bottomsheetmenu.BottomSheetMenuFragment; +import com.seafile.seadroid2.ui.bottomsheetmenu.BottomSheetHelper; +import com.seafile.seadroid2.ui.bottomsheetmenu.BottomSheetMenuFragment; import com.seafile.seadroid2.databinding.LayoutFrameSwipeRvBinding; import com.seafile.seadroid2.framework.data.db.entities.FileTransferEntity; import com.seafile.seadroid2.enums.TransferAction; diff --git a/app/src/main/res/drawable/baseline_activities_24.xml b/app/src/main/res/drawable/baseline_activities_24.xml index 999954588..d1eea04d0 100644 --- a/app/src/main/res/drawable/baseline_activities_24.xml +++ b/app/src/main/res/drawable/baseline_activities_24.xml @@ -5,8 +5,8 @@ android:viewportHeight="24"> + android:fillColor="#999999"/> + android:fillColor="#999999"/> diff --git a/app/src/main/res/drawable/baseline_activities_bold_24.xml b/app/src/main/res/drawable/baseline_activities_bold_24.xml index c78f2f5df..d61da030c 100644 --- a/app/src/main/res/drawable/baseline_activities_bold_24.xml +++ b/app/src/main/res/drawable/baseline_activities_bold_24.xml @@ -6,9 +6,9 @@ + android:fillColor="#999999" + android:strokeColor="#999999"/> + android:fillColor="#999999"/> diff --git a/app/src/main/res/drawable/baseline_add_24.xml b/app/src/main/res/drawable/baseline_add_24.xml index f0e955833..63c79f09e 100644 --- a/app/src/main/res/drawable/baseline_add_24.xml +++ b/app/src/main/res/drawable/baseline_add_24.xml @@ -4,6 +4,6 @@ android:viewportWidth="24" android:viewportHeight="24"> diff --git a/app/src/main/res/drawable/baseline_arrow_back_24.xml b/app/src/main/res/drawable/baseline_arrow_back_24.xml index 2f142289d..b35bd4d7a 100644 --- a/app/src/main/res/drawable/baseline_arrow_back_24.xml +++ b/app/src/main/res/drawable/baseline_arrow_back_24.xml @@ -5,6 +5,6 @@ android:viewportWidth="24" android:viewportHeight="24"> diff --git a/app/src/main/res/drawable/baseline_arrow_forward_24.xml b/app/src/main/res/drawable/baseline_arrow_forward_24.xml index 6710c2253..17d33ccec 100644 --- a/app/src/main/res/drawable/baseline_arrow_forward_24.xml +++ b/app/src/main/res/drawable/baseline_arrow_forward_24.xml @@ -5,6 +5,6 @@ android:viewportWidth="24" android:viewportHeight="24"> diff --git a/app/src/main/res/drawable/baseline_delete_20.xml b/app/src/main/res/drawable/baseline_delete_20.xml deleted file mode 100644 index 08172bb13..000000000 --- a/app/src/main/res/drawable/baseline_delete_20.xml +++ /dev/null @@ -1,36 +0,0 @@ - - - - - - - diff --git a/app/src/main/res/drawable/baseline_edit_24.xml b/app/src/main/res/drawable/baseline_edit_24.xml index 8e18d509d..7598117c7 100644 --- a/app/src/main/res/drawable/baseline_edit_24.xml +++ b/app/src/main/res/drawable/baseline_edit_24.xml @@ -5,5 +5,5 @@ android:viewportHeight="24"> + android:fillColor="#999999"/> diff --git a/app/src/main/res/drawable/baseline_expandable_24.xml b/app/src/main/res/drawable/baseline_expandable_24.xml deleted file mode 100644 index 4f2c72d8e..000000000 --- a/app/src/main/res/drawable/baseline_expandable_24.xml +++ /dev/null @@ -1,13 +0,0 @@ - - - - - - diff --git a/app/src/main/res/drawable/baseline_folder_24.xml b/app/src/main/res/drawable/baseline_folder_24.xml index dd89a805c..a134eee6a 100644 --- a/app/src/main/res/drawable/baseline_folder_24.xml +++ b/app/src/main/res/drawable/baseline_folder_24.xml @@ -6,11 +6,9 @@ + android:strokeColor="#999999"/> + android:strokeColor="#999999"/> diff --git a/app/src/main/res/drawable/baseline_folder_read_only_24.xml b/app/src/main/res/drawable/baseline_folder_read_only_24.xml index 469d6dfdd..b64b14455 100644 --- a/app/src/main/res/drawable/baseline_folder_read_only_24.xml +++ b/app/src/main/res/drawable/baseline_folder_read_only_24.xml @@ -10,15 +10,13 @@ + android:strokeColor="#999999"/> + android:strokeColor="#999999"/> diff --git a/app/src/main/res/drawable/baseline_more_vert_24.xml b/app/src/main/res/drawable/baseline_more_vert_24.xml index 59dbd60f5..7bf4a0055 100644 --- a/app/src/main/res/drawable/baseline_more_vert_24.xml +++ b/app/src/main/res/drawable/baseline_more_vert_24.xml @@ -4,6 +4,6 @@ android:viewportWidth="24" android:viewportHeight="24"> diff --git a/app/src/main/res/drawable/baseline_repo_bold_24.xml b/app/src/main/res/drawable/baseline_repo_bold_24.xml index df4e2588d..6be907ce2 100644 --- a/app/src/main/res/drawable/baseline_repo_bold_24.xml +++ b/app/src/main/res/drawable/baseline_repo_bold_24.xml @@ -8,34 +8,34 @@ android:strokeLineJoin="round" android:strokeWidth="1.3" android:fillColor="#00000000" - android:strokeColor="@color/bar_icon_tint_color" + android:strokeColor="#999999" android:strokeLineCap="round"/> diff --git a/app/src/main/res/drawable/baseline_repo_encrypted_24.xml b/app/src/main/res/drawable/baseline_repo_encrypted_24.xml index e4e2b5536..e10b83090 100644 --- a/app/src/main/res/drawable/baseline_repo_encrypted_24.xml +++ b/app/src/main/res/drawable/baseline_repo_encrypted_24.xml @@ -17,7 +17,7 @@ android:strokeLineJoin="round" android:strokeWidth="1" android:fillColor="#00000000" - android:strokeColor="@color/item_icon_tint_color" + android:strokeColor="#999999" android:strokeLineCap="round"/> @@ -28,7 +28,7 @@ android:strokeLineJoin="round" android:strokeWidth="1" android:fillColor="#00000000" - android:strokeColor="@color/item_icon_tint_color" + android:strokeColor="#999999" android:strokeLineCap="round"/> @@ -39,7 +39,7 @@ android:strokeLineJoin="round" android:strokeWidth="1" android:fillColor="#00000000" - android:strokeColor="@color/item_icon_tint_color" + android:strokeColor="#999999" android:strokeLineCap="round"/> @@ -50,7 +50,7 @@ android:strokeLineJoin="round" android:strokeWidth="1" android:fillColor="#00000000" - android:strokeColor="@color/item_icon_tint_color" + android:strokeColor="#999999" android:strokeLineCap="round"/> @@ -61,7 +61,7 @@ android:strokeLineJoin="round" android:strokeWidth="1" android:fillColor="#00000000" - android:strokeColor="@color/item_icon_tint_color" + android:strokeColor="#999999" android:strokeLineCap="round"/> @@ -71,20 +71,20 @@ android:strokeLineJoin="round" android:strokeWidth="1" android:fillColor="#00000000" - android:strokeColor="@color/item_icon_tint_color" + android:strokeColor="#999999" android:strokeLineCap="round"/> diff --git a/app/src/main/res/drawable/baseline_repo_readonly_24.xml b/app/src/main/res/drawable/baseline_repo_readonly_24.xml index fa02eeeed..e78b31668 100644 --- a/app/src/main/res/drawable/baseline_repo_readonly_24.xml +++ b/app/src/main/res/drawable/baseline_repo_readonly_24.xml @@ -17,7 +17,7 @@ android:strokeLineJoin="round" android:strokeWidth="1" android:fillColor="#00000000" - android:strokeColor="@color/item_icon_tint_color" + android:strokeColor="#999999" android:strokeLineCap="round"/> @@ -28,7 +28,7 @@ android:strokeLineJoin="round" android:strokeWidth="1" android:fillColor="#00000000" - android:strokeColor="@color/item_icon_tint_color" + android:strokeColor="#999999" android:strokeLineCap="round"/> @@ -39,7 +39,7 @@ android:strokeLineJoin="round" android:strokeWidth="1" android:fillColor="#00000000" - android:strokeColor="@color/item_icon_tint_color" + android:strokeColor="#999999" android:strokeLineCap="round"/> @@ -50,7 +50,7 @@ android:strokeLineJoin="round" android:strokeWidth="1" android:fillColor="#00000000" - android:strokeColor="@color/item_icon_tint_color" + android:strokeColor="#999999" android:strokeLineCap="round"/> @@ -61,7 +61,7 @@ android:strokeLineJoin="round" android:strokeWidth="1" android:fillColor="#00000000" - android:strokeColor="@color/item_icon_tint_color" + android:strokeColor="#999999" android:strokeLineCap="round"/> @@ -71,27 +71,27 @@ android:strokeLineJoin="round" android:strokeWidth="1" android:fillColor="#00000000" - android:strokeColor="@color/item_icon_tint_color" + android:strokeColor="#999999" android:strokeLineCap="round"/> diff --git a/app/src/main/res/drawable/baseline_schedule_24.xml b/app/src/main/res/drawable/baseline_schedule_24.xml deleted file mode 100644 index 0c89ae89c..000000000 --- a/app/src/main/res/drawable/baseline_schedule_24.xml +++ /dev/null @@ -1,27 +0,0 @@ - - - - diff --git a/app/src/main/res/drawable/baseline_search_24.xml b/app/src/main/res/drawable/baseline_search_24.xml index 968087902..9e4d91f15 100644 --- a/app/src/main/res/drawable/baseline_search_24.xml +++ b/app/src/main/res/drawable/baseline_search_24.xml @@ -6,8 +6,7 @@ + android:fillColor="#999999" + android:pathData="M9.901,15.683C12.794,15.683 15.139,13.329 15.139,10.425C15.139,7.522 12.794,5.168 9.901,5.168C7.009,5.168 4.664,7.522 4.664,10.425C4.664,13.329 7.009,15.683 9.901,15.683ZM14.849,14.502L20.08,19.753C20.307,19.981 20.307,20.351 20.08,20.579C19.852,20.807 19.484,20.807 19.257,20.579L14.032,15.334C12.917,16.28 11.475,16.851 9.901,16.851C6.366,16.851 3.5,13.974 3.5,10.425C3.5,6.877 6.366,4 9.901,4C13.436,4 16.302,6.877 16.302,10.425C16.302,11.973 15.757,13.393 14.849,14.502Z" /> diff --git a/app/src/main/res/drawable/baseline_settings_24.xml b/app/src/main/res/drawable/baseline_settings_24.xml deleted file mode 100644 index 294213597..000000000 --- a/app/src/main/res/drawable/baseline_settings_24.xml +++ /dev/null @@ -1,12 +0,0 @@ - - - - diff --git a/app/src/main/res/drawable/baseline_sort_24.xml b/app/src/main/res/drawable/baseline_sort_24.xml index f5fbabdd8..9fe129669 100644 --- a/app/src/main/res/drawable/baseline_sort_24.xml +++ b/app/src/main/res/drawable/baseline_sort_24.xml @@ -4,6 +4,6 @@ android:viewportWidth="24" android:viewportHeight="24"> diff --git a/app/src/main/res/drawable/baseline_star_outline_24.xml b/app/src/main/res/drawable/baseline_star_outline_24.xml index 2de43f65e..0735fb743 100644 --- a/app/src/main/res/drawable/baseline_star_outline_24.xml +++ b/app/src/main/res/drawable/baseline_star_outline_24.xml @@ -4,6 +4,6 @@ android:viewportWidth="24" android:viewportHeight="24"> diff --git a/app/src/main/res/drawable/baseline_transfer_delete_24.xml b/app/src/main/res/drawable/baseline_transfer_delete_24.xml index aed69638b..7d7266725 100644 --- a/app/src/main/res/drawable/baseline_transfer_delete_24.xml +++ b/app/src/main/res/drawable/baseline_transfer_delete_24.xml @@ -5,8 +5,8 @@ android:viewportHeight="24"> + android:fillColor="#999999"/> + android:fillColor="#999999"/> diff --git a/app/src/main/res/drawable/baseline_transfer_restart_24.xml b/app/src/main/res/drawable/baseline_transfer_restart_24.xml index 34029764a..73520d07e 100644 --- a/app/src/main/res/drawable/baseline_transfer_restart_24.xml +++ b/app/src/main/res/drawable/baseline_transfer_restart_24.xml @@ -4,6 +4,6 @@ android:viewportWidth="24" android:viewportHeight="24"> diff --git a/app/src/main/res/drawable/baseline_transfer_select_all2_24.xml b/app/src/main/res/drawable/baseline_transfer_select_all2_24.xml index 425ba9aec..e1d1bd32e 100644 --- a/app/src/main/res/drawable/baseline_transfer_select_all2_24.xml +++ b/app/src/main/res/drawable/baseline_transfer_select_all2_24.xml @@ -5,8 +5,8 @@ android:viewportHeight="24"> + android:fillColor="#999999"/> + android:fillColor="#999999"/> diff --git a/app/src/main/res/drawable/baseline_transfer_select_all_24.xml b/app/src/main/res/drawable/baseline_transfer_select_all_24.xml deleted file mode 100644 index 61c51c8aa..000000000 --- a/app/src/main/res/drawable/baseline_transfer_select_all_24.xml +++ /dev/null @@ -1,9 +0,0 @@ - - - diff --git a/app/src/main/res/drawable/ic_check_square_solid.xml b/app/src/main/res/drawable/ic_check_square_solid.xml index e251c6c4c..5daf886a0 100644 --- a/app/src/main/res/drawable/ic_check_square_solid.xml +++ b/app/src/main/res/drawable/ic_check_square_solid.xml @@ -5,5 +5,5 @@ android:viewportHeight="32"> + android:fillColor="#666666"/> diff --git a/app/src/main/res/drawable/ic_checkbox_checked.xml b/app/src/main/res/drawable/ic_checkbox_checked.xml new file mode 100644 index 000000000..f11148b17 --- /dev/null +++ b/app/src/main/res/drawable/ic_checkbox_checked.xml @@ -0,0 +1,9 @@ + + + diff --git a/app/src/main/res/drawable/ic_checkbox_unchecked.xml b/app/src/main/res/drawable/ic_checkbox_unchecked.xml new file mode 100644 index 000000000..9327481c5 --- /dev/null +++ b/app/src/main/res/drawable/ic_checkbox_unchecked.xml @@ -0,0 +1,9 @@ + + + diff --git a/app/src/main/res/drawable/ic_setup_32.xml b/app/src/main/res/drawable/ic_setup_32.xml new file mode 100644 index 000000000..0df7a9a07 --- /dev/null +++ b/app/src/main/res/drawable/ic_setup_32.xml @@ -0,0 +1,12 @@ + + + + diff --git a/app/src/main/res/drawable/seekbar_for_webview_style.xml b/app/src/main/res/drawable/seekbar_for_webview_style.xml index 3dd305187..5501769bc 100644 --- a/app/src/main/res/drawable/seekbar_for_webview_style.xml +++ b/app/src/main/res/drawable/seekbar_for_webview_style.xml @@ -3,13 +3,12 @@ - + + android:endColor="@color/progress_bar_background_color" + android:startColor="@color/progress_bar_background_color" /> @@ -17,13 +16,12 @@ - + + android:endColor="@color/progress_bar_second_color" + android:startColor="@color/progress_bar_second_color" /> diff --git a/app/src/main/res/layout/activity_main.xml b/app/src/main/res/layout/activity_main.xml index caa9a11e1..6ad01c3a6 100644 --- a/app/src/main/res/layout/activity_main.xml +++ b/app/src/main/res/layout/activity_main.xml @@ -1,37 +1,54 @@ - + android:layout_height="match_parent"> - + android:layout_height="match_parent" + android:orientation="vertical"> - + - + + + + + - + + - - \ No newline at end of file + android:layout_height="400dp" + android:layout_gravity="bottom" + android:background="@color/material_blue_grey_500" + app:behavior_hideable="false" + app:behavior_peekHeight="200dp" + app:layout_behavior="@string/bottom_sheet_behavior" /> + diff --git a/app/src/main/res/layout/bottom_sheet_item_grid.xml b/app/src/main/res/layout/bottom_sheet_item_grid.xml index fbb639334..c8a4a1b1a 100644 --- a/app/src/main/res/layout/bottom_sheet_item_grid.xml +++ b/app/src/main/res/layout/bottom_sheet_item_grid.xml @@ -1,5 +1,6 @@ \ No newline at end of file diff --git a/app/src/main/res/layout/item_dirent.xml b/app/src/main/res/layout/item_dirent.xml index ecdff6837..8eb05c42f 100644 --- a/app/src/main/res/layout/item_dirent.xml +++ b/app/src/main/res/layout/item_dirent.xml @@ -20,17 +20,6 @@ app:layout_constraintTop_toTopOf="parent" app:shapeAppearance="@style/ShapeCorner4Style" /> - @@ -90,19 +79,37 @@ - + app:layout_constraintTop_toTopOf="parent"> + + + + + + + @@ -22,20 +22,6 @@ app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toTopOf="parent" /> - - - @@ -47,31 +47,34 @@ tools:text="@string/app_name" /> - + app:layout_constraintTop_toTopOf="parent"> - + + + + - \ No newline at end of file + \ No newline at end of file diff --git a/app/src/main/res/layout/item_sdoc_comment.xml b/app/src/main/res/layout/item_sdoc_comment.xml index ee802319c..073f33b27 100644 --- a/app/src/main/res/layout/item_sdoc_comment.xml +++ b/app/src/main/res/layout/item_sdoc_comment.xml @@ -54,15 +54,15 @@ app:layout_constraintStart_toEndOf="@+id/comment_user_avatar" app:layout_constraintTop_toBottomOf="@+id/comment_nick_name" /> - - - - - - - - - + + tools:src="@drawable/baseline_more_vert_24" /> @@ -67,7 +67,6 @@ app:layout_constraintTop_toTopOf="parent" app:tint="@color/item_subtitle_color" /> - - - - - - + app:layout_constraintEnd_toEndOf="parent" + app:tint="@color/item_subtitle_color" /> diff --git a/app/src/main/res/layout/layout_bottom_sheet_recycler_menu.xml b/app/src/main/res/layout/layout_bottom_sheet_menu_dialog.xml similarity index 83% rename from app/src/main/res/layout/layout_bottom_sheet_recycler_menu.xml rename to app/src/main/res/layout/layout_bottom_sheet_menu_dialog.xml index db17813f2..6b8a44625 100644 --- a/app/src/main/res/layout/layout_bottom_sheet_recycler_menu.xml +++ b/app/src/main/res/layout/layout_bottom_sheet_menu_dialog.xml @@ -1,5 +1,6 @@ - - - + android:paddingBottom="16dp" + tools:listitem="@layout/bottom_sheet_item_grid" /> \ No newline at end of file diff --git a/app/src/main/res/layout/layout_bottom_sheet_menu_view.xml b/app/src/main/res/layout/layout_bottom_sheet_menu_view.xml new file mode 100644 index 000000000..215702e73 --- /dev/null +++ b/app/src/main/res/layout/layout_bottom_sheet_menu_view.xml @@ -0,0 +1,28 @@ + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/layout_dialog_button_positive_negative_loading.xml b/app/src/main/res/layout/layout_dialog_button_positive_negative_loading.xml index c7983b7ab..aaebe067d 100644 --- a/app/src/main/res/layout/layout_dialog_button_positive_negative_loading.xml +++ b/app/src/main/res/layout/layout_dialog_button_positive_negative_loading.xml @@ -10,22 +10,22 @@ android:id="@+id/text_view_negative" android:layout_width="wrap_content" android:layout_height="?actionBarSize" - android:foreground="?selectableItemBackground" + android:foreground="@drawable/selection_control_ripple" android:gravity="center" android:paddingHorizontal="?attr/dialogPreferredPadding" android:text="@string/cancel" - android:textColor="@color/md_theme_primary" /> + android:textColor="?attr/colorPrimary" /> + android:textColor="?attr/colorPrimary" /> diff --git a/app/src/main/res/layout/layout_fast_rv.xml b/app/src/main/res/layout/layout_fast_rv.xml index 1eae37595..f24b54fa0 100644 --- a/app/src/main/res/layout/layout_fast_rv.xml +++ b/app/src/main/res/layout/layout_fast_rv.xml @@ -1,6 +1,5 @@ @@ -20,4 +19,6 @@ android:id="@+id/sticky_container" android:layout_width="match_parent" android:layout_height="wrap_content" /> - \ No newline at end of file + + + diff --git a/app/src/main/res/layout/toolbar_actionbar_progress_bar.xml b/app/src/main/res/layout/toolbar_actionbar_progress_bar.xml index 8432a0469..3e6b6c73d 100644 --- a/app/src/main/res/layout/toolbar_actionbar_progress_bar.xml +++ b/app/src/main/res/layout/toolbar_actionbar_progress_bar.xml @@ -19,8 +19,8 @@ android:layout_height="wrap_content" android:indeterminate="false" android:max="100" - android:maxHeight="2dp" - android:minHeight="2dp" + android:maxHeight="1dp" + android:minHeight="1dp" android:progress="10" android:progressDrawable="@drawable/seekbar_for_webview_style" /> diff --git a/app/src/main/res/layout/view_dialog_message_textview.xml b/app/src/main/res/layout/view_dialog_message_textview.xml index 6018aaed5..fe091bd06 100644 --- a/app/src/main/res/layout/view_dialog_message_textview.xml +++ b/app/src/main/res/layout/view_dialog_message_textview.xml @@ -4,9 +4,6 @@ android:id="@+id/message_view" android:layout_width="match_parent" android:layout_height="wrap_content" - android:layout_marginHorizontal="4dp" android:gravity="start" - android:textAppearance="?android:attr/textAppearanceLarge" - android:textColor="@color/grey" - android:textSize="16sp" + android:textAppearance="?android:attr/textAppearanceSmall" tools:text="@string/app_name" /> \ No newline at end of file diff --git a/app/src/main/res/menu/bottom_navigation_menu.xml b/app/src/main/res/menu/bottom_navigation_menu.xml index 223c06ab1..f4ca6c31b 100644 --- a/app/src/main/res/menu/bottom_navigation_menu.xml +++ b/app/src/main/res/menu/bottom_navigation_menu.xml @@ -8,7 +8,7 @@ \ No newline at end of file diff --git a/app/src/main/res/menu/bottom_sdoc_menu.xml b/app/src/main/res/menu/bottom_sdoc_menu.xml index b4d114084..3373e07ba 100644 --- a/app/src/main/res/menu/bottom_sdoc_menu.xml +++ b/app/src/main/res/menu/bottom_sdoc_menu.xml @@ -3,25 +3,25 @@ xmlns:app="http://schemas.android.com/apk/res-auto"> \ No newline at end of file diff --git a/app/src/main/res/menu/bottom_sheet_op_dirent.xml b/app/src/main/res/menu/bottom_sheet_op_dirent.xml new file mode 100644 index 000000000..88213bf53 --- /dev/null +++ b/app/src/main/res/menu/bottom_sheet_op_dirent.xml @@ -0,0 +1,53 @@ + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/menu/bottom_sheet_op_file.xml b/app/src/main/res/menu/bottom_sheet_op_file.xml index 81c93cc49..be0f18ce4 100644 --- a/app/src/main/res/menu/bottom_sheet_op_file.xml +++ b/app/src/main/res/menu/bottom_sheet_op_file.xml @@ -47,7 +47,7 @@ \ No newline at end of file diff --git a/app/src/main/res/menu/browser_menu.xml b/app/src/main/res/menu/browser_menu.xml index 2152704c1..73f7c5fcd 100644 --- a/app/src/main/res/menu/browser_menu.xml +++ b/app/src/main/res/menu/browser_menu.xml @@ -1,12 +1,6 @@ - - - - - - - + + + + + + + + + + + + + + + + + + + + + android:title="@string/transfer_list_select_all" + app:showAsAction="ifRoom" /> diff --git a/app/src/main/res/values-night/colors.xml b/app/src/main/res/values-night/colors.xml index f4fc885a1..1360d9ec6 100644 --- a/app/src/main/res/values-night/colors.xml +++ b/app/src/main/res/values-night/colors.xml @@ -54,17 +54,24 @@ #4A4A4A #86848B - - #2c2c2c - #2c2c2c + + @color/material_grey_919 + @color/material_grey_911 + @color/material_grey_50 + @color/material_grey_50 + @color/material_grey_400 + + @color/material_grey_700 + @color/material_grey_919 + @color/material_grey_919 + @color/material_grey_824 + + #d3d3d3 #7F7F7F #9a9a9a - #191919 - #f2f2f2 - #f2f2f2 @color/blue_500 @color/white diff --git a/app/src/main/res/values-night/styles.xml b/app/src/main/res/values-night/styles.xml index f0ddbbce5..cb9c14401 100644 --- a/app/src/main/res/values-night/styles.xml +++ b/app/src/main/res/values-night/styles.xml @@ -80,16 +80,4 @@ #ff000000 14sp - - - diff --git a/app/src/main/res/values-sw600dp/dimens.xml b/app/src/main/res/values-sw600dp/dimens.xml index 274b0f164..ef9019a5a 100644 --- a/app/src/main/res/values-sw600dp/dimens.xml +++ b/app/src/main/res/values-sw600dp/dimens.xml @@ -8,13 +8,7 @@ 25dp - - 64dp - 48dp - - 32dp - 32dp 14sp 70dp 48dp diff --git a/app/src/main/res/values/colors.xml b/app/src/main/res/values/colors.xml index 05ae1fe06..efae4f3c3 100644 --- a/app/src/main/res/values/colors.xml +++ b/app/src/main/res/values/colors.xml @@ -57,18 +57,24 @@ #4A4A4A #86848B - + + @color/material_grey_50 + @color/material_grey_88 + @color/material_grey_900 + @color/material_grey_900 + @color/material_grey_666 + + @color/material_grey_50 + @color/material_grey_500 + #E3E3E3 #F1F1F1 + #333333 - @color/material_grey_500 + @color/material_grey_599 #A0A0A0 - #f9f9f9 - #191919 - #191919 - @color/material_grey_666 @color/blue_500 @color/white diff --git a/app/src/main/res/values/colors_material.xml b/app/src/main/res/values/colors_material.xml index c4e98eb77..658be85e3 100644 --- a/app/src/main/res/values/colors_material.xml +++ b/app/src/main/res/values/colors_material.xml @@ -259,17 +259,23 @@ #3E2723 #FAFAFA + #F8F8F8 #F5F5F5 #EEEEEE #E0E0E0 #BDBDBD #9E9E9E + #999999 #757575 #666666 #616161 #424242 #212121 + #222222 + #191919 + #111111 + #ECEFF1 #CFD8DC #B0BEC5 diff --git a/app/src/main/res/values/ids.xml b/app/src/main/res/values/ids.xml index 7d1becc26..cc9727188 100644 --- a/app/src/main/res/values/ids.xml +++ b/app/src/main/res/values/ids.xml @@ -1,4 +1,16 @@ + + + + + + + + + + + + diff --git a/app/src/main/res/values/styles.xml b/app/src/main/res/values/styles.xml index 3d4f925bb..f1b3c242b 100644 --- a/app/src/main/res/values/styles.xml +++ b/app/src/main/res/values/styles.xml @@ -60,18 +60,25 @@ - -