org.opencms.ui.dialogs.availability.CmsAvailabilityDialog Maven / Gradle / Ivy
Show all versions of opencms-test Show documentation
/*
* This library is part of OpenCms -
* the Open Source Content Management System
*
* Copyright (c) Alkacon Software GmbH & Co. KG (http://www.alkacon.com)
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* For further information about Alkacon Software, please see the
* company website: http://www.alkacon.com
*
* For further information about OpenCms, please see the
* project website: http://www.opencms.org
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
package org.opencms.ui.dialogs.availability;
import org.opencms.file.CmsObject;
import org.opencms.file.CmsProperty;
import org.opencms.file.CmsPropertyDefinition;
import org.opencms.file.CmsResource;
import org.opencms.file.CmsResourceFilter;
import org.opencms.file.types.I_CmsResourceType;
import org.opencms.gwt.shared.CmsPrincipalBean;
import org.opencms.loader.CmsLoaderException;
import org.opencms.lock.CmsLockActionRecord;
import org.opencms.lock.CmsLockActionRecord.LockChange;
import org.opencms.lock.CmsLockException;
import org.opencms.lock.CmsLockUtil;
import org.opencms.main.CmsException;
import org.opencms.main.CmsLog;
import org.opencms.main.OpenCms;
import org.opencms.security.CmsAccessControlEntry;
import org.opencms.security.I_CmsPrincipal;
import org.opencms.ui.A_CmsUI;
import org.opencms.ui.CmsVaadinUtils;
import org.opencms.ui.I_CmsDialogContext;
import org.opencms.ui.components.CmsBasicDialog;
import org.opencms.ui.components.CmsOkCancelActionHandler;
import org.opencms.ui.components.CmsResourceInfo;
import org.opencms.util.CmsStringUtil;
import org.opencms.util.CmsUUID;
import org.opencms.workplace.CmsWorkplace;
import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import org.apache.commons.logging.Log;
import com.vaadin.data.Property.ValueChangeEvent;
import com.vaadin.data.Property.ValueChangeListener;
import com.vaadin.data.Validator;
import com.vaadin.ui.Button;
import com.vaadin.ui.Button.ClickEvent;
import com.vaadin.ui.Button.ClickListener;
import com.vaadin.ui.CheckBox;
import com.vaadin.ui.DateField;
import com.vaadin.ui.Panel;
import com.vaadin.ui.TextField;
import com.vaadin.ui.VerticalLayout;
/**
* Availability dialog.
*/
public class CmsAvailabilityDialog extends CmsBasicDialog {
/** Logger for this class. */
private static final Log LOG = CmsLog.getLog(CmsAvailabilityDialog.class);
/** Serial version id. */
private static final long serialVersionUID = 1L;
/** Availability info. */
private CmsAvailabilityInfoBean m_availabilityInfo;
/** The cancel button. */
private Button m_cancelButton;
/** The dialog context. */
private I_CmsDialogContext m_dialogContext;
/** Date field. */
private DateField m_expiredField;
/** Initial value for 'notification enabled. */
private Boolean m_initialNotificationEnabled = Boolean.FALSE;
/** Initial value for notification interval. */
private String m_initialNotificationInterval = "";
/** 'Modify siblings' check box. */
private CheckBox m_modifySiblingsField;
/** 'enable notification' check box. */
private CheckBox m_notificationEnabledField;
/** Field for the notification interval. */
private TextField m_notificationIntervalField;
/** Panel for notifications. */
private Panel m_notificationPanel;
/** OK button. */
private Button m_okButton;
/** Date field. */
private DateField m_releasedField;
/** Option to reset the expiration. */
private CheckBox m_resetExpired;
/** Option to reset the relase date. */
private CheckBox m_resetReleased;
/** Container for responsibles widgets. */
private VerticalLayout m_responsiblesContainer;
/** Panel for 'Responsibles'. */
private Panel m_responsiblesPanel;
/** Option to enable subresource modification. */
private CheckBox m_subresourceModificationField;
/**
* Creates a new instance.
*
* @param dialogContext the dialog context
*/
public CmsAvailabilityDialog(I_CmsDialogContext dialogContext) {
super();
m_dialogContext = dialogContext;
CmsObject cms = dialogContext.getCms();
CmsVaadinUtils.readAndLocalizeDesign(
this,
OpenCms.getWorkplaceManager().getMessages(A_CmsUI.get().getLocale()),
null);
List resources = dialogContext.getResources();
m_notificationIntervalField.addValidator(new Validator() {
/** Serial version id. */
private static final long serialVersionUID = 1L;
public void validate(Object value) throws InvalidValueException {
String strValue = ((String)value).trim();
if (!strValue.matches("[0-9]*")) {
throw new InvalidValueException(
CmsVaadinUtils.getMessageText(org.opencms.ui.Messages.GUI_VALIDATOR_EMPTY_OR_NUMBER_0));
}
}
});
boolean hasSiblings = false;
for (CmsResource resource : m_dialogContext.getResources()) {
hasSiblings |= resource.getSiblingCount() > 1;
if (hasSiblings) {
break;
}
}
m_modifySiblingsField.setVisible(hasSiblings);
if (resources.size() == 1) {
CmsResource onlyResource = resources.get(0);
if (onlyResource.getDateReleased() != CmsResource.DATE_RELEASED_DEFAULT) {
m_releasedField.setValue(new Date(onlyResource.getDateReleased()));
}
if (onlyResource.getDateExpired() != CmsResource.DATE_EXPIRED_DEFAULT) {
m_expiredField.setValue(new Date(onlyResource.getDateExpired()));
}
initNotification();
Map responsibles = m_availabilityInfo.getResponsibles();
if (!responsibles.isEmpty()) {
m_responsiblesPanel.setVisible(true);
m_notificationPanel.setVisible(true);
for (Map.Entry entry : responsibles.entrySet()) {
CmsPrincipalBean principal = entry.getKey();
String icon = principal.isGroup()
? CmsWorkplace.getResourceUri("buttons/group.png")
: CmsWorkplace.getResourceUri("buttons/user.png");
String subtitle = "";
if (entry.getValue() != null) {
subtitle = cms.getRequestContext().removeSiteRoot(entry.getValue());
}
CmsResourceInfo infoWidget = new CmsResourceInfo(entry.getKey().getName(), subtitle, icon);
m_responsiblesContainer.addComponent(infoWidget);
}
}
} else {
boolean showNotification = false;
resourceLoop: for (CmsResource resource : resources) {
try {
List aces = cms.getAccessControlEntries(cms.getSitePath(resource));
for (CmsAccessControlEntry ace : aces) {
if (ace.isResponsible()) {
showNotification = true;
break resourceLoop;
}
}
} catch (CmsException e) {
LOG.error(e.getLocalizedMessage(), e);
}
}
m_notificationPanel.setVisible(showNotification);
}
m_notificationEnabledField.setValue(m_initialNotificationEnabled);
m_notificationIntervalField.setValue(m_initialNotificationInterval);
boolean hasFolders = false;
for (CmsResource resource : resources) {
if (resource.isFolder()) {
hasFolders = true;
}
}
m_subresourceModificationField.setVisible(hasFolders);
initResetCheckbox(m_resetReleased, m_releasedField);
initResetCheckbox(m_resetExpired, m_expiredField);
m_okButton.addClickListener(new ClickListener() {
private static final long serialVersionUID = 1L;
public void buttonClick(ClickEvent event) {
submit();
}
});
m_cancelButton.addClickListener(new ClickListener() {
private static final long serialVersionUID = 1L;
public void buttonClick(ClickEvent event) {
cancel();
}
});
displayResourceInfo(m_dialogContext.getResources());
setActionHandler(new CmsOkCancelActionHandler() {
private static final long serialVersionUID = 1L;
@Override
protected void cancel() {
CmsAvailabilityDialog.this.cancel();
}
@Override
protected void ok() {
submit();
}
});
}
/**
* Initializes the values for the notification widgets.
*/
public void initNotification() {
if (m_dialogContext.getResources().size() == 1) {
CmsResource resource = m_dialogContext.getResources().get(0);
try {
m_availabilityInfo = getAvailabilityInfo(A_CmsUI.getCmsObject(), resource);
m_initialNotificationInterval = "" + m_availabilityInfo.getNotificationInterval();
m_initialNotificationEnabled = Boolean.valueOf(m_availabilityInfo.isNotificationEnabled());
} catch (CmsLoaderException e) {
LOG.error(e.getLocalizedMessage(), e);
} catch (CmsException e) {
LOG.error(e.getLocalizedMessage(), e);
}
}
}
/**
* Actually performs the availability change.
*
* @return the ids of the changed resources
*
* @throws CmsException if something goes wrong
*/
protected List changeAvailability() throws CmsException {
Date released = m_releasedField.getValue();
Date expired = m_expiredField.getValue();
boolean resetReleased = m_resetReleased.getValue().booleanValue();
boolean resetExpired = m_resetExpired.getValue().booleanValue();
boolean modifySubresources = m_subresourceModificationField.getValue().booleanValue();
List changedIds = new ArrayList();
for (CmsResource resource : m_dialogContext.getResources()) {
changeAvailability(resource, released, resetReleased, expired, resetExpired, modifySubresources);
changedIds.add(resource.getStructureId());
}
String notificationInterval = m_notificationIntervalField.getValue().trim();
int notificationIntervalInt = 0;
try {
notificationIntervalInt = Integer.parseInt(notificationInterval);
} catch (NumberFormatException e) {
LOG.warn(e.getLocalizedMessage(), e);
}
Boolean notificationEnabled = m_notificationEnabledField.getValue();
boolean notificationSettingsUnchanged = notificationInterval.equals(m_initialNotificationInterval)
&& notificationEnabled.equals(m_initialNotificationEnabled);
CmsObject cms = A_CmsUI.getCmsObject();
if (!notificationSettingsUnchanged) {
for (CmsResource resource : m_dialogContext.getResources()) {
performSingleResourceNotification(
A_CmsUI.getCmsObject(),
cms.getSitePath(resource),
notificationEnabled.booleanValue(),
notificationIntervalInt,
m_modifySiblingsField.getValue().booleanValue());
}
}
return changedIds;
}
/**
* Cancels the dialog.
*/
void cancel() {
m_dialogContext.finish(new ArrayList());
}
/**
* Returns the availability info.
*
* @param cms the cms context
* @param res the resource
*
* @return the info
*
* @throws CmsException if reading the info fails
*/
CmsAvailabilityInfoBean getAvailabilityInfo(CmsObject cms, CmsResource res) throws CmsException {
CmsAvailabilityInfoBean result = new CmsAvailabilityInfoBean();
String resourceSitePath = cms.getRequestContext().removeSiteRoot(res.getRootPath());
result.setVfsPath(resourceSitePath);
I_CmsResourceType type = OpenCms.getResourceManager().getResourceType(res.getTypeId());
result.setResType(type.getTypeName());
result.setDateReleased(res.getDateReleased());
result.setDateExpired(res.getDateExpired());
String notificationInterval = cms.readPropertyObject(
res,
CmsPropertyDefinition.PROPERTY_NOTIFICATION_INTERVAL,
false).getValue();
if (CmsStringUtil.isNotEmptyOrWhitespaceOnly(notificationInterval)) {
result.setNotificationInterval(Integer.valueOf(notificationInterval).intValue());
}
String notificationEnabled = cms.readPropertyObject(
res,
CmsPropertyDefinition.PROPERTY_ENABLE_NOTIFICATION,
false).getValue();
if (CmsStringUtil.isNotEmptyOrWhitespaceOnly(notificationEnabled)) {
result.setNotificationEnabled(Boolean.valueOf(notificationEnabled).booleanValue());
}
result.setHasSiblings(cms.readSiblings(resourceSitePath, CmsResourceFilter.ALL).size() > 1);
result.setResponsibles(getResponsibles(cms, res.getRootPath()));
return result;
}
/**
* Returns the responsibles.
*
* @param cms the cms context
* @param rootPath the resource root path
*
* @return the responsibles
*
* @throws CmsException in case reading the resource fails
*/
Map getResponsibles(CmsObject cms, String rootPath) throws CmsException {
Map result = new HashMap();
List parentResources = new ArrayList();
String resourceSitePath = cms.getRequestContext().removeSiteRoot(rootPath);
// get all parent folders of the current file
try {
parentResources = cms.readPath(resourceSitePath, CmsResourceFilter.IGNORE_EXPIRATION);
} catch (CmsException e) {
LOG.error(e.getLocalizedMessage(), e);
}
for (CmsResource resource : parentResources) {
String storedSiteRoot = cms.getRequestContext().getSiteRoot();
String sitePath = cms.getRequestContext().removeSiteRoot(resource.getRootPath());
try {
cms.getRequestContext().setSiteRoot("/");
List entries = cms.getAccessControlEntries(resource.getRootPath(), false);
for (CmsAccessControlEntry ace : entries) {
if (ace.isResponsible()) {
I_CmsPrincipal principal = cms.lookupPrincipal(ace.getPrincipal());
if (principal != null) {
CmsPrincipalBean prinBean = new CmsPrincipalBean(
principal.getName(),
principal.getDescription(),
principal.isGroup());
if (!resource.getRootPath().equals(rootPath)) {
if (resource.getRootPath().startsWith(storedSiteRoot)) {
result.put(prinBean, sitePath);
} else {
result.put(prinBean, resource.getRootPath());
}
} else {
result.put(prinBean, null);
}
}
}
}
} catch (CmsException e) {
LOG.info(
"Problem with reading responsible users for "
+ resource.getName()
+ " : "
+ e.getLocalizedMessage(),
e);
} finally {
cms.getRequestContext().setSiteRoot(storedSiteRoot);
}
}
return result;
}
/**
* Submits the dialog.
*/
void submit() {
if (validate()) {
try {
m_dialogContext.finish(changeAvailability());
} catch (Throwable t) {
m_dialogContext.error(t);
}
}
}
/**
* Changes availability.
*
* @param resource the resource
* @param released release date
* @param resetReleased reset release date
* @param expired expiration date
* @param resetExpired reset expiration date
* @param modifySubresources modify children
*
* @throws CmsException if something goes wrong
*/
private void changeAvailability(
CmsResource resource,
Date released,
boolean resetReleased,
Date expired,
boolean resetExpired,
boolean modifySubresources) throws CmsException {
CmsObject cms = m_dialogContext.getCms();
CmsLockActionRecord lockActionRecord = CmsLockUtil.ensureLock(cms, resource);
try {
long newDateReleased;
if (resetReleased || (released != null)) {
newDateReleased = released != null ? released.getTime() : CmsResource.DATE_RELEASED_DEFAULT;
cms.setDateReleased(resource, newDateReleased, modifySubresources);
}
long newDateExpired;
if (resetExpired || (expired != null)) {
newDateExpired = expired != null ? expired.getTime() : CmsResource.DATE_EXPIRED_DEFAULT;
cms.setDateExpired(resource, newDateExpired, modifySubresources);
}
} finally {
if (lockActionRecord.getChange() == LockChange.locked) {
try {
cms.unlockResource(resource);
} catch (CmsLockException e) {
LOG.warn(e.getLocalizedMessage(), e);
}
}
}
}
/**
* Creates a reset checkbox which can enable / disable a date field.
*
* @param box the check box
* @param field the date field
*/
private void initResetCheckbox(CheckBox box, final DateField field) {
box.addValueChangeListener(new ValueChangeListener() {
private static final long serialVersionUID = 1L;
public void valueChange(ValueChangeEvent event) {
Boolean value = (Boolean)(event.getProperty().getValue());
if (value.booleanValue()) {
field.clear();
field.setEnabled(false);
} else {
field.setEnabled(true);
}
}
});
}
/**
* Performs the notification operations on a single resource.
*
* @param cms the CMS context
* @param resName the VFS path of the resource
* @param enableNotification if the notification is activated
* @param notificationInterval the notification interval in days
* @param modifySiblings flag indicating to include resource siblings
*
* @throws CmsException if the availability and notification operations fail
*/
private void performSingleResourceNotification(
CmsObject cms,
String resName,
boolean enableNotification,
int notificationInterval,
boolean modifySiblings) throws CmsException {
List resources = new ArrayList();
if (modifySiblings) {
// modify all siblings of a resource
resources = cms.readSiblings(resName, CmsResourceFilter.IGNORE_EXPIRATION);
} else {
// modify only resource without siblings
resources.add(cms.readResource(resName, CmsResourceFilter.IGNORE_EXPIRATION));
}
Iterator i = resources.iterator();
while (i.hasNext()) {
CmsResource resource = i.next();
String resourcePath = cms.getRequestContext().removeSiteRoot(resource.getRootPath());
// lock resource if auto lock is enabled
CmsLockActionRecord lockRecord = CmsLockUtil.ensureLock(cms, resource);
try {
// write notification settings
writeProperty(
cms,
resourcePath,
CmsPropertyDefinition.PROPERTY_NOTIFICATION_INTERVAL,
String.valueOf(notificationInterval));
writeProperty(
cms,
resourcePath,
CmsPropertyDefinition.PROPERTY_ENABLE_NOTIFICATION,
String.valueOf(enableNotification));
} finally {
if (lockRecord.getChange() == LockChange.locked) {
cms.unlockResource(resource);
}
}
}
}
/**
* Validates release / expiration.
*
* @return true if the fields are valid.
*/
private boolean validate() {
return m_releasedField.isValid() && m_expiredField.isValid();
}
/**
* Writes a property value for a resource.
*
* @param cms the cms context
* @param resourcePath the path of the resource
* @param propertyName the name of the property
* @param propertyValue the new value of the property
*
* @throws CmsException if something goes wrong
*/
private void writeProperty(CmsObject cms, String resourcePath, String propertyName, String propertyValue)
throws CmsException {
if (CmsStringUtil.isEmpty(propertyValue)) {
propertyValue = CmsProperty.DELETE_VALUE;
}
CmsProperty newProp = new CmsProperty();
newProp.setName(propertyName);
CmsProperty oldProp = cms.readPropertyObject(resourcePath, propertyName, false);
if (oldProp.isNullProperty()) {
// property value was not already set
if (OpenCms.getWorkplaceManager().isDefaultPropertiesOnStructure()) {
newProp.setStructureValue(propertyValue);
} else {
newProp.setResourceValue(propertyValue);
}
} else {
if (oldProp.getStructureValue() != null) {
newProp.setStructureValue(propertyValue);
newProp.setResourceValue(oldProp.getResourceValue());
} else {
newProp.setResourceValue(propertyValue);
}
}
newProp.setAutoCreatePropertyDefinition(true);
String oldStructureValue = oldProp.getStructureValue();
String newStructureValue = newProp.getStructureValue();
if (CmsStringUtil.isEmpty(oldStructureValue)) {
oldStructureValue = CmsProperty.DELETE_VALUE;
}
if (CmsStringUtil.isEmpty(newStructureValue)) {
newStructureValue = CmsProperty.DELETE_VALUE;
}
String oldResourceValue = oldProp.getResourceValue();
String newResourceValue = newProp.getResourceValue();
if (CmsStringUtil.isEmpty(oldResourceValue)) {
oldResourceValue = CmsProperty.DELETE_VALUE;
}
if (CmsStringUtil.isEmpty(newResourceValue)) {
newResourceValue = CmsProperty.DELETE_VALUE;
}
// change property only if it has been changed
if (!oldResourceValue.equals(newResourceValue) || !oldStructureValue.equals(newStructureValue)) {
cms.writePropertyObject(resourcePath, newProp);
}
}
}