mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-21 20:22:12 +01:00
8858b6cf8d
Summary: Fixes T10562. I left this behavior sort of ambiguous in the original implementation because I didn't anticipate or stumble across this situation. It's easy to fix: when you reply to a ghost, just put the reply in the exact same place as the ghost (even if it's a different diff), so they always move/ghost/port/thread together. Test Plan: See T10562 for reproduction steps and a "before" picture. Here's the after picture: {F1168983} The two comments at the bottom are pre-fix, and exhibit the bug. The comment at the top is post-fix, and appears adjacent to the original correctly. Reviewers: chad Reviewed By: chad Subscribers: eadler Maniphest Tasks: T10562 Differential Revision: https://secure.phabricator.com/D15458
407 lines
13 KiB
PHP
407 lines
13 KiB
PHP
<?php
|
|
|
|
abstract class PhabricatorInlineCommentController
|
|
extends PhabricatorController {
|
|
|
|
abstract protected function createComment();
|
|
abstract protected function loadComment($id);
|
|
abstract protected function loadCommentForEdit($id);
|
|
abstract protected function loadCommentForDone($id);
|
|
abstract protected function loadCommentByPHID($phid);
|
|
abstract protected function loadObjectOwnerPHID(
|
|
PhabricatorInlineCommentInterface $inline);
|
|
abstract protected function deleteComment(
|
|
PhabricatorInlineCommentInterface $inline);
|
|
abstract protected function undeleteComment(
|
|
PhabricatorInlineCommentInterface $inline);
|
|
abstract protected function saveComment(
|
|
PhabricatorInlineCommentInterface $inline);
|
|
|
|
protected function hideComments(array $ids) {
|
|
throw new PhutilMethodNotImplementedException();
|
|
}
|
|
|
|
protected function showComments(array $ids) {
|
|
throw new PhutilMethodNotImplementedException();
|
|
}
|
|
|
|
private $changesetID;
|
|
private $isNewFile;
|
|
private $isOnRight;
|
|
private $lineNumber;
|
|
private $lineLength;
|
|
private $commentText;
|
|
private $operation;
|
|
private $commentID;
|
|
private $renderer;
|
|
private $replyToCommentPHID;
|
|
|
|
public function getCommentID() {
|
|
return $this->commentID;
|
|
}
|
|
|
|
public function getOperation() {
|
|
return $this->operation;
|
|
}
|
|
|
|
public function getCommentText() {
|
|
return $this->commentText;
|
|
}
|
|
|
|
public function getLineLength() {
|
|
return $this->lineLength;
|
|
}
|
|
|
|
public function getLineNumber() {
|
|
return $this->lineNumber;
|
|
}
|
|
|
|
public function getIsOnRight() {
|
|
return $this->isOnRight;
|
|
}
|
|
|
|
public function getChangesetID() {
|
|
return $this->changesetID;
|
|
}
|
|
|
|
public function getIsNewFile() {
|
|
return $this->isNewFile;
|
|
}
|
|
|
|
public function setRenderer($renderer) {
|
|
$this->renderer = $renderer;
|
|
return $this;
|
|
}
|
|
|
|
public function getRenderer() {
|
|
return $this->renderer;
|
|
}
|
|
|
|
public function setReplyToCommentPHID($phid) {
|
|
$this->replyToCommentPHID = $phid;
|
|
return $this;
|
|
}
|
|
|
|
public function getReplyToCommentPHID() {
|
|
return $this->replyToCommentPHID;
|
|
}
|
|
|
|
public function processRequest() {
|
|
$request = $this->getRequest();
|
|
$user = $request->getUser();
|
|
|
|
$this->readRequestParameters();
|
|
|
|
$op = $this->getOperation();
|
|
switch ($op) {
|
|
case 'busy':
|
|
if ($request->isFormPost()) {
|
|
return new AphrontAjaxResponse();
|
|
}
|
|
|
|
return $this->newDialog()
|
|
->setTitle(pht('Already Editing'))
|
|
->appendParagraph(
|
|
pht(
|
|
'You are already editing an inline comment. Finish editing '.
|
|
'your current comment before adding new comments.'))
|
|
->addCancelButton('/')
|
|
->addSubmitButton(pht('Jump to Inline'));
|
|
case 'hide':
|
|
case 'show':
|
|
if (!$request->validateCSRF()) {
|
|
return new Aphront404Response();
|
|
}
|
|
|
|
$ids = $request->getStrList('ids');
|
|
if ($ids) {
|
|
if ($op == 'hide') {
|
|
$this->hideComments($ids);
|
|
} else {
|
|
$this->showComments($ids);
|
|
}
|
|
}
|
|
|
|
return id(new AphrontAjaxResponse())->setContent(array());
|
|
case 'done':
|
|
if (!$request->validateCSRF()) {
|
|
return new Aphront404Response();
|
|
}
|
|
$inline = $this->loadCommentForDone($this->getCommentID());
|
|
|
|
$is_draft_state = false;
|
|
switch ($inline->getFixedState()) {
|
|
case PhabricatorInlineCommentInterface::STATE_DRAFT:
|
|
$next_state = PhabricatorInlineCommentInterface::STATE_UNDONE;
|
|
break;
|
|
case PhabricatorInlineCommentInterface::STATE_UNDRAFT:
|
|
$next_state = PhabricatorInlineCommentInterface::STATE_DONE;
|
|
break;
|
|
case PhabricatorInlineCommentInterface::STATE_DONE:
|
|
$next_state = PhabricatorInlineCommentInterface::STATE_UNDRAFT;
|
|
$is_draft_state = true;
|
|
break;
|
|
default:
|
|
case PhabricatorInlineCommentInterface::STATE_UNDONE:
|
|
$next_state = PhabricatorInlineCommentInterface::STATE_DRAFT;
|
|
$is_draft_state = true;
|
|
break;
|
|
}
|
|
|
|
$inline->setFixedState($next_state)->save();
|
|
|
|
return id(new AphrontAjaxResponse())
|
|
->setContent(
|
|
array(
|
|
'draftState' => $is_draft_state,
|
|
));
|
|
case 'delete':
|
|
case 'undelete':
|
|
case 'refdelete':
|
|
if (!$request->validateCSRF()) {
|
|
return new Aphront404Response();
|
|
}
|
|
|
|
// NOTE: For normal deletes, we just process the delete immediately
|
|
// and show an "Undo" action. For deletes by reference from the
|
|
// preview ("refdelete"), we prompt first (because the "Undo" may
|
|
// not draw, or may not be easy to locate).
|
|
|
|
if ($op == 'refdelete') {
|
|
if (!$request->isFormPost()) {
|
|
return $this->newDialog()
|
|
->setTitle(pht('Really delete comment?'))
|
|
->addHiddenInput('id', $this->getCommentID())
|
|
->addHiddenInput('op', $op)
|
|
->appendParagraph(pht('Delete this inline comment?'))
|
|
->addCancelButton('#')
|
|
->addSubmitButton(pht('Delete'));
|
|
}
|
|
}
|
|
|
|
$is_delete = ($op == 'delete' || $op == 'refdelete');
|
|
|
|
$inline = $this->loadCommentForEdit($this->getCommentID());
|
|
|
|
if ($is_delete) {
|
|
$this->deleteComment($inline);
|
|
} else {
|
|
$this->undeleteComment($inline);
|
|
}
|
|
|
|
return $this->buildEmptyResponse();
|
|
case 'edit':
|
|
$inline = $this->loadCommentForEdit($this->getCommentID());
|
|
$text = $this->getCommentText();
|
|
|
|
if ($request->isFormPost()) {
|
|
if (strlen($text)) {
|
|
$inline->setContent($text);
|
|
$this->saveComment($inline);
|
|
return $this->buildRenderedCommentResponse(
|
|
$inline,
|
|
$this->getIsOnRight());
|
|
} else {
|
|
$this->deleteComment($inline);
|
|
return $this->buildEmptyResponse();
|
|
}
|
|
}
|
|
|
|
$edit_dialog = $this->buildEditDialog();
|
|
$edit_dialog->setTitle(pht('Edit Inline Comment'));
|
|
|
|
$edit_dialog->addHiddenInput('id', $this->getCommentID());
|
|
$edit_dialog->addHiddenInput('op', 'edit');
|
|
|
|
$edit_dialog->appendChild(
|
|
$this->renderTextArea(
|
|
nonempty($text, $inline->getContent())));
|
|
|
|
$view = $this->buildScaffoldForView($edit_dialog);
|
|
|
|
return id(new AphrontAjaxResponse())
|
|
->setContent($view->render());
|
|
case 'create':
|
|
$text = $this->getCommentText();
|
|
|
|
if (!$request->isFormPost() || !strlen($text)) {
|
|
return $this->buildEmptyResponse();
|
|
}
|
|
|
|
$inline = $this->createComment()
|
|
->setChangesetID($this->getChangesetID())
|
|
->setAuthorPHID($user->getPHID())
|
|
->setLineNumber($this->getLineNumber())
|
|
->setLineLength($this->getLineLength())
|
|
->setIsNewFile($this->getIsNewFile())
|
|
->setContent($text);
|
|
|
|
if ($this->getReplyToCommentPHID()) {
|
|
$inline->setReplyToCommentPHID($this->getReplyToCommentPHID());
|
|
}
|
|
|
|
$this->saveComment($inline);
|
|
|
|
return $this->buildRenderedCommentResponse(
|
|
$inline,
|
|
$this->getIsOnRight());
|
|
case 'reply':
|
|
default:
|
|
$edit_dialog = $this->buildEditDialog();
|
|
|
|
if ($this->getOperation() == 'reply') {
|
|
$edit_dialog->setTitle(pht('Reply to Inline Comment'));
|
|
} else {
|
|
$edit_dialog->setTitle(pht('New Inline Comment'));
|
|
}
|
|
|
|
// NOTE: We read the values from the client (the display values), not
|
|
// the values from the database (the original values) when replying.
|
|
// In particular, when replying to a ghost comment which was moved
|
|
// across diffs and then moved backward to the most recent visible
|
|
// line, we want to reply on the display line (which exists), not on
|
|
// the comment's original line (which may not exist in this changeset).
|
|
$is_new = $this->getIsNewFile();
|
|
$number = $this->getLineNumber();
|
|
$length = $this->getLineLength();
|
|
|
|
$edit_dialog->addHiddenInput('op', 'create');
|
|
$edit_dialog->addHiddenInput('is_new', $is_new);
|
|
$edit_dialog->addHiddenInput('number', $number);
|
|
$edit_dialog->addHiddenInput('length', $length);
|
|
|
|
$text_area = $this->renderTextArea($this->getCommentText());
|
|
$edit_dialog->appendChild($text_area);
|
|
|
|
$view = $this->buildScaffoldForView($edit_dialog);
|
|
|
|
return id(new AphrontAjaxResponse())
|
|
->setContent($view->render());
|
|
}
|
|
}
|
|
|
|
private function readRequestParameters() {
|
|
$request = $this->getRequest();
|
|
|
|
// NOTE: This isn't necessarily a DifferentialChangeset ID, just an
|
|
// application identifier for the changeset. In Diffusion, it's a Path ID.
|
|
$this->changesetID = $request->getInt('changesetID');
|
|
|
|
$this->isNewFile = (int)$request->getBool('is_new');
|
|
$this->isOnRight = $request->getBool('on_right');
|
|
$this->lineNumber = $request->getInt('number');
|
|
$this->lineLength = $request->getInt('length');
|
|
$this->commentText = $request->getStr('text');
|
|
$this->commentID = $request->getInt('id');
|
|
$this->operation = $request->getStr('op');
|
|
$this->renderer = $request->getStr('renderer');
|
|
$this->replyToCommentPHID = $request->getStr('replyToCommentPHID');
|
|
|
|
if ($this->getReplyToCommentPHID()) {
|
|
$reply_phid = $this->getReplyToCommentPHID();
|
|
$reply_comment = $this->loadCommentByPHID($reply_phid);
|
|
if (!$reply_comment) {
|
|
throw new Exception(
|
|
pht('Failed to load comment "%s".', $reply_phid));
|
|
}
|
|
|
|
// When replying, force the new comment into the same location as the
|
|
// old comment. If we don't do this, replying to a ghost comment from
|
|
// diff A while viewing diff B can end up placing the two comments in
|
|
// different places while viewing diff C, because the porting algorithm
|
|
// makes a different decision. Forcing the comments to bind to the same
|
|
// place makes sure they stick together no matter which diff is being
|
|
// viewed. See T10562 for discussion.
|
|
|
|
$this->changesetID = $reply_comment->getChangesetID();
|
|
$this->isNewFile = $reply_comment->getIsNewFile();
|
|
$this->lineNumber = $reply_comment->getLineNumber();
|
|
$this->lineLength = $reply_comment->getLineLength();
|
|
}
|
|
}
|
|
|
|
private function buildEditDialog() {
|
|
$request = $this->getRequest();
|
|
$user = $request->getUser();
|
|
|
|
$edit_dialog = id(new PHUIDiffInlineCommentEditView())
|
|
->setUser($user)
|
|
->setSubmitURI($request->getRequestURI())
|
|
->setIsOnRight($this->getIsOnRight())
|
|
->setIsNewFile($this->getIsNewFile())
|
|
->setNumber($this->getLineNumber())
|
|
->setLength($this->getLineLength())
|
|
->setRenderer($this->getRenderer())
|
|
->setReplyToCommentPHID($this->getReplyToCommentPHID())
|
|
->setChangesetID($this->getChangesetID());
|
|
|
|
return $edit_dialog;
|
|
}
|
|
|
|
private function buildEmptyResponse() {
|
|
return id(new AphrontAjaxResponse())
|
|
->setContent(
|
|
array(
|
|
'markup' => '',
|
|
));
|
|
}
|
|
|
|
private function buildRenderedCommentResponse(
|
|
PhabricatorInlineCommentInterface $inline,
|
|
$on_right) {
|
|
|
|
$request = $this->getRequest();
|
|
$user = $request->getUser();
|
|
|
|
$engine = new PhabricatorMarkupEngine();
|
|
$engine->setViewer($user);
|
|
$engine->addObject(
|
|
$inline,
|
|
PhabricatorInlineCommentInterface::MARKUP_FIELD_BODY);
|
|
$engine->process();
|
|
|
|
$phids = array($user->getPHID());
|
|
|
|
$handles = $this->loadViewerHandles($phids);
|
|
$object_owner_phid = $this->loadObjectOwnerPHID($inline);
|
|
|
|
$view = id(new PHUIDiffInlineCommentDetailView())
|
|
->setUser($user)
|
|
->setInlineComment($inline)
|
|
->setIsOnRight($on_right)
|
|
->setMarkupEngine($engine)
|
|
->setHandles($handles)
|
|
->setEditable(true)
|
|
->setCanMarkDone(false)
|
|
->setObjectOwnerPHID($object_owner_phid);
|
|
|
|
$view = $this->buildScaffoldForView($view);
|
|
|
|
return id(new AphrontAjaxResponse())
|
|
->setContent(
|
|
array(
|
|
'inlineCommentID' => $inline->getID(),
|
|
'markup' => $view->render(),
|
|
));
|
|
}
|
|
|
|
private function renderTextArea($text) {
|
|
return id(new PhabricatorRemarkupControl())
|
|
->setUser($this->getRequest()->getUser())
|
|
->setSigil('differential-inline-comment-edit-textarea')
|
|
->setName('text')
|
|
->setValue($text)
|
|
->setDisableFullScreen(true);
|
|
}
|
|
|
|
private function buildScaffoldForView(PHUIDiffInlineCommentView $view) {
|
|
$renderer = DifferentialChangesetHTMLRenderer::getHTMLRendererByKey(
|
|
$this->getRenderer());
|
|
|
|
$view = $renderer->getRowScaffoldForInline($view);
|
|
|
|
return id(new PHUIDiffInlineCommentTableScaffold())
|
|
->addRowScaffold($view);
|
|
}
|
|
|
|
}
|