James Moger
2013-12-09 5e3521f8496511db4df45f011ea72f25623ad90f
src/main/java/com/gitblit/wicket/pages/TicketsPage.java
@@ -1,5 +1,5 @@
/*
 * Copyright 2011 gitblit.com.
 * Copyright 2013 gitblit.com.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
@@ -15,58 +15,864 @@
 */
package com.gitblit.wicket.pages;
import java.io.Serializable;
import java.text.MessageFormat;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Set;
import java.util.TreeSet;
import org.apache.wicket.Component;
import org.apache.wicket.PageParameters;
import org.apache.wicket.behavior.SimpleAttributeModifier;
import org.apache.wicket.markup.html.basic.Label;
import org.apache.wicket.markup.html.form.TextField;
import org.apache.wicket.markup.html.link.BookmarkablePageLink;
import org.apache.wicket.markup.html.panel.Fragment;
import org.apache.wicket.markup.repeater.Item;
import org.apache.wicket.markup.repeater.data.DataView;
import org.apache.wicket.markup.repeater.data.ListDataProvider;
import org.apache.wicket.model.IModel;
import org.apache.wicket.model.Model;
import org.apache.wicket.request.target.basic.RedirectRequestTarget;
import com.gitblit.Constants;
import com.gitblit.Constants.AccessPermission;
import com.gitblit.Keys;
import com.gitblit.models.RegistrantAccessPermission;
import com.gitblit.models.TicketModel;
import com.gitblit.models.TicketModel.Status;
import com.gitblit.models.UserModel;
import com.gitblit.tickets.QueryBuilder;
import com.gitblit.tickets.QueryResult;
import com.gitblit.tickets.TicketIndexer.Lucene;
import com.gitblit.tickets.TicketLabel;
import com.gitblit.tickets.TicketMilestone;
import com.gitblit.tickets.TicketResponsible;
import com.gitblit.utils.ArrayUtils;
import com.gitblit.utils.StringUtils;
import com.gitblit.utils.TicgitUtils;
import com.gitblit.wicket.GitBlitWebSession;
import com.gitblit.wicket.SessionlessForm;
import com.gitblit.wicket.WicketUtils;
import com.gitblit.wicket.panels.GravatarImage;
import com.gitblit.wicket.panels.LinkPanel;
public class TicketsPage extends RepositoryPage {
public class TicketsPage extends TicketBasePage {
   final TicketResponsible any;
   public static final String [] openStatii = new String [] { Status.New.name().toLowerCase(), Status.Open.name().toLowerCase() };
   public static final String [] closedStatii = new String [] { "!" + Status.New.name().toLowerCase(), "!" + Status.Open.name().toLowerCase() };
   public TicketsPage(PageParameters params) {
      super(params);
      List<TicketModel> tickets = TicgitUtils.getTickets(getRepository());
      if (!app().tickets().isReady()) {
         // tickets prohibited
         setResponsePage(SummaryPage.class, WicketUtils.newRepositoryParameter(repositoryName));
      } else if (!app().tickets().hasTickets(getRepositoryModel())) {
         // no tickets for this repository
         setResponsePage(NoTicketsPage.class, WicketUtils.newRepositoryParameter(repositoryName));
      } else {
         String id = WicketUtils.getObject(params);
         if (id != null) {
            // view the ticket with the TicketPage
            setResponsePage(TicketPage.class, params);
         }
      }
      ListDataProvider<TicketModel> ticketsDp = new ListDataProvider<TicketModel>(tickets);
      DataView<TicketModel> ticketsView = new DataView<TicketModel>("ticket", ticketsDp) {
      // set stateless page preference
      setStatelessHint(true);
      any = new TicketResponsible("any", "[* TO *]", null);
      UserModel user = GitBlitWebSession.get().getUser();
      boolean isAuthenticated = user != null && user.isAuthenticated;
      final String [] statiiParam = params.getStringArray(Lucene.status.name());
      final String assignedToParam = params.getString(Lucene.responsible.name(), null);
      final String milestoneParam = params.getString(Lucene.milestone.name(), null);
      final String queryParam = params.getString("q", null);
      final String searchParam = params.getString("s", null);
      final String sortBy = Lucene.fromString(params.getString("sort", Lucene.created.name())).name();
      final boolean desc = !"asc".equals(params.getString("direction", "desc"));
      // add search form
      TicketSearchForm searchForm = new TicketSearchForm("ticketSearchForm", repositoryName, searchParam);
      add(searchForm);
      searchForm.setTranslatedAttributes();
      final String activeQuery;
      if (!StringUtils.isEmpty(searchParam)) {
         activeQuery = searchParam;
      } else if (StringUtils.isEmpty(queryParam)) {
         activeQuery = "";
      } else {
         activeQuery = queryParam;
      }
      // build Lucene query from defaults and request parameters
      QueryBuilder qb = new QueryBuilder(queryParam);
      if (!qb.containsField(Lucene.rid.name())) {
         // specify the repository
         qb.and(Lucene.rid.matches(getRepositoryModel().getRID()));
      }
      if (!qb.containsField(Lucene.responsible.name())) {
         // specify the responsible
         qb.and(Lucene.responsible.matches(assignedToParam));
      }
      if (!qb.containsField(Lucene.milestone.name())) {
         // specify the milestone
         qb.and(Lucene.milestone.matches(milestoneParam));
      }
      if (!qb.containsField(Lucene.status.name()) && !ArrayUtils.isEmpty(statiiParam)) {
         // specify the states
         boolean not = false;
         QueryBuilder q = new QueryBuilder();
         for (String state : statiiParam) {
            if (state.charAt(0) == '!') {
               not = true;
               q.and(Lucene.status.doesNotMatch(state.substring(1)));
            } else {
               q.or(Lucene.status.matches(state));
            }
         }
         if (not) {
            qb.and(q.toString());
         } else {
            qb.and(q.toSubquery().toString());
         }
      }
      final String luceneQuery = qb.build();
      // open milestones
      List<TicketMilestone> milestones = app().tickets().getMilestones(getRepositoryModel(), Status.Open);
      TicketMilestone currentMilestone = null;
      if (!StringUtils.isEmpty(milestoneParam)) {
         for (TicketMilestone tm : milestones) {
            if (tm.name.equals(milestoneParam)) {
               // get the milestone (queries the index)
               currentMilestone = app().tickets().getMilestone(getRepositoryModel(), milestoneParam);
               break;
            }
         }
         if (currentMilestone == null) {
            // milestone not found, create a temporary one
            currentMilestone = new TicketMilestone(milestoneParam);
         }
      }
      Fragment milestonePanel;
      if (currentMilestone == null) {
         milestonePanel = new Fragment("milestonePanel", "noMilestoneFragment", this);
         add(milestonePanel);
      } else {
         milestonePanel = new Fragment("milestonePanel", "milestoneProgressFragment", this);
         milestonePanel.add(new Label("currentMilestone", currentMilestone.name));
         if (currentMilestone.due == null) {
            milestonePanel.add(new Label("currentDueDate", getString("gb.notSpecified")));
         } else {
            milestonePanel.add(WicketUtils.createDateLabel("currentDueDate", currentMilestone.due, GitBlitWebSession
                  .get().getTimezone(), getTimeUtils(), false));
         }
         Label label = new Label("progress");
         WicketUtils.setCssStyle(label, "width:" + currentMilestone.getProgress() + "%;");
         milestonePanel.add(label);
         milestonePanel.add(new LinkPanel("openTickets", null,
               currentMilestone.getOpenTickets() + " open",
               TicketsPage.class,
               queryParameters(null, currentMilestone.name, openStatii, null, sortBy, desc, 1)));
         milestonePanel.add(new LinkPanel("closedTickets", null,
               currentMilestone.getClosedTickets() + " closed",
               TicketsPage.class,
               queryParameters(null, currentMilestone.name, closedStatii, null, sortBy, desc, 1)));
         milestonePanel.add(new Label("totalTickets", currentMilestone.getTotalTickets() + " total"));
         add(milestonePanel);
      }
      Fragment milestoneDropdown = new Fragment("milestoneDropdown", "milestoneDropdownFragment", this);
      PageParameters resetMilestone = queryParameters(queryParam, null, statiiParam, assignedToParam, sortBy, desc, 1);
      milestoneDropdown.add(new BookmarkablePageLink<Void>("resetMilestone", TicketsPage.class, resetMilestone));
      ListDataProvider<TicketMilestone> milestonesDp = new ListDataProvider<TicketMilestone>(milestones);
      DataView<TicketMilestone> milestonesMenu = new DataView<TicketMilestone>("milestone", milestonesDp) {
         private static final long serialVersionUID = 1L;
         int counter;
         public void populateItem(final Item<TicketModel> item) {
            final TicketModel entry = item.getModelObject();
            Label stateLabel = new Label("ticketState", entry.state);
            WicketUtils.setTicketCssClass(stateLabel, entry.state);
            item.add(stateLabel);
            item.add(WicketUtils.createDateLabel("ticketDate", entry.date, GitBlitWebSession
                  .get().getTimezone(), getTimeUtils()));
            item.add(new Label("ticketHandler", StringUtils.trimString(
                  entry.handler.toLowerCase(), 30)));
            item.add(new LinkPanel("ticketTitle", "list subject", StringUtils.trimString(
                  entry.title, 80), TicketPage.class, newPathParameter(entry.name)));
         @Override
         public void populateItem(final Item<TicketMilestone> item) {
            final TicketMilestone tm = item.getModelObject();
            PageParameters params = queryParameters(queryParam, tm.name, statiiParam, assignedToParam, sortBy, desc, 1);
            item.add(new LinkPanel("milestoneLink", null, tm.name, TicketsPage.class, params).setRenderBodyOnly(true));
         }
      };
      milestoneDropdown.add(milestonesMenu);
      milestonePanel.add(milestoneDropdown);
            WicketUtils.setAlternatingBackground(item, counter);
            counter++;
      // search or query tickets
      int page = Math.max(1,  WicketUtils.getPage(params));
      int pageSize = app().settings().getInteger(Keys.tickets.perPage, 25);
      List<QueryResult> results;
      if (StringUtils.isEmpty(searchParam)) {
         results = app().tickets().queryFor(luceneQuery, page, pageSize, sortBy, desc);
      } else {
         results = app().tickets().searchFor(getRepositoryModel(), searchParam, page, pageSize);
      }
      int totalResults = results.size() == 0 ? 0 : results.get(0).totalResults;
      // standard queries
      add(new BookmarkablePageLink<Void>("changesQuery", TicketsPage.class,
            queryParameters(
                  Lucene.type.matches(TicketModel.Type.Proposal.name()),
                  milestoneParam,
                  statiiParam,
                  assignedToParam,
                  sortBy,
                  desc,
                  1)));
      add(new BookmarkablePageLink<Void>("bugsQuery", TicketsPage.class,
            queryParameters(
                  Lucene.type.matches(TicketModel.Type.Bug.name()),
                  milestoneParam,
                  statiiParam,
                  assignedToParam,
                  sortBy,
                  desc,
                  1)));
      add(new BookmarkablePageLink<Void>("enhancementsQuery", TicketsPage.class,
            queryParameters(
                  Lucene.type.matches(TicketModel.Type.Enhancement.name()),
                  milestoneParam,
                  statiiParam,
                  assignedToParam,
                  sortBy,
                  desc,
                  1)));
      add(new BookmarkablePageLink<Void>("tasksQuery", TicketsPage.class,
            queryParameters(
                  Lucene.type.matches(TicketModel.Type.Task.name()),
                  milestoneParam,
                  statiiParam,
                  assignedToParam,
                  sortBy,
                  desc,
                  1)));
      add(new BookmarkablePageLink<Void>("questionsQuery", TicketsPage.class,
            queryParameters(
                  Lucene.type.matches(TicketModel.Type.Question.name()),
                  milestoneParam,
                  statiiParam,
                  assignedToParam,
                  sortBy,
                  desc,
                  1)));
      add(new BookmarkablePageLink<Void>("resetQuery", TicketsPage.class,
            queryParameters(
                  null,
                  milestoneParam,
                  openStatii,
                  null,
                  null,
                  true,
                  1)));
      if (isAuthenticated) {
         add(new Label("userDivider"));
         add(new BookmarkablePageLink<Void>("createdQuery", TicketsPage.class,
               queryParameters(
                     Lucene.createdby.matches(user.username),
                     milestoneParam,
                     statiiParam,
                     assignedToParam,
                     sortBy,
                     desc,
                     1)));
         add(new BookmarkablePageLink<Void>("watchedQuery", TicketsPage.class,
               queryParameters(
                     Lucene.watchedby.matches(user.username),
                     milestoneParam,
                     statiiParam,
                     assignedToParam,
                     sortBy,
                     desc,
                     1)));
         add(new BookmarkablePageLink<Void>("mentionsQuery", TicketsPage.class,
               queryParameters(
                     Lucene.mentions.matches(user.username),
                     milestoneParam,
                     statiiParam,
                     assignedToParam,
                     sortBy,
                     desc,
                     1)));
      } else {
         add(new Label("userDivider").setVisible(false));
         add(new Label("createdQuery").setVisible(false));
         add(new Label("watchedQuery").setVisible(false));
         add(new Label("mentionsQuery").setVisible(false));
      }
      Set<TicketQuery> dynamicQueries = new TreeSet<TicketQuery>();
      for (TicketLabel label : app().tickets().getLabels(getRepositoryModel())) {
         String q = QueryBuilder.q(Lucene.labels.matches(label.name)).build();
         dynamicQueries.add(new TicketQuery(label.name, q).color(label.color));
      }
      for (QueryResult ticket : results) {
         if (!StringUtils.isEmpty(ticket.topic)) {
            String q = QueryBuilder.q(Lucene.topic.matches(ticket.topic)).build();
            dynamicQueries.add(new TicketQuery(ticket.topic, q));
         }
         if (!ArrayUtils.isEmpty(ticket.labels)) {
            for (String label : ticket.labels) {
               String q = QueryBuilder.q(Lucene.labels.matches(label)).build();
               dynamicQueries.add(new TicketQuery(label, q));
            }
         }
      }
      if (dynamicQueries.size() == 0) {
         add(new Label("dynamicQueries").setVisible(false));
      } else {
         Fragment fragment = new Fragment("dynamicQueries", "dynamicQueriesFragment", this);
         ListDataProvider<TicketQuery> dynamicQueriesDp = new ListDataProvider<TicketQuery>(new ArrayList<TicketQuery>(dynamicQueries));
         DataView<TicketQuery> dynamicQueriesList = new DataView<TicketQuery>("dynamicQuery", dynamicQueriesDp) {
            private static final long serialVersionUID = 1L;
            @Override
            public void populateItem(final Item<TicketQuery> item) {
               final TicketQuery tq = item.getModelObject();
               Component swatch = new Label("swatch", "&nbsp;").setEscapeModelStrings(false);
               if (StringUtils.isEmpty(tq.color)) {
                  // calculate a color
                  tq.color = StringUtils.getColor(tq.name);
               }
               String background = MessageFormat.format("background-color:{0};", tq.color);
               swatch.add(new SimpleAttributeModifier("style", background));
               item.add(swatch);
               if (activeQuery.contains(tq.query)) {
                  // selected
                  String q = QueryBuilder.q(activeQuery).remove(tq.query).build();
                  PageParameters params = queryParameters(q, milestoneParam, statiiParam, assignedToParam, sortBy, desc, 1);
                  item.add(new LinkPanel("link", "active", tq.name, TicketsPage.class, params).setRenderBodyOnly(true));
                  Label checked = new Label("checked");
                  WicketUtils.setCssClass(checked, "iconic-o-x");
                  item.add(checked);
                  item.add(new SimpleAttributeModifier("style", background));
               } else {
                  // unselected
                  String q = QueryBuilder.q(queryParam).toSubquery().and(tq.query).build();
                  PageParameters params = queryParameters(q, milestoneParam, statiiParam, assignedToParam, sortBy, desc, 1);
                  item.add(new LinkPanel("link", null, tq.name, TicketsPage.class, params).setRenderBodyOnly(true));
                  item.add(new Label("checked").setVisible(false));
               }
            }
         };
         fragment.add(dynamicQueriesList);
         add(fragment);
      }
      // states
      if (ArrayUtils.isEmpty(statiiParam)) {
         add(new Label("selectedStatii", getString("gb.all")));
      } else {
         add(new Label("selectedStatii", StringUtils.flattenStrings(Arrays.asList(statiiParam), ",")));
      }
      add(new BookmarkablePageLink<Void>("openTickets", TicketsPage.class, queryParameters(queryParam, milestoneParam, openStatii, assignedToParam, sortBy, desc, 1)));
      add(new BookmarkablePageLink<Void>("closedTickets", TicketsPage.class, queryParameters(queryParam, milestoneParam, closedStatii, assignedToParam, sortBy, desc, 1)));
      add(new BookmarkablePageLink<Void>("allTickets", TicketsPage.class, queryParameters(queryParam, milestoneParam, null, assignedToParam, sortBy, desc, 1)));
      // by status
      List<Status> statii = Arrays.asList(Status.values());
      ListDataProvider<Status> resolutionsDp = new ListDataProvider<Status>(statii);
      DataView<Status> statiiLinks = new DataView<Status>("statii", resolutionsDp) {
         private static final long serialVersionUID = 1L;
         @Override
         public void populateItem(final Item<Status> item) {
            final Status status = item.getModelObject();
            PageParameters p = queryParameters(queryParam, milestoneParam, new String [] { status.name().toLowerCase() }, assignedToParam, sortBy, desc, 1);
            String css = getStatusClass(status);
            item.add(new LinkPanel("statusLink", css, status.toString(), TicketsPage.class, p).setRenderBodyOnly(true));
         }
      };
      add(statiiLinks);
      // responsible filter
      List<TicketResponsible> responsibles = new ArrayList<TicketResponsible>();
      for (RegistrantAccessPermission perm : app().repositories().getUserAccessPermissions(getRepositoryModel())) {
         if (perm.permission.atLeast(AccessPermission.PUSH)) {
            UserModel u = app().users().getUserModel(perm.registrant);
            responsibles.add(new TicketResponsible(u));
         }
      }
      Collections.sort(responsibles);
      responsibles.add(0, any);
      TicketResponsible currentResponsible = null;
      for (TicketResponsible u : responsibles) {
         if (u.username.equals(assignedToParam)) {
            currentResponsible = u;
            break;
         }
      }
      add(new Label("currentResponsible", currentResponsible == null ? "" : currentResponsible.displayname));
      ListDataProvider<TicketResponsible> responsibleDp = new ListDataProvider<TicketResponsible>(responsibles);
      DataView<TicketResponsible> responsibleMenu = new DataView<TicketResponsible>("responsible", responsibleDp) {
         private static final long serialVersionUID = 1L;
         @Override
         public void populateItem(final Item<TicketResponsible> item) {
            final TicketResponsible u = item.getModelObject();
            PageParameters params = queryParameters(queryParam, milestoneParam, statiiParam, u.username, sortBy, desc, 1);
            item.add(new LinkPanel("responsibleLink", null, u.displayname, TicketsPage.class, params).setRenderBodyOnly(true));
         }
      };
      add(responsibleMenu);
      PageParameters resetResponsibleParams = queryParameters(queryParam, milestoneParam, statiiParam, null, sortBy, desc, 1);
      add(new BookmarkablePageLink<Void>("resetResponsible", TicketsPage.class, resetResponsibleParams));
      List<TicketSort> sortChoices = new ArrayList<TicketSort>();
      sortChoices.add(new TicketSort(getString("gb.sortNewest"), Lucene.created.name(), true));
      sortChoices.add(new TicketSort(getString("gb.sortOldest"), Lucene.created.name(), false));
      sortChoices.add(new TicketSort(getString("gb.sortMostRecentlyUpdated"), Lucene.updated.name(), true));
      sortChoices.add(new TicketSort(getString("gb.sortLeastRecentlyUpdated"), Lucene.updated.name(), false));
      sortChoices.add(new TicketSort(getString("gb.sortMostComments"), Lucene.comments.name(), true));
      sortChoices.add(new TicketSort(getString("gb.sortLeastComments"), Lucene.comments.name(), false));
      sortChoices.add(new TicketSort(getString("gb.sortMostPatchsetRevisions"), Lucene.patchsets.name(), true));
      sortChoices.add(new TicketSort(getString("gb.sortLeastPatchsetRevisions"), Lucene.patchsets.name(), false));
      sortChoices.add(new TicketSort(getString("gb.sortMostVotes"), Lucene.votes.name(), true));
      sortChoices.add(new TicketSort(getString("gb.sortLeastVotes"), Lucene.votes.name(), false));
      TicketSort currentSort = sortChoices.get(0);
      for (TicketSort ts : sortChoices) {
         if (ts.sortBy.equals(sortBy) && desc == ts.desc) {
            currentSort = ts;
            break;
         }
      }
      add(new Label("currentSort", currentSort.name));
      ListDataProvider<TicketSort> sortChoicesDp = new ListDataProvider<TicketSort>(sortChoices);
      DataView<TicketSort> sortMenu = new DataView<TicketSort>("sort", sortChoicesDp) {
         private static final long serialVersionUID = 1L;
         @Override
         public void populateItem(final Item<TicketSort> item) {
            final TicketSort ts = item.getModelObject();
            PageParameters params = queryParameters(queryParam, milestoneParam, statiiParam, assignedToParam, ts.sortBy, ts.desc, 1);
            item.add(new LinkPanel("sortLink", null, ts.name, TicketsPage.class, params).setRenderBodyOnly(true));
         }
      };
      add(sortMenu);
      // paging links
      buildPager(queryParam, milestoneParam, statiiParam, assignedToParam, sortBy, desc, page, pageSize, results.size(), totalResults);
      ListDataProvider<QueryResult> resultsDataProvider = new ListDataProvider<QueryResult>(results);
      DataView<QueryResult> ticketsView = new DataView<QueryResult>("ticket", resultsDataProvider) {
         private static final long serialVersionUID = 1L;
         @Override
         public void populateItem(final Item<QueryResult> item) {
            final QueryResult ticket = item.getModelObject();
            item.add(getStateIcon("state", ticket.type, ticket.status));
            item.add(new Label("id", "" + ticket.number));
            UserModel creator = app().users().getUserModel(ticket.createdBy);
            if (creator != null) {
               item.add(new LinkPanel("createdBy", null, creator.getDisplayName(),
                  UserPage.class, WicketUtils.newUsernameParameter(ticket.createdBy)));
            } else {
               item.add(new Label("createdBy", ticket.createdBy));
            }
            item.add(WicketUtils.createDateLabel("createDate", ticket.createdAt, GitBlitWebSession
                  .get().getTimezone(), getTimeUtils(), false));
            if (ticket.updatedAt == null) {
               item.add(new Label("updated").setVisible(false));
            } else {
               Fragment updated = new Fragment("updated", "updatedFragment", this);
               UserModel updater = app().users().getUserModel(ticket.updatedBy);
               if (updater != null) {
                  updated.add(new LinkPanel("updatedBy", null, updater.getDisplayName(),
                        UserPage.class, WicketUtils.newUsernameParameter(ticket.updatedBy)));
               } else {
                  updated.add(new Label("updatedBy", ticket.updatedBy));
               }
               updated.add(WicketUtils.createDateLabel("updateDate", ticket.updatedAt, GitBlitWebSession
                     .get().getTimezone(), getTimeUtils(), false));
               item.add(updated);
            }
            item.add(new LinkPanel("title", "list subject", StringUtils.trimString(
                  ticket.title, Constants.LEN_SHORTLOG), TicketsPage.class, newTicketParameter(ticket)));
            ListDataProvider<String> labelsProvider = new ListDataProvider<String>(ticket.getLabels());
            DataView<String> labelsView = new DataView<String>("labels", labelsProvider) {
               private static final long serialVersionUID = 1L;
               @Override
               public void populateItem(final Item<String> labelItem) {
                  String content = messageProcessor().processPlainCommitMessage(getRepository(), repositoryName, labelItem.getModelObject());
                  Label label = new Label("label", content);
                  label.setEscapeModelStrings(false);
                  TicketLabel tLabel = app().tickets().getLabel(getRepositoryModel(), labelItem.getModelObject());
                  String background = MessageFormat.format("background-color:{0};", tLabel.color);
                  label.add(new SimpleAttributeModifier("style", background));
                  labelItem.add(label);
               }
            };
            item.add(labelsView);
            if (StringUtils.isEmpty(ticket.responsible)) {
               item.add(new Label("responsible").setVisible(false));
            } else {
               UserModel responsible = app().users().getUserModel(ticket.responsible);
               if (responsible == null) {
                  responsible = new UserModel(ticket.responsible);
               }
               GravatarImage avatar = new GravatarImage("responsible", responsible.getDisplayName(),
                     responsible.emailAddress, null, 16, true);
               avatar.setTooltip(getString("gb.responsible") + ": " + responsible.getDisplayName());
               item.add(avatar);
            }
            // votes indicator
            Label v = new Label("votes", "" + ticket.votesCount);
            WicketUtils.setHtmlTooltip(v, getString("gb.votes"));
            item.add(v.setVisible(ticket.votesCount > 0));
            // watching indicator
            item.add(new Label("watching").setVisible(ticket.isWatching(GitBlitWebSession.get().getUsername())));
            // status indicator
            String css = getLozengeClass(ticket.status, true);
            Label l = new Label("status", ticket.status.toString());
            WicketUtils.setCssClass(l, css);
            item.add(l);
            // add the ticket indicators/icons
            List<Indicator> indicators = new ArrayList<Indicator>();
            // comments
            if (ticket.commentsCount > 0) {
               int count = ticket.commentsCount;
               String pattern = "gb.nComments";
               if (count == 1) {
                  pattern = "gb.oneComment";
               }
               indicators.add(new Indicator("fa fa-comment", count, pattern));
            }
            // participants
            if (!ArrayUtils.isEmpty(ticket.participants)) {
               int count = ticket.participants.size();
               if (count > 1) {
                  String pattern = "gb.nParticipants";
                  indicators.add(new Indicator("fa fa-user", count, pattern));
               }
            }
            // attachments
            if (!ArrayUtils.isEmpty(ticket.attachments)) {
               int count = ticket.attachments.size();
               String pattern = "gb.nAttachments";
               if (count == 1) {
                  pattern = "gb.oneAttachment";
               }
               indicators.add(new Indicator("fa fa-file", count, pattern));
            }
            // patchset revisions
            if (ticket.patchset != null) {
               int count = ticket.patchset.commits;
               String pattern = "gb.nCommits";
               if (count == 1) {
                  pattern = "gb.oneCommit";
               }
               indicators.add(new Indicator("fa fa-code", count, pattern));
            }
            // milestone
            if (!StringUtils.isEmpty(ticket.milestone)) {
               indicators.add(new Indicator("fa fa-bullseye", ticket.milestone));
            }
            ListDataProvider<Indicator> indicatorsDp = new ListDataProvider<Indicator>(indicators);
            DataView<Indicator> indicatorsView = new DataView<Indicator>("indicators", indicatorsDp) {
               private static final long serialVersionUID = 1L;
               @Override
               public void populateItem(final Item<Indicator> item) {
                  Indicator indicator = item.getModelObject();
                  String tooltip = indicator.getTooltip();
                  Label icon = new Label("icon");
                  WicketUtils.setCssClass(icon, indicator.css);
                  item.add(icon);
                  if (indicator.count > 0) {
                     Label count = new Label("count", "" + indicator.count);
                     item.add(count.setVisible(!StringUtils.isEmpty(tooltip)));
                  } else {
                     item.add(new Label("count").setVisible(false));
                  }
                  WicketUtils.setHtmlTooltip(item, tooltip);
               }
            };
            item.add(indicatorsView);
         }
      };
      add(ticketsView);
      DataView<TicketMilestone> milestonesList = new DataView<TicketMilestone>("milestoneList", milestonesDp) {
         private static final long serialVersionUID = 1L;
         @Override
         public void populateItem(final Item<TicketMilestone> item) {
            final TicketMilestone tm = item.getModelObject();
            item.add(new Label("milestoneName", tm.name));
            item.add(new Label("milestoneState", tm.status.name()));
            item.add(new Label("milestoneDue", tm.due == null ? getString("gb.notSpecified") : tm.due.toString()));
         }
      };
      add(milestonesList);
   }
   protected PageParameters newPathParameter(String path) {
      return WicketUtils.newPathParameter(repositoryName, objectId, path);
   protected PageParameters queryParameters(
         String query,
         String milestone,
         String[] states,
         String assignedTo,
         String sort,
         boolean descending,
         int page) {
      PageParameters params = WicketUtils.newRepositoryParameter(repositoryName);
      if (!StringUtils.isEmpty(query)) {
         params.add("q", query);
      }
      if (!StringUtils.isEmpty(milestone)) {
         params.add(Lucene.milestone.name(), milestone);
      }
      if (!ArrayUtils.isEmpty(states)) {
         for (String state : states) {
            params.add(Lucene.status.name(), state);
         }
      }
      if (!StringUtils.isEmpty(assignedTo)) {
         params.add(Lucene.responsible.name(), assignedTo);
      }
      if (!StringUtils.isEmpty(sort)) {
         params.add("sort", sort);
      }
      if (!descending) {
         params.add("direction", "asc");
      }
      if (page > 1) {
         params.add("pg", "" + page);
      }
      return params;
   }
   protected PageParameters newTicketParameter(QueryResult ticket) {
      return WicketUtils.newObjectParameter(repositoryName, "" + ticket.number);
   }
   @Override
   protected String getPageName() {
      return getString("gb.tickets");
   }
   protected void buildPager(
         final String query,
         final String milestone,
         final String [] states,
         final String assignedTo,
         final String sort,
         final boolean desc,
         final int page,
         int pageSize,
         int count,
         int total) {
      boolean showNav = total > (2 * pageSize);
      boolean allowPrev = page > 1;
      boolean allowNext = (pageSize * (page - 1) + count) < total;
      add(new BookmarkablePageLink<Void>("prevLink", TicketsPage.class, queryParameters(query, milestone, states, assignedTo, sort, desc, page - 1)).setEnabled(allowPrev).setVisible(showNav));
      add(new BookmarkablePageLink<Void>("nextLink", TicketsPage.class, queryParameters(query, milestone, states, assignedTo, sort, desc, page + 1)).setEnabled(allowNext).setVisible(showNav));
      if (total <= pageSize) {
         add(new Label("pageLink").setVisible(false));
         return;
      }
      // determine page numbers to display
      int pages = count == 0 ? 0 : ((total / pageSize) + (total % pageSize == 0 ? 0 : 1));
      // preferred number of pagelinks
      int segments = 5;
      if (pages < segments) {
         // not enough data for preferred number of page links
         segments = pages;
      }
      int minpage = Math.min(Math.max(1, page - 2), pages - (segments - 1));
      int maxpage = Math.min(pages, minpage + (segments - 1));
      List<Integer> sequence = new ArrayList<Integer>();
      for (int i = minpage; i <= maxpage; i++) {
         sequence.add(i);
      }
      ListDataProvider<Integer> pagesDp = new ListDataProvider<Integer>(sequence);
      DataView<Integer> pagesView = new DataView<Integer>("pageLink", pagesDp) {
         private static final long serialVersionUID = 1L;
         @Override
         public void populateItem(final Item<Integer> item) {
            final Integer i = item.getModelObject();
            LinkPanel link = new LinkPanel("page", null, "" + i, TicketsPage.class, queryParameters(query, milestone, states, assignedTo, sort, desc, i));
            link.setRenderBodyOnly(true);
            if (i == page) {
               WicketUtils.setCssClass(item, "active");
            }
            item.add(link);
         }
      };
      add(pagesView);
   }
   private class Indicator implements Serializable {
      private static final long serialVersionUID = 1L;
      final String css;
      final int count;
      final String tooltip;
      Indicator(String css, String tooltip) {
         this.css = css;
         this.tooltip = tooltip;
         this.count = 0;
      }
      Indicator(String css, int count, String pattern) {
         this.css = css;
         this.count = count;
         this.tooltip = StringUtils.isEmpty(pattern) ? "" : MessageFormat.format(getString(pattern), count);
      }
      String getTooltip() {
         return tooltip;
      }
   }
   private class TicketQuery implements Serializable, Comparable<TicketQuery> {
      private static final long serialVersionUID = 1L;
      final String name;
      final String query;
      String color;
      TicketQuery(String name, String query) {
         this.name = name;
         this.query = query;
      }
      TicketQuery color(String value) {
         this.color = value;
         return this;
      }
      @Override
      public boolean equals(Object o) {
         if (o instanceof TicketQuery) {
            return ((TicketQuery) o).query.equals(query);
         }
         return false;
      }
      @Override
      public int hashCode() {
         return query.hashCode();
      }
      @Override
      public int compareTo(TicketQuery o) {
         return query.compareTo(o.query);
      }
   }
   private class TicketSort implements Serializable {
      private static final long serialVersionUID = 1L;
      final String name;
      final String sortBy;
      final boolean desc;
      TicketSort(String name, String sortBy, boolean desc) {
         this.name = name;
         this.sortBy = sortBy;
         this.desc = desc;
      }
   }
   private class TicketSearchForm extends SessionlessForm<Void> implements Serializable {
      private static final long serialVersionUID = 1L;
      private final String repositoryName;
      private final IModel<String> searchBoxModel;;
      public TicketSearchForm(String id, String repositoryName, String text) {
         super(id, TicketsPage.this.getClass(), TicketsPage.this.getPageParameters());
         this.repositoryName = repositoryName;
         this.searchBoxModel = new Model<String>(text == null ? "" : text);
         TextField<String> searchBox = new TextField<String>("ticketSearchBox", searchBoxModel);
         add(searchBox);
      }
      void setTranslatedAttributes() {
         WicketUtils.setHtmlTooltip(get("ticketSearchBox"),
               MessageFormat.format(getString("gb.searchTicketsTooltip"), repositoryName));
         WicketUtils.setInputPlaceholder(get("ticketSearchBox"), getString("gb.searchTickets"));
      }
      @Override
      public void onSubmit() {
         String searchString = searchBoxModel.getObject();
         if (StringUtils.isEmpty(searchString)) {
            // redirect to self to avoid wicket page update bug
            String absoluteUrl = getCanonicalUrl();
            getRequestCycle().setRequestTarget(new RedirectRequestTarget(absoluteUrl));
            return;
         }
         // use an absolute url to workaround Wicket-Tomcat problems with
         // mounted url parameters (issue-111)
         PageParameters params = WicketUtils.newRepositoryParameter(repositoryName);
         params.add("s", searchString);
         String absoluteUrl = getCanonicalUrl(TicketsPage.class, params);
         getRequestCycle().setRequestTarget(new RedirectRequestTarget(absoluteUrl));
      }
   }
}