James Moger
2014-09-25 54cc7d7c2483d7ca100a5db47f4e1e98bd97c7fe
commit | author | age
859deb 1 /*
JM 2  * Copyright 2011 gitblit.com.
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 com.gitblit.wicket.pages;
17
18 import java.text.MessageFormat;
19 import java.util.ArrayList;
20 import java.util.Arrays;
21 import java.util.Calendar;
22 import java.util.Collections;
23 import java.util.Date;
24 import java.util.HashMap;
25 import java.util.HashSet;
26 import java.util.LinkedHashSet;
27 import java.util.List;
28 import java.util.Map;
29 import java.util.Set;
30 import java.util.TreeSet;
31 import java.util.concurrent.atomic.AtomicInteger;
32 import java.util.regex.Pattern;
33
34 import org.apache.wicket.MarkupContainer;
35 import org.apache.wicket.PageParameters;
36 import org.apache.wicket.behavior.HeaderContributor;
37 import org.apache.wicket.markup.html.IHeaderContributor;
38 import org.apache.wicket.markup.html.IHeaderResponse;
39 import org.apache.wicket.markup.html.basic.Label;
40 import org.apache.wicket.markup.html.form.PasswordTextField;
41 import org.apache.wicket.markup.html.form.TextField;
42 import org.apache.wicket.markup.html.link.BookmarkablePageLink;
43 import org.apache.wicket.markup.html.panel.Fragment;
44 import org.apache.wicket.markup.repeater.Item;
45 import org.apache.wicket.markup.repeater.data.DataView;
46 import org.apache.wicket.markup.repeater.data.ListDataProvider;
47 import org.apache.wicket.model.IModel;
48 import org.apache.wicket.model.Model;
ec7ed8 49 import org.apache.wicket.protocol.http.WebRequest;
859deb 50 import org.apache.wicket.protocol.http.WebResponse;
JM 51
52 import com.gitblit.Constants;
53 import com.gitblit.Keys;
7a401a 54 import com.gitblit.extensions.NavLinkExtension;
859deb 55 import com.gitblit.extensions.UserMenuExtension;
JM 56 import com.gitblit.models.Menu.ExternalLinkMenuItem;
57 import com.gitblit.models.Menu.MenuDivider;
58 import com.gitblit.models.Menu.MenuItem;
59 import com.gitblit.models.Menu.PageLinkMenuItem;
60 import com.gitblit.models.Menu.ParameterMenuItem;
61 import com.gitblit.models.Menu.ToggleMenuItem;
7a401a 62 import com.gitblit.models.NavLink;
JM 63 import com.gitblit.models.NavLink.PageNavLink;
859deb 64 import com.gitblit.models.RepositoryModel;
JM 65 import com.gitblit.models.TeamModel;
66 import com.gitblit.models.UserModel;
67 import com.gitblit.utils.ModelUtils;
68 import com.gitblit.utils.StringUtils;
69 import com.gitblit.wicket.GitBlitWebSession;
70 import com.gitblit.wicket.SessionlessForm;
71 import com.gitblit.wicket.WicketUtils;
72 import com.gitblit.wicket.panels.GravatarImage;
73 import com.gitblit.wicket.panels.LinkPanel;
74 import com.gitblit.wicket.panels.NavigationPanel;
75
76 /**
77  * Root page is a topbar, navigable page like Repositories, Users, or
78  * Federation.
79  *
80  * @author James Moger
81  *
82  */
83 public abstract class RootPage extends BasePage {
84
85     boolean showAdmin;
86
87     IModel<String> username = new Model<String>("");
88     IModel<String> password = new Model<String>("");
89     List<RepositoryModel> repositoryModels = new ArrayList<RepositoryModel>();
90
91     public RootPage() {
92         super();
93     }
94
95     public RootPage(PageParameters params) {
96         super(params);
97     }
98
99     @Override
100     protected void setupPage(String repositoryName, String pageName) {
101
102         // CSS header overrides
103         add(new HeaderContributor(new IHeaderContributor() {
104             private static final long serialVersionUID = 1L;
105
106             @Override
107             public void renderHead(IHeaderResponse response) {
108                 StringBuilder buffer = new StringBuilder();
109                 buffer.append("<style type=\"text/css\">\n");
110                 buffer.append(".navbar-inner {\n");
111                 final String headerBackground = app().settings().getString(Keys.web.headerBackgroundColor, null);
112                 if (!StringUtils.isEmpty(headerBackground)) {
113                     buffer.append(MessageFormat.format("background-color: {0};\n", headerBackground));
114                 }
115                 final String headerBorder = app().settings().getString(Keys.web.headerBorderColor, null);
116                 if (!StringUtils.isEmpty(headerBorder)) {
117                     buffer.append(MessageFormat.format("border-bottom: 1px solid {0} !important;\n", headerBorder));
118                 }
119                 buffer.append("}\n");
120                 final String headerBorderFocus = app().settings().getString(Keys.web.headerBorderFocusColor, null);
121                 if (!StringUtils.isEmpty(headerBorderFocus)) {
122                     buffer.append(".navbar ul li:focus, .navbar .active {\n");
123                     buffer.append(MessageFormat.format("border-bottom: 4px solid {0};\n", headerBorderFocus));
124                     buffer.append("}\n");
125                 }
126                 final String headerForeground = app().settings().getString(Keys.web.headerForegroundColor, null);
127                 if (!StringUtils.isEmpty(headerForeground)) {
128                     buffer.append(".navbar ul.nav li a {\n");
129                     buffer.append(MessageFormat.format("color: {0};\n", headerForeground));
130                     buffer.append("}\n");
131                     buffer.append(".navbar ul.nav .active a {\n");
132                     buffer.append(MessageFormat.format("color: {0};\n", headerForeground));
133                     buffer.append("}\n");
134                 }
135                 final String headerHover = app().settings().getString(Keys.web.headerHoverColor, null);
136                 if (!StringUtils.isEmpty(headerHover)) {
137                     buffer.append(".navbar ul.nav li a:hover {\n");
138                     buffer.append(MessageFormat.format("color: {0} !important;\n", headerHover));
139                     buffer.append("}\n");
140                 }
141                 buffer.append("</style>\n");
142                 response.renderString(buffer.toString());
143                 }
144             }));
145
146         boolean authenticateView = app().settings().getBoolean(Keys.web.authenticateViewPages, false);
147         boolean authenticateAdmin = app().settings().getBoolean(Keys.web.authenticateAdminPages, true);
148         boolean allowAdmin = app().settings().getBoolean(Keys.web.allowAdministration, true);
149         boolean allowLucene = app().settings().getBoolean(Keys.web.allowLuceneIndexing, true);
150         boolean isLoggedIn = GitBlitWebSession.get().isLoggedIn();
151
152         if (authenticateAdmin) {
153             showAdmin = allowAdmin && GitBlitWebSession.get().canAdmin();
154             // authentication requires state and session
155             setStatelessHint(false);
156         } else {
157             showAdmin = allowAdmin;
158             if (authenticateView) {
159                 // authentication requires state and session
160                 setStatelessHint(false);
161             } else {
162                 // no authentication required, no state and no session required
163                 setStatelessHint(true);
164             }
165         }
166
167         if (authenticateView || authenticateAdmin) {
168             if (isLoggedIn) {
169                 UserMenu userFragment = new UserMenu("userPanel", "userMenuFragment", RootPage.this);
170                 add(userFragment);
171             } else {
172                 LoginForm loginForm = new LoginForm("userPanel", "loginFormFragment", RootPage.this);
173                 add(loginForm);
174             }
175         } else {
176             add(new Label("userPanel").setVisible(false));
177         }
178
179         // navigation links
7a401a 180         List<NavLink> navLinks = new ArrayList<NavLink>();
859deb 181         if (!authenticateView || (authenticateView && isLoggedIn)) {
7a401a 182             navLinks.add(new PageNavLink(isLoggedIn ? "gb.myDashboard" : "gb.dashboard", MyDashboardPage.class,
859deb 183                     getRootPageParameters()));
JM 184             if (isLoggedIn && app().tickets().isReady()) {
7a401a 185                 navLinks.add(new PageNavLink("gb.myTickets", MyTicketsPage.class));
859deb 186             }
7a401a 187             navLinks.add(new PageNavLink("gb.repositories", RepositoriesPage.class,
859deb 188                     getRootPageParameters()));
7a401a 189             navLinks.add(new PageNavLink("gb.activity", ActivityPage.class, getRootPageParameters()));
859deb 190             if (allowLucene) {
7a401a 191                 navLinks.add(new PageNavLink("gb.search", LuceneSearchPage.class));
859deb 192             }
JM 193
194             if (!authenticateView || (authenticateView && isLoggedIn)) {
7a401a 195                 addDropDownMenus(navLinks);
JM 196             }
197
198             UserModel user = UserModel.ANONYMOUS;
199             if (isLoggedIn) {
200                 user = GitBlitWebSession.get().getUser();
201             }
202
203             // add nav link extensions
204             List<NavLinkExtension> extensions = app().plugins().getExtensions(NavLinkExtension.class);
205             for (NavLinkExtension ext : extensions) {
206                 navLinks.addAll(ext.getNavLinks(user));
859deb 207             }
JM 208         }
209
7a401a 210         NavigationPanel navPanel = new NavigationPanel("navPanel", getRootNavPageClass(), navLinks);
859deb 211         add(navPanel);
JM 212
213         // display an error message cached from a redirect
214         String cachedMessage = GitBlitWebSession.get().clearErrorMessage();
215         if (!StringUtils.isEmpty(cachedMessage)) {
216             error(cachedMessage);
217         } else if (showAdmin) {
218             int pendingProposals = app().federation().getPendingFederationProposals().size();
219             if (pendingProposals == 1) {
220                 info(getString("gb.OneProposalToReview"));
221             } else if (pendingProposals > 1) {
222                 info(MessageFormat.format(getString("gb.nFederationProposalsToReview"),
223                         pendingProposals));
224             }
225         }
226
227         super.setupPage(repositoryName, pageName);
228     }
229
230     protected Class<? extends BasePage> getRootNavPageClass() {
231         return getClass();
232     }
233
234     private PageParameters getRootPageParameters() {
235         if (reusePageParameters()) {
236             PageParameters pp = getPageParameters();
237             if (pp != null) {
238                 PageParameters params = new PageParameters(pp);
239                 // remove named project parameter
240                 params.remove("p");
241
242                 // remove named repository parameter
243                 params.remove("r");
244
245                 // remove named user parameter
246                 params.remove("user");
247
248                 // remove days back parameter if it is the default value
249                 if (params.containsKey("db")
250                         && params.getInt("db") == app().settings().getInteger(Keys.web.activityDuration, 7)) {
251                     params.remove("db");
252                 }
253                 return params;
254             }
255         }
256         return null;
257     }
258
259     protected boolean reusePageParameters() {
260         return false;
261     }
262
263     private void loginUser(UserModel user) {
264         if (user != null) {
265             // Set the user into the session
266             GitBlitWebSession session = GitBlitWebSession.get();
267             // issue 62: fix session fixation vulnerability
268             session.replaceSession();
269             session.setUser(user);
270
271             // Set Cookie
272             if (app().settings().getBoolean(Keys.web.allowCookieAuthentication, false)) {
ec7ed8 273                 WebRequest request = (WebRequest) getRequestCycle().getRequest();
859deb 274                 WebResponse response = (WebResponse) getRequestCycle().getResponse();
ec7ed8 275                 app().authentication().setCookie(request.getHttpServletRequest(),
JM 276                         response.getHttpServletResponse(), user);
859deb 277             }
JM 278
279             if (!session.continueRequest()) {
280                 PageParameters params = getPageParameters();
281                 if (params == null) {
282                     // redirect to this page
283                     setResponsePage(getClass());
284                 } else {
285                     // Strip username and password and redirect to this page
286                     params.remove("username");
287                     params.remove("password");
288                     setResponsePage(getClass(), params);
289                 }
290             }
291         }
292     }
293
294     protected List<RepositoryModel> getRepositoryModels() {
295         if (repositoryModels.isEmpty()) {
296             final UserModel user = GitBlitWebSession.get().getUser();
297             List<RepositoryModel> repositories = app().repositories().getRepositoryModels(user);
298             repositoryModels.addAll(repositories);
299             Collections.sort(repositoryModels);
300         }
301         return repositoryModels;
302     }
303
7a401a 304     protected void addDropDownMenus(List<NavLink> navLinks) {
859deb 305
JM 306     }
307
308     protected List<com.gitblit.models.Menu.MenuItem> getRepositoryFilterItems(PageParameters params) {
309         final UserModel user = GitBlitWebSession.get().getUser();
310         Set<MenuItem> filters = new LinkedHashSet<MenuItem>();
311         List<RepositoryModel> repositories = getRepositoryModels();
312
313         // accessible repositories by federation set
314         Map<String, AtomicInteger> setMap = new HashMap<String, AtomicInteger>();
315         for (RepositoryModel repository : repositories) {
316             for (String set : repository.federationSets) {
317                 String key = set.toLowerCase();
318                 if (setMap.containsKey(key)) {
319                     setMap.get(key).incrementAndGet();
320                 } else {
321                     setMap.put(key, new AtomicInteger(1));
322                 }
323             }
324         }
325         if (setMap.size() > 0) {
326             List<String> sets = new ArrayList<String>(setMap.keySet());
327             Collections.sort(sets);
328             for (String set : sets) {
329                 filters.add(new ToggleMenuItem(MessageFormat.format("{0} ({1})", set,
330                         setMap.get(set).get()), "set", set, params));
331             }
332             // divider
333             filters.add(new MenuDivider());
334         }
335
336         // user's team memberships
337         if (user != null && user.teams.size() > 0) {
338             List<TeamModel> teams = new ArrayList<TeamModel>(user.teams);
339             Collections.sort(teams);
340             for (TeamModel team : teams) {
341                 filters.add(new ToggleMenuItem(MessageFormat.format("{0} ({1})", team.name,
342                         team.repositories.size()), "team", team.name, params));
343             }
344             // divider
345             filters.add(new MenuDivider());
346         }
347
348         // custom filters
349         String customFilters = app().settings().getString(Keys.web.customFilters, null);
350         if (!StringUtils.isEmpty(customFilters)) {
351             boolean addedExpression = false;
352             List<String> expressions = StringUtils.getStringsFromValue(customFilters, "!!!");
353             for (String expression : expressions) {
354                 if (!StringUtils.isEmpty(expression)) {
355                     addedExpression = true;
356                     filters.add(new ToggleMenuItem(null, "x", expression, params));
357                 }
358             }
359             // if we added any custom expressions, add a divider
360             if (addedExpression) {
361                 filters.add(new MenuDivider());
362             }
363         }
364         return new ArrayList<MenuItem>(filters);
365     }
366
367     protected List<MenuItem> getTimeFilterItems(PageParameters params) {
368         // days back choices - additive parameters
369         int daysBack = app().settings().getInteger(Keys.web.activityDuration, 7);
370         int maxDaysBack = app().settings().getInteger(Keys.web.activityDurationMaximum, 30);
371         if (daysBack < 1) {
372             daysBack = 7;
373         }
374         if (daysBack > maxDaysBack) {
375             daysBack = maxDaysBack;
376         }
377         PageParameters clonedParams;
378         if (params == null) {
379             clonedParams = new PageParameters();
380         } else {
381             clonedParams = new PageParameters(params);
382         }
383
384         if (!clonedParams.containsKey("db")) {
385             clonedParams.put("db",  daysBack);
386         }
387
388         List<MenuItem> items = new ArrayList<MenuItem>();
389         Set<Integer> choicesSet = new TreeSet<Integer>(app().settings().getIntegers(Keys.web.activityDurationChoices));
390         if (choicesSet.isEmpty()) {
391              choicesSet.addAll(Arrays.asList(1, 3, 7, 14, 21, 28));
392         }
393         List<Integer> choices = new ArrayList<Integer>(choicesSet);
394         Collections.sort(choices);
395         String lastDaysPattern = getString("gb.lastNDays");
396         for (Integer db : choices) {
397             if (db == 1) {
398                 items.add(new ParameterMenuItem(getString("gb.time.today"), "db", db.toString(), clonedParams));
399             } else {
400                 String txt = MessageFormat.format(lastDaysPattern, db);
401                 items.add(new ParameterMenuItem(txt, "db", db.toString(), clonedParams));
402             }
403         }
404         items.add(new MenuDivider());
405         return items;
406     }
407
408     protected List<RepositoryModel> getRepositories(PageParameters params) {
409         if (params == null) {
410             return getRepositoryModels();
411         }
412
413         boolean hasParameter = false;
414         String projectName = WicketUtils.getProjectName(params);
415         String userName = WicketUtils.getUsername(params);
416         if (StringUtils.isEmpty(projectName)) {
417             if (!StringUtils.isEmpty(userName)) {
418                 projectName = ModelUtils.getPersonalPath(userName);
419             }
420         }
421         String repositoryName = WicketUtils.getRepositoryName(params);
422         String set = WicketUtils.getSet(params);
423         String regex = WicketUtils.getRegEx(params);
424         String team = WicketUtils.getTeam(params);
425         int daysBack = params.getInt("db", 0);
426         int maxDaysBack = app().settings().getInteger(Keys.web.activityDurationMaximum, 30);
427
428         List<RepositoryModel> availableModels = getRepositoryModels();
429         Set<RepositoryModel> models = new HashSet<RepositoryModel>();
430
431         if (!StringUtils.isEmpty(repositoryName)) {
432             // try named repository
433             hasParameter = true;
434             for (RepositoryModel model : availableModels) {
435                 if (model.name.equalsIgnoreCase(repositoryName)) {
436                     models.add(model);
437                     break;
438                 }
439             }
440         }
441
442         if (!StringUtils.isEmpty(projectName)) {
443             // try named project
444             hasParameter = true;
445             if (projectName.equalsIgnoreCase(app().settings().getString(Keys.web.repositoryRootGroupName, "main"))) {
446                 // root project/group
447                 for (RepositoryModel model : availableModels) {
448                     if (model.name.indexOf('/') == -1) {
449                         models.add(model);
450                     }
451                 }
452             } else {
453                 // named project/group
454                 String group = projectName.toLowerCase() + "/";
455                 for (RepositoryModel model : availableModels) {
456                     if (model.name.toLowerCase().startsWith(group)) {
457                         models.add(model);
458                     }
459                 }
460             }
461         }
462
463         if (!StringUtils.isEmpty(regex)) {
464             // filter the repositories by the regex
465             hasParameter = true;
466             Pattern pattern = Pattern.compile(regex);
467             for (RepositoryModel model : availableModels) {
468                 if (pattern.matcher(model.name).find()) {
469                     models.add(model);
470                 }
471             }
472         }
473
474         if (!StringUtils.isEmpty(set)) {
475             // filter the repositories by the specified sets
476             hasParameter = true;
477             List<String> sets = StringUtils.getStringsFromValue(set, ",");
478             for (RepositoryModel model : availableModels) {
479                 for (String curr : sets) {
480                     if (model.federationSets.contains(curr)) {
481                         models.add(model);
482                     }
483                 }
484             }
485         }
486
487         if (!StringUtils.isEmpty(team)) {
488             // filter the repositories by the specified teams
489             hasParameter = true;
490             List<String> teams = StringUtils.getStringsFromValue(team, ",");
491
492             // need TeamModels first
493             List<TeamModel> teamModels = new ArrayList<TeamModel>();
494             for (String name : teams) {
495                 TeamModel teamModel = app().users().getTeamModel(name);
496                 if (teamModel != null) {
497                     teamModels.add(teamModel);
498                 }
499             }
500
501             // brute-force our way through finding the matching models
502             for (RepositoryModel repositoryModel : availableModels) {
503                 for (TeamModel teamModel : teamModels) {
504                     if (teamModel.hasRepositoryPermission(repositoryModel.name)) {
505                         models.add(repositoryModel);
506                     }
507                 }
508             }
509         }
510
511         if (!hasParameter) {
512             models.addAll(availableModels);
513         }
514
515         // time-filter the list
516         if (daysBack > 0) {
517             if (maxDaysBack > 0 && daysBack > maxDaysBack) {
518                 daysBack = maxDaysBack;
519             }
520             Calendar cal = Calendar.getInstance();
521             cal.set(Calendar.HOUR_OF_DAY, 0);
522             cal.set(Calendar.MINUTE, 0);
523             cal.set(Calendar.SECOND, 0);
524             cal.set(Calendar.MILLISECOND, 0);
525             cal.add(Calendar.DATE, -1 * daysBack);
526             Date threshold = cal.getTime();
527             Set<RepositoryModel> timeFiltered = new HashSet<RepositoryModel>();
528             for (RepositoryModel model : models) {
529                 if (model.lastChange.after(threshold)) {
530                     timeFiltered.add(model);
531                 }
532             }
533             models = timeFiltered;
534         }
535
536         List<RepositoryModel> list = new ArrayList<RepositoryModel>(models);
537         Collections.sort(list);
538         return list;
539     }
540
541     /**
542      * Inline login form.
543      */
544     private class LoginForm extends Fragment {
545         private static final long serialVersionUID = 1L;
546
547         public LoginForm(String id, String markupId, MarkupContainer markupProvider) {
548             super(id, markupId, markupProvider);
549             setRenderBodyOnly(true);
550
551             SessionlessForm<Void> loginForm = new SessionlessForm<Void>("loginForm", RootPage.this.getClass(), getPageParameters()) {
552
553                 private static final long serialVersionUID = 1L;
554
555                 @Override
556                 public void onSubmit() {
557                     String username = RootPage.this.username.getObject();
558                     char[] password = RootPage.this.password.getObject().toCharArray();
559
560                     UserModel user = app().authentication().authenticate(username, password);
561                     if (user == null) {
562                         error(getString("gb.invalidUsernameOrPassword"));
563                     } else if (user.username.equals(Constants.FEDERATION_USER)) {
564                         // disallow the federation user from logging in via the
565                         // web ui
566                         error(getString("gb.invalidUsernameOrPassword"));
567                         user = null;
568                     } else {
569                         loginUser(user);
570                     }
571                 }
572             };
573             TextField<String> unameField = new TextField<String>("username", username);
574             WicketUtils.setInputPlaceholder(unameField, markupProvider.getString("gb.username"));
575             loginForm.add(unameField);
576             PasswordTextField pwField = new PasswordTextField("password", password);
577             WicketUtils.setInputPlaceholder(pwField, markupProvider.getString("gb.password"));
578             loginForm.add(pwField);
579             add(loginForm);
580         }
581     }
582
583     /**
584      * Menu for the authenticated user.
585      */
586     class UserMenu extends Fragment {
587
588         private static final long serialVersionUID = 1L;
589
590         public UserMenu(String id, String markupId, MarkupContainer markupProvider) {
591             super(id, markupId, markupProvider);
592             setRenderBodyOnly(true);
593         }
594
595         @Override
596         protected void onInitialize() {
597             super.onInitialize();
598
599             GitBlitWebSession session = GitBlitWebSession.get();
600             UserModel user = session.getUser();
601             boolean editCredentials = app().authentication().supportsCredentialChanges(user);
602             boolean standardLogin = session.authenticationType.isStandard();
603
604             if (app().settings().getBoolean(Keys.web.allowGravatar, true)) {
605                 add(new GravatarImage("username", user, "navbarGravatar", 20, false));
606             } else {
607                 add(new Label("username", user.getDisplayName()));
608             }
609
610             List<MenuItem> standardItems = new ArrayList<MenuItem>();
611             standardItems.add(new MenuDivider());
612             if (user.canAdmin() || user.canCreate()) {
0047fb 613                 standardItems.add(new PageLinkMenuItem("gb.newRepository", app().getNewRepositoryPage()));
859deb 614             }
JM 615             standardItems.add(new PageLinkMenuItem("gb.myProfile", UserPage.class,
616                     WicketUtils.newUsernameParameter(user.username)));
617             if (editCredentials) {
618                 standardItems.add(new PageLinkMenuItem("gb.changePassword", ChangePasswordPage.class));
619             }
620             standardItems.add(new MenuDivider());
621             add(newSubmenu("standardMenu", user.getDisplayName(), standardItems));
622
623             if (showAdmin) {
624                 // admin menu
625                 List<MenuItem> adminItems = new ArrayList<MenuItem>();
626                 adminItems.add(new MenuDivider());
627                 adminItems.add(new PageLinkMenuItem("gb.users", UsersPage.class));
628                 adminItems.add(new PageLinkMenuItem("gb.teams", TeamsPage.class));
629
630                 boolean showRegistrations = app().federation().canFederate()
631                         && app().settings().getBoolean(Keys.web.showFederationRegistrations, false);
632                 if (showRegistrations) {
633                     adminItems.add(new PageLinkMenuItem("gb.federation", FederationPage.class));
634                 }
635                 adminItems.add(new MenuDivider());
636
637                 add(newSubmenu("adminMenu", getString("gb.administration"), adminItems));
638             } else {
639                 add(new Label("adminMenu").setVisible(false));
640             }
641
642             // plugin extension items
643             List<MenuItem> extensionItems = new ArrayList<MenuItem>();
644             List<UserMenuExtension> extensions = app().plugins().getExtensions(UserMenuExtension.class);
645             for (UserMenuExtension ext : extensions) {
646                 List<MenuItem> items = ext.getMenuItems(user);
647                 extensionItems.addAll(items);
648             }
649
650             if (extensionItems.isEmpty()) {
651                 // no extension items
652                 add(new Label("extensionsMenu").setVisible(false));
653             } else {
654                 // found extension items
655                 extensionItems.add(0, new MenuDivider());
656                 add(newSubmenu("extensionsMenu", getString("gb.extensions"), extensionItems));
657                 extensionItems.add(new MenuDivider());
658             }
659
660             add(new BookmarkablePageLink<Void>("logout",
661                     LogoutPage.class).setVisible(standardLogin));
662         }
663
664         /**
665          * Creates a submenu.  This is not actually submenu because we're using
666          * an older Twitter Bootstrap which is pre-submenu.
667          *
668          * @param wicketId
669          * @param submenuTitle
670          * @param menuItems
671          * @return a submenu fragment
672          */
673         private Fragment newSubmenu(String wicketId, String submenuTitle, List<MenuItem> menuItems) {
674             Fragment submenu = new Fragment(wicketId, "submenuFragment", this);
675             submenu.add(new Label("submenuTitle", submenuTitle).setRenderBodyOnly(true));
676             ListDataProvider<MenuItem> menuItemsDp = new ListDataProvider<MenuItem>(menuItems);
677             DataView<MenuItem> submenuItems = new DataView<MenuItem>("submenuItem", menuItemsDp) {
678                 private static final long serialVersionUID = 1L;
679
680                 @Override
681                 public void populateItem(final Item<MenuItem> menuItem) {
682                     final MenuItem item = menuItem.getModelObject();
683                     String name = item.toString();
684                     try {
685                         // try to lookup translation
686                         name = getString(name);
687                     } catch (Exception e) {
688                     }
689                     if (item instanceof PageLinkMenuItem) {
690                         // link to another Wicket page
691                         PageLinkMenuItem pageLink = (PageLinkMenuItem) item;
692                         menuItem.add(new LinkPanel("submenuLink", null, null, name, pageLink.getPageClass(),
693                                 pageLink.getPageParameters(), false).setRenderBodyOnly(true));
694                     } else if (item instanceof ExternalLinkMenuItem) {
695                         // link to a specified href
696                         ExternalLinkMenuItem extLink = (ExternalLinkMenuItem) item;
697                         menuItem.add(new LinkPanel("submenuLink", null, name, extLink.getHref(),
698                                 extLink.openInNewWindow()).setRenderBodyOnly(true));
699                     } else if (item instanceof MenuDivider) {
700                         // divider
701                         menuItem.add(new Label("submenuLink").setRenderBodyOnly(true));
702                         WicketUtils.setCssClass(menuItem, "divider");
703                     }
704                 }
705             };
706             submenu.add(submenuItems);
707             submenu.setRenderBodyOnly(true);
708             return submenu;
709         }
710     }
711 }