diff --git a/composer.json b/composer.json index 7031c4c..a0295b1 100644 --- a/composer.json +++ b/composer.json @@ -15,6 +15,7 @@ "symfony/dotenv": "4.3.*", "symfony/flex": "^1.3.1", "symfony/framework-bundle": "4.3.*", + "symfony/polyfill-intl-messageformatter": "^1.15", "symfony/security-bundle": "4.3.*", "symfony/translation": "4.3.*", "symfony/twig-bundle": "4.3.*", diff --git a/composer.lock b/composer.lock index b5616ea..3156faa 100644 --- a/composer.lock +++ b/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "cc98799ce9bc12d4405acb67e867d8b7", + "content-hash": "421672a1e764c0f375493f05beb754dd", "packages": [ { "name": "craue/formflow-bundle", @@ -2104,6 +2104,83 @@ ], "time": "2020-03-09T19:04:49+00:00" }, + { + "name": "symfony/polyfill-intl-messageformatter", + "version": "v1.15.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/polyfill-intl-messageformatter.git", + "reference": "3326c736f61bbb2d030622ff128a590b41ed46b5" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/polyfill-intl-messageformatter/zipball/3326c736f61bbb2d030622ff128a590b41ed46b5", + "reference": "3326c736f61bbb2d030622ff128a590b41ed46b5", + "shasum": "" + }, + "require": { + "php": ">=5.3.3" + }, + "suggest": { + "ext-intl": "For best performance" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.15-dev" + } + }, + "autoload": { + "psr-4": { + "Symfony\\Polyfill\\Intl\\MessageFormatter\\": "" + }, + "files": [ + "bootstrap.php" + ], + "classmap": [ + "Resources/stubs" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Symfony polyfill for intl's MessageFormatter class and related functions", + "homepage": "https://symfony.com", + "keywords": [ + "compatibility", + "intl", + "messageformatter", + "polyfill", + "portable", + "shim" + ], + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2020-02-27T09:26:54+00:00" + }, { "name": "symfony/polyfill-mbstring", "version": "v1.15.0", diff --git a/config/services.yaml b/config/services.yaml index c67634b..8cc42d5 100644 --- a/config/services.yaml +++ b/config/services.yaml @@ -33,3 +33,6 @@ services: mastodon.api: class: App\Services\Mastodon_api public: true + app.form: + class: App\Form\ComposeType + arguments: ['@security.helper','@translator.default'] \ No newline at end of file diff --git a/public/css/style.css b/public/css/style.css index ec9f585..d116a6e 100644 --- a/public/css/style.css +++ b/public/css/style.css @@ -24,3 +24,10 @@ html, body { background-color: #f5f5f5; } +.switch_width{ + width: 50px; +} + +ul.options { + list-style: none; +} \ No newline at end of file diff --git a/public/js/emojionearea.js b/public/js/emojionearea.js index 3c924eb..02d59bb 100644 --- a/public/js/emojionearea.js +++ b/public/js/emojionearea.js @@ -1463,7 +1463,7 @@ document = window.document || {}; if (typeof value.acct == 'undefined') { return shortnameTo(value, self.emojiTemplate); }else{ - return "@"+value.acct; + return "@"+value.acct+ " "; } }, cache: true, diff --git a/src/Form/ComposeType.php b/src/Form/ComposeType.php index 93ab715..2fc8a02 100644 --- a/src/Form/ComposeType.php +++ b/src/Form/ComposeType.php @@ -10,33 +10,37 @@ namespace App\Form; use App\SocialEntity\Compose; +use App\SocialEntity\MastodonAccount; +use DateTime; use Symfony\Component\Form\AbstractType; use Symfony\Component\Form\Extension\Core\Type\CheckboxType; use Symfony\Component\Form\Extension\Core\Type\CollectionType; -use Symfony\Component\Form\Extension\Core\Type\FileType; +use Symfony\Component\Form\Extension\Core\Type\DateTimeType; use Symfony\Component\Form\Extension\Core\Type\SubmitType; use Symfony\Component\Form\Extension\Core\Type\TextareaType; use Symfony\Component\Form\Extension\Core\Type\TextType; use Symfony\Component\Form\Extension\Core\Type\TimezoneType; use Symfony\Component\Form\FormBuilderInterface; -use Symfony\Component\HttpFoundation\File\File; use Symfony\Component\OptionsResolver\OptionsResolver; use Symfony\Component\Form\Extension\Core\Type\ChoiceType; use Symfony\Component\Security\Core\Security; +use Symfony\Component\Translation\Translator; class ComposeType extends AbstractType { private $securityContext; + private $translator; - public function __construct(Security $securityContext) + public function __construct(Security $securityContext, Translator $translator) { $this->securityContext = $securityContext; + $this->translator = $translator; } public function buildForm(FormBuilderInterface $builder, array $options) { - /**@var $user \App\SocialEntity\MastodonAccount**/ + /**@var $user MastodonAccount*/ $user = $options['user']; if( $user->getDefaultSensitivity()) { @@ -75,9 +79,9 @@ class ComposeType extends AbstractType { 'label' => 'page.schedule.form.timeZone', 'translation_domain' => 'fediplan']); $builder->add('sensitive', CheckboxType::class, $checkbox); - $builder->add('scheduled_at', \Symfony\Component\Form\Extension\Core\Type\DateTimeType::class,[ + $builder->add('scheduled_at', DateTimeType::class,[ 'widget' => 'single_text', - "data" => new \DateTime(), + "data" => new DateTime(), 'label' => 'page.schedule.form.scheduled_at', 'translation_domain' => 'fediplan']); $builder->add('Send', SubmitType::class, @@ -85,6 +89,34 @@ class ComposeType extends AbstractType { 'label' => 'page.schedule.form.send', 'translation_domain' => 'fediplan']); + $builder->add('poll_option_1', TextType::class, ['required' => false]); + $builder->add('poll_option_2', TextType::class, ['required' => false]); + $builder->add('poll_options', CollectionType::class, + [ + 'entry_type' => PollOptionType::class, + 'allow_add' => true, + 'prototype' => true, + 'allow_delete' => true, + 'required' => false, + ]); + $builder->add('poll_multiple', CheckboxType::class, + ['required' => false, 'label' => 'page.schedule.form.multiple', + 'translation_domain' => 'fediplan']); + $builder->add('poll_expires_at', ChoiceType::class, + [ + 'choices' => [ + $this->translator->trans('poll.duration_m', ['minutes' => 5], 'fediplan') => 5*60, + $this->translator->trans('poll.duration_m', ['minutes' => 30], 'fediplan') => 30*60, + $this->translator->trans('poll.duration_h', ['hours' => 1], 'fediplan') => 60*60, + $this->translator->trans('poll.duration_h', ['hours' => 6], 'fediplan') => 6*60*60, + $this->translator->trans('poll.duration_d', ['days' => 1], 'fediplan') => 24*60*60, + $this->translator->trans('poll.duration_d', ['days' => 3], 'fediplan') => 3*24*60*60, + $this->translator->trans('poll.duration_d', ['days' => 7], 'fediplan') => 7*24*60*60, + + ], + 'required' => false, + 'label' => 'page.schedule.form.end_in', + 'translation_domain' => 'fediplan']); } diff --git a/src/Form/PollOptionType.php b/src/Form/PollOptionType.php new file mode 100644 index 0000000..abf8f80 --- /dev/null +++ b/src/Form/PollOptionType.php @@ -0,0 +1,45 @@ +securityContext = $securityContext; + } + + public function buildForm(FormBuilderInterface $builder, array $options) + { + $builder->add('option', TextType::class, + [ + 'required' => false, + 'attr'=> ['class'=>'form-control'] + ]); + } + + + public function configureOptions(OptionsResolver $resolver) + { + $resolver->setDefaults([ + 'data_class' => PollOption::class, + 'translation_domain' => 'fediplan' + ]); + } + +} \ No newline at end of file diff --git a/src/Services/Mastodon_api.php b/src/Services/Mastodon_api.php index 1a7cdf0..c727fab 100644 --- a/src/Services/Mastodon_api.php +++ b/src/Services/Mastodon_api.php @@ -249,7 +249,7 @@ class Mastodon_api { if (strpos($value, 'Link: ') !== false) { preg_match( - "/min_id=([0-9a-zA-Z]{1,})/", + "/min_id=([0-9a-zA-Z]+)/", $value, $matches ); @@ -259,7 +259,7 @@ class Mastodon_api { } if (strpos($value, 'Link: ') !== false) { preg_match( - "/max_id=([0-9a-zA-Z]{1,})/", + "/max_id=([0-9a-zA-Z]+)/", $value, $matches ); diff --git a/src/SocialEntity/Compose.php b/src/SocialEntity/Compose.php index d2d5a6a..6c3d220 100644 --- a/src/SocialEntity/Compose.php +++ b/src/SocialEntity/Compose.php @@ -30,8 +30,16 @@ class Compose private $timeZone; - /** @var Poll */ - private $poll; + /** @var PollOption[] */ + private $poll_options; + /** @var int */ + private $poll_expires_at; + /** @var bool */ + private $poll_multiple; + /** @var PollOption */ + private $poll_option_1; + /** @var PollOption */ + private $poll_option_2; public function __construct() { @@ -159,19 +167,83 @@ class Compose } /** - * @return Poll + * @return PollOption[] */ - public function getPoll(): Poll + public function getPollOptions(): ?array { - return $this->poll; + return $this->poll_options; } /** - * @param Poll $poll + * @param PollOption[] $poll_options */ - public function setPoll(Poll $poll): void + public function setPollOptions(?array $poll_options): void { - $this->poll = $poll; + $this->poll_options = $poll_options; + } + + /** + * @return int + */ + public function getPollExpiresAt(): ?int + { + return $this->poll_expires_at; + } + + /** + * @param int $poll_expires_at + */ + public function setPollExpiresAt(?int $poll_expires_at): void + { + $this->poll_expires_at = $poll_expires_at; + } + + /** + * @return bool + */ + public function isPollMultiple(): ?bool + { + return $this->poll_multiple; + } + + /** + * @param bool $poll_multiple + */ + public function setPollMultiple(?bool $poll_multiple): void + { + $this->poll_multiple = $poll_multiple; + } + + /** + * @return PollOption + */ + public function getPollOption1(): ?PollOption + { + return $this->poll_option_1; + } + + /** + * @param PollOption $poll_option_1 + */ + public function setPollOption1(?PollOption $poll_option_1): void + { + $this->poll_option_1 = $poll_option_1; + } + + /** + * @return PollOption + */ + public function getPollOption2(): ?PollOption + { + return $this->poll_option_2; + } + + /** + * @param PollOption $poll_option_2 + */ + public function setPollOption2(?PollOption $poll_option_2): void + { + $this->poll_option_2 = $poll_option_2; } diff --git a/symfony.lock b/symfony.lock index 42155cd..2851ba5 100644 --- a/symfony.lock +++ b/symfony.lock @@ -160,6 +160,9 @@ "symfony/polyfill-intl-idn": { "version": "v1.12.0" }, + "symfony/polyfill-intl-messageformatter": { + "version": "v1.15.0" + }, "symfony/polyfill-mbstring": { "version": "v1.12.0" }, diff --git a/templates/fediplan/schedule.html.twig b/templates/fediplan/schedule.html.twig index 7868480..bb194be 100644 --- a/templates/fediplan/schedule.html.twig +++ b/templates/fediplan/schedule.html.twig @@ -4,7 +4,7 @@ {% block content %} {% include 'nav.html.twig' %} -

Schedule

+

Schedule for {{ app.user.avatar }} {{ convertAccountEmoji(app.user, app.user.displayName) | raw }} (@{{ app.user.acct}}@{{ instance }})

{% for type, messages in app.session.flashbag.all() %} {% for message in messages %} @@ -25,51 +25,59 @@ {% endfor %} -
-
-
-
-
- -
-
-

{{ convertAccountEmoji(app.user, app.user.displayName) | raw }}

-

@{{ app.user.acct }}

-
-
- -
-
-
{{ form_start(form, {'attr': {'novalidate': 'novalidate'}}) }}
-
-
-
- {{ form_label(form.content_warning) }} - {{ form_widget(form.content_warning, {'attr': {'class': 'form-control', 'data-emojiable':'true'}}) }} - {% if not form.content_warning.vars.errors is empty %} - +
+ {{ form_label(form.content_warning) }} + {{ form_widget(form.content_warning, {'attr': {'class': 'form-control', 'data-emojiable':'true'}}) }} + {% if not form.content_warning.vars.errors is empty %} + {% for errorItem in form.content_warning.vars.errors %} {{ errorItem.message }} {% endfor %} - {% endif %} -
-
+ {% endif %} +
+ +
+ {{ form_label(form.content) }} + {{ form_widget(form.content, {'attr': {'class': 'form-control','id':'composer_content','data-emojiable':'true'}}) }} + {% if not form.content.vars.errors is empty %} + + {% for errorItem in form.content.vars.errors %} + {{ errorItem.message }} + {% endfor %} + + {% endif %}
-
-
- {{ form_label(form.content) }} - {{ form_widget(form.content, {'attr': {'class': 'form-control','id':'composer_content','data-emojiable':'true'}}) }} - {% if not form.content.vars.errors is empty %} +
+
+   0 +
+
+
+
+ {{ form_label(form.visibility) }}   + {{ form_widget(form.visibility, {'attr': {'class': 'form-control'}}) }} + {% if not form.visibility.vars.errors is empty %} - {% for errorItem in form.content.vars.errors %} + {% for errorItem in form.visibility.vars.errors %} + {{ errorItem.message }} + {% endfor %} + + {% endif %} +
+
+
+
+ {{ form_label(form.sensitive) }}   + {{ form_widget(form.sensitive, {'attr': {'class': 'form-control','data-toggle':'toggle', 'data-on': 'common.yes'|trans , 'data-off':'common.no'|trans}}) }} + {% if not form.sensitive.vars.errors is empty %} + + {% for errorItem in form.sensitive.vars.errors %} {{ errorItem.message }} {% endfor %} @@ -77,49 +85,8 @@
- -
-
-
- {{ 'common.counter'|trans }}: 0 -
-
-
- -
-
-
- {{ form_label(form.visibility) }} - {{ form_widget(form.visibility, {'attr': {'class': 'form-control'}}) }} - {% if not form.visibility.vars.errors is empty %} - - {% for errorItem in form.visibility.vars.errors %} - {{ errorItem.message }} - {% endfor %} - - {% endif %} -
-
-
- -
-
-
- {{ form_label(form.sensitive) }}   - {{ form_widget(form.sensitive, {'attr': {'class': 'form-control','data-toggle':'toggle', 'data-on': 'common.yes'|trans , 'data-off':'common.no'|trans}}) }} - {% if not form.sensitive.vars.errors is empty %} - - {% for errorItem in form.sensitive.vars.errors %} - {{ errorItem.message }} - {% endfor %} - - {% endif %} -
-
-
- -
-
+
+
{{ form_label(form.scheduled_at) }} {{ form_widget(form.scheduled_at, {'attr': {'class': 'form-control'}}) }} @@ -132,7 +99,7 @@ {% endif %}
-
+
{{ form_label(form.timeZone) }} {{ form_widget(form.timeZone, {'attr': {'class': 'form-control'}}) }} @@ -147,10 +114,70 @@
+ + +
    +
    + {{ form_label(form.poll_option_1) }} + {{ form_widget(form.poll_option_1, {'attr': {'class': 'form-control'}}) }} + {% if not form.poll_option_1.vars.errors is empty %} + + {% for errorItem in form.poll_option_1.vars.errors %} + {{ errorItem.message }} + {% endfor %} + + {% endif %} +
    +
    + {{ form_label(form.poll_option_2) }} + {{ form_widget(form.poll_option_2, {'attr': {'class': 'form-control'}}) }} + {% if not form.poll_option_2.vars.errors is empty %} + + {% for errorItem in form.poll_option_2.vars.errors %} + {{ errorItem.message }} + {% endfor %} + + {% endif %} +
    +
+
+
+
+
+ {{ form_label(form.poll_multiple) }} + {{ form_widget(form.poll_multiple, {'attr': {'class': 'form-control','data-toggle':'toggle', 'data-on': 'common.yes'|trans , 'data-off':'common.no'|trans}}) }} + {% if not form.poll_multiple.vars.errors is empty %} + + {% for errorItem in form.poll_multiple.vars.errors %} + {{ errorItem.message }} + {% endfor %} + + {% endif %} +
+
+
+ {{ form_label(form.poll_expires_at) }}    + {{ form_widget(form.poll_expires_at, {'attr': {'class': 'form-control'}}) }} + {% if not form.poll_expires_at.vars.errors is empty %} + + {% for errorItem in form.poll_expires_at.vars.errors %} + {{ errorItem.message }} + {% endfor %} + + {% endif %} +
+
+
+
+
+
+ +
{{ form_end(form) }}
@@ -189,7 +216,7 @@ >
@@ -449,6 +476,42 @@