2012-08-10 19:44:04 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
final class PonderAnswer extends PonderDAO
|
2013-07-26 22:19:49 +02:00
|
|
|
implements
|
|
|
|
PhabricatorMarkupInterface,
|
|
|
|
PonderVotableInterface,
|
2013-07-29 02:23:04 +02:00
|
|
|
PhabricatorPolicyInterface,
|
2013-10-25 21:52:00 +02:00
|
|
|
PhabricatorFlaggableInterface,
|
2013-07-29 02:23:04 +02:00
|
|
|
PhabricatorSubscribableInterface,
|
|
|
|
PhabricatorTokenReceiverInterface {
|
2012-08-10 19:44:04 +02:00
|
|
|
|
|
|
|
const MARKUP_FIELD_CONTENT = 'markup:content';
|
|
|
|
|
|
|
|
protected $authorPHID;
|
|
|
|
protected $questionID;
|
|
|
|
|
|
|
|
protected $content;
|
|
|
|
protected $contentSource;
|
|
|
|
|
|
|
|
protected $voteCount;
|
|
|
|
private $vote;
|
2013-07-26 22:13:38 +02:00
|
|
|
private $question = self::ATTACHABLE;
|
adding comments to ponder
Summary: This is pretty spartan, but it does the job.
Test Plan:
Patch, update storage, add some comment
to your favorite question or answer.
Reviewers: nh, vrana, epriestley
Reviewed By: epriestley
CC: aran, Korvin, starruler, syrneus, me.here, victorzarate7
Maniphest Tasks: T1645
Differential Revision: https://secure.phabricator.com/D3471
2012-09-11 21:13:20 +02:00
|
|
|
private $comments;
|
2012-08-10 19:44:04 +02:00
|
|
|
|
2013-07-29 04:28:00 +02:00
|
|
|
private $userVotes = array();
|
|
|
|
|
2013-07-26 22:13:38 +02:00
|
|
|
public function attachQuestion(PonderQuestion $question = null) {
|
2012-08-10 19:44:04 +02:00
|
|
|
$this->question = $question;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getQuestion() {
|
2013-07-26 22:13:38 +02:00
|
|
|
return $this->assertAttached($this->question);
|
2012-08-10 19:44:04 +02:00
|
|
|
}
|
|
|
|
|
2013-09-19 00:15:25 +02:00
|
|
|
public function getURI() {
|
|
|
|
return '/Q'.$this->getQuestionID().'#A'.$this->getID();
|
|
|
|
}
|
|
|
|
|
2012-08-10 19:44:04 +02:00
|
|
|
public function setUserVote($vote) {
|
|
|
|
$this->vote = $vote['data'];
|
|
|
|
if (!$this->vote) {
|
2013-07-28 03:37:17 +02:00
|
|
|
$this->vote = PonderVote::VOTE_NONE;
|
2012-08-10 19:44:04 +02:00
|
|
|
}
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2013-07-29 04:28:00 +02:00
|
|
|
public function attachUserVote($user_phid, $vote) {
|
|
|
|
$this->vote = $vote;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2012-08-10 19:44:04 +02:00
|
|
|
public function getUserVote() {
|
|
|
|
return $this->vote;
|
|
|
|
}
|
|
|
|
|
adding comments to ponder
Summary: This is pretty spartan, but it does the job.
Test Plan:
Patch, update storage, add some comment
to your favorite question or answer.
Reviewers: nh, vrana, epriestley
Reviewed By: epriestley
CC: aran, Korvin, starruler, syrneus, me.here, victorzarate7
Maniphest Tasks: T1645
Differential Revision: https://secure.phabricator.com/D3471
2012-09-11 21:13:20 +02:00
|
|
|
public function setComments($comments) {
|
|
|
|
$this->comments = $comments;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getComments() {
|
|
|
|
return $this->comments;
|
|
|
|
}
|
|
|
|
|
2012-08-10 19:44:04 +02:00
|
|
|
public function getConfiguration() {
|
|
|
|
return array(
|
|
|
|
self::CONFIG_AUX_PHID => true,
|
|
|
|
) + parent::getConfiguration();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function generatePHID() {
|
|
|
|
return PhabricatorPHID::generateNewPHID(
|
2013-07-26 22:19:49 +02:00
|
|
|
PonderPHIDTypeAnswer::TYPECONST);
|
2012-08-10 19:44:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public function setContentSource(PhabricatorContentSource $content_source) {
|
|
|
|
$this->contentSource = $content_source->serialize();
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getContentSource() {
|
|
|
|
return PhabricatorContentSource::newFromSerialized($this->contentSource);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getMarkupField() {
|
|
|
|
return self::MARKUP_FIELD_CONTENT;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Markup interface
|
|
|
|
|
|
|
|
public function getMarkupFieldKey($field) {
|
|
|
|
$hash = PhabricatorHash::digest($this->getMarkupText($field));
|
|
|
|
$id = $this->getID();
|
|
|
|
return "ponder:A{$id}:{$field}:{$hash}";
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getMarkupText($field) {
|
|
|
|
return $this->getContent();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function newMarkupEngine($field) {
|
2013-08-05 19:47:26 +02:00
|
|
|
return PhabricatorMarkupEngine::getEngine();
|
2012-08-10 19:44:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public function didMarkupText(
|
|
|
|
$field,
|
|
|
|
$output,
|
|
|
|
PhutilMarkupEngine $engine) {
|
|
|
|
return $output;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function shouldUseMarkupCache($field) {
|
|
|
|
return (bool)$this->getID();
|
|
|
|
}
|
|
|
|
|
|
|
|
// votable interface
|
|
|
|
public function getUserVoteEdgeType() {
|
|
|
|
return PhabricatorEdgeConfig::TYPE_VOTING_USER_HAS_ANSWER;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getVotablePHID() {
|
|
|
|
return $this->getPHID();
|
|
|
|
}
|
2013-07-26 22:19:49 +02:00
|
|
|
|
|
|
|
|
|
|
|
/* -( PhabricatorPolicyInterface )----------------------------------------- */
|
|
|
|
|
|
|
|
|
|
|
|
public function getCapabilities() {
|
|
|
|
return array(
|
|
|
|
PhabricatorPolicyCapability::CAN_VIEW,
|
|
|
|
PhabricatorPolicyCapability::CAN_EDIT,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getPolicy($capability) {
|
|
|
|
switch ($capability) {
|
|
|
|
case PhabricatorPolicyCapability::CAN_VIEW:
|
|
|
|
return $this->getQuestion()->getPolicy($capability);
|
|
|
|
case PhabricatorPolicyCapability::CAN_EDIT:
|
|
|
|
return PhabricatorPolicies::POLICY_NOONE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function hasAutomaticCapability($capability, PhabricatorUser $viewer) {
|
|
|
|
switch ($capability) {
|
|
|
|
case PhabricatorPolicyCapability::CAN_VIEW:
|
2013-09-27 17:43:41 +02:00
|
|
|
if ($this->getAuthorPHID() == $viewer->getPHID()) {
|
|
|
|
return true;
|
|
|
|
}
|
2013-07-26 22:19:49 +02:00
|
|
|
return $this->getQuestion()->hasAutomaticCapability(
|
|
|
|
$capability,
|
|
|
|
$viewer);
|
|
|
|
case PhabricatorPolicyCapability::CAN_EDIT:
|
|
|
|
return ($this->getAuthorPHID() == $viewer->getPHID());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-29 02:23:04 +02:00
|
|
|
|
2013-09-27 17:43:41 +02:00
|
|
|
public function describeAutomaticCapability($capability) {
|
|
|
|
$out = array();
|
|
|
|
$out[] = pht("The author of an answer can always view and edit it.");
|
|
|
|
switch ($capability) {
|
|
|
|
case PhabricatorPolicyCapability::CAN_VIEW:
|
|
|
|
$out[] = pht(
|
|
|
|
"The user who asks a question can always view the answers.");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return $out;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-29 02:23:04 +02:00
|
|
|
/* -( PhabricatorTokenReceiverInterface )---------------------------------- */
|
|
|
|
|
|
|
|
|
|
|
|
public function getUsersToNotifyOfTokenGiven() {
|
|
|
|
return array(
|
|
|
|
$this->getAuthorPHID(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* -( PhabricatorSubscribableInterface )----------------------------------- */
|
|
|
|
|
|
|
|
|
|
|
|
public function isAutomaticallySubscribed($phid) {
|
|
|
|
return ($phid == $this->getAuthorPHID());
|
|
|
|
}
|
|
|
|
|
Make Projects a PhabricatorSubscribableInterface, but with restricted defaults
Summary:
Ref T4379. I want project subscriptions to work like this (yell if this seems whacky, since it makes subscriptions mean somethign a little different for projects than they do for other objects):
- You can only subscribe to a project if you're a project member.
- When you're added as a member, you're added as a subscriber.
- When you're removed as a member, you're removed as a subscriber.
- While you're a member, you can optionally unsubscribe.
From a UI perspective:
- We don't show the subscriber list, since it's going to be some uninteresting subset of the member list.
- We don't show CC transactions in history, since they're an uninteresting near-approximation of the membership transactions.
- You only see the subscription controls if you're a member.
To do this, I've augmented `PhabricatorSubscribableInterface` with two new methods. It would be nice if we were on PHP 5.4+ and could just use traits for this, but we should get data about version usage before we think about this. For now, copy/paste the default implementations into every implementing class.
Then, I implemented the interface in `PhabricatorProject` but with alternate defaults.
Test Plan:
- Used the normal interaction on existing objects.
- This has no actual effect on projects, verified no subscription stuff mysteriously appeared.
- Hit the new error case by fiddling with the UI.
Reviewers: btrahan
Reviewed By: btrahan
CC: chad, aran
Maniphest Tasks: T4379
Differential Revision: https://secure.phabricator.com/D8165
2014-02-10 23:29:17 +01:00
|
|
|
public function shouldShowSubscribersProperty() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function shouldAllowSubscription($phid) {
|
|
|
|
return true;
|
|
|
|
}
|
2013-07-29 02:23:04 +02:00
|
|
|
|
2012-08-10 19:44:04 +02:00
|
|
|
}
|