2017-06-12 11:24:17 -07:00
|
|
|
<?php
|
|
|
|
|
|
|
|
final class DifferentialRevisionInlinesController
|
|
|
|
extends DifferentialController {
|
|
|
|
|
|
|
|
public function shouldAllowPublic() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function handleRequest(AphrontRequest $request) {
|
|
|
|
$viewer = $this->getViewer();
|
|
|
|
$id = $request->getURIData('id');
|
|
|
|
|
|
|
|
$revision = id(new DifferentialRevisionQuery())
|
|
|
|
->withIDs(array($id))
|
|
|
|
->setViewer($viewer)
|
|
|
|
->needDiffIDs(true)
|
|
|
|
->executeOne();
|
|
|
|
if (!$revision) {
|
|
|
|
return new Aphront404Response();
|
|
|
|
}
|
|
|
|
|
|
|
|
$revision_monogram = $revision->getMonogram();
|
|
|
|
$revision_uri = $revision->getURI();
|
|
|
|
$revision_title = $revision->getTitle();
|
|
|
|
|
2020-05-07 13:02:06 -07:00
|
|
|
$inlines = id(new DifferentialDiffInlineCommentQuery())
|
2017-06-12 11:24:17 -07:00
|
|
|
->setViewer($viewer)
|
2020-05-07 13:02:06 -07:00
|
|
|
->withRevisionPHIDs(array($revision->getPHID()))
|
|
|
|
->withPublishedComments(true)
|
|
|
|
->execute();
|
|
|
|
$inlines = mpull($inlines, 'newInlineCommentObject');
|
2017-06-12 11:24:17 -07:00
|
|
|
|
|
|
|
$crumbs = $this->buildApplicationCrumbs();
|
|
|
|
$crumbs->addTextCrumb($revision_monogram, $revision_uri);
|
|
|
|
$crumbs->addTextCrumb(pht('Inline Comments'));
|
|
|
|
$crumbs->setBorder(true);
|
|
|
|
|
|
|
|
$content = $this->renderInlineTable($revision, $inlines);
|
|
|
|
$header = $this->buildHeader($revision);
|
|
|
|
|
|
|
|
$view = id(new PHUITwoColumnView())
|
|
|
|
->setHeader($header)
|
|
|
|
->setFooter($content);
|
|
|
|
|
|
|
|
return $this->newPage()
|
|
|
|
->setTitle(
|
|
|
|
array(
|
|
|
|
"{$revision_monogram} {$revision_title}",
|
|
|
|
pht('Inlines'),
|
|
|
|
))
|
|
|
|
->setCrumbs($crumbs)
|
|
|
|
->appendChild($view);
|
|
|
|
}
|
|
|
|
|
|
|
|
private function buildHeader(DifferentialRevision $revision) {
|
|
|
|
$viewer = $this->getViewer();
|
|
|
|
|
|
|
|
$button = id(new PHUIButtonView())
|
|
|
|
->setTag('a')
|
|
|
|
->setIcon('fa-chevron-left')
|
|
|
|
->setHref($revision->getURI())
|
|
|
|
->setText(pht('Back to Revision'));
|
|
|
|
|
|
|
|
return id(new PHUIHeaderView())
|
|
|
|
->setHeader($revision->getTitle())
|
|
|
|
->setUser($viewer)
|
|
|
|
->setHeaderIcon('fa-cog')
|
|
|
|
->addActionLink($button);
|
|
|
|
}
|
|
|
|
|
|
|
|
private function renderInlineTable(
|
|
|
|
DifferentialRevision $revision,
|
|
|
|
array $inlines) {
|
|
|
|
|
|
|
|
$viewer = $this->getViewer();
|
|
|
|
$inlines = id(new PHUIDiffInlineThreader())
|
|
|
|
->reorderAndThreadCommments($inlines);
|
|
|
|
|
|
|
|
$handle_phids = array();
|
|
|
|
$changeset_ids = array();
|
|
|
|
foreach ($inlines as $inline) {
|
|
|
|
$handle_phids[] = $inline->getAuthorPHID();
|
|
|
|
$changeset_ids[] = $inline->getChangesetID();
|
|
|
|
}
|
|
|
|
$handles = $viewer->loadHandles($handle_phids);
|
|
|
|
$handles = iterator_to_array($handles);
|
|
|
|
|
|
|
|
if ($changeset_ids) {
|
|
|
|
$changesets = id(new DifferentialChangesetQuery())
|
|
|
|
->setViewer($viewer)
|
|
|
|
->withIDs($changeset_ids)
|
|
|
|
->execute();
|
|
|
|
$changesets = mpull($changesets, null, 'getID');
|
|
|
|
} else {
|
|
|
|
$changesets = array();
|
|
|
|
}
|
|
|
|
|
|
|
|
$current_changeset = head($revision->getDiffIDs());
|
|
|
|
|
|
|
|
$rows = array();
|
|
|
|
foreach ($inlines as $inline) {
|
|
|
|
$status_icons = array();
|
|
|
|
|
|
|
|
$c_id = $inline->getChangesetID();
|
|
|
|
$d_id = $changesets[$c_id]->getDiffID();
|
|
|
|
|
|
|
|
if ($d_id == $current_changeset) {
|
|
|
|
$diff_id = phutil_tag('strong', array(), pht('Current'));
|
|
|
|
} else {
|
|
|
|
$diff_id = pht('Diff %d', $d_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
$reviewer = $handles[$inline->getAuthorPHID()]->renderLink();
|
|
|
|
$now = PhabricatorTime::getNow();
|
|
|
|
$then = $inline->getDateModified();
|
|
|
|
$datetime = phutil_format_relative_time($now - $then);
|
|
|
|
|
|
|
|
$comment_href = $revision->getURI().'#inline-'.$inline->getID();
|
|
|
|
$comment = phutil_tag(
|
|
|
|
'a',
|
|
|
|
array(
|
|
|
|
'href' => $comment_href,
|
|
|
|
),
|
|
|
|
$inline->getContent());
|
|
|
|
|
|
|
|
$state = $inline->getFixedState();
|
Make "editing" state persistent for inline comments
Summary:
Ref T13513. This is mostly an infrastructure cleanup change.
In a perfect world, this would be a series of several changes, but they're tightly interconnected and don't have an obvious clean, nontrivial partition (or, at least, I don't see one). Followup changes will exercise this code repeatedly and all of these individual mutations are "obviously good", so I'm not too worried about the breadth of this change.
---
Inline comments are stored as transaction comments in the `PhabricatorAuditTransactionComment` and `DifferentialTransactionComment` classes.
On top of these two storage classes sit `PhabricatorAuditInlineComment` and `DifferentialInlineComment`. Historically, these were an indirection layer over significantly different storage classes, but nowadays both storage classes look pretty similar and most of the logic is actually the same. Prior to this change, these two classes were about 80% copy/pastes of one another.
Part of the reason they're so copy/pastey is that they implement a parent `Interface`. They are the only classes which implement this interface, and the interface does not provide any correctness guarantees (the storage objects are not actually constrained by it).
To simplify this:
- Make `PhabricatorInlineCommentInterface` an abstract base class instead.
- Lift as much code out of the `Audit` and `Differential` subclasses as possible.
- Delete methods which no longer have callers, or have only trivial callers.
---
Inline comments have two `View` rendering classes, `DetailView` and `EditView`. They share very little code.
Partly, this is because `EditView` does not take an `$inline` object. Historically, it needed to be able to operate on inlines that did not have an ID yet, and even further back in history this was probably just an outgrowth of a simple `<form />`.
These classes can be significantly simplified by passing an `$inline` to the `EditView`, instead of individually setting all the properties on the `View` itself. This allows the `DetailView` and `EditView` classes to share a lot of code.
The `EditView` can not fully render its content. Move the content rendering code into the view.
---
Prior to this change, some operations need to work on inlines that don't have an inline ID yet (we assign an ID the first time you "Save" a comment). Since "editing" comments will now be saved, we can instead create a row immediately.
This means that all the inline code can always rely on having a valid ID to work with, even if that ID corresponds to an empty, draft, "isEditing" comment. This simplifies more code in `EditView` and allows the "create" and "reply" code to be merged in `PhabricatorInlineCommentController`.
---
Client-side inline events are currently handled through a mixture of `ChangesetList` listeners (good) and ad-hoc row-level listeners (less good). In particular, the "save", "cancel", and "undo" events are row-level. All other events are list-level.
Move all events to list-level. This is supported by all inlines now having an ID at all stages of their lifecycle.
This allows some of the client behavior to be simplified. It currently depends on binding complex ad-hoc dictionaries into event handlers in `_drawRows()`, but it seems like almost all of this code can be removed. In fact, no more than one row ever seems to be drawn, so this code can probably be simplified further.
---
Finally, save an "isEditing" state. When we rebuild a revision on the client, click the "edit" button if it's in this state. This is a little hacky, but simpler to get into a stable state, since the row layout of an inline depends on a "view row" followed by an "edit row".
Test Plan:
- Created comments on either side of a diff.
- Edited a comment, reloaded, saw edit stick.
- Saved comments, reloaded, saw save stick.
- Edited a comment, typed text, cancelled, "unedited" to get state back.
- Created a comment, typed text, cancelled, "unedited" to get state back.
- Deleted a comment, "undeleted" to get state back.
Weirdness / known issues:
- Drafts don't autosave yet.
- Fixed in D21187:
- When you create an empty comment then reload, you get an empty editor. This is a bit silly.
- "Cancel" does not save state, but should, once drafts autosave.
- Mostly fixed in D21188:
- "Editing" comments aren't handled specially by the overall submission flow.
- "Editing" comments submitted in that state try to edit themselves again on load, which doesn't work.
Subscribers: jmeador
Maniphest Tasks: T13513
Differential Revision: https://secure.phabricator.com/D21186
2020-04-28 14:45:54 -07:00
|
|
|
if ($state == PhabricatorInlineComment::STATE_DONE) {
|
2017-06-12 11:24:17 -07:00
|
|
|
$status_icons[] = id(new PHUIIconView())
|
|
|
|
->setIcon('fa-check green')
|
|
|
|
->addClass('mmr');
|
|
|
|
} else if ($inline->getReplyToCommentPHID() &&
|
|
|
|
$inline->getAuthorPHID() == $revision->getAuthorPHID()) {
|
|
|
|
$status_icons[] = id(new PHUIIconView())
|
|
|
|
->setIcon('fa-commenting-o blue')
|
|
|
|
->addClass('mmr');
|
|
|
|
} else {
|
|
|
|
$status_icons[] = id(new PHUIIconView())
|
|
|
|
->setIcon('fa-circle-o grey')
|
|
|
|
->addClass('mmr');
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if ($inline->getReplyToCommentPHID()) {
|
|
|
|
$reply_icon = id(new PHUIIconView())
|
|
|
|
->setIcon('fa-reply mmr darkgrey');
|
|
|
|
$comment = array($reply_icon, $comment);
|
|
|
|
}
|
|
|
|
|
|
|
|
$rows[] = array(
|
|
|
|
$diff_id,
|
|
|
|
$status_icons,
|
|
|
|
$reviewer,
|
|
|
|
AphrontTableView::renderSingleDisplayLine($comment),
|
|
|
|
$datetime,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$table = new AphrontTableView($rows);
|
|
|
|
$table->setHeaders(
|
|
|
|
array(
|
|
|
|
pht('Diff'),
|
|
|
|
pht('Status'),
|
|
|
|
pht('Reviewer'),
|
|
|
|
pht('Comment'),
|
|
|
|
pht('Created'),
|
|
|
|
));
|
|
|
|
$table->setColumnClasses(
|
|
|
|
array(
|
|
|
|
'',
|
|
|
|
'',
|
|
|
|
'',
|
|
|
|
'wide',
|
|
|
|
'right',
|
|
|
|
));
|
|
|
|
$table->setColumnVisibility(
|
|
|
|
array(
|
|
|
|
true,
|
|
|
|
true,
|
|
|
|
true,
|
|
|
|
true,
|
|
|
|
true,
|
|
|
|
));
|
|
|
|
|
|
|
|
return id(new PHUIObjectBoxView())
|
|
|
|
->setHeaderText(pht('Inline Comments'))
|
|
|
|
->setBackground(PHUIObjectBoxView::BLUE_PROPERTY)
|
|
|
|
->setTable($table);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|