React.js同圢/ナニバヌサルアプリケヌションをれロから構築したす。 パヌト1スタックをたずめる

画像

圌女がこの蚘事を読んだずきの劻の顔


私は6ヶ月前にどこかで芋぀けたいず思う䞀連の蚘事を曞くこずにしたした。 䞻にReact.jsでクヌルなアプリケヌションの開発を開始したいが、最近の完党なフロント゚ンド開発のために知っおおく必芁のあるさたざたなテクノロゞヌやツヌルの動物園ぞのアプロヌチ方法がわからない人に興味がありたす。


おそらく最初から、おそらく最も人気のあるシナリオを実装したいず思いたす。RESTAPIを提䟛するサヌバヌ郚分がありたす 。 䞀郚のメ゜ッドでは、Webアプリケヌションのナヌザヌがログむンする必芁がありたす。


目次


1同型アプリケヌションの基本スタックを構築する
2 ルヌティングずブヌトストラップを䜿甚した簡単なアプリケヌションを䜜成したす
3 APIず認蚌ずの盞互䜜甚を実装したす


囜際化、テストの䜜成、展開、 CSSオプションなどの問題は括匧の倖に残りたす。これらの質問ははるかに倚様で、それぞれが個別の蚘事のブロックに描画されるためです。 おそらく、需芁があれば、埌でそれらに戻るでしょう。


泚 githubには、アプリケヌションの基盀ずしお䜿甚できる倚くのすばらしいボむラヌプレヌトたたは既にビルドされたスタックがありたすが、各パッケヌゞずコヌドの各行が䜕をするのかを理解しお、独自のスタックを組み立おる方がはるかに正しいようです。 これを1回行うこずを孊習するず、次回はスタックを収集するのに5分以䞊かかるこずはほずんどありたせん。


泚2読者によっお準備のレベルが異なるず想定しおいるため、カッタヌの䞋にツヌルの長い説明を隠しお、蚘事が無限に長く芋えないようにしたす。


さあ、行こう


1.同圢Webアプリケヌションを開発したす。


同圢たたは汎甚アプリケヌションずは、アプリケヌションのJavaScriptコヌドをサヌバヌずクラむアントの䞡方で実行できるこずを意味したす。 このメカニズムはReactの匷みの1぀であり、ナヌザヌがコンテンツにはるかに高速にアクセスできるようにしたす。 以䞋では「同圢」ずいう甚語を䜿甚したすが、これはさらに䞀般的ですが、「同圢」ず「普遍的」は同䞀であるず理解するこずが重芁です。


同型アプリケヌションの詳现をご芧ください。
画像

ナヌザヌの芳点から芋るず、Webアプリケヌションずの盞互䜜甚は次のずおりです。


1ブラりザがWebアプリケヌションにリク゚ストを送信したす。
2 Node.jsのサヌバヌ偎はJavaScriptを実行したす 。 必芁に応じお、プロセスはAPI芁求も実行したす 。 その結果、完成したHTMLペヌゞがクラむアントに送信されたす。
3ナヌザヌはほずんど瞬時にペヌゞコンテンツを受け取りたす。 この時点で、クラむアント偎のJavaScriptがバックグラりンドでダりンロヌドおよび初期化され、アプリケヌションが起動したす。 最も重芁なこずは、ナヌザヌは埓来のクラむアントサむド JavaScript アプリケヌションの堎合のように、2秒以䞊たっおからではなく、ほが即座にコンテンツにアクセスできるこずです。
4a JavaScript がクラむアントでロヌドに倱敗したか゚ラヌで実行された堎合、リンクをクリックするず、サヌバヌぞの通垞のリク゚ストが実行され、プロセスの最初のステップに戻りたす。
4bすべおが正垞である堎合、リンクはアプリケヌションによっおむンタヌセプトされたす。 必芁に応じお、 API リク゚ストが実行され 、クラむアント偎の JavaScript *がリク゚ストされたペヌゞを生成しおレンダリングしたす。 このアプロヌチにより、トラフィックが枛少し、アプリケヌションの生産性が向䞊したす。


なぜそれがクヌルなのですか
1ナヌザヌは2秒以䞊コンテンツをより速く受信したす。 これは、モバむルむンタヌネットがあたりない堎合、たたは条件付き䞭囜にいる堎合に特に圓おはたりたす。 利益は、クラむアントのJavaScriptがダりンロヌドされるのを埅぀必芁がないずいう事実によるものですが、これは200kb以䞊であり、瞮小ず圧瞮を考慮しおいたす。 たた、 JavaScriptの初期化には時間がかかる堎合がありたす。 初期化埌にクラむアントAPIリク゚ストを行う必芁性を远加し、モバむルむンタヌネット䞊で非垞に顕著な遅延が発生するこずが倚いこずを思い出すず、同圢アプロヌチがアプリケヌションをナヌザヌにずっおより快適にするこずが明らかになりたす。
2゚ラヌのためにクラむアントのJavaScriptアプリケヌションが動䜜を停止した堎合、サむトはナヌザヌにずっお圹に立たなくなる可胜性が高くなりたす。 同型の堎合、ナヌザヌが望んでいるこずを実行できる可胜性は十分にありたす。


実装に関しお


server.jsずclient.jsの 2぀の゚ントリポむントがありたす 。
Server.jsはノヌドサヌバヌによっお䜿甚されたす。 その䞭で、 ゚クスプレスたたは別のWebサヌバヌを実行し、リク゚スト凊理ず他のサヌバヌ固有のビゞネスロゞックをその䞭に配眮したす。
Client.js-ブラりザヌの゚ントリポむント。 ここに顧客固有のビゞネスロゞックを配眮したす。
Reactアプリケヌションは、クラむアントずサヌバヌの䞡方で共有されたす。 これは、アプリケヌション党䜓の゜ヌスコヌドの90〜95以䞊を占めおいたす。これは、同型/普遍的なアプロヌチの本質です。 実装プロセスでは、これが実際にどのように機胜するかを確認したす。


新しいプロゞェクトを䜜成する


Node.jsずnpmパッケヌゞマネヌゞャヌをむンストヌルする

䞀芋、ノヌドのバヌゞョン付きバヌゞョンは少し奇劙に芋えるかもしれたせん。 混乱しないように、v4.xがLTSブランチであり、v5.xが実隓的ブランチであり、v6.xが2016幎10月1日以降の将来のLTSであるこずを知るだけで十分です。 最新のLTSバヌゞョンをむンストヌルするこずをお勧めしたす。぀たり、蚘事の公開日は4回目です。これにより、プラットフォヌム自䜓のバグずの非垞に䞍快な衝突を防ぐこずができたす。 私たちの目的では、2぀の間に特別な違いはありたせん。


リンクhttps://nodejs.org/en/download/に埓うこずにより、 node.jsおよびプラットフォヌム甚のnpmパッケヌゞマネヌゞャヌをダりンロヌドしおむンストヌルできたす。


mkdir habr-app && cd habr-app npm init 

すべおのnpmの質問に぀いお、 Enterボタンを安党に抌しおデフォルト倀を遞択できたす。 その結果、 package.jsonファむルがプロゞェクトのルヌトディレクトリに衚瀺されたす。


2.開発プロセス


近幎のJavaScriptは飛躍的に発展しおおり、ナヌザヌに぀いおは蚀えたせん。 残念ながら、「クヌルな技術的機胜」のために芖聎者の倧郚分を犠牲にする顧客はほずんどいないため、ブラりザの最新バヌゞョンのみをサポヌトするこずはできたせん。 幞いなこずに、プログラマヌが最新の蚀語構成を䜿甚しお自分に合った方法で蚘述できるツヌルが発明されたため、非垞に叀いブラりザヌでも正しく動䜜するコヌドを取埗できたす。


バベル


Babelは、 CoffeeScript 、 TypeScript、およびその他の蚀語アドオンを含むJavaScriptの方蚀をJavaScript ES5に倉換するコンパむラヌです。これは、 babel-polyfillを远加するず、 IE8を含むほずんどすべおのブラりザヌでサポヌトされたす。 Babelの匷みは、プラグむンによるモゞュヌル性ず拡匵性です。 たずえば、叀いブラりザでは動䜜しないこずを心配するこずなく、最新のJavaScriptチップを䜿甚できるようになりたした。


反応のコンポヌネントを倉換するには、 babel-preset-reactプリセットを䜿甚したす。 JavaScriptデコレヌタが倧奜きなので、 babel-plugin-transform-decorators-legacyパッケヌゞも必芁になりたす。 コヌドを叀いブラりザヌで正垞に動䜜させるには、 babel-polyfillパッケヌゞをむンストヌルしたす 。ES6 / ES7方蚀で曞き蟌むには、それぞれbabel- preset-es2015ずbabel- preset -stage-0が必芁です。


 npm i --save babel-core babel-plugin-transform-decorators-legacy babel-polyfill babel-preset-es2015 babel-preset-react babel-preset-stage-0 

アプリケヌションのサヌバヌ偎にもbabelが必芁なため、これらの䟝存関係はプロゞェクトの䟝存関係ずしおむンストヌルする必芁がありたす。


クラむアント偎のJavaScriptを構築するために必芁なパッケヌゞをむンストヌルしたす。 本番環境では必芁ないため、開発の䟝存関係ずしおむンストヌルしたす。


.babelrc


babelを起動するず、プロゞェクトのルヌトにある.babelrcファむルにアクセスしたす。 このファむルには、䜿甚されおいるプリセットずプラグむンの構成ずリストが保存されおいたす。


このファむルを䜜成する


 { "presets": [ "es2015", "react", "stage-0" ], "plugins": [ "transform-decorators-legacy" ] } 

3.組み立お


Babelに必芁なプラグむンをむンストヌルしたしたが、だれがい぀それを起動する必芁がありたすか この問題に進む時が来たした。


泚最近のWebアプリケヌションプロゞェクトは、デスクトップたたはモバむルアプリケヌションプロゞェクトず倧差ありたせん。倖郚ラむブラリ、 MVCパラダむムに最も䞀臎する可胜性が高いファむル、リ゜ヌス、スタむルファむルなどが含たれたす。 このようなプレれンテヌションはプログラマにずっお非垞に䟿利ですが、ナヌザヌにずっおは䟿利ではありたせん。 JavaScriptプロゞェクトのすべおの゜ヌスコヌドず䜿甚するラむブラリを取埗し、䞍芁なものをすべお砎棄し、1぀の倧きなファむルに結合しお瞮小化を適甚するず、出力ファむルは元のセットよりも10倍以䞊少なくなりたす。 たた、アプリケヌションのすべおのロゞックをダりンロヌドするために必芁なブラりザリク゚ストは数癟ではなく、1぀だけです。 どちらもパフォヌマンスにずっお非垞に重芁です。 ずころで、さたざたな方蚀 LESS 、 SASSなどを含むCSSリ゜ヌスに同じロゞックが適甚されたす。


この䟿利な䜜業はwebpackによっお行われたす 。


泚同じ目的で、コレクタヌを䜿甚できたすgrunt、gulp、bower、browserifyなど。ただし、Reactの歎史的には、webpackが最も頻繁に䜿甚されたした。


webpackおよびwebpack-dev-serverの詳现

りェブパック


画像

Webpack操䜜アルゎリズム


webpackをパむプラむンず考える最も簡単な方法。 Webpackは、提䟛された゚ントリポむントを取埗し、途䞭で遭遇するすべおの䟝存関係を順番に凊理したす。 JavaScriptたたはその方蚀で蚘述されたすべおのコヌドは、 バベルを通過しお1぀の倧きなJavaScript ES5ファむルにブラむンドされたす。 ここでは、これがどのように機胜するかに぀いお詳しく説明する䟡倀がありたす。 コヌドずnode_modules webpackで䜿甚されるコヌドのそれぞれが必芁たたはむンポヌトされ、最終アセンブリの独自の小さなモゞュヌルに割り圓おられたす。 自分のコヌドたたは䜿甚するラむブラリのコヌドが同じ関数に䟝存する堎合、 Webpackモゞュヌルずしお䞀床だけ最終アセンブリに到達し、それに䟝存するすべおのコヌドは同じものを参照したす最終アセンブリのモゞュヌル。 webpackビルドプロセスのもう1぀のクヌルな機胜は、 lodashのような巚倧なラむブラリを䜿甚しおいるが、次のような特定の関数のみが必芁であるず明瀺的に述べおいるこずです。


 import assign from 'lodash/assign'; 

ラむブラリ党䜓ではなく、ラむブラリの䜿甚枈み郚分のみが最終アセンブリに入りたす。これにより、アセンブルされたファむルのサむズが倧幅に削枛されたす。


泚これは、䜿甚するラむブラリがモゞュヌル性をサポヌトしおいる堎合にのみ機胜したす。 このため、䜜者は自分のプロゞェクトでラむブラリMoment.js、XRegExpなどを䜿甚するこずを拒吊したした。


webpack構成のプロゞェクトのさたざたなタむプのファむルに぀いお、それを凊理するロヌダヌたたはロヌダヌのチェヌンを定矩したす。


webpack-dev-server


毎回、プロゞェクト党䜓を再組み立おするのは非垞に費甚がかかる可胜性がありたす。䞭芏暡のプロゞェクトの堎合、アセンブリは30秒以䞊に簡単に達するこずができたす。 この問題を解決するには、開発䞭にwebpack-dev-serverを䜿甚するず非垞に䟿利です。 これはサヌドパヌティのサヌバヌアプリケヌションで、起動するずリ゜ヌスの完党なアセンブリを䜜成し、アクセスするずRAMからリ゜ヌスの最新バヌゞョンを提䟛したす。 開発䞭に個々のwebpack-dev-serverファむルを倉曎するず、その堎で倉曎されたファむルのみが再コンパむルされ、最終アセンブリの叀いモゞュヌルが新しいモゞュヌルに眮き換えられたす。 再構築が必芁なのはプロゞェクト党䜓ではなく、1぀のファむルのみであるため、1秒以䞊かかるこずはめったにありたせん。


もちろん 、 本番環境では構築しないため、 Webpackずwebpack-dev-serverを開発の䟝存関係ずしおむンストヌルしたす 。


 npm i --save-dev webpack@1.13.2 webpack-dev-server 

泚 この蚘事の執筆時および公開時には、webpack 1のバヌゞョンが関連しおいたしたが、2016幎9月22日以降、webpack 2ベヌタがデフォルトでむンストヌルされたす。


さお、アセンブリの構成ファむルを䜜成する必芁がありたす。 プロゞェクトのルヌトにファむルを䜜成したす


webpack.config.js


 global.Promise = require('bluebird'); var webpack = require('webpack'); var path = require('path'); var ExtractTextPlugin = require('extract-text-webpack-plugin'); var CleanWebpackPlugin = require('clean-webpack-plugin'); var publicPath = 'http://localhost:8050/public/assets'; var cssName = process.env.NODE_ENV === 'production' ? 'styles-[hash].css' : 'styles.css'; var jsName = process.env.NODE_ENV === 'production' ? 'bundle-[hash].js' : 'bundle.js'; var plugins = [ new webpack.DefinePlugin({ 'process.env': { BROWSER: JSON.stringify(true), NODE_ENV: JSON.stringify(process.env.NODE_ENV || 'development') } }), new ExtractTextPlugin(cssName) ]; if (process.env.NODE_ENV === 'production') { plugins.push( new CleanWebpackPlugin([ 'public/assets/' ], { root: __dirname, verbose: true, dry: false }) ); plugins.push(new webpack.optimize.DedupePlugin()); plugins.push(new webpack.optimize.OccurenceOrderPlugin()); } module.exports = { entry: ['babel-polyfill', './src/client.js'], debug: process.env.NODE_ENV !== 'production', resolve: { root: path.join(__dirname, 'src'), modulesDirectories: ['node_modules'], extensions: ['', '.js', '.jsx'] }, plugins, output: { path: `${__dirname}/public/assets/`, filename: jsName, publicPath }, module: { loaders: [ { test: /\.css$/, loader: ExtractTextPlugin.extract('style-loader', 'css-loader!postcss-loader') }, { test: /\.less$/, loader: ExtractTextPlugin.extract('style-loader', 'css-loader!postcss-loader!less-loader') }, { test: /\.gif$/, loader: 'url-loader?limit=10000&mimetype=image/gif' }, { test: /\.jpg$/, loader: 'url-loader?limit=10000&mimetype=image/jpg' }, { test: /\.png$/, loader: 'url-loader?limit=10000&mimetype=image/png' }, { test: /\.svg/, loader: 'url-loader?limit=26000&mimetype=image/svg+xml' }, { test: /\.(woff|woff2|ttf|eot)/, loader: 'url-loader?limit=1' }, { test: /\.jsx?$/, loader: 'babel', exclude: [/node_modules/, /public/] }, { test: /\.json$/, loader: 'json-loader' }, ] }, devtool: process.env.NODE_ENV !== 'production' ? 'source-map' : null, devServer: { headers: { 'Access-Control-Allow-Origin': '*' } } }; 

泚これは生産的なプロゞェクトの蚭定䟋であるため、芋た目よりも少し耇雑に芋えたす。


構成の説明

だから
1 bluebirdプロゞェクトのpromiseの実装を䜿甚するこずを発衚したす。 事実䞊の暙準。
2本番環境では、リ゜ヌスキャッシングを効果的に管理するために、各ファむルにアセンブリのハッシュを持たせる必芁がありたす。 収集されたリ゜ヌスの叀いバヌゞョンが気にならないように、次のビルドの前に察応するディレクトリをクリアするclean-webpack-pluginを䜿甚したす。
3ビルドプロセス䞭にextract-text-webpack-pluginは、すべおのcss / less / sass /任意の䟝存関係を探し、最埌にそれらを単䞀のCSSファむルに描画したす。
4 DefinePluginを䜿甚しお、グロヌバルアセンブリ倉数、 DedupePluginおよびOccurenceOrderPlugin-最適化プラグむンを蚭定したす。 これらのプラグむンの詳现に぀いおは、ドキュメントをご芧ください。
5 babel-polyfillずclient.jsを゚ントリポむントずしお指定したす。 1぀目はJavaScriptコヌドを叀いブラりザヌで実行できるようにし、2぀目はクラむアントWebアプリケヌションの゚ントリポむントであり、埌で蚘述したす。
6 解決ずは、アプリケヌションのコヌドを蚘述するずき


 import SomeClass from './SomeClass'; 

webpackは、指定されたファむルが芋぀からないず報告する前に、 SomeClass.jsたたはSomeClass.jsxファむルでSomeClassを探したす。
7次に、プラグむンのリストを枡し、 出力 webpackがアセンブリ埌にファむルを配眮するディレクトリを指定したす。
8最も興味深いのは、 ロヌダヌのリストです。 ここで、䞊蚘のコンベダを定矩したす。 適切な拡匵子を持぀ファむルに適甚されたす。 この問題は別の蚘事にたずめられおいるため、ドキュメント内のロヌダヌずそのパラメヌタヌを決定する圢匏をよく理解するこずをお勧めしたす。 完党に根拠がないように、ロヌダヌの蚭蚈に焊点を圓おたす。


 test: /\.less$/, loader: ExtractTextPlugin.extract('style-loader', 'css-loader!postcss-loader!less-loader') 

ここで、拡匵子の小さいファむルが芋぀かった堎合、 css-loaderPostcss-loaderLess-loaderチェヌンを䜿甚するExtractTextPluginに枡す必芁がありたす。 右から巊に読みたす。぀たり、最初のless-loaderは .lessファむルを凊理し、結果はpostcss-loaderになり、凊理されたcss-loaderのコンテンツを枡したす。
9webpackドキュメントのdevtoolに぀いお読むこずもお勧めしたす。
10最埌に、パラメヌタヌwebpack-dev-serverを远加で指定したす 。 この堎合、 webpack-dev-serverずアプリケヌションが異なるポヌトで動䜜するため、Access-Control-Allow-Originを指定するこずが重芁です。これは、 CORS問題を解決する必芁があるこずを意味したす。


構成で参照するラむブラリはむンストヌルされたせん。 プロゞェクトによっおは、 bluebirdのみが䜿甚できたす。他のすべおはアセンブリ専甚です。


 npm i --save bluebird npm i --save-dev babel-loader clean-webpack-plugin css-loader extract-text-webpack-plugin file-loader html-loader json-loader less less-loader postcss-loader style-loader url-loader 

たた、いく぀かの新しいpackage.jsonスクリプトを远加したす。ビルドおよびwebpack devサヌバヌを実行したす。


  "scripts": { "build": "NODE_ENV='production' webpack -p", "webpack-devserver": "webpack-dev-server --debug --hot --devtool eval-source-map --output-pathinfo --watch --colors --inline --content-base public --port 8050 --host 0.0.0.0" } 

曎新PMのwrewolfずNeropのナヌザヌはそれぞれ、Windowsスクリプトでは異なるように芋えるべきであるず報告したした。


  "scripts": { "build": "set NODE_ENV='production' && webpack -p", "webpack-devserver": "webpack-dev-server --debug --hot --devtool eval-source-map --output-pathinfo --watch --colors --inline --content-base public --port 8050 --host 0.0.0.0" } 

プロゞェクトルヌトにsrcフォルダヌを䜜成し、その䞭に空のclient.jsファむルを䜜成したす。


スクリプトをテストしたす。コン゜ヌルでnpm run buildず入力し、他のコン゜ヌルりィンドりでnpm run webpack-devserverず入力したす。 間違いがなければ、次に進みたす。


4. ESLint


これはオプションの項目ですが、非垞に䟿利です。 ESLintは、゜ヌスコヌドに提瀺されるルヌルのコレクションです。 コヌドの䜜成プロセス䞭にプログラマがこれらのルヌルの1぀以䞊に違反するず、 webpackのビルド䞭に゚ラヌが発生したす。 したがっお、すべおのWebアプリケヌションコヌドは、倉数の呜名、むンデント、特定の構成芁玠の䜿甚の犁止などを含む単䞀のスタむルで蚘述されたす。


プロゞェクトルヌトの.eslintrcファむルにルヌルのリストを配眮したす。 ESLintおよびルヌルの詳现に぀いおは、プロゞェクトのWebサむトをご芧ください。


.eslintrc
 { "parser": "babel-eslint", "plugins": [ "react" ], "env": { "browser": true, "node": true, "mocha": true, "es6": true }, "ecmaFeatures": { "arrowFunctions": true, "blockBindings": true, "classes": true, "defaultParams": true, "destructuring": true, "forOf": true, "generators": false, "modules": true, "objectLiteralComputedProperties": true, "objectLiteralDuplicateProperties": false, "objectLiteralShorthandMethods": true, "objectLiteralShorthandProperties": true, "restParams": true, "spread": true, "superInFunctions": true, "templateStrings": true, "jsx": true }, "rules":{ // Possible errors "comma-dangle": [2, "never"], "no-cond-assign": [2, "always"], "no-constant-condition": 2, "no-control-regex": 2, "no-dupe-args": 2, "no-dupe-keys": 2, "no-duplicate-case": 2, "no-empty-character-class": 2, "no-empty": 2, "no-extra-boolean-cast": 0, "no-extra-parens": [2, "functions"], "no-extra-semi": 2, "no-func-assign": 2, "no-inner-declarations": 2, "no-invalid-regexp": 2, "no-irregular-whitespace": 2, "no-negated-in-lhs": 2, "no-obj-calls": 2, "no-regex-spaces": 2, "no-sparse-arrays": 2, "no-unreachable": 2, "use-isnan": 2, "valid-typeof": 2, "no-unexpected-multiline": 0, // Best Practices "block-scoped-var": 2, "complexity": [2, 40], "curly": [2, "multi-line"], "default-case": 2, "dot-notation": [2, { "allowKeywords": true }], "eqeqeq": 2, "guard-for-in": 2, "no-alert": 1, "no-caller": 2, "no-case-declarations": 2, "no-div-regex": 0, "no-else-return": 2, "no-eq-null": 2, "no-eval": 2, "no-extend-native": 2, "no-extra-bind": 2, "no-fallthrough": 2, "no-floating-decimal": 2, "no-implied-eval": 2, "no-iterator": 2, "no-labels": 2, "no-lone-blocks": 2, "no-loop-func": 2, "no-multi-str": 2, "no-native-reassign": 2, "no-new": 2, "no-new-func": 2, "no-new-wrappers": 2, "no-octal": 2, "no-octal-escape": 2, "no-param-reassign": [2, { "props": true }], "no-proto": 2, "no-redeclare": 2, "no-script-url": 2, "no-self-compare": 2, "no-sequences": 2, "no-unused-expressions": 2, "no-useless-call": 2, "no-with": 2, "radix": 2, "wrap-iife": [2, "outside"], "yoda": 2, // ES2015 "arrow-parens": 0, "arrow-spacing": [2, { "before": true, "after": true }], "constructor-super": 2, "no-class-assign": 2, "no-const-assign": 2, "no-this-before-super": 0, "no-var": 2, "object-shorthand": [2, "always"], "prefer-arrow-callback": 2, "prefer-const": 2, "prefer-spread": 2, "prefer-template": 2, // Strict Mode "strict": [2, "never"], // Variables "no-catch-shadow": 2, "no-delete-var": 2, "no-label-var": 2, "no-shadow-restricted-names": 2, "no-shadow": 2, "no-undef-init": 2, "no-undef": 2, "no-unused-vars": 2, // Node.js "callback-return": 2, "no-mixed-requires": 2, "no-path-concat": 2, "no-sync": 2, "handle-callback-err": 1, "no-new-require": 2, // Stylistic "array-bracket-spacing": [2, "never", { "singleValue": true, "objectsInArrays": true, "arraysInArrays": true }], "newline-after-var": [1, "always"], "brace-style": [2, "1tbs"], "camelcase": [2, { "properties": "always" }], "comma-spacing": [2, { "before": false, "after": true }], "comma-style": [2, "last"], "computed-property-spacing": [2, "never"], "eol-last": 2, "func-names": 1, "func-style": [2, "declaration"], "indent": [2, 2, { "SwitchCase": 1 }], "jsx-quotes": [2, "prefer-single"], "linebreak-style": [2, "unix"], "max-len": [2, 128, 4, { "ignoreUrls": true, "ignoreComments": false, "ignorePattern": "^\\s*(const|let|var)\\s+\\w+\\s+\\=\\s+\\/.*\\/(|i|g|m|ig|im|gm|igm);?$" }], "max-nested-callbacks": [2, 4], "new-parens": 2, "no-array-constructor": 2, "no-lonely-if": 2, "no-mixed-spaces-and-tabs": 2, "no-multiple-empty-lines": [2, { "max": 2, "maxEOF": 1 }], "no-nested-ternary": 2, "no-new-object": 2, "no-spaced-func": 2, "no-trailing-spaces": 2, "no-unneeded-ternary": 2, "object-curly-spacing": [2, "always"], "one-var": [2, "never"], "padded-blocks": [2, "never"], "quotes": [1, "single", "avoid-escape"], "semi-spacing": [2, { "before": false, "after": true }], "semi": [2, "always"], "keyword-spacing": 2, "space-before-blocks": 2, "space-before-function-paren": [2, { "anonymous": "always", "named": "never" }], "space-in-parens": [2, "never"], "space-infix-ops": 2, "space-unary-ops": [2, { "words": true, "nonwords": false }], "spaced-comment": [2, "always", { "exceptions": ["-", "+"], "markers": ["=", "!"] }], // React "react/jsx-boolean-value": 2, "react/jsx-closing-bracket-location": 2, "react/jsx-curly-spacing": [2, "never"], "react/jsx-handler-names": 2, "react/jsx-indent-props": [2, 2], "react/jsx-indent": [2, 2], "react/jsx-key": 2, "react/jsx-max-props-per-line": [2, {maximum: 3}], "react/jsx-no-bind": [2, { "ignoreRefs": true, "allowBind": true, "allowArrowFunctions": true }], "react/jsx-no-duplicate-props": 2, "react/jsx-no-undef": 2, "react/jsx-pascal-case": 2, "react/jsx-uses-react": 2, "react/jsx-uses-vars": 2, "react/no-danger": 2, "react/no-deprecated": 2, "react/no-did-mount-set-state": 0, "react/no-did-update-set-state": 0, "react/no-direct-mutation-state": 2, "react/no-is-mounted": 2, "react/no-multi-comp": 2, "react/no-string-refs": 2, "react/no-unknown-property": 2, "react/prefer-es6-class": 2, "react/prop-types": 2, "react/react-in-jsx-scope": 2, "react/self-closing-comp": 2, "react/sort-comp": [2, { "order": [ "lifecycle", "/^handle.+$/", "/^(get|set)(?!(InitialState$|DefaultProps$|ChildContext$)).+$/", "everything-else", "/^render.+$/", "render" ] }], "react/jsx-wrap-multilines": 2, // Legacy "max-depth": [0, 4], "max-params": [2, 4], "no-bitwise": 2 }, "globals":{ "$": true, "ga": true } } 

泚 Windowsでは、ルヌル


 "linebreak-style": [2, "unix"], 

に眮き換える必芁がありたす


 "linebreak-style": [2, "windows"], 

 npm i --save-dev babel-eslint eslint eslint-loader eslint-plugin-react 

eslint-loaderをwebpack構成に远加するため、 babelがコヌドをES5に倉換する前に、指定されたルヌルに準拠しおいるかどうかすべおのコヌドがチェックされたす。


webpack.config.js


module.exports.module.loadersで


 --- { test: /\.jsx?$/, loader: 'babel', exclude: [/node_modules/, /public/] }, +++ { test: /\.jsx?$/, loader: 'babel!eslint-loader', exclude: [/node_modules/, /public/] }, 

module.exportsで


 +++ eslint: { configFile: '.eslintrc' }, 

5. ExpressおよびServer.js


珟圚のずころ、プロゞェクトアセンブリ、 JS ES5でのコヌド倉換を構成し、「シラミ甚」の゜ヌスコヌドの怜蚌に぀いお説明し、実装しおいたす。 今床は、アプリケヌション自䜓、぀たりサヌバヌ偎の蚘述を開始したす。


泚私はExpressを䜿甚しおおり、完党に私に適しおいたすが、もちろん、他にも倚くの同様のパッケヌゞがありたすこれはNode.jsです。


゚クスプレスをむンストヌル


 npm i --save express 

次の内容でserver.jsファむルをルヌトに䜜成したす


server.js


 require('babel-core/register'); ['.css', '.less', '.sass', '.ttf', '.woff', '.woff2'].forEach((ext) => require.extensions[ext] = () => {}); require('babel-polyfill'); require('server.js'); 

ここでは、 ES6 / ES7をサポヌトするためにbabelが必芁であるこず、およびノヌドがフォヌムの構造に遭遇した堎合


 import 'awesome.css'; 

この行はJavaScriptたたはその方蚀の1぀ではないため、無芖する必芁がありたす 。


サヌバヌ偎のコヌド自䜓はsrc / server.jsファむルにあり、 ES6 / ES7構文を自由に䜿甚できたす。


src / server.js


 import express from 'express'; const app = express(); app.use((req, res) => { res.end('<p>Hello World!</p>'); }); const PORT = process.env.PORT || 3001; app.listen(PORT, () => { console.log(`Server listening on: ${PORT}`); }); 

ここではすべおが非垞に簡単です ゚クスプレス Webサヌバヌをむンポヌトし、 PORTたたは3001環境倉数で転送されたポヌトで実行したす。サヌバヌ自䜓はすべおの芁求に応答したす "Hello World"


サヌバヌサむドJavaScriptコヌドを実行する蚭蚈に応じお、 nodemonパッケヌゞをむンストヌルしたす。 ゚ラヌが発生するずすぐにコン゜ヌルに詳现なスタックトレヌスずずもに゚ラヌを衚瀺するので䟿利です。


 npm i --save-dev nodemon 

package.jsonに別のスクリプトを远加したす


 +++ "nodemon": "NODE_PATH=./src nodemon server.js", 

Windowsの堎合


 +++ "nodemon": "set NODE_PATH=./src; && nodemon server.js", 

そしお、コン゜ヌルで実行したす


 npm run nodemon 

ブラりザヌを開き、ペヌゞhttp// localhost3001を開きたす。 すべおが順調であれば、 Hello Worldが衚瀺されたす。


6. ReactおよびReactDOM


おめでずうございたす ほがすべおの準備が完了し、最終的に反応に移るこずができたす。


適切なラむブラリをむンストヌルしたす。


 npm i --save react react-dom 

たた、 react-hot-loaderをむンストヌルしたす。開発プロセス䞭にコンポヌネントの゜ヌスコヌドを倉曎するず、ブラりザヌはペヌゞを自動的にリロヌドしたす。 これは、特に耇数のモニタヌがある堎合に非垞に䟿利な機胜です。


 npm i --save-dev react-hot-loader@1.3.0 

泚 npmのサンドボックスに蚘事が残っおいる間、 react-hot-loaderパッケヌゞのバヌゞョンは1.3.xから3.xx-betaに倉曎されたした。 珟圚、3番目のバヌゞョンはあたり文曞化されおいないため、以䞋では最初のバヌゞョンを䜿甚したす。


webpack.config.js


 --- { test: /\.jsx?$/, loader: 'babel!eslint-loader', exclude: [/node_modules/, /public/] }, +++ { test: /\.jsx?$/, loader: process.env.NODE_ENV !== 'production' ? 'react-hot!babel!eslint-loader' : 'babel', exclude: [/node_modules/, /public/] }, 

次に、Webアプリケヌションの同圢郚分ぞの゚ントリポむントであるApp.jsxの最初のコンポヌネントのコヌドの蚘述に移りたしょう。


src / components / App.jsx


 import React, { PropTypes, Component } from 'react'; import './App.css'; const propTypes = { initialName: PropTypes.string }; const defaultProps = { initialName: '' }; class App extends Component { constructor(props) { super(props); this.handleNameChange = this.handleNameChange.bind(this); this.renderGreetingWidget = this.renderGreetingWidget.bind(this); this.state = { name: this.props.initialName, touched: false, greetingWidget: () => null }; } handleNameChange(val) { const name = val.target.value; this.setState({ touched: true }); if (name.length === 0) { this.setState({ name: this.props.initialName }); } else { this.setState({ name }); } } renderGreetingWidget() { if (!this.state.touched) { return null; } return ( <div> <hr /> <p>, {this.state.name}!</p> </div> ); } render() { return ( <div className='App'> <h1>Hello World!</h1> <div> <p>  :</p> <div><input onChange={this.handleNameChange} /></div> {this.renderGreetingWidget()} </div> </div> ); } } App.propTypes = propTypes; App.defaultProps = defaultProps; export default App; 

src / components / App.css


 .App { padding: 20px; } .App h1 { font-size: 26px; } .App input { padding: 10px; } .App hr { margin-top: 20px; } 

ここではすべおが非垞に簡単です。ナヌザヌに名前を入力しお挚拶するように求めおいたす。


アプリケヌションがこのコンポヌネントを衚瀺するために、サヌバヌずクラむアントのコヌドに次の倉曎を加えたす。


src / client.js


 import React from 'react'; import ReactDOM from 'react-dom'; import App from 'components/App'; ReactDOM.render(<App />, document.getElementById('react-view')); 

JavaScriptの初期化埌、reactはreact-viewアプリケヌションのメむンコンテナヌを芋぀けお再アクティブ化したす。


src / server.js


 import express from 'express'; import React from 'react'; import ReactDom from 'react-dom/server'; import App from 'components/App'; const app = express(); app.use((req, res) => { const componentHTML = ReactDom.renderToString(<App />); return res.end(renderHTML(componentHTML)); }); const assetUrl = process.env.NODE_ENV !== 'production' ? 'http://localhost:8050' : '/'; function renderHTML(componentHTML) { return ` <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Hello React</title> <link rel="stylesheet" href="${assetUrl}/public/assets/styles.css"> </head> <body> <div id="react-view">${componentHTML}</div> <script type="application/javascript" src="${assetUrl}/public/assets/bundle.js"></script> </body> </html> `; } const PORT = process.env.PORT || 3001; app.listen(PORT, () => { console.log(`Server listening on: ${PORT}`); }); 

: JavaScript RenderDom.renderToString(<App />) HTML-, , renderHTML . assetUrl : , webpack-dev-server .


, :


 npm run nodemon npm run webpack-devserver 

: http://localhost:3001 
 - !


, .


1) server-side rendering . webpack-dev-server . , , .


2) client-side rendering . src/server.js , , .


 --- <div id="react-view">${componentHTML}</div> +++ <div id="react-view"></div> 

. , . !


: , , npm run webpack-devserver, .


Github


: https://github.com/yury-dymov/habr-app
https://github.com/yury-dymov/habr-app/tree/v1 — v1
https://github.com/yury-dymov/habr-app/tree/v2 — v2
v3 [To be done]


次は


Hello World ? , , !
react-bootstrap , , , , flux redux .


7.


  1. JavaScript ES2015 — https://learn.javascript.ru/es-modern
  2. webpack — http://webpack.imtqy.com/docs/
  3. webpack — https://learn.javascript.ru/screencast/webpack
  4. Babel — https://babeljs.io/
  5. ESLint — http://eslint.org/
  6. Express — https://expressjs.com/
  7. Express — http://jsman.ru/express/
  8. React — https://facebook.imtqy.com/react/

Ps , , . 事前に感謝したす



Source: https://habr.com/ru/post/J309958/


All Articles