EditShareFragment.java 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366
  1. /**
  2. * ownCloud Android client application
  3. *
  4. * @author masensio
  5. * @author David A. Velasco
  6. * Copyright (C) 2015 ownCloud Inc.
  7. *
  8. * This program is free software: you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License version 2,
  10. * as published by the Free Software Foundation.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  19. *
  20. */
  21. package com.owncloud.android.ui.fragment;
  22. import android.accounts.Account;
  23. import android.os.Bundle;
  24. import android.support.v4.app.Fragment;
  25. import android.view.LayoutInflater;
  26. import android.view.View;
  27. import android.view.ViewGroup;
  28. import android.widget.Button;
  29. import android.widget.CheckBox;
  30. import android.widget.CompoundButton;
  31. import android.widget.TextView;
  32. import android.widget.Toast;
  33. import com.owncloud.android.R;
  34. import com.owncloud.android.authentication.AccountUtils;
  35. import com.owncloud.android.datamodel.OCFile;
  36. import com.owncloud.android.lib.common.utils.Log_OC;
  37. import com.owncloud.android.lib.resources.shares.OCShare;
  38. import com.owncloud.android.lib.resources.shares.ShareType;
  39. import com.owncloud.android.ui.activity.FileActivity;
  40. public class EditShareFragment extends Fragment {
  41. private static final String TAG = EditShareFragment.class.getSimpleName();
  42. /** The fragment initialization parameters */
  43. private static final String ARG_SHARE = "SHARE";
  44. private static final String ARG_FILE = "FILE";
  45. private static final String ARG_ACCOUNT = "ACCOUNT";
  46. /** Ids of CheckBoxes depending on R.id.canEdit CheckBox */
  47. private static final int sSubordinateCheckBoxIds[] = {
  48. R.id.canEditCreateCheckBox,
  49. R.id.canEditChangeCheckBox,
  50. R.id.canEditDeleteCheckBox
  51. };
  52. /** Share to show & edit, received as a parameter in construction time */
  53. private OCShare mShare;
  54. /** File bound to mShare, received as a parameter in construction time */
  55. private OCFile mFile;
  56. /** OC account holding the shared file, received as a parameter in construction time */
  57. private Account mAccount;
  58. /** Listener for changes on privilege checkboxes */
  59. private CompoundButton.OnCheckedChangeListener mOnPrivilegeChangeListener;
  60. /**
  61. * Public factory method to create new EditShareFragment instances.
  62. *
  63. * @param shareToEdit An {@link OCShare} to show and edit in the fragment
  64. * @param sharedFile The {@link OCFile} bound to 'shareToEdit'
  65. * @param account The ownCloud account holding 'sharedFile'
  66. * @return A new instance of fragment EditShareFragment.
  67. */
  68. public static EditShareFragment newInstance(OCShare shareToEdit, OCFile sharedFile, Account account) {
  69. EditShareFragment fragment = new EditShareFragment();
  70. Bundle args = new Bundle();
  71. args.putParcelable(ARG_SHARE, shareToEdit);
  72. args.putParcelable(ARG_FILE, sharedFile);
  73. args.putParcelable(ARG_ACCOUNT, account);
  74. fragment.setArguments(args);
  75. return fragment;
  76. }
  77. /**
  78. * Required empty public constructor
  79. */
  80. public EditShareFragment() {
  81. }
  82. /**
  83. * {@inheritDoc}
  84. */
  85. @Override
  86. public void onCreate(Bundle savedInstanceState) {
  87. super.onCreate(savedInstanceState);
  88. Log_OC.d(TAG, "onCreate");
  89. if (getArguments() != null) {
  90. mShare = getArguments().getParcelable(ARG_SHARE);
  91. mFile = getArguments().getParcelable(ARG_FILE);
  92. mAccount = getArguments().getParcelable(ARG_ACCOUNT);
  93. }
  94. }
  95. /**
  96. * {@inheritDoc}
  97. */
  98. @Override
  99. public View onCreateView(LayoutInflater inflater, ViewGroup container,
  100. Bundle savedInstanceState) {
  101. Log_OC.d(TAG, "onCreateView");
  102. // Inflate the layout for this fragment
  103. View view = inflater.inflate(R.layout.edit_share_layout, container, false);
  104. // Setup layout
  105. initPrivileges(view);
  106. initUnshareButton(view);
  107. initDoneButton(view);
  108. return view;
  109. }
  110. /**
  111. * Binds listener for user actions to enable or disable a privilege on the edited share
  112. * to the views receiving the user events.
  113. *
  114. * @param editShareView Root view in the fragment.
  115. */
  116. private void initPrivileges(View editShareView) {
  117. mOnPrivilegeChangeListener = new OnPrivilegeChangeListener();
  118. int sharePermissions = mShare.getPermissions();
  119. CheckBox checkBox;
  120. checkBox = (CheckBox) editShareView.findViewById(R.id.canShareCheckBox);
  121. checkBox.setChecked((sharePermissions & OCShare.SHARE_PERMISSION_FLAG) > 0);
  122. checkBox.setOnCheckedChangeListener(mOnPrivilegeChangeListener);
  123. checkBox = (CheckBox) editShareView.findViewById(R.id.canEditCheckBox);
  124. int anyUpdatePermission =
  125. OCShare.CREATE_PERMISSION_FLAG |
  126. OCShare.UPDATE_PERMISSION_FLAG |
  127. OCShare.DELETE_PERMISSION_FLAG
  128. ;
  129. boolean canEdit = (sharePermissions & anyUpdatePermission) > 0;
  130. checkBox.setChecked(canEdit);
  131. checkBox.setOnCheckedChangeListener(mOnPrivilegeChangeListener);
  132. if (mFile.isFolder()) {
  133. checkBox = (CheckBox) editShareView.findViewById(R.id.canEditCreateCheckBox);
  134. if (canEdit) {
  135. checkBox.setVisibility(View.VISIBLE);
  136. checkBox.setChecked((sharePermissions & OCShare.CREATE_PERMISSION_FLAG) > 0);
  137. }
  138. checkBox.setOnCheckedChangeListener(mOnPrivilegeChangeListener);
  139. checkBox = (CheckBox) editShareView.findViewById(R.id.canEditChangeCheckBox);
  140. if (canEdit) {
  141. checkBox.setVisibility(View.VISIBLE);
  142. checkBox.setChecked((sharePermissions & OCShare.UPDATE_PERMISSION_FLAG) > 0);
  143. }
  144. checkBox.setOnCheckedChangeListener(mOnPrivilegeChangeListener);
  145. checkBox = (CheckBox) editShareView.findViewById(R.id.canEditDeleteCheckBox);
  146. if (canEdit) {
  147. checkBox.setVisibility(View.VISIBLE);
  148. checkBox.setChecked((sharePermissions & OCShare.DELETE_PERMISSION_FLAG) > 0);
  149. }
  150. checkBox.setOnCheckedChangeListener(mOnPrivilegeChangeListener);
  151. } // else, trust in visibility GONE in R.layout.edit_share_layout
  152. }
  153. /**
  154. * Listener for user actions that enable or disable a privilege
  155. */
  156. private class OnPrivilegeChangeListener
  157. implements CompoundButton.OnCheckedChangeListener {
  158. /**
  159. * Called by R.id.shareViaLinkSectionSwitch to create or delete a public link.
  160. *
  161. * @param checkBoxView {@link CheckBox} toggled by the user
  162. * @param isChecked New switch state.
  163. */
  164. @Override
  165. public void onCheckedChanged(CompoundButton checkBoxView, boolean isChecked) {
  166. if (!isResumed()) {
  167. // very important, setCheched(...) is called automatically during
  168. // Fragment recreation on device rotations
  169. return;
  170. }
  171. /// else, getView() cannot be NULL
  172. CheckBox subordinate;
  173. switch(checkBoxView.getId()) {
  174. case R.id.canShareCheckBox:
  175. Log_OC.v(TAG, "canShareCheckBox toggled to " + isChecked);
  176. /// TODO?
  177. // option 1: direct update approach
  178. /*
  179. ((FileActivity) getActivity()).getFileOperationsHelper().
  180. setPrivilegeToShare(mFile, mShare, privilege, isChecked);
  181. */
  182. // option 2: nothing?
  183. break;
  184. case R.id.canEditCheckBox:
  185. Log_OC.v(TAG, "canEditCheckBox toggled to " + isChecked);
  186. /// sync subordinate CheckBoxes
  187. if (mFile.isFolder()) {
  188. if (isChecked) {
  189. for (int i = 0; i < sSubordinateCheckBoxIds.length; i++) {
  190. //noinspection ConstantConditions, prevented in the method beginning
  191. subordinate = (CheckBox) getView().findViewById(sSubordinateCheckBoxIds[i]);
  192. subordinate.setVisibility(View.VISIBLE);
  193. if (!subordinate.isChecked()) {
  194. toggleDisablingListener(subordinate);
  195. }
  196. }
  197. } else {
  198. for (int i = 0; i < sSubordinateCheckBoxIds.length; i++) {
  199. //noinspection ConstantConditions, prevented in the method beginning
  200. subordinate = (CheckBox) getView().findViewById(sSubordinateCheckBoxIds[i]);
  201. subordinate.setVisibility(View.GONE);
  202. if (subordinate.isChecked()) {
  203. toggleDisablingListener(subordinate);
  204. }
  205. }
  206. }
  207. }
  208. /// TODO - anything else?; only if modification-on-change approach is taken
  209. break;
  210. case R.id.canEditCreateCheckBox:
  211. Log_OC.v(TAG, "canEditCreateCheckBox toggled to " + isChecked);
  212. syncCanEditCheckBox(checkBoxView, isChecked);
  213. /// TODO - anything else?; only if modification-on-change approach is taken
  214. break;
  215. case R.id.canEditChangeCheckBox:
  216. Log_OC.v(TAG, "canEditChangeCheckBox toggled to " + isChecked);
  217. syncCanEditCheckBox(checkBoxView, isChecked);
  218. /// TODO - anything else?; only if modification-on-change approach is taken
  219. break;
  220. case R.id.canEditDeleteCheckBox:
  221. Log_OC.v(TAG, "canEditDeleteCheckBox toggled to " + isChecked);
  222. syncCanEditCheckBox(checkBoxView, isChecked);
  223. /// TODO - anything else?; only if modification-on-change approach is taken
  224. break;
  225. }
  226. // undo the toggle to grant the view will be correct if any intermediate dialog is cancelled or
  227. // the create/delete operation fails
  228. // ONLY if direct update approach is followed
  229. /*
  230. checkBoxView.setOnCheckedChangeListener(null);
  231. checkBoxView.toggle();
  232. checkBoxView.setOnCheckedChangeListener(mOnShareViaLinkSwitchCheckedChangeListener);
  233. */
  234. }
  235. /**
  236. * Sync value of "can edit" CheckBox according to a change in one of its subordinate checkboxes.
  237. *
  238. * If all the subordinates are disabled, "can edit" has to be disabled.
  239. *
  240. * If any subordinate is enabled, "can edit" has to be enabled.
  241. *
  242. * @param subordinateCheckBoxView Subordinate CheckBox that was changed.
  243. * @param isChecked 'true' iif subordinateCheckBoxView was checked.
  244. */
  245. private void syncCanEditCheckBox(View subordinateCheckBoxView, boolean isChecked) {
  246. CheckBox canEditCheckBox = (CheckBox) getView().findViewById(R.id.canEditCheckBox);
  247. if (isChecked) {
  248. if (!canEditCheckBox.isChecked()) {
  249. toggleDisablingListener(canEditCheckBox);
  250. }
  251. } else {
  252. boolean allDisabled = true;
  253. for (int i=0; allDisabled && i<sSubordinateCheckBoxIds.length; i++) {
  254. allDisabled &=
  255. sSubordinateCheckBoxIds[i] == subordinateCheckBoxView.getId() ||
  256. !((CheckBox) getView().findViewById(sSubordinateCheckBoxIds[i])).isChecked()
  257. ;
  258. }
  259. if (canEditCheckBox.isChecked() && allDisabled) {
  260. toggleDisablingListener(canEditCheckBox);
  261. for (int i=0; i<sSubordinateCheckBoxIds.length; i++) {
  262. getView().findViewById(sSubordinateCheckBoxIds[i]).setVisibility(View.GONE);
  263. }
  264. }
  265. }
  266. }
  267. /**
  268. * Toggle value of received CheckBox granting that its change listener is not called.
  269. *
  270. * @param checkBoxView CheckBox to toggle without reporting to the change listener
  271. */
  272. private void toggleDisablingListener(CheckBox checkBoxView) {
  273. checkBoxView.setOnCheckedChangeListener(null);
  274. checkBoxView.toggle();
  275. checkBoxView.setOnCheckedChangeListener(mOnPrivilegeChangeListener);
  276. }
  277. }
  278. /**
  279. * Binds listener for user interactions on the 'unshare' button with the button itself.
  280. *
  281. * @param editShareView Root view in the fragment.
  282. */
  283. private void initUnshareButton(View editShareView) {
  284. TextView unshareButton = (TextView) editShareView.findViewById(R.id.unshareButton);
  285. unshareButton.setOnClickListener(new View.OnClickListener() {
  286. @Override
  287. public void onClick(View v) {
  288. ((FileActivity) getActivity()).getFileOperationsHelper().
  289. unshareFileWithUserOrGroup(
  290. mFile,
  291. mShare.getShareType(),
  292. mShare.getShareWith()
  293. )
  294. ;
  295. }
  296. });
  297. }
  298. /**
  299. * Binds listener for user interactions on the 'done' button with the button itself.
  300. *
  301. * @param editShareView Root view in the fragment.
  302. */
  303. private void initDoneButton(View editShareView) {
  304. TextView doneButton = (TextView) editShareView.findViewById(R.id.doneButton);
  305. doneButton.setOnClickListener(new View.OnClickListener() {
  306. @Override
  307. public void onClick(View v) {
  308. Toast.makeText(getActivity(), "TODO - update permissions in server", Toast.LENGTH_LONG).show();
  309. }
  310. });
  311. }
  312. @Override
  313. public void onActivityCreated(Bundle savedInstanceState) {
  314. super.onActivityCreated(savedInstanceState);
  315. Log_OC.d(TAG, "onActivityCreated");
  316. getActivity().setTitle(mShare.getSharedWithDisplayName());
  317. }
  318. }