我正在尝试在JSF 1.2中扩展NavigationHandler,并保留一堆被访问的页面(更准确地说,是视图ID,以及动作和输出)。
我尝试实现自定义操作,例如“go_back”,这会将我带回上一页。
我当前的NavigationHandler:
import javax.faces.application.NavigationHandler;
import javax.faces.context.FacesContext;
import java.util.Stack;
public class NavigationManager extends NavigationHandler {
NavigationHandler _base;
private Stack trailPointStack;
public NavigationManager(NavigationHandler base) {
super();
_base = base;
trailPointStack = new Stack();
}
@Override
public void handleNavigation(FacesContext fc, String actionMethod, String actionName) {
NavigationPoint point;
String currentAction = actionName;
String currentMethod = actionMethod;
if (actionName.equals("go_back") && (trailPointStack.size() > 0)) {
point = (NavigationPoint) trailPointStack.pop();//better check if there is something in there
//currentAction = null;
//currentMethod = null;
currentAction = point.getAction();
currentMethod = point.getActionMethod();
fc.getViewRoot().setViewId(point.getViewId());
} else {
point = new NavigationPoint(actionName, actionMethod, fc.getViewRoot().getViewId());
trailPointStack.push(point);
}
//check stack size to be less than 6 items
while (trailPointStack.size() > 5) {
trailPointStack.removeElementAt(0);
}
_base.handleNavigation(fc, currentMethod, currentAction);
}
}
NavigationPoint只是一个包含3个字符串的简单类,用于actionName,actionMethod和ViewId。
我的导航规则,在faces-config.xml中:
<navigation-rule>
<description>Index to subpages</description>
<from-view-id>/index.jsp</from-view-id>
<navigation-case>
<from-outcome>i_to_1</from-outcome>
<to-view-id>/page_a1.jsp</to-view-id>
<redirect />
</navigation-case>
<navigation-case>
<from-outcome>to_page_a2</from-outcome>
<to-view-id>/page_a2.jsp</to-view-id>
<redirect />
</navigation-case>
</navigation-rule>
<navigation-rule>
<description>From page a1</description>
<from-view-id>/page_a1.jsp</from-view-id>
<navigation-case>
<from-outcome>to_page_a2</from-outcome>
<to-view-id>/page_a2.jsp</to-view-id>
<redirect />
</navigation-case>
</navigation-rule>
<navigation-rule>
<description>From page a2</description>
<from-view-id>/page_a2.jsp</from-view-id>
<navigation-case>
<from-outcome>to_page_a1</from-outcome>
<to-view-id>/page_a1.jsp</to-view-id>
<redirect />
</navigation-case>
<navigation-case>
<from-outcome>to_index</from-outcome>
<to-view-id>/index.jsp</to-view-id>
<redirect />
</navigation-case>
</navigation-rule>
我只有3个页面,index.jsp,page_a1.jsp和page_a2.jsp。
您可以在导航案例中看到它们之间的连接。我想要的是能够从page_a2.jsp“go_back”到page_a1.jsp或index.jsp。
正常导航工作正常:索引 - &gt; P1 - &gt; P2 - &gt; P1 - &gt; P2 - &gt;指数;没问题。
如果我这样做:索引 - &gt; P1 - &gt; P2
我会在堆栈上:
底
1:index.jsp / i_to_1 - &gt; page_a1.jsp
2:page_a1.jsp / to_page_a2 - &gt; page_a2.jsp
顶
当我从P2尝试“go_back”时,我希望它返回到第1页。它不会(页面刚重新加载)。 如果我第二次尝试,它会起作用。
我认为这是因为在第一次尝试时,我从堆栈中弹出,并尝试使用“to_page_a2”操作 - 失败。 第二次,它从堆栈中再次弹出,但现在它尝试使用“i_to_1”,而且......不知怎的,它可以工作。
任何人都可以帮我吗?我希望我的解释足够清楚 - 如果没有,请问。
也欢迎任何类似的想法。我应该提到的是,2天前我开始使用JSF,而且我不太清楚那里发生的一切。
谢谢你, 亚历
答案 0 :(得分:1)
好的,我找到了一个解决方案(或者,更好的说,一个补丁,因为我不确定它是否工作得太好了。)
在我执行“go_back”之前,我正在弹出2个项目,而不是一个。
Ex:Index - &gt; P1 - &gt; P2 - go_back;我应该去P1(P2 - &gt; P1)
为了能够达到P1,我实际上必须进入“索引”点,然后尝试去P1。 (我必须在前一步前往我想要回归的页面):
筹码:
1:pop P1(我想要它,所以我不能使用它;我只是从堆栈中删除它)
2:pop index - 使用Index point转到P1(即使我实际上在P2中)
更新的代码:
import javax.faces.application.NavigationHandler;
import javax.faces.context.FacesContext;
import java.util.Stack;
public class NavigationManager extends NavigationHandler {
NavigationHandler _base;
private Stack trailPointStack;
public NavigationManager(NavigationHandler base) {
super();
_base = base;
trailPointStack = new Stack();
}
@Override
public void handleNavigation(FacesContext fc, String actionMethod, String actionName) {
NavigationPoint point;
String currentAction = actionName;
String currentMethod = actionMethod;
if (actionName.equals("go_back") && (trailPointStack.size() > 0)) {//"go_back" is the action that triggers the back move
if (trailPointStack.size() == 1) {
//It is the first page we visit (the second page in stack)
point = (NavigationPoint) trailPointStack.pop();
currentAction = null;
currentMethod = point.getActionMethod();
} else {
trailPointStack.pop();//we get rid of the prev, and we go from prev-prev to prev :D
point = (NavigationPoint) trailPointStack.pop();//actually the second one in the stack
currentAction = point.getAction();
currentMethod = point.getActionMethod();
}
fc.getViewRoot().setViewId(point.getViewId());
} else {
point = new NavigationPoint(actionName, actionMethod, fc.getViewRoot().getViewId());
trailPointStack.push(point);
}
//check stack size to be max 10 items
while (trailPointStack.size() > 10) {
trailPointStack.removeElementAt(0);
}
_base.handleNavigation(fc, currentMethod, currentAction);
}
}
我希望这有助于某人!
另外,我确信这可能效果不好,我还没有测试太多。
欢迎任何其他想法。
感谢
亚历
答案 1 :(得分:0)
我已经实现了类似的东西; 我还在javascript中拦截了“退格”键,以便单击隐藏的commandLink,触发导航到堆栈中的上一个视图(或您需要的任何组合/逻辑)。
在我的用例中,我已经扁平化了导航案例的逻辑,以便在代码中更轻松地从一点到另一点导航:
public static void navigate(String action, String outcome) {
FacesContext fc = FacesContext.getCurrentInstance();
NavigationHandler nh = fc.getApplication().getNavigationHandler();
nh.handleNavigation(fc, action, outcome);
}