diff --git a/composer.json b/composer.json index 4c74ff3..7023ca9 100644 --- a/composer.json +++ b/composer.json @@ -6,7 +6,7 @@ "ext-ctype": "*", "ext-iconv": "*", "craue/formflow-bundle": "^3.2", - "friendsofsymfony/jsrouting-bundle": "^2.3", + "friendsofsymfony/jsrouting-bundle": "^2.4", "sensio/framework-extra-bundle": "^5.4", "symfony/asset": "4.3.*", "symfony/console": "4.3.*", diff --git a/composer.lock b/composer.lock index 76afe72..7b977a1 100644 --- a/composer.lock +++ b/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#composer-lock-the-lock-file", "This file is @generated automatically" ], - "content-hash": "26bef0bc0c89d795031fc70f28fdc8bd", + "content-hash": "428bb8a6c6ea6826f24ac19126f513ac", "packages": [ { "name": "craue/formflow-bundle", @@ -581,16 +581,16 @@ }, { "name": "friendsofsymfony/jsrouting-bundle", - "version": "2.3.1", + "version": "2.4.0", "source": { "type": "git", "url": "https://github.com/FriendsOfSymfony/FOSJsRoutingBundle.git", - "reference": "f6c9ee6857bce5924fbd54d4c6176a4dfa9de5b4" + "reference": "e42ed450eac2b61d5fcba9cd834c294a429e9a40" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/FriendsOfSymfony/FOSJsRoutingBundle/zipball/f6c9ee6857bce5924fbd54d4c6176a4dfa9de5b4", - "reference": "f6c9ee6857bce5924fbd54d4c6176a4dfa9de5b4", + "url": "https://api.github.com/repos/FriendsOfSymfony/FOSJsRoutingBundle/zipball/e42ed450eac2b61d5fcba9cd834c294a429e9a40", + "reference": "e42ed450eac2b61d5fcba9cd834c294a429e9a40", "shasum": "" }, "require": { @@ -623,13 +623,13 @@ "MIT" ], "authors": [ - { - "name": "FriendsOfSymfony Community", - "homepage": "https://github.com/friendsofsymfony/FOSJsRoutingBundle/contributors" - }, { "name": "William Durand", "email": "will+git@drnd.me" + }, + { + "name": "FriendsOfSymfony Community", + "homepage": "https://github.com/friendsofsymfony/FOSJsRoutingBundle/contributors" } ], "description": "A pretty nice way to expose your Symfony2 routing to client applications.", @@ -639,7 +639,7 @@ "javascript", "routing" ], - "time": "2019-06-17T18:22:41+00:00" + "time": "2019-08-10T15:40:05+00:00" }, { "name": "psr/cache", diff --git a/src/Controller/FediPlanController.php b/src/Controller/FediPlanController.php index a9683cf..5c0741d 100644 --- a/src/Controller/FediPlanController.php +++ b/src/Controller/FediPlanController.php @@ -16,10 +16,11 @@ use App\SocialEntity\Compose; use App\SocialEntity\MastodonAccount; use DateTime; use DateTimeZone; +use Sensio\Bundle\FrameworkExtraBundle\Configuration\Method; use Symfony\Bundle\FrameworkBundle\Controller\AbstractController; use Symfony\Component\Form\FormError; +use Symfony\Component\HttpFoundation\JsonResponse; use Symfony\Component\HttpFoundation\Request; -use Symfony\Component\HttpFoundation\Session\Session; use Symfony\Component\Routing\Annotation\Route; use Symfony\Component\Security\Core\Authorization\AuthorizationCheckerInterface; use Symfony\Component\Security\Http\Event\InteractiveLoginEvent; @@ -213,7 +214,58 @@ class FediPlanController extends AbstractController public function scheduled() { - return $this->render("fediplan/index.html.twig"); + return $this->render("fediplan/scheduled.html.twig"); + } + + + /** + * @Route("/scheduled/messages/{max_id}", name="load_more", options={"expose"=true}) + */ + public function loadMoreAction(Request $request, Mastodon_api $mastodon_api, String $max_id = null){ + + + $user = $this->getUser(); + /** @var $mastodon_api Mastodon_api */ + $mastodon_api->set_url("https://" . $user->getInstance()); + + $token = explode(" " ,$user->getToken())[1]; + $type = explode(" ", $user->getToken())[0]; + $mastodon_api->set_token($token, $type); + + $params = []; + + if( $max_id != null){ + $params['max_id'] = $max_id; + } + $scheduled_reply = []; + try { + $scheduled_reply = $mastodon_api->get_scheduled($params); + } catch (\ErrorException $e) { + } + $statuses = $mastodon_api->getScheduledStatuses($scheduled_reply['response'], $this->getUser()); + $data['max_id'] = $scheduled_reply['max_id']; + $data['html'] = $this->renderView('fediplan/Ajax/layout.html.twig', ['statuses' => $statuses]); + return new JsonResponse($data); + } + + /** + * @Method({"POST"}) + * @Route("/scheduled/delete/messages/{id}", name="delete_message", options={"expose"=true}) + */ + public function deleteMessage(Request $request, Mastodon_api $mastodon_api, String $id = null){ + + + $user = $this->getUser(); + /** @var $mastodon_api Mastodon_api */ + $mastodon_api->set_url("https://" . $user->getInstance()); + $token = explode(" " ,$user->getToken())[1]; + $type = explode(" ", $user->getToken())[0]; + $mastodon_api->set_token($token, $type); + $response = []; + try { + $response = $mastodon_api->delete_scheduled($id); + } catch (\ErrorException $e) {} + return new JsonResponse($response); } /** diff --git a/src/Services/Mastodon_api.php b/src/Services/Mastodon_api.php index 601ee88..a8cb54c 100644 --- a/src/Services/Mastodon_api.php +++ b/src/Services/Mastodon_api.php @@ -170,7 +170,6 @@ class Mastodon_api { 'Authorization' => $this->token['token_type'] . ' ' . $this->token['access_token'] ); } - $url = $this->mastodon_url.$url; $response = $this->get_content_remote($url,$parameters); return $response; @@ -230,7 +229,6 @@ class Mastodon_api { $url .= "&max_id=".$parameters['body']['max_id']; $parameters['body'] = []; } - if( isset($parameters["method"]) && $parameters['method'] == "POST" ) $response = $curl->post($url, $parameters['body'] ); else if( isset($parameters["method"]) && $parameters['method'] == "GET" ) @@ -240,7 +238,8 @@ class Mastodon_api { else if( isset($parameters["method"]) && $parameters['method'] == "PATCH" ) $response = $curl->patch($url, $parameters['body'] ); else if( isset($parameters["method"]) && $parameters['method'] == "DELETE" ) - $response = $curl->delete($url, $parameters['body'] ); + $response = $curl->delete($url); + $min_id = null; $max_id = null; if( $response->response_headers) { @@ -1048,6 +1047,22 @@ class Mastodon_api { return $response; } + + /** + * delete_scheduled + * + * Deleting a scheduled status + * + * @param int $id + * + * @return array $response empty + * @throws \ErrorException + */ + public function delete_scheduled ($id) { + $response = $this->_delete('/api/v1/scheduled_statuses/'.$id); + return $response; + } + /** * statuses_reblog * @@ -1108,6 +1123,20 @@ class Mastodon_api { return $response; } + + /** + * scheduled_statuses + * + * @param array $parameters + * + * @return array $response + * @throws \ErrorException + */ + public function get_scheduled ($parameters = array()) { + $response = $this->_get('/api/v1/scheduled_statuses/', $parameters); + return $response; + } + /** * timelines_home * @@ -1338,6 +1367,8 @@ class Mastodon_api { return $statuses; } + + /** * getSingleNotification Hydrate a Notification from API reply * @param $notificationParams @@ -1452,6 +1483,95 @@ class Mastodon_api { } + + + /** + * getScheduledStatuses Hydrate an array of Scheduled Status from API reply + * @param $statusParams + * @return array + */ + public function getScheduledStatuses($statusParams, $account){ + $statuses = []; + foreach ($statusParams as $statusParam) + $statuses[] = $this->getSingleScheduledStatus($statusParam, $account); + return $statuses; + } + + + /** + * getSingleScheduledStatus Hydrate a scheduled Status from API reply + * @param $statusParams + * @return Status + */ + public function getSingleScheduledStatus($statusParams, $account){ + + $status = new Status(); + $status->setId($statusParams['id']); + $status->setInReplyToId($statusParams['params']['in_reply_to_id']); + $status->setContent($statusParams['params']['text']); + $status->setScheduledAt($this->stringToDate($statusParams['scheduled_at'])); + $status->setAccount($account); + if( isset($statusParams['emojis']) && count($statusParams['emojis']) > 0){ + $emojis = []; + foreach ($statusParams['emojis'] as $_e){ + $emoji = new Emoji(); + $emoji->setUrl($_e['url']); + $emoji->setShortcode($_e['shortcode']); + $emoji->setStaticUrl($_e['static_url']); + $emoji->setVisibleInPicker($_e['visible_in_picker']); + $emojis[] = $emoji; + } + $status->setEmojis($emojis); + } + $status->setSensitive($statusParams['params']['sensitive']); + $status->setSpoilerText($statusParams['params']['spoiler_text']);; + $status->setVisibility($statusParams['params']['visibility']); + if( isset($statusParams['media_attachments']) && count($statusParams['media_attachments']) > 0){ + $media_attachments = []; + foreach ($statusParams['media_attachments'] as $_m){ + $attachment = new Attachment(); + $attachment->setId($_m['id']); + $attachment->setUrl($_m['url']); + $attachment->setType($_m['type']); + if($_m['remote_url']) + $attachment->setRemoteUrl($_m['remote_url']); + $attachment->setPreviewUrl($_m['preview_url']); + if($_m['text_url']) + $attachment->setTextUrl($_m['text_url']); + $attachment->setMeta(serialize($_m['meta'])); + if($_m['description']) + $attachment->setDescription($_m['description']); + $media_attachments[] = $attachment; + } + $status->setMediaAttachments($media_attachments); + } + if( isset($statusParams['mentions']) && count($statusParams['mentions']) > 0){ + $mentions = []; + foreach ($statusParams['mentions'] as $_m){ + $mention = new Mention(); + $mention->setUrl($_m['url']); + $mention->setAcct($_m['acct']); + $mention->setUsername($_m['username']); + $mention->setId($_m['id']); + $mentions[] = $mention; + } + $status->setMentions($mentions); + } + + if( isset($statusParams['tags']) && count($statusParams['tags']) > 0){ + $tags = []; + foreach ($statusParams['tags'] as $_t){ + $tag = new Tag(); + $tag->setUrl($_t['url']); + $tag->setName($_t['name']); + $tag->setHistory(isset($_t['history'])?$_t['history']:[]); + $tags[] = $tag; + } + $status->setTags($tags); + } + return $status; + } + /** * getSingleAttachment Hydrate an Attachment from API reply * @param $mediaParams diff --git a/src/SocialEntity/Status.php b/src/SocialEntity/Status.php index d8d012f..f087d20 100644 --- a/src/SocialEntity/Status.php +++ b/src/SocialEntity/Status.php @@ -4,9 +4,6 @@ namespace App\SocialEntity; -use App\Entity\Emoji; -use App\Entity\MastodonAccount; - class Status { /** @var string */ @@ -25,6 +22,8 @@ class Status private $content; /** @var \DateTime */ private $created_at; + /** @var \DateTime */ + private $scheduled_at; /** @var Emoji[] */ private $emojis = []; /** @var int */ @@ -175,9 +174,9 @@ class Status } /** - * @return \DateTime + * @return \DateTime|null */ - public function getCreatedAt(): \DateTime + public function getCreatedAt(): ?\DateTime { return $this->created_at; } @@ -185,11 +184,28 @@ class Status /** * @param \DateTime $created_at */ - public function setCreatedAt(\DateTime $created_at): void + public function setCreatedAt(?\DateTime $created_at): void { $this->created_at = $created_at; } + + /** + * @return \DateTime|null + */ + public function getScheduledAt(): ?\DateTime + { + return $this->scheduled_at; + } + + /** + * @param \DateTime $scheduled_at + */ + public function setScheduledAt(\DateTime $scheduled_at): void + { + $this->scheduled_at = $scheduled_at; + } + /** * @return Emoji[] */ @@ -321,7 +337,7 @@ class Status /** * @return string */ - public function getSpoilerText(): string + public function getSpoilerText(): ?string { return $this->spoiler_text; } @@ -329,7 +345,7 @@ class Status /** * @param string $spoiler_text */ - public function setSpoilerText(string $spoiler_text): void + public function setSpoilerText(?string $spoiler_text): void { $this->spoiler_text = $spoiler_text; } diff --git a/templates/fediplan/Ajax/layout.html.twig b/templates/fediplan/Ajax/layout.html.twig new file mode 100644 index 0000000..0bc63cc --- /dev/null +++ b/templates/fediplan/Ajax/layout.html.twig @@ -0,0 +1,43 @@ +{# @var status \App\SocialEntity\Status #} +{% for status in statuses %} + +
+ {% if status.spoilerText is defined %}
+ {{ status.spoilerText }}
+ {% endif %}
+ {{ status.content }}
+