Wunderschön means very gorgeous and wonderful in German. This project is a kind of online shopping website which is filled with sketches of rabbits and its products.
Wunderschön is built with Ruby version 2.4.1 and Rails 5.1.
- Ruby Version 4.2.1
- Rails Version 5.1.1
- NodeJS Version >= 6.4.0
Clone It!
$ git clone git@github.com:Maxwell-Alexius/Wunderschon.git
Go into the project folder:
$ cd Wunderschon
Bundle it:
$ bundle install
Create and migrate database:
$ bundle exec rails db:create
$ bundle exec rails db:migrate
If yarn
haven't installed in your OS, see installation of yarn. After yarn installed, run this command to install JS packages via yarn:
$ yarn init
You can choose to open Rails server and open webpacker
(JS develop server) separately, that requires you to open additional two terminal and run two commands separately:
# Rails server
$ bundle exec rails server
# Webpacker JS develop server
$ ./bin/webpack-dev-server
Or you can integrate both server into a terminal via one command:
$ foreman start
It will automatically start both Rails server and JS webpacker develop server in the same terminal.
Excluded the Rails default gems, the info of the included Ruby gems are listed below (in an alphabetical order):
aasm
- Perfect simulation of state machineawesome_print
- Pretty print for Ruby objects with style in consolebootstrap-sass
- Sass support for Bootstrap 2 and 3factory_girl_rails
- Factory for producing records with different trait in a fast, easy and efficient wayfaker
- Generating fake datakaminari
- An awesome pagination toolparanois
- Prevention of hard destroy. In other words, rather than deleting the data, it just hide it.rails-controller-spec
- For Rails controller tests with RSpecrspec-rails
- Brilliant testing framework for Railsrubocop
- Static Ruby code analyzer and syntax checker based on Ruby style guidesettingslogic
- Simple, straightforward settings solutionsimple_form
- Namely, simple form parsershoulda-matchers
- Testing matcherswebpacker
- An integration of JS webpack development with Rails
Using Rails 5.1+ provided new feature: Loving Javascript integrate with Yarn JS dependencies management. You can think of Yarn as the JS version of the bundler.
- Add gem
webpacker
to theGemfile
and executebundle install
gem 'webpacker'
$ bundle install
- Remember to check and update your
NodeJS
version! (>= 6.4.0)
$ node -v
- After install the gem,
webpacker
provide several rake tasks. To check it, runbin/rails webpacker
you should get the following output:
$ bin/rails webpacker
Available webpacker tasks are:
webpacker:install Installs and setup webpack with yarn
webpacker:compile Compiles webpack bundles based on environment
webpacker:check_node Verifies if Node.js is installed
webpacker:check_yarn Verifies if yarn is installed
webpacker:check_webpack_binstubs Verifies that bin/webpack & bin/webpack-dev-server are present
webpacker:verify_install Verifies if webpacker is installed
webpacker:yarn_install Support for older Rails versions. Install all JavaScript dependencies as specified via Yarn
webpacker:install:react Installs and setup example React component
webpacker:install:vue Installs and setup example Vue component
webpacker:install:angular Installs and setup example Angular component
webpacker:install:elm Installs and setup example Elm component
- Choose an installation command, e.g. if you want to install with
React
, run
$ bundle exec rake webpacker:install:react
-
After
webpacker
installed, you can put your ES6 JS files in theapp/javascript
directory. This is where thewebpacker
compiles your JS file. -
Run the command below to compile the JS files. The compiled files will be placed in
public/packs
directory.
$ bundle exec webpacker:compile
webpacker
gem provide helper methodjavascript_pack_tag
for loading JS files in your views. For instance, to load the compiledapplication.js
inapplication.html.erb
, in your view:
<html>
<head>
<!-- ... Other Stylesheet or Meta Tags ... -->
<%= javascript_include_tag 'application', 'data-turbolinks-track': 'reload' %>
<!-- Load compiled application.js -->
<%= javascript_pack_tag 'application' %>
</head>
...
foreman
can help us manage to execute rails server with webpacker
watch mode at the same time. Install the gem first, but not to include in your project Gemfile!
$ gem install foreman
Create the Procfile
in the app root directory and add the content:
web: bundle exec rails server
webpacker: ./bin/webpack-dev-server
Running the command will start the rails server and start the webpacker
watch mode:
$ foreman start
For more information, see foreman
gem GitHub
It is very easy to use Yarn, similar to node package manager, initialize Yarn first:
$ yarn init
Use the command to install any node package:
$ yarn add [package-name]
For example, adding jquery
via yarn is simple:
$ yarn add jquery
Webpack prefers the original, unmodified source code of a library, rather than the packaged “dist” code. In Rails config/webpack/shared.js
, add the alias
key in the resolve
object with an object which aliases the "jquery" with the directory of the jQuery file:
// ... Omitted
module.exports = {
// ... Omitted
resolve: {
alias: {
jquery: 'jquery/src/jquery'
},
// ... Omitted
}
}
In order to make jQuery available to other modules, we can use the webpack.ProvidePlugin
makes a module available as a variable in every other module required by webpack. Therefore, in the same file, add the statement in the plugins
part of the export module object:
// ... Omitted
module.exports = {
// ... Omitted
plugins: [
// Omitted
new webpack.ProvidePlugin({
$: 'jquery',
jQuery: 'jquery',
jquery: 'jquery'
})
],
// ... Omitted
}
Lastly, we can then require jQuery in the pack. Import jQuery into app/javascript/packs/application.js
and make it available globally using the window
object:
import jQuery from 'jquery'
window.jQuery = jQuery
For more information, you can read:
webpacker
Gem Github- Embracing Change: Rails 5.1 Adopts Yarn, Webpack, and the JS Ecosystem
- Introducing Webpacker
jQuery
- Common but powerful JS library for fundamental DOM manipulations, events, traversal, animation and simple AJAXReact.js
- Component based dynamic, reactive and declarative UI interface designaxios
- Promise based HTTP client for browser and NodeJSRx.js
- Powerful reactive programming with observable patterns implemented in JavaScriptPIXI.js
- Super fast HTML 5 2D rendering engine that uses webGL with canvas fallback
Make sure you have webpacker already installed, if not, run:
$ bin/rails webpacker:install
Then just run the webpacker built-in command and it'll serve you well:
$ bin/rails webpacker:install:react
It will automatically configure the package.json
, .babelrc
and install dependencies including the babel-preset-react
which enables you to use JSX
syntax during development.
[Deprecated]
Additionally, you should also install prop-types
package using Yarn:
$ yarn add prop-types
Because React.PropTypes
is deprecated as of React version 15.5, so if you want to use type checking
(and it is highly recommended), you should also install the prop-types
library instead.
[Update]
When install react
with webpacker
, it also helps you install the prop-types
library which you don't need to install it by yourself.
Note: Please make sure you have setup webpacker
with react
.
Add react-rails
into your Gemfile
and run bundle install
. After installation, run:
$ rails generate react:install
Which generates:
components/
directory for your React componentsReactRailsUJS
setup inpacks/application.js
- packs/server_rendering.js for server-side rendering
Please visit react-rails
repository for more information. This README only covers the step to setup ReactJS environment.
In the application/javascript/packs/application.js
, you can see that react-rails
automatically add several lines which enables you to controller React components rendering:
// Support component names relative to this directory:
var componentRequireContext = require.context('/components', true)
var ReactRailsUJS = require('react_ujs')
ReactRailsUJS.useContext(componentRequireContext)
Add the statement ReactRailsUJS.mountComponents
in the document ready event, which enables to use the simple React component helper:
$(document).ready(function() {
ReactRailsUJS.mountComponents()
})
So when we want to render the component file, say hello.jsx
, we can use the react_component
helper to render it:
<%= react_component 'hello' %>
We can also pass in default props
with a Ruby hash which is very helpful:
<%= react_component 'hello', greetings: 'hello', name: 'Maxwell-Alexius' %>
After installation, you can manage and develop React component in app/javascript/packs/components/
directory. To create a component, for example, the hello-world
program, create a new file called hello-world.jsx
and below is the example code:
import React from 'react'
import ReactDOM from 'react-dom'
import PropTypes from 'prop-types'
/* Declare Your Component */
class HelloWorld extends React.Component {
constructor(props) {
super(props)
/* You can initialize your states here */
// this.states = { ... }
}
static defaultProps = {
appName: 'Hello World!'
}; // <----- Do not forget the semicolon!
/* You can declare life-cycle methods in this component declaration block */
// componentWillMount() { ... }
// componentDidMount() { ... }
// shouldComponentReceiveProps() { ... }
// ... other life-cycle methods
render() {
return (
<div className="hello-world">
{this.props.appName}
</div>
)
}
}
/* Checking Props' Types */
HelloWorld.propTypes = {
appName: PropTypes.string.isRequired
}
/* Render Your Component */
// You should have <div id="app"></div> in your HTML file
ReactDOM.render(
<HelloWorld />,
document.getElementById('app')
)
For example, to nest another component in our HelloWorld
component, let's create another directory which is called components/hello_world/
and include another component file inner-component.jsx
. It should use the export default
to export our component as a module:
import React from 'react'
import PropTypes from 'prop-types'
class InnerComponent extends React.Component {
constructor(props) {
super(props)
}
render() {
return (
<div className="inner-component">
Hello, This is the InnerComponent!
</div>
)
}
}
export default InnerComponent
And then include the module to our root component file. Using JSX syntax to use the component in the render
method:
import React from 'react'
import ReactDOM from 'react-dom'
import PropTypes from 'prop-types'
import InnerComponent from './hello_world/inner-component'
class HelloWorld extends React.Component {
/* ... Omitted ... */
render() {
return (
<div className="hello-world">
<InnerComponent />
</div>
)
}
}
(Drafting...)
(Drafting...)
(Drafting...)
(Drafting...)
(Drafting...)
Using React.js
to construct the shopping cart component UI interface. Its structure is presented below:
- shopping-cart (Root Component)
- message
- icon
- wrapper
- header
- body
- caption
- *item
- quantity-field
- footer
Hint: * means list-like components
(Drafting...)
(Drafting...)
(Drafting...)
(Drafting...)
(Drafting...)