. Version History list for a file (available in the File Details screen)

- View previous file version content
 - Sorted by most recent version first
 - Revert to a previous version action
. Some more changes to Create Web Project wizard from Usability Review

git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@4756 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
This commit is contained in:
Kevin Roast
2007-01-08 19:13:45 +00:00
parent e4ca6831b6
commit 8b10b42a07
7 changed files with 202 additions and 16 deletions

View File

@@ -128,6 +128,11 @@ public class AVMNode extends Node implements Map<String, Object>
return this.avmRef.getPath(); return this.avmRef.getPath();
} }
public final AVMNodeDescriptor getDescriptor()
{
return avmRef;
}
public int getVersion() public int getVersion()
{ {
return this.version; return this.version;

View File

@@ -16,11 +16,31 @@
*/ */
package org.alfresco.web.bean.wcm; package org.alfresco.web.bean.wcm;
import java.io.Serializable;
import java.text.MessageFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap;
import java.util.List; import java.util.List;
import java.util.Map;
import javax.faces.context.FacesContext;
import javax.faces.event.ActionEvent;
import javax.transaction.UserTransaction;
import org.alfresco.repo.avm.AVMNodeConverter;
import org.alfresco.repo.avm.actions.AVMRevertToVersionAction;
import org.alfresco.service.cmr.action.Action;
import org.alfresco.service.cmr.action.ActionService;
import org.alfresco.service.cmr.avm.AVMNodeDescriptor;
import org.alfresco.util.Pair;
import org.alfresco.web.app.Application;
import org.alfresco.web.app.servlet.DownloadContentServlet; import org.alfresco.web.app.servlet.DownloadContentServlet;
import org.alfresco.web.bean.repository.Node; import org.alfresco.web.bean.repository.Node;
import org.alfresco.web.bean.repository.Repository;
import org.alfresco.web.ui.common.Utils; import org.alfresco.web.ui.common.Utils;
import org.alfresco.web.ui.common.component.UIActionLink;
import org.alfresco.web.ui.common.component.data.UIRichList;
/** /**
* Backing bean for File Details page. * Backing bean for File Details page.
@@ -29,6 +49,36 @@ import org.alfresco.web.ui.common.Utils;
*/ */
public class FileDetailsBean extends AVMDetailsBean public class FileDetailsBean extends AVMDetailsBean
{ {
/** Action service bean reference */
private ActionService actionService;
// ------------------------------------------------------------------------------
// Construction
/**
* Default constructor
*/
public FileDetailsBean()
{
super();
// initial state of some panels that don't use the default
panels.put("version-history-panel", false);
}
// ------------------------------------------------------------------------------
// Bean getters and setters
/**
* @param actionService The actionService to set.
*/
public void setActionService(ActionService actionService)
{
this.actionService = actionService;
}
/** /**
* @see org.alfresco.web.bean.wcm.AVMDetailsBean#getAvmNode() * @see org.alfresco.web.bean.wcm.AVMDetailsBean#getAvmNode()
*/ */
@@ -92,4 +142,81 @@ public class FileDetailsBean extends AVMDetailsBean
{ {
return (List)this.avmBrowseBean.getFiles(); return (List)this.avmBrowseBean.getFiles();
} }
/**
* @return version history list for a node
*/
public List<Map<String, Object>> getVersionHistory()
{
AVMNode avmNode = getAvmNode();
List<AVMNodeDescriptor> history = this.avmService.getHistory(avmNode.getDescriptor(), -1);
List<Map<String, Object>> wrappers = new ArrayList<Map<String, Object>>(history.size());
for (AVMNodeDescriptor record : history)
{
Map<String, Object> wrapper = new HashMap<String, Object>(4, 1.0f);
wrapper.put("version", record.getVersionID());
wrapper.put("strVersion", Integer.toString(record.getVersionID()));
wrapper.put("modifiedDate", new Date(record.getModDate()));
List<Pair<Integer, String>> paths = this.avmService.getPaths(record);
if (paths.size() != 0)
{
// display the first path as any will show the same content
Pair<Integer, String> path = paths.get(0);
wrapper.put("url", DownloadContentServlet.generateBrowserURL(
AVMNodeConverter.ToNodeRef(path.getFirst(), path.getSecond()), avmNode.getName()));
}
wrappers.add(wrapper);
}
return wrappers;
}
/**
* Revert a node back to a previous version
*/
public void revertNode(ActionEvent event)
{
UIActionLink link = (UIActionLink)event.getComponent();
Map<String, String> params = link.getParameterMap();
int version = Integer.parseInt(params.get("version"));
UserTransaction tx = null;
try
{
FacesContext context = FacesContext.getCurrentInstance();
tx = Repository.getUserTransaction(context, false);
tx.begin();
Map<String, Serializable> args = new HashMap<String, Serializable>(1, 1.0f);
List<AVMNodeDescriptor> history = this.avmService.getHistory(getAvmNode().getDescriptor(), -1);
// the history list should contain the version ID we are looking for
for (AVMNodeDescriptor record : history)
{
if (record.getVersionID() == version)
{
// the action expects the HEAD revision as the noderef and
// the to-revert param as the previous version to revert to
Action action = this.actionService.createAction(AVMRevertToVersionAction.NAME, args);
args.put(AVMRevertToVersionAction.TOREVERT, record);
this.actionService.executeAction(action, getAvmNode().getNodeRef());
// clear the version history list after a revert ready for refresh
UIRichList versionList = (UIRichList)link.findComponent("version-history-list");
versionList.setValue(null);
// reset the action node reference as the version ID has changed
avmBrowseBean.setAvmActionNode(new AVMNode(avmService.lookup(-1, getAvmNode().getPath())));
break;
}
}
tx.commit();
}
catch (Throwable err)
{
err.printStackTrace(System.err);
Utils.addErrorMessage(MessageFormat.format(Application.getMessage(
FacesContext.getCurrentInstance(), Repository.ERROR_GENERIC), err.getMessage()), err);
try { if (tx != null) {tx.rollback();} } catch (Exception tex) {}
}
}
} }

View File

@@ -324,9 +324,11 @@ public class UIUserSandboxes extends SelfRenderingComponent
out.write("<table cellspacing=2 cellpadding=2 border=0 width=100%><tr><td>"); out.write("<table cellspacing=2 cellpadding=2 border=0 width=100%><tr><td>");
// show the icon for the sandbox as a clickable browse link image // show the icon for the sandbox as a clickable browse link image
// this is currently identical to the sandbox_browse action as below // this is currently identical to the sandbox_browse action as below
Utils.encodeRecursive(context, aquireAction( UIActionLink browseAction = aquireAction(
context, mainStore, username, ACT_SANDBOX_ICON, WebResources.IMAGE_USERSANDBOX_32, context, mainStore, username, ACT_SANDBOX_ICON, WebResources.IMAGE_USERSANDBOX_32,
"#{AVMBrowseBean.setupSandboxAction}", "browseSandbox")); "#{AVMBrowseBean.setupSandboxAction}", "browseSandbox");
browseAction.setShowLink(false);
Utils.encodeRecursive(context, browseAction);
out.write("</td><td width=100%>"); out.write("</td><td width=100%>");
out.write("<b>"); out.write("<b>");
out.write(bundle.getString(MSG_USERNAME)); out.write(bundle.getString(MSG_USERNAME));
@@ -337,6 +339,11 @@ public class UIUserSandboxes extends SelfRenderingComponent
out.write(")</td><td><nobr>"); out.write(")</td><td><nobr>");
// direct actions for a sandbox // direct actions for a sandbox
Utils.encodeRecursive(context, aquireAction(
context, mainStore, username, ACT_SANDBOX_BROWSE, "/images/icons/space_small.gif",
"#{AVMBrowseBean.setupSandboxAction}", "browseSandbox"));
out.write("&nbsp;&nbsp;");
String websiteUrl = AVMConstants.buildWebappUrl(mainStore, getWebapp()); String websiteUrl = AVMConstants.buildWebappUrl(mainStore, getWebapp());
Map requestMap = context.getExternalContext().getRequestMap(); Map requestMap = context.getExternalContext().getRequestMap();
requestMap.put(REQUEST_PREVIEW_REF, websiteUrl); requestMap.put(REQUEST_PREVIEW_REF, websiteUrl);
@@ -349,24 +356,20 @@ public class UIUserSandboxes extends SelfRenderingComponent
Utils.encodeRecursive(context, aquireAction( Utils.encodeRecursive(context, aquireAction(
context, mainStore, username, ACT_SANDBOX_SUBMITALL, "/images/icons/submit_all.gif", context, mainStore, username, ACT_SANDBOX_SUBMITALL, "/images/icons/submit_all.gif",
"#{AVMBrowseBean.setupAllItemsAction}", "dialog:submitSandboxItems")); "#{AVMBrowseBean.setupAllItemsAction}", "dialog:submitSandboxItems"));
out.write("&nbsp;"); out.write("&nbsp;&nbsp;");
Utils.encodeRecursive(context, aquireAction( Utils.encodeRecursive(context, aquireAction(
context, mainStore, username, ACT_SANDBOX_REVERTALL, "/images/icons/revert_all.gif", context, mainStore, username, ACT_SANDBOX_REVERTALL, "/images/icons/revert_all.gif",
"#{AVMBrowseBean.setupAllItemsAction}", "dialog:revertAllItems")); "#{AVMBrowseBean.setupAllItemsAction}", "dialog:revertAllItems"));
out.write("&nbsp;"); out.write("&nbsp;&nbsp;");
if (isManager) if (isManager)
{ {
Utils.encodeRecursive(context, aquireAction( Utils.encodeRecursive(context, aquireAction(
context, mainStore, username, ACT_REMOVE_SANDBOX, "/images/icons/delete_sandbox.gif", context, mainStore, username, ACT_REMOVE_SANDBOX, "/images/icons/delete_sandbox.gif",
"#{AVMBrowseBean.setupSandboxAction}", "dialog:deleteSandbox")); "#{AVMBrowseBean.setupSandboxAction}", "dialog:deleteSandbox"));
out.write("&nbsp;");
} }
Utils.encodeRecursive(context, aquireAction(
context, mainStore, username, ACT_SANDBOX_BROWSE, "/images/icons/space_small.gif",
"#{AVMBrowseBean.setupSandboxAction}", "browseSandbox"));
out.write("</nobr></td></tr>"); out.write("</nobr></td></tr>");
// modified items panel // modified items panel
@@ -674,11 +677,11 @@ public class UIUserSandboxes extends SelfRenderingComponent
// output multi-select actions for this user // output multi-select actions for this user
out.write("<tr><td colspan=8>"); out.write("<tr><td colspan=8>");
out.write(bundle.getString(MSG_SELECTED)); out.write(bundle.getString(MSG_SELECTED));
out.write(":&nbsp;"); out.write(":&nbsp;&nbsp;");
Utils.encodeRecursive(fc, aquireAction( Utils.encodeRecursive(fc, aquireAction(
fc, userStore, username, ACT_SANDBOX_SUBMITSELECTED, "/images/icons/submit_all.gif", fc, userStore, username, ACT_SANDBOX_SUBMITSELECTED, "/images/icons/submit_all.gif",
"#{AVMBrowseBean.setupSandboxAction}", "dialog:submitSandboxItems")); "#{AVMBrowseBean.setupSandboxAction}", "dialog:submitSandboxItems"));
out.write("&nbsp;"); out.write("&nbsp;&nbsp;");
Utils.encodeRecursive(fc, aquireAction( Utils.encodeRecursive(fc, aquireAction(
fc, userStore, username, ACT_SANDBOX_REVERTSELECTED, "/images/icons/revert_all.gif", fc, userStore, username, ACT_SANDBOX_REVERTSELECTED, "/images/icons/revert_all.gif",
"#{AVMBrowseBean.setupSandboxAction}", "dialog:revertSelectedItems")); "#{AVMBrowseBean.setupSandboxAction}", "dialog:revertSelectedItems"));
@@ -925,7 +928,7 @@ public class UIUserSandboxes extends SelfRenderingComponent
control.setRendererType(UIActions.RENDERER_ACTIONLINK); control.setRendererType(UIActions.RENDERER_ACTIONLINK);
control.setId(id); control.setId(id);
control.setValue(Application.getMessage(fc, name)); control.setValue(Application.getMessage(fc, name));
control.setShowLink(icon != null ? false : true); //control.setShowLink(icon != null ? false : true);
control.setImage(icon); control.setImage(icon);
if (actionListener != null) if (actionListener != null)

View File

@@ -2601,6 +2601,10 @@
<property-name>avmBrowseBean</property-name> <property-name>avmBrowseBean</property-name>
<value>#{AVMBrowseBean}</value> <value>#{AVMBrowseBean}</value>
</managed-property> </managed-property>
<managed-property>
<property-name>actionService</property-name>
<value>#{ActionService}</value>
</managed-property>
</managed-bean> </managed-bean>
<managed-bean> <managed-bean>

View File

@@ -354,7 +354,7 @@
<f:facet name="header"> <f:facet name="header">
<h:outputText value="#{msg.actions}"/> <h:outputText value="#{msg.actions}"/>
</f:facet> </f:facet>
<a:actionLink id="view-link" value="View" href="#{r.url}" target="new" /> <a:actionLink id="view-link" value="#{msg.view}" href="#{r.url}" target="new" />
</a:column> </a:column>
<a:dataPager styleClass="pager" /> <a:dataPager styleClass="pager" />

View File

@@ -117,9 +117,12 @@
<td align=left width=32><a:actionLink image="/images/icons/sandbox_large.gif" showLink="false" value="#{msg.staging_sandbox}" actionListener="#{AVMBrowseBean.setupSandboxAction}" action="browseSandbox" /></td> <td align=left width=32><a:actionLink image="/images/icons/sandbox_large.gif" showLink="false" value="#{msg.staging_sandbox}" actionListener="#{AVMBrowseBean.setupSandboxAction}" action="browseSandbox" /></td>
<td align=left><h:outputText value="#{msg.staging_sandbox}" styleClass="mainSubTitle" /></td> <td align=left><h:outputText value="#{msg.staging_sandbox}" styleClass="mainSubTitle" /></td>
<td align=right> <td align=right>
<a:actionLink id="actPreview" value="#{msg.sandbox_preview}" image="/images/icons/preview_website.gif" showLink="false" href="#{AVMBrowseBean.stagingPreviewUrl}" target="new" /> <a:actionLink id="actBrowse" value="#{msg.sandbox_browse}" image="/images/icons/space_small.gif" actionListener="#{AVMBrowseBean.setupSandboxAction}" action="browseSandbox" />
<a:actionLink id="actSnap" value="#{msg.sandbox_snapshot}" image="/images/icons/create_snapshot.gif" showLink="false" actionListener="#{AVMBrowseBean.setupSandboxAction}" action="dialog:snapshotSandbox" /> &nbsp;&nbsp;
<a:actionLink id="actBrowse" value="#{msg.sandbox_browse}" image="/images/icons/space_small.gif" showLink="false" actionListener="#{AVMBrowseBean.setupSandboxAction}" action="browseSandbox" /> <a:actionLink id="actPreview" value="#{msg.sandbox_preview}" image="/images/icons/preview_website.gif" href="#{AVMBrowseBean.stagingPreviewUrl}" target="new" />
<%-- Disabled action for GA
<a:actionLink id="actSnap" value="#{msg.sandbox_snapshot}" image="/images/icons/create_snapshot.gif" actionListener="#{AVMBrowseBean.setupSandboxAction}" action="dialog:snapshotSandbox" />
--%>
</td> </td>
</tr> </tr>
<tr> <tr>

View File

@@ -169,6 +169,50 @@
</table> </table>
</a:panel> </a:panel>
<div style="padding:4px"></div>
<a:panel label="#{msg.version_history}" id="version-history-panel" progressive="true"
border="white" bgcolor="white" titleBorder="blue" titleBgcolor="#D3E6FE"
expanded='#{FileDetailsBean.panels["version-history-panel"]}' expandedActionListener="#{FileDetailsBean.expandPanel}">
<a:richList id="version-history-list" viewMode="details" value="#{FileDetailsBean.versionHistory}"
var="r" styleClass="recordSet" headerStyleClass="recordSetHeader"
rowStyleClass="recordSetRow" altRowStyleClass="recordSetRowAlt" width="100%"
pageSize="10" initialSortColumn="version" initialSortDescending="true">
<%-- Primary column --%>
<a:column id="col1" primary="true" width="100" style="padding:2px;text-align:left">
<f:facet name="header">
<a:sortLink label="#{msg.version}" value="version" styleClass="header"/>
</f:facet>
<a:actionLink id="label" value="#{r.version}" href="#{r.url}" target="new" />
</a:column>
<%-- Modified Date column --%>
<a:column id="col2" style="text-align:left; white-space:nowrap">
<f:facet name="header">
<a:sortLink label="#{msg.modified_date}" value="modifiedDate" styleClass="header"/>
</f:facet>
<h:outputText id="date" value="#{r.modifiedDate}">
<a:convertXMLDate type="both" pattern="#{msg.date_time_pattern}" />
</h:outputText>
</a:column>
<%-- actions --%>
<a:column id="col3" style="text-align: left">
<f:facet name="header">
<h:outputText value="#{msg.actions}"/>
</f:facet>
<a:actionLink id="view-link" value="#{msg.view}" href="#{r.url}" target="new" />
<a:actionLink id="revert-link" value="#{msg.revert}" style="padding-left:4px" actionListener="#{FileDetailsBean.revertNode}">
<f:param name="version" value="#{r.strVersion}" />
</a:actionLink>
</a:column>
<a:dataPager styleClass="pager" />
</a:richList>
</a:panel>
</td> </td>
<td valign="top"> <td valign="top">