Port ApprovalTable to new REST API
The table now renders from a new-style ChangeInfo JSON object rather than a ChangeDetail object. The structure of the new JSON object is more suited to rendering the approval table, so the code should be simplified somewhat. Change-Id: I708068b6a5e359ebf5065bb52b23b78557634969
This commit is contained in:
parent
f22f39ebad
commit
051e7313ba
@ -21,8 +21,10 @@ import com.google.gerrit.reviewdb.client.PatchSetApproval;
|
|||||||
import java.sql.Timestamp;
|
import java.sql.Timestamp;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Comparator;
|
import java.util.Comparator;
|
||||||
|
import java.util.HashMap;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
public class ApprovalDetail {
|
public class ApprovalDetail {
|
||||||
@ -46,6 +48,7 @@ public class ApprovalDetail {
|
|||||||
|
|
||||||
private transient Set<String> approved;
|
private transient Set<String> approved;
|
||||||
private transient Set<String> rejected;
|
private transient Set<String> rejected;
|
||||||
|
private transient Map<String, Integer> values;
|
||||||
private transient int hasNonZero;
|
private transient int hasNonZero;
|
||||||
private transient Timestamp sortOrder = EG_D;
|
private transient Timestamp sortOrder = EG_D;
|
||||||
|
|
||||||
@ -69,10 +72,12 @@ public class ApprovalDetail {
|
|||||||
canRemove = removeable;
|
canRemove = removeable;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Deprecated
|
||||||
public List<PatchSetApproval> getPatchSetApprovals() {
|
public List<PatchSetApproval> getPatchSetApprovals() {
|
||||||
return approvals;
|
return approvals;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Deprecated
|
||||||
public PatchSetApproval getPatchSetApproval(ApprovalCategory.Id category) {
|
public PatchSetApproval getPatchSetApproval(ApprovalCategory.Id category) {
|
||||||
for (PatchSetApproval psa : approvals) {
|
for (PatchSetApproval psa : approvals) {
|
||||||
if (psa.getCategoryId().equals(category)) {
|
if (psa.getCategoryId().equals(category)) {
|
||||||
@ -87,12 +92,15 @@ public class ApprovalDetail {
|
|||||||
sortOrder = ApprovalDetail.EG_0;
|
sortOrder = ApprovalDetail.EG_0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Deprecated
|
||||||
public void add(final PatchSetApproval ca) {
|
public void add(final PatchSetApproval ca) {
|
||||||
approvals.add(ca);
|
approvals.add(ca);
|
||||||
|
|
||||||
final Timestamp g = ca.getGranted();
|
final Timestamp g = ca.getGranted();
|
||||||
if (g != null && g.compareTo(sortOrder) < 0) {
|
if (g != null && g.compareTo(sortOrder) < 0) {
|
||||||
sortOrder = g;
|
sortOrder = g;
|
||||||
|
// Value is not set, but code calling this deprecated method does not
|
||||||
|
// call getValue.
|
||||||
}
|
}
|
||||||
if (ca.getValue() != 0) {
|
if (ca.getValue() != 0) {
|
||||||
hasNonZero = 1;
|
hasNonZero = 1;
|
||||||
@ -120,6 +128,13 @@ public class ApprovalDetail {
|
|||||||
votable.add(label);
|
votable.add(label);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void value(String label, int value) {
|
||||||
|
if (values == null) {
|
||||||
|
values = new HashMap<String, Integer>();
|
||||||
|
}
|
||||||
|
values.put(label, value);
|
||||||
|
}
|
||||||
|
|
||||||
public boolean isApproved(String label) {
|
public boolean isApproved(String label) {
|
||||||
return approved != null && approved.contains(label);
|
return approved != null && approved.contains(label);
|
||||||
}
|
}
|
||||||
@ -131,4 +146,12 @@ public class ApprovalDetail {
|
|||||||
public boolean canVote(String label) {
|
public boolean canVote(String label) {
|
||||||
return votable != null && votable.contains(label);
|
return votable != null && votable.contains(label);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public int getValue(String label) {
|
||||||
|
if (values == null) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
Integer v = values.get(label);
|
||||||
|
return v != null ? v : 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,31 @@
|
|||||||
|
// Copyright (C) 2013 The Android Open Source Project
|
||||||
|
//
|
||||||
|
// 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.google.gerrit.client.changes;
|
||||||
|
|
||||||
|
import com.google.gwt.core.client.JavaScriptObject;
|
||||||
|
|
||||||
|
public class AccountInfo extends JavaScriptObject {
|
||||||
|
public final native int _account_id() /*-{ return this._account_id || 0; }-*/;
|
||||||
|
public final native String name() /*-{ return this.name; }-*/;
|
||||||
|
public final native String email() /*-{ return this.email; }-*/;
|
||||||
|
|
||||||
|
public static native AccountInfo create(int id, String name,
|
||||||
|
String email) /*-{
|
||||||
|
return {'_account_id': id, 'name': name, 'email': email};
|
||||||
|
}-*/;
|
||||||
|
|
||||||
|
protected AccountInfo() {
|
||||||
|
}
|
||||||
|
}
|
@ -14,28 +14,29 @@
|
|||||||
|
|
||||||
package com.google.gerrit.client.changes;
|
package com.google.gerrit.client.changes;
|
||||||
|
|
||||||
|
import static com.google.gerrit.reviewdb.client.ApprovalCategoryValue.formatValue;
|
||||||
|
|
||||||
import com.google.gerrit.client.ConfirmationCallback;
|
import com.google.gerrit.client.ConfirmationCallback;
|
||||||
import com.google.gerrit.client.ConfirmationDialog;
|
import com.google.gerrit.client.ConfirmationDialog;
|
||||||
import com.google.gerrit.client.ErrorDialog;
|
import com.google.gerrit.client.ErrorDialog;
|
||||||
import com.google.gerrit.client.FormatUtil;
|
|
||||||
import com.google.gerrit.client.Gerrit;
|
import com.google.gerrit.client.Gerrit;
|
||||||
import com.google.gerrit.client.patches.PatchUtil;
|
import com.google.gerrit.client.changes.ChangeInfo.ApprovalInfo;
|
||||||
|
import com.google.gerrit.client.changes.ChangeInfo.LabelInfo;
|
||||||
import com.google.gerrit.client.rpc.GerritCallback;
|
import com.google.gerrit.client.rpc.GerritCallback;
|
||||||
import com.google.gerrit.client.ui.AccountLink;
|
import com.google.gerrit.client.rpc.NativeMap;
|
||||||
|
import com.google.gerrit.client.rpc.NativeString;
|
||||||
|
import com.google.gerrit.client.rpc.Natives;
|
||||||
import com.google.gerrit.client.ui.AddMemberBox;
|
import com.google.gerrit.client.ui.AddMemberBox;
|
||||||
|
import com.google.gerrit.client.ui.InlineHyperlink;
|
||||||
import com.google.gerrit.client.ui.ReviewerSuggestOracle;
|
import com.google.gerrit.client.ui.ReviewerSuggestOracle;
|
||||||
import com.google.gerrit.common.data.AccountInfoCache;
|
import com.google.gerrit.common.PageLinks;
|
||||||
import com.google.gerrit.common.data.ApprovalDetail;
|
import com.google.gerrit.common.data.ApprovalDetail;
|
||||||
import com.google.gerrit.common.data.ApprovalType;
|
import com.google.gerrit.common.data.ApprovalType;
|
||||||
import com.google.gerrit.common.data.ApprovalTypes;
|
import com.google.gerrit.common.data.ApprovalTypes;
|
||||||
import com.google.gerrit.common.data.ChangeDetail;
|
|
||||||
import com.google.gerrit.common.data.PatchSetPublishDetail;
|
|
||||||
import com.google.gerrit.common.data.ReviewerResult;
|
|
||||||
import com.google.gerrit.common.data.SubmitRecord;
|
import com.google.gerrit.common.data.SubmitRecord;
|
||||||
import com.google.gerrit.reviewdb.client.Account;
|
import com.google.gerrit.reviewdb.client.Account;
|
||||||
import com.google.gerrit.reviewdb.client.Change;
|
|
||||||
import com.google.gerrit.reviewdb.client.PatchSetApproval;
|
|
||||||
import com.google.gwt.core.client.JavaScriptObject;
|
import com.google.gwt.core.client.JavaScriptObject;
|
||||||
|
import com.google.gwt.core.client.JsArray;
|
||||||
import com.google.gwt.event.dom.client.ClickEvent;
|
import com.google.gwt.event.dom.client.ClickEvent;
|
||||||
import com.google.gwt.event.dom.client.ClickHandler;
|
import com.google.gwt.event.dom.client.ClickHandler;
|
||||||
import com.google.gwt.user.client.DOM;
|
import com.google.gwt.user.client.DOM;
|
||||||
@ -51,24 +52,34 @@ import com.google.gwt.user.client.ui.Widget;
|
|||||||
import com.google.gwtexpui.safehtml.client.SafeHtmlBuilder;
|
import com.google.gwtexpui.safehtml.client.SafeHtmlBuilder;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Collections;
|
import java.util.Collection;
|
||||||
import java.util.Comparator;
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
/** Displays a table of {@link ApprovalDetail} objects for a change record. */
|
/** Displays a table of {@link ApprovalDetail} objects for a change record. */
|
||||||
public class ApprovalTable extends Composite {
|
public class ApprovalTable extends Composite {
|
||||||
|
static short parseLabelValue(String value) {
|
||||||
|
if (value.charAt(0) == ' ' || value.charAt(0) == '+') {
|
||||||
|
value = value.substring(1);
|
||||||
|
}
|
||||||
|
return Short.parseShort(value);
|
||||||
|
}
|
||||||
|
|
||||||
private final ApprovalTypes types;
|
private final ApprovalTypes types;
|
||||||
private final Grid table;
|
private final Grid table;
|
||||||
private final Widget missing;
|
private final Widget missing;
|
||||||
private final Panel addReviewer;
|
private final Panel addReviewer;
|
||||||
private final ReviewerSuggestOracle reviewerSuggestOracle;
|
private final ReviewerSuggestOracle reviewerSuggestOracle;
|
||||||
private final AddMemberBox addMemberBox;
|
private final AddMemberBox addMemberBox;
|
||||||
private Change.Id changeId;
|
private ChangeInfo lastChange;
|
||||||
private AccountInfoCache accountCache = AccountInfoCache.empty();
|
private Map<Integer, Integer> rows;
|
||||||
|
|
||||||
public ApprovalTable() {
|
public ApprovalTable() {
|
||||||
|
rows = new HashMap<Integer, Integer>();
|
||||||
types = Gerrit.getConfig().getApprovalTypes();
|
types = Gerrit.getConfig().getApprovalTypes();
|
||||||
table = new Grid(1, 3);
|
table = new Grid(1, 3);
|
||||||
table.addStyleName(Gerrit.RESOURCES.css().infoTable());
|
table.addStyleName(Gerrit.RESOURCES.css().infoTable());
|
||||||
@ -104,7 +115,7 @@ public class ApprovalTable extends Composite {
|
|||||||
setStyleName(Gerrit.RESOURCES.css().approvalTable());
|
setStyleName(Gerrit.RESOURCES.css().approvalTable());
|
||||||
}
|
}
|
||||||
|
|
||||||
private void displayHeader(List<String> labels) {
|
private void displayHeader(Collection<String> labels) {
|
||||||
table.resizeColumns(2 + labels.size());
|
table.resizeColumns(2 + labels.size());
|
||||||
|
|
||||||
final CellFormatter fmt = table.getCellFormatter();
|
final CellFormatter fmt = table.getCellFormatter();
|
||||||
@ -126,263 +137,212 @@ public class ApprovalTable extends Composite {
|
|||||||
fmt.addStyleName(0, col - 1, Gerrit.RESOURCES.css().rightmost());
|
fmt.addStyleName(0, col - 1, Gerrit.RESOURCES.css().rightmost());
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setAccountInfoCache(final AccountInfoCache aic) {
|
void display(ChangeInfo change) {
|
||||||
assert aic != null;
|
lastChange = change;
|
||||||
accountCache = aic;
|
reviewerSuggestOracle.setChange(change.legacy_id());
|
||||||
}
|
Map<Integer, ApprovalDetail> byUser =
|
||||||
|
new LinkedHashMap<Integer, ApprovalDetail>();
|
||||||
|
Map<Integer, AccountInfo> accounts =
|
||||||
|
new LinkedHashMap<Integer, AccountInfo>();
|
||||||
|
List<String> missingLabels = initLabels(change, accounts, byUser);
|
||||||
|
|
||||||
private AccountLink link(final Account.Id id) {
|
removeAllChildren(missing.getElement());
|
||||||
return AccountLink.link(accountCache, id);
|
for (String label : missingLabels) {
|
||||||
}
|
addMissingLabel(Util.M.needApproval(label));
|
||||||
|
|
||||||
void display(PatchSetPublishDetail detail) {
|
|
||||||
doDisplay(detail.getChange(), detail.getApprovals(),
|
|
||||||
detail.getSubmitRecords());
|
|
||||||
}
|
|
||||||
|
|
||||||
void display(ChangeDetail detail) {
|
|
||||||
doDisplay(detail.getChange(), detail.getApprovals(),
|
|
||||||
detail.getSubmitRecords());
|
|
||||||
}
|
|
||||||
|
|
||||||
private void doDisplay(Change change, List<ApprovalDetail> approvals,
|
|
||||||
List<SubmitRecord> submitRecords) {
|
|
||||||
changeId = change.getId();
|
|
||||||
reviewerSuggestOracle.setChange(changeId);
|
|
||||||
|
|
||||||
List<String> columns = new ArrayList<String>();
|
|
||||||
|
|
||||||
final Element missingList = missing.getElement();
|
|
||||||
while (DOM.getChildCount(missingList) > 0) {
|
|
||||||
DOM.removeChild(missingList, DOM.getChild(missingList, 0));
|
|
||||||
}
|
|
||||||
missing.setVisible(false);
|
|
||||||
|
|
||||||
if (submitRecords != null) {
|
|
||||||
HashSet<String> reportedMissing = new HashSet<String>();
|
|
||||||
|
|
||||||
HashMap<Account.Id, ApprovalDetail> byUser =
|
|
||||||
new HashMap<Account.Id, ApprovalDetail>();
|
|
||||||
for (ApprovalDetail ad : approvals) {
|
|
||||||
byUser.put(ad.getAccount(), ad);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (SubmitRecord rec : submitRecords) {
|
|
||||||
if (rec.labels == null) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (SubmitRecord.Label lbl : rec.labels) {
|
|
||||||
if (!columns.contains(lbl.label)) {
|
|
||||||
columns.add(lbl.label);
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (lbl.status) {
|
|
||||||
case OK: {
|
|
||||||
ApprovalDetail ad = byUser.get(lbl.appliedBy);
|
|
||||||
if (ad != null) {
|
|
||||||
ad.approved(lbl.label);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case REJECT: {
|
|
||||||
ApprovalDetail ad = byUser.get(lbl.appliedBy);
|
|
||||||
if (ad != null) {
|
|
||||||
ad.rejected(lbl.label);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case MAY:
|
|
||||||
break;
|
|
||||||
|
|
||||||
case NEED:
|
|
||||||
case IMPOSSIBLE:
|
|
||||||
if (reportedMissing.add(lbl.label)) {
|
|
||||||
Element li = DOM.createElement("li");
|
|
||||||
li.setClassName(Gerrit.RESOURCES.css().missingApproval());
|
|
||||||
DOM.setInnerText(li, Util.M.needApproval(lbl.label));
|
|
||||||
DOM.appendChild(missingList, li);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
missing.setVisible(!reportedMissing.isEmpty());
|
|
||||||
|
|
||||||
} else {
|
|
||||||
for (ApprovalDetail ad : approvals) {
|
|
||||||
for (PatchSetApproval psa : ad.getPatchSetApprovals()) {
|
|
||||||
ApprovalType legacyType = types.byId(psa.getCategoryId());
|
|
||||||
if (legacyType == null) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
String labelName = legacyType.getCategory().getLabelName();
|
|
||||||
if (psa.getValue() != 0 ) {
|
|
||||||
if (psa.getValue() == legacyType.getMax().getValue()) {
|
|
||||||
ad.approved(labelName);
|
|
||||||
} else if (psa.getValue() == legacyType.getMin().getValue()) {
|
|
||||||
ad.rejected(labelName);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!columns.contains(labelName)) {
|
|
||||||
columns.add(labelName);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Collections.sort(columns, new Comparator<String>() {
|
|
||||||
@Override
|
|
||||||
public int compare(String o1, String o2) {
|
|
||||||
ApprovalType a = types.byLabel(o1);
|
|
||||||
ApprovalType b = types.byLabel(o2);
|
|
||||||
int cmp = 0;
|
|
||||||
if (a != null && b != null) {
|
|
||||||
cmp = a.getCategory().getPosition() - b.getCategory().getPosition();
|
|
||||||
}
|
|
||||||
if (cmp == 0) {
|
|
||||||
cmp = o1.compareTo(o2);
|
|
||||||
}
|
|
||||||
return cmp;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (approvals.isEmpty()) {
|
if (byUser.isEmpty()) {
|
||||||
table.setVisible(false);
|
table.setVisible(false);
|
||||||
} else {
|
} else {
|
||||||
displayHeader(columns);
|
displayHeader(change.labels());
|
||||||
table.resizeRows(1 + approvals.size());
|
table.resizeRows(1 + byUser.size());
|
||||||
for (int i = 0; i < approvals.size(); i++) {
|
int i = 1;
|
||||||
displayRow(i + 1, approvals.get(i), change, columns);
|
for (ApprovalDetail ad : byUser.values()) {
|
||||||
|
displayRow(i++, ad, change, accounts.get(ad.getAccount().get()));
|
||||||
}
|
}
|
||||||
table.setVisible(true);
|
table.setVisible(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
addReviewer.setVisible(Gerrit.isSignedIn());
|
|
||||||
|
|
||||||
if (Gerrit.getConfig().testChangeMerge()
|
if (Gerrit.getConfig().testChangeMerge()
|
||||||
&& !change.isMergeable()) {
|
&& !change.mergeable()) {
|
||||||
Element li = DOM.createElement("li");
|
addMissingLabel(Util.C.messageNeedsRebaseOrHasDependency());
|
||||||
li.setClassName(Gerrit.RESOURCES.css().missingApproval());
|
|
||||||
DOM.setInnerText(li, Util.C.messageNeedsRebaseOrHasDependency());
|
|
||||||
DOM.appendChild(missingList, li);
|
|
||||||
missing.setVisible(true);
|
|
||||||
}
|
}
|
||||||
|
missing.setVisible(DOM.getChildCount(missing.getElement()) > 0);
|
||||||
|
addReviewer.setVisible(Gerrit.isSignedIn());
|
||||||
|
}
|
||||||
|
|
||||||
|
private void removeAllChildren(Element el) {
|
||||||
|
for (int i = DOM.getChildCount(el) - 1; i >= 0; i--) {
|
||||||
|
DOM.removeChild(el, DOM.getChild(el, i));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void addMissingLabel(String text) {
|
||||||
|
Element li = DOM.createElement("li");
|
||||||
|
li.setClassName(Gerrit.RESOURCES.css().missingApproval());
|
||||||
|
DOM.setInnerText(li, text);
|
||||||
|
DOM.appendChild(missing.getElement(), li);
|
||||||
|
}
|
||||||
|
|
||||||
|
private Set<Integer> removableReviewers(ChangeInfo change) {
|
||||||
|
Set<Integer> result =
|
||||||
|
new HashSet<Integer>(change.removable_reviewers().length());
|
||||||
|
for (int i = 0; i < change.removable_reviewers().length(); i++) {
|
||||||
|
result.add(change.removable_reviewers().get(i)._account_id());
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<String> initLabels(ChangeInfo change,
|
||||||
|
Map<Integer, AccountInfo> accounts,
|
||||||
|
Map<Integer, ApprovalDetail> byUser) {
|
||||||
|
Set<Integer> removableReviewers = removableReviewers(change);
|
||||||
|
List<String> missing = new ArrayList<String>();
|
||||||
|
for (String name : change.labels()) {
|
||||||
|
LabelInfo label = change.label(name);
|
||||||
|
|
||||||
|
String min = null;
|
||||||
|
String max = null;
|
||||||
|
for (String v : label.values()) {
|
||||||
|
if (min == null) {
|
||||||
|
min = v;
|
||||||
|
}
|
||||||
|
if (v.startsWith("+")) {
|
||||||
|
max = v;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (label.status() == SubmitRecord.Label.Status.NEED) {
|
||||||
|
missing.add(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (label.all() != null) {
|
||||||
|
for (ApprovalInfo ai : Natives.asList(label.all())) {
|
||||||
|
if (!accounts.containsKey(ai._account_id())) {
|
||||||
|
accounts.put(ai._account_id(), ai);
|
||||||
|
}
|
||||||
|
int id = ai._account_id();
|
||||||
|
ApprovalDetail ad = byUser.get(id);
|
||||||
|
if (ad == null) {
|
||||||
|
ad = new ApprovalDetail(new Account.Id(id));
|
||||||
|
ad.setCanRemove(removableReviewers.contains(id));
|
||||||
|
byUser.put(id, ad);
|
||||||
|
}
|
||||||
|
ad.votable(name);
|
||||||
|
ad.value(name, ai.value());
|
||||||
|
if (formatValue(ai.value()).equals(max)) {
|
||||||
|
ad.approved(name);
|
||||||
|
} else if (formatValue(ai.value()).equals(min)) {
|
||||||
|
ad.rejected(name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return missing;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void doAddReviewer() {
|
private void doAddReviewer() {
|
||||||
final String reviewer = addMemberBox.getText();
|
String reviewer = addMemberBox.getText();
|
||||||
if (reviewer.length() == 0) {
|
if (!reviewer.isEmpty()) {
|
||||||
return;
|
addMemberBox.setEnabled(false);
|
||||||
|
addReviewer(reviewer, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
addMemberBox.setEnabled(false);
|
|
||||||
final List<String> reviewers = new ArrayList<String>();
|
|
||||||
reviewers.add(reviewer);
|
|
||||||
|
|
||||||
addReviewers(reviewers, false);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void addReviewers(final List<String> reviewers,
|
private static class PostInput extends JavaScriptObject {
|
||||||
final boolean confirmed) {
|
static PostInput create(String reviewer, boolean confirmed) {
|
||||||
PatchUtil.DETAIL_SVC.addReviewers(changeId, reviewers, confirmed,
|
PostInput input = createObject().cast();
|
||||||
new GerritCallback<ReviewerResult>() {
|
input.init(reviewer, confirmed);
|
||||||
public void onSuccess(final ReviewerResult result) {
|
return input;
|
||||||
|
}
|
||||||
|
|
||||||
|
private native void init(String reviewer, boolean confirmed) /*-{
|
||||||
|
this.reviewer = reviewer;
|
||||||
|
if (confirmed) {
|
||||||
|
this.confirmed = true;
|
||||||
|
}
|
||||||
|
}-*/;
|
||||||
|
|
||||||
|
protected PostInput() {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class ReviewerInfo extends AccountInfo {
|
||||||
|
final Set<String> approvals() {
|
||||||
|
return Natives.keys(_approvals());
|
||||||
|
}
|
||||||
|
final native String approval(String l) /*-{ return this.approvals[l]; }-*/;
|
||||||
|
private final native NativeMap<NativeString> _approvals() /*-{ return this.approvals; }-*/;
|
||||||
|
|
||||||
|
protected ReviewerInfo() {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class PostResult extends JavaScriptObject {
|
||||||
|
final native JsArray<ReviewerInfo> reviewers() /*-{ return this.reviewers; }-*/;
|
||||||
|
final native boolean confirm() /*-{ return this.confirm || false; }-*/;
|
||||||
|
final native String error() /*-{ return this.error; }-*/;
|
||||||
|
|
||||||
|
protected PostResult() {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void addReviewer(final String reviewer, boolean confirmed) {
|
||||||
|
ChangeApi.reviewers(lastChange.legacy_id().get()).post(
|
||||||
|
PostInput.create(reviewer, confirmed),
|
||||||
|
new GerritCallback<PostResult>() {
|
||||||
|
public void onSuccess(PostResult result) {
|
||||||
addMemberBox.setEnabled(true);
|
addMemberBox.setEnabled(true);
|
||||||
addMemberBox.setText("");
|
addMemberBox.setText("");
|
||||||
|
if (result.error() == null) {
|
||||||
final ChangeDetail changeDetail = result.getChange();
|
reload();
|
||||||
if (changeDetail != null) {
|
} else if (result.confirm()) {
|
||||||
setAccountInfoCache(changeDetail.getAccounts());
|
askForConfirmation(result.error());
|
||||||
display(changeDetail);
|
} else {
|
||||||
}
|
new ErrorDialog(new SafeHtmlBuilder().append(result.error()));
|
||||||
|
|
||||||
if (!result.getErrors().isEmpty()) {
|
|
||||||
final SafeHtmlBuilder r = new SafeHtmlBuilder();
|
|
||||||
for (final ReviewerResult.Error e : result.getErrors()) {
|
|
||||||
switch (e.getType()) {
|
|
||||||
case REVIEWER_NOT_FOUND:
|
|
||||||
r.append(Util.M.reviewerNotFound(e.getName()));
|
|
||||||
break;
|
|
||||||
|
|
||||||
case ACCOUNT_INACTIVE:
|
|
||||||
r.append(Util.M.accountInactive(e.getName()));
|
|
||||||
break;
|
|
||||||
|
|
||||||
case CHANGE_NOT_VISIBLE:
|
|
||||||
r.append(Util.M.changeNotVisibleTo(e.getName()));
|
|
||||||
break;
|
|
||||||
|
|
||||||
case GROUP_EMPTY:
|
|
||||||
r.append(Util.M.groupIsEmpty(e.getName()));
|
|
||||||
break;
|
|
||||||
|
|
||||||
case GROUP_HAS_TOO_MANY_MEMBERS:
|
|
||||||
if (result.askForConfirmation() && !confirmed) {
|
|
||||||
askForConfirmation(e.getName(), result.getMemberCount());
|
|
||||||
return;
|
|
||||||
} else {
|
|
||||||
r.append(Util.M.groupHasTooManyMembers(e.getName()));
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case GROUP_NOT_ALLOWED:
|
|
||||||
r.append(Util.M.groupIsNotAllowed(e.getName()));
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
r.append(e.getName());
|
|
||||||
r.append(" - ");
|
|
||||||
r.append(e.getType());
|
|
||||||
r.br();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
new ErrorDialog(r).center();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void askForConfirmation(final String groupName,
|
private void askForConfirmation(String text) {
|
||||||
final int memberCount) {
|
String title = Util.C
|
||||||
final SafeHtmlBuilder b = new SafeHtmlBuilder();
|
.approvalTableAddManyReviewersConfirmationDialogTitle();
|
||||||
b.openElement("b");
|
ConfirmationDialog confirmationDialog = new ConfirmationDialog(
|
||||||
b.append(Util.M
|
title, new SafeHtmlBuilder().append(text),
|
||||||
.groupManyMembersConfirmation(groupName, memberCount));
|
new ConfirmationCallback() {
|
||||||
b.closeElement("b");
|
@Override
|
||||||
final ConfirmationDialog confirmationDialog =
|
public void onOk() {
|
||||||
new ConfirmationDialog(Util.C
|
addReviewer(reviewer, true);
|
||||||
.approvalTableAddManyReviewersConfirmationDialogTitle(),
|
}
|
||||||
b.toSafeHtml(), new ConfirmationCallback() {
|
});
|
||||||
@Override
|
|
||||||
public void onOk() {
|
|
||||||
addReviewers(reviewers, true);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
confirmationDialog.center();
|
confirmationDialog.center();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onFailure(final Throwable caught) {
|
public void onFailure(final Throwable caught) {
|
||||||
addMemberBox.setEnabled(true);
|
addMemberBox.setEnabled(true);
|
||||||
super.onFailure(caught);
|
if (isNoSuchEntity(caught)) {
|
||||||
|
new ErrorDialog(Util.M.reviewerNotFound(reviewer)).center();
|
||||||
|
} else {
|
||||||
|
super.onFailure(caught);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void displayRow(final int row, final ApprovalDetail ad,
|
private void displayRow(int row, final ApprovalDetail ad, ChangeInfo change,
|
||||||
final Change change, List<String> columns) {
|
AccountInfo account) {
|
||||||
final CellFormatter fmt = table.getCellFormatter();
|
final CellFormatter fmt = table.getCellFormatter();
|
||||||
int col = 0;
|
int col = 0;
|
||||||
|
|
||||||
table.setWidget(row, col++, link(ad.getAccount()));
|
table.setWidget(row, col++, new InlineHyperlink(account.name(),
|
||||||
|
PageLinks.toAccountQuery(account.name())));
|
||||||
|
rows.put(account._account_id(), row);
|
||||||
|
|
||||||
if (ad.canRemove()) {
|
if (ad.canRemove()) {
|
||||||
final PushButton remove = new PushButton( //
|
final PushButton remove = new PushButton( //
|
||||||
new Image(Util.R.removeReviewerNormal()), //
|
new Image(Util.R.removeReviewerNormal()), //
|
||||||
new Image(Util.R.removeReviewerPressed()));
|
new Image(Util.R.removeReviewerPressed()));
|
||||||
remove.setTitle(Util.M.removeReviewer( //
|
remove.setTitle(Util.M.removeReviewer(account.name()));
|
||||||
FormatUtil.name(accountCache.get(ad.getAccount()))));
|
|
||||||
remove.setStyleName(Gerrit.RESOURCES.css().removeReviewer());
|
remove.setStyleName(Gerrit.RESOURCES.css().removeReviewer());
|
||||||
remove.addStyleName(Gerrit.RESOURCES.css().link());
|
remove.addStyleName(Gerrit.RESOURCES.css().link());
|
||||||
remove.addClickHandler(new ClickHandler() {
|
remove.addClickHandler(new ClickHandler() {
|
||||||
@ -397,7 +357,7 @@ public class ApprovalTable extends Composite {
|
|||||||
}
|
}
|
||||||
fmt.setStyleName(row, col++, Gerrit.RESOURCES.css().removeReviewerCell());
|
fmt.setStyleName(row, col++, Gerrit.RESOURCES.css().removeReviewerCell());
|
||||||
|
|
||||||
for (String labelName : columns) {
|
for (String labelName : change.labels()) {
|
||||||
fmt.setStyleName(row, col, Gerrit.RESOURCES.css().approvalscore());
|
fmt.setStyleName(row, col, Gerrit.RESOURCES.css().approvalscore());
|
||||||
if (!ad.canVote(labelName)) {
|
if (!ad.canVote(labelName)) {
|
||||||
fmt.addStyleName(row, col, Gerrit.RESOURCES.css().notVotable());
|
fmt.addStyleName(row, col, Gerrit.RESOURCES.css().notVotable());
|
||||||
@ -411,6 +371,7 @@ public class ApprovalTable extends Composite {
|
|||||||
table.setWidget(row, col, new Image(Gerrit.RESOURCES.greenCheck()));
|
table.setWidget(row, col, new Image(Gerrit.RESOURCES.greenCheck()));
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
|
// TODO: support arbitrary labels.
|
||||||
ApprovalType legacyType = types.byLabel(labelName);
|
ApprovalType legacyType = types.byLabel(labelName);
|
||||||
if (legacyType == null) {
|
if (legacyType == null) {
|
||||||
table.clearCell(row, col);
|
table.clearCell(row, col);
|
||||||
@ -418,15 +379,14 @@ public class ApprovalTable extends Composite {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
PatchSetApproval ca = ad.getPatchSetApproval(legacyType.getCategory().getId());
|
int v = ad.getValue(labelName);
|
||||||
if (ca == null || ca.getValue() == 0) {
|
if (v == 0) {
|
||||||
table.clearCell(row, col);
|
table.clearCell(row, col);
|
||||||
col++;
|
col++;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
String vstr = String.valueOf(ad.getValue(labelName));
|
||||||
String vstr = String.valueOf(ca.getValue());
|
if (v > 0) {
|
||||||
if (ca.getValue() > 0) {
|
|
||||||
vstr = "+" + vstr;
|
vstr = "+" + vstr;
|
||||||
fmt.addStyleName(row, col, Gerrit.RESOURCES.css().posscore());
|
fmt.addStyleName(row, col, Gerrit.RESOURCES.css().posscore());
|
||||||
} else {
|
} else {
|
||||||
@ -442,19 +402,19 @@ public class ApprovalTable extends Composite {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void reload() {
|
private void reload() {
|
||||||
Util.DETAIL_SVC.changeDetail(changeId,
|
ChangeApi.detail(lastChange.legacy_id().get(),
|
||||||
new GerritCallback<ChangeDetail>() {
|
new GerritCallback<ChangeInfo>() {
|
||||||
@Override
|
@Override
|
||||||
public void onSuccess(ChangeDetail result) {
|
public void onSuccess(ChangeInfo result) {
|
||||||
display(result);
|
display(result);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void doRemove(ApprovalDetail ad, final PushButton remove) {
|
private void doRemove(ApprovalDetail ad, final PushButton remove) {
|
||||||
remove.setEnabled(false);
|
remove.setEnabled(false);
|
||||||
ChangeApi.reviewer(changeId.get(), ad.getAccount().get()).delete(
|
ChangeApi.reviewer(lastChange.legacy_id().get(), ad.getAccount().get())
|
||||||
new GerritCallback<JavaScriptObject>() {
|
.delete(new GerritCallback<JavaScriptObject>() {
|
||||||
@Override
|
@Override
|
||||||
public void onSuccess(JavaScriptObject result) {
|
public void onSuccess(JavaScriptObject result) {
|
||||||
reload();
|
reload();
|
||||||
|
@ -69,6 +69,10 @@ public class ChangeApi {
|
|||||||
return change(id.getParentKey().get()).view("revisions").id(id.get());
|
return change(id.getParentKey().get()).view("revisions").id(id.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static RestApi reviewers(int id) {
|
||||||
|
return change(id).view("reviewers");
|
||||||
|
}
|
||||||
|
|
||||||
public static RestApi reviewer(int id, int reviewer) {
|
public static RestApi reviewer(int id, int reviewer) {
|
||||||
return change(id).view("reviewers").id(reviewer);
|
return change(id).view("reviewers").id(reviewer);
|
||||||
}
|
}
|
||||||
|
@ -106,13 +106,6 @@ public class ChangeInfo extends JavaScriptObject {
|
|||||||
protected ChangeInfo() {
|
protected ChangeInfo() {
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class AccountInfo extends JavaScriptObject {
|
|
||||||
public final native String name() /*-{ return this.name; }-*/;
|
|
||||||
|
|
||||||
protected AccountInfo() {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class LabelInfo extends JavaScriptObject {
|
public static class LabelInfo extends JavaScriptObject {
|
||||||
public final SubmitRecord.Label.Status status() {
|
public final SubmitRecord.Label.Status status() {
|
||||||
if (approved() != null) {
|
if (approved() != null) {
|
||||||
@ -136,6 +129,7 @@ public class ChangeInfo extends JavaScriptObject {
|
|||||||
public final native JsArray<ApprovalInfo> all() /*-{ return this.all; }-*/;
|
public final native JsArray<ApprovalInfo> all() /*-{ return this.all; }-*/;
|
||||||
|
|
||||||
private final native NativeMap<NativeString> _values() /*-{ return this.values; }-*/;
|
private final native NativeMap<NativeString> _values() /*-{ return this.values; }-*/;
|
||||||
|
|
||||||
public final Set<String> values() {
|
public final Set<String> values() {
|
||||||
return Natives.keys(_values());
|
return Natives.keys(_values());
|
||||||
}
|
}
|
||||||
@ -154,10 +148,7 @@ public class ChangeInfo extends JavaScriptObject {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class ApprovalInfo extends JavaScriptObject {
|
public static class ApprovalInfo extends AccountInfo {
|
||||||
public final native int _account_id() /*-{ return this._account_id || 0; }-*/;
|
|
||||||
public final native String name() /*-{ return this.name; }-*/;
|
|
||||||
public final native String email() /*-{ return this.email; }-*/;
|
|
||||||
public final native short value() /*-{ return this.value; }-*/;
|
public final native short value() /*-{ return this.value; }-*/;
|
||||||
|
|
||||||
protected ApprovalInfo() {
|
protected ApprovalInfo() {
|
||||||
|
@ -16,6 +16,7 @@ package com.google.gerrit.client.changes;
|
|||||||
|
|
||||||
import com.google.gerrit.client.Dispatcher;
|
import com.google.gerrit.client.Dispatcher;
|
||||||
import com.google.gerrit.client.Gerrit;
|
import com.google.gerrit.client.Gerrit;
|
||||||
|
import com.google.gerrit.client.rpc.GerritCallback;
|
||||||
import com.google.gerrit.client.ui.CommentPanel;
|
import com.google.gerrit.client.ui.CommentPanel;
|
||||||
import com.google.gerrit.client.ui.ComplexDisclosurePanel;
|
import com.google.gerrit.client.ui.ComplexDisclosurePanel;
|
||||||
import com.google.gerrit.client.ui.ExpandAllCommand;
|
import com.google.gerrit.client.ui.ExpandAllCommand;
|
||||||
@ -62,6 +63,7 @@ public class ChangeScreen extends Screen
|
|||||||
private final Change.Id changeId;
|
private final Change.Id changeId;
|
||||||
private final PatchSet.Id openPatchSetId;
|
private final PatchSet.Id openPatchSetId;
|
||||||
private ChangeDetailCache detailCache;
|
private ChangeDetailCache detailCache;
|
||||||
|
private com.google.gerrit.client.changes.ChangeInfo changeInfo;
|
||||||
|
|
||||||
private ChangeDescriptionBlock descriptionBlock;
|
private ChangeDescriptionBlock descriptionBlock;
|
||||||
private ApprovalTable approvals;
|
private ApprovalTable approvals;
|
||||||
@ -255,9 +257,20 @@ public class ChangeScreen extends Screen
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onValueChange(ValueChangeEvent<ChangeDetail> event) {
|
public void onValueChange(final ValueChangeEvent<ChangeDetail> event) {
|
||||||
if (isAttached()) {
|
if (isAttached()) {
|
||||||
display(event.getValue());
|
// Until this screen is fully migrated to the new API, this call must be
|
||||||
|
// sequential, because we can't start an async get at the source of every
|
||||||
|
// call that might trigger a value change.
|
||||||
|
ChangeApi.detail(event.getValue().getChange().getId().get(),
|
||||||
|
new GerritCallback<com.google.gerrit.client.changes.ChangeInfo>() {
|
||||||
|
@Override
|
||||||
|
public void onSuccess(
|
||||||
|
com.google.gerrit.client.changes.ChangeInfo result) {
|
||||||
|
changeInfo = result;
|
||||||
|
display(event.getValue());
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -273,7 +286,6 @@ public class ChangeScreen extends Screen
|
|||||||
}
|
}
|
||||||
|
|
||||||
dependencies.setAccountInfoCache(detail.getAccounts());
|
dependencies.setAccountInfoCache(detail.getAccounts());
|
||||||
approvals.setAccountInfoCache(detail.getAccounts());
|
|
||||||
|
|
||||||
descriptionBlock.display(detail.getChange(),
|
descriptionBlock.display(detail.getChange(),
|
||||||
detail.isStarred(),
|
detail.isStarred(),
|
||||||
@ -282,7 +294,7 @@ public class ChangeScreen extends Screen
|
|||||||
detail.getAccounts(), detail.getSubmitTypeRecord());
|
detail.getAccounts(), detail.getSubmitTypeRecord());
|
||||||
dependsOn.display(detail.getDependsOn());
|
dependsOn.display(detail.getDependsOn());
|
||||||
neededBy.display(detail.getNeededBy());
|
neededBy.display(detail.getNeededBy());
|
||||||
approvals.display(detail);
|
approvals.display(changeInfo);
|
||||||
|
|
||||||
patchesList.clear();
|
patchesList.clear();
|
||||||
if (detail.getCurrentPatchSetDetail().getInfo().getParents().size() > 1) {
|
if (detail.getCurrentPatchSetDetail().getInfo().getParents().size() > 1) {
|
||||||
|
@ -14,6 +14,8 @@
|
|||||||
|
|
||||||
package com.google.gerrit.client.changes;
|
package com.google.gerrit.client.changes;
|
||||||
|
|
||||||
|
import static com.google.gerrit.client.changes.ApprovalTable.parseLabelValue;
|
||||||
|
|
||||||
import com.google.gerrit.client.Gerrit;
|
import com.google.gerrit.client.Gerrit;
|
||||||
import com.google.gerrit.client.changes.ChangeInfo.ApprovalInfo;
|
import com.google.gerrit.client.changes.ChangeInfo.ApprovalInfo;
|
||||||
import com.google.gerrit.client.changes.ChangeInfo.LabelInfo;
|
import com.google.gerrit.client.changes.ChangeInfo.LabelInfo;
|
||||||
@ -242,14 +244,13 @@ public class PublishCommentScreen extends AccountScreen implements
|
|||||||
mwrap.add(message);
|
mwrap.add(message);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void initApprovals(final PatchSetPublishDetail r, final Panel body) {
|
private void initApprovals(Panel body) {
|
||||||
for (String labelName : change.labels()) {
|
for (String labelName : change.labels()) {
|
||||||
initLabel(r, labelName, body);
|
initLabel(labelName, body);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void initLabel(PatchSetPublishDetail r, String labelName,
|
private void initLabel(String labelName, Panel body) {
|
||||||
Panel body) {
|
|
||||||
JsArrayString nativeValues = change.permitted_values(labelName);
|
JsArrayString nativeValues = change.permitted_values(labelName);
|
||||||
if (nativeValues == null || nativeValues.length() == 0) {
|
if (nativeValues == null || nativeValues.length() == 0) {
|
||||||
return;
|
return;
|
||||||
@ -302,10 +303,8 @@ public class PublishCommentScreen extends AccountScreen implements
|
|||||||
r.getSubmitTypeRecord());
|
r.getSubmitTypeRecord());
|
||||||
|
|
||||||
if (r.getChange().getStatus().isOpen()) {
|
if (r.getChange().getStatus().isOpen()) {
|
||||||
initApprovals(r, approvalPanel);
|
initApprovals(approvalPanel);
|
||||||
|
approvals.display(change);
|
||||||
approvals.setAccountInfoCache(r.getAccounts());
|
|
||||||
approvals.display(r);
|
|
||||||
} else {
|
} else {
|
||||||
approvals.setVisible(false);
|
approvals.setVisible(false);
|
||||||
}
|
}
|
||||||
@ -451,13 +450,6 @@ public class PublishCommentScreen extends AccountScreen implements
|
|||||||
Gerrit.display(PageLinks.toChange(ck), new ChangeScreen(ck));
|
Gerrit.display(PageLinks.toChange(ck), new ChangeScreen(ck));
|
||||||
}
|
}
|
||||||
|
|
||||||
private static short parseLabelValue(String value) {
|
|
||||||
if (value.charAt(0) == ' ' || value.charAt(0) == '+') {
|
|
||||||
value = value.substring(1);
|
|
||||||
}
|
|
||||||
return Short.parseShort(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static class ValueRadioButton extends RadioButton {
|
private static class ValueRadioButton extends RadioButton {
|
||||||
final LabelInfo label;
|
final LabelInfo label;
|
||||||
final String value;
|
final String value;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user