Compare commits
	
		
			207 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 9302369aa5 | ||
|  | a0047fdca0 | ||
|  | a20509b41e | ||
|  | 281c577cf8 | ||
|  | f9a0d8f2b9 | ||
|  | 4de211b80a | ||
|  | 063a1c2a8b | ||
|  | a9ca5ce920 | ||
|  | d7a17b5e8b | ||
|  | 34e2a06de0 | ||
|  | 4c1a02fa73 | ||
|  | b21db9bbde | ||
|  | 42bcbd36b7 | ||
|  | 0393a64a90 | ||
|  | d68868ca14 | ||
|  | e20895f251 | ||
|  | 12cea76634 | ||
|  | b4bc594c5a | ||
|  | 82884ac5c4 | ||
|  | 886829e96c | ||
|  | 62a94ebed4 | ||
|  | ac17309faf | ||
|  | dd23ae031f | ||
|  | 51f2eca887 | ||
|  | bdf6d0a684 | ||
|  | b15482ce71 | ||
|  | 74320971e2 | ||
|  | eee3b32b77 | ||
|  | df03042a6e | ||
|  | 9927df83ad | ||
|  | 4c6b5dbe96 | ||
|  | 85e97ecab6 | ||
|  | dc1ebd45a3 | ||
|  | f0d4c7d7ab | ||
|  | 82ab9736d5 | ||
|  | a62039df27 | ||
|  | 15fab79cfa | ||
|  | eeaec39888 | ||
|  | b8efb5daed | ||
|  | 2b3b44ebbc | ||
|  | 1b57d4dd3a | ||
|  | d937a59997 | ||
|  | 706e534455 | ||
|  | ff78c1177a | ||
|  | c6b7c77229 | ||
|  | e20258a2e5 | ||
|  | 7fb850e987 | ||
|  | 1c5b0e3334 | ||
|  | 740f8a95a9 | ||
|  | 0ea4478b68 | ||
|  | fd87e5a53b | ||
|  | 57fe4102ea | ||
|  | bf7757cbbc | ||
|  | 1266c66f79 | ||
|  | d07983b56d | ||
|  | 662b8eefe8 | ||
|  | 520d147803 | ||
|  | 32987004c9 | ||
|  | 31ac5f0e00 | ||
|  | 269a445c0b | ||
|  | 2b51b4094c | ||
|  | 1104ac35d3 | ||
|  | a78f66c069 | ||
|  | 8c0e77d688 | ||
|  | 7a45d382ea | ||
|  | 5a551b530a | ||
|  | e84fecb7e9 | ||
|  | 801eee0ff3 | ||
|  | bc4a726c24 | ||
|  | fc2155019b | ||
|  | 53b7b81b43 | ||
|  | 6f609dc4b4 | ||
|  | 3dce6cbbd7 | ||
|  | 2bcc81700c | ||
|  | 53e95c4efc | ||
|  | 08deec4c84 | ||
|  | 2590aac863 | ||
|  | 3d1d3d9a20 | ||
|  | 8f638a2bf2 | ||
|  | 9d9b1aff1e | ||
|  | bfdcf76a64 | ||
|  | b380e9d2cb | ||
|  | 58cede4808 | ||
|  | 6be72a3ec6 | ||
|  | 2864e5e077 | ||
|  | 24cafd73a2 | ||
|  | 4a2fc2d444 | ||
|  | e1ebf36352 | ||
|  | ae6dd08121 | ||
|  | e28b33c89c | ||
|  | dc6e031364 | ||
|  | 9dd5e329ab | ||
|  | 3e90987c8b | ||
|  | 2151fd3150 | ||
|  | ad207456d6 | ||
|  | 9e3d24a150 | ||
|  | 556c07df1f | ||
|  | 3023725936 | ||
|  | 3e4b01b47d | ||
|  | 19e8b861a2 | ||
|  | 7d7df877ef | ||
|  | c73a1fb537 | ||
|  | f6bc6399e2 | ||
|  | 031a5a8f92 | ||
|  | 6d7e05ec1f | ||
|  | 58bca7b1e4 | ||
|  | 1c25853842 | ||
|  | 546257bc7f | ||
|  | fbef909c2a | ||
|  | c3ec1e87b8 | ||
|  | 48e27c47a7 | ||
|  | 1f1838420f | ||
|  | 20150659e6 | ||
|  | 8087aa83d4 | ||
|  | 249b0fe107 | ||
|  | a6682a3000 | ||
|  | 4112a0631f | ||
|  | 0e6c4cb796 | ||
|  | 92aaa55f06 | ||
|  | 5df8e30415 | ||
|  | 60f247c2e7 | ||
|  | cf7e840990 | ||
|  | 252d0fe020 | ||
|  | 2fb722397d | ||
|  | 07f7192bc3 | ||
|  | fcb9533549 | ||
|  | 7bb8b0b2fc | ||
|  | 2b1190065c | ||
|  | 56720ba590 | ||
|  | e5aa4128f6 | ||
|  | f9e7336296 | ||
|  | 07cca6e364 | ||
|  | 54b42901df | ||
|  | d200e041fe | ||
|  | 49a285ce15 | ||
|  | cfd7b7a0b7 | ||
|  | 36376b5e23 | ||
|  | eb97bd8af6 | ||
|  | 4c0a85ef9b | ||
|  | 64cc129225 | ||
|  | 97fc2da2e0 | ||
|  | 889ada5ee2 | ||
|  | 3f16caaa50 | ||
|  | 5d5c0f4f43 | ||
|  | 1032f3994f | ||
|  | cbbeec05be | ||
|  | e618edf85a | ||
|  | b6e2e999bd | ||
|  | 782224c991 | ||
|  | 84cfee2488 | ||
|  | 7bea1530f4 | ||
|  | 47b0c61853 | ||
|  | 864c4d869f | ||
|  | d8cd9000d9 | ||
|  | d307ee79e9 | ||
|  | cf01326cc1 | ||
|  | d48779cf7b | ||
|  | 8a588145d5 | ||
|  | 8abe9e9058 | ||
|  | 15c0f6ae56 | ||
|  | da3adc0a73 | ||
|  | 0338c16f9f | ||
|  | 38d072446b | ||
|  | 8ae9bd0eea | ||
|  | 5521e94e24 | ||
|  | 763a2f8511 | ||
|  | 60f962eedc | ||
|  | 47d56438da | ||
|  | 0692991b54 | ||
|  | 6705463ed0 | ||
|  | a2a4bf4e78 | ||
|  | b254e6ca5f | ||
|  | 29609fbb6a | ||
|  | d37a56c07c | ||
|  | 2cea4592a3 | ||
|  | 512feab222 | ||
|  | 5e111ce16d | ||
|  | 4080569c2d | ||
|  | 2cbb8e8cd1 | ||
|  | 3e9236b343 | ||
|  | 89c77fe225 | ||
|  | e843f62f47 | ||
|  | ec487166db | ||
|  | 37b267e2ab | ||
|  | 3de22a82bf | ||
|  | e4080772b5 | ||
|  | 781105293c | ||
|  | 0cb329f63a | ||
|  | 0129f5eada | ||
|  | 22da775a85 | ||
|  | d556be2968 | ||
|  | 4f337c020a | ||
|  | 02f7f3619a | ||
|  | 20fee786b1 | ||
|  | 74777599cf | ||
|  | 1ba3725473 | ||
|  | fdb0848e08 | ||
|  | 8392ddbf87 | ||
|  | 049381b284 | ||
|  | 09d81defcd | ||
|  | 3810d98cd8 | ||
|  | 26b2a6a71e | ||
|  | edf9a5e4fc | ||
|  | c710069c12 | ||
|  | 990d6dd565 | ||
|  | 402da46ff6 | ||
|  | e7099d8d9e | 
| @@ -1,21 +1,36 @@ | ||||
| engines: | ||||
| version: "2" | ||||
| checks: | ||||
|   argument-count: | ||||
|     enabled: false | ||||
|   complex-logic: | ||||
|     enabled: false | ||||
|   file-lines: | ||||
|     enabled: false | ||||
|   method-complexity: | ||||
|     enabled: false | ||||
|   method-count: | ||||
|     enabled: false | ||||
|   method-lines: | ||||
|     enabled: false | ||||
|   nested-control-flow: | ||||
|     enabled: false | ||||
|   return-statements: | ||||
|     enabled: false | ||||
|   similar-code: | ||||
|     enabled: false | ||||
|   identical-code: | ||||
|     enabled: false | ||||
| plugins: | ||||
|   brakeman: | ||||
|     enabled: true | ||||
|   bundler-audit: | ||||
|     enabled: true | ||||
|   duplication: | ||||
|     enabled: false | ||||
|   eslint: | ||||
|     enabled: true | ||||
|   rubocop: | ||||
|     enabled: true | ||||
|   scss-lint: | ||||
|     enabled: true | ||||
| ratings: | ||||
|   paths: | ||||
|   - "**.rb" | ||||
|   - "**.js" | ||||
|   - "**.scss" | ||||
| exclude_paths: | ||||
| exclude_patterns: | ||||
| - spec/ | ||||
| - vendor/asset | ||||
|   | ||||
							
								
								
									
										30
									
								
								.env.nanobox
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								.env.nanobox
									
									
									
									
									
								
							| @@ -35,6 +35,17 @@ PAPERCLIP_SECRET=$PAPERCLIP_SECRET | ||||
| SECRET_KEY_BASE=$SECRET_KEY_BASE | ||||
| OTP_SECRET=$OTP_SECRET | ||||
|  | ||||
| # VAPID keys (used for push notifications) | ||||
| # You can generate the keys using the following command (first is the private key, second is the public one) | ||||
| # You should only generate this once per instance. If you later decide to change it, all push subscription will | ||||
| # be invalidated, requiring the users to access the website again to resubscribe. | ||||
| # | ||||
| # Generate with `rake mastodon:webpush:generate_vapid_key` task (`nanobox run bundle exec rake mastodon:webpush:generate_vapid_key`) | ||||
| # | ||||
| # For more information visit https://rossta.net/blog/using-the-web-push-api-with-vapid.html | ||||
| VAPID_PRIVATE_KEY=$VAPID_PRIVATE_KEY | ||||
| VAPID_PUBLIC_KEY=$VAPID_PUBLIC_KEY | ||||
|  | ||||
| # Registrations | ||||
| # Single user mode will disable registrations and redirect frontpage to the first profile | ||||
| # SINGLE_USER_MODE=true | ||||
| @@ -62,7 +73,7 @@ SMTP_FROM_ADDRESS=notifications@${APP_NAME}.nanoapp.io | ||||
| #SMTP_CA_FILE=/etc/ssl/certs/ca-certificates.crt | ||||
| #SMTP_OPENSSL_VERIFY_MODE=peer | ||||
| #SMTP_ENABLE_STARTTLS_AUTO=true | ||||
|  | ||||
| #SMTP_TLS=true | ||||
|  | ||||
| # Optional user upload path and URL (images, avatars). Default is :rails_root/public/system. If you set this variable, you are responsible for making your HTTP server (eg. nginx) serve these files. | ||||
| # PAPERCLIP_ROOT_PATH=/var/lib/mastodon/public-system | ||||
| @@ -91,6 +102,23 @@ SMTP_FROM_ADDRESS=notifications@${APP_NAME}.nanoapp.io | ||||
| # S3_ENDPOINT= | ||||
| # S3_SIGNATURE_VERSION= | ||||
|  | ||||
| # Swift (optional) | ||||
| # SWIFT_ENABLED=true | ||||
| # SWIFT_USERNAME= | ||||
| # For Keystone V3, the value for SWIFT_TENANT should be the project name | ||||
| # SWIFT_TENANT= | ||||
| # SWIFT_PASSWORD= | ||||
| # Keystone V2 and V3 URLs are supported. Use a V3 URL if possible to avoid | ||||
| # issues with token rate-limiting during high load. | ||||
| # SWIFT_AUTH_URL= | ||||
| # SWIFT_CONTAINER= | ||||
| # SWIFT_OBJECT_URL= | ||||
| # SWIFT_REGION= | ||||
| # Defaults to 'default' | ||||
| # SWIFT_DOMAIN_NAME= | ||||
| # Defaults to 60 seconds. Set to 0 to disable | ||||
| # SWIFT_CACHE_TTL= | ||||
|  | ||||
| # Optional alias for S3 if you want to use Cloudfront or Cloudflare in front | ||||
| # S3_CLOUDFRONT_HOST= | ||||
|  | ||||
|   | ||||
| @@ -27,11 +27,14 @@ addons: | ||||
|   apt: | ||||
|     sources: | ||||
|     - trusty-media | ||||
|     - sourceline: deb https://dl.yarnpkg.com/debian/ stable main | ||||
|       key_url: https://dl.yarnpkg.com/debian/pubkey.gpg | ||||
|     packages: | ||||
|     - ffmpeg | ||||
|     - libicu-dev | ||||
|     - libprotobuf-dev | ||||
|     - protobuf-compiler | ||||
|     - libicu-dev | ||||
|     - yarn | ||||
|  | ||||
| rvm: | ||||
|   - 2.3.4 | ||||
| @@ -42,7 +45,6 @@ services: | ||||
|  | ||||
| install: | ||||
|   - nvm install | ||||
|   - npm install -g yarn | ||||
|   - bundle install --path=vendor/bundle --without development production --retry=3 --jobs=16 | ||||
|   - yarn install | ||||
|  | ||||
| @@ -54,4 +56,4 @@ before_script: | ||||
| script: | ||||
|   - travis_retry bundle exec parallel_test spec/ --group-by filesize --type rspec | ||||
|   - yarn test | ||||
|   - bundle exec i18n-tasks unused | ||||
|   - bundle exec i18n-tasks check-normalized && bundle exec i18n-tasks unused | ||||
|   | ||||
							
								
								
									
										46
									
								
								CODE_OF_CONDUCT.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								CODE_OF_CONDUCT.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | ||||
| # Contributor Covenant Code of Conduct | ||||
|  | ||||
| ## Our Pledge | ||||
|  | ||||
| In the interest of fostering an open and welcoming environment, we as contributors and maintainers pledge to making participation in our project and our community a harassment-free experience for everyone, regardless of age, body size, disability, ethnicity, gender identity and expression, level of experience, nationality, personal appearance, race, religion, or sexual identity and orientation. | ||||
|  | ||||
| ## Our Standards | ||||
|  | ||||
| Examples of behavior that contributes to creating a positive environment include: | ||||
|  | ||||
| * Using welcoming and inclusive language | ||||
| * Being respectful of differing viewpoints and experiences | ||||
| * Gracefully accepting constructive criticism | ||||
| * Focusing on what is best for the community | ||||
| * Showing empathy towards other community members | ||||
|  | ||||
| Examples of unacceptable behavior by participants include: | ||||
|  | ||||
| * The use of sexualized language or imagery and unwelcome sexual attention or advances | ||||
| * Trolling, insulting/derogatory comments, and personal or political attacks | ||||
| * Public or private harassment | ||||
| * Publishing others' private information, such as a physical or electronic address, without explicit permission | ||||
| * Other conduct which could reasonably be considered inappropriate in a professional setting | ||||
|  | ||||
| ## Our Responsibilities | ||||
|  | ||||
| Project maintainers are responsible for clarifying the standards of acceptable behavior and are expected to take appropriate and fair corrective action in response to any instances of unacceptable behavior. | ||||
|  | ||||
| Project maintainers have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, or to ban temporarily or permanently any contributor for other behaviors that they deem inappropriate, threatening, offensive, or harmful. | ||||
|  | ||||
| ## Scope | ||||
|  | ||||
| This Code of Conduct applies both within project spaces and in public spaces when an individual is representing the project or its community. Examples of representing a project or community include using an official project e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event. Representation of a project may be further defined and clarified by project maintainers. | ||||
|  | ||||
| ## Enforcement | ||||
|  | ||||
| Instances of abusive, harassing, or otherwise unacceptable behavior may be reported by contacting the project team at eugen@zeonfederated.com. The project team will review and investigate all complaints, and will respond in a way that it deems appropriate to the circumstances. The project team is obligated to maintain confidentiality with regard to the reporter of an incident. Further details of specific enforcement policies may be posted separately. | ||||
|  | ||||
| Project maintainers who do not follow or enforce the Code of Conduct in good faith may face temporary or permanent repercussions as determined by other members of the project's leadership. | ||||
|  | ||||
| ## Attribution | ||||
|  | ||||
| This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4, available at [http://contributor-covenant.org/version/1/4][version] | ||||
|  | ||||
| [homepage]: http://contributor-covenant.org | ||||
| [version]: http://contributor-covenant.org/version/1/4/ | ||||
| @@ -7,8 +7,8 @@ ENV UID=991 GID=991 \ | ||||
|     RAILS_SERVE_STATIC_FILES=true \ | ||||
|     RAILS_ENV=production NODE_ENV=production | ||||
|  | ||||
| ARG YARN_VERSION=1.1.0 | ||||
| ARG YARN_DOWNLOAD_SHA256=171c1f9ee93c488c0d774ac6e9c72649047c3f896277d88d0f805266519430f3 | ||||
| ARG YARN_VERSION=1.3.2 | ||||
| ARG YARN_DOWNLOAD_SHA256=6cfe82e530ef0837212f13e45c1565ba53f5199eec2527b85ecbcd88bf26821d | ||||
| ARG LIBICONV_VERSION=1.15 | ||||
| ARG LIBICONV_DOWNLOAD_SHA256=ccf536620a45458d26ba83887a983b96827001e92a13847b45e4925cc8913178 | ||||
|  | ||||
| @@ -48,7 +48,7 @@ RUN apk -U upgrade \ | ||||
|  && rm yarn.tar.gz \ | ||||
|  && mv /tmp/src/yarn-v$YARN_VERSION /opt/yarn \ | ||||
|  && ln -s /opt/yarn/bin/yarn /usr/local/bin/yarn \ | ||||
|  && wget -O libiconv.tar.gz "http://ftp.gnu.org/pub/gnu/libiconv/libiconv-$LIBICONV_VERSION.tar.gz" \ | ||||
|  && wget -O libiconv.tar.gz "https://ftp.gnu.org/pub/gnu/libiconv/libiconv-$LIBICONV_VERSION.tar.gz" \ | ||||
|  && echo "$LIBICONV_DOWNLOAD_SHA256 *libiconv.tar.gz" | sha256sum -c - \ | ||||
|  && tar -xzf libiconv.tar.gz -C /tmp/src \ | ||||
|  && rm libiconv.tar.gz \ | ||||
|   | ||||
							
								
								
									
										33
									
								
								Gemfile
									
									
									
									
									
								
							
							
						
						
									
										33
									
								
								Gemfile
									
									
									
									
									
								
							| @@ -14,8 +14,10 @@ gem 'pg', '~> 0.20' | ||||
| gem 'pghero', '~> 1.7' | ||||
| gem 'dotenv-rails', '~> 2.2' | ||||
|  | ||||
| gem 'aws-sdk', '~> 2.9' | ||||
| gem 'fog-openstack', '~> 0.1' | ||||
| gem 'fog-aws', '~> 1.4', require: false | ||||
| gem 'fog-core', '~> 1.45' | ||||
| gem 'fog-local', '~> 0.4', require: false | ||||
| gem 'fog-openstack', '~> 0.1', require: false | ||||
| gem 'paperclip', '~> 5.1' | ||||
| gem 'paperclip-av-transcoder', '~> 0.6' | ||||
|  | ||||
| @@ -38,16 +40,15 @@ gem 'http', '~> 2.2' | ||||
| gem 'http_accept_language', '~> 2.1' | ||||
| gem 'httplog', '~> 0.99' | ||||
| gem 'idn-ruby', require: 'idn' | ||||
| gem 'kaminari', '~> 1.0' | ||||
| gem 'kaminari', '~> 1.1' | ||||
| gem 'link_header', '~> 0.0' | ||||
| gem 'mime-types', '~> 3.1' | ||||
| gem 'nokogiri', '~> 1.7' | ||||
| gem 'nokogiri', '~> 1.8' | ||||
| gem 'nsa', '~> 0.2' | ||||
| gem 'oj', '~> 3.0' | ||||
| gem 'oj', '~> 3.3' | ||||
| gem 'ostatus2', '~> 2.0' | ||||
| gem 'ox', '~> 2.5' | ||||
| gem 'ox', '~> 2.8' | ||||
| gem 'pundit', '~> 1.1' | ||||
| gem 'rabl', '~> 0.13' | ||||
| gem 'rack-attack', '~> 5.0' | ||||
| gem 'rack-cors', '~> 0.4', require: 'rack/cors' | ||||
| gem 'rack-timeout', '~> 0.4' | ||||
| @@ -75,15 +76,15 @@ gem 'json-ld-preloaded', '~> 2.2.1' | ||||
| gem 'rdf-normalize', '~> 0.3.1' | ||||
|  | ||||
| group :development, :test do | ||||
|   gem 'fabrication', '~> 2.16' | ||||
|   gem 'fabrication', '~> 2.18' | ||||
|   gem 'fuubar', '~> 2.2' | ||||
|   gem 'i18n-tasks', '~> 0.9', require: false | ||||
|   gem 'pry-rails', '~> 0.3' | ||||
|   gem 'rspec-rails', '~> 3.6' | ||||
|   gem 'rspec-rails', '~> 3.7' | ||||
| end | ||||
|  | ||||
| group :test do | ||||
|   gem 'capybara', '~> 2.14' | ||||
|   gem 'capybara', '~> 2.15' | ||||
|   gem 'climate_control', '~> 0.2' | ||||
|   gem 'faker', '~> 1.7' | ||||
|   gem 'microformats', '~> 4.0' | ||||
| @@ -91,13 +92,13 @@ group :test do | ||||
|   gem 'rspec-sidekiq', '~> 3.0' | ||||
|   gem 'simplecov', '~> 0.14', require: false | ||||
|   gem 'webmock', '~> 3.0' | ||||
|   gem 'parallel_tests', '~> 2.14' | ||||
|   gem 'parallel_tests', '~> 2.17' | ||||
| end | ||||
|  | ||||
| group :development do | ||||
|   gem 'active_record_query_trace', '~> 1.5' | ||||
|   gem 'annotate', '~> 2.7' | ||||
|   gem 'better_errors', '~> 2.1' | ||||
|   gem 'better_errors', '~> 2.4' | ||||
|   gem 'binding_of_caller', '~> 0.7' | ||||
|   gem 'bullet', '~> 5.5' | ||||
|   gem 'letter_opener', '~> 1.4' | ||||
| @@ -105,15 +106,15 @@ group :development do | ||||
|   gem 'rubocop', require: false | ||||
|   gem 'brakeman', '~> 4.0', require: false | ||||
|   gem 'bundler-audit', '~> 0.6', require: false | ||||
|   gem 'scss_lint', '~> 0.53', require: false | ||||
|   gem 'scss_lint', '~> 0.55', require: false | ||||
|  | ||||
|   gem 'capistrano', '~> 3.8' | ||||
|   gem 'capistrano-rails', '~> 1.2' | ||||
|   gem 'capistrano', '~> 3.10' | ||||
|   gem 'capistrano-rails', '~> 1.3' | ||||
|   gem 'capistrano-rbenv', '~> 2.1' | ||||
|   gem 'capistrano-yarn', '~> 2.0' | ||||
| end | ||||
|  | ||||
| group :production do | ||||
|   gem 'lograge', '~> 0.5' | ||||
|   gem 'lograge', '~> 0.7' | ||||
|   gem 'redis-rails', '~> 5.0' | ||||
| end | ||||
|   | ||||
							
								
								
									
										267
									
								
								Gemfile.lock
									
									
									
									
									
								
							
							
						
						
									
										267
									
								
								Gemfile.lock
									
									
									
									
									
								
							| @@ -24,11 +24,11 @@ GEM | ||||
|       erubi (~> 1.4) | ||||
|       rails-dom-testing (~> 2.0) | ||||
|       rails-html-sanitizer (~> 1.0, >= 1.0.3) | ||||
|     active_model_serializers (0.10.6) | ||||
|     active_model_serializers (0.10.7) | ||||
|       actionpack (>= 4.1, < 6) | ||||
|       activemodel (>= 4.1, < 6) | ||||
|       case_transform (>= 0.2) | ||||
|       jsonapi-renderer (>= 0.1.1.beta1, < 0.2) | ||||
|       jsonapi-renderer (>= 0.1.1.beta1, < 0.3) | ||||
|     active_record_query_trace (1.5.4) | ||||
|     activejob (5.1.4) | ||||
|       activesupport (= 5.1.4) | ||||
| @@ -57,25 +57,17 @@ GEM | ||||
|       encryptor (~> 3.0.0) | ||||
|     av (0.9.0) | ||||
|       cocaine (~> 0.5.3) | ||||
|     aws-sdk (2.10.46) | ||||
|       aws-sdk-resources (= 2.10.46) | ||||
|     aws-sdk-core (2.10.46) | ||||
|       aws-sigv4 (~> 1.0) | ||||
|       jmespath (~> 1.0) | ||||
|     aws-sdk-resources (2.10.46) | ||||
|       aws-sdk-core (= 2.10.46) | ||||
|     aws-sigv4 (1.0.2) | ||||
|     bcrypt (3.1.11) | ||||
|     better_errors (2.3.0) | ||||
|     better_errors (2.4.0) | ||||
|       coderay (>= 1.0.0) | ||||
|       erubi (>= 1.0.0) | ||||
|       rack (>= 0.9.0) | ||||
|     binding_of_caller (0.7.2) | ||||
|     binding_of_caller (0.7.3) | ||||
|       debug_inspector (>= 0.0.1) | ||||
|     bootsnap (1.1.3) | ||||
|     bootsnap (1.1.5) | ||||
|       msgpack (~> 1.0) | ||||
|     brakeman (4.0.1) | ||||
|     browser (2.5.1) | ||||
|     browser (2.5.2) | ||||
|     builder (3.2.3) | ||||
|     bullet (5.6.1) | ||||
|       activesupport (>= 3.0.0) | ||||
| @@ -83,23 +75,23 @@ GEM | ||||
|     bundler-audit (0.6.0) | ||||
|       bundler (~> 1.2) | ||||
|       thor (~> 0.18) | ||||
|     capistrano (3.9.1) | ||||
|     capistrano (3.10.0) | ||||
|       airbrussh (>= 1.0.0) | ||||
|       i18n | ||||
|       rake (>= 10.0.0) | ||||
|       sshkit (>= 1.9.0) | ||||
|     capistrano-bundler (1.2.0) | ||||
|     capistrano-bundler (1.3.0) | ||||
|       capistrano (~> 3.1) | ||||
|       sshkit (~> 1.2) | ||||
|     capistrano-rails (1.3.0) | ||||
|     capistrano-rails (1.3.1) | ||||
|       capistrano (~> 3.1) | ||||
|       capistrano-bundler (~> 1.1) | ||||
|     capistrano-rbenv (2.1.1) | ||||
|     capistrano-rbenv (2.1.3) | ||||
|       capistrano (~> 3.1) | ||||
|       sshkit (~> 1.3) | ||||
|     capistrano-yarn (2.0.2) | ||||
|       capistrano (~> 3.0) | ||||
|     capybara (2.15.1) | ||||
|     capybara (2.16.1) | ||||
|       addressable | ||||
|       mini_mime (>= 0.1.3) | ||||
|       nokogiri (>= 1.3.3) | ||||
| @@ -110,7 +102,7 @@ GEM | ||||
|       activesupport | ||||
|     charlock_holmes (0.7.5) | ||||
|     chunky_png (1.3.8) | ||||
|     cld3 (3.2.0) | ||||
|     cld3 (3.2.1) | ||||
|       ffi (>= 1.1.0, < 1.10.0) | ||||
|     climate_control (0.2.0) | ||||
|     cocaine (0.5.8) | ||||
| @@ -121,7 +113,7 @@ GEM | ||||
|     connection_pool (2.2.1) | ||||
|     crack (0.4.3) | ||||
|       safe_yaml (~> 1.0.0) | ||||
|     crass (1.0.2) | ||||
|     crass (1.0.3) | ||||
|     debug_inspector (0.0.3) | ||||
|     devise (4.3.0) | ||||
|       bcrypt (~> 3.0) | ||||
| @@ -129,11 +121,11 @@ GEM | ||||
|       railties (>= 4.1.0, < 5.2) | ||||
|       responders | ||||
|       warden (~> 1.2.3) | ||||
|     devise-two-factor (3.0.0) | ||||
|       activesupport | ||||
|     devise-two-factor (3.0.2) | ||||
|       activesupport (< 5.2) | ||||
|       attr_encrypted (>= 1.3, < 4, != 2) | ||||
|       devise (~> 4.0) | ||||
|       railties | ||||
|       railties (< 5.2) | ||||
|       rotp (~> 2.0) | ||||
|     diff-lcs (1.3) | ||||
|     docile (1.1.5) | ||||
| @@ -150,16 +142,21 @@ GEM | ||||
|       thread | ||||
|       thread_safe | ||||
|     encryptor (3.0.0) | ||||
|     erubi (1.6.1) | ||||
|     et-orbi (1.0.5) | ||||
|     erubi (1.7.0) | ||||
|     et-orbi (1.0.8) | ||||
|       tzinfo | ||||
|     excon (0.59.0) | ||||
|     execjs (2.7.0) | ||||
|     fabrication (2.16.3) | ||||
|     fabrication (2.18.0) | ||||
|     faker (1.8.4) | ||||
|       i18n (~> 0.5) | ||||
|     fast_blank (1.0.0) | ||||
|     ffi (1.9.18) | ||||
|     fog-aws (1.4.1) | ||||
|       fog-core (~> 1.38) | ||||
|       fog-json (~> 1.0) | ||||
|       fog-xml (~> 0.1) | ||||
|       ipaddress (~> 0.8) | ||||
|     fog-core (1.45.0) | ||||
|       builder | ||||
|       excon (~> 0.58) | ||||
| @@ -167,22 +164,27 @@ GEM | ||||
|     fog-json (1.0.2) | ||||
|       fog-core (~> 1.0) | ||||
|       multi_json (~> 1.10) | ||||
|     fog-openstack (0.1.21) | ||||
|     fog-local (0.4.0) | ||||
|       fog-core (~> 1.27) | ||||
|     fog-openstack (0.1.22) | ||||
|       fog-core (>= 1.40) | ||||
|       fog-json (>= 1.0) | ||||
|       ipaddress (>= 0.8) | ||||
|     fog-xml (0.1.3) | ||||
|       fog-core | ||||
|       nokogiri (>= 1.5.11, < 2.0.0) | ||||
|     formatador (0.2.5) | ||||
|     fuubar (2.2.0) | ||||
|       rspec-core (~> 3.0) | ||||
|       ruby-progressbar (~> 1.4) | ||||
|     globalid (0.4.0) | ||||
|     globalid (0.4.1) | ||||
|       activesupport (>= 4.2.0) | ||||
|     goldfinger (2.0.1) | ||||
|       addressable (~> 2.5) | ||||
|       http (~> 2.2) | ||||
|       nokogiri (~> 1.8) | ||||
|       oj (~> 3.0) | ||||
|     hamlit (2.8.4) | ||||
|     hamlit (2.8.5) | ||||
|       temple (>= 0.8.0) | ||||
|       thor | ||||
|       tilt | ||||
| @@ -194,7 +196,7 @@ GEM | ||||
|     hamster (3.0.0) | ||||
|       concurrent-ruby (~> 1.0) | ||||
|     hashdiff (0.3.7) | ||||
|     highline (1.7.8) | ||||
|     highline (1.7.10) | ||||
|     hiredis (0.6.1) | ||||
|     hkdf (0.3.0) | ||||
|     htmlentities (4.3.4) | ||||
| @@ -211,8 +213,9 @@ GEM | ||||
|     httplog (0.99.7) | ||||
|       colorize | ||||
|       rack | ||||
|     i18n (0.8.6) | ||||
|     i18n-tasks (0.9.18) | ||||
|     i18n (0.9.1) | ||||
|       concurrent-ruby (~> 1.0) | ||||
|     i18n-tasks (0.9.19) | ||||
|       activesupport (>= 4.0.2) | ||||
|       ast (>= 2.1.0) | ||||
|       easy_translate (>= 0.5.0) | ||||
| @@ -225,29 +228,28 @@ GEM | ||||
|     idn-ruby (0.1.0) | ||||
|     ipaddress (0.8.3) | ||||
|     iso-639 (0.2.8) | ||||
|     jmespath (1.3.1) | ||||
|     json (2.1.0) | ||||
|     json-ld (2.1.5) | ||||
|     json-ld (2.1.7) | ||||
|       multi_json (~> 1.12) | ||||
|       rdf (~> 2.2) | ||||
|       rdf (~> 2.2, >= 2.2.8) | ||||
|     json-ld-preloaded (2.2.2) | ||||
|       json-ld (~> 2.1, >= 2.1.5) | ||||
|       multi_json (~> 1.11) | ||||
|       rdf (~> 2.2) | ||||
|     jsonapi-renderer (0.1.3) | ||||
|     jwt (1.5.6) | ||||
|     kaminari (1.0.1) | ||||
|     jsonapi-renderer (0.2.0) | ||||
|     jwt (2.1.0) | ||||
|     kaminari (1.1.1) | ||||
|       activesupport (>= 4.1.0) | ||||
|       kaminari-actionview (= 1.0.1) | ||||
|       kaminari-activerecord (= 1.0.1) | ||||
|       kaminari-core (= 1.0.1) | ||||
|     kaminari-actionview (1.0.1) | ||||
|       kaminari-actionview (= 1.1.1) | ||||
|       kaminari-activerecord (= 1.1.1) | ||||
|       kaminari-core (= 1.1.1) | ||||
|     kaminari-actionview (1.1.1) | ||||
|       actionview | ||||
|       kaminari-core (= 1.0.1) | ||||
|     kaminari-activerecord (1.0.1) | ||||
|       kaminari-core (= 1.1.1) | ||||
|     kaminari-activerecord (1.1.1) | ||||
|       activerecord | ||||
|       kaminari-core (= 1.0.1) | ||||
|     kaminari-core (1.0.1) | ||||
|       kaminari-core (= 1.1.1) | ||||
|     kaminari-core (1.1.1) | ||||
|     launchy (2.4.3) | ||||
|       addressable (~> 2.3) | ||||
|     letter_opener (1.4.1) | ||||
| @@ -257,18 +259,19 @@ GEM | ||||
|       letter_opener (~> 1.0) | ||||
|       railties (>= 3.2) | ||||
|     link_header (0.0.8) | ||||
|     lograge (0.6.0) | ||||
|     lograge (0.7.1) | ||||
|       actionpack (>= 4, < 5.2) | ||||
|       activesupport (>= 4, < 5.2) | ||||
|       railties (>= 4, < 5.2) | ||||
|       request_store (~> 1.0) | ||||
|     loofah (2.0.3) | ||||
|     loofah (2.1.1) | ||||
|       crass (~> 1.0.2) | ||||
|       nokogiri (>= 1.5.9) | ||||
|     mail (2.6.6) | ||||
|       mime-types (>= 1.16, < 4) | ||||
|     mail (2.7.0) | ||||
|       mini_mime (>= 0.1.1) | ||||
|     mario-redis-lock (1.2.0) | ||||
|       redis (~> 3, >= 3.0.5) | ||||
|     method_source (0.8.2) | ||||
|     method_source (0.9.0) | ||||
|     microformats (4.0.7) | ||||
|       json | ||||
|       nokogiri | ||||
| @@ -276,8 +279,8 @@ GEM | ||||
|       mime-types-data (~> 3.2015) | ||||
|     mime-types-data (3.2016.0521) | ||||
|     mimemagic (0.3.2) | ||||
|     mini_mime (0.1.4) | ||||
|     mini_portile2 (2.2.0) | ||||
|     mini_mime (1.0.0) | ||||
|     mini_portile2 (2.3.0) | ||||
|     minitest (5.10.3) | ||||
|     msgpack (1.1.0) | ||||
|     multi_json (1.12.2) | ||||
| @@ -285,8 +288,8 @@ GEM | ||||
|       net-ssh (>= 2.6.5) | ||||
|     net-ssh (4.2.0) | ||||
|     nio4r (2.1.0) | ||||
|     nokogiri (1.8.0) | ||||
|       mini_portile2 (~> 2.2.0) | ||||
|     nokogiri (1.8.1) | ||||
|       mini_portile2 (~> 2.3.0) | ||||
|     nokogumbo (1.4.13) | ||||
|       nokogiri | ||||
|     nsa (0.2.4) | ||||
| @@ -294,15 +297,15 @@ GEM | ||||
|       concurrent-ruby (~> 1.0.0) | ||||
|       sidekiq (>= 3.5.0) | ||||
|       statsd-ruby (~> 1.2.0) | ||||
|     oj (3.3.5) | ||||
|     openssl (2.0.5) | ||||
|     oj (3.3.9) | ||||
|     openssl (2.0.6) | ||||
|     orm_adapter (0.5.0) | ||||
|     ostatus2 (2.0.1) | ||||
|       addressable (~> 2.4) | ||||
|       http (~> 2.0) | ||||
|       nokogiri (~> 1.6) | ||||
|       openssl (~> 2.0) | ||||
|     ox (2.6.0) | ||||
|     ox (2.8.2) | ||||
|     paperclip (5.1.0) | ||||
|       activemodel (>= 4.2.0) | ||||
|       activesupport (>= 4.2.0) | ||||
| @@ -313,27 +316,24 @@ GEM | ||||
|       av (~> 0.9.0) | ||||
|       paperclip (>= 2.5.2) | ||||
|     parallel (1.12.0) | ||||
|     parallel_tests (2.15.0) | ||||
|     parallel_tests (2.19.0) | ||||
|       parallel | ||||
|     parser (2.4.0.0) | ||||
|       ast (~> 2.2) | ||||
|     parser (2.4.0.2) | ||||
|       ast (~> 2.3) | ||||
|     pg (0.21.0) | ||||
|     pghero (1.7.0) | ||||
|       activerecord | ||||
|     pkg-config (1.2.7) | ||||
|     pkg-config (1.2.8) | ||||
|     powerpack (0.1.1) | ||||
|     pry (0.10.4) | ||||
|     pry (0.11.3) | ||||
|       coderay (~> 1.1.0) | ||||
|       method_source (~> 0.8.1) | ||||
|       slop (~> 3.4) | ||||
|       method_source (~> 0.9.0) | ||||
|     pry-rails (0.3.6) | ||||
|       pry (>= 0.10.4) | ||||
|     public_suffix (3.0.0) | ||||
|     puma (3.10.0) | ||||
|     public_suffix (3.0.1) | ||||
|     puma (3.11.0) | ||||
|     pundit (1.1.0) | ||||
|       activesupport (>= 3.0.0) | ||||
|     rabl (0.13.1) | ||||
|       activesupport (>= 2.3.14) | ||||
|     rack (2.0.3) | ||||
|     rack-attack (5.0.1) | ||||
|       rack | ||||
| @@ -342,7 +342,7 @@ GEM | ||||
|       rack | ||||
|     rack-proxy (0.6.2) | ||||
|       rack | ||||
|     rack-test (0.7.0) | ||||
|     rack-test (0.8.2) | ||||
|       rack (>= 1.0, < 3) | ||||
|     rack-timeout (0.4.2) | ||||
|     rails (5.1.4) | ||||
| @@ -379,31 +379,34 @@ GEM | ||||
|       thor (>= 0.18.1, < 2.0) | ||||
|     rainbow (2.2.2) | ||||
|       rake | ||||
|     rake (12.1.0) | ||||
|     rdf (2.2.9) | ||||
|     rake (12.3.0) | ||||
|     rb-fsevent (0.10.2) | ||||
|     rb-inotify (0.9.10) | ||||
|       ffi (>= 0.5.0, < 2) | ||||
|     rdf (2.2.12) | ||||
|       hamster (~> 3.0) | ||||
|       link_header (~> 0.0, >= 0.0.8) | ||||
|     rdf-normalize (0.3.2) | ||||
|       rdf (~> 2.0) | ||||
|     redis (3.3.3) | ||||
|     redis-actionpack (5.0.1) | ||||
|     redis (3.3.5) | ||||
|     redis-actionpack (5.0.2) | ||||
|       actionpack (>= 4.0, < 6) | ||||
|       redis-rack (>= 1, < 3) | ||||
|       redis-store (>= 1.1.0, < 1.4.0) | ||||
|     redis-activesupport (5.0.3) | ||||
|       redis-store (>= 1.1.0, < 2) | ||||
|     redis-activesupport (5.0.4) | ||||
|       activesupport (>= 3, < 6) | ||||
|       redis-store (~> 1.3.0) | ||||
|     redis-namespace (1.5.3) | ||||
|       redis (~> 3.0, >= 3.0.4) | ||||
|     redis-rack (2.0.2) | ||||
|       redis-store (>= 1.3, < 2) | ||||
|     redis-namespace (1.6.0) | ||||
|       redis (>= 3.0.4) | ||||
|     redis-rack (2.0.3) | ||||
|       rack (>= 1.5, < 3) | ||||
|       redis-store (>= 1.2, < 1.4) | ||||
|       redis-store (>= 1.2, < 2) | ||||
|     redis-rails (5.0.2) | ||||
|       redis-actionpack (>= 5.0, < 6) | ||||
|       redis-activesupport (>= 5.0, < 6) | ||||
|       redis-store (>= 1.2, < 2) | ||||
|     redis-store (1.3.0) | ||||
|       redis (>= 2.2) | ||||
|     redis-store (1.4.1) | ||||
|       redis (>= 2.2, < 5) | ||||
|     request_store (1.3.2) | ||||
|     responders (2.4.0) | ||||
|       actionpack (>= 4.2.0, < 5.3) | ||||
| @@ -411,27 +414,27 @@ GEM | ||||
|     rotp (2.1.2) | ||||
|     rqrcode (0.10.1) | ||||
|       chunky_png (~> 1.0) | ||||
|     rspec-core (3.6.0) | ||||
|       rspec-support (~> 3.6.0) | ||||
|     rspec-expectations (3.6.0) | ||||
|     rspec-core (3.7.0) | ||||
|       rspec-support (~> 3.7.0) | ||||
|     rspec-expectations (3.7.0) | ||||
|       diff-lcs (>= 1.2.0, < 2.0) | ||||
|       rspec-support (~> 3.6.0) | ||||
|     rspec-mocks (3.6.0) | ||||
|       rspec-support (~> 3.7.0) | ||||
|     rspec-mocks (3.7.0) | ||||
|       diff-lcs (>= 1.2.0, < 2.0) | ||||
|       rspec-support (~> 3.6.0) | ||||
|     rspec-rails (3.6.1) | ||||
|       rspec-support (~> 3.7.0) | ||||
|     rspec-rails (3.7.2) | ||||
|       actionpack (>= 3.0) | ||||
|       activesupport (>= 3.0) | ||||
|       railties (>= 3.0) | ||||
|       rspec-core (~> 3.6.0) | ||||
|       rspec-expectations (~> 3.6.0) | ||||
|       rspec-mocks (~> 3.6.0) | ||||
|       rspec-support (~> 3.6.0) | ||||
|       rspec-core (~> 3.7.0) | ||||
|       rspec-expectations (~> 3.7.0) | ||||
|       rspec-mocks (~> 3.7.0) | ||||
|       rspec-support (~> 3.7.0) | ||||
|     rspec-sidekiq (3.0.3) | ||||
|       rspec-core (~> 3.0, >= 3.0.0) | ||||
|       sidekiq (>= 2.4.0) | ||||
|     rspec-support (3.6.0) | ||||
|     rubocop (0.50.0) | ||||
|     rspec-support (3.7.0) | ||||
|     rubocop (0.51.0) | ||||
|       parallel (~> 1.10) | ||||
|       parser (>= 2.3.3.1, < 3.0) | ||||
|       powerpack (~> 0.1) | ||||
| @@ -439,7 +442,7 @@ GEM | ||||
|       ruby-progressbar (~> 1.7) | ||||
|       unicode-display_width (~> 1.0, >= 1.0.1) | ||||
|     ruby-oembed (0.12.0) | ||||
|     ruby-progressbar (1.8.3) | ||||
|     ruby-progressbar (1.9.0) | ||||
|     rufus-scheduler (3.4.2) | ||||
|       et-orbi (~> 1.0) | ||||
|     safe_yaml (1.0.4) | ||||
| @@ -447,20 +450,24 @@ GEM | ||||
|       crass (~> 1.0.2) | ||||
|       nokogiri (>= 1.4.4) | ||||
|       nokogumbo (~> 1.4.1) | ||||
|     sass (3.4.25) | ||||
|     scss_lint (0.54.0) | ||||
|     sass (3.5.3) | ||||
|       sass-listen (~> 4.0.0) | ||||
|     sass-listen (4.0.0) | ||||
|       rb-fsevent (~> 0.9, >= 0.9.4) | ||||
|       rb-inotify (~> 0.9, >= 0.9.7) | ||||
|     scss_lint (0.56.0) | ||||
|       rake (>= 0.9, < 13) | ||||
|       sass (~> 3.4.20) | ||||
|     sidekiq (5.0.4) | ||||
|       sass (~> 3.5.3) | ||||
|     sidekiq (5.0.5) | ||||
|       concurrent-ruby (~> 1.0) | ||||
|       connection_pool (~> 2.2, >= 2.2.0) | ||||
|       rack-protection (>= 1.5.0) | ||||
|       redis (~> 3.3, >= 3.3.3) | ||||
|       redis (>= 3.3.4, < 5) | ||||
|     sidekiq-bulk (0.1.1) | ||||
|       activesupport | ||||
|       sidekiq | ||||
|     sidekiq-scheduler (2.1.9) | ||||
|       redis (~> 3) | ||||
|     sidekiq-scheduler (2.1.10) | ||||
|       redis (>= 3, < 5) | ||||
|       rufus-scheduler (~> 3.2) | ||||
|       sidekiq (>= 3) | ||||
|       tilt (>= 1.4.0) | ||||
| @@ -477,7 +484,6 @@ GEM | ||||
|       json (>= 1.8, < 3) | ||||
|       simplecov-html (~> 0.10.0) | ||||
|     simplecov-html (0.10.2) | ||||
|     slop (3.6.0) | ||||
|     sprockets (3.7.1) | ||||
|       concurrent-ruby (~> 1.0) | ||||
|       rack (> 1, < 3) | ||||
| @@ -485,7 +491,7 @@ GEM | ||||
|       actionpack (>= 4.0) | ||||
|       activesupport (>= 4.0) | ||||
|       sprockets (>= 3.0.0) | ||||
|     sshkit (1.14.0) | ||||
|     sshkit (1.15.1) | ||||
|       net-scp (>= 1.1.2) | ||||
|       net-ssh (>= 2.8.0) | ||||
|     statsd-ruby (1.2.1) | ||||
| @@ -500,9 +506,9 @@ GEM | ||||
|     tilt (2.0.8) | ||||
|     twitter-text (1.14.7) | ||||
|       unf (~> 0.1.0) | ||||
|     tzinfo (1.2.3) | ||||
|     tzinfo (1.2.4) | ||||
|       thread_safe (~> 0.1) | ||||
|     tzinfo-data (1.2017.2) | ||||
|     tzinfo-data (1.2017.3) | ||||
|       tzinfo (>= 1.0.0) | ||||
|     uglifier (3.2.0) | ||||
|       execjs (>= 0.3.0, < 3) | ||||
| @@ -513,20 +519,20 @@ GEM | ||||
|     uniform_notifier (1.10.0) | ||||
|     warden (1.2.7) | ||||
|       rack (>= 1.0) | ||||
|     webmock (3.1.0) | ||||
|     webmock (3.1.1) | ||||
|       addressable (>= 2.3.6) | ||||
|       crack (>= 0.3.2) | ||||
|       hashdiff | ||||
|     webpacker (3.0.1) | ||||
|     webpacker (3.0.2) | ||||
|       activesupport (>= 4.2) | ||||
|       rack-proxy (>= 0.6.1) | ||||
|       railties (>= 4.2) | ||||
|     webpush (0.3.2) | ||||
|     webpush (0.3.3) | ||||
|       hkdf (~> 0.2) | ||||
|       jwt | ||||
|       jwt (~> 2.0) | ||||
|     websocket-driver (0.6.5) | ||||
|       websocket-extensions (>= 0.1.0) | ||||
|     websocket-extensions (0.1.2) | ||||
|     websocket-extensions (0.1.3) | ||||
|     xpath (2.1.0) | ||||
|       nokogiri (~> 1.3) | ||||
|  | ||||
| @@ -538,19 +544,18 @@ DEPENDENCIES | ||||
|   active_record_query_trace (~> 1.5) | ||||
|   addressable (~> 2.5) | ||||
|   annotate (~> 2.7) | ||||
|   aws-sdk (~> 2.9) | ||||
|   better_errors (~> 2.1) | ||||
|   better_errors (~> 2.4) | ||||
|   binding_of_caller (~> 0.7) | ||||
|   bootsnap | ||||
|   brakeman (~> 4.0) | ||||
|   browser | ||||
|   bullet (~> 5.5) | ||||
|   bundler-audit (~> 0.6) | ||||
|   capistrano (~> 3.8) | ||||
|   capistrano-rails (~> 1.2) | ||||
|   capistrano (~> 3.10) | ||||
|   capistrano-rails (~> 1.3) | ||||
|   capistrano-rbenv (~> 2.1) | ||||
|   capistrano-yarn (~> 2.0) | ||||
|   capybara (~> 2.14) | ||||
|   capybara (~> 2.15) | ||||
|   charlock_holmes (~> 0.7.5) | ||||
|   cld3 (~> 3.2.0) | ||||
|   climate_control (~> 0.2) | ||||
| @@ -558,9 +563,12 @@ DEPENDENCIES | ||||
|   devise-two-factor (~> 3.0) | ||||
|   doorkeeper (~> 4.2) | ||||
|   dotenv-rails (~> 2.2) | ||||
|   fabrication (~> 2.16) | ||||
|   fabrication (~> 2.18) | ||||
|   faker (~> 1.7) | ||||
|   fast_blank (~> 1.0) | ||||
|   fog-aws (~> 1.4) | ||||
|   fog-core (~> 1.45) | ||||
|   fog-local (~> 0.4) | ||||
|   fog-openstack (~> 0.1) | ||||
|   fuubar (~> 2.2) | ||||
|   goldfinger (~> 2.0) | ||||
| @@ -574,29 +582,28 @@ DEPENDENCIES | ||||
|   idn-ruby | ||||
|   iso-639 | ||||
|   json-ld-preloaded (~> 2.2.1) | ||||
|   kaminari (~> 1.0) | ||||
|   kaminari (~> 1.1) | ||||
|   letter_opener (~> 1.4) | ||||
|   letter_opener_web (~> 1.3) | ||||
|   link_header (~> 0.0) | ||||
|   lograge (~> 0.5) | ||||
|   lograge (~> 0.7) | ||||
|   mario-redis-lock (~> 1.2) | ||||
|   microformats (~> 4.0) | ||||
|   mime-types (~> 3.1) | ||||
|   nokogiri (~> 1.7) | ||||
|   nokogiri (~> 1.8) | ||||
|   nsa (~> 0.2) | ||||
|   oj (~> 3.0) | ||||
|   oj (~> 3.3) | ||||
|   ostatus2 (~> 2.0) | ||||
|   ox (~> 2.5) | ||||
|   ox (~> 2.8) | ||||
|   paperclip (~> 5.1) | ||||
|   paperclip-av-transcoder (~> 0.6) | ||||
|   parallel_tests (~> 2.14) | ||||
|   parallel_tests (~> 2.17) | ||||
|   pg (~> 0.20) | ||||
|   pghero (~> 1.7) | ||||
|   pkg-config (~> 1.2) | ||||
|   pry-rails (~> 0.3) | ||||
|   puma (~> 3.10) | ||||
|   pundit (~> 1.1) | ||||
|   rabl (~> 0.13) | ||||
|   rack-attack (~> 5.0) | ||||
|   rack-cors (~> 0.4) | ||||
|   rack-timeout (~> 0.4) | ||||
| @@ -609,12 +616,12 @@ DEPENDENCIES | ||||
|   redis-namespace (~> 1.5) | ||||
|   redis-rails (~> 5.0) | ||||
|   rqrcode (~> 0.10) | ||||
|   rspec-rails (~> 3.6) | ||||
|   rspec-rails (~> 3.7) | ||||
|   rspec-sidekiq (~> 3.0) | ||||
|   rubocop | ||||
|   ruby-oembed (~> 0.12) | ||||
|   sanitize (~> 4.4) | ||||
|   scss_lint (~> 0.53) | ||||
|   scss_lint (~> 0.55) | ||||
|   sidekiq (~> 5.0) | ||||
|   sidekiq-bulk (~> 0.1.1) | ||||
|   sidekiq-scheduler (~> 2.1) | ||||
| @@ -635,4 +642,4 @@ RUBY VERSION | ||||
|    ruby 2.4.2p198 | ||||
|  | ||||
| BUNDLED WITH | ||||
|    1.15.4 | ||||
|    1.16.0 | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
|  | ||||
| ======== | ||||
|  | ||||
| [][travis] | ||||
| [][code_climate] | ||||
| [][travis] | ||||
| [][code_climate] | ||||
|  | ||||
| [travis]: https://travis-ci.org/tootsuite/mastodon | ||||
| [code_climate]: https://codeclimate.com/github/tootsuite/mastodon | ||||
|   | ||||
| @@ -1,23 +1,28 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class Admin::AccountModerationNotesController < Admin::BaseController | ||||
| module Admin | ||||
|   class AccountModerationNotesController < BaseController | ||||
|     before_action :set_account_moderation_note, only: [:destroy] | ||||
|  | ||||
|     def create | ||||
|       authorize AccountModerationNote, :create? | ||||
|  | ||||
|       @account_moderation_note = current_account.account_moderation_notes.new(resource_params) | ||||
|  | ||||
|       if @account_moderation_note.save | ||||
|       @target_account = @account_moderation_note.target_account | ||||
|       redirect_to admin_account_path(@target_account.id), notice: I18n.t('admin.account_moderation_notes.created_msg') | ||||
|         redirect_to admin_account_path(@account_moderation_note.target_account_id), notice: I18n.t('admin.account_moderation_notes.created_msg') | ||||
|       else | ||||
|         @account          = @account_moderation_note.target_account | ||||
|         @moderation_notes = @account.targeted_moderation_notes.latest | ||||
|  | ||||
|         render template: 'admin/accounts/show' | ||||
|       end | ||||
|     end | ||||
|  | ||||
|     def destroy | ||||
|     @account_moderation_note = AccountModerationNote.find(params[:id]) | ||||
|     @target_account = @account_moderation_note.target_account | ||||
|     @account_moderation_note.destroy | ||||
|     redirect_to admin_account_path(@target_account.id), notice: I18n.t('admin.account_moderation_notes.destroyed_msg') | ||||
|       authorize @account_moderation_note, :destroy? | ||||
|       @account_moderation_note.destroy! | ||||
|       redirect_to admin_account_path(@account_moderation_note.target_account_id), notice: I18n.t('admin.account_moderation_notes.destroyed_msg') | ||||
|     end | ||||
|  | ||||
|     private | ||||
| @@ -28,4 +33,9 @@ class Admin::AccountModerationNotesController < Admin::BaseController | ||||
|         :target_account_id | ||||
|       ) | ||||
|     end | ||||
|  | ||||
|     def set_account_moderation_note | ||||
|       @account_moderation_note = AccountModerationNote.find(params[:id]) | ||||
|     end | ||||
|   end | ||||
| end | ||||
|   | ||||
| @@ -2,29 +2,57 @@ | ||||
|  | ||||
| module Admin | ||||
|   class AccountsController < BaseController | ||||
|     before_action :set_account, only: [:show, :subscribe, :unsubscribe, :redownload] | ||||
|     before_action :set_account, only: [:show, :subscribe, :unsubscribe, :redownload, :enable, :disable, :memorialize] | ||||
|     before_action :require_remote_account!, only: [:subscribe, :unsubscribe, :redownload] | ||||
|     before_action :require_local_account!, only: [:enable, :disable, :memorialize] | ||||
|  | ||||
|     def index | ||||
|       authorize :account, :index? | ||||
|       @accounts = filtered_accounts.page(params[:page]) | ||||
|     end | ||||
|  | ||||
|     def show | ||||
|       authorize @account, :show? | ||||
|       @account_moderation_note = current_account.account_moderation_notes.new(target_account: @account) | ||||
|       @moderation_notes = @account.targeted_moderation_notes.latest | ||||
|     end | ||||
|  | ||||
|     def subscribe | ||||
|       authorize @account, :subscribe? | ||||
|       Pubsubhubbub::SubscribeWorker.perform_async(@account.id) | ||||
|       redirect_to admin_account_path(@account.id) | ||||
|     end | ||||
|  | ||||
|     def unsubscribe | ||||
|       authorize @account, :unsubscribe? | ||||
|       Pubsubhubbub::UnsubscribeWorker.perform_async(@account.id) | ||||
|       redirect_to admin_account_path(@account.id) | ||||
|     end | ||||
|  | ||||
|     def memorialize | ||||
|       authorize @account, :memorialize? | ||||
|       @account.memorialize! | ||||
|       log_action :memorialize, @account | ||||
|       redirect_to admin_account_path(@account.id) | ||||
|     end | ||||
|  | ||||
|     def enable | ||||
|       authorize @account.user, :enable? | ||||
|       @account.user.enable! | ||||
|       log_action :enable, @account.user | ||||
|       redirect_to admin_account_path(@account.id) | ||||
|     end | ||||
|  | ||||
|     def disable | ||||
|       authorize @account.user, :disable? | ||||
|       @account.user.disable! | ||||
|       log_action :disable, @account.user | ||||
|       redirect_to admin_account_path(@account.id) | ||||
|     end | ||||
|  | ||||
|     def redownload | ||||
|       authorize @account, :redownload? | ||||
|  | ||||
|       @account.reset_avatar! | ||||
|       @account.reset_header! | ||||
|       @account.save! | ||||
| @@ -42,6 +70,10 @@ module Admin | ||||
|       redirect_to admin_account_path(@account.id) if @account.local? | ||||
|     end | ||||
|  | ||||
|     def require_local_account! | ||||
|       redirect_to admin_account_path(@account.id) unless @account.local? && @account.user.present? | ||||
|     end | ||||
|  | ||||
|     def filtered_accounts | ||||
|       AccountFilter.new(filter_params).results | ||||
|     end | ||||
|   | ||||
							
								
								
									
										9
									
								
								app/controllers/admin/action_logs_controller.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								app/controllers/admin/action_logs_controller.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| module Admin | ||||
|   class ActionLogsController < BaseController | ||||
|     def index | ||||
|       @action_logs = Admin::ActionLog.page(params[:page]) | ||||
|     end | ||||
|   end | ||||
| end | ||||
| @@ -2,7 +2,10 @@ | ||||
|  | ||||
| module Admin | ||||
|   class BaseController < ApplicationController | ||||
|     before_action :require_admin! | ||||
|     include Authorization | ||||
|     include AccountableConcern | ||||
|  | ||||
|     before_action :require_staff! | ||||
|  | ||||
|     layout 'admin' | ||||
|   end | ||||
|   | ||||
| @@ -2,15 +2,19 @@ | ||||
|  | ||||
| module Admin | ||||
|   class ConfirmationsController < BaseController | ||||
|     before_action :set_user | ||||
|  | ||||
|     def create | ||||
|       account_user.confirm | ||||
|       authorize @user, :confirm? | ||||
|       @user.confirm! | ||||
|       log_action :confirm, @user | ||||
|       redirect_to admin_accounts_path | ||||
|     end | ||||
|  | ||||
|     private | ||||
|  | ||||
|     def account_user | ||||
|       Account.find(params[:account_id]).user || raise(ActiveRecord::RecordNotFound) | ||||
|     def set_user | ||||
|       @user = Account.find(params[:account_id]).user || raise(ActiveRecord::RecordNotFound) | ||||
|     end | ||||
|   end | ||||
| end | ||||
|   | ||||
| @@ -5,47 +5,73 @@ module Admin | ||||
|     before_action :set_custom_emoji, except: [:index, :new, :create] | ||||
|  | ||||
|     def index | ||||
|       @custom_emojis = filtered_custom_emojis.page(params[:page]) | ||||
|       authorize :custom_emoji, :index? | ||||
|       @custom_emojis = filtered_custom_emojis.eager_load(:local_counterpart).page(params[:page]) | ||||
|     end | ||||
|  | ||||
|     def new | ||||
|       authorize :custom_emoji, :create? | ||||
|       @custom_emoji = CustomEmoji.new | ||||
|     end | ||||
|  | ||||
|     def create | ||||
|       authorize :custom_emoji, :create? | ||||
|  | ||||
|       @custom_emoji = CustomEmoji.new(resource_params) | ||||
|  | ||||
|       if @custom_emoji.save | ||||
|         log_action :create, @custom_emoji | ||||
|         redirect_to admin_custom_emojis_path, notice: I18n.t('admin.custom_emojis.created_msg') | ||||
|       else | ||||
|         render :new | ||||
|       end | ||||
|     end | ||||
|  | ||||
|     def update | ||||
|       authorize @custom_emoji, :update? | ||||
|  | ||||
|       if @custom_emoji.update(resource_params) | ||||
|         log_action :update, @custom_emoji | ||||
|         redirect_to admin_custom_emojis_path, notice: I18n.t('admin.custom_emojis.updated_msg') | ||||
|       else | ||||
|         redirect_to admin_custom_emojis_path, notice: I18n.t('admin.custom_emojis.update_failed_msg') | ||||
|       end | ||||
|     end | ||||
|  | ||||
|     def destroy | ||||
|       @custom_emoji.destroy | ||||
|       authorize @custom_emoji, :destroy? | ||||
|       @custom_emoji.destroy! | ||||
|       log_action :destroy, @custom_emoji | ||||
|       redirect_to admin_custom_emojis_path, notice: I18n.t('admin.custom_emojis.destroyed_msg') | ||||
|     end | ||||
|  | ||||
|     def copy | ||||
|       emoji = CustomEmoji.new(domain: nil, shortcode: @custom_emoji.shortcode, image: @custom_emoji.image) | ||||
|       authorize @custom_emoji, :copy? | ||||
|  | ||||
|       emoji = CustomEmoji.find_or_initialize_by(domain: nil, shortcode: @custom_emoji.shortcode) | ||||
|       emoji.image = @custom_emoji.image | ||||
|  | ||||
|       if emoji.save | ||||
|         log_action :create, emoji | ||||
|         flash[:notice] = I18n.t('admin.custom_emojis.copied_msg') | ||||
|       else | ||||
|         flash[:alert] = I18n.t('admin.custom_emojis.copy_failed_msg') | ||||
|       end | ||||
|  | ||||
|       redirect_to admin_custom_emojis_path(params[:page]) | ||||
|       redirect_to admin_custom_emojis_path(page: params[:page]) | ||||
|     end | ||||
|  | ||||
|     def enable | ||||
|       authorize @custom_emoji, :enable? | ||||
|       @custom_emoji.update!(disabled: false) | ||||
|       log_action :enable, @custom_emoji | ||||
|       redirect_to admin_custom_emojis_path, notice: I18n.t('admin.custom_emojis.enabled_msg') | ||||
|     end | ||||
|  | ||||
|     def disable | ||||
|       authorize @custom_emoji, :disable? | ||||
|       @custom_emoji.update!(disabled: true) | ||||
|       log_action :disable, @custom_emoji | ||||
|       redirect_to admin_custom_emojis_path, notice: I18n.t('admin.custom_emojis.disabled_msg') | ||||
|     end | ||||
|  | ||||
| @@ -56,7 +82,7 @@ module Admin | ||||
|     end | ||||
|  | ||||
|     def resource_params | ||||
|       params.require(:custom_emoji).permit(:shortcode, :image) | ||||
|       params.require(:custom_emoji).permit(:shortcode, :image, :visible_in_picker) | ||||
|     end | ||||
|  | ||||
|     def filtered_custom_emojis | ||||
|   | ||||
| @@ -5,28 +5,37 @@ module Admin | ||||
|     before_action :set_domain_block, only: [:show, :destroy] | ||||
|  | ||||
|     def index | ||||
|       authorize :domain_block, :index? | ||||
|       @domain_blocks = DomainBlock.page(params[:page]) | ||||
|     end | ||||
|  | ||||
|     def new | ||||
|       authorize :domain_block, :create? | ||||
|       @domain_block = DomainBlock.new | ||||
|     end | ||||
|  | ||||
|     def create | ||||
|       authorize :domain_block, :create? | ||||
|  | ||||
|       @domain_block = DomainBlock.new(resource_params) | ||||
|  | ||||
|       if @domain_block.save | ||||
|         DomainBlockWorker.perform_async(@domain_block.id) | ||||
|         log_action :create, @domain_block | ||||
|         redirect_to admin_domain_blocks_path, notice: I18n.t('admin.domain_blocks.created_msg') | ||||
|       else | ||||
|         render :new | ||||
|       end | ||||
|     end | ||||
|  | ||||
|     def show; end | ||||
|     def show | ||||
|       authorize @domain_block, :show? | ||||
|     end | ||||
|  | ||||
|     def destroy | ||||
|       authorize @domain_block, :destroy? | ||||
|       UnblockDomainService.new.call(@domain_block, retroactive_unblock?) | ||||
|       log_action :destroy, @domain_block | ||||
|       redirect_to admin_domain_blocks_path, notice: I18n.t('admin.domain_blocks.destroyed_msg') | ||||
|     end | ||||
|  | ||||
|   | ||||
| @@ -5,17 +5,22 @@ module Admin | ||||
|     before_action :set_email_domain_block, only: [:show, :destroy] | ||||
|  | ||||
|     def index | ||||
|       authorize :email_domain_block, :index? | ||||
|       @email_domain_blocks = EmailDomainBlock.page(params[:page]) | ||||
|     end | ||||
|  | ||||
|     def new | ||||
|       authorize :email_domain_block, :create? | ||||
|       @email_domain_block = EmailDomainBlock.new | ||||
|     end | ||||
|  | ||||
|     def create | ||||
|       authorize :email_domain_block, :create? | ||||
|  | ||||
|       @email_domain_block = EmailDomainBlock.new(resource_params) | ||||
|  | ||||
|       if @email_domain_block.save | ||||
|         log_action :create, @email_domain_block | ||||
|         redirect_to admin_email_domain_blocks_path, notice: I18n.t('admin.email_domain_blocks.created_msg') | ||||
|       else | ||||
|         render :new | ||||
| @@ -23,7 +28,9 @@ module Admin | ||||
|     end | ||||
|  | ||||
|     def destroy | ||||
|       @email_domain_block.destroy | ||||
|       authorize @email_domain_block, :destroy? | ||||
|       @email_domain_block.destroy! | ||||
|       log_action :destroy, @email_domain_block | ||||
|       redirect_to admin_email_domain_blocks_path, notice: I18n.t('admin.email_domain_blocks.destroyed_msg') | ||||
|     end | ||||
|  | ||||
|   | ||||
| @@ -3,10 +3,12 @@ | ||||
| module Admin | ||||
|   class InstancesController < BaseController | ||||
|     def index | ||||
|       authorize :instance, :index? | ||||
|       @instances = ordered_instances | ||||
|     end | ||||
|  | ||||
|     def resubscribe | ||||
|       authorize :instance, :resubscribe? | ||||
|       params.require(:by_domain) | ||||
|       Pubsubhubbub::SubscribeWorker.push_bulk(subscribeable_accounts.pluck(:id)) | ||||
|       redirect_to admin_instances_path | ||||
|   | ||||
							
								
								
									
										47
									
								
								app/controllers/admin/invites_controller.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								app/controllers/admin/invites_controller.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| module Admin | ||||
|   class InvitesController < BaseController | ||||
|     def index | ||||
|       authorize :invite, :index? | ||||
|  | ||||
|       @invites = filtered_invites.includes(user: :account).page(params[:page]) | ||||
|       @invite  = Invite.new | ||||
|     end | ||||
|  | ||||
|     def create | ||||
|       authorize :invite, :create? | ||||
|  | ||||
|       @invite      = Invite.new(resource_params) | ||||
|       @invite.user = current_user | ||||
|  | ||||
|       if @invite.save | ||||
|         redirect_to admin_invites_path | ||||
|       else | ||||
|         @invites = Invite.page(params[:page]) | ||||
|         render :index | ||||
|       end | ||||
|     end | ||||
|  | ||||
|     def destroy | ||||
|       @invite = Invite.find(params[:id]) | ||||
|       authorize @invite, :destroy? | ||||
|       @invite.expire! | ||||
|       redirect_to admin_invites_path | ||||
|     end | ||||
|  | ||||
|     private | ||||
|  | ||||
|     def resource_params | ||||
|       params.require(:invite).permit(:max_uses, :expires_in) | ||||
|     end | ||||
|  | ||||
|     def filtered_invites | ||||
|       InviteFilter.new(filter_params).results | ||||
|     end | ||||
|  | ||||
|     def filter_params | ||||
|       params.permit(:available, :expired) | ||||
|     end | ||||
|   end | ||||
| end | ||||
| @@ -2,26 +2,29 @@ | ||||
|  | ||||
| module Admin | ||||
|   class ReportedStatusesController < BaseController | ||||
|     include Authorization | ||||
|  | ||||
|     before_action :set_report | ||||
|     before_action :set_status, only: [:update, :destroy] | ||||
|  | ||||
|     def create | ||||
|       @form = Form::StatusBatch.new(form_status_batch_params) | ||||
|       flash[:alert] = t('admin.statuses.failed_to_execute') unless @form.save | ||||
|       authorize :status, :update? | ||||
|  | ||||
|       @form         = Form::StatusBatch.new(form_status_batch_params.merge(current_account: current_account)) | ||||
|       flash[:alert] = I18n.t('admin.statuses.failed_to_execute') unless @form.save | ||||
|  | ||||
|       redirect_to admin_report_path(@report) | ||||
|     end | ||||
|  | ||||
|     def update | ||||
|       @status.update(status_params) | ||||
|       authorize @status, :update? | ||||
|       @status.update!(status_params) | ||||
|       log_action :update, @status | ||||
|       redirect_to admin_report_path(@report) | ||||
|     end | ||||
|  | ||||
|     def destroy | ||||
|       authorize @status, :destroy? | ||||
|       RemovalWorker.perform_async(@status.id) | ||||
|       log_action :destroy, @status | ||||
|       render json: @status | ||||
|     end | ||||
|  | ||||
|   | ||||
| @@ -5,14 +5,17 @@ module Admin | ||||
|     before_action :set_report, except: [:index] | ||||
|  | ||||
|     def index | ||||
|       authorize :report, :index? | ||||
|       @reports = filtered_reports.page(params[:page]) | ||||
|     end | ||||
|  | ||||
|     def show | ||||
|       authorize @report, :show? | ||||
|       @form = Form::StatusBatch.new | ||||
|     end | ||||
|  | ||||
|     def update | ||||
|       authorize @report, :update? | ||||
|       process_report | ||||
|       redirect_to admin_report_path(@report) | ||||
|     end | ||||
| @@ -22,12 +25,17 @@ module Admin | ||||
|     def process_report | ||||
|       case params[:outcome].to_s | ||||
|       when 'resolve' | ||||
|         @report.update(action_taken_by_current_attributes) | ||||
|         @report.update!(action_taken_by_current_attributes) | ||||
|         log_action :resolve, @report | ||||
|       when 'suspend' | ||||
|         Admin::SuspensionWorker.perform_async(@report.target_account.id) | ||||
|         log_action :resolve, @report | ||||
|         log_action :suspend, @report.target_account | ||||
|         resolve_all_target_account_reports | ||||
|       when 'silence' | ||||
|         @report.target_account.update(silenced: true) | ||||
|         @report.target_account.update!(silenced: true) | ||||
|         log_action :resolve, @report | ||||
|         log_action :silence, @report.target_account | ||||
|         resolve_all_target_account_reports | ||||
|       else | ||||
|         raise ActiveRecord::RecordNotFound | ||||
|   | ||||
| @@ -2,17 +2,19 @@ | ||||
|  | ||||
| module Admin | ||||
|   class ResetsController < BaseController | ||||
|     before_action :set_account | ||||
|     before_action :set_user | ||||
|  | ||||
|     def create | ||||
|       @account.user.send_reset_password_instructions | ||||
|       authorize @user, :reset_password? | ||||
|       @user.send_reset_password_instructions | ||||
|       log_action :reset_password, @user | ||||
|       redirect_to admin_accounts_path | ||||
|     end | ||||
|  | ||||
|     private | ||||
|  | ||||
|     def set_account | ||||
|       @account = Account.find(params[:account_id]) | ||||
|     def set_user | ||||
|       @user = Account.find(params[:account_id]).user || raise(ActiveRecord::RecordNotFound) | ||||
|     end | ||||
|   end | ||||
| end | ||||
|   | ||||
							
								
								
									
										27
									
								
								app/controllers/admin/roles_controller.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								app/controllers/admin/roles_controller.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| module Admin | ||||
|   class RolesController < BaseController | ||||
|     before_action :set_user | ||||
|  | ||||
|     def promote | ||||
|       authorize @user, :promote? | ||||
|       @user.promote! | ||||
|       log_action :promote, @user | ||||
|       redirect_to admin_account_path(@user.account_id) | ||||
|     end | ||||
|  | ||||
|     def demote | ||||
|       authorize @user, :demote? | ||||
|       @user.demote! | ||||
|       log_action :demote, @user | ||||
|       redirect_to admin_account_path(@user.account_id) | ||||
|     end | ||||
|  | ||||
|     private | ||||
|  | ||||
|     def set_user | ||||
|       @user = Account.find(params[:account_id]).user || raise(ActiveRecord::RecordNotFound) | ||||
|     end | ||||
|   end | ||||
| end | ||||
| @@ -13,14 +13,17 @@ module Admin | ||||
|       closed_registrations_message | ||||
|       open_deletion | ||||
|       timeline_preview | ||||
|       show_staff_badge | ||||
|       bootstrap_timeline_accounts | ||||
|       thumbnail | ||||
|       min_invite_role | ||||
|     ).freeze | ||||
|  | ||||
|     BOOLEAN_SETTINGS = %w( | ||||
|       open_registrations | ||||
|       open_deletion | ||||
|       timeline_preview | ||||
|       show_staff_badge | ||||
|     ).freeze | ||||
|  | ||||
|     UPLOAD_SETTINGS = %w( | ||||
| @@ -28,10 +31,13 @@ module Admin | ||||
|     ).freeze | ||||
|  | ||||
|     def edit | ||||
|       authorize :settings, :show? | ||||
|       @admin_settings = Form::AdminSettings.new | ||||
|     end | ||||
|  | ||||
|     def update | ||||
|       authorize :settings, :update? | ||||
|  | ||||
|       settings_params.each do |key, value| | ||||
|         if UPLOAD_SETTINGS.include?(key) | ||||
|           upload = SiteUpload.where(var: key).first_or_initialize(var: key) | ||||
|   | ||||
| @@ -5,12 +5,16 @@ module Admin | ||||
|     before_action :set_account | ||||
|  | ||||
|     def create | ||||
|       @account.update(silenced: true) | ||||
|       authorize @account, :silence? | ||||
|       @account.update!(silenced: true) | ||||
|       log_action :silence, @account | ||||
|       redirect_to admin_accounts_path | ||||
|     end | ||||
|  | ||||
|     def destroy | ||||
|       @account.update(silenced: false) | ||||
|       authorize @account, :unsilence? | ||||
|       @account.update!(silenced: false) | ||||
|       log_action :unsilence, @account | ||||
|       redirect_to admin_accounts_path | ||||
|     end | ||||
|  | ||||
|   | ||||
| @@ -2,8 +2,6 @@ | ||||
|  | ||||
| module Admin | ||||
|   class StatusesController < BaseController | ||||
|     include Authorization | ||||
|  | ||||
|     helper_method :current_params | ||||
|  | ||||
|     before_action :set_account | ||||
| @@ -12,31 +10,39 @@ module Admin | ||||
|     PER_PAGE = 20 | ||||
|  | ||||
|     def index | ||||
|       authorize :status, :index? | ||||
|  | ||||
|       @statuses = @account.statuses | ||||
|  | ||||
|       if params[:media] | ||||
|         account_media_status_ids = @account.media_attachments.attached.reorder(nil).select(:status_id).distinct | ||||
|         @statuses.merge!(Status.where(id: account_media_status_ids)) | ||||
|       end | ||||
|       @statuses = @statuses.preload(:media_attachments, :mentions).page(params[:page]).per(PER_PAGE) | ||||
|  | ||||
|       @statuses = @statuses.preload(:media_attachments, :mentions).page(params[:page]).per(PER_PAGE) | ||||
|       @form     = Form::StatusBatch.new | ||||
|     end | ||||
|  | ||||
|     def create | ||||
|       @form = Form::StatusBatch.new(form_status_batch_params) | ||||
|       flash[:alert] = t('admin.statuses.failed_to_execute') unless @form.save | ||||
|       authorize :status, :update? | ||||
|  | ||||
|       @form         = Form::StatusBatch.new(form_status_batch_params.merge(current_account: current_account)) | ||||
|       flash[:alert] = I18n.t('admin.statuses.failed_to_execute') unless @form.save | ||||
|  | ||||
|       redirect_to admin_account_statuses_path(@account.id, current_params) | ||||
|     end | ||||
|  | ||||
|     def update | ||||
|       @status.update(status_params) | ||||
|       authorize @status, :update? | ||||
|       @status.update!(status_params) | ||||
|       log_action :update, @status | ||||
|       redirect_to admin_account_statuses_path(@account.id, current_params) | ||||
|     end | ||||
|  | ||||
|     def destroy | ||||
|       authorize @status, :destroy? | ||||
|       RemovalWorker.perform_async(@status.id) | ||||
|       log_action :destroy, @status | ||||
|       render json: @status | ||||
|     end | ||||
|  | ||||
| @@ -60,6 +66,7 @@ module Admin | ||||
|  | ||||
|     def current_params | ||||
|       page = (params[:page] || 1).to_i | ||||
|  | ||||
|       { | ||||
|         media: params[:media], | ||||
|         page: page > 1 && page, | ||||
|   | ||||
| @@ -3,6 +3,7 @@ | ||||
| module Admin | ||||
|   class SubscriptionsController < BaseController | ||||
|     def index | ||||
|       authorize :subscription, :index? | ||||
|       @subscriptions = ordered_subscriptions.page(requested_page) | ||||
|     end | ||||
|  | ||||
|   | ||||
| @@ -5,12 +5,16 @@ module Admin | ||||
|     before_action :set_account | ||||
|  | ||||
|     def create | ||||
|       authorize @account, :suspend? | ||||
|       Admin::SuspensionWorker.perform_async(@account.id) | ||||
|       log_action :suspend, @account | ||||
|       redirect_to admin_accounts_path | ||||
|     end | ||||
|  | ||||
|     def destroy | ||||
|       @account.update(suspended: false) | ||||
|       authorize @account, :unsuspend? | ||||
|       @account.unsuspend! | ||||
|       log_action :unsuspend, @account | ||||
|       redirect_to admin_accounts_path | ||||
|     end | ||||
|  | ||||
|   | ||||
| @@ -5,7 +5,9 @@ module Admin | ||||
|     before_action :set_user | ||||
|  | ||||
|     def destroy | ||||
|       authorize @user, :disable_2fa? | ||||
|       @user.disable_two_factor! | ||||
|       log_action :disable_2fa, @user | ||||
|       redirect_to admin_accounts_path | ||||
|     end | ||||
|  | ||||
|   | ||||
| @@ -17,12 +17,13 @@ class Api::V1::Accounts::SearchController < Api::BaseController | ||||
|     AccountSearchService.new.call( | ||||
|       params[:q], | ||||
|       limit_param(DEFAULT_ACCOUNTS_LIMIT), | ||||
|       resolving_search?, | ||||
|       current_account | ||||
|       current_account, | ||||
|       resolve: truthy_param?(:resolve), | ||||
|       following: truthy_param?(:following) | ||||
|     ) | ||||
|   end | ||||
|  | ||||
|   def resolving_search? | ||||
|     params[:resolve] == 'true' | ||||
|   def truthy_param?(key) | ||||
|     params[key] == 'true' | ||||
|   end | ||||
| end | ||||
|   | ||||
| @@ -13,9 +13,9 @@ class Api::V1::AccountsController < Api::BaseController | ||||
|   end | ||||
|  | ||||
|   def follow | ||||
|     FollowService.new.call(current_user.account, @account.acct) | ||||
|     FollowService.new.call(current_user.account, @account.acct, reblogs: params[:reblogs]) | ||||
|  | ||||
|     options = @account.locked? ? {} : { following_map: { @account.id => true }, requested_map: { @account.id => false } } | ||||
|     options = @account.locked? ? {} : { following_map: { @account.id => { reblogs: params[:reblogs] } }, requested_map: { @account.id => false } } | ||||
|  | ||||
|     render json: @account, serializer: REST::RelationshipSerializer, relationships: relationships(options) | ||||
|   end | ||||
| @@ -26,7 +26,7 @@ class Api::V1::AccountsController < Api::BaseController | ||||
|   end | ||||
|  | ||||
|   def mute | ||||
|     MuteService.new.call(current_user.account, @account) | ||||
|     MuteService.new.call(current_user.account, @account, notifications: params[:notifications]) | ||||
|     render json: @account, serializer: REST::RelationshipSerializer, relationships: relationships | ||||
|   end | ||||
|  | ||||
| @@ -51,7 +51,7 @@ class Api::V1::AccountsController < Api::BaseController | ||||
|     @account = Account.find(params[:id]) | ||||
|   end | ||||
|  | ||||
|   def relationships(options = {}) | ||||
|   def relationships(**options) | ||||
|     AccountRelationshipsPresenter.new([@account.id], current_user.account_id, options) | ||||
|   end | ||||
| end | ||||
|   | ||||
							
								
								
									
										97
									
								
								app/controllers/api/v1/lists/accounts_controller.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								app/controllers/api/v1/lists/accounts_controller.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,97 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class Api::V1::Lists::AccountsController < Api::BaseController | ||||
|   before_action -> { doorkeeper_authorize! :read },    only: [:show] | ||||
|   before_action -> { doorkeeper_authorize! :write }, except: [:show] | ||||
|  | ||||
|   before_action :require_user! | ||||
|   before_action :set_list | ||||
|  | ||||
|   after_action :insert_pagination_headers, only: :show | ||||
|  | ||||
|   def show | ||||
|     @accounts = load_accounts | ||||
|     render json: @accounts, each_serializer: REST::AccountSerializer | ||||
|   end | ||||
|  | ||||
|   def create | ||||
|     ApplicationRecord.transaction do | ||||
|       list_accounts.each do |account| | ||||
|         @list.accounts << account | ||||
|       end | ||||
|     end | ||||
|  | ||||
|     render_empty | ||||
|   end | ||||
|  | ||||
|   def destroy | ||||
|     ListAccount.where(list: @list, account_id: account_ids).destroy_all | ||||
|     render_empty | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def set_list | ||||
|     @list = List.where(account: current_account).find(params[:list_id]) | ||||
|   end | ||||
|  | ||||
|   def load_accounts | ||||
|     if unlimited? | ||||
|       @list.accounts.all | ||||
|     else | ||||
|       @list.accounts.paginate_by_max_id(limit_param(DEFAULT_ACCOUNTS_LIMIT), params[:max_id], params[:since_id]) | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def list_accounts | ||||
|     Account.find(account_ids) | ||||
|   end | ||||
|  | ||||
|   def account_ids | ||||
|     Array(resource_params[:account_ids]) | ||||
|   end | ||||
|  | ||||
|   def resource_params | ||||
|     params.permit(account_ids: []) | ||||
|   end | ||||
|  | ||||
|   def insert_pagination_headers | ||||
|     set_pagination_headers(next_path, prev_path) | ||||
|   end | ||||
|  | ||||
|   def next_path | ||||
|     return if unlimited? | ||||
|  | ||||
|     if records_continue? | ||||
|       api_v1_list_accounts_url pagination_params(max_id: pagination_max_id) | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def prev_path | ||||
|     return if unlimited? | ||||
|  | ||||
|     unless @accounts.empty? | ||||
|       api_v1_list_accounts_url pagination_params(since_id: pagination_since_id) | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def pagination_max_id | ||||
|     @accounts.last.id | ||||
|   end | ||||
|  | ||||
|   def pagination_since_id | ||||
|     @accounts.first.id | ||||
|   end | ||||
|  | ||||
|   def records_continue? | ||||
|     @accounts.size == limit_param(DEFAULT_ACCOUNTS_LIMIT) | ||||
|   end | ||||
|  | ||||
|   def pagination_params(core_params) | ||||
|     params.permit(:limit).merge(core_params) | ||||
|   end | ||||
|  | ||||
|   def unlimited? | ||||
|     params[:limit] == '0' | ||||
|   end | ||||
| end | ||||
							
								
								
									
										79
									
								
								app/controllers/api/v1/lists_controller.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								app/controllers/api/v1/lists_controller.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,79 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class Api::V1::ListsController < Api::BaseController | ||||
|   LISTS_LIMIT = 50 | ||||
|  | ||||
|   before_action -> { doorkeeper_authorize! :read },    only: [:index, :show] | ||||
|   before_action -> { doorkeeper_authorize! :write }, except: [:index, :show] | ||||
|  | ||||
|   before_action :require_user! | ||||
|   before_action :set_list, except: [:index, :create] | ||||
|  | ||||
|   after_action :insert_pagination_headers, only: :index | ||||
|  | ||||
|   def index | ||||
|     @lists = List.where(account: current_account).paginate_by_max_id(limit_param(LISTS_LIMIT), params[:max_id], params[:since_id]) | ||||
|     render json: @lists, each_serializer: REST::ListSerializer | ||||
|   end | ||||
|  | ||||
|   def show | ||||
|     render json: @list, serializer: REST::ListSerializer | ||||
|   end | ||||
|  | ||||
|   def create | ||||
|     @list = List.create!(list_params.merge(account: current_account)) | ||||
|     render json: @list, serializer: REST::ListSerializer | ||||
|   end | ||||
|  | ||||
|   def update | ||||
|     @list.update!(list_params) | ||||
|     render json: @list, serializer: REST::ListSerializer | ||||
|   end | ||||
|  | ||||
|   def destroy | ||||
|     @list.destroy! | ||||
|     render_empty | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def set_list | ||||
|     @list = List.where(account: current_account).find(params[:id]) | ||||
|   end | ||||
|  | ||||
|   def list_params | ||||
|     params.permit(:title) | ||||
|   end | ||||
|  | ||||
|   def insert_pagination_headers | ||||
|     set_pagination_headers(next_path, prev_path) | ||||
|   end | ||||
|  | ||||
|   def next_path | ||||
|     if records_continue? | ||||
|       api_v1_lists_url pagination_params(max_id: pagination_max_id) | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def prev_path | ||||
|     unless @lists.empty? | ||||
|       api_v1_lists_url pagination_params(since_id: pagination_since_id) | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def pagination_max_id | ||||
|     @lists.last.id | ||||
|   end | ||||
|  | ||||
|   def pagination_since_id | ||||
|     @lists.first.id | ||||
|   end | ||||
|  | ||||
|   def records_continue? | ||||
|     @lists.size == limit_param(LISTS_LIMIT) | ||||
|   end | ||||
|  | ||||
|   def pagination_params(core_params) | ||||
|     params.permit(:limit).merge(core_params) | ||||
|   end | ||||
| end | ||||
| @@ -19,7 +19,7 @@ class Api::V1::ReportsController < Api::BaseController | ||||
|       comment: report_params[:comment] | ||||
|     ) | ||||
|  | ||||
|     User.admins.includes(:account).each { |u| AdminMailer.new_report(u.account, @report).deliver_later } | ||||
|     User.staff.includes(:account).each { |u| AdminMailer.new_report(u.account, @report).deliver_later } | ||||
|  | ||||
|     render json: @report, serializer: REST::ReportSerializer | ||||
|   end | ||||
|   | ||||
| @@ -1,6 +1,8 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class Api::V1::SearchController < Api::BaseController | ||||
|   include Authorization | ||||
|  | ||||
|   RESULTS_LIMIT = 5 | ||||
|  | ||||
|   before_action -> { doorkeeper_authorize! :read } | ||||
| @@ -9,12 +11,24 @@ class Api::V1::SearchController < Api::BaseController | ||||
|   respond_to :json | ||||
|  | ||||
|   def index | ||||
|     @search = Search.new(search_results) | ||||
|     @search = Search.new(search) | ||||
|     render json: @search, serializer: REST::SearchSerializer | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def search | ||||
|     search_results.tap do |search| | ||||
|       search[:statuses].keep_if do |status| | ||||
|         begin | ||||
|           authorize status, :show? | ||||
|         rescue Mastodon::NotPermittedError | ||||
|           false | ||||
|         end | ||||
|       end | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def search_results | ||||
|     SearchService.new.call( | ||||
|       params[:q], | ||||
|   | ||||
| @@ -31,7 +31,7 @@ class Api::V1::Timelines::HomeController < Api::BaseController | ||||
|   end | ||||
|  | ||||
|   def account_home_feed | ||||
|     Feed.new(:home, current_account) | ||||
|     HomeFeed.new(current_account) | ||||
|   end | ||||
|  | ||||
|   def insert_pagination_headers | ||||
|   | ||||
							
								
								
									
										66
									
								
								app/controllers/api/v1/timelines/list_controller.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								app/controllers/api/v1/timelines/list_controller.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class Api::V1::Timelines::ListController < Api::BaseController | ||||
|   before_action -> { doorkeeper_authorize! :read } | ||||
|   before_action :require_user! | ||||
|   before_action :set_list | ||||
|   before_action :set_statuses | ||||
|  | ||||
|   after_action :insert_pagination_headers, unless: -> { @statuses.empty? } | ||||
|  | ||||
|   def show | ||||
|     render json: @statuses, | ||||
|            each_serializer: REST::StatusSerializer, | ||||
|            relationships: StatusRelationshipsPresenter.new(@statuses, current_user.account_id) | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def set_list | ||||
|     @list = List.where(account: current_account).find(params[:id]) | ||||
|   end | ||||
|  | ||||
|   def set_statuses | ||||
|     @statuses = cached_list_statuses | ||||
|   end | ||||
|  | ||||
|   def cached_list_statuses | ||||
|     cache_collection list_statuses, Status | ||||
|   end | ||||
|  | ||||
|   def list_statuses | ||||
|     list_feed.get( | ||||
|       limit_param(DEFAULT_STATUSES_LIMIT), | ||||
|       params[:max_id], | ||||
|       params[:since_id] | ||||
|     ) | ||||
|   end | ||||
|  | ||||
|   def list_feed | ||||
|     ListFeed.new(@list) | ||||
|   end | ||||
|  | ||||
|   def insert_pagination_headers | ||||
|     set_pagination_headers(next_path, prev_path) | ||||
|   end | ||||
|  | ||||
|   def pagination_params(core_params) | ||||
|     params.permit(:limit).merge(core_params) | ||||
|   end | ||||
|  | ||||
|   def next_path | ||||
|     api_v1_timelines_list_url params[:id], pagination_params(max_id: pagination_max_id) | ||||
|   end | ||||
|  | ||||
|   def prev_path | ||||
|     api_v1_timelines_list_url params[:id], pagination_params(since_id: pagination_since_id) | ||||
|   end | ||||
|  | ||||
|   def pagination_max_id | ||||
|     @statuses.last.id | ||||
|   end | ||||
|  | ||||
|   def pagination_since_id | ||||
|     @statuses.first.id | ||||
|   end | ||||
| end | ||||
| @@ -18,6 +18,7 @@ class ApplicationController < ActionController::Base | ||||
|   rescue_from ActionController::RoutingError, with: :not_found | ||||
|   rescue_from ActiveRecord::RecordNotFound, with: :not_found | ||||
|   rescue_from ActionController::InvalidAuthenticityToken, with: :unprocessable_entity | ||||
|   rescue_from Mastodon::NotPermittedError, with: :forbidden | ||||
|  | ||||
|   before_action :store_current_location, except: :raise_not_found, unless: :devise_controller? | ||||
|   before_action :check_suspension, if: :user_signed_in? | ||||
| @@ -40,6 +41,10 @@ class ApplicationController < ActionController::Base | ||||
|     redirect_to root_path unless current_user&.admin? | ||||
|   end | ||||
|  | ||||
|   def require_staff! | ||||
|     redirect_to root_path unless current_user&.staff? | ||||
|   end | ||||
|  | ||||
|   def check_suspension | ||||
|     forbidden if current_user.account.suspended? | ||||
|   end | ||||
| @@ -99,7 +104,7 @@ class ApplicationController < ActionController::Base | ||||
|     unless uncached_ids.empty? | ||||
|       uncached = klass.where(id: uncached_ids).with_includes.map { |item| [item.id, item] }.to_h | ||||
|  | ||||
|       uncached.values.each do |item| | ||||
|       uncached.each_value do |item| | ||||
|         Rails.cache.write(item.cache_key, item) | ||||
|       end | ||||
|     end | ||||
|   | ||||
| @@ -16,13 +16,16 @@ class Auth::RegistrationsController < Devise::RegistrationsController | ||||
|  | ||||
|   def build_resource(hash = nil) | ||||
|     super(hash) | ||||
|  | ||||
|     resource.locale      = I18n.locale | ||||
|     resource.invite_code = params[:invite_code] if resource.invite_code.blank? | ||||
|  | ||||
|     resource.build_account if resource.account.nil? | ||||
|   end | ||||
|  | ||||
|   def configure_sign_up_params | ||||
|     devise_parameter_sanitizer.permit(:sign_up) do |u| | ||||
|       u.permit({ account_attributes: [:username] }, :email, :password, :password_confirmation) | ||||
|       u.permit({ account_attributes: [:username] }, :email, :password, :password_confirmation, :invite_code) | ||||
|     end | ||||
|   end | ||||
|  | ||||
| @@ -35,7 +38,19 @@ class Auth::RegistrationsController < Devise::RegistrationsController | ||||
|   end | ||||
|  | ||||
|   def check_enabled_registrations | ||||
|     redirect_to root_path if single_user_mode? || !Setting.open_registrations | ||||
|     redirect_to root_path if single_user_mode? || !allowed_registrations? | ||||
|   end | ||||
|  | ||||
|   def allowed_registrations? | ||||
|     Setting.open_registrations || (invite_code.present? && Invite.find_by(code: invite_code)&.valid_for_use?) | ||||
|   end | ||||
|  | ||||
|   def invite_code | ||||
|     if params[:user] | ||||
|       params[:user][:invite_code] | ||||
|     else | ||||
|       params[:invite_code] | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   private | ||||
|   | ||||
| @@ -62,7 +62,7 @@ class Auth::SessionsController < Devise::SessionsController | ||||
|  | ||||
|     if user_params[:otp_attempt].present? && session[:otp_user_id] | ||||
|       authenticate_with_two_factor_via_otp(user) | ||||
|     elsif user && user.valid_password?(user_params[:password]) | ||||
|     elsif user&.valid_password?(user_params[:password]) | ||||
|       prompt_for_two_factor(user) | ||||
|     end | ||||
|   end | ||||
|   | ||||
							
								
								
									
										9
									
								
								app/controllers/concerns/accountable_concern.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								app/controllers/concerns/accountable_concern.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| module AccountableConcern | ||||
|   extend ActiveSupport::Concern | ||||
|  | ||||
|   def log_action(action, target) | ||||
|     Admin::ActionLog.create(account: current_account, action: action, target: target) | ||||
|   end | ||||
| end | ||||
| @@ -2,6 +2,7 @@ | ||||
|  | ||||
| module Authorization | ||||
|   extend ActiveSupport::Concern | ||||
|  | ||||
|   include Pundit | ||||
|  | ||||
|   def pundit_user | ||||
|   | ||||
							
								
								
									
										43
									
								
								app/controllers/invites_controller.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								app/controllers/invites_controller.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class InvitesController < ApplicationController | ||||
|   include Authorization | ||||
|  | ||||
|   layout 'admin' | ||||
|  | ||||
|   before_action :authenticate_user! | ||||
|  | ||||
|   def index | ||||
|     authorize :invite, :create? | ||||
|  | ||||
|     @invites = Invite.where(user: current_user) | ||||
|     @invite  = Invite.new(expires_in: 1.day.to_i) | ||||
|   end | ||||
|  | ||||
|   def create | ||||
|     authorize :invite, :create? | ||||
|  | ||||
|     @invite      = Invite.new(resource_params) | ||||
|     @invite.user = current_user | ||||
|  | ||||
|     if @invite.save | ||||
|       redirect_to invites_path | ||||
|     else | ||||
|       @invites = Invite.where(user: current_user) | ||||
|       render :index | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def destroy | ||||
|     @invite = Invite.where(user: current_user).find(params[:id]) | ||||
|     authorize @invite, :destroy? | ||||
|     @invite.expire! | ||||
|     redirect_to invites_path | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def resource_params | ||||
|     params.require(:invite).permit(:max_uses, :expires_in) | ||||
|   end | ||||
| end | ||||
							
								
								
									
										33
									
								
								app/controllers/settings/migrations_controller.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								app/controllers/settings/migrations_controller.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| class Settings::MigrationsController < ApplicationController | ||||
|   layout 'admin' | ||||
|  | ||||
|   before_action :authenticate_user! | ||||
|  | ||||
|   def show | ||||
|     @migration = Form::Migration.new(account: current_account.moved_to_account) | ||||
|   end | ||||
|  | ||||
|   def update | ||||
|     @migration = Form::Migration.new(resource_params) | ||||
|  | ||||
|     if @migration.valid? && migration_account_changed? | ||||
|       current_account.update!(moved_to_account: @migration.account) | ||||
|       ActivityPub::UpdateDistributionWorker.perform_async(current_account.id) | ||||
|       redirect_to settings_migration_path, notice: I18n.t('migrations.updated_msg') | ||||
|     else | ||||
|       render :show | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def resource_params | ||||
|     params.require(:migration).permit(:acct) | ||||
|   end | ||||
|  | ||||
|   def migration_account_changed? | ||||
|     current_account.moved_to_account_id != @migration.account&.id | ||||
|   end | ||||
| end | ||||
| @@ -26,7 +26,7 @@ class Settings::NotificationsController < ApplicationController | ||||
|   def user_settings_params | ||||
|     params.require(:user).permit( | ||||
|       notification_emails: %i(follow follow_request reblog favourite mention digest), | ||||
|       interactions: %i(must_be_follower must_be_following) | ||||
|       interactions: %i(must_be_follower must_be_following must_be_following_dm) | ||||
|     ) | ||||
|   end | ||||
| end | ||||
|   | ||||
| @@ -6,12 +6,10 @@ module WellKnown | ||||
|  | ||||
|     def show | ||||
|       @account = Account.find_local!(username_from_resource) | ||||
|       @canonical_account_uri = @account.to_webfinger_s | ||||
|       @magic_key = pem_to_magic_key(@account.keypair.public_key) | ||||
|  | ||||
|       respond_to do |format| | ||||
|         format.any(:json, :html) do | ||||
|           render formats: :json, content_type: 'application/jrd+json' | ||||
|           render json: @account, serializer: WebfingerSerializer, content_type: 'application/jrd+json' | ||||
|         end | ||||
|  | ||||
|         format.xml do | ||||
| @@ -35,21 +33,6 @@ module WellKnown | ||||
|       WebfingerResource.new(resource_user).username | ||||
|     end | ||||
|  | ||||
|     def pem_to_magic_key(public_key) | ||||
|       modulus, exponent = [public_key.n, public_key.e].map do |component| | ||||
|         result = [] | ||||
|  | ||||
|         until component.zero? | ||||
|           result << [component % 256].pack('C') | ||||
|           component >>= 8 | ||||
|         end | ||||
|  | ||||
|         result.reverse.join | ||||
|       end | ||||
|  | ||||
|       (['RSA'] + [modulus, exponent].map { |n| Base64.urlsafe_encode64(n) }).join('.') | ||||
|     end | ||||
|  | ||||
|     def resource_param | ||||
|       params.require(:resource) | ||||
|     end | ||||
|   | ||||
							
								
								
									
										103
									
								
								app/helpers/admin/action_logs_helper.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										103
									
								
								app/helpers/admin/action_logs_helper.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,103 @@ | ||||
| # frozen_string_literal: true | ||||
|  | ||||
| module Admin::ActionLogsHelper | ||||
|   def log_target(log) | ||||
|     if log.target | ||||
|       linkable_log_target(log.target) | ||||
|     else | ||||
|       log_target_from_history(log.target_type, log.recorded_changes) | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def linkable_log_target(record) | ||||
|     case record.class.name | ||||
|     when 'Account' | ||||
|       link_to record.acct, admin_account_path(record.id) | ||||
|     when 'User' | ||||
|       link_to record.account.acct, admin_account_path(record.account_id) | ||||
|     when 'CustomEmoji' | ||||
|       record.shortcode | ||||
|     when 'Report' | ||||
|       link_to "##{record.id}", admin_report_path(record) | ||||
|     when 'DomainBlock', 'EmailDomainBlock' | ||||
|       link_to record.domain, "https://#{record.domain}" | ||||
|     when 'Status' | ||||
|       link_to record.account.acct, TagManager.instance.url_for(record) | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def log_target_from_history(type, attributes) | ||||
|     case type | ||||
|     when 'CustomEmoji' | ||||
|       attributes['shortcode'] | ||||
|     when 'DomainBlock', 'EmailDomainBlock' | ||||
|       link_to attributes['domain'], "https://#{attributes['domain']}" | ||||
|     when 'Status' | ||||
|       tmp_status = Status.new(attributes) | ||||
|       link_to tmp_status.account.acct, TagManager.instance.url_for(tmp_status) | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def relevant_log_changes(log) | ||||
|     if log.target_type == 'CustomEmoji' && [:enable, :disable, :destroy].include?(log.action) | ||||
|       log.recorded_changes.slice('domain') | ||||
|     elsif log.target_type == 'CustomEmoji' && log.action == :update | ||||
|       log.recorded_changes.slice('domain', 'visible_in_picker') | ||||
|     elsif log.target_type == 'User' && [:promote, :demote].include?(log.action) | ||||
|       log.recorded_changes.slice('moderator', 'admin') | ||||
|     elsif log.target_type == 'DomainBlock' | ||||
|       log.recorded_changes.slice('severity', 'reject_media') | ||||
|     elsif log.target_type == 'Status' && log.action == :update | ||||
|       log.recorded_changes.slice('sensitive') | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def log_extra_attributes(hash) | ||||
|     safe_join(hash.to_a.map { |key, value| safe_join([content_tag(:span, key, class: 'diff-key'), '=', log_change(value)]) }, ' ') | ||||
|   end | ||||
|  | ||||
|   def log_change(val) | ||||
|     return content_tag(:span, val, class: 'diff-neutral') unless val.is_a?(Array) | ||||
|     safe_join([content_tag(:span, val.first, class: 'diff-old'), content_tag(:span, val.last, class: 'diff-new')], '→') | ||||
|   end | ||||
|  | ||||
|   def icon_for_log(log) | ||||
|     case log.target_type | ||||
|     when 'Account', 'User' | ||||
|       'user' | ||||
|     when 'CustomEmoji' | ||||
|       'file' | ||||
|     when 'Report' | ||||
|       'flag' | ||||
|     when 'DomainBlock' | ||||
|       'lock' | ||||
|     when 'EmailDomainBlock' | ||||
|       'envelope' | ||||
|     when 'Status' | ||||
|       'pencil' | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def class_for_log_icon(log) | ||||
|     case log.action | ||||
|     when :enable, :unsuspend, :unsilence, :confirm, :promote, :resolve | ||||
|       'positive' | ||||
|     when :create | ||||
|       opposite_verbs?(log) ? 'negative' : 'positive' | ||||
|     when :update, :reset_password, :disable_2fa, :memorialize | ||||
|       'neutral' | ||||
|     when :demote, :silence, :disable, :suspend | ||||
|       'negative' | ||||
|     when :destroy | ||||
|       opposite_verbs?(log) ? 'positive' : 'negative' | ||||
|     else | ||||
|       '' | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def opposite_verbs?(log) | ||||
|     %w(DomainBlock EmailDomainBlock).include?(log.target_type) | ||||
|   end | ||||
| end | ||||
| @@ -3,8 +3,9 @@ | ||||
| module Admin::FilterHelper | ||||
|   ACCOUNT_FILTERS = %i(local remote by_domain silenced suspended recent username display_name email ip).freeze | ||||
|   REPORT_FILTERS = %i(resolved account_id target_account_id).freeze | ||||
|   INVITE_FILTER = %i(available expired).freeze | ||||
|  | ||||
|   FILTERS = ACCOUNT_FILTERS + REPORT_FILTERS | ||||
|   FILTERS = ACCOUNT_FILTERS + REPORT_FILTERS + INVITE_FILTER | ||||
|  | ||||
|   def filter_link_to(text, link_to_params, link_class_params = link_to_params) | ||||
|     new_url = filtered_url_for(link_to_params) | ||||
| @@ -12,13 +13,13 @@ module Admin::FilterHelper | ||||
|     link_to text, new_url, class: filter_link_class(new_class) | ||||
|   end | ||||
|  | ||||
|   def table_link_to(icon, text, path, options = {}) | ||||
|   def table_link_to(icon, text, path, **options) | ||||
|     link_to safe_join([fa_icon(icon), text]), path, options.merge(class: 'table-action-link') | ||||
|   end | ||||
|  | ||||
|   def selected?(more_params) | ||||
|     new_url = filtered_url_for(more_params) | ||||
|     filter_link_class(new_url) == 'selected' ? true : false | ||||
|     filter_link_class(new_url) == 'selected' | ||||
|   end | ||||
|  | ||||
|   private | ||||
|   | ||||
| @@ -5,7 +5,7 @@ module ApplicationHelper | ||||
|     current_page?(path) ? 'active' : '' | ||||
|   end | ||||
|  | ||||
|   def active_link_to(label, path, options = {}) | ||||
|   def active_link_to(label, path, **options) | ||||
|     link_to label, path, options.merge(class: active_nav_class(path)) | ||||
|   end | ||||
|  | ||||
| @@ -35,6 +35,11 @@ module ApplicationHelper | ||||
|     Rails.env.production? ? site_title : "#{site_title} (Dev)" | ||||
|   end | ||||
|  | ||||
|   def can?(action, record) | ||||
|     return false if record.nil? | ||||
|     policy(record).public_send("#{action}?") | ||||
|   end | ||||
|  | ||||
|   def fa_icon(icon, attributes = {}) | ||||
|     class_names = attributes[:class]&.split(' ') || [] | ||||
|     class_names << 'fa' | ||||
| @@ -43,6 +48,10 @@ module ApplicationHelper | ||||
|     content_tag(:i, nil, attributes.merge(class: class_names.join(' '))) | ||||
|   end | ||||
|  | ||||
|   def custom_emoji_tag(custom_emoji) | ||||
|     image_tag(custom_emoji.image.url, class: 'emojione', alt: ":#{custom_emoji.shortcode}:") | ||||
|   end | ||||
|  | ||||
|   def opengraph(property, content) | ||||
|     tag(:meta, content: content, property: property) | ||||
|   end | ||||
|   | ||||
| @@ -9,6 +9,28 @@ module JsonLdHelper | ||||
|     value.is_a?(Array) ? value.first : value | ||||
|   end | ||||
|  | ||||
|   # The url attribute can be a string, an array of strings, or an array of objects. | ||||
|   # The objects could include a mimeType. Not-included mimeType means it's text/html. | ||||
|   def url_to_href(value, preferred_type = nil) | ||||
|     single_value = if value.is_a?(Array) && !value.first.is_a?(String) | ||||
|                      value.find { |link| preferred_type.nil? || ((link['mimeType'].presence || 'text/html') == preferred_type) } | ||||
|                    elsif value.is_a?(Array) | ||||
|                      value.first | ||||
|                    else | ||||
|                      value | ||||
|                    end | ||||
|  | ||||
|     if single_value.nil? || single_value.is_a?(String) | ||||
|       single_value | ||||
|     else | ||||
|       single_value['href'] | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def as_array(value) | ||||
|     value.is_a?(Array) ? value : [value] | ||||
|   end | ||||
|  | ||||
|   def value_or_id(value) | ||||
|     value.is_a?(String) || value.nil? ? value : value['id'] | ||||
|   end | ||||
|   | ||||
| @@ -11,7 +11,7 @@ module RoutingHelper | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def full_asset_url(source, options = {}) | ||||
|   def full_asset_url(source, **options) | ||||
|     source = ActionController::Base.helpers.asset_url(source, options) unless use_storage? | ||||
|  | ||||
|     URI.join(root_url, source).to_s | ||||
|   | ||||
| @@ -105,12 +105,13 @@ export function fetchAccountFail(id, error) { | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| export function followAccount(id) { | ||||
| export function followAccount(id, reblogs = true) { | ||||
|   return (dispatch, getState) => { | ||||
|     const alreadyFollowing = getState().getIn(['relationships', id, 'following']); | ||||
|     dispatch(followAccountRequest(id)); | ||||
|  | ||||
|     api(getState).post(`/api/v1/accounts/${id}/follow`).then(response => { | ||||
|       dispatch(followAccountSuccess(response.data)); | ||||
|     api(getState).post(`/api/v1/accounts/${id}/follow`, { reblogs }).then(response => { | ||||
|       dispatch(followAccountSuccess(response.data, alreadyFollowing)); | ||||
|     }).catch(error => { | ||||
|       dispatch(followAccountFail(error)); | ||||
|     }); | ||||
| @@ -136,10 +137,11 @@ export function followAccountRequest(id) { | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| export function followAccountSuccess(relationship) { | ||||
| export function followAccountSuccess(relationship, alreadyFollowing) { | ||||
|   return { | ||||
|     type: ACCOUNT_FOLLOW_SUCCESS, | ||||
|     relationship, | ||||
|     alreadyFollowing, | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| @@ -241,11 +243,11 @@ export function unblockAccountFail(error) { | ||||
| }; | ||||
|  | ||||
|  | ||||
| export function muteAccount(id) { | ||||
| export function muteAccount(id, notifications) { | ||||
|   return (dispatch, getState) => { | ||||
|     dispatch(muteAccountRequest(id)); | ||||
|  | ||||
|     api(getState).post(`/api/v1/accounts/${id}/mute`).then(response => { | ||||
|     api(getState).post(`/api/v1/accounts/${id}/mute`, { notifications }).then(response => { | ||||
|       // Pass in entire statuses map so we can use it to filter stuff in different parts of the reducers | ||||
|       dispatch(muteAccountSuccess(response.data, getState().get('statuses'))); | ||||
|     }).catch(error => { | ||||
|   | ||||
							
								
								
									
										313
									
								
								app/javascript/mastodon/actions/lists.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										313
									
								
								app/javascript/mastodon/actions/lists.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,313 @@ | ||||
| import api from '../api'; | ||||
|  | ||||
| export const LIST_FETCH_REQUEST = 'LIST_FETCH_REQUEST'; | ||||
| export const LIST_FETCH_SUCCESS = 'LIST_FETCH_SUCCESS'; | ||||
| export const LIST_FETCH_FAIL    = 'LIST_FETCH_FAIL'; | ||||
|  | ||||
| export const LISTS_FETCH_REQUEST = 'LISTS_FETCH_REQUEST'; | ||||
| export const LISTS_FETCH_SUCCESS = 'LISTS_FETCH_SUCCESS'; | ||||
| export const LISTS_FETCH_FAIL    = 'LISTS_FETCH_FAIL'; | ||||
|  | ||||
| export const LIST_EDITOR_TITLE_CHANGE = 'LIST_EDITOR_TITLE_CHANGE'; | ||||
| export const LIST_EDITOR_RESET        = 'LIST_EDITOR_RESET'; | ||||
| export const LIST_EDITOR_SETUP        = 'LIST_EDITOR_SETUP'; | ||||
|  | ||||
| export const LIST_CREATE_REQUEST = 'LIST_CREATE_REQUEST'; | ||||
| export const LIST_CREATE_SUCCESS = 'LIST_CREATE_SUCCESS'; | ||||
| export const LIST_CREATE_FAIL    = 'LIST_CREATE_FAIL'; | ||||
|  | ||||
| export const LIST_UPDATE_REQUEST = 'LIST_UPDATE_REQUEST'; | ||||
| export const LIST_UPDATE_SUCCESS = 'LIST_UPDATE_SUCCESS'; | ||||
| export const LIST_UPDATE_FAIL    = 'LIST_UPDATE_FAIL'; | ||||
|  | ||||
| export const LIST_DELETE_REQUEST = 'LIST_DELETE_REQUEST'; | ||||
| export const LIST_DELETE_SUCCESS = 'LIST_DELETE_SUCCESS'; | ||||
| export const LIST_DELETE_FAIL    = 'LIST_DELETE_FAIL'; | ||||
|  | ||||
| export const LIST_ACCOUNTS_FETCH_REQUEST = 'LIST_ACCOUNTS_FETCH_REQUEST'; | ||||
| export const LIST_ACCOUNTS_FETCH_SUCCESS = 'LIST_ACCOUNTS_FETCH_SUCCESS'; | ||||
| export const LIST_ACCOUNTS_FETCH_FAIL    = 'LIST_ACCOUNTS_FETCH_FAIL'; | ||||
|  | ||||
| export const LIST_EDITOR_SUGGESTIONS_CHANGE = 'LIST_EDITOR_SUGGESTIONS_CHANGE'; | ||||
| export const LIST_EDITOR_SUGGESTIONS_READY  = 'LIST_EDITOR_SUGGESTIONS_READY'; | ||||
| export const LIST_EDITOR_SUGGESTIONS_CLEAR  = 'LIST_EDITOR_SUGGESTIONS_CLEAR'; | ||||
|  | ||||
| export const LIST_EDITOR_ADD_REQUEST = 'LIST_EDITOR_ADD_REQUEST'; | ||||
| export const LIST_EDITOR_ADD_SUCCESS = 'LIST_EDITOR_ADD_SUCCESS'; | ||||
| export const LIST_EDITOR_ADD_FAIL    = 'LIST_EDITOR_ADD_FAIL'; | ||||
|  | ||||
| export const LIST_EDITOR_REMOVE_REQUEST = 'LIST_EDITOR_REMOVE_REQUEST'; | ||||
| export const LIST_EDITOR_REMOVE_SUCCESS = 'LIST_EDITOR_REMOVE_SUCCESS'; | ||||
| export const LIST_EDITOR_REMOVE_FAIL    = 'LIST_EDITOR_REMOVE_FAIL'; | ||||
|  | ||||
| export const fetchList = id => (dispatch, getState) => { | ||||
|   if (getState().getIn(['lists', id])) { | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   dispatch(fetchListRequest(id)); | ||||
|  | ||||
|   api(getState).get(`/api/v1/lists/${id}`) | ||||
|     .then(({ data }) => dispatch(fetchListSuccess(data))) | ||||
|     .catch(err => dispatch(fetchListFail(id, err))); | ||||
| }; | ||||
|  | ||||
| export const fetchListRequest = id => ({ | ||||
|   type: LIST_FETCH_REQUEST, | ||||
|   id, | ||||
| }); | ||||
|  | ||||
| export const fetchListSuccess = list => ({ | ||||
|   type: LIST_FETCH_SUCCESS, | ||||
|   list, | ||||
| }); | ||||
|  | ||||
| export const fetchListFail = (id, error) => ({ | ||||
|   type: LIST_FETCH_FAIL, | ||||
|   id, | ||||
|   error, | ||||
| }); | ||||
|  | ||||
| export const fetchLists = () => (dispatch, getState) => { | ||||
|   dispatch(fetchListsRequest()); | ||||
|  | ||||
|   api(getState).get('/api/v1/lists') | ||||
|     .then(({ data }) => dispatch(fetchListsSuccess(data))) | ||||
|     .catch(err => dispatch(fetchListsFail(err))); | ||||
| }; | ||||
|  | ||||
| export const fetchListsRequest = () => ({ | ||||
|   type: LISTS_FETCH_REQUEST, | ||||
| }); | ||||
|  | ||||
| export const fetchListsSuccess = lists => ({ | ||||
|   type: LISTS_FETCH_SUCCESS, | ||||
|   lists, | ||||
| }); | ||||
|  | ||||
| export const fetchListsFail = error => ({ | ||||
|   type: LISTS_FETCH_FAIL, | ||||
|   error, | ||||
| }); | ||||
|  | ||||
| export const submitListEditor = shouldReset => (dispatch, getState) => { | ||||
|   const listId = getState().getIn(['listEditor', 'listId']); | ||||
|   const title  = getState().getIn(['listEditor', 'title']); | ||||
|  | ||||
|   if (listId === null) { | ||||
|     dispatch(createList(title, shouldReset)); | ||||
|   } else { | ||||
|     dispatch(updateList(listId, title, shouldReset)); | ||||
|   } | ||||
| }; | ||||
|  | ||||
| export const setupListEditor = listId => (dispatch, getState) => { | ||||
|   dispatch({ | ||||
|     type: LIST_EDITOR_SETUP, | ||||
|     list: getState().getIn(['lists', listId]), | ||||
|   }); | ||||
|  | ||||
|   dispatch(fetchListAccounts(listId)); | ||||
| }; | ||||
|  | ||||
| export const changeListEditorTitle = value => ({ | ||||
|   type: LIST_EDITOR_TITLE_CHANGE, | ||||
|   value, | ||||
| }); | ||||
|  | ||||
| export const createList = (title, shouldReset) => (dispatch, getState) => { | ||||
|   dispatch(createListRequest()); | ||||
|  | ||||
|   api(getState).post('/api/v1/lists', { title }).then(({ data }) => { | ||||
|     dispatch(createListSuccess(data)); | ||||
|  | ||||
|     if (shouldReset) { | ||||
|       dispatch(resetListEditor()); | ||||
|     } | ||||
|   }).catch(err => dispatch(createListFail(err))); | ||||
| }; | ||||
|  | ||||
| export const createListRequest = () => ({ | ||||
|   type: LIST_CREATE_REQUEST, | ||||
| }); | ||||
|  | ||||
| export const createListSuccess = list => ({ | ||||
|   type: LIST_CREATE_SUCCESS, | ||||
|   list, | ||||
| }); | ||||
|  | ||||
| export const createListFail = error => ({ | ||||
|   type: LIST_CREATE_FAIL, | ||||
|   error, | ||||
| }); | ||||
|  | ||||
| export const updateList = (id, title, shouldReset) => (dispatch, getState) => { | ||||
|   dispatch(updateListRequest(id)); | ||||
|  | ||||
|   api(getState).put(`/api/v1/lists/${id}`, { title }).then(({ data }) => { | ||||
|     dispatch(updateListSuccess(data)); | ||||
|  | ||||
|     if (shouldReset) { | ||||
|       dispatch(resetListEditor()); | ||||
|     } | ||||
|   }).catch(err => dispatch(updateListFail(id, err))); | ||||
| }; | ||||
|  | ||||
| export const updateListRequest = id => ({ | ||||
|   type: LIST_UPDATE_REQUEST, | ||||
|   id, | ||||
| }); | ||||
|  | ||||
| export const updateListSuccess = list => ({ | ||||
|   type: LIST_UPDATE_SUCCESS, | ||||
|   list, | ||||
| }); | ||||
|  | ||||
| export const updateListFail = (id, error) => ({ | ||||
|   type: LIST_UPDATE_FAIL, | ||||
|   id, | ||||
|   error, | ||||
| }); | ||||
|  | ||||
| export const resetListEditor = () => ({ | ||||
|   type: LIST_EDITOR_RESET, | ||||
| }); | ||||
|  | ||||
| export const deleteList = id => (dispatch, getState) => { | ||||
|   dispatch(deleteListRequest(id)); | ||||
|  | ||||
|   api(getState).delete(`/api/v1/lists/${id}`) | ||||
|     .then(() => dispatch(deleteListSuccess(id))) | ||||
|     .catch(err => dispatch(deleteListFail(id, err))); | ||||
| }; | ||||
|  | ||||
| export const deleteListRequest = id => ({ | ||||
|   type: LIST_DELETE_REQUEST, | ||||
|   id, | ||||
| }); | ||||
|  | ||||
| export const deleteListSuccess = id => ({ | ||||
|   type: LIST_DELETE_SUCCESS, | ||||
|   id, | ||||
| }); | ||||
|  | ||||
| export const deleteListFail = (id, error) => ({ | ||||
|   type: LIST_DELETE_FAIL, | ||||
|   id, | ||||
|   error, | ||||
| }); | ||||
|  | ||||
| export const fetchListAccounts = listId => (dispatch, getState) => { | ||||
|   dispatch(fetchListAccountsRequest(listId)); | ||||
|  | ||||
|   api(getState).get(`/api/v1/lists/${listId}/accounts`, { params: { limit: 0 } }) | ||||
|     .then(({ data }) => dispatch(fetchListAccountsSuccess(listId, data))) | ||||
|     .catch(err => dispatch(fetchListAccountsFail(listId, err))); | ||||
| }; | ||||
|  | ||||
| export const fetchListAccountsRequest = id => ({ | ||||
|   type: LIST_ACCOUNTS_FETCH_REQUEST, | ||||
|   id, | ||||
| }); | ||||
|  | ||||
| export const fetchListAccountsSuccess = (id, accounts, next) => ({ | ||||
|   type: LIST_ACCOUNTS_FETCH_SUCCESS, | ||||
|   id, | ||||
|   accounts, | ||||
|   next, | ||||
| }); | ||||
|  | ||||
| export const fetchListAccountsFail = (id, error) => ({ | ||||
|   type: LIST_ACCOUNTS_FETCH_FAIL, | ||||
|   id, | ||||
|   error, | ||||
| }); | ||||
|  | ||||
| export const fetchListSuggestions = q => (dispatch, getState) => { | ||||
|   const params = { | ||||
|     q, | ||||
|     resolve: false, | ||||
|     limit: 4, | ||||
|     following: true, | ||||
|   }; | ||||
|  | ||||
|   api(getState).get('/api/v1/accounts/search', { params }) | ||||
|     .then(({ data }) => dispatch(fetchListSuggestionsReady(q, data))); | ||||
| }; | ||||
|  | ||||
| export const fetchListSuggestionsReady = (query, accounts) => ({ | ||||
|   type: LIST_EDITOR_SUGGESTIONS_READY, | ||||
|   query, | ||||
|   accounts, | ||||
| }); | ||||
|  | ||||
| export const clearListSuggestions = () => ({ | ||||
|   type: LIST_EDITOR_SUGGESTIONS_CLEAR, | ||||
| }); | ||||
|  | ||||
| export const changeListSuggestions = value => ({ | ||||
|   type: LIST_EDITOR_SUGGESTIONS_CHANGE, | ||||
|   value, | ||||
| }); | ||||
|  | ||||
| export const addToListEditor = accountId => (dispatch, getState) => { | ||||
|   dispatch(addToList(getState().getIn(['listEditor', 'listId']), accountId)); | ||||
| }; | ||||
|  | ||||
| export const addToList = (listId, accountId) => (dispatch, getState) => { | ||||
|   dispatch(addToListRequest(listId, accountId)); | ||||
|  | ||||
|   api(getState).post(`/api/v1/lists/${listId}/accounts`, { account_ids: [accountId] }) | ||||
|     .then(() => dispatch(addToListSuccess(listId, accountId))) | ||||
|     .catch(err => dispatch(addToListFail(listId, accountId, err))); | ||||
| }; | ||||
|  | ||||
| export const addToListRequest = (listId, accountId) => ({ | ||||
|   type: LIST_EDITOR_ADD_REQUEST, | ||||
|   listId, | ||||
|   accountId, | ||||
| }); | ||||
|  | ||||
| export const addToListSuccess = (listId, accountId) => ({ | ||||
|   type: LIST_EDITOR_ADD_SUCCESS, | ||||
|   listId, | ||||
|   accountId, | ||||
| }); | ||||
|  | ||||
| export const addToListFail = (listId, accountId, error) => ({ | ||||
|   type: LIST_EDITOR_ADD_FAIL, | ||||
|   listId, | ||||
|   accountId, | ||||
|   error, | ||||
| }); | ||||
|  | ||||
| export const removeFromListEditor = accountId => (dispatch, getState) => { | ||||
|   dispatch(removeFromList(getState().getIn(['listEditor', 'listId']), accountId)); | ||||
| }; | ||||
|  | ||||
| export const removeFromList = (listId, accountId) => (dispatch, getState) => { | ||||
|   dispatch(removeFromListRequest(listId, accountId)); | ||||
|  | ||||
|   api(getState).delete(`/api/v1/lists/${listId}/accounts`, { params: { account_ids: [accountId] } }) | ||||
|     .then(() => dispatch(removeFromListSuccess(listId, accountId))) | ||||
|     .catch(err => dispatch(removeFromListFail(listId, accountId, err))); | ||||
| }; | ||||
|  | ||||
| export const removeFromListRequest = (listId, accountId) => ({ | ||||
|   type: LIST_EDITOR_REMOVE_REQUEST, | ||||
|   listId, | ||||
|   accountId, | ||||
| }); | ||||
|  | ||||
| export const removeFromListSuccess = (listId, accountId) => ({ | ||||
|   type: LIST_EDITOR_REMOVE_SUCCESS, | ||||
|   listId, | ||||
|   accountId, | ||||
| }); | ||||
|  | ||||
| export const removeFromListFail = (listId, accountId, error) => ({ | ||||
|   type: LIST_EDITOR_REMOVE_FAIL, | ||||
|   listId, | ||||
|   accountId, | ||||
|   error, | ||||
| }); | ||||
| @@ -1,5 +1,6 @@ | ||||
| import api, { getLinks } from '../api'; | ||||
| import { fetchRelationships } from './accounts'; | ||||
| import { openModal } from './modal'; | ||||
|  | ||||
| export const MUTES_FETCH_REQUEST = 'MUTES_FETCH_REQUEST'; | ||||
| export const MUTES_FETCH_SUCCESS = 'MUTES_FETCH_SUCCESS'; | ||||
| @@ -9,6 +10,9 @@ export const MUTES_EXPAND_REQUEST = 'MUTES_EXPAND_REQUEST'; | ||||
| export const MUTES_EXPAND_SUCCESS = 'MUTES_EXPAND_SUCCESS'; | ||||
| export const MUTES_EXPAND_FAIL    = 'MUTES_EXPAND_FAIL'; | ||||
|  | ||||
| export const MUTES_INIT_MODAL = 'MUTES_INIT_MODAL'; | ||||
| export const MUTES_TOGGLE_HIDE_NOTIFICATIONS = 'MUTES_TOGGLE_HIDE_NOTIFICATIONS'; | ||||
|  | ||||
| export function fetchMutes() { | ||||
|   return (dispatch, getState) => { | ||||
|     dispatch(fetchMutesRequest()); | ||||
| @@ -80,3 +84,20 @@ export function expandMutesFail(error) { | ||||
|     error, | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| export function initMuteModal(account) { | ||||
|   return dispatch => { | ||||
|     dispatch({ | ||||
|       type: MUTES_INIT_MODAL, | ||||
|       account, | ||||
|     }); | ||||
|  | ||||
|     dispatch(openModal('MUTE')); | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export function toggleHideNotifications() { | ||||
|   return dispatch => { | ||||
|     dispatch({ type: MUTES_TOGGLE_HIDE_NOTIFICATIONS }); | ||||
|   }; | ||||
| } | ||||
|   | ||||
| @@ -4,12 +4,13 @@ export const PINNED_STATUSES_FETCH_REQUEST = 'PINNED_STATUSES_FETCH_REQUEST'; | ||||
| export const PINNED_STATUSES_FETCH_SUCCESS = 'PINNED_STATUSES_FETCH_SUCCESS'; | ||||
| export const PINNED_STATUSES_FETCH_FAIL = 'PINNED_STATUSES_FETCH_FAIL'; | ||||
|  | ||||
| import { me } from '../initial_state'; | ||||
|  | ||||
| export function fetchPinnedStatuses() { | ||||
|   return (dispatch, getState) => { | ||||
|     dispatch(fetchPinnedStatusesRequest()); | ||||
|  | ||||
|     const accountId = getState().getIn(['meta', 'me']); | ||||
|     api(getState).get(`/api/v1/accounts/${accountId}/statuses`, { params: { pinned: true } }).then(response => { | ||||
|     api(getState).get(`/api/v1/accounts/${me}/statuses`, { params: { pinned: true } }).then(response => { | ||||
|       dispatch(fetchPinnedStatusesSuccess(response.data, null)); | ||||
|     }).catch(error => { | ||||
|       dispatch(fetchPinnedStatusesFail(error)); | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import createStream from '../stream'; | ||||
| import { connectStream } from '../stream'; | ||||
| import { | ||||
|   updateTimeline, | ||||
|   deleteFromTimelines, | ||||
| @@ -12,42 +12,19 @@ import { getLocale } from '../locales'; | ||||
| const { messages } = getLocale(); | ||||
|  | ||||
| export function connectTimelineStream (timelineId, path, pollingRefresh = null) { | ||||
|   return (dispatch, getState) => { | ||||
|     const streamingAPIBaseURL = getState().getIn(['meta', 'streaming_api_base_url']); | ||||
|     const accessToken = getState().getIn(['meta', 'access_token']); | ||||
|  | ||||
|   return connectStream (path, pollingRefresh, (dispatch, getState) => { | ||||
|     const locale = getState().getIn(['meta', 'locale']); | ||||
|     let polling = null; | ||||
|  | ||||
|     const setupPolling = () => { | ||||
|       polling = setInterval(() => { | ||||
|         pollingRefresh(dispatch); | ||||
|       }, 20000); | ||||
|     }; | ||||
|  | ||||
|     const clearPolling = () => { | ||||
|       if (polling) { | ||||
|         clearInterval(polling); | ||||
|         polling = null; | ||||
|       } | ||||
|     }; | ||||
|  | ||||
|     const subscription = createStream(streamingAPIBaseURL, accessToken, path, { | ||||
|  | ||||
|       connected () { | ||||
|         if (pollingRefresh) { | ||||
|           clearPolling(); | ||||
|         } | ||||
|     return { | ||||
|       onConnect() { | ||||
|         dispatch(connectTimeline(timelineId)); | ||||
|       }, | ||||
|  | ||||
|       disconnected () { | ||||
|         if (pollingRefresh) { | ||||
|           setupPolling(); | ||||
|         } | ||||
|       onDisconnect() { | ||||
|         dispatch(disconnectTimeline(timelineId)); | ||||
|       }, | ||||
|  | ||||
|       received (data) { | ||||
|       onReceive (data) { | ||||
|         switch(data.event) { | ||||
|         case 'update': | ||||
|           dispatch(updateTimeline(timelineId, JSON.parse(data.payload))); | ||||
| @@ -60,26 +37,8 @@ export function connectTimelineStream (timelineId, path, pollingRefresh = null) | ||||
|           break; | ||||
|         } | ||||
|       }, | ||||
|  | ||||
|       reconnected () { | ||||
|         if (pollingRefresh) { | ||||
|           clearPolling(); | ||||
|           pollingRefresh(dispatch); | ||||
|         } | ||||
|         dispatch(connectTimeline(timelineId)); | ||||
|       }, | ||||
|  | ||||
|     }; | ||||
|   }); | ||||
|  | ||||
|     const disconnect = () => { | ||||
|       if (subscription) { | ||||
|         subscription.close(); | ||||
|       } | ||||
|       clearPolling(); | ||||
|     }; | ||||
|  | ||||
|     return disconnect; | ||||
|   }; | ||||
| } | ||||
|  | ||||
| function refreshHomeTimelineAndNotification (dispatch) { | ||||
| @@ -92,3 +51,4 @@ export const connectCommunityStream = () => connectTimelineStream('community', ' | ||||
| export const connectMediaStream = () => connectTimelineStream('community', 'public:local'); | ||||
| export const connectPublicStream = () => connectTimelineStream('public', 'public'); | ||||
| export const connectHashtagStream = (tag) => connectTimelineStream(`hashtag:${tag}`, `hashtag&tag=${tag}`); | ||||
| export const connectListStream = (id) => connectTimelineStream(`list:${id}`, `list&list=${id}`); | ||||
|   | ||||
| @@ -118,6 +118,7 @@ export const refreshCommunityTimeline    = () => refreshTimeline('community', '/ | ||||
| export const refreshAccountTimeline      = accountId => refreshTimeline(`account:${accountId}`, `/api/v1/accounts/${accountId}/statuses`); | ||||
| export const refreshAccountMediaTimeline = accountId => refreshTimeline(`account:${accountId}:media`, `/api/v1/accounts/${accountId}/statuses`, { only_media: true }); | ||||
| export const refreshHashtagTimeline      = hashtag => refreshTimeline(`hashtag:${hashtag}`, `/api/v1/timelines/tag/${hashtag}`); | ||||
| export const refreshListTimeline         = id => refreshTimeline(`list:${id}`, `/api/v1/timelines/list/${id}`); | ||||
|  | ||||
| export function refreshTimelineFail(timeline, error, skipLoading) { | ||||
|   return { | ||||
| @@ -158,6 +159,7 @@ export const expandCommunityTimeline    = () => expandTimeline('community', '/ap | ||||
| export const expandAccountTimeline      = accountId => expandTimeline(`account:${accountId}`, `/api/v1/accounts/${accountId}/statuses`); | ||||
| export const expandAccountMediaTimeline = accountId => expandTimeline(`account:${accountId}:media`, `/api/v1/accounts/${accountId}/statuses`, { only_media: true }); | ||||
| export const expandHashtagTimeline      = hashtag => expandTimeline(`hashtag:${hashtag}`, `/api/v1/timelines/tag/${hashtag}`); | ||||
| export const expandListTimeline         = id => expandTimeline(`list:${id}`, `/api/v1/timelines/list/${id}`); | ||||
|  | ||||
| export function expandTimelineRequest(timeline) { | ||||
|   return { | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import Permalink from './permalink'; | ||||
| import IconButton from './icon_button'; | ||||
| import { defineMessages, injectIntl } from 'react-intl'; | ||||
| import ImmutablePureComponent from 'react-immutable-pure-component'; | ||||
| import { me } from '../initial_state'; | ||||
|  | ||||
| const messages = defineMessages({ | ||||
|   follow: { id: 'account.follow', defaultMessage: 'Follow' }, | ||||
| @@ -14,6 +15,8 @@ const messages = defineMessages({ | ||||
|   requested: { id: 'account.requested', defaultMessage: 'Awaiting approval' }, | ||||
|   unblock: { id: 'account.unblock', defaultMessage: 'Unblock @{name}' }, | ||||
|   unmute: { id: 'account.unmute', defaultMessage: 'Unmute @{name}' }, | ||||
|   mute_notifications: { id: 'account.mute_notifications', defaultMessage: 'Mute notifications from @{name}' }, | ||||
|   unmute_notifications: { id: 'account.unmute_notifications', defaultMessage: 'Unmute notifications from @{name}' }, | ||||
| }); | ||||
|  | ||||
| @injectIntl | ||||
| @@ -21,7 +24,6 @@ export default class Account extends ImmutablePureComponent { | ||||
|  | ||||
|   static propTypes = { | ||||
|     account: ImmutablePropTypes.map.isRequired, | ||||
|     me: PropTypes.string.isRequired, | ||||
|     onFollow: PropTypes.func.isRequired, | ||||
|     onBlock: PropTypes.func.isRequired, | ||||
|     onMute: PropTypes.func.isRequired, | ||||
| @@ -41,8 +43,16 @@ export default class Account extends ImmutablePureComponent { | ||||
|     this.props.onMute(this.props.account); | ||||
|   } | ||||
|  | ||||
|   handleMuteNotifications = () => { | ||||
|     this.props.onMuteNotifications(this.props.account, true); | ||||
|   } | ||||
|  | ||||
|   handleUnmuteNotifications = () => { | ||||
|     this.props.onMuteNotifications(this.props.account, false); | ||||
|   } | ||||
|  | ||||
|   render () { | ||||
|     const { account, me, intl, hidden } = this.props; | ||||
|     const { account, intl, hidden } = this.props; | ||||
|  | ||||
|     if (!account) { | ||||
|       return <div />; | ||||
| @@ -70,7 +80,18 @@ export default class Account extends ImmutablePureComponent { | ||||
|       } else if (blocking) { | ||||
|         buttons = <IconButton active icon='unlock-alt' title={intl.formatMessage(messages.unblock, { name: account.get('username') })} onClick={this.handleBlock} />; | ||||
|       } else if (muting) { | ||||
|         buttons = <IconButton active icon='volume-up' title={intl.formatMessage(messages.unmute, { name: account.get('username') })} onClick={this.handleMute} />; | ||||
|         let hidingNotificationsButton; | ||||
|         if (account.getIn(['relationship', 'muting_notifications'])) { | ||||
|           hidingNotificationsButton = <IconButton active icon='bell' title={intl.formatMessage(messages.unmute_notifications, { name: account.get('username') })} onClick={this.handleUnmuteNotifications} />; | ||||
|         } else { | ||||
|           hidingNotificationsButton = <IconButton active icon='bell-slash' title={intl.formatMessage(messages.mute_notifications, { name: account.get('username')  })} onClick={this.handleMuteNotifications} />; | ||||
|         } | ||||
|         buttons = ( | ||||
|           <div> | ||||
|             <IconButton active icon='volume-up' title={intl.formatMessage(messages.unmute, { name: account.get('username') })} onClick={this.handleMute} /> | ||||
|             {hidingNotificationsButton} | ||||
|           </div> | ||||
|         ); | ||||
|       } else { | ||||
|         buttons = <IconButton icon={following ? 'user-times' : 'user-plus'} title={intl.formatMessage(following ? messages.unfollow : messages.follow)} onClick={this.handleFollow} active={following} />; | ||||
|       } | ||||
|   | ||||
| @@ -209,6 +209,7 @@ export default class AutosuggestTextarea extends ImmutablePureComponent { | ||||
|             onBlur={this.onBlur} | ||||
|             onPaste={this.onPaste} | ||||
|             style={style} | ||||
|             aria-autocomplete='list' | ||||
|           /> | ||||
|         </label> | ||||
|  | ||||
|   | ||||
| @@ -137,7 +137,9 @@ export default class ColumnHeader extends React.PureComponent { | ||||
|       <div className={wrapperClassName}> | ||||
|         <h1 tabIndex={focusable ? 0 : null} role='button' className={buttonClassName} aria-label={title} onClick={this.handleTitleClick}> | ||||
|           <i className={`fa fa-fw fa-${icon} column-header__icon`} /> | ||||
|           <span className='column-header__title'> | ||||
|             {title} | ||||
|           </span> | ||||
|  | ||||
|           <div className='column-header__buttons'> | ||||
|             {backButton} | ||||
|   | ||||
| @@ -110,7 +110,7 @@ export default class Dropdown extends React.PureComponent { | ||||
|     icon: PropTypes.string.isRequired, | ||||
|     items: PropTypes.array.isRequired, | ||||
|     size: PropTypes.number.isRequired, | ||||
|     ariaLabel: PropTypes.string, | ||||
|     title: PropTypes.string, | ||||
|     disabled: PropTypes.bool, | ||||
|     status: ImmutablePropTypes.map, | ||||
|     isUserTouching: PropTypes.func, | ||||
| @@ -120,7 +120,7 @@ export default class Dropdown extends React.PureComponent { | ||||
|   }; | ||||
|  | ||||
|   static defaultProps = { | ||||
|     ariaLabel: 'Menu', | ||||
|     title: 'Menu', | ||||
|   }; | ||||
|  | ||||
|   state = { | ||||
| @@ -186,14 +186,14 @@ export default class Dropdown extends React.PureComponent { | ||||
|   } | ||||
|  | ||||
|   render () { | ||||
|     const { icon, items, size, ariaLabel, disabled } = this.props; | ||||
|     const { icon, items, size, title, disabled } = this.props; | ||||
|     const { expanded } = this.state; | ||||
|  | ||||
|     return ( | ||||
|       <div onKeyDown={this.handleKeyDown}> | ||||
|         <IconButton | ||||
|           icon={icon} | ||||
|           title={ariaLabel} | ||||
|           title={title} | ||||
|           active={expanded} | ||||
|           disabled={disabled} | ||||
|           size={size} | ||||
|   | ||||
| @@ -72,6 +72,25 @@ export default class IconButton extends React.PureComponent { | ||||
|       overlayed: overlay, | ||||
|     }); | ||||
|  | ||||
|     if (!animate) { | ||||
|       // Perf optimization: avoid unnecessary <Motion> components unless | ||||
|       // we actually need to animate. | ||||
|       return ( | ||||
|         <button | ||||
|           aria-label={title} | ||||
|           aria-pressed={pressed} | ||||
|           aria-expanded={expanded} | ||||
|           title={title} | ||||
|           className={classes} | ||||
|           onClick={this.handleClick} | ||||
|           style={style} | ||||
|           tabIndex={tabIndex} | ||||
|         > | ||||
|           <i className={`fa fa-fw fa-${icon}`} aria-hidden='true' /> | ||||
|         </button> | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     return ( | ||||
|       <Motion defaultStyle={{ rotate: active ? -360 : 0 }} style={{ rotate: animate ? spring(active ? -360 : 0, { stiffness: 120, damping: 7 }) : 0 }}> | ||||
|         {({ rotate }) => | ||||
|   | ||||
| @@ -6,6 +6,7 @@ import IconButton from './icon_button'; | ||||
| import { defineMessages, injectIntl, FormattedMessage } from 'react-intl'; | ||||
| import { isIOS } from '../is_mobile'; | ||||
| import classNames from 'classnames'; | ||||
| import { autoPlayGif } from '../initial_state'; | ||||
|  | ||||
| const messages = defineMessages({ | ||||
|   toggle_visible: { id: 'media_gallery.toggle_visible', defaultMessage: 'Toggle visibility' }, | ||||
| @@ -23,11 +24,9 @@ class Item extends React.PureComponent { | ||||
|     index: PropTypes.number.isRequired, | ||||
|     size: PropTypes.number.isRequired, | ||||
|     onClick: PropTypes.func.isRequired, | ||||
|     autoPlayGif: PropTypes.bool, | ||||
|   }; | ||||
|  | ||||
|   static defaultProps = { | ||||
|     autoPlayGif: false, | ||||
|     standalone: false, | ||||
|     index: 0, | ||||
|     size: 1, | ||||
| @@ -47,7 +46,7 @@ class Item extends React.PureComponent { | ||||
|   } | ||||
|  | ||||
|   hoverToPlay () { | ||||
|     const { attachment, autoPlayGif } = this.props; | ||||
|     const { attachment } = this.props; | ||||
|     return !autoPlayGif && attachment.get('type') === 'gifv'; | ||||
|   } | ||||
|  | ||||
| @@ -139,7 +138,7 @@ class Item extends React.PureComponent { | ||||
|         </a> | ||||
|       ); | ||||
|     } else if (attachment.get('type') === 'gifv') { | ||||
|       const autoPlay = !isIOS() && this.props.autoPlayGif; | ||||
|       const autoPlay = !isIOS() && autoPlayGif; | ||||
|  | ||||
|       thumbnail = ( | ||||
|         <div className={classNames('media-gallery__gifv', { autoplay: autoPlay })}> | ||||
| @@ -181,11 +180,9 @@ export default class MediaGallery extends React.PureComponent { | ||||
|     height: PropTypes.number.isRequired, | ||||
|     onOpenMedia: PropTypes.func.isRequired, | ||||
|     intl: PropTypes.object.isRequired, | ||||
|     autoPlayGif: PropTypes.bool, | ||||
|   }; | ||||
|  | ||||
|   static defaultProps = { | ||||
|     autoPlayGif: false, | ||||
|     standalone: false, | ||||
|   }; | ||||
|  | ||||
| @@ -261,9 +258,9 @@ export default class MediaGallery extends React.PureComponent { | ||||
|       const size = media.take(4).size; | ||||
|  | ||||
|       if (this.isStandaloneEligible()) { | ||||
|         children = <Item standalone onClick={this.handleClick} attachment={media.get(0)} autoPlayGif={this.props.autoPlayGif} />; | ||||
|         children = <Item standalone onClick={this.handleClick} attachment={media.get(0)} />; | ||||
|       } else { | ||||
|         children = media.take(4).map((attachment, i) => <Item key={attachment.get('id')} onClick={this.handleClick} attachment={attachment} autoPlayGif={this.props.autoPlayGif} index={i} size={size} />); | ||||
|         children = media.take(4).map((attachment, i) => <Item key={attachment.get('id')} onClick={this.handleClick} attachment={attachment} index={i} size={size} />); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React, { PureComponent } from 'react'; | ||||
| import { ScrollContainer } from 'react-router-scroll'; | ||||
| import { ScrollContainer } from 'react-router-scroll-4'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import IntersectionObserverArticleContainer from '../containers/intersection_observer_article_container'; | ||||
| import LoadMore from './load_more'; | ||||
|   | ||||
| @@ -36,9 +36,6 @@ export default class Status extends ImmutablePureComponent { | ||||
|     onBlock: PropTypes.func, | ||||
|     onEmbed: PropTypes.func, | ||||
|     onHeightChange: PropTypes.func, | ||||
|     me: PropTypes.string, | ||||
|     boostModal: PropTypes.bool, | ||||
|     autoPlayGif: PropTypes.bool, | ||||
|     muted: PropTypes.bool, | ||||
|     hidden: PropTypes.bool, | ||||
|     onMoveUp: PropTypes.func, | ||||
| @@ -54,9 +51,6 @@ export default class Status extends ImmutablePureComponent { | ||||
|   updateOnProps = [ | ||||
|     'status', | ||||
|     'account', | ||||
|     'me', | ||||
|     'boostModal', | ||||
|     'autoPlayGif', | ||||
|     'muted', | ||||
|     'hidden', | ||||
|   ] | ||||
| @@ -197,7 +191,7 @@ export default class Status extends ImmutablePureComponent { | ||||
|       } else { | ||||
|         media = ( | ||||
|           <Bundle fetchComponent={MediaGallery} loading={this.renderLoadingMediaGallery} > | ||||
|             {Component => <Component media={status.get('media_attachments')} sensitive={status.get('sensitive')} height={110} onOpenMedia={this.props.onOpenMedia} autoPlayGif={this.props.autoPlayGif} />} | ||||
|             {Component => <Component media={status.get('media_attachments')} sensitive={status.get('sensitive')} height={110} onOpenMedia={this.props.onOpenMedia} />} | ||||
|           </Bundle> | ||||
|         ); | ||||
|       } | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import IconButton from './icon_button'; | ||||
| import DropdownMenuContainer from '../containers/dropdown_menu_container'; | ||||
| import { defineMessages, injectIntl } from 'react-intl'; | ||||
| import ImmutablePureComponent from 'react-immutable-pure-component'; | ||||
| import { me } from '../initial_state'; | ||||
|  | ||||
| const messages = defineMessages({ | ||||
|   delete: { id: 'status.delete', defaultMessage: 'Delete' }, | ||||
| @@ -47,7 +48,6 @@ export default class StatusActionBar extends ImmutablePureComponent { | ||||
|     onEmbed: PropTypes.func, | ||||
|     onMuteConversation: PropTypes.func, | ||||
|     onPin: PropTypes.func, | ||||
|     me: PropTypes.string, | ||||
|     withDismiss: PropTypes.bool, | ||||
|     intl: PropTypes.object.isRequired, | ||||
|   }; | ||||
| @@ -56,7 +56,6 @@ export default class StatusActionBar extends ImmutablePureComponent { | ||||
|   // evaluate to false. See react-immutable-pure-component for usage. | ||||
|   updateOnProps = [ | ||||
|     'status', | ||||
|     'me', | ||||
|     'withDismiss', | ||||
|   ] | ||||
|  | ||||
| @@ -116,7 +115,7 @@ export default class StatusActionBar extends ImmutablePureComponent { | ||||
|   } | ||||
|  | ||||
|   render () { | ||||
|     const { status, me, intl, withDismiss } = this.props; | ||||
|     const { status, intl, withDismiss } = this.props; | ||||
|  | ||||
|     const mutingConversation = status.get('muted'); | ||||
|     const anonymousAccess    = !me; | ||||
| @@ -180,7 +179,7 @@ export default class StatusActionBar extends ImmutablePureComponent { | ||||
|         {shareButton} | ||||
|  | ||||
|         <div className='status__action-bar-dropdown'> | ||||
|           <DropdownMenuContainer disabled={anonymousAccess} status={status} items={menu} icon='ellipsis-h' size={18} direction='right' ariaLabel={intl.formatMessage(messages.more)} /> | ||||
|           <DropdownMenuContainer disabled={anonymousAccess} status={status} items={menu} icon='ellipsis-h' size={18} direction='right' title={intl.formatMessage(messages.more)} /> | ||||
|         </div> | ||||
|       </div> | ||||
|     ); | ||||
|   | ||||
| @@ -12,6 +12,8 @@ import { | ||||
|   unmuteAccount, | ||||
| } from '../actions/accounts'; | ||||
| import { openModal } from '../actions/modal'; | ||||
| import { initMuteModal } from '../actions/mutes'; | ||||
| import { unfollowModal } from '../initial_state'; | ||||
|  | ||||
| const messages = defineMessages({ | ||||
|   unfollowConfirm: { id: 'confirmations.unfollow.confirm', defaultMessage: 'Unfollow' }, | ||||
| @@ -22,8 +24,6 @@ const makeMapStateToProps = () => { | ||||
|  | ||||
|   const mapStateToProps = (state, props) => ({ | ||||
|     account: getAccount(state, props.id), | ||||
|     me: state.getIn(['meta', 'me']), | ||||
|     unfollowModal: state.getIn(['meta', 'unfollow_modal']), | ||||
|   }); | ||||
|  | ||||
|   return mapStateToProps; | ||||
| @@ -33,7 +33,7 @@ const mapDispatchToProps = (dispatch, { intl }) => ({ | ||||
|  | ||||
|   onFollow (account) { | ||||
|     if (account.getIn(['relationship', 'following']) || account.getIn(['relationship', 'requested'])) { | ||||
|       if (this.unfollowModal) { | ||||
|       if (unfollowModal) { | ||||
|         dispatch(openModal('CONFIRM', { | ||||
|           message: <FormattedMessage id='confirmations.unfollow.message' defaultMessage='Are you sure you want to unfollow {name}?' values={{ name: <strong>@{account.get('acct')}</strong> }} />, | ||||
|           confirm: intl.formatMessage(messages.unfollowConfirm), | ||||
| @@ -59,10 +59,14 @@ const mapDispatchToProps = (dispatch, { intl }) => ({ | ||||
|     if (account.getIn(['relationship', 'muting'])) { | ||||
|       dispatch(unmuteAccount(account.get('id'))); | ||||
|     } else { | ||||
|       dispatch(muteAccount(account.get('id'))); | ||||
|       dispatch(initMuteModal(account)); | ||||
|     } | ||||
|   }, | ||||
|  | ||||
|  | ||||
|   onMuteNotifications (account, notifications) { | ||||
|     dispatch(muteAccount(account.get('id'), notifications)); | ||||
|   }, | ||||
| }); | ||||
|  | ||||
| export default injectIntl(connect(makeMapStateToProps, mapDispatchToProps)(Account)); | ||||
|   | ||||
| @@ -6,15 +6,14 @@ import { hydrateStore } from '../actions/store'; | ||||
| import { IntlProvider, addLocaleData } from 'react-intl'; | ||||
| import { getLocale } from '../locales'; | ||||
| import Compose from '../features/standalone/compose'; | ||||
| import initialState from '../initial_state'; | ||||
|  | ||||
| const { localeData, messages } = getLocale(); | ||||
| addLocaleData(localeData); | ||||
|  | ||||
| const store = configureStore(); | ||||
| const initialStateContainer = document.getElementById('initial-state'); | ||||
|  | ||||
| if (initialStateContainer !== null) { | ||||
|   const initialState = JSON.parse(initialStateContainer.textContent); | ||||
| if (initialState) { | ||||
|   store.dispatch(hydrateStore(initialState)); | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -4,18 +4,19 @@ import PropTypes from 'prop-types'; | ||||
| import configureStore from '../store/configureStore'; | ||||
| import { showOnboardingOnce } from '../actions/onboarding'; | ||||
| import { BrowserRouter, Route } from 'react-router-dom'; | ||||
| import { ScrollContext } from 'react-router-scroll'; | ||||
| import { ScrollContext } from 'react-router-scroll-4'; | ||||
| import UI from '../features/ui'; | ||||
| import { hydrateStore } from '../actions/store'; | ||||
| import { connectUserStream } from '../actions/streaming'; | ||||
| import { IntlProvider, addLocaleData } from 'react-intl'; | ||||
| import { getLocale } from '../locales'; | ||||
| import initialState from '../initial_state'; | ||||
|  | ||||
| const { localeData, messages } = getLocale(); | ||||
| addLocaleData(localeData); | ||||
|  | ||||
| export const store = configureStore(); | ||||
| const hydrateAction = hydrateStore(JSON.parse(document.getElementById('initial-state').textContent)); | ||||
| const hydrateAction = hydrateStore(initialState); | ||||
| store.dispatch(hydrateAction); | ||||
|  | ||||
| export default class Mastodon extends React.PureComponent { | ||||
|   | ||||
| @@ -14,20 +14,18 @@ import { | ||||
|   pin, | ||||
|   unpin, | ||||
| } from '../actions/interactions'; | ||||
| import { | ||||
|   blockAccount, | ||||
|   muteAccount, | ||||
| } from '../actions/accounts'; | ||||
| import { blockAccount } from '../actions/accounts'; | ||||
| import { muteStatus, unmuteStatus, deleteStatus } from '../actions/statuses'; | ||||
| import { initMuteModal } from '../actions/mutes'; | ||||
| import { initReport } from '../actions/reports'; | ||||
| import { openModal } from '../actions/modal'; | ||||
| import { defineMessages, injectIntl, FormattedMessage } from 'react-intl'; | ||||
| import { boostModal, deleteModal } from '../initial_state'; | ||||
|  | ||||
| const messages = defineMessages({ | ||||
|   deleteConfirm: { id: 'confirmations.delete.confirm', defaultMessage: 'Delete' }, | ||||
|   deleteMessage: { id: 'confirmations.delete.message', defaultMessage: 'Are you sure you want to delete this status?' }, | ||||
|   blockConfirm: { id: 'confirmations.block.confirm', defaultMessage: 'Block' }, | ||||
|   muteConfirm: { id: 'confirmations.mute.confirm', defaultMessage: 'Mute' }, | ||||
| }); | ||||
|  | ||||
| const makeMapStateToProps = () => { | ||||
| @@ -35,10 +33,6 @@ const makeMapStateToProps = () => { | ||||
|  | ||||
|   const mapStateToProps = (state, props) => ({ | ||||
|     status: getStatus(state, props.id), | ||||
|     me: state.getIn(['meta', 'me']), | ||||
|     boostModal: state.getIn(['meta', 'boost_modal']), | ||||
|     deleteModal: state.getIn(['meta', 'delete_modal']), | ||||
|     autoPlayGif: state.getIn(['meta', 'auto_play_gif']), | ||||
|   }); | ||||
|  | ||||
|   return mapStateToProps; | ||||
| @@ -58,7 +52,7 @@ const mapDispatchToProps = (dispatch, { intl }) => ({ | ||||
|     if (status.get('reblogged')) { | ||||
|       dispatch(unreblog(status)); | ||||
|     } else { | ||||
|       if (e.shiftKey || !this.boostModal) { | ||||
|       if (e.shiftKey || !boostModal) { | ||||
|         this.onModalReblog(status); | ||||
|       } else { | ||||
|         dispatch(openModal('BOOST', { status, onReblog: this.onModalReblog })); | ||||
| @@ -87,7 +81,7 @@ const mapDispatchToProps = (dispatch, { intl }) => ({ | ||||
|   }, | ||||
|  | ||||
|   onDelete (status) { | ||||
|     if (!this.deleteModal) { | ||||
|     if (!deleteModal) { | ||||
|       dispatch(deleteStatus(status.get('id'))); | ||||
|     } else { | ||||
|       dispatch(openModal('CONFIRM', { | ||||
| @@ -123,11 +117,7 @@ const mapDispatchToProps = (dispatch, { intl }) => ({ | ||||
|   }, | ||||
|  | ||||
|   onMute (account) { | ||||
|     dispatch(openModal('CONFIRM', { | ||||
|       message: <FormattedMessage id='confirmations.mute.message' defaultMessage='Are you sure you want to mute {name}?' values={{ name: <strong>@{account.get('acct')}</strong> }} />, | ||||
|       confirm: intl.formatMessage(messages.muteConfirm), | ||||
|       onConfirm: () => dispatch(muteAccount(account.get('id'))), | ||||
|     })); | ||||
|     dispatch(initMuteModal(account)); | ||||
|   }, | ||||
|  | ||||
|   onMuteConversation (status) { | ||||
|   | ||||
| @@ -7,15 +7,14 @@ import { IntlProvider, addLocaleData } from 'react-intl'; | ||||
| import { getLocale } from '../locales'; | ||||
| import PublicTimeline from '../features/standalone/public_timeline'; | ||||
| import HashtagTimeline from '../features/standalone/hashtag_timeline'; | ||||
| import initialState from '../initial_state'; | ||||
|  | ||||
| const { localeData, messages } = getLocale(); | ||||
| addLocaleData(localeData); | ||||
|  | ||||
| const store = configureStore(); | ||||
| const initialStateContainer = document.getElementById('initial-state'); | ||||
|  | ||||
| if (initialStateContainer !== null) { | ||||
|   const initialState = JSON.parse(initialStateContainer.textContent); | ||||
| if (initialState) { | ||||
|   store.dispatch(hydrateStore(initialState)); | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -4,6 +4,7 @@ import PropTypes from 'prop-types'; | ||||
| import DropdownMenuContainer from '../../../containers/dropdown_menu_container'; | ||||
| import { Link } from 'react-router-dom'; | ||||
| import { defineMessages, injectIntl, FormattedMessage, FormattedNumber } from 'react-intl'; | ||||
| import { me } from '../../../initial_state'; | ||||
|  | ||||
| const messages = defineMessages({ | ||||
|   mention: { id: 'account.mention', defaultMessage: 'Mention @{name}' }, | ||||
| @@ -19,6 +20,8 @@ const messages = defineMessages({ | ||||
|   media: { id: 'account.media', defaultMessage: 'Media' }, | ||||
|   blockDomain: { id: 'account.block_domain', defaultMessage: 'Hide everything from {domain}' }, | ||||
|   unblockDomain: { id: 'account.unblock_domain', defaultMessage: 'Unhide {domain}' }, | ||||
|   hideReblogs: { id: 'account.hide_reblogs', defaultMessage: 'Hide boosts from @{name}' }, | ||||
|   showReblogs: { id: 'account.show_reblogs', defaultMessage: 'Show boosts from @{name}' }, | ||||
| }); | ||||
|  | ||||
| @injectIntl | ||||
| @@ -26,10 +29,10 @@ export default class ActionBar extends React.PureComponent { | ||||
|  | ||||
|   static propTypes = { | ||||
|     account: ImmutablePropTypes.map.isRequired, | ||||
|     me: PropTypes.string.isRequired, | ||||
|     onFollow: PropTypes.func, | ||||
|     onBlock: PropTypes.func.isRequired, | ||||
|     onMention: PropTypes.func.isRequired, | ||||
|     onReblogToggle: PropTypes.func.isRequired, | ||||
|     onReport: PropTypes.func.isRequired, | ||||
|     onMute: PropTypes.func.isRequired, | ||||
|     onBlockDomain: PropTypes.func.isRequired, | ||||
| @@ -44,7 +47,7 @@ export default class ActionBar extends React.PureComponent { | ||||
|   } | ||||
|  | ||||
|   render () { | ||||
|     const { account, me, intl } = this.props; | ||||
|     const { account, intl } = this.props; | ||||
|  | ||||
|     let menu = []; | ||||
|     let extraInfo = ''; | ||||
| @@ -60,6 +63,14 @@ export default class ActionBar extends React.PureComponent { | ||||
|     if (account.get('id') === me) { | ||||
|       menu.push({ text: intl.formatMessage(messages.edit_profile), href: '/settings/profile' }); | ||||
|     } else { | ||||
|       if (account.getIn(['relationship', 'following'])) { | ||||
|         if (account.getIn(['relationship', 'showing_reblogs'])) { | ||||
|           menu.push({ text: intl.formatMessage(messages.hideReblogs, { name: account.get('username') }), action: this.props.onReblogToggle }); | ||||
|         } else { | ||||
|           menu.push({ text: intl.formatMessage(messages.showReblogs, { name: account.get('username') }), action: this.props.onReblogToggle }); | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       if (account.getIn(['relationship', 'muting'])) { | ||||
|         menu.push({ text: intl.formatMessage(messages.unmute, { name: account.get('username') }), action: this.props.onMute }); | ||||
|       } else { | ||||
|   | ||||
| @@ -5,8 +5,9 @@ import { defineMessages, injectIntl, FormattedMessage } from 'react-intl'; | ||||
| import IconButton from '../../../components/icon_button'; | ||||
| import Motion from '../../ui/util/optional_motion'; | ||||
| import spring from 'react-motion/lib/spring'; | ||||
| import { connect } from 'react-redux'; | ||||
| import ImmutablePureComponent from 'react-immutable-pure-component'; | ||||
| import { autoPlayGif, me } from '../../../initial_state'; | ||||
| import classNames from 'classnames'; | ||||
|  | ||||
| const messages = defineMessages({ | ||||
|   unfollow: { id: 'account.unfollow', defaultMessage: 'Unfollow' }, | ||||
| @@ -14,19 +15,10 @@ const messages = defineMessages({ | ||||
|   requested: { id: 'account.requested', defaultMessage: 'Awaiting approval. Click to cancel follow request' }, | ||||
| }); | ||||
|  | ||||
| const makeMapStateToProps = () => { | ||||
|   const mapStateToProps = state => ({ | ||||
|     autoPlayGif: state.getIn(['meta', 'auto_play_gif']), | ||||
|   }); | ||||
|  | ||||
|   return mapStateToProps; | ||||
| }; | ||||
|  | ||||
| class Avatar extends ImmutablePureComponent { | ||||
|  | ||||
|   static propTypes = { | ||||
|     account: ImmutablePropTypes.map.isRequired, | ||||
|     autoPlayGif: PropTypes.bool.isRequired, | ||||
|   }; | ||||
|  | ||||
|   state = { | ||||
| @@ -44,7 +36,7 @@ class Avatar extends ImmutablePureComponent { | ||||
|   } | ||||
|  | ||||
|   render () { | ||||
|     const { account, autoPlayGif }   = this.props; | ||||
|     const { account }   = this.props; | ||||
|     const { isHovered } = this.state; | ||||
|  | ||||
|     return ( | ||||
| @@ -71,20 +63,17 @@ class Avatar extends ImmutablePureComponent { | ||||
|  | ||||
| } | ||||
|  | ||||
| @connect(makeMapStateToProps) | ||||
| @injectIntl | ||||
| export default class Header extends ImmutablePureComponent { | ||||
|  | ||||
|   static propTypes = { | ||||
|     account: ImmutablePropTypes.map, | ||||
|     me: PropTypes.string.isRequired, | ||||
|     onFollow: PropTypes.func.isRequired, | ||||
|     intl: PropTypes.object.isRequired, | ||||
|     autoPlayGif: PropTypes.bool.isRequired, | ||||
|   }; | ||||
|  | ||||
|   render () { | ||||
|     const { account, me, intl } = this.props; | ||||
|     const { account, intl } = this.props; | ||||
|  | ||||
|     if (!account) { | ||||
|       return null; | ||||
| @@ -114,6 +103,10 @@ export default class Header extends ImmutablePureComponent { | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     if (account.get('moved')) { | ||||
|       actionBtn = ''; | ||||
|     } | ||||
|  | ||||
|     if (account.get('locked')) { | ||||
|       lockedIcon = <i className='fa fa-lock' />; | ||||
|     } | ||||
| @@ -122,9 +115,9 @@ export default class Header extends ImmutablePureComponent { | ||||
|     const displayNameHtml = { __html: account.get('display_name_html') }; | ||||
|  | ||||
|     return ( | ||||
|       <div className='account__header' style={{ backgroundImage: `url(${account.get('header')})` }}> | ||||
|       <div className={classNames('account__header', { inactive: !!account.get('moved') })} style={{ backgroundImage: `url(${account.get('header')})` }}> | ||||
|         <div> | ||||
|           <Avatar account={account} autoPlayGif={this.props.autoPlayGif} /> | ||||
|           <Avatar account={account} /> | ||||
|  | ||||
|           <span className='account__header__display-name' dangerouslySetInnerHTML={displayNameHtml} /> | ||||
|           <span className='account__header__username'>@{account.get('acct')} {lockedIcon}</span> | ||||
|   | ||||
| @@ -12,14 +12,13 @@ import { getAccountGallery } from '../../selectors'; | ||||
| import MediaItem from './components/media_item'; | ||||
| import HeaderContainer from '../account_timeline/containers/header_container'; | ||||
| import { FormattedMessage } from 'react-intl'; | ||||
| import { ScrollContainer } from 'react-router-scroll'; | ||||
| import { ScrollContainer } from 'react-router-scroll-4'; | ||||
| import LoadMore from '../../components/load_more'; | ||||
|  | ||||
| const mapStateToProps = (state, props) => ({ | ||||
|   medias: getAccountGallery(state, props.params.accountId), | ||||
|   isLoading: state.getIn(['timelines', `account:${props.params.accountId}:media`, 'isLoading']), | ||||
|   hasMore: !!state.getIn(['timelines', `account:${props.params.accountId}:media`, 'next']), | ||||
|   autoPlayGif: state.getIn(['meta', 'auto_play_gif']), | ||||
| }); | ||||
|  | ||||
| @connect(mapStateToProps) | ||||
| @@ -31,7 +30,6 @@ export default class AccountGallery extends ImmutablePureComponent { | ||||
|     medias: ImmutablePropTypes.list.isRequired, | ||||
|     isLoading: PropTypes.bool, | ||||
|     hasMore: PropTypes.bool, | ||||
|     autoPlayGif: PropTypes.bool, | ||||
|   }; | ||||
|  | ||||
|   componentDidMount () { | ||||
| @@ -67,7 +65,7 @@ export default class AccountGallery extends ImmutablePureComponent { | ||||
|   } | ||||
|  | ||||
|   render () { | ||||
|     const { medias, autoPlayGif, isLoading, hasMore } = this.props; | ||||
|     const { medias, isLoading, hasMore } = this.props; | ||||
|  | ||||
|     let loadMore = null; | ||||
|  | ||||
| @@ -100,7 +98,6 @@ export default class AccountGallery extends ImmutablePureComponent { | ||||
|                 <MediaItem | ||||
|                   key={media.get('id')} | ||||
|                   media={media} | ||||
|                   autoPlayGif={autoPlayGif} | ||||
|                 /> | ||||
|               )} | ||||
|               {loadMore} | ||||
|   | ||||
| @@ -5,15 +5,16 @@ import InnerHeader from '../../account/components/header'; | ||||
| import ActionBar from '../../account/components/action_bar'; | ||||
| import MissingIndicator from '../../../components/missing_indicator'; | ||||
| import ImmutablePureComponent from 'react-immutable-pure-component'; | ||||
| import MovedNote from './moved_note'; | ||||
|  | ||||
| export default class Header extends ImmutablePureComponent { | ||||
|  | ||||
|   static propTypes = { | ||||
|     account: ImmutablePropTypes.map, | ||||
|     me: PropTypes.string.isRequired, | ||||
|     onFollow: PropTypes.func.isRequired, | ||||
|     onBlock: PropTypes.func.isRequired, | ||||
|     onMention: PropTypes.func.isRequired, | ||||
|     onReblogToggle: PropTypes.func.isRequired, | ||||
|     onReport: PropTypes.func.isRequired, | ||||
|     onMute: PropTypes.func.isRequired, | ||||
|     onBlockDomain: PropTypes.func.isRequired, | ||||
| @@ -40,6 +41,10 @@ export default class Header extends ImmutablePureComponent { | ||||
|     this.props.onReport(this.props.account); | ||||
|   } | ||||
|  | ||||
|   handleReblogToggle = () => { | ||||
|     this.props.onReblogToggle(this.props.account); | ||||
|   } | ||||
|  | ||||
|   handleMute = () => { | ||||
|     this.props.onMute(this.props.account); | ||||
|   } | ||||
| @@ -61,7 +66,7 @@ export default class Header extends ImmutablePureComponent { | ||||
|   } | ||||
|  | ||||
|   render () { | ||||
|     const { account, me } = this.props; | ||||
|     const { account } = this.props; | ||||
|  | ||||
|     if (account === null) { | ||||
|       return <MissingIndicator />; | ||||
| @@ -69,17 +74,18 @@ export default class Header extends ImmutablePureComponent { | ||||
|  | ||||
|     return ( | ||||
|       <div className='account-timeline__header'> | ||||
|         {account.get('moved') && <MovedNote from={account} to={account.get('moved')} />} | ||||
|  | ||||
|         <InnerHeader | ||||
|           account={account} | ||||
|           me={me} | ||||
|           onFollow={this.handleFollow} | ||||
|         /> | ||||
|  | ||||
|         <ActionBar | ||||
|           account={account} | ||||
|           me={me} | ||||
|           onBlock={this.handleBlock} | ||||
|           onMention={this.handleMention} | ||||
|           onReblogToggle={this.handleReblogToggle} | ||||
|           onReport={this.handleReport} | ||||
|           onMute={this.handleMute} | ||||
|           onBlockDomain={this.handleBlockDomain} | ||||
|   | ||||
| @@ -0,0 +1,48 @@ | ||||
| import React from 'react'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import ImmutablePropTypes from 'react-immutable-proptypes'; | ||||
| import { FormattedMessage } from 'react-intl'; | ||||
| import ImmutablePureComponent from 'react-immutable-pure-component'; | ||||
| import AvatarOverlay from '../../../components/avatar_overlay'; | ||||
| import DisplayName from '../../../components/display_name'; | ||||
|  | ||||
| export default class MovedNote extends ImmutablePureComponent { | ||||
|  | ||||
|   static contextTypes = { | ||||
|     router: PropTypes.object, | ||||
|   }; | ||||
|  | ||||
|   static propTypes = { | ||||
|     from: ImmutablePropTypes.map.isRequired, | ||||
|     to: ImmutablePropTypes.map.isRequired, | ||||
|   }; | ||||
|  | ||||
|   handleAccountClick = e => { | ||||
|     if (e.button === 0) { | ||||
|       e.preventDefault(); | ||||
|       this.context.router.history.push(`/accounts/${this.props.to.get('id')}`); | ||||
|     } | ||||
|  | ||||
|     e.stopPropagation(); | ||||
|   } | ||||
|  | ||||
|   render () { | ||||
|     const { from, to } = this.props; | ||||
|     const displayNameHtml = { __html: from.get('display_name_html') }; | ||||
|  | ||||
|     return ( | ||||
|       <div className='account__moved-note'> | ||||
|         <div className='account__moved-note__message'> | ||||
|           <div className='account__moved-note__icon-wrapper'><i className='fa fa-fw fa-suitcase account__moved-note__icon' /></div> | ||||
|           <FormattedMessage id='account.moved_to' defaultMessage='{name} has moved to:' values={{ name: <strong dangerouslySetInnerHTML={displayNameHtml} /> }} /> | ||||
|         </div> | ||||
|  | ||||
|         <a href={to.get('url')} onClick={this.handleAccountClick} className='detailed-status__display-name'> | ||||
|           <div className='detailed-status__display-avatar'><AvatarOverlay account={to} friend={from} /></div> | ||||
|           <DisplayName account={to} /> | ||||
|         </a> | ||||
|       </div> | ||||
|     ); | ||||
|   } | ||||
|  | ||||
| } | ||||
| @@ -7,19 +7,19 @@ import { | ||||
|   unfollowAccount, | ||||
|   blockAccount, | ||||
|   unblockAccount, | ||||
|   muteAccount, | ||||
|   unmuteAccount, | ||||
| } from '../../../actions/accounts'; | ||||
| import { mentionCompose } from '../../../actions/compose'; | ||||
| import { initMuteModal } from '../../../actions/mutes'; | ||||
| import { initReport } from '../../../actions/reports'; | ||||
| import { openModal } from '../../../actions/modal'; | ||||
| import { blockDomain, unblockDomain } from '../../../actions/domain_blocks'; | ||||
| import { defineMessages, injectIntl, FormattedMessage } from 'react-intl'; | ||||
| import { unfollowModal } from '../../../initial_state'; | ||||
|  | ||||
| const messages = defineMessages({ | ||||
|   unfollowConfirm: { id: 'confirmations.unfollow.confirm', defaultMessage: 'Unfollow' }, | ||||
|   blockConfirm: { id: 'confirmations.block.confirm', defaultMessage: 'Block' }, | ||||
|   muteConfirm: { id: 'confirmations.mute.confirm', defaultMessage: 'Mute' }, | ||||
|   blockDomainConfirm: { id: 'confirmations.domain_block.confirm', defaultMessage: 'Hide entire domain' }, | ||||
| }); | ||||
|  | ||||
| @@ -28,8 +28,6 @@ const makeMapStateToProps = () => { | ||||
|  | ||||
|   const mapStateToProps = (state, { accountId }) => ({ | ||||
|     account: getAccount(state, accountId), | ||||
|     me: state.getIn(['meta', 'me']), | ||||
|     unfollowModal: state.getIn(['meta', 'unfollow_modal']), | ||||
|   }); | ||||
|  | ||||
|   return mapStateToProps; | ||||
| @@ -39,7 +37,7 @@ const mapDispatchToProps = (dispatch, { intl }) => ({ | ||||
|  | ||||
|   onFollow (account) { | ||||
|     if (account.getIn(['relationship', 'following']) || account.getIn(['relationship', 'requested'])) { | ||||
|       if (this.unfollowModal) { | ||||
|       if (unfollowModal) { | ||||
|         dispatch(openModal('CONFIRM', { | ||||
|           message: <FormattedMessage id='confirmations.unfollow.message' defaultMessage='Are you sure you want to unfollow {name}?' values={{ name: <strong>@{account.get('acct')}</strong> }} />, | ||||
|           confirm: intl.formatMessage(messages.unfollowConfirm), | ||||
| @@ -69,6 +67,14 @@ const mapDispatchToProps = (dispatch, { intl }) => ({ | ||||
|     dispatch(mentionCompose(account, router)); | ||||
|   }, | ||||
|  | ||||
|   onReblogToggle (account) { | ||||
|     if (account.getIn(['relationship', 'show_reblogs'])) { | ||||
|       dispatch(followAccount(account.get('id'), false)); | ||||
|     } else { | ||||
|       dispatch(followAccount(account.get('id'), true)); | ||||
|     } | ||||
|   }, | ||||
|  | ||||
|   onReport (account) { | ||||
|     dispatch(initReport(account)); | ||||
|   }, | ||||
| @@ -77,11 +83,7 @@ const mapDispatchToProps = (dispatch, { intl }) => ({ | ||||
|     if (account.getIn(['relationship', 'muting'])) { | ||||
|       dispatch(unmuteAccount(account.get('id'))); | ||||
|     } else { | ||||
|       dispatch(openModal('CONFIRM', { | ||||
|         message: <FormattedMessage id='confirmations.mute.message' defaultMessage='Are you sure you want to mute {name}?' values={{ name: <strong>@{account.get('acct')}</strong> }} />, | ||||
|         confirm: intl.formatMessage(messages.muteConfirm), | ||||
|         onConfirm: () => dispatch(muteAccount(account.get('id'))), | ||||
|       })); | ||||
|       dispatch(initMuteModal(account)); | ||||
|     } | ||||
|   }, | ||||
|  | ||||
|   | ||||
| @@ -16,7 +16,6 @@ const mapStateToProps = (state, props) => ({ | ||||
|   statusIds: state.getIn(['timelines', `account:${props.params.accountId}`, 'items'], ImmutableList()), | ||||
|   isLoading: state.getIn(['timelines', `account:${props.params.accountId}`, 'isLoading']), | ||||
|   hasMore: !!state.getIn(['timelines', `account:${props.params.accountId}`, 'next']), | ||||
|   me: state.getIn(['meta', 'me']), | ||||
| }); | ||||
|  | ||||
| @connect(mapStateToProps) | ||||
| @@ -28,7 +27,6 @@ export default class AccountTimeline extends ImmutablePureComponent { | ||||
|     statusIds: ImmutablePropTypes.list, | ||||
|     isLoading: PropTypes.bool, | ||||
|     hasMore: PropTypes.bool, | ||||
|     me: PropTypes.string.isRequired, | ||||
|   }; | ||||
|  | ||||
|   componentWillMount () { | ||||
| @@ -50,7 +48,7 @@ export default class AccountTimeline extends ImmutablePureComponent { | ||||
|   } | ||||
|  | ||||
|   render () { | ||||
|     const { statusIds, isLoading, hasMore, me } = this.props; | ||||
|     const { statusIds, isLoading, hasMore } = this.props; | ||||
|  | ||||
|     if (!statusIds && isLoading) { | ||||
|       return ( | ||||
| @@ -70,7 +68,6 @@ export default class AccountTimeline extends ImmutablePureComponent { | ||||
|           statusIds={statusIds} | ||||
|           isLoading={isLoading} | ||||
|           hasMore={hasMore} | ||||
|           me={me} | ||||
|           onScrollToBottom={this.handleScrollToBottom} | ||||
|         /> | ||||
|       </Column> | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import { connect } from 'react-redux'; | ||||
| import ImmutablePropTypes from 'react-immutable-proptypes'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import LoadingIndicator from '../../components/loading_indicator'; | ||||
| import { ScrollContainer } from 'react-router-scroll'; | ||||
| import { ScrollContainer } from 'react-router-scroll-4'; | ||||
| import Column from '../ui/components/column'; | ||||
| import ColumnBackButtonSlim from '../../components/column_back_button_slim'; | ||||
| import AccountContainer from '../../containers/account_container'; | ||||
|   | ||||
| @@ -41,7 +41,6 @@ export default class ComposeForm extends ImmutablePureComponent { | ||||
|     preselectDate: PropTypes.instanceOf(Date), | ||||
|     is_submitting: PropTypes.bool, | ||||
|     is_uploading: PropTypes.bool, | ||||
|     me: PropTypes.string, | ||||
|     onChange: PropTypes.func.isRequired, | ||||
|     onSubmit: PropTypes.func.isRequired, | ||||
|     onClearSuggestions: PropTypes.func.isRequired, | ||||
|   | ||||
| @@ -157,7 +157,6 @@ class EmojiPickerMenu extends React.PureComponent { | ||||
|     intl: PropTypes.object.isRequired, | ||||
|     skinTone: PropTypes.number.isRequired, | ||||
|     onSkinTone: PropTypes.func.isRequired, | ||||
|     autoPlay: PropTypes.bool, | ||||
|   }; | ||||
|  | ||||
|   static defaultProps = { | ||||
| @@ -235,7 +234,7 @@ class EmojiPickerMenu extends React.PureComponent { | ||||
|   } | ||||
|  | ||||
|   render () { | ||||
|     const { loading, style, intl, custom_emojis, autoPlay, skinTone, frequentlyUsedEmojis } = this.props; | ||||
|     const { loading, style, intl, custom_emojis, skinTone, frequentlyUsedEmojis } = this.props; | ||||
|  | ||||
|     if (loading) { | ||||
|       return <div style={{ width: 299 }} />; | ||||
| @@ -250,7 +249,7 @@ class EmojiPickerMenu extends React.PureComponent { | ||||
|           perLine={8} | ||||
|           emojiSize={22} | ||||
|           sheetSize={32} | ||||
|           custom={buildCustomEmojis(custom_emojis, autoPlay)} | ||||
|           custom={buildCustomEmojis(custom_emojis)} | ||||
|           color='' | ||||
|           emoji='' | ||||
|           set='twitter' | ||||
| @@ -284,7 +283,6 @@ export default class EmojiPickerDropdown extends React.PureComponent { | ||||
|   static propTypes = { | ||||
|     custom_emojis: ImmutablePropTypes.list, | ||||
|     frequentlyUsedEmojis: PropTypes.arrayOf(PropTypes.string), | ||||
|     autoPlay: PropTypes.bool, | ||||
|     intl: PropTypes.object.isRequired, | ||||
|     onPickEmoji: PropTypes.func.isRequired, | ||||
|     onSkinTone: PropTypes.func.isRequired, | ||||
| @@ -346,7 +344,7 @@ export default class EmojiPickerDropdown extends React.PureComponent { | ||||
|   } | ||||
|  | ||||
|   render () { | ||||
|     const { intl, onPickEmoji, autoPlay, onSkinTone, skinTone, frequentlyUsedEmojis } = this.props; | ||||
|     const { intl, onPickEmoji, onSkinTone, skinTone, frequentlyUsedEmojis } = this.props; | ||||
|     const title = intl.formatMessage(messages.emoji); | ||||
|     const { active, loading } = this.state; | ||||
|  | ||||
| @@ -366,7 +364,6 @@ export default class EmojiPickerDropdown extends React.PureComponent { | ||||
|             loading={loading} | ||||
|             onClose={this.onHideDropdown} | ||||
|             onPick={onPickEmoji} | ||||
|             autoPlay={autoPlay} | ||||
|             onSkinTone={onSkinTone} | ||||
|             skinTone={skinTone} | ||||
|             frequentlyUsedEmojis={frequentlyUsedEmojis} | ||||
|   | ||||
| @@ -11,7 +11,7 @@ export default class NavigationBar extends ImmutablePureComponent { | ||||
|  | ||||
|   static propTypes = { | ||||
|     account: ImmutablePropTypes.map.isRequired, | ||||
|     onClose: PropTypes.func.isRequired, | ||||
|     onClose: PropTypes.func, | ||||
|   }; | ||||
|  | ||||
|   render () { | ||||
|   | ||||
| @@ -68,7 +68,7 @@ export default class Upload extends ImmutablePureComponent { | ||||
|       <div className='compose-form__upload' onMouseEnter={this.handleMouseEnter} onMouseLeave={this.handleMouseLeave}> | ||||
|         <Motion defaultStyle={{ scale: 0.8 }} style={{ scale: spring(1, { stiffness: 180, damping: 12 }) }}> | ||||
|           {({ scale }) => ( | ||||
|             <div className='compose-form__upload-thumbnail' style={{ transform: `translateZ(0) scale(${scale})`, backgroundImage: `url(${media.get('preview_url')})` }}> | ||||
|             <div className='compose-form__upload-thumbnail' style={{ transform: `scale(${scale})`, backgroundImage: `url(${media.get('preview_url')})` }}> | ||||
|               <IconButton icon='times' title={intl.formatMessage(messages.undo)} size={36} onClick={this.handleUndoClick} /> | ||||
|  | ||||
|               <div className={classNames('compose-form__upload-description', { active })}> | ||||
|   | ||||
| @@ -22,7 +22,6 @@ const mapStateToProps = state => ({ | ||||
|   preselectDate: state.getIn(['compose', 'preselectDate']), | ||||
|   is_submitting: state.getIn(['compose', 'is_submitting']), | ||||
|   is_uploading: state.getIn(['compose', 'is_uploading']), | ||||
|   me: state.getIn(['compose', 'me']), | ||||
|   showSearch: state.getIn(['search', 'submitted']) && !state.getIn(['search', 'hidden']), | ||||
| }); | ||||
|  | ||||
|   | ||||
| @@ -46,7 +46,7 @@ const getFrequentlyUsedEmojis = createSelector([ | ||||
|  | ||||
| const getCustomEmojis = createSelector([ | ||||
|   state => state.get('custom_emojis'), | ||||
| ], emojis => emojis.sort((a, b) => { | ||||
| ], emojis => emojis.filter(e => e.get('visible_in_picker')).sort((a, b) => { | ||||
|   const aShort = a.get('shortcode').toLowerCase(); | ||||
|   const bShort = b.get('shortcode').toLowerCase(); | ||||
|  | ||||
| @@ -61,7 +61,6 @@ const getCustomEmojis = createSelector([ | ||||
|  | ||||
| const mapStateToProps = state => ({ | ||||
|   custom_emojis: getCustomEmojis(state), | ||||
|   autoPlay: state.getIn(['meta', 'auto_play_gif']), | ||||
|   skinTone: state.getIn(['settings', 'skinTone']), | ||||
|   frequentlyUsedEmojis: getFrequentlyUsedEmojis(state), | ||||
| }); | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| import { connect }   from 'react-redux'; | ||||
| import NavigationBar from '../components/navigation_bar'; | ||||
| import { me } from '../../../initial_state'; | ||||
|  | ||||
| const mapStateToProps = state => { | ||||
|   return { | ||||
|     account: state.getIn(['accounts', state.getIn(['meta', 'me'])]), | ||||
|     account: state.getIn(['accounts', me]), | ||||
|   }; | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -47,7 +47,7 @@ class SensitiveButton extends React.PureComponent { | ||||
|             'compose-form__sensitive-button--visible': visible, | ||||
|           }); | ||||
|           return ( | ||||
|             <div className={className} style={{ transform: `translateZ(0) scale(${scale})` }}> | ||||
|             <div className={className} style={{ transform: `scale(${scale})` }}> | ||||
|               <IconButton | ||||
|                 className='compose-form__sensitive-button__icon' | ||||
|                 title={intl.formatMessage(messages.title)} | ||||
|   | ||||
| @@ -3,9 +3,10 @@ import { connect } from 'react-redux'; | ||||
| import Warning from '../components/warning'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import { FormattedMessage } from 'react-intl'; | ||||
| import { me } from '../../../initial_state'; | ||||
|  | ||||
| const mapStateToProps = state => ({ | ||||
|   needsLockWarning: state.getIn(['compose', 'privacy']) === 'private' && !state.getIn(['accounts', state.getIn(['meta', 'me']), 'locked']), | ||||
|   needsLockWarning: state.getIn(['compose', 'privacy']) === 'private' && !state.getIn(['accounts', me, 'locked']), | ||||
| }); | ||||
|  | ||||
| const WarningWrapper = ({ needsLockWarning }) => { | ||||
|   | ||||
| @@ -5,5 +5,5 @@ const urlPlaceholder = 'xxxxxxxxxxxxxxxxxxxxxxx'; | ||||
| export function countableText(inputText) { | ||||
|   return inputText | ||||
|     .replace(urlRegex, urlPlaceholder) | ||||
|     .replace(/(?:^|[^\/\w])@(([a-z0-9_]+)@[a-z0-9\.\-]+[a-z0-9]+)/ig, '@$2'); | ||||
|     .replace(/(^|[^\/\w])@(([a-z0-9_]+)@[a-z0-9\.\-]+[a-z0-9]+)/ig, '$1@$3'); | ||||
| }; | ||||
|   | ||||
| @@ -57,5 +57,21 @@ describe('emoji', () => { | ||||
|     it('does an emoji whose filename is irregular', () => { | ||||
|       expect(emojify('↙️')).toEqual('<img draggable="false" class="emojione" alt="↙️" title=":arrow_lower_left:" src="/emoji/2199.svg" />'); | ||||
|     }); | ||||
|  | ||||
|     it('avoid emojifying on invisible text', () => { | ||||
|       expect(emojify('<a href="http://example.com/test%F0%9F%98%84"><span class="invisible">http://</span><span class="ellipsis">example.com/te</span><span class="invisible">st😄</span></a>')) | ||||
|         .toEqual('<a href="http://example.com/test%F0%9F%98%84"><span class="invisible">http://</span><span class="ellipsis">example.com/te</span><span class="invisible">st😄</span></a>'); | ||||
|       expect(emojify('<span class="invisible">:luigi:</span>', { ':luigi:': { static_url: 'luigi.exe' } })) | ||||
|         .toEqual('<span class="invisible">:luigi:</span>'); | ||||
|     }); | ||||
|  | ||||
|     it('avoid emojifying on invisible text with nested tags', () => { | ||||
|       expect(emojify('<span class="invisible">😄<span class="foo">bar</span>😴</span>😇')) | ||||
|         .toEqual('<span class="invisible">😄<span class="foo">bar</span>😴</span><img draggable="false" class="emojione" alt="😇" title=":innocent:" src="/emoji/1f607.svg" />'); | ||||
|       expect(emojify('<span class="invisible">😄<span class="invisible">😕</span>😴</span>😇')) | ||||
|         .toEqual('<span class="invisible">😄<span class="invisible">😕</span>😴</span><img draggable="false" class="emojione" alt="😇" title=":innocent:" src="/emoji/1f607.svg" />'); | ||||
|       expect(emojify('<span class="invisible">😄<br/>😴</span>😇')) | ||||
|         .toEqual('<span class="invisible">😄<br/>😴</span><img draggable="false" class="emojione" alt="😇" title=":innocent:" src="/emoji/1f607.svg" />'); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| import { autoPlayGif } from '../../initial_state'; | ||||
| import unicodeMapping from './emoji_unicode_mapping_light'; | ||||
| import Trie from 'substring-trie'; | ||||
|  | ||||
| @@ -5,13 +6,13 @@ const trie = new Trie(Object.keys(unicodeMapping)); | ||||
|  | ||||
| const assetHost = process.env.CDN_HOST || ''; | ||||
|  | ||||
| let allowAnimations = false; | ||||
|  | ||||
| const emojify = (str, customEmojis = {}) => { | ||||
|   let rtn = ''; | ||||
|   const tagCharsWithoutEmojis = '<&'; | ||||
|   const tagCharsWithEmojis = Object.keys(customEmojis).length ? '<&:' : '<&'; | ||||
|   let rtn = '', tagChars = tagCharsWithEmojis, invisible = 0; | ||||
|   for (;;) { | ||||
|     let match, i = 0, tag; | ||||
|     while (i < str.length && (tag = '<&:'.indexOf(str[i])) === -1 && !(match = trie.search(str.slice(i)))) { | ||||
|     while (i < str.length && (tag = tagChars.indexOf(str[i])) === -1 && (invisible || !(match = trie.search(str.slice(i))))) { | ||||
|       i += str.codePointAt(i) < 65536 ? 1 : 2; | ||||
|     } | ||||
|     let rend, replacement = ''; | ||||
| @@ -27,7 +28,7 @@ const emojify = (str, customEmojis = {}) => { | ||||
|         // now got a replacee as ':shortname:' | ||||
|         // if you want additional emoji handler, add statements below which set replacement and return true. | ||||
|         if (shortname in customEmojis) { | ||||
|           const filename = allowAnimations ? customEmojis[shortname].url : customEmojis[shortname].static_url; | ||||
|           const filename = autoPlayGif ? customEmojis[shortname].url : customEmojis[shortname].static_url; | ||||
|           replacement = `<img draggable="false" class="emojione" alt="${shortname}" title="${shortname}" src="${filename}" />`; | ||||
|           return true; | ||||
|         } | ||||
| @@ -35,7 +36,26 @@ const emojify = (str, customEmojis = {}) => { | ||||
|       })()) rend = ++i; | ||||
|     } else if (tag >= 0) { // <, & | ||||
|       rend = str.indexOf('>;'[tag], i + 1) + 1; | ||||
|       if (!rend) break; | ||||
|       if (!rend) { | ||||
|         break; | ||||
|       } | ||||
|       if (tag === 0) { | ||||
|         if (invisible) { | ||||
|           if (str[i + 1] === '/') { // closing tag | ||||
|             if (!--invisible) { | ||||
|               tagChars = tagCharsWithEmojis; | ||||
|             } | ||||
|           } else if (str[rend - 2] !== '/') { // opening tag | ||||
|             invisible++; | ||||
|           } | ||||
|         } else { | ||||
|           if (str.startsWith('<span class="invisible">', i)) { | ||||
|             // avoid emojifying on invisible text | ||||
|             invisible = 1; | ||||
|             tagChars = tagCharsWithoutEmojis; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|       i = rend; | ||||
|     } else { // matched to unicode emoji | ||||
|       const { filename, shortCode } = unicodeMapping[match]; | ||||
| @@ -51,14 +71,12 @@ const emojify = (str, customEmojis = {}) => { | ||||
|  | ||||
| export default emojify; | ||||
|  | ||||
| export const buildCustomEmojis = (customEmojis, overrideAllowAnimations = false) => { | ||||
| export const buildCustomEmojis = (customEmojis) => { | ||||
|   const emojis = []; | ||||
|  | ||||
|   allowAnimations = overrideAllowAnimations; | ||||
|  | ||||
|   customEmojis.forEach(emoji => { | ||||
|     const shortcode = emoji.get('shortcode'); | ||||
|     const url       = allowAnimations ? emoji.get('url') : emoji.get('static_url'); | ||||
|     const url       = autoPlayGif ? emoji.get('url') : emoji.get('static_url'); | ||||
|     const name      = shortcode.replace(':', ''); | ||||
|  | ||||
|     emojis.push({ | ||||
|   | ||||
| @@ -64,14 +64,14 @@ Object.keys(emojiMap).forEach(key => { | ||||
|  | ||||
| Object.keys(emojiIndex.emojis).forEach(key => { | ||||
|   const { native } = emojiIndex.emojis[key]; | ||||
|   const { short_names, search, unified } = emojiMartData.emojis[key]; | ||||
|   let { short_names, search, unified } = emojiMartData.emojis[key]; | ||||
|   if (short_names[0] !== key) { | ||||
|     throw new Error('The compresser expects the first short_code to be the ' + | ||||
|       'key. It may need to be rewritten if the emoji change such that this ' + | ||||
|       'is no longer the case.'); | ||||
|   } | ||||
|  | ||||
|   short_names.splice(0, 1); // first short name can be inferred from the key | ||||
|   short_names = short_names.slice(1); // first short name can be inferred from the key | ||||
|  | ||||
|   const searchData = [native, short_names, search]; | ||||
|   if (unicodeToUnifiedName(native) !== unified) { | ||||
|   | ||||
| @@ -4,7 +4,7 @@ import PropTypes from 'prop-types'; | ||||
| import ImmutablePropTypes from 'react-immutable-proptypes'; | ||||
| import LoadingIndicator from '../../components/loading_indicator'; | ||||
| import { fetchFavourites } from '../../actions/interactions'; | ||||
| import { ScrollContainer } from 'react-router-scroll'; | ||||
| import { ScrollContainer } from 'react-router-scroll-4'; | ||||
| import AccountContainer from '../../containers/account_container'; | ||||
| import Column from '../ui/components/column'; | ||||
| import ColumnBackButton from '../../components/column_back_button'; | ||||
|   | ||||
| @@ -3,7 +3,7 @@ import { connect } from 'react-redux'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import ImmutablePropTypes from 'react-immutable-proptypes'; | ||||
| import LoadingIndicator from '../../components/loading_indicator'; | ||||
| import { ScrollContainer } from 'react-router-scroll'; | ||||
| import { ScrollContainer } from 'react-router-scroll-4'; | ||||
| import Column from '../ui/components/column'; | ||||
| import ColumnBackButtonSlim from '../../components/column_back_button_slim'; | ||||
| import AccountAuthorizeContainer from './containers/account_authorize_container'; | ||||
|   | ||||
| @@ -8,7 +8,7 @@ import { | ||||
|   fetchFollowers, | ||||
|   expandFollowers, | ||||
| } from '../../actions/accounts'; | ||||
| import { ScrollContainer } from 'react-router-scroll'; | ||||
| import { ScrollContainer } from 'react-router-scroll-4'; | ||||
| import AccountContainer from '../../containers/account_container'; | ||||
| import Column from '../ui/components/column'; | ||||
| import HeaderContainer from '../account_timeline/containers/header_container'; | ||||
|   | ||||
| @@ -8,7 +8,7 @@ import { | ||||
|   fetchFollowing, | ||||
|   expandFollowing, | ||||
| } from '../../actions/accounts'; | ||||
| import { ScrollContainer } from 'react-router-scroll'; | ||||
| import { ScrollContainer } from 'react-router-scroll-4'; | ||||
| import AccountContainer from '../../containers/account_container'; | ||||
| import Column from '../ui/components/column'; | ||||
| import HeaderContainer from '../account_timeline/containers/header_container'; | ||||
|   | ||||
| @@ -7,6 +7,7 @@ import { connect } from 'react-redux'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import ImmutablePropTypes from 'react-immutable-proptypes'; | ||||
| import ImmutablePureComponent from 'react-immutable-pure-component'; | ||||
| import { me } from '../../initial_state'; | ||||
|  | ||||
| const messages = defineMessages({ | ||||
|   heading: { id: 'getting_started.heading', defaultMessage: 'Getting started' }, | ||||
| @@ -24,10 +25,12 @@ const messages = defineMessages({ | ||||
|   mutes: { id: 'navigation_bar.mutes', defaultMessage: 'Muted users' }, | ||||
|   info: { id: 'navigation_bar.info', defaultMessage: 'Extended information' }, | ||||
|   pins: { id: 'navigation_bar.pins', defaultMessage: 'Pinned toots' }, | ||||
|   lists: { id: 'navigation_bar.lists', defaultMessage: 'Lists' }, | ||||
|   keyboard_shortcuts: { id: 'navigation_bar.keyboard_shortcuts', defaultMessage: 'Keyboard shortcuts' }, | ||||
| }); | ||||
|  | ||||
| const mapStateToProps = state => ({ | ||||
|   me: state.getIn(['accounts', state.getIn(['meta', 'me'])]), | ||||
|   myAccount: state.getIn(['accounts', me]), | ||||
|   columns: state.getIn(['settings', 'columns']), | ||||
| }); | ||||
|  | ||||
| @@ -37,13 +40,13 @@ export default class GettingStarted extends ImmutablePureComponent { | ||||
|  | ||||
|   static propTypes = { | ||||
|     intl: PropTypes.object.isRequired, | ||||
|     me: ImmutablePropTypes.map.isRequired, | ||||
|     myAccount: ImmutablePropTypes.map.isRequired, | ||||
|     columns: ImmutablePropTypes.list, | ||||
|     multiColumn: PropTypes.bool, | ||||
|   }; | ||||
|  | ||||
|   render () { | ||||
|     const { intl, me, columns, multiColumn } = this.props; | ||||
|     const { intl, myAccount, columns, multiColumn } = this.props; | ||||
|  | ||||
|     let navItems = []; | ||||
|  | ||||
| @@ -68,15 +71,17 @@ export default class GettingStarted extends ImmutablePureComponent { | ||||
|     navItems = navItems.concat([ | ||||
|       <ColumnLink key='4' icon='star' text={intl.formatMessage(messages.favourites)} to='/favourites' />, | ||||
|       <ColumnLink key='5' icon='thumb-tack' text={intl.formatMessage(messages.pins)} to='/pinned' />, | ||||
|       <ColumnLink key='9' icon='bars' text={intl.formatMessage(messages.lists)} to='/lists' />, | ||||
|     ]); | ||||
|  | ||||
|     if (me.get('locked')) { | ||||
|     if (myAccount.get('locked')) { | ||||
|       navItems.push(<ColumnLink key='6' icon='users' text={intl.formatMessage(messages.follow_requests)} to='/follow_requests' />); | ||||
|     } | ||||
|  | ||||
|     navItems = navItems.concat([ | ||||
|       <ColumnLink key='7' icon='volume-off' text={intl.formatMessage(messages.mutes)} to='/mutes' />, | ||||
|       <ColumnLink key='8' icon='ban' text={intl.formatMessage(messages.blocks)} to='/blocks' />, | ||||
|       <ColumnLink key='10' icon='question' text={intl.formatMessage(messages.keyboard_shortcuts)} to='/keyboard-shortcuts' hideOnMobile />, | ||||
|     ]); | ||||
|  | ||||
|     return ( | ||||
|   | ||||
							
								
								
									
										98
									
								
								app/javascript/mastodon/features/keyboard_shortcuts/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								app/javascript/mastodon/features/keyboard_shortcuts/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,98 @@ | ||||
| import React from 'react'; | ||||
| import Column from '../ui/components/column'; | ||||
| import ColumnBackButtonSlim from '../../components/column_back_button_slim'; | ||||
| import { defineMessages, injectIntl, FormattedMessage } from 'react-intl'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import ImmutablePureComponent from 'react-immutable-pure-component'; | ||||
|  | ||||
| const messages = defineMessages({ | ||||
|   heading: { id: 'keyboard_shortcuts.heading', defaultMessage: 'Keyboard Shortcuts' }, | ||||
| }); | ||||
|  | ||||
| @injectIntl | ||||
| export default class KeyboardShortcuts extends ImmutablePureComponent { | ||||
|  | ||||
|   static propTypes = { | ||||
|     intl: PropTypes.object.isRequired, | ||||
|     multiColumn: PropTypes.bool, | ||||
|   }; | ||||
|  | ||||
|   render () { | ||||
|     const { intl } = this.props; | ||||
|  | ||||
|     return ( | ||||
|       <Column icon='question' heading={intl.formatMessage(messages.heading)}> | ||||
|         <ColumnBackButtonSlim /> | ||||
|         <div className='keyboard-shortcuts scrollable optionally-scrollable'> | ||||
|           <table> | ||||
|             <thead> | ||||
|               <tr> | ||||
|                 <th><FormattedMessage id='keyboard_shortcuts.hotkey' defaultMessage='Hotkey' /></th> | ||||
|                 <th><FormattedMessage id='keyboard_shortcuts.description' defaultMessage='Description' /></th> | ||||
|               </tr> | ||||
|             </thead> | ||||
|             <tbody> | ||||
|               <tr> | ||||
|                 <td><code>r</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.reply' defaultMessage='to reply' /></td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td><code>m</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.mention' defaultMessage='to mention author' /></td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td><code>f</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.favourite' defaultMessage='to favourite' /></td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td><code>b</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.boost' defaultMessage='to boost' /></td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td><code>enter</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.enter' defaultMessage='to open status' /></td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td><code>up</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.up' defaultMessage='to move up in the list' /></td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td><code>down</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.down' defaultMessage='to move down in the list' /></td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td><code>1</code>-<code>9</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.column' defaultMessage='to focus a status in one of the columns' /></td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td><code>n</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.compose' defaultMessage='to focus the compose textarea' /></td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td><code>alt</code>+<code>n</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.toot' defaultMessage='to start a brand new toot' /></td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td><code>backspace</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.back' defaultMessage='to navigate back' /></td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td><code>s</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.search' defaultMessage='to focus search' /></td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td><code>esc</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.unfocus' defaultMessage='to un-focus compose textarea/search' /></td> | ||||
|               </tr> | ||||
|               <tr> | ||||
|                 <td><code>?</code></td> | ||||
|                 <td><FormattedMessage id='keyboard_shortcuts.legend' defaultMessage='to display this legend' /></td> | ||||
|               </tr> | ||||
|             </tbody> | ||||
|           </table> | ||||
|         </div> | ||||
|       </Column> | ||||
|     ); | ||||
|   } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,77 @@ | ||||
| import React from 'react'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import { connect } from 'react-redux'; | ||||
| import { makeGetAccount } from '../../../selectors'; | ||||
| import ImmutablePureComponent from 'react-immutable-pure-component'; | ||||
| import ImmutablePropTypes from 'react-immutable-proptypes'; | ||||
| import Avatar from '../../../components/avatar'; | ||||
| import DisplayName from '../../../components/display_name'; | ||||
| import IconButton from '../../../components/icon_button'; | ||||
| import { defineMessages, injectIntl } from 'react-intl'; | ||||
| import { removeFromListEditor, addToListEditor } from '../../../actions/lists'; | ||||
|  | ||||
| const messages = defineMessages({ | ||||
|   remove: { id: 'lists.account.remove', defaultMessage: 'Remove from list' }, | ||||
|   add: { id: 'lists.account.add', defaultMessage: 'Add to list' }, | ||||
| }); | ||||
|  | ||||
| const makeMapStateToProps = () => { | ||||
|   const getAccount = makeGetAccount(); | ||||
|  | ||||
|   const mapStateToProps = (state, { accountId, added }) => ({ | ||||
|     account: getAccount(state, accountId), | ||||
|     added: typeof added === 'undefined' ? state.getIn(['listEditor', 'accounts', 'items']).includes(accountId) : added, | ||||
|   }); | ||||
|  | ||||
|   return mapStateToProps; | ||||
| }; | ||||
|  | ||||
| const mapDispatchToProps = (dispatch, { accountId }) => ({ | ||||
|   onRemove: () => dispatch(removeFromListEditor(accountId)), | ||||
|   onAdd: () => dispatch(addToListEditor(accountId)), | ||||
| }); | ||||
|  | ||||
| @connect(makeMapStateToProps, mapDispatchToProps) | ||||
| @injectIntl | ||||
| export default class Account extends ImmutablePureComponent { | ||||
|  | ||||
|   static propTypes = { | ||||
|     account: ImmutablePropTypes.map.isRequired, | ||||
|     intl: PropTypes.object.isRequired, | ||||
|     onRemove: PropTypes.func.isRequired, | ||||
|     onAdd: PropTypes.func.isRequired, | ||||
|     added: PropTypes.bool, | ||||
|   }; | ||||
|  | ||||
|   static defaultProps = { | ||||
|     added: false, | ||||
|   }; | ||||
|  | ||||
|   render () { | ||||
|     const { account, intl, onRemove, onAdd, added } = this.props; | ||||
|  | ||||
|     let button; | ||||
|  | ||||
|     if (added) { | ||||
|       button = <IconButton icon='times' title={intl.formatMessage(messages.remove)} onClick={onRemove} />; | ||||
|     } else { | ||||
|       button = <IconButton icon='plus' title={intl.formatMessage(messages.add)} onClick={onAdd} />; | ||||
|     } | ||||
|  | ||||
|     return ( | ||||
|       <div className='account'> | ||||
|         <div className='account__wrapper'> | ||||
|           <div className='account__display-name'> | ||||
|             <div className='account__avatar-wrapper'><Avatar account={account} size={36} /></div> | ||||
|             <DisplayName account={account} /> | ||||
|           </div> | ||||
|  | ||||
|           <div className='account__relationship'> | ||||
|             {button} | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     ); | ||||
|   } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,75 @@ | ||||
| import React from 'react'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import { connect } from 'react-redux'; | ||||
| import { defineMessages, injectIntl } from 'react-intl'; | ||||
| import { fetchListSuggestions, clearListSuggestions, changeListSuggestions } from '../../../actions/lists'; | ||||
| import classNames from 'classnames'; | ||||
|  | ||||
| const messages = defineMessages({ | ||||
|   search: { id: 'lists.search', defaultMessage: 'Search among people you follow' }, | ||||
| }); | ||||
|  | ||||
| const mapStateToProps = state => ({ | ||||
|   value: state.getIn(['listEditor', 'suggestions', 'value']), | ||||
| }); | ||||
|  | ||||
| const mapDispatchToProps = dispatch => ({ | ||||
|   onSubmit: value => dispatch(fetchListSuggestions(value)), | ||||
|   onClear: () => dispatch(clearListSuggestions()), | ||||
|   onChange: value => dispatch(changeListSuggestions(value)), | ||||
| }); | ||||
|  | ||||
| @connect(mapStateToProps, mapDispatchToProps) | ||||
| @injectIntl | ||||
| export default class Search extends React.PureComponent { | ||||
|  | ||||
|   static propTypes = { | ||||
|     intl: PropTypes.object.isRequired, | ||||
|     value: PropTypes.string.isRequired, | ||||
|     onChange: PropTypes.func.isRequired, | ||||
|     onSubmit: PropTypes.func.isRequired, | ||||
|     onClear: PropTypes.func.isRequired, | ||||
|   }; | ||||
|  | ||||
|   handleChange = e => { | ||||
|     this.props.onChange(e.target.value); | ||||
|   } | ||||
|  | ||||
|   handleKeyUp = e => { | ||||
|     if (e.keyCode === 13) { | ||||
|       this.props.onSubmit(this.props.value); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   handleClear = () => { | ||||
|     this.props.onClear(); | ||||
|   } | ||||
|  | ||||
|   render () { | ||||
|     const { value, intl } = this.props; | ||||
|     const hasValue = value.length > 0; | ||||
|  | ||||
|     return ( | ||||
|       <div className='list-editor__search search'> | ||||
|         <label> | ||||
|           <span style={{ display: 'none' }}>{intl.formatMessage(messages.search)}</span> | ||||
|  | ||||
|           <input | ||||
|             className='search__input' | ||||
|             type='text' | ||||
|             value={value} | ||||
|             onChange={this.handleChange} | ||||
|             onKeyUp={this.handleKeyUp} | ||||
|             placeholder={intl.formatMessage(messages.search)} | ||||
|           /> | ||||
|         </label> | ||||
|  | ||||
|         <div role='button' tabIndex='0' className='search__icon' onClick={this.handleClear}> | ||||
|           <i className={classNames('fa fa-search', { active: !hasValue })} /> | ||||
|           <i aria-label={intl.formatMessage(messages.search)} className={classNames('fa fa-times-circle', { active: hasValue })} /> | ||||
|         </div> | ||||
|       </div> | ||||
|     ); | ||||
|   } | ||||
|  | ||||
| } | ||||
							
								
								
									
										80
									
								
								app/javascript/mastodon/features/list_editor/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								app/javascript/mastodon/features/list_editor/index.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,80 @@ | ||||
| import React from 'react'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import ImmutablePropTypes from 'react-immutable-proptypes'; | ||||
| import { connect } from 'react-redux'; | ||||
| import ImmutablePureComponent from 'react-immutable-pure-component'; | ||||
| import { injectIntl } from 'react-intl'; | ||||
| import { setupListEditor, clearListSuggestions, resetListEditor } from '../../actions/lists'; | ||||
| import Account from './components/account'; | ||||
| import Search from './components/search'; | ||||
| import Motion from '../ui/util/optional_motion'; | ||||
| import spring from 'react-motion/lib/spring'; | ||||
|  | ||||
| const mapStateToProps = state => ({ | ||||
|   title: state.getIn(['listEditor', 'title']), | ||||
|   accountIds: state.getIn(['listEditor', 'accounts', 'items']), | ||||
|   searchAccountIds: state.getIn(['listEditor', 'suggestions', 'items']), | ||||
| }); | ||||
|  | ||||
| const mapDispatchToProps = dispatch => ({ | ||||
|   onInitialize: listId => dispatch(setupListEditor(listId)), | ||||
|   onClear: () => dispatch(clearListSuggestions()), | ||||
|   onReset: () => dispatch(resetListEditor()), | ||||
| }); | ||||
|  | ||||
| @connect(mapStateToProps, mapDispatchToProps) | ||||
| @injectIntl | ||||
| export default class ListEditor extends ImmutablePureComponent { | ||||
|  | ||||
|   static propTypes = { | ||||
|     listId: PropTypes.string.isRequired, | ||||
|     onClose: PropTypes.func.isRequired, | ||||
|     intl: PropTypes.object.isRequired, | ||||
|     onInitialize: PropTypes.func.isRequired, | ||||
|     onClear: PropTypes.func.isRequired, | ||||
|     onReset: PropTypes.func.isRequired, | ||||
|     title: PropTypes.string.isRequired, | ||||
|     accountIds: ImmutablePropTypes.list.isRequired, | ||||
|     searchAccountIds: ImmutablePropTypes.list.isRequired, | ||||
|   }; | ||||
|  | ||||
|   componentDidMount () { | ||||
|     const { onInitialize, listId } = this.props; | ||||
|     onInitialize(listId); | ||||
|   } | ||||
|  | ||||
|   componentWillUnmount () { | ||||
|     const { onReset } = this.props; | ||||
|     onReset(); | ||||
|   } | ||||
|  | ||||
|   render () { | ||||
|     const { title, accountIds, searchAccountIds, onClear } = this.props; | ||||
|     const showSearch = searchAccountIds.size > 0; | ||||
|  | ||||
|     return ( | ||||
|       <div className='modal-root__modal list-editor'> | ||||
|         <h4>{title}</h4> | ||||
|  | ||||
|         <Search /> | ||||
|  | ||||
|         <div className='drawer__pager'> | ||||
|           <div className='drawer__inner list-editor__accounts'> | ||||
|             {accountIds.map(accountId => <Account key={accountId} accountId={accountId} added />)} | ||||
|           </div> | ||||
|  | ||||
|           {showSearch && <div role='button' tabIndex='-1' className='drawer__backdrop' onClick={onClear} />} | ||||
|  | ||||
|           <Motion defaultStyle={{ x: -100 }} style={{ x: spring(showSearch ? 0 : -100, { stiffness: 210, damping: 20 }) }}> | ||||
|             {({ x }) => | ||||
|               <div className='drawer__inner backdrop' style={{ transform: x === 0 ? null : `translateX(${x}%)`, visibility: x === -100 ? 'hidden' : 'visible' }}> | ||||
|                 {searchAccountIds.map(accountId => <Account key={accountId} accountId={accountId} />)} | ||||
|               </div> | ||||
|             } | ||||
|           </Motion> | ||||
|         </div> | ||||
|       </div> | ||||
|     ); | ||||
|   } | ||||
|  | ||||
| } | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user