├── .DS_Store
├── .browserslistrc
├── .gitignore
├── .ruby-version
├── Gemfile
├── Gemfile.lock
├── README.md
├── Rakefile
├── app
├── .DS_Store
├── assets
│ ├── .DS_Store
│ ├── config
│ │ └── manifest.js
│ ├── images
│ │ ├── .DS_Store
│ │ ├── .keep
│ │ ├── home_page.png
│ │ └── loader.gif
│ └── stylesheets
│ │ ├── application.css
│ │ └── home.scss
├── channels
│ └── application_cable
│ │ ├── channel.rb
│ │ └── connection.rb
├── controllers
│ ├── api
│ │ └── v1
│ │ │ └── blogs_controller.rb
│ ├── application_controller.rb
│ ├── concerns
│ │ └── .keep
│ └── home_controller.rb
├── helpers
│ ├── application_helper.rb
│ └── home_helper.rb
├── javascript
│ ├── .DS_Store
│ ├── channels
│ │ ├── consumer.js
│ │ └── index.js
│ ├── components
│ │ ├── App.js
│ │ ├── Blog.js
│ │ ├── Blogs.js
│ │ └── Home.js
│ ├── packs
│ │ ├── application.js
│ │ └── index.jsx
│ ├── routes
│ │ └── Index.js
│ ├── shared
│ │ ├── Alert.js
│ │ ├── BlogForm.js
│ │ ├── Header.js
│ │ └── PageLoader.js
│ └── stylesheets
│ │ ├── App.css
│ │ ├── application.scss
│ │ └── tailwind.config.js
├── jobs
│ └── application_job.rb
├── mailers
│ └── application_mailer.rb
├── models
│ ├── application_record.rb
│ ├── blog.rb
│ └── concerns
│ │ └── .keep
└── views
│ ├── home
│ └── index.html.erb
│ └── layouts
│ ├── application.html.erb
│ ├── mailer.html.erb
│ └── mailer.text.erb
├── babel.config.js
├── bin
├── bundle
├── rails
├── rake
├── setup
├── spring
├── webpack
├── webpack-dev-server
└── yarn
├── config.ru
├── config
├── application.rb
├── boot.rb
├── cable.yml
├── credentials.yml.enc
├── database.yml
├── environment.rb
├── environments
│ ├── development.rb
│ ├── production.rb
│ └── test.rb
├── initializers
│ ├── application_controller_renderer.rb
│ ├── assets.rb
│ ├── backtrace_silencers.rb
│ ├── content_security_policy.rb
│ ├── cookies_serializer.rb
│ ├── filter_parameter_logging.rb
│ ├── inflections.rb
│ ├── mime_types.rb
│ └── wrap_parameters.rb
├── locales
│ └── en.yml
├── puma.rb
├── routes.rb
├── spring.rb
├── storage.yml
├── webpack
│ ├── development.js
│ ├── environment.js
│ ├── production.js
│ └── test.js
└── webpacker.yml
├── db
├── migrate
│ └── 20200811034930_create_blogs.rb
├── schema.rb
└── seeds.rb
├── lib
├── assets
│ └── .keep
└── tasks
│ └── .keep
├── log
└── .keep
├── package.json
├── postcss.config.js
├── public
├── 404.html
├── 422.html
├── 500.html
├── apple-touch-icon-precomposed.png
├── apple-touch-icon.png
├── favicon.ico
└── robots.txt
├── storage
└── .keep
├── tmp
├── .keep
└── pids
│ └── .keep
├── vendor
└── .keep
└── yarn.lock
/.DS_Store:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/.DS_Store
--------------------------------------------------------------------------------
/.browserslistrc:
--------------------------------------------------------------------------------
1 | defaults
2 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # See https://help.github.com/articles/ignoring-files for more about ignoring files.
2 | #
3 | # If you find yourself ignoring temporary files generated by your text editor
4 | # or operating system, you probably want to add a global ignore instead:
5 | # git config --global core.excludesfile '~/.gitignore_global'
6 |
7 | # Ignore bundler config.
8 | /.bundle
9 |
10 | # Ignore all logfiles and tempfiles.
11 | /log/*
12 | /tmp/*
13 | !/log/.keep
14 | !/tmp/.keep
15 |
16 | # Ignore pidfiles, but keep the directory.
17 | /tmp/pids/*
18 | !/tmp/pids/
19 | !/tmp/pids/.keep
20 |
21 | # Ignore uploaded files in development.
22 | /storage/*
23 | !/storage/.keep
24 |
25 | /public/assets
26 | .byebug_history
27 |
28 | # Ignore master key for decrypting credentials and more.
29 | /config/master.key
30 |
31 | /public/packs
32 | /public/packs-test
33 | /node_modules
34 | /yarn-error.log
35 | yarn-debug.log*
36 | .yarn-integrity
37 |
--------------------------------------------------------------------------------
/.ruby-version:
--------------------------------------------------------------------------------
1 | ruby-2.7.0
2 |
--------------------------------------------------------------------------------
/Gemfile:
--------------------------------------------------------------------------------
1 | source 'https://rubygems.org'
2 | git_source(:github) { |repo| "https://github.com/#{repo}.git" }
3 |
4 | ruby '2.7.0'
5 |
6 | # Bundle edge Rails instead: gem 'rails', github: 'rails/rails'
7 | gem 'rails', '~> 6.0.3', '>= 6.0.3.2'
8 | # Use postgresql as the database for Active Record
9 | gem 'pg', '>= 0.18', '< 2.0'
10 | # Use Puma as the app server
11 | gem 'puma', '~> 4.1'
12 | # Use SCSS for stylesheets
13 | gem 'sass-rails', '>= 6'
14 | # Transpile app-like JavaScript. Read more: https://github.com/rails/webpacker
15 | gem 'webpacker', '~> 4.0'
16 | # Turbolinks makes navigating your web application faster. Read more: https://github.com/turbolinks/turbolinks
17 | gem 'turbolinks', '~> 5'
18 | # Build JSON APIs with ease. Read more: https://github.com/rails/jbuilder
19 | gem 'jbuilder', '~> 2.7'
20 | # Use Redis adapter to run Action Cable in production
21 | # gem 'redis', '~> 4.0'
22 | # Use Active Model has_secure_password
23 | # gem 'bcrypt', '~> 3.1.7'
24 |
25 | # Use Active Storage variant
26 | # gem 'image_processing', '~> 1.2'
27 |
28 | # Reduces boot times through caching; required in config/boot.rb
29 | gem 'bootsnap', '>= 1.4.2', require: false
30 |
31 | group :development, :test do
32 | # Call 'byebug' anywhere in the code to stop execution and get a debugger console
33 | gem 'byebug', platforms: [:mri, :mingw, :x64_mingw]
34 | end
35 |
36 | group :development do
37 | # Access an interactive console on exception pages or by calling 'console' anywhere in the code.
38 | gem 'web-console', '>= 3.3.0'
39 | gem 'listen', '~> 3.2'
40 | # Spring speeds up development by keeping your application running in the background. Read more: https://github.com/rails/spring
41 | gem 'spring'
42 | gem 'spring-watcher-listen', '~> 2.0.0'
43 | end
44 |
45 | # Windows does not include zoneinfo files, so bundle the tzinfo-data gem
46 | gem 'tzinfo-data', platforms: [:mingw, :mswin, :x64_mingw, :jruby]
47 |
--------------------------------------------------------------------------------
/Gemfile.lock:
--------------------------------------------------------------------------------
1 | GEM
2 | remote: https://rubygems.org/
3 | specs:
4 | actioncable (6.0.3.2)
5 | actionpack (= 6.0.3.2)
6 | nio4r (~> 2.0)
7 | websocket-driver (>= 0.6.1)
8 | actionmailbox (6.0.3.2)
9 | actionpack (= 6.0.3.2)
10 | activejob (= 6.0.3.2)
11 | activerecord (= 6.0.3.2)
12 | activestorage (= 6.0.3.2)
13 | activesupport (= 6.0.3.2)
14 | mail (>= 2.7.1)
15 | actionmailer (6.0.3.2)
16 | actionpack (= 6.0.3.2)
17 | actionview (= 6.0.3.2)
18 | activejob (= 6.0.3.2)
19 | mail (~> 2.5, >= 2.5.4)
20 | rails-dom-testing (~> 2.0)
21 | actionpack (6.0.3.2)
22 | actionview (= 6.0.3.2)
23 | activesupport (= 6.0.3.2)
24 | rack (~> 2.0, >= 2.0.8)
25 | rack-test (>= 0.6.3)
26 | rails-dom-testing (~> 2.0)
27 | rails-html-sanitizer (~> 1.0, >= 1.2.0)
28 | actiontext (6.0.3.2)
29 | actionpack (= 6.0.3.2)
30 | activerecord (= 6.0.3.2)
31 | activestorage (= 6.0.3.2)
32 | activesupport (= 6.0.3.2)
33 | nokogiri (>= 1.8.5)
34 | actionview (6.0.3.2)
35 | activesupport (= 6.0.3.2)
36 | builder (~> 3.1)
37 | erubi (~> 1.4)
38 | rails-dom-testing (~> 2.0)
39 | rails-html-sanitizer (~> 1.1, >= 1.2.0)
40 | activejob (6.0.3.2)
41 | activesupport (= 6.0.3.2)
42 | globalid (>= 0.3.6)
43 | activemodel (6.0.3.2)
44 | activesupport (= 6.0.3.2)
45 | activerecord (6.0.3.2)
46 | activemodel (= 6.0.3.2)
47 | activesupport (= 6.0.3.2)
48 | activestorage (6.0.3.2)
49 | actionpack (= 6.0.3.2)
50 | activejob (= 6.0.3.2)
51 | activerecord (= 6.0.3.2)
52 | marcel (~> 0.3.1)
53 | activesupport (6.0.3.2)
54 | concurrent-ruby (~> 1.0, >= 1.0.2)
55 | i18n (>= 0.7, < 2)
56 | minitest (~> 5.1)
57 | tzinfo (~> 1.1)
58 | zeitwerk (~> 2.2, >= 2.2.2)
59 | bindex (0.8.1)
60 | bootsnap (1.4.7)
61 | msgpack (~> 1.0)
62 | builder (3.2.4)
63 | byebug (11.1.3)
64 | concurrent-ruby (1.1.7)
65 | crass (1.0.6)
66 | erubi (1.9.0)
67 | ffi (1.13.1)
68 | globalid (0.4.2)
69 | activesupport (>= 4.2.0)
70 | i18n (1.8.5)
71 | concurrent-ruby (~> 1.0)
72 | jbuilder (2.10.0)
73 | activesupport (>= 5.0.0)
74 | listen (3.2.1)
75 | rb-fsevent (~> 0.10, >= 0.10.3)
76 | rb-inotify (~> 0.9, >= 0.9.10)
77 | loofah (2.6.0)
78 | crass (~> 1.0.2)
79 | nokogiri (>= 1.5.9)
80 | mail (2.7.1)
81 | mini_mime (>= 0.1.1)
82 | marcel (0.3.3)
83 | mimemagic (~> 0.3.2)
84 | method_source (1.0.0)
85 | mimemagic (0.3.5)
86 | mini_mime (1.0.2)
87 | mini_portile2 (2.4.0)
88 | minitest (5.14.1)
89 | msgpack (1.3.3)
90 | nio4r (2.5.2)
91 | nokogiri (1.10.10)
92 | mini_portile2 (~> 2.4.0)
93 | pg (1.2.3)
94 | puma (4.3.5)
95 | nio4r (~> 2.0)
96 | rack (2.2.3)
97 | rack-proxy (0.6.5)
98 | rack
99 | rack-test (1.1.0)
100 | rack (>= 1.0, < 3)
101 | rails (6.0.3.2)
102 | actioncable (= 6.0.3.2)
103 | actionmailbox (= 6.0.3.2)
104 | actionmailer (= 6.0.3.2)
105 | actionpack (= 6.0.3.2)
106 | actiontext (= 6.0.3.2)
107 | actionview (= 6.0.3.2)
108 | activejob (= 6.0.3.2)
109 | activemodel (= 6.0.3.2)
110 | activerecord (= 6.0.3.2)
111 | activestorage (= 6.0.3.2)
112 | activesupport (= 6.0.3.2)
113 | bundler (>= 1.3.0)
114 | railties (= 6.0.3.2)
115 | sprockets-rails (>= 2.0.0)
116 | rails-dom-testing (2.0.3)
117 | activesupport (>= 4.2.0)
118 | nokogiri (>= 1.6)
119 | rails-html-sanitizer (1.3.0)
120 | loofah (~> 2.3)
121 | railties (6.0.3.2)
122 | actionpack (= 6.0.3.2)
123 | activesupport (= 6.0.3.2)
124 | method_source
125 | rake (>= 0.8.7)
126 | thor (>= 0.20.3, < 2.0)
127 | rake (13.0.1)
128 | rb-fsevent (0.10.4)
129 | rb-inotify (0.10.1)
130 | ffi (~> 1.0)
131 | sass-rails (6.0.0)
132 | sassc-rails (~> 2.1, >= 2.1.1)
133 | sassc (2.4.0)
134 | ffi (~> 1.9)
135 | sassc-rails (2.1.2)
136 | railties (>= 4.0.0)
137 | sassc (>= 2.0)
138 | sprockets (> 3.0)
139 | sprockets-rails
140 | tilt
141 | spring (2.1.0)
142 | spring-watcher-listen (2.0.1)
143 | listen (>= 2.7, < 4.0)
144 | spring (>= 1.2, < 3.0)
145 | sprockets (4.0.2)
146 | concurrent-ruby (~> 1.0)
147 | rack (> 1, < 3)
148 | sprockets-rails (3.2.1)
149 | actionpack (>= 4.0)
150 | activesupport (>= 4.0)
151 | sprockets (>= 3.0.0)
152 | thor (1.0.1)
153 | thread_safe (0.3.6)
154 | tilt (2.0.10)
155 | turbolinks (5.2.1)
156 | turbolinks-source (~> 5.2)
157 | turbolinks-source (5.2.0)
158 | tzinfo (1.2.7)
159 | thread_safe (~> 0.1)
160 | web-console (4.0.4)
161 | actionview (>= 6.0.0)
162 | activemodel (>= 6.0.0)
163 | bindex (>= 0.4.0)
164 | railties (>= 6.0.0)
165 | webpacker (4.2.2)
166 | activesupport (>= 4.2)
167 | rack-proxy (>= 0.6.1)
168 | railties (>= 4.2)
169 | websocket-driver (0.7.3)
170 | websocket-extensions (>= 0.1.0)
171 | websocket-extensions (0.1.5)
172 | zeitwerk (2.4.0)
173 |
174 | PLATFORMS
175 | ruby
176 |
177 | DEPENDENCIES
178 | bootsnap (>= 1.4.2)
179 | byebug
180 | jbuilder (~> 2.7)
181 | listen (~> 3.2)
182 | pg (>= 0.18, < 2.0)
183 | puma (~> 4.1)
184 | rails (~> 6.0.3, >= 6.0.3.2)
185 | sass-rails (>= 6)
186 | spring
187 | spring-watcher-listen (~> 2.0.0)
188 | turbolinks (~> 5)
189 | tzinfo-data
190 | web-console (>= 3.3.0)
191 | webpacker (~> 4.0)
192 |
193 | RUBY VERSION
194 | ruby 2.7.0p0
195 |
196 | BUNDLED WITH
197 | 2.1.2
198 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 |
ReactRails Blog App
2 |
3 |
4 |
5 | [](https://www.linkedin.com/in/muhammad-ahmad20/)
6 | [](mailto:muhammad.ahmad8043@gmail.com)
7 |
8 | This app shows you how to create a CRUD app with ROR on back-end and React on front-end with Tailwindcss.
9 |
10 | 
11 |
12 | ## This project will help you:
13 |
14 | - To configure and create a CRUD Rails application with React.
15 | - To configure and use Tailwind in React-Rails(RR) app.
16 | - To show custom alert messages by using Tailwind.
17 |
18 | ## What this App does?
19 |
20 | - By this app, you will be able to:
21 |
22 | - Create a new Blog.
23 | - Edit a Blog.
24 | - Delete a Blog.
25 | - Get a speicific Blog.
26 | - Get list of all Blogs.
27 |
28 | ## Built with
29 |
30 | - Ruby on Rails
31 | - Reactjs (react hooks, react router)
32 | - React-icons
33 | - Tailwind used for styling
34 |
35 | ## Setup
36 |
37 | - Clone the repository
38 | - Use `cd `
39 | - Run `bundle install`
40 | - Run `npm install`
41 | - Run `rails db:create`
42 | - Run `rails db:migrate`
43 | - Run `rails db:seed`
44 | - Run `rails s`
45 | Now check it on browser `localhost:3000`
46 |
47 | ## Author
48 |
49 | Muhammad Ahmad
50 |
51 | - Github: [@Ahmad](https://github.com/MA-Ahmad)
52 |
53 | - LinkedIn: [Muhammad Ahmad](https://www.linkedin.com/in/muhammad-ahmad20/)
54 |
55 | ## 🤝 Contributing
56 |
57 | Contributions, issues and feature requests are welcome! Start by:
58 |
59 | - Forking the project
60 | - Cloning the project to your local machine
61 | - `cd` into the project directory
62 | - Run `git checkout -b your-branch-name`
63 | - Make your contributions
64 | - Push your branch up to your forked repository
65 | - Open a Pull Request with a detailed description to the development branch of the original project for a review
66 |
67 | ## Show your support
68 |
69 | Give a ⭐️ if you like this project!
--------------------------------------------------------------------------------
/Rakefile:
--------------------------------------------------------------------------------
1 | # Add your own tasks in files placed in lib/tasks ending in .rake,
2 | # for example lib/tasks/capistrano.rake, and they will automatically be available to Rake.
3 |
4 | require_relative 'config/application'
5 |
6 | Rails.application.load_tasks
7 |
--------------------------------------------------------------------------------
/app/.DS_Store:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/app/.DS_Store
--------------------------------------------------------------------------------
/app/assets/.DS_Store:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/app/assets/.DS_Store
--------------------------------------------------------------------------------
/app/assets/config/manifest.js:
--------------------------------------------------------------------------------
1 | //= link_tree ../images
2 | //= link_directory ../stylesheets .css
3 |
--------------------------------------------------------------------------------
/app/assets/images/.DS_Store:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/app/assets/images/.DS_Store
--------------------------------------------------------------------------------
/app/assets/images/.keep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/app/assets/images/.keep
--------------------------------------------------------------------------------
/app/assets/images/home_page.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/app/assets/images/home_page.png
--------------------------------------------------------------------------------
/app/assets/images/loader.gif:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/app/assets/images/loader.gif
--------------------------------------------------------------------------------
/app/assets/stylesheets/application.css:
--------------------------------------------------------------------------------
1 | /*
2 | * This is a manifest file that'll be compiled into application.css, which will include all the files
3 | * listed below.
4 | *
5 | * Any CSS and SCSS file within this directory, lib/assets/stylesheets, or any plugin's
6 | * vendor/assets/stylesheets directory can be referenced here using a relative path.
7 | *
8 | * You're free to add application-wide styles to this file and they'll appear at the bottom of the
9 | * compiled file so the styles you add here take precedence over styles defined in any other CSS/SCSS
10 | * files in this directory. Styles in this file should be added after the last require_* statement.
11 | * It is generally better to create a new file per style scope.
12 | *
13 | *= require_tree .
14 | *= require_self
15 | */
16 |
--------------------------------------------------------------------------------
/app/assets/stylesheets/home.scss:
--------------------------------------------------------------------------------
1 | // Place all the styles related to the home controller here.
2 | // They will automatically be included in application.css.
3 | // You can use Sass (SCSS) here: https://sass-lang.com/
4 |
--------------------------------------------------------------------------------
/app/channels/application_cable/channel.rb:
--------------------------------------------------------------------------------
1 | module ApplicationCable
2 | class Channel < ActionCable::Channel::Base
3 | end
4 | end
5 |
--------------------------------------------------------------------------------
/app/channels/application_cable/connection.rb:
--------------------------------------------------------------------------------
1 | module ApplicationCable
2 | class Connection < ActionCable::Connection::Base
3 | end
4 | end
5 |
--------------------------------------------------------------------------------
/app/controllers/api/v1/blogs_controller.rb:
--------------------------------------------------------------------------------
1 | class Api::V1::BlogsController < ApplicationController
2 | before_action :find_blog, except: [:index, :new, :create]
3 |
4 | def index
5 | @blogs = Blog.all.order(created_at: :desc)
6 | render json: @blogs
7 | end
8 |
9 | def show
10 | if @blog
11 | render json: @blog
12 | else
13 | render json: @blog.errors
14 | end
15 | end
16 |
17 | def new
18 | @blog = Blog.new
19 | render json: @blog
20 | end
21 |
22 | def create
23 | @blog = Blog.create!(blog_params)
24 | if @blog
25 | render json: @blog
26 | else
27 | render json: @blog.errors
28 | end
29 | end
30 |
31 | def edit
32 | render json: @blog
33 | end
34 |
35 | def update
36 | if @blog.update(blog_params)
37 | render json: @blog
38 | else
39 | render json: @blog.errors
40 | end
41 | end
42 |
43 | def destroy
44 | if @blog.destroy
45 | render json: { message: 'Blog deleted!' }
46 | else
47 | render json: @blog.error
48 | end
49 | end
50 |
51 | private
52 |
53 | def find_blog
54 | @blog = Blog.find(params[:id])
55 | end
56 |
57 | def blog_params
58 | params.require(:blog).permit(:title, :author, :content, :image)
59 | end
60 | end
--------------------------------------------------------------------------------
/app/controllers/application_controller.rb:
--------------------------------------------------------------------------------
1 | class ApplicationController < ActionController::Base
2 | end
3 |
--------------------------------------------------------------------------------
/app/controllers/concerns/.keep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/app/controllers/concerns/.keep
--------------------------------------------------------------------------------
/app/controllers/home_controller.rb:
--------------------------------------------------------------------------------
1 | class HomeController < ApplicationController
2 | def index
3 | end
4 | end
5 |
--------------------------------------------------------------------------------
/app/helpers/application_helper.rb:
--------------------------------------------------------------------------------
1 | module ApplicationHelper
2 | end
3 |
--------------------------------------------------------------------------------
/app/helpers/home_helper.rb:
--------------------------------------------------------------------------------
1 | module HomeHelper
2 | end
3 |
--------------------------------------------------------------------------------
/app/javascript/.DS_Store:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/app/javascript/.DS_Store
--------------------------------------------------------------------------------
/app/javascript/channels/consumer.js:
--------------------------------------------------------------------------------
1 | // Action Cable provides the framework to deal with WebSockets in Rails.
2 | // You can generate new channels where WebSocket features live using the `rails generate channel` command.
3 |
4 | import { createConsumer } from "@rails/actioncable"
5 |
6 | export default createConsumer()
7 |
--------------------------------------------------------------------------------
/app/javascript/channels/index.js:
--------------------------------------------------------------------------------
1 | // Load all the channels within this directory and all subdirectories.
2 | // Channel files must be named *_channel.js.
3 |
4 | const channels = require.context('.', true, /_channel\.js$/)
5 | channels.keys().forEach(channels)
6 |
--------------------------------------------------------------------------------
/app/javascript/components/App.js:
--------------------------------------------------------------------------------
1 | // import Routes from "../routes/Index";
2 |
3 | // export default () => <>{Routes}>;
4 |
5 | import React, { useState, useEffect } from "react";
6 | import { BrowserRouter as Router, Route, Switch } from "react-router-dom";
7 | import Home from "./Home";
8 | import Blogs from "./Blogs";
9 | import Blog from "./Blog";
10 | import BlogForm from "../shared/BlogForm";
11 | import Header from "../shared/Header";
12 |
13 | const App = () => {
14 | const [isDataUpdated, setIsDataUpdated] = useState(false);
15 | const [editAlert, setEditAlert] = useState(false);
16 | const [createAlert, setCreateAlert] = useState(false);
17 |
18 | const handleData = status => {
19 | setIsDataUpdated(status);
20 | };
21 |
22 | const handleEditAlert = status => {
23 | setEditAlert(status);
24 | };
25 |
26 | const handleCreateAlert = status => {
27 | setCreateAlert(status);
28 | };
29 |
30 | return (
31 |
32 |
33 |
34 |
35 | {
39 | return (
40 |
47 | );
48 | }}
49 | />
50 | {
54 | return (
55 |
62 | );
63 | }}
64 | />
65 | {
69 | return (
70 |
77 | );
78 | }}
79 | />
80 | {
84 | return (
85 |
92 | );
93 | }}
94 | />
95 |
96 |
97 |
98 |
99 | );
100 | };
101 |
102 | export default App;
103 |
--------------------------------------------------------------------------------
/app/javascript/components/Blog.js:
--------------------------------------------------------------------------------
1 | import React, { useState, useEffect } from "react";
2 | import PageLoader from "../shared/PageLoader";
3 |
4 | const Blog = ({ match }) => {
5 | const [blog, setBlog] = useState({});
6 | useEffect(() => {
7 | fetch(`/api/v1/blogs/${match.params.id}`)
8 | .then(response => response.json())
9 | .then(response => {
10 | setBlog(response);
11 | });
12 | }, []);
13 |
14 | return (
15 | <>
16 | {blog.id ? (
17 |
18 |
19 |

24 |
25 |
26 |
27 |
28 | Title:
29 | {blog.title}
30 |
31 |
32 | Title:
33 | {blog.author}
34 |
35 |
36 |
Content:
37 |
{blog.content}
38 |
39 |
40 |
41 |
42 | ) : (
43 |
44 | )}
45 | >
46 | );
47 | };
48 |
49 | export default Blog;
50 |
--------------------------------------------------------------------------------
/app/javascript/components/Blogs.js:
--------------------------------------------------------------------------------
1 | import React, { useEffect, useState } from "react";
2 | import { Link } from "react-router-dom";
3 | import Dotdotdot from "react-dotdotdot";
4 | import Alert from "../shared/Alert";
5 | import { Img } from "react-image";
6 | import PageLoader from "../shared/PageLoader";
7 |
8 | const Blogs = props => {
9 | const [isDataUpdated, setIsDataUpdated] = useState(props.isDataUpdated);
10 | const [blogs, setBlogs] = useState([]);
11 | const [createAlert, setCreateAlert] = useState(props.createAlert);
12 | const [editAlert, setEditAlert] = useState(props.editAlert);
13 | const [showAlert, setShowAlert] = useState(false);
14 | const [color, setColor] = useState("");
15 | const [message, setMessage] = useState("");
16 |
17 | const fetchRequest = () => {
18 | const url = "/api/v1/blogs";
19 | fetch(url)
20 | .then(response => response.json())
21 | .then(response => {
22 | setBlogs(response);
23 | });
24 | setIsDataUpdated(false);
25 | };
26 |
27 | useEffect(() => {
28 | if (createAlert) {
29 | setShowAlert(true);
30 | setColor("teal");
31 | setMessage("Blog created successfully");
32 | } else if (editAlert) {
33 | setShowAlert(true);
34 | setColor("teal");
35 | setMessage("Blog updated successfully");
36 | }
37 | fetchRequest();
38 | }, [isDataUpdated]);
39 |
40 | const deleteBlog = id => {
41 | const token = document.querySelector('meta[name="csrf-token"]').content;
42 | fetch(`/api/v1/blogs/destroy/${id}`, {
43 | method: "DELETE",
44 | headers: {
45 | "X-CSRF-Token": token,
46 | "Content-Type": "application/json"
47 | }
48 | })
49 | .then(response => {
50 | if (response.ok) {
51 | return response.json();
52 | }
53 | throw new Error("Network response was not ok.");
54 | })
55 | .then(() => props.history.push("/"))
56 | .catch(error => console.log(error.message));
57 | // setBlogId(id);
58 | setIsDataUpdated(true);
59 | setShowAlert(true);
60 | setColor("red");
61 | setMessage("Blog deleted successfully");
62 | setEditAlert(false);
63 | setCreateAlert(false);
64 | };
65 |
66 | return (
67 | <>
68 | {showAlert ? : null}
69 |
70 | {blogs &&
71 | blogs.map(blog => {
72 | return (
73 |
77 |
78 |

}
83 | />
84 |
85 |
86 |
87 | {blog.title}
88 |
89 |
90 | {blog.content}
91 |
92 |
93 |
94 |
98 | {blog.author}
99 |
100 |
101 |
102 |
103 |
104 | Edit
105 |
106 |
107 | deleteBlog(blog.id)}
109 | className="inline-block bg-red-200 rounded-full px-3 py-1 ml-2 text-sm font-semibold text-gray-700 cursor-pointer"
110 | >
111 | Delete
112 |
113 |
114 |
115 |
116 | );
117 | })}
118 |
119 | >
120 | );
121 | };
122 |
123 | export default Blogs;
124 |
--------------------------------------------------------------------------------
/app/javascript/components/Home.js:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import Blogs from "./Blogs";
3 |
4 | const Home = props => {
5 | return ;
6 | };
7 |
8 | export default Home;
9 |
--------------------------------------------------------------------------------
/app/javascript/packs/application.js:
--------------------------------------------------------------------------------
1 | // This file is automatically compiled by Webpack, along with any other files
2 | // present in this directory. You're encouraged to place your actual application logic in
3 | // a relevant structure within app/javascript and only use these pack files to reference
4 | // that code so it'll be compiled.
5 |
6 | require("@rails/ujs").start();
7 | require("turbolinks").start();
8 | require("@rails/activestorage").start();
9 | require("channels");
10 |
11 | // Tailwind
12 | import "stylesheets/application";
13 | // Uncomment to copy all static images under ../images to the output folder and reference
14 | // them with the image_pack_tag helper in views (e.g <%= image_pack_tag 'rails.png' %>)
15 | // or the `imagePath` JavaScript helper below.
16 | //
17 | // const images = require.context('../images', true)
18 | // const imagePath = (name) => images(name, true)
19 |
--------------------------------------------------------------------------------
/app/javascript/packs/index.jsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import ReactDOM from "react-dom";
3 | import App from "../components/App";
4 |
5 | document.addEventListener("DOMContentLoaded", () => {
6 | ReactDOM.render(
7 | ,
8 | document.body.appendChild(document.createElement("div"))
9 | );
10 | });
11 |
--------------------------------------------------------------------------------
/app/javascript/routes/Index.js:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import { BrowserRouter as Router, Route, Switch } from "react-router-dom";
3 | import Home from "../components/Home";
4 | import Blogs from "../components/Blogs";
5 | import Blog from "../components/Blog";
6 | import BlogForm from "../shared/BlogForm";
7 | import Header from "../shared/Header";
8 |
9 | export default (
10 |
11 |
12 |
13 |
14 |
15 |
16 | {
20 | return ;
21 | }}
22 | />
23 | {
27 | return ;
28 | }}
29 | />
30 |
31 |
32 |
33 |
34 | );
35 |
--------------------------------------------------------------------------------
/app/javascript/shared/Alert.js:
--------------------------------------------------------------------------------
1 | import React, { useState } from "react";
2 | import { FiBell } from "react-icons/fi";
3 |
4 | const Alert = ({ color, message }) => {
5 | const [showAlert, setShowAlert] = useState(true);
6 | return (
7 | <>
8 | {showAlert ? (
9 |
16 |
17 |
18 |
19 |
20 | {message}
21 |
22 |
28 |
29 | ) : null}
30 | >
31 | );
32 | };
33 |
34 | export default Alert;
35 |
--------------------------------------------------------------------------------
/app/javascript/shared/BlogForm.js:
--------------------------------------------------------------------------------
1 | import React, { useState, useEffect } from "react";
2 | import Alert from "./Alert";
3 |
4 | const BlogForm = props => {
5 | const { match, history, editMode, handleData } = props;
6 | const [alert, setAlert] = useState(false);
7 | const [color, setColor] = useState("");
8 | const [message, setMessage] = useState("");
9 | const [title, setTitle] = useState("");
10 | const [author, setAuthor] = useState("");
11 | const [content, setContent] = useState("");
12 |
13 | useEffect(() => {
14 | if (editMode) {
15 | fetch(`/api/v1/blogs/edit/${match.params.id}`)
16 | .then(response => response.json())
17 | .then(response => {
18 | setTitle(response.title);
19 | setAuthor(response.author);
20 | setContent(response.content);
21 | });
22 | } else {
23 | setTitle("");
24 | setAuthor("");
25 | setContent("");
26 | }
27 | }, [editMode]);
28 |
29 | const sendRequest = url => {
30 | const body = {
31 | title,
32 | author,
33 | content: content.replace(/\n/g, "
")
34 | };
35 |
36 | const token = document.querySelector('meta[name="csrf-token"]').content;
37 | fetch(url, {
38 | method: editMode ? "PUT" : "POST",
39 | headers: {
40 | "X-CSRF-Token": token,
41 | "Content-Type": "application/json"
42 | },
43 | body: JSON.stringify(body)
44 | })
45 | .then(response => {
46 | if (response.ok) {
47 | return response.json();
48 | }
49 | throw new Error("Network response was not ok.");
50 | })
51 | .then(response => {
52 | if (editMode) {
53 | props.handleEditAlert(true);
54 | props.handleCreateAlert(false);
55 | } else {
56 | props.handleCreateAlert(true);
57 | props.handleEditAlert(false);
58 | }
59 | return history.push("/");
60 | })
61 | .catch(error => console.log(error.message));
62 | };
63 |
64 | const handleSubmit = event => {
65 | if (title.length == 0 || author.length == 0) {
66 | setAlert(true);
67 | setColor("red");
68 | setMessage("Please fill the required fields");
69 | } else {
70 | const url = editMode
71 | ? `/api/v1/blogs/update/${match.params.id}`
72 | : "/api/v1/blogs/create";
73 | sendRequest(url);
74 | handleData(true);
75 | }
76 | event.preventDefault();
77 | };
78 |
79 | return (
80 | <>
81 | {alert ? : null}
82 |
147 | >
148 | );
149 | };
150 |
151 | export default BlogForm;
152 |
--------------------------------------------------------------------------------
/app/javascript/shared/Header.js:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import { NavLink, Link } from "react-router-dom";
3 | import { FaGithub } from "react-icons/fa";
4 |
5 | const Header = () => {
6 | return (
7 |
8 |
62 |
63 |
75 |
76 |
77 | );
78 | };
79 |
80 | export default Header;
81 |
--------------------------------------------------------------------------------
/app/javascript/shared/PageLoader.js:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import LoaderGif from "images/loader.gif";
3 |
4 | const PageLoader = () => {
5 | return (
6 |
7 |

8 |
9 | );
10 | };
11 |
12 | export default PageLoader;
13 |
--------------------------------------------------------------------------------
/app/javascript/stylesheets/App.css:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/app/javascript/stylesheets/App.css
--------------------------------------------------------------------------------
/app/javascript/stylesheets/application.scss:
--------------------------------------------------------------------------------
1 | @import "tailwindcss/base";
2 | @import "tailwindcss/components";
3 | @import "tailwindcss/utilities";
4 |
5 | nav li a.active {
6 | border-color: #7f9cf5;
7 | }
8 |
--------------------------------------------------------------------------------
/app/javascript/stylesheets/tailwind.config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | purge: [],
3 | target: 'relaxed',
4 | prefix: '',
5 | important: false,
6 | separator: ':',
7 | theme: {
8 | screens: {
9 | sm: '640px',
10 | md: '768px',
11 | lg: '1024px',
12 | xl: '1280px',
13 | },
14 | colors: {
15 | transparent: 'transparent',
16 | current: 'currentColor',
17 |
18 | black: '#000',
19 | white: '#fff',
20 |
21 | gray: {
22 | 100: '#f7fafc',
23 | 200: '#edf2f7',
24 | 300: '#e2e8f0',
25 | 400: '#cbd5e0',
26 | 500: '#a0aec0',
27 | 600: '#718096',
28 | 700: '#4a5568',
29 | 800: '#2d3748',
30 | 900: '#1a202c',
31 | },
32 | red: {
33 | 100: '#fff5f5',
34 | 200: '#fed7d7',
35 | 300: '#feb2b2',
36 | 400: '#fc8181',
37 | 500: '#f56565',
38 | 600: '#e53e3e',
39 | 700: '#c53030',
40 | 800: '#9b2c2c',
41 | 900: '#742a2a',
42 | },
43 | orange: {
44 | 100: '#fffaf0',
45 | 200: '#feebc8',
46 | 300: '#fbd38d',
47 | 400: '#f6ad55',
48 | 500: '#ed8936',
49 | 600: '#dd6b20',
50 | 700: '#c05621',
51 | 800: '#9c4221',
52 | 900: '#7b341e',
53 | },
54 | yellow: {
55 | 100: '#fffff0',
56 | 200: '#fefcbf',
57 | 300: '#faf089',
58 | 400: '#f6e05e',
59 | 500: '#ecc94b',
60 | 600: '#d69e2e',
61 | 700: '#b7791f',
62 | 800: '#975a16',
63 | 900: '#744210',
64 | },
65 | green: {
66 | 100: '#f0fff4',
67 | 200: '#c6f6d5',
68 | 300: '#9ae6b4',
69 | 400: '#68d391',
70 | 500: '#48bb78',
71 | 600: '#38a169',
72 | 700: '#2f855a',
73 | 800: '#276749',
74 | 900: '#22543d',
75 | },
76 | teal: {
77 | 100: '#e6fffa',
78 | 200: '#b2f5ea',
79 | 300: '#81e6d9',
80 | 400: '#4fd1c5',
81 | 500: '#38b2ac',
82 | 600: '#319795',
83 | 700: '#2c7a7b',
84 | 800: '#285e61',
85 | 900: '#234e52',
86 | },
87 | blue: {
88 | 100: '#ebf8ff',
89 | 200: '#bee3f8',
90 | 300: '#90cdf4',
91 | 400: '#63b3ed',
92 | 500: '#4299e1',
93 | 600: '#3182ce',
94 | 700: '#2b6cb0',
95 | 800: '#2c5282',
96 | 900: '#2a4365',
97 | },
98 | indigo: {
99 | 100: '#ebf4ff',
100 | 200: '#c3dafe',
101 | 300: '#a3bffa',
102 | 400: '#7f9cf5',
103 | 500: '#667eea',
104 | 600: '#5a67d8',
105 | 700: '#4c51bf',
106 | 800: '#434190',
107 | 900: '#3c366b',
108 | },
109 | purple: {
110 | 100: '#faf5ff',
111 | 200: '#e9d8fd',
112 | 300: '#d6bcfa',
113 | 400: '#b794f4',
114 | 500: '#9f7aea',
115 | 600: '#805ad5',
116 | 700: '#6b46c1',
117 | 800: '#553c9a',
118 | 900: '#44337a',
119 | },
120 | pink: {
121 | 100: '#fff5f7',
122 | 200: '#fed7e2',
123 | 300: '#fbb6ce',
124 | 400: '#f687b3',
125 | 500: '#ed64a6',
126 | 600: '#d53f8c',
127 | 700: '#b83280',
128 | 800: '#97266d',
129 | 900: '#702459',
130 | },
131 | },
132 | spacing: {
133 | px: '1px',
134 | '0': '0',
135 | '1': '0.25rem',
136 | '2': '0.5rem',
137 | '3': '0.75rem',
138 | '4': '1rem',
139 | '5': '1.25rem',
140 | '6': '1.5rem',
141 | '8': '2rem',
142 | '10': '2.5rem',
143 | '12': '3rem',
144 | '16': '4rem',
145 | '20': '5rem',
146 | '24': '6rem',
147 | '32': '8rem',
148 | '40': '10rem',
149 | '48': '12rem',
150 | '56': '14rem',
151 | '64': '16rem',
152 | },
153 | backgroundColor: theme => theme('colors'),
154 | backgroundOpacity: theme => theme('opacity'),
155 | backgroundPosition: {
156 | bottom: 'bottom',
157 | center: 'center',
158 | left: 'left',
159 | 'left-bottom': 'left bottom',
160 | 'left-top': 'left top',
161 | right: 'right',
162 | 'right-bottom': 'right bottom',
163 | 'right-top': 'right top',
164 | top: 'top',
165 | },
166 | backgroundSize: {
167 | auto: 'auto',
168 | cover: 'cover',
169 | contain: 'contain',
170 | },
171 | borderColor: theme => ({
172 | ...theme('colors'),
173 | default: theme('colors.gray.300', 'currentColor'),
174 | }),
175 | borderOpacity: theme => theme('opacity'),
176 | borderRadius: {
177 | none: '0',
178 | sm: '0.125rem',
179 | default: '0.25rem',
180 | md: '0.375rem',
181 | lg: '0.5rem',
182 | full: '9999px',
183 | },
184 | borderWidth: {
185 | default: '1px',
186 | '0': '0',
187 | '2': '2px',
188 | '4': '4px',
189 | '8': '8px',
190 | },
191 | boxShadow: {
192 | xs: '0 0 0 1px rgba(0, 0, 0, 0.05)',
193 | sm: '0 1px 2px 0 rgba(0, 0, 0, 0.05)',
194 | default: '0 1px 3px 0 rgba(0, 0, 0, 0.1), 0 1px 2px 0 rgba(0, 0, 0, 0.06)',
195 | md: '0 4px 6px -1px rgba(0, 0, 0, 0.1), 0 2px 4px -1px rgba(0, 0, 0, 0.06)',
196 | lg: '0 10px 15px -3px rgba(0, 0, 0, 0.1), 0 4px 6px -2px rgba(0, 0, 0, 0.05)',
197 | xl: '0 20px 25px -5px rgba(0, 0, 0, 0.1), 0 10px 10px -5px rgba(0, 0, 0, 0.04)',
198 | '2xl': '0 25px 50px -12px rgba(0, 0, 0, 0.25)',
199 | inner: 'inset 0 2px 4px 0 rgba(0, 0, 0, 0.06)',
200 | outline: '0 0 0 3px rgba(66, 153, 225, 0.5)',
201 | none: 'none',
202 | },
203 | container: {},
204 | cursor: {
205 | auto: 'auto',
206 | default: 'default',
207 | pointer: 'pointer',
208 | wait: 'wait',
209 | text: 'text',
210 | move: 'move',
211 | 'not-allowed': 'not-allowed',
212 | },
213 | divideColor: theme => theme('borderColor'),
214 | divideOpacity: theme => theme('borderOpacity'),
215 | divideWidth: theme => theme('borderWidth'),
216 | fill: {
217 | current: 'currentColor',
218 | },
219 | flex: {
220 | '1': '1 1 0%',
221 | auto: '1 1 auto',
222 | initial: '0 1 auto',
223 | none: 'none',
224 | },
225 | flexGrow: {
226 | '0': '0',
227 | default: '1',
228 | },
229 | flexShrink: {
230 | '0': '0',
231 | default: '1',
232 | },
233 | fontFamily: {
234 | sans: [
235 | 'system-ui',
236 | '-apple-system',
237 | 'BlinkMacSystemFont',
238 | '"Segoe UI"',
239 | 'Roboto',
240 | '"Helvetica Neue"',
241 | 'Arial',
242 | '"Noto Sans"',
243 | 'sans-serif',
244 | '"Apple Color Emoji"',
245 | '"Segoe UI Emoji"',
246 | '"Segoe UI Symbol"',
247 | '"Noto Color Emoji"',
248 | ],
249 | serif: ['Georgia', 'Cambria', '"Times New Roman"', 'Times', 'serif'],
250 | mono: ['Menlo', 'Monaco', 'Consolas', '"Liberation Mono"', '"Courier New"', 'monospace'],
251 | },
252 | fontSize: {
253 | xs: '0.75rem',
254 | sm: '0.875rem',
255 | base: '1rem',
256 | lg: '1.125rem',
257 | xl: '1.25rem',
258 | '2xl': '1.5rem',
259 | '3xl': '1.875rem',
260 | '4xl': '2.25rem',
261 | '5xl': '3rem',
262 | '6xl': '4rem',
263 | },
264 | fontWeight: {
265 | hairline: '100',
266 | thin: '200',
267 | light: '300',
268 | normal: '400',
269 | medium: '500',
270 | semibold: '600',
271 | bold: '700',
272 | extrabold: '800',
273 | black: '900',
274 | },
275 | height: theme => ({
276 | auto: 'auto',
277 | ...theme('spacing'),
278 | full: '100%',
279 | screen: '100vh',
280 | }),
281 | inset: {
282 | '0': '0',
283 | auto: 'auto',
284 | },
285 | letterSpacing: {
286 | tighter: '-0.05em',
287 | tight: '-0.025em',
288 | normal: '0',
289 | wide: '0.025em',
290 | wider: '0.05em',
291 | widest: '0.1em',
292 | },
293 | lineHeight: {
294 | none: '1',
295 | tight: '1.25',
296 | snug: '1.375',
297 | normal: '1.5',
298 | relaxed: '1.625',
299 | loose: '2',
300 | '3': '.75rem',
301 | '4': '1rem',
302 | '5': '1.25rem',
303 | '6': '1.5rem',
304 | '7': '1.75rem',
305 | '8': '2rem',
306 | '9': '2.25rem',
307 | '10': '2.5rem',
308 | },
309 | listStyleType: {
310 | none: 'none',
311 | disc: 'disc',
312 | decimal: 'decimal',
313 | },
314 | margin: (theme, { negative }) => ({
315 | auto: 'auto',
316 | ...theme('spacing'),
317 | ...negative(theme('spacing')),
318 | }),
319 | maxHeight: {
320 | full: '100%',
321 | screen: '100vh',
322 | },
323 | maxWidth: (theme, { breakpoints }) => ({
324 | none: 'none',
325 | xs: '20rem',
326 | sm: '24rem',
327 | md: '28rem',
328 | lg: '32rem',
329 | xl: '36rem',
330 | '2xl': '42rem',
331 | '3xl': '48rem',
332 | '4xl': '56rem',
333 | '5xl': '64rem',
334 | '6xl': '72rem',
335 | full: '100%',
336 | ...breakpoints(theme('screens')),
337 | }),
338 | minHeight: {
339 | '0': '0',
340 | full: '100%',
341 | screen: '100vh',
342 | },
343 | minWidth: {
344 | '0': '0',
345 | full: '100%',
346 | },
347 | objectPosition: {
348 | bottom: 'bottom',
349 | center: 'center',
350 | left: 'left',
351 | 'left-bottom': 'left bottom',
352 | 'left-top': 'left top',
353 | right: 'right',
354 | 'right-bottom': 'right bottom',
355 | 'right-top': 'right top',
356 | top: 'top',
357 | },
358 | opacity: {
359 | '0': '0',
360 | '25': '0.25',
361 | '50': '0.5',
362 | '75': '0.75',
363 | '100': '1',
364 | },
365 | order: {
366 | first: '-9999',
367 | last: '9999',
368 | none: '0',
369 | '1': '1',
370 | '2': '2',
371 | '3': '3',
372 | '4': '4',
373 | '5': '5',
374 | '6': '6',
375 | '7': '7',
376 | '8': '8',
377 | '9': '9',
378 | '10': '10',
379 | '11': '11',
380 | '12': '12',
381 | },
382 | padding: theme => theme('spacing'),
383 | placeholderColor: theme => theme('colors'),
384 | placeholderOpacity: theme => theme('opacity'),
385 | space: (theme, { negative }) => ({
386 | ...theme('spacing'),
387 | ...negative(theme('spacing')),
388 | }),
389 | stroke: {
390 | current: 'currentColor',
391 | },
392 | strokeWidth: {
393 | '0': '0',
394 | '1': '1',
395 | '2': '2',
396 | },
397 | textColor: theme => theme('colors'),
398 | textOpacity: theme => theme('opacity'),
399 | width: theme => ({
400 | auto: 'auto',
401 | ...theme('spacing'),
402 | '1/2': '50%',
403 | '1/3': '33.333333%',
404 | '2/3': '66.666667%',
405 | '1/4': '25%',
406 | '2/4': '50%',
407 | '3/4': '75%',
408 | '1/5': '20%',
409 | '2/5': '40%',
410 | '3/5': '60%',
411 | '4/5': '80%',
412 | '1/6': '16.666667%',
413 | '2/6': '33.333333%',
414 | '3/6': '50%',
415 | '4/6': '66.666667%',
416 | '5/6': '83.333333%',
417 | '1/12': '8.333333%',
418 | '2/12': '16.666667%',
419 | '3/12': '25%',
420 | '4/12': '33.333333%',
421 | '5/12': '41.666667%',
422 | '6/12': '50%',
423 | '7/12': '58.333333%',
424 | '8/12': '66.666667%',
425 | '9/12': '75%',
426 | '10/12': '83.333333%',
427 | '11/12': '91.666667%',
428 | full: '100%',
429 | screen: '100vw',
430 | }),
431 | zIndex: {
432 | auto: 'auto',
433 | '0': '0',
434 | '10': '10',
435 | '20': '20',
436 | '30': '30',
437 | '40': '40',
438 | '50': '50',
439 | },
440 | gap: theme => theme('spacing'),
441 | gridTemplateColumns: {
442 | none: 'none',
443 | '1': 'repeat(1, minmax(0, 1fr))',
444 | '2': 'repeat(2, minmax(0, 1fr))',
445 | '3': 'repeat(3, minmax(0, 1fr))',
446 | '4': 'repeat(4, minmax(0, 1fr))',
447 | '5': 'repeat(5, minmax(0, 1fr))',
448 | '6': 'repeat(6, minmax(0, 1fr))',
449 | '7': 'repeat(7, minmax(0, 1fr))',
450 | '8': 'repeat(8, minmax(0, 1fr))',
451 | '9': 'repeat(9, minmax(0, 1fr))',
452 | '10': 'repeat(10, minmax(0, 1fr))',
453 | '11': 'repeat(11, minmax(0, 1fr))',
454 | '12': 'repeat(12, minmax(0, 1fr))',
455 | },
456 | gridColumn: {
457 | auto: 'auto',
458 | 'span-1': 'span 1 / span 1',
459 | 'span-2': 'span 2 / span 2',
460 | 'span-3': 'span 3 / span 3',
461 | 'span-4': 'span 4 / span 4',
462 | 'span-5': 'span 5 / span 5',
463 | 'span-6': 'span 6 / span 6',
464 | 'span-7': 'span 7 / span 7',
465 | 'span-8': 'span 8 / span 8',
466 | 'span-9': 'span 9 / span 9',
467 | 'span-10': 'span 10 / span 10',
468 | 'span-11': 'span 11 / span 11',
469 | 'span-12': 'span 12 / span 12',
470 | },
471 | gridColumnStart: {
472 | auto: 'auto',
473 | '1': '1',
474 | '2': '2',
475 | '3': '3',
476 | '4': '4',
477 | '5': '5',
478 | '6': '6',
479 | '7': '7',
480 | '8': '8',
481 | '9': '9',
482 | '10': '10',
483 | '11': '11',
484 | '12': '12',
485 | '13': '13',
486 | },
487 | gridColumnEnd: {
488 | auto: 'auto',
489 | '1': '1',
490 | '2': '2',
491 | '3': '3',
492 | '4': '4',
493 | '5': '5',
494 | '6': '6',
495 | '7': '7',
496 | '8': '8',
497 | '9': '9',
498 | '10': '10',
499 | '11': '11',
500 | '12': '12',
501 | '13': '13',
502 | },
503 | gridTemplateRows: {
504 | none: 'none',
505 | '1': 'repeat(1, minmax(0, 1fr))',
506 | '2': 'repeat(2, minmax(0, 1fr))',
507 | '3': 'repeat(3, minmax(0, 1fr))',
508 | '4': 'repeat(4, minmax(0, 1fr))',
509 | '5': 'repeat(5, minmax(0, 1fr))',
510 | '6': 'repeat(6, minmax(0, 1fr))',
511 | },
512 | gridRow: {
513 | auto: 'auto',
514 | 'span-1': 'span 1 / span 1',
515 | 'span-2': 'span 2 / span 2',
516 | 'span-3': 'span 3 / span 3',
517 | 'span-4': 'span 4 / span 4',
518 | 'span-5': 'span 5 / span 5',
519 | 'span-6': 'span 6 / span 6',
520 | },
521 | gridRowStart: {
522 | auto: 'auto',
523 | '1': '1',
524 | '2': '2',
525 | '3': '3',
526 | '4': '4',
527 | '5': '5',
528 | '6': '6',
529 | '7': '7',
530 | },
531 | gridRowEnd: {
532 | auto: 'auto',
533 | '1': '1',
534 | '2': '2',
535 | '3': '3',
536 | '4': '4',
537 | '5': '5',
538 | '6': '6',
539 | '7': '7',
540 | },
541 | transformOrigin: {
542 | center: 'center',
543 | top: 'top',
544 | 'top-right': 'top right',
545 | right: 'right',
546 | 'bottom-right': 'bottom right',
547 | bottom: 'bottom',
548 | 'bottom-left': 'bottom left',
549 | left: 'left',
550 | 'top-left': 'top left',
551 | },
552 | scale: {
553 | '0': '0',
554 | '50': '.5',
555 | '75': '.75',
556 | '90': '.9',
557 | '95': '.95',
558 | '100': '1',
559 | '105': '1.05',
560 | '110': '1.1',
561 | '125': '1.25',
562 | '150': '1.5',
563 | },
564 | rotate: {
565 | '-180': '-180deg',
566 | '-90': '-90deg',
567 | '-45': '-45deg',
568 | '0': '0',
569 | '45': '45deg',
570 | '90': '90deg',
571 | '180': '180deg',
572 | },
573 | translate: (theme, { negative }) => ({
574 | ...theme('spacing'),
575 | ...negative(theme('spacing')),
576 | '-full': '-100%',
577 | '-1/2': '-50%',
578 | '1/2': '50%',
579 | full: '100%',
580 | }),
581 | skew: {
582 | '-12': '-12deg',
583 | '-6': '-6deg',
584 | '-3': '-3deg',
585 | '0': '0',
586 | '3': '3deg',
587 | '6': '6deg',
588 | '12': '12deg',
589 | },
590 | transitionProperty: {
591 | none: 'none',
592 | all: 'all',
593 | default: 'background-color, border-color, color, fill, stroke, opacity, box-shadow, transform',
594 | colors: 'background-color, border-color, color, fill, stroke',
595 | opacity: 'opacity',
596 | shadow: 'box-shadow',
597 | transform: 'transform',
598 | },
599 | transitionTimingFunction: {
600 | linear: 'linear',
601 | in: 'cubic-bezier(0.4, 0, 1, 1)',
602 | out: 'cubic-bezier(0, 0, 0.2, 1)',
603 | 'in-out': 'cubic-bezier(0.4, 0, 0.2, 1)',
604 | },
605 | transitionDuration: {
606 | '75': '75ms',
607 | '100': '100ms',
608 | '150': '150ms',
609 | '200': '200ms',
610 | '300': '300ms',
611 | '500': '500ms',
612 | '700': '700ms',
613 | '1000': '1000ms',
614 | },
615 | transitionDelay: {
616 | '75': '75ms',
617 | '100': '100ms',
618 | '150': '150ms',
619 | '200': '200ms',
620 | '300': '300ms',
621 | '500': '500ms',
622 | '700': '700ms',
623 | '1000': '1000ms',
624 | },
625 | animation: {
626 | none: 'none',
627 | spin: 'spin 1s linear infinite',
628 | ping: 'ping 1s cubic-bezier(0, 0, 0.2, 1) infinite',
629 | pulse: 'pulse 2s cubic-bezier(0.4, 0, 0.6, 1) infinite',
630 | bounce: 'bounce 1s infinite',
631 | },
632 | keyframes: {
633 | spin: {
634 | from: { transform: 'rotate(0deg)' },
635 | to: { transform: 'rotate(360deg)' },
636 | },
637 | ping: {
638 | '0%': { transform: 'scale(1)', opacity: '1' },
639 | '75%, 100%': { transform: 'scale(2)', opacity: '0' },
640 | },
641 | pulse: {
642 | '0%, 100%': { opacity: '1' },
643 | '50%': { opacity: '.5' },
644 | },
645 | bounce: {
646 | '0%, 100%': {
647 | transform: 'translateY(-25%)',
648 | animationTimingFunction: 'cubic-bezier(0.8,0,1,1)',
649 | },
650 | '50%': {
651 | transform: 'translateY(0)',
652 | animationTimingFunction: 'cubic-bezier(0,0,0.2,1)',
653 | },
654 | },
655 | },
656 | },
657 | variants: {
658 | accessibility: ['responsive', 'focus'],
659 | alignContent: ['responsive'],
660 | alignItems: ['responsive'],
661 | alignSelf: ['responsive'],
662 | appearance: ['responsive'],
663 | backgroundAttachment: ['responsive'],
664 | backgroundColor: ['responsive', 'hover', 'focus'],
665 | backgroundOpacity: ['responsive', 'hover', 'focus'],
666 | backgroundPosition: ['responsive'],
667 | backgroundRepeat: ['responsive'],
668 | backgroundSize: ['responsive'],
669 | borderCollapse: ['responsive'],
670 | borderColor: ['responsive', 'hover', 'focus'],
671 | borderOpacity: ['responsive', 'hover', 'focus'],
672 | borderRadius: ['responsive'],
673 | borderStyle: ['responsive'],
674 | borderWidth: ['responsive'],
675 | boxShadow: ['responsive', 'hover', 'focus'],
676 | boxSizing: ['responsive'],
677 | container: ['responsive'],
678 | cursor: ['responsive'],
679 | display: ['responsive'],
680 | divideColor: ['responsive'],
681 | divideOpacity: ['responsive'],
682 | divideWidth: ['responsive'],
683 | fill: ['responsive'],
684 | flex: ['responsive'],
685 | flexDirection: ['responsive'],
686 | flexGrow: ['responsive'],
687 | flexShrink: ['responsive'],
688 | flexWrap: ['responsive'],
689 | float: ['responsive'],
690 | clear: ['responsive'],
691 | fontFamily: ['responsive'],
692 | fontSize: ['responsive'],
693 | fontSmoothing: ['responsive'],
694 | fontStyle: ['responsive'],
695 | fontWeight: ['responsive', 'hover', 'focus'],
696 | height: ['responsive'],
697 | inset: ['responsive'],
698 | justifyContent: ['responsive'],
699 | letterSpacing: ['responsive'],
700 | lineHeight: ['responsive'],
701 | listStylePosition: ['responsive'],
702 | listStyleType: ['responsive'],
703 | margin: ['responsive'],
704 | maxHeight: ['responsive'],
705 | maxWidth: ['responsive'],
706 | minHeight: ['responsive'],
707 | minWidth: ['responsive'],
708 | objectFit: ['responsive'],
709 | objectPosition: ['responsive'],
710 | opacity: ['responsive', 'hover', 'focus'],
711 | order: ['responsive'],
712 | outline: ['responsive', 'focus'],
713 | overflow: ['responsive'],
714 | overscrollBehavior: ['responsive'],
715 | padding: ['responsive'],
716 | placeholderColor: ['responsive', 'focus'],
717 | placeholderOpacity: ['responsive', 'focus'],
718 | pointerEvents: ['responsive'],
719 | position: ['responsive'],
720 | resize: ['responsive'],
721 | space: ['responsive'],
722 | stroke: ['responsive'],
723 | strokeWidth: ['responsive'],
724 | tableLayout: ['responsive'],
725 | textAlign: ['responsive'],
726 | textColor: ['responsive', 'hover', 'focus'],
727 | textOpacity: ['responsive', 'hover', 'focus'],
728 | textDecoration: ['responsive', 'hover', 'focus'],
729 | textTransform: ['responsive'],
730 | userSelect: ['responsive'],
731 | verticalAlign: ['responsive'],
732 | visibility: ['responsive'],
733 | whitespace: ['responsive'],
734 | width: ['responsive'],
735 | wordBreak: ['responsive'],
736 | zIndex: ['responsive'],
737 | gap: ['responsive'],
738 | gridAutoFlow: ['responsive'],
739 | gridTemplateColumns: ['responsive'],
740 | gridColumn: ['responsive'],
741 | gridColumnStart: ['responsive'],
742 | gridColumnEnd: ['responsive'],
743 | gridTemplateRows: ['responsive'],
744 | gridRow: ['responsive'],
745 | gridRowStart: ['responsive'],
746 | gridRowEnd: ['responsive'],
747 | transform: ['responsive'],
748 | transformOrigin: ['responsive'],
749 | scale: ['responsive', 'hover', 'focus'],
750 | rotate: ['responsive', 'hover', 'focus'],
751 | translate: ['responsive', 'hover', 'focus'],
752 | skew: ['responsive', 'hover', 'focus'],
753 | transitionProperty: ['responsive'],
754 | transitionTimingFunction: ['responsive'],
755 | transitionDuration: ['responsive'],
756 | transitionDelay: ['responsive'],
757 | animation: ['responsive'],
758 | },
759 | corePlugins: {},
760 | plugins: [],
761 | }
762 |
--------------------------------------------------------------------------------
/app/jobs/application_job.rb:
--------------------------------------------------------------------------------
1 | class ApplicationJob < ActiveJob::Base
2 | # Automatically retry jobs that encountered a deadlock
3 | # retry_on ActiveRecord::Deadlocked
4 |
5 | # Most jobs are safe to ignore if the underlying records are no longer available
6 | # discard_on ActiveJob::DeserializationError
7 | end
8 |
--------------------------------------------------------------------------------
/app/mailers/application_mailer.rb:
--------------------------------------------------------------------------------
1 | class ApplicationMailer < ActionMailer::Base
2 | default from: 'from@example.com'
3 | layout 'mailer'
4 | end
5 |
--------------------------------------------------------------------------------
/app/models/application_record.rb:
--------------------------------------------------------------------------------
1 | class ApplicationRecord < ActiveRecord::Base
2 | self.abstract_class = true
3 | end
4 |
--------------------------------------------------------------------------------
/app/models/blog.rb:
--------------------------------------------------------------------------------
1 | class Blog < ApplicationRecord
2 | end
3 |
--------------------------------------------------------------------------------
/app/models/concerns/.keep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/app/models/concerns/.keep
--------------------------------------------------------------------------------
/app/views/home/index.html.erb:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/app/views/home/index.html.erb
--------------------------------------------------------------------------------
/app/views/layouts/application.html.erb:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | ReactRailsBlog
5 | <%= csrf_meta_tags %>
6 | <%= csp_meta_tag %>
7 |
8 |
9 |
10 | <%= stylesheet_link_tag 'application', media: 'all', 'data-turbolinks-track': 'reload' %>
11 | <%= stylesheet_pack_tag 'application', 'data-turbolinks-track': 'reload' %>
12 | <%= javascript_pack_tag 'application', 'data-turbolinks-track': 'reload' %>
13 | <%= javascript_pack_tag 'index' %>
14 |
15 |
16 |
17 | <%= yield %>
18 |
19 |
20 |
--------------------------------------------------------------------------------
/app/views/layouts/mailer.html.erb:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
8 |
9 |
10 |
11 | <%= yield %>
12 |
13 |
14 |
--------------------------------------------------------------------------------
/app/views/layouts/mailer.text.erb:
--------------------------------------------------------------------------------
1 | <%= yield %>
2 |
--------------------------------------------------------------------------------
/babel.config.js:
--------------------------------------------------------------------------------
1 | module.exports = function(api) {
2 | var validEnv = ['development', 'test', 'production']
3 | var currentEnv = api.env()
4 | var isDevelopmentEnv = api.env('development')
5 | var isProductionEnv = api.env('production')
6 | var isTestEnv = api.env('test')
7 |
8 | if (!validEnv.includes(currentEnv)) {
9 | throw new Error(
10 | 'Please specify a valid `NODE_ENV` or ' +
11 | '`BABEL_ENV` environment variables. Valid values are "development", ' +
12 | '"test", and "production". Instead, received: ' +
13 | JSON.stringify(currentEnv) +
14 | '.'
15 | )
16 | }
17 |
18 | return {
19 | presets: [
20 | isTestEnv && [
21 | '@babel/preset-env',
22 | {
23 | targets: {
24 | node: 'current'
25 | },
26 | modules: 'commonjs'
27 | },
28 | '@babel/preset-react'
29 | ],
30 | (isProductionEnv || isDevelopmentEnv) && [
31 | '@babel/preset-env',
32 | {
33 | forceAllTransforms: true,
34 | useBuiltIns: 'entry',
35 | corejs: 3,
36 | modules: false,
37 | exclude: ['transform-typeof-symbol']
38 | }
39 | ],
40 | [
41 | '@babel/preset-react',
42 | {
43 | development: isDevelopmentEnv || isTestEnv,
44 | useBuiltIns: true
45 | }
46 | ]
47 | ].filter(Boolean),
48 | plugins: [
49 | 'babel-plugin-macros',
50 | '@babel/plugin-syntax-dynamic-import',
51 | isTestEnv && 'babel-plugin-dynamic-import-node',
52 | '@babel/plugin-transform-destructuring',
53 | [
54 | '@babel/plugin-proposal-class-properties',
55 | {
56 | loose: true
57 | }
58 | ],
59 | [
60 | '@babel/plugin-proposal-object-rest-spread',
61 | {
62 | useBuiltIns: true
63 | }
64 | ],
65 | [
66 | '@babel/plugin-transform-runtime',
67 | {
68 | helpers: false,
69 | regenerator: true,
70 | corejs: false
71 | }
72 | ],
73 | [
74 | '@babel/plugin-transform-regenerator',
75 | {
76 | async: false
77 | }
78 | ],
79 | isProductionEnv && [
80 | 'babel-plugin-transform-react-remove-prop-types',
81 | {
82 | removeImport: true
83 | }
84 | ]
85 | ].filter(Boolean)
86 | }
87 | }
88 |
--------------------------------------------------------------------------------
/bin/bundle:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env ruby
2 | # frozen_string_literal: true
3 |
4 | #
5 | # This file was generated by Bundler.
6 | #
7 | # The application 'bundle' is installed as part of a gem, and
8 | # this file is here to facilitate running it.
9 | #
10 |
11 | require "rubygems"
12 |
13 | m = Module.new do
14 | module_function
15 |
16 | def invoked_as_script?
17 | File.expand_path($0) == File.expand_path(__FILE__)
18 | end
19 |
20 | def env_var_version
21 | ENV["BUNDLER_VERSION"]
22 | end
23 |
24 | def cli_arg_version
25 | return unless invoked_as_script? # don't want to hijack other binstubs
26 | return unless "update".start_with?(ARGV.first || " ") # must be running `bundle update`
27 | bundler_version = nil
28 | update_index = nil
29 | ARGV.each_with_index do |a, i|
30 | if update_index && update_index.succ == i && a =~ Gem::Version::ANCHORED_VERSION_PATTERN
31 | bundler_version = a
32 | end
33 | next unless a =~ /\A--bundler(?:[= ](#{Gem::Version::VERSION_PATTERN}))?\z/
34 | bundler_version = $1
35 | update_index = i
36 | end
37 | bundler_version
38 | end
39 |
40 | def gemfile
41 | gemfile = ENV["BUNDLE_GEMFILE"]
42 | return gemfile if gemfile && !gemfile.empty?
43 |
44 | File.expand_path("../../Gemfile", __FILE__)
45 | end
46 |
47 | def lockfile
48 | lockfile =
49 | case File.basename(gemfile)
50 | when "gems.rb" then gemfile.sub(/\.rb$/, gemfile)
51 | else "#{gemfile}.lock"
52 | end
53 | File.expand_path(lockfile)
54 | end
55 |
56 | def lockfile_version
57 | return unless File.file?(lockfile)
58 | lockfile_contents = File.read(lockfile)
59 | return unless lockfile_contents =~ /\n\nBUNDLED WITH\n\s{2,}(#{Gem::Version::VERSION_PATTERN})\n/
60 | Regexp.last_match(1)
61 | end
62 |
63 | def bundler_version
64 | @bundler_version ||=
65 | env_var_version || cli_arg_version ||
66 | lockfile_version
67 | end
68 |
69 | def bundler_requirement
70 | return "#{Gem::Requirement.default}.a" unless bundler_version
71 |
72 | bundler_gem_version = Gem::Version.new(bundler_version)
73 |
74 | requirement = bundler_gem_version.approximate_recommendation
75 |
76 | return requirement unless Gem::Version.new(Gem::VERSION) < Gem::Version.new("2.7.0")
77 |
78 | requirement += ".a" if bundler_gem_version.prerelease?
79 |
80 | requirement
81 | end
82 |
83 | def load_bundler!
84 | ENV["BUNDLE_GEMFILE"] ||= gemfile
85 |
86 | activate_bundler
87 | end
88 |
89 | def activate_bundler
90 | gem_error = activation_error_handling do
91 | gem "bundler", bundler_requirement
92 | end
93 | return if gem_error.nil?
94 | require_error = activation_error_handling do
95 | require "bundler/version"
96 | end
97 | return if require_error.nil? && Gem::Requirement.new(bundler_requirement).satisfied_by?(Gem::Version.new(Bundler::VERSION))
98 | warn "Activating bundler (#{bundler_requirement}) failed:\n#{gem_error.message}\n\nTo install the version of bundler this project requires, run `gem install bundler -v '#{bundler_requirement}'`"
99 | exit 42
100 | end
101 |
102 | def activation_error_handling
103 | yield
104 | nil
105 | rescue StandardError, LoadError => e
106 | e
107 | end
108 | end
109 |
110 | m.load_bundler!
111 |
112 | if m.invoked_as_script?
113 | load Gem.bin_path("bundler", "bundle")
114 | end
115 |
--------------------------------------------------------------------------------
/bin/rails:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env ruby
2 | begin
3 | load File.expand_path('../spring', __FILE__)
4 | rescue LoadError => e
5 | raise unless e.message.include?('spring')
6 | end
7 | APP_PATH = File.expand_path('../config/application', __dir__)
8 | require_relative '../config/boot'
9 | require 'rails/commands'
10 |
--------------------------------------------------------------------------------
/bin/rake:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env ruby
2 | begin
3 | load File.expand_path('../spring', __FILE__)
4 | rescue LoadError => e
5 | raise unless e.message.include?('spring')
6 | end
7 | require_relative '../config/boot'
8 | require 'rake'
9 | Rake.application.run
10 |
--------------------------------------------------------------------------------
/bin/setup:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env ruby
2 | require 'fileutils'
3 |
4 | # path to your application root.
5 | APP_ROOT = File.expand_path('..', __dir__)
6 |
7 | def system!(*args)
8 | system(*args) || abort("\n== Command #{args} failed ==")
9 | end
10 |
11 | FileUtils.chdir APP_ROOT do
12 | # This script is a way to setup or update your development environment automatically.
13 | # This script is idempotent, so that you can run it at anytime and get an expectable outcome.
14 | # Add necessary setup steps to this file.
15 |
16 | puts '== Installing dependencies =='
17 | system! 'gem install bundler --conservative'
18 | system('bundle check') || system!('bundle install')
19 |
20 | # Install JavaScript dependencies
21 | # system('bin/yarn')
22 |
23 | # puts "\n== Copying sample files =="
24 | # unless File.exist?('config/database.yml')
25 | # FileUtils.cp 'config/database.yml.sample', 'config/database.yml'
26 | # end
27 |
28 | puts "\n== Preparing database =="
29 | system! 'bin/rails db:prepare'
30 |
31 | puts "\n== Removing old logs and tempfiles =="
32 | system! 'bin/rails log:clear tmp:clear'
33 |
34 | puts "\n== Restarting application server =="
35 | system! 'bin/rails restart'
36 | end
37 |
--------------------------------------------------------------------------------
/bin/spring:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env ruby
2 |
3 | # This file loads Spring without using Bundler, in order to be fast.
4 | # It gets overwritten when you run the `spring binstub` command.
5 |
6 | unless defined?(Spring)
7 | require 'rubygems'
8 | require 'bundler'
9 |
10 | lockfile = Bundler::LockfileParser.new(Bundler.default_lockfile.read)
11 | spring = lockfile.specs.detect { |spec| spec.name == 'spring' }
12 | if spring
13 | Gem.use_paths Gem.dir, Bundler.bundle_path.to_s, *Gem.path
14 | gem 'spring', spring.version
15 | require 'spring/binstub'
16 | end
17 | end
18 |
--------------------------------------------------------------------------------
/bin/webpack:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env ruby
2 |
3 | ENV["RAILS_ENV"] ||= ENV["RACK_ENV"] || "development"
4 | ENV["NODE_ENV"] ||= "development"
5 |
6 | require "pathname"
7 | ENV["BUNDLE_GEMFILE"] ||= File.expand_path("../../Gemfile",
8 | Pathname.new(__FILE__).realpath)
9 |
10 | require "bundler/setup"
11 |
12 | require "webpacker"
13 | require "webpacker/webpack_runner"
14 |
15 | APP_ROOT = File.expand_path("..", __dir__)
16 | Dir.chdir(APP_ROOT) do
17 | Webpacker::WebpackRunner.run(ARGV)
18 | end
19 |
--------------------------------------------------------------------------------
/bin/webpack-dev-server:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env ruby
2 |
3 | ENV["RAILS_ENV"] ||= ENV["RACK_ENV"] || "development"
4 | ENV["NODE_ENV"] ||= "development"
5 |
6 | require "pathname"
7 | ENV["BUNDLE_GEMFILE"] ||= File.expand_path("../../Gemfile",
8 | Pathname.new(__FILE__).realpath)
9 |
10 | require "bundler/setup"
11 |
12 | require "webpacker"
13 | require "webpacker/dev_server_runner"
14 |
15 | APP_ROOT = File.expand_path("..", __dir__)
16 | Dir.chdir(APP_ROOT) do
17 | Webpacker::DevServerRunner.run(ARGV)
18 | end
19 |
--------------------------------------------------------------------------------
/bin/yarn:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env ruby
2 | APP_ROOT = File.expand_path('..', __dir__)
3 | Dir.chdir(APP_ROOT) do
4 | begin
5 | exec "yarnpkg", *ARGV
6 | rescue Errno::ENOENT
7 | $stderr.puts "Yarn executable was not detected in the system."
8 | $stderr.puts "Download Yarn at https://yarnpkg.com/en/docs/install"
9 | exit 1
10 | end
11 | end
12 |
--------------------------------------------------------------------------------
/config.ru:
--------------------------------------------------------------------------------
1 | # This file is used by Rack-based servers to start the application.
2 |
3 | require_relative 'config/environment'
4 |
5 | run Rails.application
6 |
--------------------------------------------------------------------------------
/config/application.rb:
--------------------------------------------------------------------------------
1 | require_relative 'boot'
2 |
3 | require "rails"
4 | # Pick the frameworks you want:
5 | require "active_model/railtie"
6 | require "active_job/railtie"
7 | require "active_record/railtie"
8 | require "active_storage/engine"
9 | require "action_controller/railtie"
10 | require "action_mailer/railtie"
11 | require "action_mailbox/engine"
12 | require "action_text/engine"
13 | require "action_view/railtie"
14 | require "action_cable/engine"
15 | require "sprockets/railtie"
16 | # require "rails/test_unit/railtie"
17 |
18 | # Require the gems listed in Gemfile, including any gems
19 | # you've limited to :test, :development, or :production.
20 | Bundler.require(*Rails.groups)
21 |
22 | module ReactRailsBlog
23 | class Application < Rails::Application
24 | # Initialize configuration defaults for originally generated Rails version.
25 | config.load_defaults 6.0
26 |
27 | # Settings in config/environments/* take precedence over those specified here.
28 | # Application configuration can go into files in config/initializers
29 | # -- all .rb files in that directory are automatically loaded after loading
30 | # the framework and any gems in your application.
31 |
32 | # Don't generate system test files.
33 | config.generators.system_tests = nil
34 | end
35 | end
36 |
--------------------------------------------------------------------------------
/config/boot.rb:
--------------------------------------------------------------------------------
1 | ENV['BUNDLE_GEMFILE'] ||= File.expand_path('../Gemfile', __dir__)
2 |
3 | require 'bundler/setup' # Set up gems listed in the Gemfile.
4 | require 'bootsnap/setup' # Speed up boot time by caching expensive operations.
5 |
--------------------------------------------------------------------------------
/config/cable.yml:
--------------------------------------------------------------------------------
1 | development:
2 | adapter: async
3 |
4 | test:
5 | adapter: test
6 |
7 | production:
8 | adapter: redis
9 | url: <%= ENV.fetch("REDIS_URL") { "redis://localhost:6379/1" } %>
10 | channel_prefix: react_rails_blog_production
11 |
--------------------------------------------------------------------------------
/config/credentials.yml.enc:
--------------------------------------------------------------------------------
1 | ceoYFBR6IBo3gJcU6OlA2naPBz6mWikItWzdG0D0mhqeNm2LepjRdwWtweu6UBiksdypC1enPSl3AQTTb6m35kP52aMXuKP8DVKIWgrO8D9N5yj8BMxKBMnIb3YVKJ6Qr8wwh5/sX8js91rYAaoUD6BMglqshy7oPB2lbYwgciL+O6bgByrkYpMzmK1Bt6g28WRC7OEF/54tkuX8kivG/pkqid/tlQrGxlWaJSMI1jXuwP5lGCEeRYKBTBLytMvpuffE8r8jyxp8PNvdgbri2rOds1U5Obd5MtKQCT534ykeXozNxAT24Q0FNGewCx1Hv5rXW9HxFeWLF3+2s2fT7I6I59RYTUD/KUe5z3lgp82czpT8XgE6ib/xA29JYUPFKWvaYytCyj+uXfv83BqFlPbKmE/iBoZ4q3e7--t9b/aLVNoAp1dGmy--D6gOSXTGFkLhfQQKoXWVuA==
--------------------------------------------------------------------------------
/config/database.yml:
--------------------------------------------------------------------------------
1 | # PostgreSQL. Versions 9.3 and up are supported.
2 | #
3 | # Install the pg driver:
4 | # gem install pg
5 | # On macOS with Homebrew:
6 | # gem install pg -- --with-pg-config=/usr/local/bin/pg_config
7 | # On macOS with MacPorts:
8 | # gem install pg -- --with-pg-config=/opt/local/lib/postgresql84/bin/pg_config
9 | # On Windows:
10 | # gem install pg
11 | # Choose the win32 build.
12 | # Install PostgreSQL and put its /bin directory on your path.
13 | #
14 | # Configure Using Gemfile
15 | # gem 'pg'
16 | #
17 | default: &default
18 | adapter: postgresql
19 | encoding: unicode
20 | # For details on connection pooling, see Rails configuration guide
21 | # https://guides.rubyonrails.org/configuring.html#database-pooling
22 | pool: <%= ENV.fetch("RAILS_MAX_THREADS") { 5 } %>
23 |
24 | development:
25 | <<: *default
26 | database: react_rails_blog_development
27 |
28 | # The specified database role being used to connect to postgres.
29 | # To create additional roles in postgres see `$ createuser --help`.
30 | # When left blank, postgres will use the default role. This is
31 | # the same name as the operating system user that initialized the database.
32 | #username: react_rails_blog
33 |
34 | # The password associated with the postgres role (username).
35 | #password:
36 |
37 | # Connect on a TCP socket. Omitted by default since the client uses a
38 | # domain socket that doesn't need configuration. Windows does not have
39 | # domain sockets, so uncomment these lines.
40 | #host: localhost
41 |
42 | # The TCP port the server listens on. Defaults to 5432.
43 | # If your server runs on a different port number, change accordingly.
44 | #port: 5432
45 |
46 | # Schema search path. The server defaults to $user,public
47 | #schema_search_path: myapp,sharedapp,public
48 |
49 | # Minimum log levels, in increasing order:
50 | # debug5, debug4, debug3, debug2, debug1,
51 | # log, notice, warning, error, fatal, and panic
52 | # Defaults to warning.
53 | #min_messages: notice
54 |
55 | # Warning: The database defined as "test" will be erased and
56 | # re-generated from your development database when you run "rake".
57 | # Do not set this db to the same as development or production.
58 | test:
59 | <<: *default
60 | database: react_rails_blog_test
61 |
62 | # As with config/credentials.yml, you never want to store sensitive information,
63 | # like your database password, in your source code. If your source code is
64 | # ever seen by anyone, they now have access to your database.
65 | #
66 | # Instead, provide the password as a unix environment variable when you boot
67 | # the app. Read https://guides.rubyonrails.org/configuring.html#configuring-a-database
68 | # for a full rundown on how to provide these environment variables in a
69 | # production deployment.
70 | #
71 | # On Heroku and other platform providers, you may have a full connection URL
72 | # available as an environment variable. For example:
73 | #
74 | # DATABASE_URL="postgres://myuser:mypass@localhost/somedatabase"
75 | #
76 | # You can use this database configuration with:
77 | #
78 | # production:
79 | # url: <%= ENV['DATABASE_URL'] %>
80 | #
81 | production:
82 | <<: *default
83 | database: react_rails_blog_production
84 | username: react_rails_blog
85 | password: <%= ENV['REACT_RAILS_BLOG_DATABASE_PASSWORD'] %>
86 |
--------------------------------------------------------------------------------
/config/environment.rb:
--------------------------------------------------------------------------------
1 | # Load the Rails application.
2 | require_relative 'application'
3 |
4 | # Initialize the Rails application.
5 | Rails.application.initialize!
6 |
--------------------------------------------------------------------------------
/config/environments/development.rb:
--------------------------------------------------------------------------------
1 | Rails.application.configure do
2 | # Settings specified here will take precedence over those in config/application.rb.
3 |
4 | # In the development environment your application's code is reloaded on
5 | # every request. This slows down response time but is perfect for development
6 | # since you don't have to restart the web server when you make code changes.
7 | config.cache_classes = false
8 |
9 | # Do not eager load code on boot.
10 | config.eager_load = false
11 |
12 | # Show full error reports.
13 | config.consider_all_requests_local = true
14 |
15 | # Enable/disable caching. By default caching is disabled.
16 | # Run rails dev:cache to toggle caching.
17 | if Rails.root.join('tmp', 'caching-dev.txt').exist?
18 | config.action_controller.perform_caching = true
19 | config.action_controller.enable_fragment_cache_logging = true
20 |
21 | config.cache_store = :memory_store
22 | config.public_file_server.headers = {
23 | 'Cache-Control' => "public, max-age=#{2.days.to_i}"
24 | }
25 | else
26 | config.action_controller.perform_caching = false
27 |
28 | config.cache_store = :null_store
29 | end
30 |
31 | # Store uploaded files on the local file system (see config/storage.yml for options).
32 | config.active_storage.service = :local
33 |
34 | # Don't care if the mailer can't send.
35 | config.action_mailer.raise_delivery_errors = false
36 |
37 | config.action_mailer.perform_caching = false
38 |
39 | # Print deprecation notices to the Rails logger.
40 | config.active_support.deprecation = :log
41 |
42 | # Raise an error on page load if there are pending migrations.
43 | config.active_record.migration_error = :page_load
44 |
45 | # Highlight code that triggered database queries in logs.
46 | config.active_record.verbose_query_logs = true
47 |
48 | # Debug mode disables concatenation and preprocessing of assets.
49 | # This option may cause significant delays in view rendering with a large
50 | # number of complex assets.
51 | config.assets.debug = true
52 |
53 | # Suppress logger output for asset requests.
54 | config.assets.quiet = true
55 |
56 | # Raises error for missing translations.
57 | # config.action_view.raise_on_missing_translations = true
58 |
59 | # Use an evented file watcher to asynchronously detect changes in source code,
60 | # routes, locales, etc. This feature depends on the listen gem.
61 | config.file_watcher = ActiveSupport::EventedFileUpdateChecker
62 | end
63 |
--------------------------------------------------------------------------------
/config/environments/production.rb:
--------------------------------------------------------------------------------
1 | Rails.application.configure do
2 | # Settings specified here will take precedence over those in config/application.rb.
3 |
4 | # Code is not reloaded between requests.
5 | config.cache_classes = true
6 |
7 | # Eager load code on boot. This eager loads most of Rails and
8 | # your application in memory, allowing both threaded web servers
9 | # and those relying on copy on write to perform better.
10 | # Rake tasks automatically ignore this option for performance.
11 | config.eager_load = true
12 |
13 | # Full error reports are disabled and caching is turned on.
14 | config.consider_all_requests_local = false
15 | config.action_controller.perform_caching = true
16 |
17 | # Ensures that a master key has been made available in either ENV["RAILS_MASTER_KEY"]
18 | # or in config/master.key. This key is used to decrypt credentials (and other encrypted files).
19 | # config.require_master_key = true
20 |
21 | # Disable serving static files from the `/public` folder by default since
22 | # Apache or NGINX already handles this.
23 | config.public_file_server.enabled = ENV['RAILS_SERVE_STATIC_FILES'].present?
24 |
25 | # Compress CSS using a preprocessor.
26 | # config.assets.css_compressor = :sass
27 |
28 | # Do not fallback to assets pipeline if a precompiled asset is missed.
29 | config.assets.compile = false
30 |
31 | # Enable serving of images, stylesheets, and JavaScripts from an asset server.
32 | # config.action_controller.asset_host = 'http://assets.example.com'
33 |
34 | # Specifies the header that your server uses for sending files.
35 | # config.action_dispatch.x_sendfile_header = 'X-Sendfile' # for Apache
36 | # config.action_dispatch.x_sendfile_header = 'X-Accel-Redirect' # for NGINX
37 |
38 | # Store uploaded files on the local file system (see config/storage.yml for options).
39 | config.active_storage.service = :local
40 |
41 | # Mount Action Cable outside main process or domain.
42 | # config.action_cable.mount_path = nil
43 | # config.action_cable.url = 'wss://example.com/cable'
44 | # config.action_cable.allowed_request_origins = [ 'http://example.com', /http:\/\/example.*/ ]
45 |
46 | # Force all access to the app over SSL, use Strict-Transport-Security, and use secure cookies.
47 | # config.force_ssl = true
48 |
49 | # Use the lowest log level to ensure availability of diagnostic information
50 | # when problems arise.
51 | config.log_level = :debug
52 |
53 | # Prepend all log lines with the following tags.
54 | config.log_tags = [ :request_id ]
55 |
56 | # Use a different cache store in production.
57 | # config.cache_store = :mem_cache_store
58 |
59 | # Use a real queuing backend for Active Job (and separate queues per environment).
60 | # config.active_job.queue_adapter = :resque
61 | # config.active_job.queue_name_prefix = "react_rails_blog_production"
62 |
63 | config.action_mailer.perform_caching = false
64 |
65 | # Ignore bad email addresses and do not raise email delivery errors.
66 | # Set this to true and configure the email server for immediate delivery to raise delivery errors.
67 | # config.action_mailer.raise_delivery_errors = false
68 |
69 | # Enable locale fallbacks for I18n (makes lookups for any locale fall back to
70 | # the I18n.default_locale when a translation cannot be found).
71 | config.i18n.fallbacks = true
72 |
73 | # Send deprecation notices to registered listeners.
74 | config.active_support.deprecation = :notify
75 |
76 | # Use default logging formatter so that PID and timestamp are not suppressed.
77 | config.log_formatter = ::Logger::Formatter.new
78 |
79 | # Use a different logger for distributed setups.
80 | # require 'syslog/logger'
81 | # config.logger = ActiveSupport::TaggedLogging.new(Syslog::Logger.new 'app-name')
82 |
83 | if ENV["RAILS_LOG_TO_STDOUT"].present?
84 | logger = ActiveSupport::Logger.new(STDOUT)
85 | logger.formatter = config.log_formatter
86 | config.logger = ActiveSupport::TaggedLogging.new(logger)
87 | end
88 |
89 | # Do not dump schema after migrations.
90 | config.active_record.dump_schema_after_migration = false
91 |
92 | # Inserts middleware to perform automatic connection switching.
93 | # The `database_selector` hash is used to pass options to the DatabaseSelector
94 | # middleware. The `delay` is used to determine how long to wait after a write
95 | # to send a subsequent read to the primary.
96 | #
97 | # The `database_resolver` class is used by the middleware to determine which
98 | # database is appropriate to use based on the time delay.
99 | #
100 | # The `database_resolver_context` class is used by the middleware to set
101 | # timestamps for the last write to the primary. The resolver uses the context
102 | # class timestamps to determine how long to wait before reading from the
103 | # replica.
104 | #
105 | # By default Rails will store a last write timestamp in the session. The
106 | # DatabaseSelector middleware is designed as such you can define your own
107 | # strategy for connection switching and pass that into the middleware through
108 | # these configuration options.
109 | # config.active_record.database_selector = { delay: 2.seconds }
110 | # config.active_record.database_resolver = ActiveRecord::Middleware::DatabaseSelector::Resolver
111 | # config.active_record.database_resolver_context = ActiveRecord::Middleware::DatabaseSelector::Resolver::Session
112 | end
113 |
--------------------------------------------------------------------------------
/config/environments/test.rb:
--------------------------------------------------------------------------------
1 | # The test environment is used exclusively to run your application's
2 | # test suite. You never need to work with it otherwise. Remember that
3 | # your test database is "scratch space" for the test suite and is wiped
4 | # and recreated between test runs. Don't rely on the data there!
5 |
6 | Rails.application.configure do
7 | # Settings specified here will take precedence over those in config/application.rb.
8 |
9 | config.cache_classes = false
10 | config.action_view.cache_template_loading = true
11 |
12 | # Do not eager load code on boot. This avoids loading your whole application
13 | # just for the purpose of running a single test. If you are using a tool that
14 | # preloads Rails for running tests, you may have to set it to true.
15 | config.eager_load = false
16 |
17 | # Configure public file server for tests with Cache-Control for performance.
18 | config.public_file_server.enabled = true
19 | config.public_file_server.headers = {
20 | 'Cache-Control' => "public, max-age=#{1.hour.to_i}"
21 | }
22 |
23 | # Show full error reports and disable caching.
24 | config.consider_all_requests_local = true
25 | config.action_controller.perform_caching = false
26 | config.cache_store = :null_store
27 |
28 | # Raise exceptions instead of rendering exception templates.
29 | config.action_dispatch.show_exceptions = false
30 |
31 | # Disable request forgery protection in test environment.
32 | config.action_controller.allow_forgery_protection = false
33 |
34 | # Store uploaded files on the local file system in a temporary directory.
35 | config.active_storage.service = :test
36 |
37 | config.action_mailer.perform_caching = false
38 |
39 | # Tell Action Mailer not to deliver emails to the real world.
40 | # The :test delivery method accumulates sent emails in the
41 | # ActionMailer::Base.deliveries array.
42 | config.action_mailer.delivery_method = :test
43 |
44 | # Print deprecation notices to the stderr.
45 | config.active_support.deprecation = :stderr
46 |
47 | # Raises error for missing translations.
48 | # config.action_view.raise_on_missing_translations = true
49 | end
50 |
--------------------------------------------------------------------------------
/config/initializers/application_controller_renderer.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # ActiveSupport::Reloader.to_prepare do
4 | # ApplicationController.renderer.defaults.merge!(
5 | # http_host: 'example.org',
6 | # https: false
7 | # )
8 | # end
9 |
--------------------------------------------------------------------------------
/config/initializers/assets.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # Version of your assets, change this if you want to expire all your assets.
4 | Rails.application.config.assets.version = '1.0'
5 |
6 | # Add additional assets to the asset load path.
7 | # Rails.application.config.assets.paths << Emoji.images_path
8 | # Add Yarn node_modules folder to the asset load path.
9 | Rails.application.config.assets.paths << Rails.root.join('node_modules')
10 |
11 | # Precompile additional assets.
12 | # application.js, application.css, and all non-JS/CSS in the app/assets
13 | # folder are already added.
14 | # Rails.application.config.assets.precompile += %w( admin.js admin.css )
15 |
--------------------------------------------------------------------------------
/config/initializers/backtrace_silencers.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # You can add backtrace silencers for libraries that you're using but don't wish to see in your backtraces.
4 | # Rails.backtrace_cleaner.add_silencer { |line| line =~ /my_noisy_library/ }
5 |
6 | # You can also remove all the silencers if you're trying to debug a problem that might stem from framework code.
7 | # Rails.backtrace_cleaner.remove_silencers!
8 |
--------------------------------------------------------------------------------
/config/initializers/content_security_policy.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # Define an application-wide content security policy
4 | # For further information see the following documentation
5 | # https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Content-Security-Policy
6 |
7 | # Rails.application.config.content_security_policy do |policy|
8 | # policy.default_src :self, :https
9 | # policy.font_src :self, :https, :data
10 | # policy.img_src :self, :https, :data
11 | # policy.object_src :none
12 | # policy.script_src :self, :https
13 | # policy.style_src :self, :https
14 | # # If you are using webpack-dev-server then specify webpack-dev-server host
15 | # policy.connect_src :self, :https, "http://localhost:3035", "ws://localhost:3035" if Rails.env.development?
16 |
17 | # # Specify URI for violation reports
18 | # # policy.report_uri "/csp-violation-report-endpoint"
19 | # end
20 |
21 | # If you are using UJS then enable automatic nonce generation
22 | # Rails.application.config.content_security_policy_nonce_generator = -> request { SecureRandom.base64(16) }
23 |
24 | # Set the nonce only to specific directives
25 | # Rails.application.config.content_security_policy_nonce_directives = %w(script-src)
26 |
27 | # Report CSP violations to a specified URI
28 | # For further information see the following documentation:
29 | # https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Content-Security-Policy-Report-Only
30 | # Rails.application.config.content_security_policy_report_only = true
31 |
--------------------------------------------------------------------------------
/config/initializers/cookies_serializer.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # Specify a serializer for the signed and encrypted cookie jars.
4 | # Valid options are :json, :marshal, and :hybrid.
5 | Rails.application.config.action_dispatch.cookies_serializer = :json
6 |
--------------------------------------------------------------------------------
/config/initializers/filter_parameter_logging.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # Configure sensitive parameters which will be filtered from the log file.
4 | Rails.application.config.filter_parameters += [:password]
5 |
--------------------------------------------------------------------------------
/config/initializers/inflections.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # Add new inflection rules using the following format. Inflections
4 | # are locale specific, and you may define rules for as many different
5 | # locales as you wish. All of these examples are active by default:
6 | # ActiveSupport::Inflector.inflections(:en) do |inflect|
7 | # inflect.plural /^(ox)$/i, '\1en'
8 | # inflect.singular /^(ox)en/i, '\1'
9 | # inflect.irregular 'person', 'people'
10 | # inflect.uncountable %w( fish sheep )
11 | # end
12 |
13 | # These inflection rules are supported but not enabled by default:
14 | # ActiveSupport::Inflector.inflections(:en) do |inflect|
15 | # inflect.acronym 'RESTful'
16 | # end
17 |
--------------------------------------------------------------------------------
/config/initializers/mime_types.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # Add new mime types for use in respond_to blocks:
4 | # Mime::Type.register "text/richtext", :rtf
5 |
--------------------------------------------------------------------------------
/config/initializers/wrap_parameters.rb:
--------------------------------------------------------------------------------
1 | # Be sure to restart your server when you modify this file.
2 |
3 | # This file contains settings for ActionController::ParamsWrapper which
4 | # is enabled by default.
5 |
6 | # Enable parameter wrapping for JSON. You can disable this by setting :format to an empty array.
7 | ActiveSupport.on_load(:action_controller) do
8 | wrap_parameters format: [:json]
9 | end
10 |
11 | # To enable root element in JSON for ActiveRecord objects.
12 | # ActiveSupport.on_load(:active_record) do
13 | # self.include_root_in_json = true
14 | # end
15 |
--------------------------------------------------------------------------------
/config/locales/en.yml:
--------------------------------------------------------------------------------
1 | # Files in the config/locales directory are used for internationalization
2 | # and are automatically loaded by Rails. If you want to use locales other
3 | # than English, add the necessary files in this directory.
4 | #
5 | # To use the locales, use `I18n.t`:
6 | #
7 | # I18n.t 'hello'
8 | #
9 | # In views, this is aliased to just `t`:
10 | #
11 | # <%= t('hello') %>
12 | #
13 | # To use a different locale, set it with `I18n.locale`:
14 | #
15 | # I18n.locale = :es
16 | #
17 | # This would use the information in config/locales/es.yml.
18 | #
19 | # The following keys must be escaped otherwise they will not be retrieved by
20 | # the default I18n backend:
21 | #
22 | # true, false, on, off, yes, no
23 | #
24 | # Instead, surround them with single quotes.
25 | #
26 | # en:
27 | # 'true': 'foo'
28 | #
29 | # To learn more, please read the Rails Internationalization guide
30 | # available at https://guides.rubyonrails.org/i18n.html.
31 |
32 | en:
33 | hello: "Hello world"
34 |
--------------------------------------------------------------------------------
/config/puma.rb:
--------------------------------------------------------------------------------
1 | # Puma can serve each request in a thread from an internal thread pool.
2 | # The `threads` method setting takes two numbers: a minimum and maximum.
3 | # Any libraries that use thread pools should be configured to match
4 | # the maximum value specified for Puma. Default is set to 5 threads for minimum
5 | # and maximum; this matches the default thread size of Active Record.
6 | #
7 | max_threads_count = ENV.fetch("RAILS_MAX_THREADS") { 5 }
8 | min_threads_count = ENV.fetch("RAILS_MIN_THREADS") { max_threads_count }
9 | threads min_threads_count, max_threads_count
10 |
11 | # Specifies the `port` that Puma will listen on to receive requests; default is 3000.
12 | #
13 | port ENV.fetch("PORT") { 3000 }
14 |
15 | # Specifies the `environment` that Puma will run in.
16 | #
17 | environment ENV.fetch("RAILS_ENV") { "development" }
18 |
19 | # Specifies the `pidfile` that Puma will use.
20 | pidfile ENV.fetch("PIDFILE") { "tmp/pids/server.pid" }
21 |
22 | # Specifies the number of `workers` to boot in clustered mode.
23 | # Workers are forked web server processes. If using threads and workers together
24 | # the concurrency of the application would be max `threads` * `workers`.
25 | # Workers do not work on JRuby or Windows (both of which do not support
26 | # processes).
27 | #
28 | # workers ENV.fetch("WEB_CONCURRENCY") { 2 }
29 |
30 | # Use the `preload_app!` method when specifying a `workers` number.
31 | # This directive tells Puma to first boot the application and load code
32 | # before forking the application. This takes advantage of Copy On Write
33 | # process behavior so workers use less memory.
34 | #
35 | # preload_app!
36 |
37 | # Allow puma to be restarted by `rails restart` command.
38 | plugin :tmp_restart
39 |
--------------------------------------------------------------------------------
/config/routes.rb:
--------------------------------------------------------------------------------
1 | Rails.application.routes.draw do
2 | namespace :api do
3 | namespace :v1 do
4 | get '/blogs', to: 'blogs#index'
5 | get '/blogs/:id', to: 'blogs#show'
6 | get '/blogs/new', to: 'blogs#new'
7 | post 'blogs/create', to: 'blogs#create'
8 | get '/blogs/edit/:id', to: 'blogs#edit'
9 | put '/blogs/update/:id', to: 'blogs#update'
10 | delete '/blogs/destroy/:id', to: 'blogs#destroy'
11 | end
12 | end
13 | root 'home#index'
14 | get '/*path', to: 'home#index'
15 | end
16 |
--------------------------------------------------------------------------------
/config/spring.rb:
--------------------------------------------------------------------------------
1 | Spring.watch(
2 | ".ruby-version",
3 | ".rbenv-vars",
4 | "tmp/restart.txt",
5 | "tmp/caching-dev.txt"
6 | )
7 |
--------------------------------------------------------------------------------
/config/storage.yml:
--------------------------------------------------------------------------------
1 | test:
2 | service: Disk
3 | root: <%= Rails.root.join("tmp/storage") %>
4 |
5 | local:
6 | service: Disk
7 | root: <%= Rails.root.join("storage") %>
8 |
9 | # Use rails credentials:edit to set the AWS secrets (as aws:access_key_id|secret_access_key)
10 | # amazon:
11 | # service: S3
12 | # access_key_id: <%= Rails.application.credentials.dig(:aws, :access_key_id) %>
13 | # secret_access_key: <%= Rails.application.credentials.dig(:aws, :secret_access_key) %>
14 | # region: us-east-1
15 | # bucket: your_own_bucket
16 |
17 | # Remember not to checkin your GCS keyfile to a repository
18 | # google:
19 | # service: GCS
20 | # project: your_project
21 | # credentials: <%= Rails.root.join("path/to/gcs.keyfile") %>
22 | # bucket: your_own_bucket
23 |
24 | # Use rails credentials:edit to set the Azure Storage secret (as azure_storage:storage_access_key)
25 | # microsoft:
26 | # service: AzureStorage
27 | # storage_account_name: your_account_name
28 | # storage_access_key: <%= Rails.application.credentials.dig(:azure_storage, :storage_access_key) %>
29 | # container: your_container_name
30 |
31 | # mirror:
32 | # service: Mirror
33 | # primary: local
34 | # mirrors: [ amazon, google, microsoft ]
35 |
--------------------------------------------------------------------------------
/config/webpack/development.js:
--------------------------------------------------------------------------------
1 | process.env.NODE_ENV = process.env.NODE_ENV || 'development'
2 |
3 | const environment = require('./environment')
4 |
5 | module.exports = environment.toWebpackConfig()
6 |
--------------------------------------------------------------------------------
/config/webpack/environment.js:
--------------------------------------------------------------------------------
1 | const { environment } = require('@rails/webpacker')
2 |
3 | module.exports = environment
4 |
--------------------------------------------------------------------------------
/config/webpack/production.js:
--------------------------------------------------------------------------------
1 | process.env.NODE_ENV = process.env.NODE_ENV || 'production'
2 |
3 | const environment = require('./environment')
4 |
5 | module.exports = environment.toWebpackConfig()
6 |
--------------------------------------------------------------------------------
/config/webpack/test.js:
--------------------------------------------------------------------------------
1 | process.env.NODE_ENV = process.env.NODE_ENV || 'development'
2 |
3 | const environment = require('./environment')
4 |
5 | module.exports = environment.toWebpackConfig()
6 |
--------------------------------------------------------------------------------
/config/webpacker.yml:
--------------------------------------------------------------------------------
1 | # Note: You must restart bin/webpack-dev-server for changes to take effect
2 |
3 | default: &default
4 | source_path: app/javascript
5 | source_entry_path: packs
6 | public_root_path: public
7 | public_output_path: packs
8 | cache_path: tmp/cache/webpacker
9 | check_yarn_integrity: false
10 | webpack_compile_output: true
11 |
12 | # Additional paths webpack should lookup modules
13 | # ['app/assets', 'engine/foo/app/assets']
14 | # resolved_paths: []
15 | resolved_paths: ['app/assets']
16 |
17 | # Reload manifest.json on all requests so we reload latest compiled packs
18 | cache_manifest: false
19 |
20 | # Extract and emit a css file
21 | extract_css: false
22 |
23 | static_assets_extensions:
24 | - .jpg
25 | - .jpeg
26 | - .png
27 | - .gif
28 | - .tiff
29 | - .ico
30 | - .svg
31 | - .eot
32 | - .otf
33 | - .ttf
34 | - .woff
35 | - .woff2
36 |
37 | extensions:
38 | - .jsx
39 | - .mjs
40 | - .js
41 | - .sass
42 | - .scss
43 | - .css
44 | - .module.sass
45 | - .module.scss
46 | - .module.css
47 | - .png
48 | - .svg
49 | - .gif
50 | - .jpeg
51 | - .jpg
52 |
53 | development:
54 | <<: *default
55 | compile: true
56 |
57 | # Verifies that correct packages and versions are installed by inspecting package.json, yarn.lock, and node_modules
58 | check_yarn_integrity: true
59 |
60 | # Reference: https://webpack.js.org/configuration/dev-server/
61 | dev_server:
62 | https: false
63 | host: localhost
64 | port: 3035
65 | public: localhost:3035
66 | hmr: false
67 | # Inline should be set to true if using HMR
68 | inline: true
69 | overlay: true
70 | compress: true
71 | disable_host_check: true
72 | use_local_ip: false
73 | quiet: false
74 | pretty: false
75 | headers:
76 | 'Access-Control-Allow-Origin': '*'
77 | watch_options:
78 | ignored: '**/node_modules/**'
79 |
80 |
81 | test:
82 | <<: *default
83 | compile: true
84 |
85 | # Compile test packs to a separate directory
86 | public_output_path: packs-test
87 |
88 | production:
89 | <<: *default
90 |
91 | # Production depends on precompilation of packs prior to booting for performance.
92 | compile: false
93 |
94 | # Extract and emit a css file
95 | extract_css: true
96 |
97 | # Cache manifest.json for performance
98 | cache_manifest: true
99 |
--------------------------------------------------------------------------------
/db/migrate/20200811034930_create_blogs.rb:
--------------------------------------------------------------------------------
1 | class CreateBlogs < ActiveRecord::Migration[6.0]
2 | def change
3 | create_table :blogs do |t|
4 | t.string :title
5 | t.string :author
6 | t.text :content
7 | t.string :image, default: 'https://bit.ly/2Z4KKcF'
8 |
9 | t.timestamps
10 | end
11 | end
12 | end
13 |
--------------------------------------------------------------------------------
/db/schema.rb:
--------------------------------------------------------------------------------
1 | # This file is auto-generated from the current state of the database. Instead
2 | # of editing this file, please use the migrations feature of Active Record to
3 | # incrementally modify your database, and then regenerate this schema definition.
4 | #
5 | # This file is the source Rails uses to define your schema when running `rails
6 | # db:schema:load`. When creating a new database, `rails db:schema:load` tends to
7 | # be faster and is potentially less error prone than running all of your
8 | # migrations from scratch. Old migrations may fail to apply correctly if those
9 | # migrations use external dependencies or application code.
10 | #
11 | # It's strongly recommended that you check this file into your version control system.
12 |
13 | ActiveRecord::Schema.define(version: 2020_08_11_034930) do
14 |
15 | # These are extensions that must be enabled in order to support this database
16 | enable_extension "plpgsql"
17 |
18 | create_table "blogs", force: :cascade do |t|
19 | t.string "title"
20 | t.string "author"
21 | t.text "content"
22 | t.string "image", default: "https://bit.ly/2Z4KKcF"
23 | t.datetime "created_at", precision: 6, null: false
24 | t.datetime "updated_at", precision: 6, null: false
25 | end
26 |
27 | end
28 |
--------------------------------------------------------------------------------
/db/seeds.rb:
--------------------------------------------------------------------------------
1 | blogs = [
2 | {
3 | title: "Rails ActiveAdmin",
4 | authorName: "Ali",
5 | content: "Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s, when an unknown printer took a galley of type and scrambled it to make a type specimen book. It has survived not only five centuries, but also the leap into electronic typesetting, remaining essentially unchanged. It was popularised in the 1960s with the release of Letraset sheets containing Lorem Ipsum passages, and more recently with desktop publishing software like Aldus PageMaker including versions of Lorem Ipsum."
6 | },
7 | {
8 | title: "Future of ROR",
9 | authorName: "Ahmad",
10 | content: "Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s, when an unknown printer took a galley of type and scrambled it to make a type specimen book. It has survived not only five centuries, but also the leap into electronic typesetting, remaining essentially unchanged. It was popularised in the 1960s with the release of Letraset sheets containing Lorem Ipsum passages, and more recently with desktop publishing software like Aldus PageMaker including versions of Lorem Ipsum."
11 | },
12 | {
13 | title: "ReactJs with Chakra",
14 | authorName: "Taimoor",
15 | content: "Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s, when an unknown printer took a galley of type and scrambled it to make a type specimen book. It has survived not only five centuries, but also the leap into electronic typesetting, remaining essentially unchanged. It was popularised in the 1960s with the release of Letraset sheets containing Lorem Ipsum passages, and more recently with desktop publishing software like Aldus PageMaker including versions of Lorem Ipsum."
16 | },
17 | {
18 | title: "ReactJS with Rails",
19 | authorName: "Hassan",
20 | content: "Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s, when an unknown printer took a galley of type and scrambled it to make a type specimen book. It has survived not only five centuries, but also the leap into electronic typesetting, remaining essentially unchanged. It was popularised in the 1960s with the release of Letraset sheets containing Lorem Ipsum passages, and more recently with desktop publishing software like Aldus PageMaker including versions of Lorem Ipsum."
21 | },
22 | {
23 | title: "React on Rails",
24 | authorName: "Umar",
25 | content: "Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s, when an unknown printer took a galley of type and scrambled it to make a type specimen book. It has survived not only five centuries, but also the leap into electronic typesetting, remaining essentially unchanged. It was popularised in the 1960s with the release of Letraset sheets containing Lorem Ipsum passages, and more recently with desktop publishing software like Aldus PageMaker including versions of Lorem Ipsum."
26 | }
27 | ]
28 |
29 | blogs.each do |b|
30 | Blog.create(title: b[:title], author: b[:authorName], content: b[:content])
31 | end
--------------------------------------------------------------------------------
/lib/assets/.keep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/lib/assets/.keep
--------------------------------------------------------------------------------
/lib/tasks/.keep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/lib/tasks/.keep
--------------------------------------------------------------------------------
/log/.keep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/log/.keep
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "react_rails_blog",
3 | "private": true,
4 | "dependencies": {
5 | "@babel/preset-react": "^7.10.4",
6 | "@rails/actioncable": "^6.0.0",
7 | "@rails/activestorage": "^6.0.0",
8 | "@rails/ujs": "^6.0.0",
9 | "@rails/webpacker": "4.2.2",
10 | "babel-plugin-transform-react-remove-prop-types": "^0.4.24",
11 | "formik": "^2.1.5",
12 | "prop-types": "^15.7.2",
13 | "react": "^16.13.1",
14 | "react-animation-components": "^3.0.0",
15 | "react-dom": "^16.13.1",
16 | "react-dotdotdot": "^1.3.1",
17 | "react-icons": "^3.10.0",
18 | "react-image": "^4.0.3",
19 | "react-router-dom": "^5.2.0",
20 | "react-transition-group": "^4.4.1",
21 | "tailwindcss": "^1.6.2",
22 | "turbolinks": "^5.2.0"
23 | },
24 | "version": "0.1.0",
25 | "devDependencies": {
26 | "webpack-dev-server": "^3.11.0"
27 | }
28 | }
29 |
--------------------------------------------------------------------------------
/postcss.config.js:
--------------------------------------------------------------------------------
1 | let environment = {
2 | plugins: [
3 | require("tailwindcss")("./app/javascript/stylesheets/tailwind.config.js"),
4 | require("autoprefixer"),
5 | require("postcss-import"),
6 | require("postcss-flexbugs-fixes"),
7 | require("postcss-preset-env")({
8 | autoprefixer: {
9 | flexbox: "no-2009"
10 | },
11 | stage: 3
12 | })
13 | ]
14 | };
15 |
16 | module.exports = environment;
17 |
--------------------------------------------------------------------------------
/public/404.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | The page you were looking for doesn't exist (404)
5 |
6 |
55 |
56 |
57 |
58 |
59 |
60 |
61 |
The page you were looking for doesn't exist.
62 |
You may have mistyped the address or the page may have moved.
63 |
64 |
If you are the application owner check the logs for more information.
65 |
66 |
67 |
68 |
--------------------------------------------------------------------------------
/public/422.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | The change you wanted was rejected (422)
5 |
6 |
55 |
56 |
57 |
58 |
59 |
60 |
61 |
The change you wanted was rejected.
62 |
Maybe you tried to change something you didn't have access to.
63 |
64 |
If you are the application owner check the logs for more information.
65 |
66 |
67 |
68 |
--------------------------------------------------------------------------------
/public/500.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | We're sorry, but something went wrong (500)
5 |
6 |
55 |
56 |
57 |
58 |
59 |
60 |
61 |
We're sorry, but something went wrong.
62 |
63 |
If you are the application owner check the logs for more information.
64 |
65 |
66 |
67 |
--------------------------------------------------------------------------------
/public/apple-touch-icon-precomposed.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/public/apple-touch-icon-precomposed.png
--------------------------------------------------------------------------------
/public/apple-touch-icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/public/apple-touch-icon.png
--------------------------------------------------------------------------------
/public/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/public/favicon.ico
--------------------------------------------------------------------------------
/public/robots.txt:
--------------------------------------------------------------------------------
1 | # See https://www.robotstxt.org/robotstxt.html for documentation on how to use the robots.txt file
2 |
--------------------------------------------------------------------------------
/storage/.keep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/storage/.keep
--------------------------------------------------------------------------------
/tmp/.keep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/tmp/.keep
--------------------------------------------------------------------------------
/tmp/pids/.keep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/tmp/pids/.keep
--------------------------------------------------------------------------------
/vendor/.keep:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/MA-Ahmad/react_rails_blog/75a9b99956c80e45fd9d454e9dfaf9b7d508d232/vendor/.keep
--------------------------------------------------------------------------------