]> git.argeo.org Git - lgpl/argeo-commons.git/blob - org.argeo.security.ui.admin/src/org/argeo/security/ui/admin/internal/parts/UserMainPage.java
Re-add org.argeo.cms.util.useradmin
[lgpl/argeo-commons.git] / org.argeo.security.ui.admin / src / org / argeo / security / ui / admin / internal / parts / UserMainPage.java
1 /*
2 * Copyright (C) 2007-2012 Argeo GmbH
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16 package org.argeo.security.ui.admin.internal.parts;
17
18 import java.util.ArrayList;
19 import java.util.Iterator;
20 import java.util.List;
21
22 import org.argeo.ArgeoException;
23 import org.argeo.cms.auth.AuthConstants;
24 import org.argeo.eclipse.ui.ColumnDefinition;
25 import org.argeo.eclipse.ui.EclipseUiUtils;
26 import org.argeo.eclipse.ui.parts.LdifUsersTable;
27 import org.argeo.jcr.ArgeoNames;
28 import org.argeo.osgi.useradmin.LdifName;
29 import org.argeo.security.ui.admin.SecurityAdminImages;
30 import org.argeo.security.ui.admin.internal.UserAdminUtils;
31 import org.argeo.security.ui.admin.internal.UserAdminWrapper;
32 import org.argeo.security.ui.admin.internal.parts.UserEditor.GroupChangeListener;
33 import org.argeo.security.ui.admin.internal.parts.UserEditor.MainInfoListener;
34 import org.argeo.security.ui.admin.internal.providers.CommonNameLP;
35 import org.argeo.security.ui.admin.internal.providers.DomainNameLP;
36 import org.argeo.security.ui.admin.internal.providers.RoleIconLP;
37 import org.argeo.security.ui.admin.internal.providers.UserFilter;
38 import org.argeo.security.ui.admin.internal.providers.UserNameLP;
39 import org.argeo.security.ui.admin.internal.providers.UserTableDefaultDClickListener;
40 import org.eclipse.jface.action.Action;
41 import org.eclipse.jface.action.ToolBarManager;
42 import org.eclipse.jface.resource.ImageDescriptor;
43 import org.eclipse.jface.viewers.ISelection;
44 import org.eclipse.jface.viewers.IStructuredSelection;
45 import org.eclipse.jface.viewers.TableViewer;
46 import org.eclipse.jface.viewers.Viewer;
47 import org.eclipse.jface.viewers.ViewerDropAdapter;
48 import org.eclipse.swt.SWT;
49 import org.eclipse.swt.dnd.DND;
50 import org.eclipse.swt.dnd.DropTargetEvent;
51 import org.eclipse.swt.dnd.TextTransfer;
52 import org.eclipse.swt.dnd.Transfer;
53 import org.eclipse.swt.dnd.TransferData;
54 import org.eclipse.swt.events.DisposeEvent;
55 import org.eclipse.swt.events.DisposeListener;
56 import org.eclipse.swt.events.ModifyEvent;
57 import org.eclipse.swt.events.ModifyListener;
58 import org.eclipse.swt.events.SelectionAdapter;
59 import org.eclipse.swt.events.SelectionEvent;
60 import org.eclipse.swt.graphics.Cursor;
61 import org.eclipse.swt.layout.GridData;
62 import org.eclipse.swt.layout.GridLayout;
63 import org.eclipse.swt.widgets.Button;
64 import org.eclipse.swt.widgets.Composite;
65 import org.eclipse.swt.widgets.Label;
66 import org.eclipse.swt.widgets.Text;
67 import org.eclipse.swt.widgets.ToolBar;
68 import org.eclipse.ui.forms.AbstractFormPart;
69 import org.eclipse.ui.forms.IManagedForm;
70 import org.eclipse.ui.forms.SectionPart;
71 import org.eclipse.ui.forms.editor.FormEditor;
72 import org.eclipse.ui.forms.editor.FormPage;
73 import org.eclipse.ui.forms.widgets.FormToolkit;
74 import org.eclipse.ui.forms.widgets.ScrolledForm;
75 import org.eclipse.ui.forms.widgets.Section;
76 import org.osgi.service.useradmin.Group;
77 import org.osgi.service.useradmin.Role;
78 import org.osgi.service.useradmin.User;
79 import org.osgi.service.useradmin.UserAdmin;
80 import org.osgi.service.useradmin.UserAdminEvent;
81
82 /** Display/edit the properties of a given user */
83 public class UserMainPage extends FormPage implements ArgeoNames {
84 final static String ID = "UserEditor.mainPage";
85
86 private final UserEditor editor;
87 private UserAdminWrapper userAdminWrapper;
88
89 // Local configuration
90 private final int PRE_TITLE_INDENT = 10;
91
92 public UserMainPage(FormEditor editor, UserAdminWrapper userAdminWrapper) {
93 super(editor, ID, "Main");
94 this.editor = (UserEditor) editor;
95 this.userAdminWrapper = userAdminWrapper;
96 }
97
98 protected void createFormContent(final IManagedForm mf) {
99 ScrolledForm form = mf.getForm();
100 Composite body = form.getBody();
101 GridLayout mainLayout = new GridLayout();
102 // mainLayout.marginRight = 10;
103 body.setLayout(mainLayout);
104 User user = editor.getDisplayedUser();
105 appendOverviewPart(body, user);
106 // Remove to ability to force the password for his own user. The user
107 // must then use the change pwd feature
108 if (!UserAdminUtils.isCurrentUser(user))
109 appendPasswordPart(body, user);
110 appendMemberOfPart(body, user);
111 }
112
113 /** Creates the general section */
114 private void appendOverviewPart(final Composite parent, final User user) {
115 FormToolkit tk = getManagedForm().getToolkit();
116
117 Section section = addSection(tk, parent, "Main information");
118 Composite body = (Composite) section.getClient();
119 body.setLayout(new GridLayout(2, false));
120
121 final Text distinguishedName = createLT(tk, body, "User Name",
122 UserAdminUtils.getProperty(user, LdifName.uid.name()));
123 distinguishedName.setEnabled(false);
124
125 final Text commonName = createLT(tk, body, "Common Name",
126 UserAdminUtils.getProperty(user, LdifName.cn.name()));
127 commonName.setEnabled(false);
128
129 final Text firstName = createLT(tk, body, "First name",
130 UserAdminUtils.getProperty(user, LdifName.givenName.name()));
131
132 final Text lastName = createLT(tk, body, "Last name",
133 UserAdminUtils.getProperty(user, LdifName.sn.name()));
134
135 final Text email = createLT(tk, body, "Email",
136 UserAdminUtils.getProperty(user, LdifName.mail.name()));
137
138 // create form part (controller)
139 AbstractFormPart part = new SectionPart((Section) body.getParent()) {
140 private MainInfoListener listener;
141
142 @Override
143 public void initialize(IManagedForm form) {
144 super.initialize(form);
145 listener = editor.new MainInfoListener(parent.getDisplay(),
146 this);
147 userAdminWrapper.addListener(listener);
148 }
149
150 @Override
151 public void dispose() {
152 userAdminWrapper.removeListener(listener);
153 super.dispose();
154 }
155
156 @SuppressWarnings("unchecked")
157 public void commit(boolean onSave) {
158 // TODO Sanity checks (mail validity...)
159 user.getProperties().put(LdifName.givenName.name(),
160 firstName.getText());
161 user.getProperties()
162 .put(LdifName.sn.name(), lastName.getText());
163 user.getProperties().put(LdifName.cn.name(),
164 commonName.getText());
165 user.getProperties().put(LdifName.mail.name(), email.getText());
166 super.commit(onSave);
167 }
168
169 @Override
170 public void refresh() {
171 distinguishedName.setText(UserAdminUtils.getProperty(user,
172 LdifName.uid.name()));
173 commonName.setText(UserAdminUtils.getProperty(user,
174 LdifName.cn.name()));
175 firstName.setText(UserAdminUtils.getProperty(user,
176 LdifName.givenName.name()));
177 lastName.setText(UserAdminUtils.getProperty(user,
178 LdifName.sn.name()));
179 email.setText(UserAdminUtils.getProperty(user,
180 LdifName.mail.name()));
181 refreshFormTitle(user);
182 super.refresh();
183 }
184 };
185
186 // Improve this: automatically generate CN when first or last name
187 // changes
188 ModifyListener cnML = new ModifyListener() {
189 private static final long serialVersionUID = 4298649222869835486L;
190
191 @Override
192 public void modifyText(ModifyEvent event) {
193 String first = firstName.getText();
194 String last = lastName.getText();
195 String cn = first.trim() + " " + last.trim() + " ";
196 cn = cn.trim();
197 commonName.setText(cn);
198 getManagedForm().getForm().setText(cn);
199 editor.updateEditorTitle(cn);
200 }
201 };
202 firstName.addModifyListener(cnML);
203 lastName.addModifyListener(cnML);
204
205 ModifyListener defaultListener = editor.new FormPartML(part);
206 firstName.addModifyListener(defaultListener);
207 lastName.addModifyListener(defaultListener);
208 email.addModifyListener(defaultListener);
209 getManagedForm().addPart(part);
210 }
211
212 /** Creates the password section */
213 private void appendPasswordPart(Composite parent, final User user) {
214 FormToolkit tk = getManagedForm().getToolkit();
215 Section section = addSection(tk, parent, "Password");
216 Composite body = (Composite) section.getClient();
217 body.setLayout(new GridLayout(2, false));
218
219 // add widgets (view)
220 final Text password1 = createLP(tk, body, "New password", "");
221 final Text password2 = createLP(tk, body, "Repeat password", "");
222
223 // create form part (controller)
224 AbstractFormPart part = new SectionPart((Section) body.getParent()) {
225 @SuppressWarnings("unchecked")
226 public void commit(boolean onSave) {
227 if (!password1.getText().equals("")
228 || !password2.getText().equals("")) {
229 if (password1.getText().equals(password2.getText())) {
230 char[] newPassword = password1.getText().toCharArray();
231 userAdminWrapper.beginTransactionIfNeeded();
232 user.getCredentials().put(null, newPassword);
233 password1.setText("");
234 password2.setText("");
235 super.commit(onSave);
236 } else {
237 password1.setText("");
238 password2.setText("");
239 throw new ArgeoException("Passwords are not equals");
240 }
241 }
242 }
243 };
244 ModifyListener defaultListener = editor.new FormPartML(part);
245 password1.addModifyListener(defaultListener);
246 password2.addModifyListener(defaultListener);
247 getManagedForm().addPart(part);
248 }
249
250 private LdifUsersTable appendMemberOfPart(final Composite parent, User user) {
251 FormToolkit tk = getManagedForm().getToolkit();
252 Section section = addSection(tk, parent, "Roles");
253 Composite body = (Composite) section.getClient();
254 body.setLayout(EclipseUiUtils.noSpaceGridLayout());
255
256 boolean isAdmin = UserAdminUtils.isUserInRole(AuthConstants.ROLE_ADMIN);
257
258 // Displayed columns
259 List<ColumnDefinition> columnDefs = new ArrayList<ColumnDefinition>();
260 columnDefs.add(new ColumnDefinition(new RoleIconLP(), "", 0, 24));
261 columnDefs.add(new ColumnDefinition(new CommonNameLP(), "Common Name",
262 150));
263 columnDefs.add(new ColumnDefinition(new DomainNameLP(), "Domain Name",
264 200));
265 // Only show technical DN to administrators
266 if (isAdmin)
267 columnDefs.add(new ColumnDefinition(new UserNameLP(),
268 "Distinguished Name", 120));
269
270 // Create and configure the table
271 final LdifUsersTable userViewerCmp = new MyUserTableViewer(body,
272 SWT.MULTI | SWT.H_SCROLL | SWT.V_SCROLL, user);
273
274 userViewerCmp.setColumnDefinitions(columnDefs);
275 if (isAdmin)
276 userViewerCmp.populateWithStaticFilters(false, false);
277 else
278 userViewerCmp.populate(true, false);
279 GridData gd = EclipseUiUtils.fillAll();
280 gd.heightHint = 300;
281 userViewerCmp.setLayoutData(gd);
282
283 // Controllers
284 TableViewer userViewer = userViewerCmp.getTableViewer();
285 userViewer.addDoubleClickListener(new UserTableDefaultDClickListener());
286 int operations = DND.DROP_COPY | DND.DROP_MOVE;
287 Transfer[] tt = new Transfer[] { TextTransfer.getInstance() };
288 GroupDropListener dropL = new GroupDropListener(userAdminWrapper,
289 userViewer, user);
290 userViewer.addDropSupport(operations, tt, dropL);
291
292 SectionPart part = new SectionPart((Section) body.getParent()) {
293
294 private GroupChangeListener listener;
295
296 @Override
297 public void initialize(IManagedForm form) {
298 super.initialize(form);
299 listener = editor.new GroupChangeListener(parent.getDisplay(),
300 this);
301 userAdminWrapper.addListener(listener);
302 }
303
304 public void commit(boolean onSave) {
305 super.commit(onSave);
306 }
307
308 @Override
309 public void dispose() {
310 userAdminWrapper.removeListener(listener);
311 super.dispose();
312 }
313
314 @Override
315 public void refresh() {
316 userViewerCmp.refresh();
317 super.refresh();
318 }
319 };
320 getManagedForm().addPart(part);
321 addRemoveAbitily(part, userViewer, user);
322 return userViewerCmp;
323 }
324
325 private class MyUserTableViewer extends LdifUsersTable {
326 private static final long serialVersionUID = 2653790051461237329L;
327
328 private Button showSystemRoleBtn;
329
330 private final User user;
331 private final UserFilter userFilter;
332
333 public MyUserTableViewer(Composite parent, int style, User user) {
334 super(parent, style, true);
335 this.user = user;
336 userFilter = new UserFilter();
337 userFilter.setShowSystemRole(false);
338 }
339
340 protected void populateStaticFilters(Composite staticFilterCmp) {
341 staticFilterCmp.setLayout(new GridLayout());
342 showSystemRoleBtn = new Button(staticFilterCmp, SWT.CHECK);
343 showSystemRoleBtn.setText("Show system roles");
344 showSystemRoleBtn.addSelectionListener(new SelectionAdapter() {
345 private static final long serialVersionUID = -7033424592697691676L;
346
347 @Override
348 public void widgetSelected(SelectionEvent e) {
349 userFilter.setShowSystemRole(showSystemRoleBtn
350 .getSelection());
351 refresh();
352 }
353 });
354 }
355
356 @Override
357 protected List<User> listFilteredElements(String filter) {
358 List<User> users = (List<User>) editor.getFlatGroups(null);
359 List<User> filteredUsers = new ArrayList<User>();
360 if (users.contains(user))
361 users.remove(user);
362 userFilter.setSearchText(filter);
363 for (User user : users)
364 if (userFilter.select(null, null, user))
365 filteredUsers.add(user);
366 return filteredUsers;
367 }
368 }
369
370 private void addRemoveAbitily(SectionPart sectionPart,
371 TableViewer userViewer, User user) {
372 Section section = sectionPart.getSection();
373 ToolBarManager toolBarManager = new ToolBarManager(SWT.FLAT);
374 ToolBar toolbar = toolBarManager.createControl(section);
375 final Cursor handCursor = new Cursor(section.getDisplay(),
376 SWT.CURSOR_HAND);
377 toolbar.setCursor(handCursor);
378 toolbar.addDisposeListener(new DisposeListener() {
379 private static final long serialVersionUID = 3882131405820522925L;
380
381 public void widgetDisposed(DisposeEvent e) {
382 if ((handCursor != null) && (handCursor.isDisposed() == false)) {
383 handCursor.dispose();
384 }
385 }
386 });
387
388 String tooltip = "Remove " + UserAdminUtils.getUsername(user)
389 + " from the below selected groups";
390 Action action = new RemoveMembershipAction(userViewer, user, tooltip,
391 SecurityAdminImages.ICON_REMOVE_DESC);
392 toolBarManager.add(action);
393 toolBarManager.update(true);
394 section.setTextClient(toolbar);
395 }
396
397 private class RemoveMembershipAction extends Action {
398 private static final long serialVersionUID = -1337713097184522588L;
399
400 private final TableViewer userViewer;
401 private final User user;
402
403 RemoveMembershipAction(TableViewer userViewer, User user, String name,
404 ImageDescriptor img) {
405 super(name, img);
406 this.userViewer = userViewer;
407 this.user = user;
408 }
409
410 @Override
411 public void run() {
412 ISelection selection = userViewer.getSelection();
413 if (selection.isEmpty())
414 return;
415
416 @SuppressWarnings("unchecked")
417 Iterator<Group> it = ((IStructuredSelection) selection).iterator();
418 List<Group> groups = new ArrayList<Group>();
419 // StringBuilder builder = new StringBuilder();
420 while (it.hasNext()) {
421 Group currGroup = it.next();
422 // String groupName = UserAdminUtils.getUsername(currGroup);
423 // builder.append(groupName).append("; ");
424 groups.add(currGroup);
425 }
426
427 // if (!MessageDialog.openQuestion(
428 // HandlerUtil.getActiveShell(event),
429 // "Re",
430 // "Are you sure that you want to delete these users?\n"
431 // + builder.substring(0, builder.length() - 2)))
432 // return null;
433
434 userAdminWrapper.beginTransactionIfNeeded();
435 for (Group group : groups) {
436 group.removeMember(user);
437 // sectionPart.refresh();
438 userAdminWrapper.notifyListeners(new UserAdminEvent(null,
439 UserAdminEvent.ROLE_CHANGED, group));
440 }
441 }
442 }
443
444 /**
445 * Defines the table as being a potential target to add group memberships
446 * (roles) to this user
447 */
448 private class GroupDropListener extends ViewerDropAdapter {
449 private static final long serialVersionUID = 2893468717831451621L;
450
451 private final UserAdminWrapper myUserAdminWrapper;
452 private final User myUser;
453
454 public GroupDropListener(UserAdminWrapper userAdminWrapper,
455 Viewer userViewer, User user) {
456 super(userViewer);
457 this.myUserAdminWrapper = userAdminWrapper;
458 this.myUser = user;
459 }
460
461 @Override
462 public boolean validateDrop(Object target, int operation,
463 TransferData transferType) {
464 // Target is always OK in a list only view
465 // TODO check if not a string
466 boolean validDrop = true;
467 return validDrop;
468 }
469
470 @Override
471 public void drop(DropTargetEvent event) {
472 String name = (String) event.data;
473 UserAdmin myUserAdmin = myUserAdminWrapper.getUserAdmin();
474 Role role = myUserAdmin.getRole(name);
475 // TODO this check should be done before.
476 if (role.getType() == Role.GROUP) {
477 // TODO check if the user is already member of this group
478 myUserAdminWrapper.beginTransactionIfNeeded();
479 Group group = (Group) role;
480 group.addMember(myUser);
481 myUserAdminWrapper.notifyListeners(new UserAdminEvent(null,
482 UserAdminEvent.ROLE_CHANGED, group));
483 }
484 super.drop(event);
485 }
486
487 @Override
488 public boolean performDrop(Object data) {
489 // userTableViewerCmp.refresh();
490 return true;
491 }
492 }
493
494 // LOCAL HELPERS
495 private void refreshFormTitle(User group) {
496 getManagedForm().getForm().setText(
497 UserAdminUtils.getProperty(group, LdifName.cn.name()));
498 }
499
500 /** Appends a section with a title */
501 private Section addSection(FormToolkit tk, Composite parent, String title) {
502 Section section = tk.createSection(parent, Section.TITLE_BAR);
503 GridData gd = EclipseUiUtils.fillWidth();
504 gd.verticalAlignment = PRE_TITLE_INDENT;
505 section.setLayoutData(gd);
506 section.setText(title);
507 // section.getMenu().setVisible(true);
508
509 Composite body = tk.createComposite(section, SWT.WRAP);
510 body.setLayoutData(EclipseUiUtils.fillAll());
511 section.setClient(body);
512
513 return section;
514 }
515
516 /** Creates label and multiline text. */
517 Text createLMT(FormToolkit toolkit, Composite body, String label,
518 String value) {
519 Label lbl = toolkit.createLabel(body, label);
520 lbl.setLayoutData(new GridData(SWT.RIGHT, SWT.CENTER, false, false));
521 Text text = toolkit.createText(body, value, SWT.BORDER | SWT.MULTI);
522 text.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, true));
523 return text;
524 }
525
526 /** Creates label and password. */
527 Text createLP(FormToolkit toolkit, Composite body, String label,
528 String value) {
529 Label lbl = toolkit.createLabel(body, label);
530 lbl.setLayoutData(new GridData(SWT.RIGHT, SWT.CENTER, false, false));
531 Text text = toolkit.createText(body, value, SWT.BORDER | SWT.PASSWORD);
532 text.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false));
533 return text;
534 }
535
536 /** Creates label and text. */
537 Text createLT(FormToolkit toolkit, Composite body, String label,
538 String value) {
539 Label lbl = toolkit.createLabel(body, label);
540 lbl.setLayoutData(new GridData(SWT.RIGHT, SWT.CENTER, false, false));
541 Text text = toolkit.createText(body, value, SWT.BORDER);
542 text.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false));
543 return text;
544 }
545 }