Commit 4d3d4506 authored by Guus der Kinderen's avatar Guus der Kinderen Committed by akrherz

OF-1347: Improve Admin Console Group Edit page (#822)

This commit adds some UX improvements to the 'edit group' page on the admin console.
parent 0770fa60
...@@ -978,16 +978,20 @@ group.delete.delete=Delete Group ...@@ -978,16 +978,20 @@ group.delete.delete=Delete Group
group.edit.title=Edit Group group.edit.title=Edit Group
group.edit.form_info=Edit group settings and add or remove group members and administrators using the \ group.edit.form_info=Edit group settings and add or remove group members and administrators using the \
forms below. forms below.
group.edit.error=Error updating group settings. Please check your error logs.
group.edit.details_info=Edit group settings below. group.edit.details_info=Edit group settings below.
group.edit.update=Group information updated successfully. group.edit.update=Group information updated successfully.
group.edit.update_add_user=User(s) added successfully. group.edit.update_add_user=User(s) added successfully.
group.edit.update_del_user=User(s) deleted successfully. group.edit.update_del_user=User(s) deleted successfully.
group.edit.update_user=User(s) updated successfully. group.edit.update_user=User(s) updated successfully.
group.edit.invalid_username=Please provide a valid username.
group.edit.update_success=Group created successfully. group.edit.update_success=Group created successfully.
group.edit.not_update=User(s) not added successfully. group.edit.not_update=User(s) not added successfully.
group.edit.share_title=Contact List (Roster) Sharing group.edit.share_title=Contact List (Roster) Sharing
group.edit.share_status_disabled=Contact list group sharing is disabled group.edit.share_status_disabled=Contact list group sharing is disabled
group.edit.share_status_enabled=Contact list group sharing is enabled group.edit.share_status_enabled=Contact list group sharing is enabled
group.edit.share_with=Share group with:
group.edit.share_group_only=Users of the same group
group.edit.share_content=You can use the form below to automatically add this group to users' contact \ group.edit.share_content=You can use the form below to automatically add this group to users' contact \
lists. <strong>When enabled</strong>, <em>this group will only appear in the contact lists of the \ lists. <strong>When enabled</strong>, <em>this group will only appear in the contact lists of the \
group&#39;s members.</em>. However, you can share this group with all users or members of other groups. group&#39;s members.</em>. However, you can share this group with all users or members of other groups.
......
...@@ -17,70 +17,70 @@ ...@@ -17,70 +17,70 @@
- limitations under the License. - limitations under the License.
--%> --%>
<%@ page import="org.jivesoftware.openfire.PresenceManager,
org.jivesoftware.openfire.group.Group,
org.jivesoftware.openfire.group.GroupManager,
org.jivesoftware.openfire.security.SecurityAuditManager,
org.jivesoftware.openfire.user.User,
org.jivesoftware.openfire.user.UserManager,
org.jivesoftware.openfire.user.UserNotFoundException"
%>
<%@ page import="gnu.inet.encoding.Stringprep"%> <%@ page import="gnu.inet.encoding.Stringprep"%>
<%@ page import="org.jivesoftware.util.LocaleUtils"%> <%@ page import="org.jivesoftware.openfire.group.Group"%>
<%@ page import="org.jivesoftware.openfire.group.GroupManager"%>
<%@ page import="org.jivesoftware.openfire.group.GroupNotFoundException"%>
<%@ page import="org.jivesoftware.openfire.security.SecurityAuditManager"%>
<%@ page import="org.jivesoftware.openfire.user.UserManager"%>
<%@ page import="org.jivesoftware.openfire.user.UserNotFoundException"%>
<%@ page import="org.jivesoftware.util.CookieUtils"%>
<%@ page import="org.jivesoftware.util.Log"%> <%@ page import="org.jivesoftware.util.Log"%>
<%@ page import="org.jivesoftware.util.ParamUtils"%> <%@ page import="org.jivesoftware.util.ParamUtils"%>
<%@ page import="org.jivesoftware.util.StringUtils"%> <%@ page import="org.jivesoftware.util.StringUtils"%>
<%@ page import="org.jivesoftware.util.CookieUtils"%>
<%@ page import="org.xmpp.packet.JID"%> <%@ page import="org.xmpp.packet.JID"%>
<%@ page import="org.xmpp.packet.Presence"%>
<%@ page import="java.io.UnsupportedEncodingException"%> <%@ page import="java.io.UnsupportedEncodingException"%>
<%@ page import="java.net.URLDecoder"%> <%@ page import="java.net.URLDecoder" %>
<%@ page import="java.net.URLEncoder"%> <%@ page import="java.net.URLEncoder" %>
<%@ page import="java.util.*" %> <%@ page import="java.util.*" %>
<%@ taglib uri="admin" prefix="admin" %>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%> <%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/fmt" prefix="fmt" %> <%@ taglib uri="http://java.sun.com/jsp/jstl/fmt" prefix="fmt" %>
<%@ taglib uri="http://java.sun.com/jsp/jstl/functions" prefix="fn" %>
<!-- Define Administration Bean --> <!-- Define Administration Bean -->
<jsp:useBean id="webManager" class="org.jivesoftware.util.WebManager"/> <jsp:useBean id="webManager" class="org.jivesoftware.util.WebManager"/>
<% webManager.init(pageContext); %> <% webManager.init(pageContext); %>
<% // Get parameters <% // Get parameters
boolean add = request.getParameter("add") != null; boolean updateDetails = request.getParameter( "updateDetails" ) != null;
boolean delete = request.getParameter("remove") != null; boolean addMember = request.getParameter( "addMember") != null;
boolean removeMember = request.getParameter( "removeMember") != null;
boolean updateMember = request.getParameter("updateMember") != null; boolean updateMember = request.getParameter("updateMember") != null;
boolean update = request.getParameter("save") != null; boolean updateContactListSettings = request.getParameter("updateContactListSettings") != null;
boolean cancel = request.getParameter("cancel") != null; boolean cancel = request.getParameter("cancel") != null;
String name = ParamUtils.getParameter( request, "name" ); // update for group name
String description = ParamUtils.getParameter( request, "description" ); // update for group description.
String username = ParamUtils.getParameter(request, "username"); String username = ParamUtils.getParameter(request, "username");
String [] adminIDs = ParamUtils.getParameters(request, "admin"); String [] adminJIDs = ParamUtils.getParameters(request, "admin");
String [] deleteMembers = ParamUtils.getParameters(request, "delete"); String [] deleteMembers = ParamUtils.getParameters(request, "delete");
String groupName = ParamUtils.getParameter(request, "group"); String groupName = ParamUtils.getParameter(request, "group");
GroupManager groupManager = webManager.getGroupManager(); GroupManager groupManager = webManager.getGroupManager();
boolean groupInfoChanged = ParamUtils.getBooleanParameter(request, "groupChanged", false); boolean groupInfoChanged = ParamUtils.getBooleanParameter(request, "groupChanged", false);
Map<String,String> errors = new HashMap<String,String>(); Map<String,String> errors = new HashMap<>();
// Get the presence manager
PresenceManager presenceManager = webManager.getPresenceManager();
UserManager userManager = webManager.getUserManager();
boolean enableRosterGroups = ParamUtils.getBooleanParameter(request,"enableRosterGroups"); boolean enableRosterGroups = ParamUtils.getBooleanParameter(request,"enableRosterGroups");
boolean shareAdditional = ParamUtils.getParameter(request, "shareContactList") != null;
String groupDisplayName = ParamUtils.getParameter(request,"groupDisplayName"); String groupDisplayName = ParamUtils.getParameter(request,"groupDisplayName");
String showGroup = ParamUtils.getParameter(request,"showGroup"); String showGroup = ParamUtils.getParameter(request,"showGroup");
String[] groupNames = ParamUtils.getParameters(request, "groupNames"); List<String> groupNames = Arrays.asList(ParamUtils.getParameters(request, "groupNames"));
Group group = groupManager.getGroup(groupName); Group group = groupManager.getGroup(groupName);
boolean success; boolean success;
StringBuffer errorBuf = new StringBuffer();
Cookie csrfCookie = CookieUtils.getCookie(request, "csrf"); Cookie csrfCookie = CookieUtils.getCookie(request, "csrf");
String csrfParam = ParamUtils.getParameter(request, "csrf"); String csrfParam = ParamUtils.getParameter(request, "csrf");
if (add || delete || updateMember || update) { if ( addMember || removeMember || updateMember || updateContactListSettings || updateDetails ) {
if (csrfCookie == null || csrfParam == null || !csrfCookie.getValue().equals(csrfParam)) { if (csrfCookie == null || csrfParam == null || !csrfCookie.getValue().equals(csrfParam)) {
add = false; addMember = false;
delete = false; removeMember = false;
update = false; updateContactListSettings = false;
updateMember = false; updateMember = false;
updateDetails = false;
errors.put("csrf", "CSRF Failure!"); errors.put("csrf", "CSRF Failure!");
} }
} }
...@@ -93,33 +93,89 @@ ...@@ -93,33 +93,89 @@
return; return;
} }
if (update) { // Handle a request to update the group details (name, description).
if (enableRosterGroups && (groupDisplayName == null || groupDisplayName.trim().length() == 0)) { if ( updateDetails )
errors.put("groupDisplayName", ""); {
// Validate
if ( name == null || name.trim().isEmpty() )
{
errors.put( "name", "Name cannot be null or empty." );
} }
if (errors.isEmpty()) { else if ( !groupName.equalsIgnoreCase( name.trim() ) )
if (enableRosterGroups) { {
if (showGroup == null || !shareAdditional) { try
showGroup = "onlyGroup"; {
webManager.getGroupManager().getGroup( name.trim(), true );
errors.put( "alreadyExists", name.trim() );
} }
if ("spefgroups".equals(showGroup)) { catch ( GroupNotFoundException e)
showGroup = "onlyGroup"; {
// intended.
}
}
if ( errors.isEmpty() )
{
try
{
webManager.getGroupManager().getGroup( groupName );
group.setName( name.trim() );
group.setDescription( description != null ? description.trim() : null );
if ( !SecurityAuditManager.getSecurityAuditProvider().blockGroupEvents() )
{
// Log the event
webManager.logEvent( "edited group " + groupName, "name = " + group.getName() + ",description = " + group.getDescription() );
}
// Successful, so redirect
response.sendRedirect( "group-edit.jsp?groupChanged=true&group=" + URLEncoder.encode( group.getName(), "UTF-8" ) );
return;
}
catch ( Exception e )
{
errors.put( "general", e.getMessage() );
} }
else {
groupNames = new String[] {};
} }
group.getProperties().put("sharedRoster.showInRoster", showGroup);
if (groupDisplayName != null) {
group.getProperties().put("sharedRoster.displayName", groupDisplayName);
} }
group.getProperties().put("sharedRoster.groupList", toList(groupNames, "UTF-8"));
// Handle a request to update the contact list settings.
if ( updateContactListSettings )
{
if (enableRosterGroups && (groupDisplayName == null || groupDisplayName.trim().length() == 0))
{
errors.put("groupDisplayName", "Group display name cannot be null or empty.");
}
if (errors.isEmpty())
{
if (enableRosterGroups)
{
if ( !"spefgroups".equals( showGroup ) )
{
// not spefgroups? Either 'everybody' or 'onlyGroups' - in any case, no 'groupList' should be empty.
groupNames = new ArrayList<>();
}
// The stored value for 'showInRoster' is either 'onlyGroups' or 'everybody', when sharing is enabled
// 'spefgroups' isn't actually saved. That's represented by 'onlyGroups' combined with a 'groupList'.
if ( !"everybody".equals( showGroup ) )
{
showGroup = "onlyGroup";
}
// update
group.getProperties().put( "sharedRoster.showInRoster", showGroup );
group.getProperties().put( "sharedRoster.displayName", groupDisplayName.trim() );
group.getProperties().put( "sharedRoster.groupList", toList( groupNames.toArray( new String[]{} ) ) );
if (!SecurityAuditManager.getSecurityAuditProvider().blockGroupEvents()) { if (!SecurityAuditManager.getSecurityAuditProvider().blockGroupEvents()) {
// Log the event // Log the event
webManager.logEvent("enabled roster groups for "+groupName, "showinroster = "+showGroup+"\ndisplayname = "+groupDisplayName+"\ngrouplist = "+toList(groupNames, "UTF-8")); webManager.logEvent("enabled roster groups for "+groupName, "showinroster = "+showGroup+"\ndisplayname = "+groupDisplayName+"\ngrouplist = "+toList(groupNames.toArray( new String[]{} ) ));
} }
} }
else { else
{
group.getProperties().put("sharedRoster.showInRoster", "nobody"); group.getProperties().put("sharedRoster.showInRoster", "nobody");
group.getProperties().put("sharedRoster.displayName", ""); group.getProperties().put("sharedRoster.displayName", "");
group.getProperties().put("sharedRoster.groupList", ""); group.getProperties().put("sharedRoster.groupList", "");
...@@ -134,107 +190,110 @@ ...@@ -134,107 +190,110 @@
response.sendRedirect("group-edit.jsp?group=" + URLEncoder.encode(groupName, "UTF-8") + "&groupChanged=true"); response.sendRedirect("group-edit.jsp?group=" + URLEncoder.encode(groupName, "UTF-8") + "&groupChanged=true");
return; return;
} }
else {
// Continue editing since there are some errors
updateMember = false;
} }
}
if (updateMember) { // Handle a request to update the privilege of an existing group member.
Set<JID> adminIDSet = new HashSet<JID>(); if (errors.isEmpty() && updateMember)
for (String adminID : adminIDs) { {
JID newAdmin = new JID(adminID); final Set<JID> newAdmins = new HashSet<>();
adminIDSet.add(newAdmin); for ( final String adminJID : adminJIDs )
boolean isAlreadyAdmin = group.getAdmins().contains(newAdmin); {
if (!isAlreadyAdmin) { newAdmins.add( new JID( adminJID ) );
// Add new admin
group.getAdmins().add(newAdmin);
} }
}
Collection<JID> admins = Collections.unmodifiableCollection(group.getAdmins()); // Process changes to admins. First, add users that are newly marked as 'admin'...
Set<JID> removeList = new HashSet<JID>(); for ( final JID newAdmin : newAdmins )
for (JID admin : admins) { {
if (!adminIDSet.contains(admin)) { if ( !group.getAdmins().contains( newAdmin ) )
removeList.add(admin); {
group.getAdmins().add( newAdmin );
} }
} }
for (JID member : removeList) {
group.getMembers().add(member); // ... all users that are unmarked should be moved from the 'admins' group to the 'members' group.
final Set<JID> oldAdmins = new HashSet<>( group.getAdmins() );
oldAdmins.removeAll( newAdmins ); // All left are no longer supposed to be admins.
for ( final JID oldAdmin : oldAdmins )
{
// Update privilege by explicitly adding the old admin to the member group.
group.getMembers().add(oldAdmin);
} }
if (!SecurityAuditManager.getSecurityAuditProvider().blockGroupEvents()) {
if (!SecurityAuditManager.getSecurityAuditProvider().blockGroupEvents())
{
// Log the event // Log the event
// TODO: Should log more here later
webManager.logEvent("updated group membership for "+groupName, null); webManager.logEvent("updated group membership for "+groupName, null);
} }
// Get admin list and compare it the admin posted list. // Get admin list and compare it the admin posted list.
response.sendRedirect("group-edit.jsp?group=" + URLEncoder.encode(groupName, "UTF-8") + "&updatesuccess=true"); response.sendRedirect("group-edit.jsp?group=" + URLEncoder.encode(groupName, "UTF-8") + "&updatesuccess=true");
return; return;
} }
else if (add && username != null) {
int count = 0; // Handle a request to add a new group member.
if ( errors.isEmpty() && addMember)
{
if ( username == null || username.trim().isEmpty() )
{
errors.put( "addMember", "username" );
}
else
{
boolean memberAdded = false;
username = username.trim(); username = username.trim();
username = username.toLowerCase(); username = username.toLowerCase();
if (username.indexOf('@') != -1) { if ( username.contains( "@" ) )
try { {
UserManager.getInstance().getUser(JID.escapeNode(username)); try
{
UserManager.getInstance().getUser( JID.escapeNode( username ) );
// That means that this user has an email address as their node. // That means that this user has an email address as their node.
username = JID.escapeNode(username); username = JID.escapeNode( username );
} }
catch (UserNotFoundException e) { catch (UserNotFoundException e)
{
// that's ok.
} }
} }
// Add to group as member by default. // Add to group as member by default.
try { try
boolean added; {
if (username.indexOf('@') == -1) { if ( !username.contains( "@" ) )
{
// No @ was found so assume this is a JID of a local user // No @ was found so assume this is a JID of a local user
username = JID.escapeNode(username); username = JID.escapeNode(username);
username = Stringprep.nodeprep(username); username = Stringprep.nodeprep(username);
UserManager.getInstance().getUser(username); UserManager.getInstance().getUser(username);
added = group.getMembers().add(webManager.getXMPPServer().createJID(username, null)); memberAdded = group.getMembers().add(webManager.getXMPPServer().createJID(username, null));
} }
else { else
{
// Admin entered a JID. Add the JID directly to the list of group members // Admin entered a JID. Add the JID directly to the list of group members
added = group.getMembers().add(new JID(username)); memberAdded = group.getMembers().add( new JID(username) );
if (!SecurityAuditManager.getSecurityAuditProvider().blockGroupEvents()) { if (!SecurityAuditManager.getSecurityAuditProvider().blockGroupEvents()) {
// Log the event // Log the event
webManager.logEvent("added group member to "+groupName, "username = "+username); webManager.logEvent("added group member to "+groupName, "username = "+username);
} }
} }
if (added) {
count++;
}
else {
errorBuf.append("<br>").append(
LocaleUtils.getLocalizedString("group.edit.already_user", Arrays.asList(username)));
}
} }
catch (Exception e) { catch ( Exception e )
{
errors.put( "general", e.getMessage() );
Log.warn("Problem adding new user to existing group", e); Log.warn("Problem adding new user to existing group", e);
errorBuf.append("<br>").append(
LocaleUtils.getLocalizedString("group.edit.inexistent_user", Arrays.asList(username)));
} }
if (count > 0) {
response.sendRedirect("group-edit.jsp?group=" + if ( memberAdded )
URLEncoder.encode(groupName, "UTF-8") + "&success=true"); {
response.sendRedirect("group-edit.jsp?group=" + URLEncoder.encode(groupName, "UTF-8") + "&success=true");
return; return;
} }
else {
success = false;
add = true;
}
} }
else if(add && username == null){
add = false;
} }
else if (delete) {
// Handle a request to remove group members.
if ( errors.isEmpty() && removeMember ) {
for (String deleteMember : deleteMembers) { for (String deleteMember : deleteMembers) {
JID member = new JID(deleteMember); JID member = new JID(deleteMember);
group.getMembers().remove(member); group.getMembers().remove(member);
...@@ -243,144 +302,171 @@ ...@@ -243,144 +302,171 @@
response.sendRedirect("group-edit.jsp?group=" + URLEncoder.encode(groupName, "UTF-8") + "&deletesuccess=true"); response.sendRedirect("group-edit.jsp?group=" + URLEncoder.encode(groupName, "UTF-8") + "&deletesuccess=true");
return; return;
} }
success = groupInfoChanged || "true".equals(request.getParameter("success")) || success = groupInfoChanged || "true".equals(request.getParameter("success")) ||
"true".equals(request.getParameter("deletesuccess")) || "true".equals(request.getParameter("deletesuccess")) ||
"true".equals(request.getParameter("updatesuccess")) || "true".equals(request.getParameter("updatesuccess")) ||
"true".equals(request.getParameter("creategroupsuccess")); "true".equals(request.getParameter("creategroupsuccess"));
if (errors.size() == 0) { showGroup = group.getProperties().get( "sharedRoster.showInRoster" );
showGroup = group.getProperties().get("sharedRoster.showInRoster"); if ( "onlyGroup".equals( showGroup ) )
enableRosterGroups = !"nobody".equals(showGroup); {
shareAdditional = "everybody".equals(showGroup); String glist = group.getProperties().get( "sharedRoster.groupList" );
if ("onlyGroup".equals(showGroup)) { List<String> l = new ArrayList<>();
String glist = group.getProperties().get("sharedRoster.groupList"); if ( glist != null )
List<String> l = new ArrayList<String>(); {
if (glist != null) { StringTokenizer tokenizer = new StringTokenizer( glist, ",\t\n\r\f" );
StringTokenizer tokenizer = new StringTokenizer(glist,",\t\n\r\f"); while ( tokenizer.hasMoreTokens() )
while (tokenizer.hasMoreTokens()) { {
String tok = tokenizer.nextToken().trim(); String tok = tokenizer.nextToken().trim();
l.add(tok.trim()); l.add( tok.trim() );
}
if (!l.isEmpty()) {
shareAdditional = true;
} }
} }
groupNames = l.toArray(new String[]{}); groupNames = l;
}
groupDisplayName = group.getProperties().get("sharedRoster.displayName");
} }
pageContext.setAttribute( "success", success );
pageContext.setAttribute( "errors", errors );
pageContext.setAttribute( "groupInfoChanged", groupInfoChanged );
pageContext.setAttribute( "group", group );
pageContext.setAttribute( "groupNames", groupNames );
final List<JID> allMembers = new ArrayList<>();
allMembers.addAll( group.getAdmins() );
allMembers.addAll( group.getMembers() );
Collections.sort( allMembers );
pageContext.setAttribute( "allMembers", allMembers );
%> %>
<html> <html>
<head> <head>
<title><fmt:message key="group.edit.title"/></title> <title><fmt:message key="group.edit.title"/></title>
<meta name="subPageID" content="group-edit"/> <meta name="subPageID" content="group-edit"/>
<meta name="extraParams" content="<%= "group="+URLEncoder.encode(groupName, "UTF-8") %>"/> <meta name="extraParams" content="group=${fn:escapeXml(param.group)}"/>
<meta name="helpPage" content="edit_group_properties.html"/> <meta name="helpPage" content="edit_group_properties.html"/>
</head> </head>
<body> <body>
<% if (webManager.getGroupManager().isReadOnly()) { %>
<div class="error">
<fmt:message key="group.read_only"/>
</div>
<% } %>
<p> <p>
<fmt:message key="group.edit.form_info" /> <fmt:message key="group.edit.form_info" />
</p> </p>
<p> <c:if test="${not empty errors['general']}">
<a href="group-summary.jsp" class="jive-link-back"><span>&laquo;</span> Back to all groups</a> <admin:infobox type="error">
</p> <fmt:message key="group.edit.error" />
</admin:infobox>
<% </c:if>
if (success) { <c:if test="${not empty errors['csrf']}">
%> <admin:infobox type="error">
<div class="jive-success"> <fmt:message key="global.csrf.failed" />
<table cellpadding="0" cellspacing="0" border="0"> </admin:infobox>
<tbody> </c:if>
<tr><td class="jive-icon"><img src="images/success-16x16.gif" width="16" height="16" border="0" alt=""></td>
<td class="jive-icon-label"> <c:if test="${empty errors and success}">
<% if (groupInfoChanged) { %> <admin:infobox type="success">
<c:choose>
<c:when test="${groupInfoChanged}">
<fmt:message key="group.edit.update" /> <fmt:message key="group.edit.update" />
<% } else if ("true".equals(request.getParameter("success"))) { %> </c:when>
<c:when test="${param.success}">
<fmt:message key="group.edit.update_add_user" /> <fmt:message key="group.edit.update_add_user" />
<% } else if ("true".equals(request.getParameter("deletesuccess"))) { %> </c:when>
<c:when test="${param.deletesuccess}">
<fmt:message key="group.edit.update_del_user" /> <fmt:message key="group.edit.update_del_user" />
<% } else if ("true".equals(request.getParameter("updatesuccess"))) { %> </c:when>
<c:when test="${param.updatesuccess}">
<fmt:message key="group.edit.update_user" /> <fmt:message key="group.edit.update_user" />
<% } else if ("true".equals(request.getParameter("creategroupsuccess"))) { %> </c:when>
<c:when test="${param.creategroupsuccess}">
<fmt:message key="group.edit.update_success" /> <fmt:message key="group.edit.update_success" />
<% </c:when>
} <c:otherwise>
%> <fmt:message key="group.edit.update" />
</td></tr> </c:otherwise>
</tbody> </c:choose>
</table> </admin:infobox>
</div><br> </c:if>
<%
}
else if(!success && add){
%>
<div class="jive-error">
<table cellpadding="0" cellspacing="0" border="0">
<tbody>
<tr><td class="jive-icon"><img src="images/error-16x16.gif" width="16" height="16" border="0" alt=""></td>
<td class="jive-icon-label">
<% if(add) { %>
<fmt:message key="group.edit.not_update" />
<%= StringUtils.escapeHTMLTags(errorBuf.toString()) %>
<% } %>
</td></tr>
</tbody>
</table>
</div><br>
<% } %>
<div class="jive-horizontalRule"></div> <!-- BEGIN group name and description -->
<fmt:message key="group.edit.edit_details" var="groupdetailsboxtitle"/>
<admin:contentBox title="${groupdetailsboxtitle}">
<form name="ff" action="group-edit.jsp"> <form name="groupdetails">
<input type="hidden" name="csrf" value="${csrf}"> <input type="hidden" name="csrf" value="${csrf}">
<input type="hidden" name="group" value="${fn:escapeXml(param.group)}"/>
<input type="hidden" name="group" value="<%= StringUtils.escapeForXML(groupName) %>"/> <c:if test="${webManager.groupManager.readOnly}">
<admin:infobox type="info"><fmt:message key="group.read_only"/></admin:infobox>
</c:if>
<table width="80%" cellpadding="3" cellspacing="0" border="0">
<tr valign="top">
<td width="1%" nowrap>
<label for="gname"><fmt:message key="group.create.group_name" /></label> *
</td>
<td width="99%">
<input type="text" name="name" size="75" maxlength="75" value="${fn:escapeXml(group.name)}" id="gname" ${webManager.groupManager.readOnly ? 'readonly' : ''} />
</td>
</tr>
<c:if test="${not empty errors['name'] or not empty errors['alreadyExists']}">
<tr valign="top">
<td></td>
<td>
<span class="jive-error-text">
<c:if test="${not empty errors['name']}"><fmt:message key="group.create.invalid_group_name" /></c:if>
<c:if test="${not empty errors['alreadyExists']}"><fmt:message key="group.create.invalid_group_info" /></c:if>
</span>
</td>
</tr>
</c:if>
<tr valign="top">
<td width="1%" nowrap>
<label for="gdesc"><fmt:message key="group.create.label_description" /></label>
</td>
<td width="99%">
<textarea name="description" cols="75" rows="3" id="gdesc" ${webManager.groupManager.readOnly ? 'readonly' : ''}><c:out value="${group.description}"/></textarea>
</td>
</tr>
<c:if test="${not empty errors['description']}">
<tr valign="top">
<td></td>
<td>
<span class="jive-error-text"><fmt:message key="group.create.invalid_description" /></span>
</td>
</tr>
</c:if>
<c:if test="${not webManager.groupManager.readOnly}">
<tr valign="top">
<td></td>
<td>
<input type="submit" name="updateDetails" value="<fmt:message key="global.save_settings" />">
</td>
</tr>
</c:if>
</table>
<!-- BEGIN group name and description --> <span class="jive-description">* <fmt:message key="group.create.required_fields" /> </span>
<div class="jive-contentBox-plain"> </form>
<% // Only show edit and delete options if the groups aren't read-only. </admin:contentBox>
if (!webManager.getGroupManager().isReadOnly()) { %> <!-- END group name and description -->
<div class="jive-contentBox-toolbox">
<a href="group-create.jsp?group=<%= URLEncoder.encode(group.getName(), "UTF-8")%>&name=<%= URLEncoder.encode(group.getName(), "UTF-8")%>&description=<%= group.getDescription() != null? URLEncoder.encode(group.getDescription(), "UTF-8") : "" %>" class="jive-link-edit"><fmt:message key="group.edit.edit_details" /></a>
<a href="group-delete.jsp?group=<%= URLEncoder.encode(group.getName(), "UTF-8")%>" class="jive-link-delete"><fmt:message key="group.edit.delete" /></a>
</div>
<% } %>
<h3>
<%= StringUtils.escapeHTMLTags(group.getName()) %>
</h3>
<p>
<%= group.getDescription() != null ? StringUtils.escapeHTMLTags(group.getDescription()) : "" %>
</p>
</div>
<!-- END group name and description -->
<!-- BEGIN contact list settings -->
<fmt:message key="group.edit.share_title" var="contactlistsettingsboxtitle"/>
<admin:contentBox title="${contactlistsettingsboxtitle}">
<!-- BEGIN contact list settings --> <form name="groupshare">
<div class="jive-contentBoxHeader"> <input type="hidden" name="csrf" value="${csrf}">
<fmt:message key="group.edit.share_title" /> <input type="hidden" name="group" value="${fn:escapeXml(param.group)}"/>
</div>
<div class="jive-contentBox">
<p> <p>
<fmt:message key="group.edit.share_content" /> <fmt:message key="group.edit.share_content" />
</p> </p>
<table cellpadding="3" cellspacing="0" border="0"> <table width="80%" cellpadding="3" cellspacing="0" border="0">
<tbody>
<tr> <tr>
<td width="1%"> <td width="1%">
<input type="radio" name="enableRosterGroups" value="false" id="rb201" <%= !enableRosterGroups ? "checked" : "" %> onClick="document.getElementById('jive-roster').style.display = 'none';"> <input type="radio" name="enableRosterGroups" value="false" id="rb201" ${group.properties['sharedRoster.showInRoster'] eq 'nobody' ? "checked" : ""} onClick="toggleReadOnly();">
</td> </td>
<td width="99%"> <td width="99%">
<label for="rb201"><fmt:message key="group.edit.share_not_in_rosters" /></label> <label for="rb201"><fmt:message key="group.edit.share_not_in_rosters" /></label>
...@@ -388,117 +474,95 @@ ...@@ -388,117 +474,95 @@
</tr> </tr>
<tr> <tr>
<td width="1%" valign="top"> <td width="1%" valign="top">
<input type="radio" name="enableRosterGroups" value="true" id="rb202" <%= enableRosterGroups ? "checked" : "" %> onClick="document.getElementById('jive-roster').style.display = 'block';"> <input type="radio" name="enableRosterGroups" value="true" id="rb202" ${group.properties['sharedRoster.showInRoster'] eq 'nobody' ? "" : "checked"} onClick="toggleReadOnly();"">
</td> </td>
<td width="99%"> <td width="99%">
<label for="rb202"><fmt:message key="group.edit.share_in_rosters" /></label> <label for="rb202"><fmt:message key="group.edit.share_in_rosters" /></label>
<div id="jive-roster" style="display: <%= !enableRosterGroups ? "none" : "block" %>;"> <div id="jive-roster">
<b><fmt:message key="group.edit.share_display_name" /></b> <b><label for="groupDisplayName"><fmt:message key="group.edit.share_display_name" /></label></b>
<input type="text" name="groupDisplayName" size="30" maxlength="100" value="<%= (groupDisplayName != null ? StringUtils.escapeForXML(groupDisplayName) : "") %>"><br> <p><input type="text" id="groupDisplayName" name="groupDisplayName" size="45" maxlength="100" value="${fn:escapeXml(group.properties['sharedRoster.displayName'])}">
<% if (errors.get("groupDisplayName") != null) { %> <c:if test="${not empty errors['groupDisplayName']}">
<span class="jive-error-text"><fmt:message key="group.edit.share_display_name" /></span><br/> <br><span class="jive-error-text"><fmt:message key="group.edit.share_display_name" /></span>
<% } %> </c:if>
<script type="text/javascript" language="JavaScript"> </p>
function toggleRosterShare() { <p><b><fmt:message key="group.edit.share_with"/></b></p>
if (document.getElementById('cb101').checked == false) {
document.getElementById('jive-rosterShare').style.display = 'none';
} else {
document.getElementById('jive-rosterShare').style.display = 'block';
document.getElementById('rb002').checked = true;
}
}
</script>
<input type="checkbox" id="cb101" name="shareContactList" onClick="toggleRosterShare();" style="vertical-align: middle;"
<%= (shareAdditional ? "checked" : "") %>>
<label for="cb101"><fmt:message key="group.edit.share_additional" /></label>
<div id="jive-rosterShare" style="display: <%= (enableRosterGroups && shareAdditional) ? "block" : "none" %>;">
<table cellpadding="2" cellspacing="0" border="0" width="100%"> <table cellpadding="2" cellspacing="0" border="0" width="100%">
<tbody>
<tr> <tr>
<td width="1%" nowrap> <td width="1%" nowrap>
<input type="radio" name="showGroup" value="everybody" id="rb002" <input type="radio" name="showGroup" value="onlyGroup" id="rb001" ${( group.properties["sharedRoster.showInRoster"] eq "nobody" ) or ( group.properties["sharedRoster.showInRoster"] eq "onlyGroup" and empty groupNames ) ? "checked" : "" }>
<%= ("everybody".equals(showGroup) ? "checked" : "") %>>
</td> </td>
<td width="99%"> <td width="99%">
<label for="rb002"><fmt:message key="group.edit.share_all_users" /></label> <label for="rb001"><fmt:message key="group.edit.share_group_only" /></label>
</td> </td>
</tr> </tr>
<tr> <tr>
<td width="1%" nowrap> <td width="1%" nowrap>
<input type="radio" name="showGroup" value="spefgroups" id="rb003" <input type="radio" name="showGroup" value="everybody" id="rb002" ${group.properties["sharedRoster.showInRoster"] eq "everybody" ? "checked" : ""}>
<%= (groupNames != null && groupNames.length > 0) ? "checked" : "" %>>
</td> </td>
<td width="99%"> <td width="99%">
<label for="rb003"><fmt:message key="group.edit.share_roster_groups" /></label> <label for="rb002"><fmt:message key="group.edit.share_all_users" /></label>
</td> </td>
</tr> </tr>
<tr> <tr>
<td width="1%" nowrap> <td width="1%" nowrap>
&nbsp; <input type="radio" name="showGroup" value="spefgroups" id="rb003" ${group.properties["sharedRoster.showInRoster"] eq "onlyGroup" and not empty groupNames ? "checked" : ""}>
</td> </td>
<td width="99%"> <td width="99%">
<select name="groupNames" size="6" onclick="this.form.showGroup[1].checked=true;" <label for="rb003"><fmt:message key="group.edit.share_roster_groups" /></label>
</td>
</tr>
<tr>
<td width="1%" nowrap></td>
<td width="99%">
<select name="groupNames" id="groupNames" size="6" onclick="this.form.showGroup[2].checked=true;"
multiple style="width:340px;font-family:verdana,arial,helvetica,sans-serif;font-size:8pt;"> multiple style="width:340px;font-family:verdana,arial,helvetica,sans-serif;font-size:8pt;">
<% for (Group g : webManager.getGroupManager().getGroups()) { <c:forEach var="g" items="${webManager.groupManager.groups}">
// Do not offer the edited group in the list of groups <!-- Do not offer the edited group in the list of groups. Members of the editing group can always see each other -->
// Members of the editing group can always see each other <c:if test="${not g.equals(group)}">
if (g.equals(group)) { <option value="${fn:escapeXml(g.name)}" ${groupNames.contains(g.name) ? "selected": ""}>
continue; <c:out value="${g.name}"/>
} </option>
%> </c:if>
</c:forEach>
<option value="<%= URLEncoder.encode(g.getName(), "UTF-8") %>"
<%= (contains(groupNames, g.getName()) ? "selected" : "") %>
><%= StringUtils.escapeHTMLTags(g.getName()) %></option>
<% } %>
</select> </select>
</td> </td>
</tr> </tr>
</tbody>
</table> </table>
</div> </div>
</div>
</td> </td>
</tr> </tr>
<tr> <tr>
<td width="1%"> <td width="1%"></td>
&nbsp;
</td>
<td width="99%"> <td width="99%">
<input type="submit" name="updateContactListSettings" value="<fmt:message key="group.edit.share_save" />">
<input type="submit" name="save" value="<fmt:message key="group.edit.share_save" />">
</td> </td>
</tr> </tr>
</tbody>
</table> </table>
</div>
<!-- END contact list settings -->
</form>
</admin:contentBox>
<!-- END contact list settings -->
</form> <!-- BEGIN group membership management -->
<fmt:message key="group.edit.members" var="groupmembersboxtitle"/>
<admin:contentBox title="${groupmembersboxtitle}">
<c:if test="${webManager.groupManager.readOnly}">
<admin:infobox type="info"><fmt:message key="group.read_only"/></admin:infobox>
</c:if>
<!-- BEGIN group membership management --> <c:if test="${not webManager.groupManager.readOnly}">
<div class="jive-contentBoxHeader">
<fmt:message key="group.edit.members" />
</div>
<div class="jive-contentBox">
<% // Only show if the group isn't read-only.
if (!webManager.getGroupManager().isReadOnly()) { %>
<p> <p>
<fmt:message key="group.edit.members_description" /> <fmt:message key="group.edit.members_description" />
</p> </p>
<form action="group-edit.jsp" method="post" name="f"> <form name="groupmembers" method="post">
<input type="hidden" name="csrf" value="${csrf}"> <input type="hidden" name="csrf" value="${csrf}">
<input type="hidden" name="group" value="<%= StringUtils.escapeForXML(groupName) %>"> <input type="hidden" name="group" value="${fn:escapeXml(param.group)}"/>
<input type="hidden" name="add" value="Add"/> <input type="hidden" name="addMember" value="addMember"/>
<table cellpadding="3" cellspacing="1" border="0" style="margin: 0 0 8px 0;"> <table cellpadding="3" cellspacing="1" border="0" style="margin: 0 0 8px 0;">
<tr> <tr>
<td nowrap width="1%"> <td nowrap width="1%">
...@@ -508,32 +572,33 @@ ...@@ -508,32 +572,33 @@
<input type="text" size="45" name="username"/> <input type="text" size="45" name="username"/>
&nbsp;<input type="submit" name="addbutton" value="<fmt:message key="global.add" />"> &nbsp;<input type="submit" name="addbutton" value="<fmt:message key="global.add" />">
</td> </td>
<c:if test="${not empty errors['addMember']}">
<td>
<span class="jive-error-text"><fmt:message key="group.edit.invalid_username"/></span>
</td>
</c:if>
</tr> </tr>
</table> </table>
</form> </form>
<% } %> </c:if>
<form action="group-edit.jsp" method="post" name="main"> <form method="post" name="main">
<input type="hidden" name="csrf" value="${csrf}"> <input type="hidden" name="csrf" value="${csrf}">
<input type="hidden" name="group" value="<%= StringUtils.escapeForXML(groupName) %>"> <input type="hidden" name="group" value="${fn:escapeXml(param.group)}"/>
<table class="jive-table" cellpadding="3" cellspacing="0" border="0" width="435"> <table class="jive-table" cellpadding="3" cellspacing="0" border="0" width="80%">
<tr> <tr>
<th>&nbsp;</th> <th>&nbsp;</th>
<th nowrap><fmt:message key="group.edit.username" /></th> <th nowrap><fmt:message key="group.edit.username" /></th>
<% // Only show if the group isn't read-only. <c:if test="${not webManager.groupManager.readOnly}">
if (!webManager.getGroupManager().isReadOnly()) { %>
<th width="1%" nowrap class="jive-table-th-center"><fmt:message key="group.edit.admin" /></th> <th width="1%" nowrap class="jive-table-th-center"><fmt:message key="group.edit.admin" /></th>
<th width="1%" nowrap class="jive-table-th-center"><fmt:message key="group.edit.remove" /></th> <th width="1%" nowrap class="jive-table-th-center"><fmt:message key="group.edit.remove" /></th>
<% } %> </c:if>
</tr> </tr>
<!-- Add admins first -->
<% <c:set var="showRemoteJIDsWarning" value="false"/>
int memberCount = group.getMembers().size() + group.getAdmins().size();
boolean showUpdateButtons = memberCount > 0; <c:if test="${empty allMembers}">
boolean showRemoteJIDsWarning = false;
if (memberCount == 0) {
%>
<tr> <tr>
<td align="center" colspan="4"> <td align="center" colspan="4">
<br> <br>
...@@ -542,115 +607,113 @@ ...@@ -542,115 +607,113 @@
<br> <br>
</td> </td>
</tr> </tr>
<% </c:if>
} <c:forEach var="member" items="${allMembers}">
else {
// Sort the list of members.
ArrayList<JID> allMembers = new ArrayList<JID>(memberCount);
allMembers.addAll(group.getMembers());
Collection<JID> admins = group.getAdmins();
allMembers.addAll(admins);
Collections.sort(allMembers);
for (JID jid:allMembers) {
boolean isLocal = webManager.getXMPPServer().isLocal(jid);
User user = null;
if (isLocal) {
try {
user = userManager.getUser(jid.getNode());
}
catch (UserNotFoundException unfe) {
// Ignore.
}
}
%>
<tr> <tr>
<td width="1%"> <td width="1%">
<% if (user != null && presenceManager.isAvailable(user)) {
Presence presence = presenceManager.getPresence(user); <c:choose>
%> <c:when test="${webManager.XMPPServer.isLocal(member)}">
<% if (presence.getShow() == null) { %> <c:choose>
<c:when test="${webManager.userManager.isRegisteredUser(member) and webManager.presenceManager.isAvailable(webManager.userManager.getUser(member))}">
<c:choose>
<c:when test="${empty webManager.presenceManager.getPresence(webManager.userManager.getUser(member)).show}">
<img src="images/im_available.gif" width="16" height="16" border="0" title="<fmt:message key="user.properties.available" />" alt="<fmt:message key="user.properties.available" />"> <img src="images/im_available.gif" width="16" height="16" border="0" title="<fmt:message key="user.properties.available" />" alt="<fmt:message key="user.properties.available" />">
<% } %> </c:when>
<% if (presence.getShow() == Presence.Show.chat) { %> <c:when test="${webManager.presenceManager.getPresence(webManager.userManager.getUser(member)).show == 'chat'}">
<img src="images/im_free_chat.gif" width="16" height="16" border="0" title="<fmt:message key="session.details.chat_available" />" alt="<fmt:message key="session.details.chat_available" />"> <img src="images/im_free_chat.gif" width="16" height="16" border="0" title="<fmt:message key="session.details.chat_available" />" alt="<fmt:message key="session.details.chat_available" />">
<% } %> </c:when>
<% if (presence.getShow() == Presence.Show.away) { %> <c:when test="${webManager.presenceManager.getPresence(webManager.userManager.getUser(member)).show == 'away'}">
<img src="images/im_away.gif" width="16" height="16" border="0" title="<fmt:message key="session.details.away" />" alt="<fmt:message key="session.details.away" />"> <img src="images/im_away.gif" width="16" height="16" border="0" title="<fmt:message key="session.details.away" />" alt="<fmt:message key="session.details.away" />">
<% } %> </c:when>
<% if (presence.getShow() == Presence.Show.xa) { %> <c:when test="${webManager.presenceManager.getPresence(webManager.userManager.getUser(member)).show == 'xa'}">
<img src="images/im_away.gif" width="16" height="16" border="0" title="<fmt:message key="session.details.extended" />" alt="<fmt:message key="session.details.extended" />"> <img src="images/im_away.gif" width="16" height="16" border="0" title="<fmt:message key="session.details.extended" />" alt="<fmt:message key="session.details.extended" />">
<% } %> </c:when>
<% if (presence.getShow() == Presence.Show.dnd) { %> <c:when test="${webManager.presenceManager.getPresence(webManager.userManager.getUser(member)).show == 'dnd'}">
<img src="images/im_dnd.gif" width="16" height="16" border="0" title="<fmt:message key="session.details.not_disturb" />" alt="<fmt:message key="session.details.not_disturb" />"> <img src="images/im_dnd.gif" width="16" height="16" border="0" title="<fmt:message key="session.details.not_disturb" />" alt="<fmt:message key="session.details.not_disturb" />">
<% } %> </c:when>
</c:choose>
<% } else { %> </c:when>
<c:otherwise>
<img src="images/im_unavailable.gif" width="16" height="16" border="0" title="<fmt:message key="user.properties.offline" />" alt="<fmt:message key="user.properties.offline" />"> <img src="images/im_unavailable.gif" width="16" height="16" border="0" title="<fmt:message key="user.properties.offline" />" alt="<fmt:message key="user.properties.offline" />">
<% } %> </c:otherwise>
</c:choose>
</c:when>
<c:otherwise>
<c:set var="showRemoteJIDsWarning" value="true"/>
&nbsp;
</c:otherwise>
</c:choose>
</td>
<td>
<c:choose>
<c:when test="${webManager.userManager.isRegisteredUser(member)}">
<a href="user-properties.jsp?username=${fn:escapeXml(webManager.userManager.getUser(member).username)}">
<c:out value="${webManager.userManager.getUser(member).username}"/>
</a>
</c:when>
<c:otherwise>
<c:out value="${member}"/> <font color="red"><b>*</b></font>
</c:otherwise>
</c:choose>
</td> </td>
<% if (user != null) { %> <c:if test="${not webManager.groupManager.readOnly}">
<td><a href="user-properties.jsp?username=<%= URLEncoder.encode(user.getUsername(), "UTF-8") %>"><%= StringUtils.escapeHTMLTags(JID.unescapeNode(user.getUsername())) %></a><% if (!isLocal) { showRemoteJIDsWarning = true; %> <font color="red"><b>*</b></font><%}%></td>
<% } else { %>
<td><%= jid %><% showRemoteJIDsWarning = true; %> <font color="red"><b>*</b></font></td>
<% } %>
<% // Only show if the group isn't read-only.
if (!webManager.getGroupManager().isReadOnly()) { %>
<td align="center"> <td align="center">
<input type="checkbox" name="admin" value="<%= jid %>" <% if (admins.contains(jid)) { %>checked<% } %>> <input type="checkbox" name="admin" value="${fn:escapeXml(member)}" ${group.admins.contains(member) ? 'checked' : ''}>
</td> </td>
<td align="center"> <td align="center">
<input type="checkbox" name="delete" value="<%= jid %>"> <input type="checkbox" name="delete" value="${fn:escapeXml(member)}">
</td> </td>
<% } %> </c:if>
</tr> </tr>
<% </c:forEach>
}
} <c:if test="${ ( not empty allMembers ) and (not webManager.groupManager.readOnly)}">
if (showUpdateButtons && !webManager.getGroupManager().isReadOnly()) {
%>
<tr> <tr>
<td colspan="2"> <td colspan="2">&nbsp;</td>
&nbsp;
</td>
<td align="center"> <td align="center">
<input type="submit" name="updateMember" value="Update"> <input type="submit" name="updateMember" value="Update">
</td> </td>
<td align="center"> <td align="center">
<input type="submit" name="remove" value="Remove"> <input type="submit" name="removeMember" value="Remove">
</td>
</tr>
<%
}
if (showRemoteJIDsWarning) {
%>
<tr>
<td colspan="4">
<font color="red">* <fmt:message key="group.edit.note" /></font>
</td> </td>
</tr> </tr>
<% </c:if>
}
%>
</table> </table>
<c:if test="${showRemoteJIDsWarning}">
<span class="jive-description"><font color="red">* <fmt:message key="group.edit.note" /></font></span>
</c:if>
</form> </form>
<script type="text/javascript"> <script type="text/javascript">
document.f.username.focus(); document.f.username.focus();
</script> </script>
</div> </admin:contentBox>
<!-- END group membership management --> <!-- END group membership management -->
<script type="text/javascript">
function toggleReadOnly()
{
var disabled = document.getElementById('rb201').checked;
document.getElementById( 'groupDisplayName' ).disabled = disabled;
document.getElementById( 'rb001' ).disabled = disabled;
document.getElementById( 'rb002' ).disabled = disabled;
document.getElementById( 'rb003' ).disabled = disabled;
document.getElementById( 'groupNames' ).disabled = disabled;
}
toggleReadOnly();
</script>
</body> </body>
</html> </html>
<%! <%!
private static String toList(String[] array, String enc) { private static String toList( String[] array ) {
if (array == null || array.length == 0) { if (array == null || array.length == 0) {
return ""; return "";
} }
...@@ -659,7 +722,7 @@ ...@@ -659,7 +722,7 @@
for (String anArray : array) { for (String anArray : array) {
String item; String item;
try { try {
item = URLDecoder.decode(anArray, enc); item = URLDecoder.decode( anArray, "UTF-8" );
} }
catch (UnsupportedEncodingException e) { catch (UnsupportedEncodingException e) {
item = anArray; item = anArray;
...@@ -669,16 +732,4 @@ ...@@ -669,16 +732,4 @@
} }
return buf.toString(); return buf.toString();
} }
private static boolean contains(String[] array, String item) {
if (array == null || array.length == 0 || item == null) {
return false;
}
for (String anArray : array) {
if (item.equals(anArray)) {
return true;
}
}
return false;
}
%> %>
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment