| | |
| | | /*
|
| | | * 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.
|
| | |
| | | */
|
| | | 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", " ").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));
|
| | | }
|
| | | }
|
| | | }
|