blob: 61e12878b1d4eb3905e115564be878e669f5df86 [file] [log] [blame]
/*
* Copyright 2014 gitblit.com.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.gitblit.wicket.pages;
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.PageParameters;
import org.apache.wicket.ajax.AjaxRequestTarget;
import org.apache.wicket.ajax.markup.html.form.AjaxButton;
import org.apache.wicket.markup.html.basic.Label;
import org.apache.wicket.markup.html.form.Button;
import org.apache.wicket.markup.html.form.DropDownChoice;
import org.apache.wicket.markup.html.form.Form;
import org.apache.wicket.markup.html.form.TextField;
import org.apache.wicket.markup.html.list.ListItem;
import org.apache.wicket.markup.html.list.ListView;
import org.apache.wicket.markup.html.panel.Fragment;
import org.apache.wicket.model.IModel;
import org.apache.wicket.model.Model;
import org.eclipse.jgit.lib.Repository;
import org.jsoup.helper.StringUtil;
import com.gitblit.Constants;
import com.gitblit.Constants.AccessPermission;
import com.gitblit.Constants.AuthorizationControl;
import com.gitblit.models.RegistrantAccessPermission;
import com.gitblit.models.TicketModel;
import com.gitblit.models.TicketModel.Change;
import com.gitblit.models.TicketModel.Field;
import com.gitblit.models.TicketModel.Status;
import com.gitblit.models.TicketModel.Type;
import com.gitblit.models.UserModel;
import com.gitblit.tickets.TicketMilestone;
import com.gitblit.tickets.TicketNotifier;
import com.gitblit.tickets.TicketResponsible;
import com.gitblit.utils.StringUtils;
import com.gitblit.wicket.GitBlitWebSession;
import com.gitblit.wicket.WicketUtils;
import com.gitblit.wicket.panels.LinkPanel;
import com.gitblit.wicket.panels.MarkdownTextArea;
import com.gitblit.wicket.panels.SimpleAjaxLink;
import com.gitblit.wicket.panels.TicketRelationEditorPanel;
import com.google.common.base.Optional;
/**
* Page for editing a ticket.
*
* @author James Moger
*
*/
public class EditTicketPage extends RepositoryPage {
static final String NIL = "<nil>";
static final String ESC_NIL = StringUtils.escapeForHtml(NIL, false);
private IModel<TicketModel.Type> typeModel;
private IModel<String> titleModel;
private MarkdownTextArea descriptionEditor;
private IModel<String> topicModel;
private IModel<String> mergeToModel;
private IModel<Status> statusModel;
private IModel<TicketResponsible> responsibleModel;
private IModel<TicketMilestone> milestoneModel;
private IModel<List<String>> dependenciesModel;
private Label descriptionPreview;
private IModel<TicketModel.Priority> priorityModel;
private IModel<TicketModel.Severity> severityModel;
public EditTicketPage(PageParameters params) {
super(params);
UserModel currentUser = GitBlitWebSession.get().getUser();
if (currentUser == null) {
currentUser = UserModel.ANONYMOUS;
}
long ticketId = 0L;
try {
String h = WicketUtils.getObject(params);
ticketId = Long.parseLong(h);
} catch (Exception e) {
setResponsePage(TicketsPage.class, WicketUtils.newRepositoryParameter(repositoryName));
}
TicketModel ticket = app().tickets().getTicket(getRepositoryModel(), ticketId);
if (ticket == null
|| !currentUser.canEdit(ticket, getRepositoryModel())
|| !app().tickets().isAcceptingTicketUpdates(getRepositoryModel())) {
setResponsePage(TicketsPage.class, WicketUtils.newObjectParameter(repositoryName, "" + ticketId));
// create a placeholder object so we don't trigger NPEs
ticket = new TicketModel();
}
typeModel = Model.of(ticket.type);
titleModel = Model.of(ticket.title);
topicModel = Model.of(ticket.topic == null ? "" : ticket.topic);
responsibleModel = Model.of();
milestoneModel = Model.of();
mergeToModel = Model.of(ticket.mergeTo == null ? getRepositoryModel().mergeTo : ticket.mergeTo);
statusModel = Model.of(ticket.status);
priorityModel = Model.of(ticket.priority);
severityModel = Model.of(ticket.severity);
dependenciesModel = Model.ofList((List) editable(ticket.getDependencies()));
setStatelessHint(false);
setOutputMarkupId(true);
Form<Void> form = new Form<Void>("editForm");
add(form);
List<Type> typeChoices;
if (ticket.isProposal()) {
typeChoices = Arrays.asList(Type.Proposal);
} else {
typeChoices = Arrays.asList(TicketModel.Type.choices());
}
form.add(new DropDownChoice<TicketModel.Type>("type", typeModel, typeChoices));
form.add(new TextField<String>("title", titleModel));
form.add(new TextField<String>("topic", topicModel));
form.setOutputMarkupId(true);
form.add(new TicketRelationEditorPanel("dependencies", dependenciesModel, getRepositoryModel()));
final IModel<String> markdownPreviewModel = Model.of(ticket.body == null ? "" : ticket.body);
descriptionPreview = new Label("descriptionPreview", markdownPreviewModel);
descriptionPreview.setEscapeModelStrings(false);
descriptionPreview.setOutputMarkupId(true);
form.add(descriptionPreview);
descriptionEditor = new MarkdownTextArea("description", markdownPreviewModel, descriptionPreview);
descriptionEditor.setRepository(repositoryName);
descriptionEditor.setText(ticket.body);
form.add(descriptionEditor);
// status
List<Status> statusChoices;
if (ticket.isClosed()) {
statusChoices = Arrays.asList(ticket.status, Status.Open);
} else if (ticket.isProposal()) {
statusChoices = Arrays.asList(TicketModel.Status.proposalWorkflow);
} else if (ticket.isBug()) {
statusChoices = Arrays.asList(TicketModel.Status.bugWorkflow);
} else {
statusChoices = Arrays.asList(TicketModel.Status.requestWorkflow);
}
Fragment status = new Fragment("status", "statusFragment", this);
status.add(new DropDownChoice<TicketModel.Status>("status", statusModel, statusChoices));
form.add(status);
List<TicketModel.Severity> severityChoices = Arrays.asList(TicketModel.Severity.choices());
form.add(new DropDownChoice<TicketModel.Severity>("severity", severityModel, severityChoices));
if (currentUser.canAdmin(ticket, getRepositoryModel())) {
// responsible
Set<String> userlist = new TreeSet<String>(ticket.getParticipants());
if (UserModel.ANONYMOUS.canPush(getRepositoryModel())
|| AuthorizationControl.AUTHENTICATED == getRepositoryModel().authorizationControl) {
// authorization is ANONYMOUS or AUTHENTICATED (i.e. all users can be set responsible)
userlist.addAll(app().users().getAllUsernames());
} else {
// authorization is by NAMED users (users with PUSH permission can be set responsible)
for (RegistrantAccessPermission rp : app().repositories().getUserAccessPermissions(getRepositoryModel())) {
if (rp.permission.atLeast(AccessPermission.PUSH)) {
userlist.add(rp.registrant);
}
}
}
List<TicketResponsible> responsibles = new ArrayList<TicketResponsible>();
for (String username : userlist) {
UserModel user = app().users().getUserModel(username);
if (user != null && !user.disabled) {
TicketResponsible responsible = new TicketResponsible(user);
responsibles.add(responsible);
if (user.username.equals(ticket.responsible)) {
responsibleModel.setObject(responsible);
}
}
}
Collections.sort(responsibles);
responsibles.add(new TicketResponsible(NIL, "", ""));
Fragment responsible = new Fragment("responsible", "responsibleFragment", this);
responsible.add(new DropDownChoice<TicketResponsible>("responsible", responsibleModel, responsibles));
form.add(responsible.setVisible(!responsibles.isEmpty()));
// milestone
List<TicketMilestone> milestones = app().tickets().getMilestones(getRepositoryModel(), Status.Open);
for (TicketMilestone milestone : milestones) {
if (milestone.name.equals(ticket.milestone)) {
milestoneModel.setObject(milestone);
break;
}
}
if (milestoneModel.getObject() == null && !StringUtils.isEmpty(ticket.milestone)) {
// ensure that this unrecognized milestone is listed
// so that we get the <nil> selection.
TicketMilestone tms = new TicketMilestone(ticket.milestone);
milestones.add(tms);
milestoneModel.setObject(tms);
}
if (!milestones.isEmpty()) {
milestones.add(new TicketMilestone(NIL));
}
// milestone
Fragment milestone = new Fragment("milestone", "milestoneFragment", this);
milestone.add(new DropDownChoice<TicketMilestone>("milestone", milestoneModel, milestones));
form.add(milestone.setVisible(!milestones.isEmpty()));
// priority
Fragment priority = new Fragment("priority", "priorityFragment", this);
List<TicketModel.Priority> priorityChoices = Arrays.asList(TicketModel.Priority.choices());
priority.add(new DropDownChoice<TicketModel.Priority>("priority", priorityModel, priorityChoices));
form.add(priority);
// mergeTo (integration branch)
List<String> branches = new ArrayList<String>();
for (String branch : getRepositoryModel().getLocalBranches()) {
// exclude ticket branches
if (!branch.startsWith(Constants.R_TICKET)) {
branches.add(Repository.shortenRefName(branch));
}
}
branches.remove(Repository.shortenRefName(getRepositoryModel().mergeTo));
branches.add(0, Repository.shortenRefName(getRepositoryModel().mergeTo));
Fragment mergeto = new Fragment("mergeto", "mergeToFragment", this);
mergeto.add(new DropDownChoice<String>("mergeto", mergeToModel, branches));
form.add(mergeto.setVisible(!branches.isEmpty()));
} else {
// user can not admin this ticket
form.add(new Label("responsible").setVisible(false));
form.add(new Label("milestone").setVisible(false));
form.add(new Label("mergeto").setVisible(false));
form.add(new Label("priority").setVisible(false));
}
form.add(new AjaxButton("update") {
private static final long serialVersionUID = 1L;
@Override
protected void onSubmit(AjaxRequestTarget target, Form<?> form) {
long ticketId = 0L;
try {
String h = WicketUtils.getObject(getPageParameters());
ticketId = Long.parseLong(h);
} catch (Exception e) {
setResponsePage(TicketsPage.class, WicketUtils.newRepositoryParameter(repositoryName));
}
TicketModel ticket = app().tickets().getTicket(getRepositoryModel(), ticketId);
String createdBy = GitBlitWebSession.get().getUsername();
Change change = new Change(createdBy);
String title = titleModel.getObject();
if (StringUtils.isEmpty(title)) {
return;
}
if (!ticket.title.equals(title)) {
// title change
change.setField(Field.title, title);
}
String description = Optional.fromNullable(descriptionEditor.getText()).or("");
if ((StringUtils.isEmpty(ticket.body) && !StringUtils.isEmpty(description))
|| (!StringUtils.isEmpty(ticket.body) && !ticket.body.equals(description))) {
// description change
change.setField(Field.body, description);
}
Status status = statusModel.getObject();
if (!ticket.status.equals(status)) {
// status change
change.setField(Field.status, status);
}
Type type = typeModel.getObject();
if (!ticket.type.equals(type)) {
// type change
change.setField(Field.type, type);
}
String topic = Optional.fromNullable(topicModel.getObject()).or("");
if ((StringUtils.isEmpty(ticket.topic) && !StringUtils.isEmpty(topic))
|| (!StringUtils.isEmpty(ticket.topic) && !ticket.topic.equals(topic))) {
// topic change
change.setField(Field.topic, topic);
}
List<String> newDependencies = (List<String>) dependenciesModel.getObject();
change.setDeltaField(Field.dependency, ticket.getDependencies(), newDependencies);
TicketResponsible responsible = responsibleModel == null ? null : responsibleModel.getObject();
if (responsible != null && !responsible.username.equals(ticket.responsible)) {
// responsible change
change.setField(Field.responsible, responsible.username);
if (!StringUtils.isEmpty(responsible.username)) {
if (!ticket.isWatching(responsible.username)) {
change.watch(responsible.username);
}
}
}
TicketMilestone milestone = milestoneModel == null ? null : milestoneModel.getObject();
if (milestone != null && !milestone.name.equals(ticket.milestone)) {
// milestone change
if (NIL.equals(milestone.name)) {
change.setField(Field.milestone, "");
} else {
change.setField(Field.milestone, milestone.name);
}
}
TicketModel.Priority priority = priorityModel.getObject();
if (!ticket.priority.equals(priority))
{
change.setField(Field.priority, priority);
}
TicketModel.Severity severity = severityModel.getObject();
if (!ticket.severity.equals(severity))
{
change.setField(Field.severity, severity);
}
String mergeTo = mergeToModel.getObject();
if ((StringUtils.isEmpty(ticket.mergeTo) && !StringUtils.isEmpty(mergeTo))
|| (!StringUtils.isEmpty(mergeTo) && !mergeTo.equals(ticket.mergeTo))) {
// integration branch change
change.setField(Field.mergeTo, mergeTo);
}
if (change.hasFieldChanges()) {
if (!ticket.isWatching(createdBy)) {
change.watch(createdBy);
}
ticket = app().tickets().updateTicket(getRepositoryModel(), ticket.number, change);
if (ticket != null) {
TicketNotifier notifier = app().tickets().createNotifier();
notifier.sendMailing(ticket);
redirectTo(TicketsPage.class, WicketUtils.newObjectParameter(getRepositoryModel().name, "" + ticket.number));
} else {
// TODO error
}
} else {
// nothing to change?!
redirectTo(TicketsPage.class, WicketUtils.newObjectParameter(getRepositoryModel().name, "" + ticket.number));
}
}
});
Button cancel = new Button("cancel") {
private static final long serialVersionUID = 1L;
@Override
public void onSubmit() {
setResponsePage(TicketsPage.class, getPageParameters());
}
};
cancel.setDefaultFormProcessing(false);
form.add(cancel);
}
private List<String> editable(List<String> list) {
// need to copy, if it's an Collection.emptyList
return new ArrayList<String>(list);
}
@Override
protected String getPageName() {
return getString("gb.editTicket");
}
@Override
protected Class<? extends BasePage> getRepoNavPageClass() {
return TicketsPage.class;
}
}