You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

MyTicketsPage.java 22KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501
  1. /*
  2. * Copyright 2014 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. import java.util.ArrayList;
  18. import java.util.Arrays;
  19. import java.util.Collection;
  20. import java.util.Collections;
  21. import java.util.HashMap;
  22. import java.util.List;
  23. import org.apache.wicket.PageParameters;
  24. import org.apache.wicket.markup.html.basic.Label;
  25. import org.apache.wicket.markup.html.link.BookmarkablePageLink;
  26. import org.apache.wicket.markup.repeater.Item;
  27. import org.apache.wicket.markup.repeater.data.DataView;
  28. import org.apache.wicket.markup.repeater.data.ListDataProvider;
  29. import com.gitblit.Keys;
  30. import com.gitblit.models.RepositoryModel;
  31. import com.gitblit.models.TicketModel;
  32. import com.gitblit.models.TicketModel.Status;
  33. import com.gitblit.models.UserModel;
  34. import com.gitblit.tickets.QueryBuilder;
  35. import com.gitblit.tickets.QueryResult;
  36. import com.gitblit.tickets.TicketIndexer.Lucene;
  37. import com.gitblit.utils.ArrayUtils;
  38. import com.gitblit.utils.StringUtils;
  39. import com.gitblit.wicket.GitBlitWebSession;
  40. import com.gitblit.wicket.TicketsUI;
  41. import com.gitblit.wicket.TicketsUI.TicketSort;
  42. import com.gitblit.wicket.WicketUtils;
  43. import com.gitblit.wicket.panels.LinkPanel;
  44. import com.gitblit.wicket.panels.TicketListPanel;
  45. import com.gitblit.wicket.panels.TicketSearchForm;
  46. import com.gitblit.wicket.panels.UserTitlePanel;
  47. /**
  48. * My Tickets page
  49. *
  50. * @author Christian Buisson
  51. * @author James Moger
  52. */
  53. public class MyTicketsPage extends RootPage {
  54. public MyTicketsPage() {
  55. this(null);
  56. }
  57. public MyTicketsPage(PageParameters params) {
  58. super(params);
  59. setupPage("", getString("gb.myTickets"));
  60. UserModel currentUser = GitBlitWebSession.get().getUser();
  61. if (currentUser == null || UserModel.ANONYMOUS.equals(currentUser)) {
  62. setRedirect(true);
  63. setResponsePage(getApplication().getHomePage());
  64. return;
  65. }
  66. final String username = currentUser.getName();
  67. final String[] statiiParam = (params == null) ? TicketsUI.openStatii : params.getStringArray(Lucene.status.name());
  68. final String assignedToParam = (params == null) ? "" : params.getString(Lucene.responsible.name(), null);
  69. final String milestoneParam = (params == null) ? "" : params.getString(Lucene.milestone.name(), null);
  70. final String queryParam = (params == null) ? null : params.getString("q", null);
  71. final String searchParam = (params == null) ? "" : params.getString("s", null);
  72. final String sortBy = (params == null) ? "" : Lucene.fromString(params.getString("sort", Lucene.created.name())).name();
  73. final String repositoryId = (params == null) ? "" : params.getString(Lucene.rid.name(), null);
  74. final boolean desc = (params == null) ? true : !"asc".equals(params.getString("direction", "desc"));
  75. // add the user title panel
  76. add(new UserTitlePanel("userTitlePanel", currentUser, getString("gb.myTickets")));
  77. // add search form
  78. add(new TicketSearchForm("ticketSearchForm", null, searchParam, getClass(), params));
  79. // standard queries
  80. add(new BookmarkablePageLink<Void>("changesQuery", MyTicketsPage.class,
  81. queryParameters(
  82. Lucene.type.matches(TicketModel.Type.Proposal.name()),
  83. milestoneParam,
  84. statiiParam,
  85. assignedToParam,
  86. sortBy,
  87. desc,
  88. repositoryId,
  89. 1)));
  90. add(new BookmarkablePageLink<Void>("bugsQuery", MyTicketsPage.class,
  91. queryParameters(
  92. Lucene.type.matches(TicketModel.Type.Bug.name()),
  93. milestoneParam,
  94. statiiParam,
  95. assignedToParam,
  96. sortBy,
  97. desc,
  98. repositoryId,
  99. 1)));
  100. add(new BookmarkablePageLink<Void>("enhancementsQuery", MyTicketsPage.class,
  101. queryParameters(
  102. Lucene.type.matches(TicketModel.Type.Enhancement.name()),
  103. milestoneParam,
  104. statiiParam,
  105. assignedToParam,
  106. sortBy,
  107. desc,
  108. repositoryId,
  109. 1)));
  110. add(new BookmarkablePageLink<Void>("tasksQuery", MyTicketsPage.class,
  111. queryParameters(
  112. Lucene.type.matches(TicketModel.Type.Task.name()),
  113. milestoneParam,
  114. statiiParam,
  115. assignedToParam,
  116. sortBy,
  117. desc,
  118. repositoryId,
  119. 1)));
  120. add(new BookmarkablePageLink<Void>("questionsQuery", MyTicketsPage.class,
  121. queryParameters(
  122. Lucene.type.matches(TicketModel.Type.Question.name()),
  123. milestoneParam,
  124. statiiParam,
  125. assignedToParam,
  126. sortBy,
  127. desc,
  128. repositoryId,
  129. 1)));
  130. add(new BookmarkablePageLink<Void>("maintenanceQuery", MyTicketsPage.class,
  131. queryParameters(
  132. Lucene.type.matches(TicketModel.Type.Maintenance.name()),
  133. milestoneParam,
  134. statiiParam,
  135. assignedToParam,
  136. sortBy,
  137. desc,
  138. repositoryId,
  139. 1)));
  140. add(new BookmarkablePageLink<Void>("resetQuery", MyTicketsPage.class,
  141. queryParameters(
  142. null,
  143. milestoneParam,
  144. TicketsUI.openStatii,
  145. null,
  146. null,
  147. true,
  148. null,
  149. 1)));
  150. add(new Label("userDivider"));
  151. add(new BookmarkablePageLink<Void>("createdQuery", MyTicketsPage.class,
  152. queryParameters(
  153. Lucene.createdby.matches(username),
  154. milestoneParam,
  155. statiiParam,
  156. assignedToParam,
  157. sortBy,
  158. desc,
  159. repositoryId,
  160. 1)));
  161. add(new BookmarkablePageLink<Void>("watchedQuery", MyTicketsPage.class,
  162. queryParameters(
  163. Lucene.watchedby.matches(username),
  164. milestoneParam,
  165. statiiParam,
  166. assignedToParam,
  167. sortBy,
  168. desc,
  169. repositoryId,
  170. 1)));
  171. add(new BookmarkablePageLink<Void>("mentionsQuery", MyTicketsPage.class,
  172. queryParameters(
  173. Lucene.mentions.matches(username),
  174. milestoneParam,
  175. statiiParam,
  176. assignedToParam,
  177. sortBy,
  178. desc,
  179. repositoryId,
  180. 1)));
  181. add(new BookmarkablePageLink<Void>("responsibleQuery", MyTicketsPage.class,
  182. queryParameters(
  183. Lucene.responsible.matches(username),
  184. milestoneParam,
  185. statiiParam,
  186. assignedToParam,
  187. sortBy,
  188. desc,
  189. repositoryId,
  190. 1)));
  191. // states
  192. if (ArrayUtils.isEmpty(statiiParam)) {
  193. add(new Label("selectedStatii", getString("gb.all")));
  194. } else {
  195. add(new Label("selectedStatii", StringUtils.flattenStrings(Arrays.asList(statiiParam), ",")));
  196. }
  197. add(new BookmarkablePageLink<Void>("openTickets", MyTicketsPage.class, queryParameters(queryParam, milestoneParam, TicketsUI.openStatii, assignedToParam, sortBy, desc, repositoryId, 1)));
  198. add(new BookmarkablePageLink<Void>("closedTickets", MyTicketsPage.class, queryParameters(queryParam, milestoneParam, TicketsUI.closedStatii, assignedToParam, sortBy, desc, repositoryId, 1)));
  199. add(new BookmarkablePageLink<Void>("allTickets", MyTicketsPage.class, queryParameters(queryParam, milestoneParam, null, assignedToParam, sortBy, desc, repositoryId, 1)));
  200. // by status
  201. List<Status> statii = new ArrayList<Status>(Arrays.asList(Status.values()));
  202. statii.remove(Status.Closed);
  203. ListDataProvider<Status> resolutionsDp = new ListDataProvider<Status>(statii);
  204. DataView<Status> statiiLinks = new DataView<Status>("statii", resolutionsDp) {
  205. private static final long serialVersionUID = 1L;
  206. @Override
  207. public void populateItem(final Item<Status> item) {
  208. final Status status = item.getModelObject();
  209. PageParameters p = queryParameters(queryParam, milestoneParam, new String [] { status.name().toLowerCase() }, assignedToParam, sortBy, desc, repositoryId, 1);
  210. String css = TicketsUI.getStatusClass(status);
  211. item.add(new LinkPanel("statusLink", css, status.toString(), MyTicketsPage.class, p).setRenderBodyOnly(true));
  212. }
  213. };
  214. add(statiiLinks);
  215. // by sort
  216. List<TicketSort> sortChoices = new ArrayList<TicketSort>();
  217. sortChoices.add(new TicketSort(getString("gb.sortNewest"), Lucene.created.name(), true));
  218. sortChoices.add(new TicketSort(getString("gb.sortOldest"), Lucene.created.name(), false));
  219. sortChoices.add(new TicketSort(getString("gb.sortMostRecentlyUpdated"), Lucene.updated.name(), true));
  220. sortChoices.add(new TicketSort(getString("gb.sortLeastRecentlyUpdated"), Lucene.updated.name(), false));
  221. sortChoices.add(new TicketSort(getString("gb.sortMostComments"), Lucene.comments.name(), true));
  222. sortChoices.add(new TicketSort(getString("gb.sortLeastComments"), Lucene.comments.name(), false));
  223. sortChoices.add(new TicketSort(getString("gb.sortMostPatchsetRevisions"), Lucene.patchsets.name(), true));
  224. sortChoices.add(new TicketSort(getString("gb.sortLeastPatchsetRevisions"), Lucene.patchsets.name(), false));
  225. sortChoices.add(new TicketSort(getString("gb.sortMostVotes"), Lucene.votes.name(), true));
  226. sortChoices.add(new TicketSort(getString("gb.sortLeastVotes"), Lucene.votes.name(), false));
  227. sortChoices.add(new TicketSort(getString("gb.sortHighestPriority"), Lucene.priority.name(), true));
  228. sortChoices.add(new TicketSort(getString("gb.sortLowestPriority"), Lucene.priority.name(), false));
  229. sortChoices.add(new TicketSort(getString("gb.sortHighestSeverity"), Lucene.severity.name(), true));
  230. sortChoices.add(new TicketSort(getString("gb.sortLowestSeverity"), Lucene.severity.name(), false));
  231. TicketSort currentSort = sortChoices.get(0);
  232. for (TicketSort ts : sortChoices) {
  233. if (ts.sortBy.equals(sortBy) && desc == ts.desc) {
  234. currentSort = ts;
  235. break;
  236. }
  237. }
  238. add(new Label("currentSort", currentSort.name));
  239. ListDataProvider<TicketSort> sortChoicesDp = new ListDataProvider<TicketSort>(sortChoices);
  240. DataView<TicketSort> sortMenu = new DataView<TicketSort>("sort", sortChoicesDp) {
  241. private static final long serialVersionUID = 1L;
  242. @Override
  243. public void populateItem(final Item<TicketSort> item) {
  244. final TicketSort ts = item.getModelObject();
  245. PageParameters params = queryParameters(queryParam, milestoneParam, statiiParam, assignedToParam, ts.sortBy, ts.desc, repositoryId, 1);
  246. item.add(new LinkPanel("sortLink", null, ts.name, MyTicketsPage.class, params).setRenderBodyOnly(true));
  247. }
  248. };
  249. add(sortMenu);
  250. // by repository
  251. final List<QueryResult> tickets =
  252. query(initializeQueryBuilder(null, username), 1, Integer.MAX_VALUE, sortBy, desc);
  253. final List<RepositoryModel> repositoryChoices = correspondingRepositories(tickets);
  254. Collections.sort(repositoryChoices);
  255. final RepositoryModel noneChoice = new RepositoryModel();
  256. noneChoice.name = getString("gb.all");
  257. repositoryChoices.add(0, noneChoice);
  258. RepositoryModel currentRepository = repositoryChoices.get(0);
  259. for (RepositoryModel r : repositoryChoices) {
  260. if (r.getRID().equals(repositoryId)) {
  261. currentRepository = r;
  262. break;
  263. }
  264. }
  265. add(new Label("currentRepository", currentRepository.toString()));
  266. ListDataProvider<RepositoryModel> repositoryChoicesDp = new ListDataProvider<RepositoryModel>(repositoryChoices);
  267. DataView<RepositoryModel> repositoryMenu = new DataView<RepositoryModel>("repository", repositoryChoicesDp) {
  268. private static final long serialVersionUID = 1L;
  269. @Override
  270. public void populateItem(final Item<RepositoryModel> item) {
  271. final RepositoryModel r = item.getModelObject();
  272. String rid = r == noneChoice ? null : r.getRID();
  273. PageParameters params = queryParameters(queryParam, milestoneParam, statiiParam, assignedToParam, sortBy, desc, rid, 1);
  274. item.add(new LinkPanel("repositoryLink", null, r.toString(), MyTicketsPage.class, params).setRenderBodyOnly(true));
  275. }
  276. };
  277. add(repositoryMenu);
  278. // Update query with filter criteria
  279. final QueryBuilder qb = initializeQueryBuilder(queryParam, username);
  280. if (!qb.containsField(Lucene.status.name()) && !ArrayUtils.isEmpty(statiiParam)) {
  281. // specify the states
  282. boolean not = false;
  283. QueryBuilder q = new QueryBuilder();
  284. for (String state : statiiParam) {
  285. if (state.charAt(0) == '!') {
  286. not = true;
  287. q.and(Lucene.status.doesNotMatch(state.substring(1)));
  288. } else {
  289. q.or(Lucene.status.matches(state));
  290. }
  291. }
  292. if (not) {
  293. qb.and(q.toString());
  294. } else {
  295. qb.and(q.toSubquery().toString());
  296. }
  297. }
  298. if (noneChoice != currentRepository && !qb.containsField(Lucene.rid.name())) {
  299. QueryBuilder q1 = new QueryBuilder();
  300. q1.and(Lucene.rid.matches(repositoryId));
  301. qb.and(q1.toSubquery().toString());
  302. }
  303. // paging links
  304. int page = (params != null) ? Math.max(1, WicketUtils.getPage(params)) : 1;
  305. int pageSize = app().settings().getInteger(Keys.tickets.perPage, 25);
  306. final List<QueryResult> allResults =
  307. StringUtils.isEmpty(searchParam) ? query(qb, page, pageSize, sortBy, desc) : search(searchParam, page, pageSize);
  308. List<QueryResult> viewableResults = new ArrayList<>(allResults.size());
  309. for (QueryResult queryResult : allResults) {
  310. RepositoryModel model = app().repositories().getRepositoryModel(currentUser, queryResult.repository);
  311. if ((model != null) && (currentUser.canView(model))) {
  312. viewableResults.add(queryResult);
  313. }
  314. }
  315. int totalResults = viewableResults.size() == 0 ? 0 : viewableResults.get(0).totalResults;
  316. buildPager(queryParam, milestoneParam, statiiParam, assignedToParam, sortBy, desc, repositoryId, page, pageSize, viewableResults.size(), totalResults);
  317. final boolean showSwatch = app().settings().getBoolean(Keys.web.repositoryListSwatches, true);
  318. add(new TicketListPanel("ticketList", viewableResults, showSwatch, true));
  319. }
  320. protected PageParameters queryParameters(
  321. String query,
  322. String milestone,
  323. String[] states,
  324. String assignedTo,
  325. String sort,
  326. boolean descending,
  327. String repositoryId,
  328. int page) {
  329. PageParameters params = WicketUtils.newRepositoryParameter("");
  330. if (!StringUtils.isEmpty(query)) {
  331. params.add("q", query);
  332. }
  333. if (!StringUtils.isEmpty(milestone)) {
  334. params.add(Lucene.milestone.name(), milestone);
  335. }
  336. if (!ArrayUtils.isEmpty(states)) {
  337. for (String state : states) {
  338. params.add(Lucene.status.name(), state);
  339. }
  340. }
  341. if (!StringUtils.isEmpty(assignedTo)) {
  342. params.add(Lucene.responsible.name(), assignedTo);
  343. }
  344. if (!StringUtils.isEmpty(sort)) {
  345. params.add("sort", sort);
  346. }
  347. if (!descending) {
  348. params.add("direction", "asc");
  349. }
  350. if (!StringUtils.isEmpty(repositoryId)) {
  351. params.add(Lucene.rid.name(), repositoryId);
  352. }
  353. if (page > 1) {
  354. params.add("pg", "" + page);
  355. }
  356. return params;
  357. }
  358. protected void buildPager(
  359. final String query,
  360. final String milestone,
  361. final String [] states,
  362. final String assignedTo,
  363. final String sort,
  364. final boolean desc,
  365. final String repositoryId,
  366. final int page,
  367. int pageSize,
  368. int count,
  369. int total) {
  370. boolean showNav = total > (2 * pageSize);
  371. boolean allowPrev = page > 1;
  372. boolean allowNext = (pageSize * (page - 1) + count) < total;
  373. add(new BookmarkablePageLink<Void>("prevLink", MyTicketsPage.class, queryParameters(query, milestone, states, assignedTo, sort, desc, repositoryId, page - 1)).setEnabled(allowPrev).setVisible(showNav));
  374. add(new BookmarkablePageLink<Void>("nextLink", MyTicketsPage.class, queryParameters(query, milestone, states, assignedTo, sort, desc, repositoryId, page + 1)).setEnabled(allowNext).setVisible(showNav));
  375. if (total <= pageSize) {
  376. add(new Label("pageLink").setVisible(false));
  377. return;
  378. }
  379. // determine page numbers to display
  380. int pages = count == 0 ? 0 : ((total / pageSize) + (total % pageSize == 0 ? 0 : 1));
  381. // preferred number of pagelinks
  382. int segments = 5;
  383. if (pages < segments) {
  384. // not enough data for preferred number of page links
  385. segments = pages;
  386. }
  387. int minpage = Math.min(Math.max(1, page - 2), pages - (segments - 1));
  388. int maxpage = Math.min(pages, minpage + (segments - 1));
  389. List<Integer> sequence = new ArrayList<Integer>();
  390. for (int i = minpage; i <= maxpage; i++) {
  391. sequence.add(i);
  392. }
  393. ListDataProvider<Integer> pagesDp = new ListDataProvider<Integer>(sequence);
  394. DataView<Integer> pagesView = new DataView<Integer>("pageLink", pagesDp) {
  395. private static final long serialVersionUID = 1L;
  396. @Override
  397. public void populateItem(final Item<Integer> item) {
  398. final Integer i = item.getModelObject();
  399. LinkPanel link = new LinkPanel("page", null, "" + i, MyTicketsPage.class, queryParameters(query, milestone, states, assignedTo, sort, desc, repositoryId, i));
  400. link.setRenderBodyOnly(true);
  401. if (i == page) {
  402. WicketUtils.setCssClass(item, "active");
  403. }
  404. item.add(link);
  405. }
  406. };
  407. add(pagesView);
  408. }
  409. private QueryBuilder initializeQueryBuilder(String queryparam, String username) {
  410. final QueryBuilder qb = new QueryBuilder(queryparam);
  411. // focused "my tickets"
  412. if (qb.containsField(Lucene.createdby.name())
  413. || qb.containsField(Lucene.responsible.name())
  414. || qb.containsField(Lucene.watchedby.name())
  415. || qb.containsField(Lucene.mentions.name())) {
  416. return qb;
  417. }
  418. // general "my tickets"
  419. return qb.andSubquery()
  420. .or(Lucene.createdby.matches(username))
  421. .or(Lucene.responsible.matches(username))
  422. .or(Lucene.watchedby.matches(username))
  423. .or(Lucene.mentions.matches(username))
  424. .endSubquery();
  425. }
  426. private List<QueryResult> query(QueryBuilder qb, int page, int pageSize, String sortBy, boolean descending) {
  427. return app().tickets().queryFor(qb.build(), page, pageSize, sortBy, descending);
  428. }
  429. private List<QueryResult> search(String searchParam, int page, int pageSize) {
  430. return app().tickets().searchFor(null, searchParam, page, pageSize);
  431. }
  432. private List<RepositoryModel> correspondingRepositories(Collection<QueryResult> tickets) {
  433. final HashMap<String, RepositoryModel> result = new HashMap<>();
  434. for (QueryResult ticket : tickets) {
  435. RepositoryModel repository = app().repositories().getRepositoryModel(ticket.repository);
  436. if (!result.containsKey(repository.getRID())) {
  437. result.put(repository.getRID(), repository);
  438. }
  439. }
  440. return new ArrayList<>(result.values());
  441. }
  442. }