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.

EditTicketPage.java 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394
  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.Collections;
  20. import java.util.List;
  21. import java.util.Set;
  22. import java.util.TreeSet;
  23. import org.apache.wicket.PageParameters;
  24. import org.apache.wicket.ajax.AjaxRequestTarget;
  25. import org.apache.wicket.ajax.markup.html.form.AjaxButton;
  26. import org.apache.wicket.markup.html.basic.Label;
  27. import org.apache.wicket.markup.html.form.Button;
  28. import org.apache.wicket.markup.html.form.DropDownChoice;
  29. import org.apache.wicket.markup.html.form.Form;
  30. import org.apache.wicket.markup.html.form.TextField;
  31. import org.apache.wicket.markup.html.panel.Fragment;
  32. import org.apache.wicket.model.IModel;
  33. import org.apache.wicket.model.Model;
  34. import org.eclipse.jgit.lib.Repository;
  35. import com.gitblit.Constants;
  36. import com.gitblit.Constants.AccessPermission;
  37. import com.gitblit.Constants.AuthorizationControl;
  38. import com.gitblit.models.RegistrantAccessPermission;
  39. import com.gitblit.models.TicketModel;
  40. import com.gitblit.models.TicketModel.Change;
  41. import com.gitblit.models.TicketModel.Field;
  42. import com.gitblit.models.TicketModel.Status;
  43. import com.gitblit.models.TicketModel.Type;
  44. import com.gitblit.models.UserModel;
  45. import com.gitblit.tickets.TicketMilestone;
  46. import com.gitblit.tickets.TicketNotifier;
  47. import com.gitblit.tickets.TicketResponsible;
  48. import com.gitblit.utils.StringUtils;
  49. import com.gitblit.wicket.GitBlitWebSession;
  50. import com.gitblit.wicket.WicketUtils;
  51. import com.gitblit.wicket.panels.MarkdownTextArea;
  52. import com.google.common.base.Optional;
  53. /**
  54. * Page for editing a ticket.
  55. *
  56. * @author James Moger
  57. *
  58. */
  59. public class EditTicketPage extends RepositoryPage {
  60. static final String NIL = "<nil>";
  61. static final String ESC_NIL = StringUtils.escapeForHtml(NIL, false);
  62. private IModel<TicketModel.Type> typeModel;
  63. private IModel<String> titleModel;
  64. private MarkdownTextArea descriptionEditor;
  65. private IModel<String> topicModel;
  66. private IModel<String> mergeToModel;
  67. private IModel<Status> statusModel;
  68. private IModel<TicketResponsible> responsibleModel;
  69. private IModel<TicketMilestone> milestoneModel;
  70. private Label descriptionPreview;
  71. private IModel<TicketModel.Priority> priorityModel;
  72. private IModel<TicketModel.Severity> severityModel;
  73. public EditTicketPage(PageParameters params) {
  74. super(params);
  75. UserModel currentUser = GitBlitWebSession.get().getUser();
  76. if (currentUser == null) {
  77. currentUser = UserModel.ANONYMOUS;
  78. }
  79. long ticketId = 0L;
  80. try {
  81. String h = WicketUtils.getObject(params);
  82. ticketId = Long.parseLong(h);
  83. } catch (Exception e) {
  84. setResponsePage(TicketsPage.class, WicketUtils.newRepositoryParameter(repositoryName));
  85. }
  86. TicketModel ticket = app().tickets().getTicket(getRepositoryModel(), ticketId);
  87. if (ticket == null
  88. || !currentUser.canEdit(ticket, getRepositoryModel())
  89. || !app().tickets().isAcceptingTicketUpdates(getRepositoryModel())) {
  90. setResponsePage(TicketsPage.class, WicketUtils.newObjectParameter(repositoryName, "" + ticketId));
  91. // create a placeholder object so we don't trigger NPEs
  92. ticket = new TicketModel();
  93. }
  94. typeModel = Model.of(ticket.type);
  95. titleModel = Model.of(ticket.title);
  96. topicModel = Model.of(ticket.topic == null ? "" : ticket.topic);
  97. responsibleModel = Model.of();
  98. milestoneModel = Model.of();
  99. mergeToModel = Model.of(ticket.mergeTo == null ? getRepositoryModel().mergeTo : ticket.mergeTo);
  100. statusModel = Model.of(ticket.status);
  101. priorityModel = Model.of(ticket.priority);
  102. severityModel = Model.of(ticket.severity);
  103. setStatelessHint(false);
  104. setOutputMarkupId(true);
  105. Form<Void> form = new Form<Void>("editForm");
  106. add(form);
  107. List<Type> typeChoices;
  108. if (ticket.isProposal()) {
  109. typeChoices = Arrays.asList(Type.Proposal);
  110. } else {
  111. typeChoices = Arrays.asList(TicketModel.Type.choices());
  112. }
  113. form.add(new DropDownChoice<TicketModel.Type>("type", typeModel, typeChoices));
  114. form.add(new TextField<String>("title", titleModel));
  115. form.add(new TextField<String>("topic", topicModel));
  116. final IModel<String> markdownPreviewModel = Model.of(ticket.body == null ? "" : ticket.body);
  117. descriptionPreview = new Label("descriptionPreview", markdownPreviewModel);
  118. descriptionPreview.setEscapeModelStrings(false);
  119. descriptionPreview.setOutputMarkupId(true);
  120. form.add(descriptionPreview);
  121. descriptionEditor = new MarkdownTextArea("description", markdownPreviewModel, descriptionPreview);
  122. descriptionEditor.setRepository(repositoryName);
  123. descriptionEditor.setText(ticket.body);
  124. form.add(descriptionEditor);
  125. // status
  126. List<Status> statusChoices;
  127. if (ticket.isClosed()) {
  128. statusChoices = Arrays.asList(ticket.status, Status.Open);
  129. } else if (ticket.isProposal()) {
  130. statusChoices = Arrays.asList(TicketModel.Status.proposalWorkflow);
  131. } else if (ticket.isBug()) {
  132. statusChoices = Arrays.asList(TicketModel.Status.bugWorkflow);
  133. } else {
  134. statusChoices = Arrays.asList(TicketModel.Status.requestWorkflow);
  135. }
  136. Fragment status = new Fragment("status", "statusFragment", this);
  137. status.add(new DropDownChoice<TicketModel.Status>("status", statusModel, statusChoices));
  138. form.add(status);
  139. List<TicketModel.Severity> severityChoices = Arrays.asList(TicketModel.Severity.choices());
  140. form.add(new DropDownChoice<TicketModel.Severity>("severity", severityModel, severityChoices));
  141. if (currentUser.canAdmin(ticket, getRepositoryModel())) {
  142. // responsible
  143. Set<String> userlist = new TreeSet<String>(ticket.getParticipants());
  144. if (UserModel.ANONYMOUS.canPush(getRepositoryModel())
  145. || AuthorizationControl.AUTHENTICATED == getRepositoryModel().authorizationControl) {
  146. // authorization is ANONYMOUS or AUTHENTICATED (i.e. all users can be set responsible)
  147. userlist.addAll(app().users().getAllUsernames());
  148. } else {
  149. // authorization is by NAMED users (users with PUSH permission can be set responsible)
  150. for (RegistrantAccessPermission rp : app().repositories().getUserAccessPermissions(getRepositoryModel())) {
  151. if (rp.permission.atLeast(AccessPermission.PUSH)) {
  152. userlist.add(rp.registrant);
  153. }
  154. }
  155. }
  156. List<TicketResponsible> responsibles = new ArrayList<TicketResponsible>();
  157. for (String username : userlist) {
  158. UserModel user = app().users().getUserModel(username);
  159. if (user != null && !user.disabled) {
  160. TicketResponsible responsible = new TicketResponsible(user);
  161. responsibles.add(responsible);
  162. if (user.username.equals(ticket.responsible)) {
  163. responsibleModel.setObject(responsible);
  164. }
  165. }
  166. }
  167. Collections.sort(responsibles);
  168. responsibles.add(new TicketResponsible(NIL, "", ""));
  169. Fragment responsible = new Fragment("responsible", "responsibleFragment", this);
  170. responsible.add(new DropDownChoice<TicketResponsible>("responsible", responsibleModel, responsibles));
  171. form.add(responsible.setVisible(!responsibles.isEmpty()));
  172. // milestone
  173. List<TicketMilestone> milestones = app().tickets().getMilestones(getRepositoryModel(), Status.Open);
  174. for (TicketMilestone milestone : milestones) {
  175. if (milestone.name.equals(ticket.milestone)) {
  176. milestoneModel.setObject(milestone);
  177. break;
  178. }
  179. }
  180. if (milestoneModel.getObject() == null && !StringUtils.isEmpty(ticket.milestone)) {
  181. // ensure that this unrecognized milestone is listed
  182. // so that we get the <nil> selection.
  183. TicketMilestone tms = new TicketMilestone(ticket.milestone);
  184. milestones.add(tms);
  185. milestoneModel.setObject(tms);
  186. }
  187. if (!milestones.isEmpty()) {
  188. milestones.add(new TicketMilestone(NIL));
  189. }
  190. // milestone
  191. Fragment milestone = new Fragment("milestone", "milestoneFragment", this);
  192. milestone.add(new DropDownChoice<TicketMilestone>("milestone", milestoneModel, milestones));
  193. form.add(milestone.setVisible(!milestones.isEmpty()));
  194. // priority
  195. Fragment priority = new Fragment("priority", "priorityFragment", this);
  196. List<TicketModel.Priority> priorityChoices = Arrays.asList(TicketModel.Priority.choices());
  197. priority.add(new DropDownChoice<TicketModel.Priority>("priority", priorityModel, priorityChoices));
  198. form.add(priority);
  199. // mergeTo (integration branch)
  200. List<String> branches = new ArrayList<String>();
  201. for (String branch : getRepositoryModel().getLocalBranches()) {
  202. // exclude ticket branches
  203. if (!branch.startsWith(Constants.R_TICKET)) {
  204. branches.add(Repository.shortenRefName(branch));
  205. }
  206. }
  207. branches.remove(Repository.shortenRefName(getRepositoryModel().mergeTo));
  208. branches.add(0, Repository.shortenRefName(getRepositoryModel().mergeTo));
  209. Fragment mergeto = new Fragment("mergeto", "mergeToFragment", this);
  210. mergeto.add(new DropDownChoice<String>("mergeto", mergeToModel, branches));
  211. form.add(mergeto.setVisible(!branches.isEmpty()));
  212. } else {
  213. // user can not admin this ticket
  214. form.add(new Label("responsible").setVisible(false));
  215. form.add(new Label("milestone").setVisible(false));
  216. form.add(new Label("mergeto").setVisible(false));
  217. form.add(new Label("priority").setVisible(false));
  218. }
  219. form.add(new AjaxButton("update") {
  220. private static final long serialVersionUID = 1L;
  221. @Override
  222. protected void onSubmit(AjaxRequestTarget target, Form<?> form) {
  223. long ticketId = 0L;
  224. try {
  225. String h = WicketUtils.getObject(getPageParameters());
  226. ticketId = Long.parseLong(h);
  227. } catch (Exception e) {
  228. setResponsePage(TicketsPage.class, WicketUtils.newRepositoryParameter(repositoryName));
  229. }
  230. TicketModel ticket = app().tickets().getTicket(getRepositoryModel(), ticketId);
  231. String createdBy = GitBlitWebSession.get().getUsername();
  232. Change change = new Change(createdBy);
  233. String title = titleModel.getObject();
  234. if (StringUtils.isEmpty(title)) {
  235. return;
  236. }
  237. if (!ticket.title.equals(title)) {
  238. // title change
  239. change.setField(Field.title, title);
  240. }
  241. String description = Optional.fromNullable(descriptionEditor.getText()).or("");
  242. if ((StringUtils.isEmpty(ticket.body) && !StringUtils.isEmpty(description))
  243. || (!StringUtils.isEmpty(ticket.body) && !ticket.body.equals(description))) {
  244. // description change
  245. change.setField(Field.body, description);
  246. }
  247. Status status = statusModel.getObject();
  248. if (!ticket.status.equals(status)) {
  249. // status change
  250. change.setField(Field.status, status);
  251. }
  252. Type type = typeModel.getObject();
  253. if (!ticket.type.equals(type)) {
  254. // type change
  255. change.setField(Field.type, type);
  256. }
  257. String topic = Optional.fromNullable(topicModel.getObject()).or("");
  258. if ((StringUtils.isEmpty(ticket.topic) && !StringUtils.isEmpty(topic))
  259. || (!StringUtils.isEmpty(ticket.topic) && !ticket.topic.equals(topic))) {
  260. // topic change
  261. change.setField(Field.topic, topic);
  262. }
  263. TicketResponsible responsible = responsibleModel == null ? null : responsibleModel.getObject();
  264. if (responsible != null && !responsible.username.equals(ticket.responsible)) {
  265. // responsible change
  266. change.setField(Field.responsible, responsible.username);
  267. if (!StringUtils.isEmpty(responsible.username)) {
  268. if (!ticket.isWatching(responsible.username)) {
  269. change.watch(responsible.username);
  270. }
  271. }
  272. }
  273. TicketMilestone milestone = milestoneModel == null ? null : milestoneModel.getObject();
  274. if (milestone != null && !milestone.name.equals(ticket.milestone)) {
  275. // milestone change
  276. if (NIL.equals(milestone.name)) {
  277. change.setField(Field.milestone, "");
  278. } else {
  279. change.setField(Field.milestone, milestone.name);
  280. }
  281. }
  282. TicketModel.Priority priority = priorityModel.getObject();
  283. if (!ticket.priority.equals(priority))
  284. {
  285. change.setField(Field.priority, priority);
  286. }
  287. TicketModel.Severity severity = severityModel.getObject();
  288. if (!ticket.severity.equals(severity))
  289. {
  290. change.setField(Field.severity, severity);
  291. }
  292. String mergeTo = mergeToModel.getObject();
  293. if ((StringUtils.isEmpty(ticket.mergeTo) && !StringUtils.isEmpty(mergeTo))
  294. || (!StringUtils.isEmpty(mergeTo) && !mergeTo.equals(ticket.mergeTo))) {
  295. // integration branch change
  296. change.setField(Field.mergeTo, mergeTo);
  297. }
  298. if (change.hasFieldChanges()) {
  299. if (!ticket.isWatching(createdBy)) {
  300. change.watch(createdBy);
  301. }
  302. ticket = app().tickets().updateTicket(getRepositoryModel(), ticket.number, change);
  303. if (ticket != null) {
  304. TicketNotifier notifier = app().tickets().createNotifier();
  305. notifier.sendMailing(ticket);
  306. redirectTo(TicketsPage.class, WicketUtils.newObjectParameter(getRepositoryModel().name, "" + ticket.number));
  307. } else {
  308. // TODO error
  309. }
  310. } else {
  311. // nothing to change?!
  312. redirectTo(TicketsPage.class, WicketUtils.newObjectParameter(getRepositoryModel().name, "" + ticket.number));
  313. }
  314. }
  315. });
  316. Button cancel = new Button("cancel") {
  317. private static final long serialVersionUID = 1L;
  318. @Override
  319. public void onSubmit() {
  320. setResponsePage(TicketsPage.class, getPageParameters());
  321. }
  322. };
  323. cancel.setDefaultFormProcessing(false);
  324. form.add(cancel);
  325. }
  326. @Override
  327. protected String getPageName() {
  328. return getString("gb.editTicket");
  329. }
  330. @Override
  331. protected Class<? extends BasePage> getRepoNavPageClass() {
  332. return TicketsPage.class;
  333. }
  334. }