angular + redux 項(xiàng)目實(shí)踐的一些探索

首先是配置$ngRedux 服務(wù)

app.config(/*@ngInject*/$ngReduxProvider => {
        let middleware = ['ngUiRouterMiddleware']

        const reducers = {
            router,         //靜態(tài)寫法
            lang: langReducer,
            session: sessionReducer,
            [actionTagManagement.name]: actionTagManagement.reducer   //動態(tài)寫法
        }

      $ngReduxProvider.createStoreWith(reducers, middleware)

項(xiàng)目中的探索

目的效果就是幾個簡單的折疊展開的效果。
按照angular的做法很簡單勉耀,通過ng-show能夠快速完成指煎,現(xiàn)在思考用redux的方式。
最初的代碼:

import html from './index.html'
import './index.scss'

let prefix = 'actionApi_'     //需要prefix來確保action的全局唯一性
const OPT_STATUS = prefix + 'OPT_STATUS'
const OPT_SAMPLE = prefix + 'OPT_SAMPLE'
const OPT_LOG = prefix + 'OPT_LOG'

const defaultState = {
    showStatus: true,
    showLog: false,
    showSample: false
}

export function reducer(state = defaultState, action) {
    switch (action.type){
        case OPT_STATUS: {
            return Object.assign({}, state, {
                    showStatus: action.data
            })
        }
        case OPT_SAMPLE: {
            return Object.assign({}, state, {
                    showSample: action.data
            })
        }
        case OPT_LOG: {
            return Object.assign({}, state, {
                    showLog: action.data
            })
        }
        default: {
            return state
        }
    }
}
export const name = 'action.api'
export const state = {
    url: '/api',
    template: html,
    controllerAs: '$ctrl',
    controller: /*@ngInject*/($scope, $ngRedux) => {
        let unSubscribe = $ngRedux.connect((state) => {
            return {
                  showStatus: state[name].showStatus,
                  showSample: state[name].showSample,
                  showLog: state[name].showLog
            }
        }, Object.assign({}))($scope)

        $scope.$on('$destroy', unSubscribe)     //銷毀監(jiān)聽器

        $scope.collapseStatus = (data) => {
            $ngRedux.dispatch(function(data){
                return {
                    type: OPT_STATUS,
                    data
                }
            }(!data))
        }       
        //function 為action 生成函數(shù)便斥,這里定義函數(shù)后立即執(zhí)行至壤,實(shí)際上生成了一個action
        $scope.collapseSample = (data) => {
            $ngRedux.dispatch(function(data){
                return {
                    type: OPT_SAMPLE,
                    data
                }
            }(!data))
        }

        $scope.collapseLog = (data) => {
            $ngRedux.dispatch(function(data){
                return {
                    type: OPT_LOG,
                    data
                }
            }(!data))
        }
       
    }
}

html 中:

<h2 ng-click="collapseSample(showSample)">
         <a href=""><span class="glyphicon glyphicon-menu-down doc-arrow" ng-class="{'doc-arrow-hide': showSample == false}"></span></a>
         <span class="doc-h2-pointer">獲取Sample數(shù)據(jù)</span>
 </h2>
 <div class="doc-hide-content" ng-show="showSample">
......
</div>

目的已經(jīng)完成了。寫法上很像angular枢纠。

$ngRedux.connect()函數(shù)的第一個參數(shù)實(shí)際上是一個函數(shù)像街,它返回一個對象。這里的state即全局的state晋渺,我們得到我們需要的本地state镰绎,并和angular中的變量建立一種綁定關(guān)系。

第二個參數(shù)是一個對象些举,表示我們進(jìn)行的一些操作 跟狱,redux是通過觸發(fā)action來更改狀態(tài)俭厚。我們的操作實(shí)際上也就是返回一些action户魏,交由reducer來更改state,從而達(dá)到改變視圖的目的

代碼看起來有點(diǎn)臃腫。讓我們進(jìn)行一下優(yōu)化:

  let unSubscribe = $ngRedux.connect((state) => {
            Object.assign({}, state[name])
        }, Object.assign({}, {
            collapseStatus: (data) => {
                return {
                    type: OPT_STATUS,
                    data: !data
                }
            },
            collapseSample: (data) => {
                return {
                    type: OPT_SAMPLE,
                    data: !data
                }
            },
            collapseLog: (data) => {
                return {
                    type: OPT_LOG,
                    data: !data
                }
            }
        }))($scope)

        $scope.$on('$destroy', unSubscribe)     //銷毀監(jiān)聽器

看起來還是有點(diǎn)別扭叼丑,前面的代碼中我們實(shí)際上聲明了三種action关翎,分別對應(yīng)于三個欄目的折疊情況。那么問題來了鸠信,當(dāng)一個頁面需要折疊的地方很多怎么辦纵寝?難道一直增加action嗎?仔細(xì)想一下星立,這幾個action本身就非常相似爽茴。考慮繼續(xù)縮減代碼绰垂,只提供一個action生成函數(shù)室奏。

這時候,為了描述每個欄目的折疊狀態(tài)劲装,我們的action生成函數(shù)需要兩個參數(shù)胧沫,一個描述是哪個欄目,一個描述折疊狀態(tài)(是展開還是折疊)

        Object.assign({}, {collapse: (option, selected) => {
            let data = {
                [option]: !selected
            }
            return {
                type: OPT_COLLAPSE, 
                data
            }
        }}))($scope)

同樣的占业,reducer函數(shù)里面也可以縮減绒怨,像這樣:

export function reducer(state = defaultState, action) {
    switch (action.type){
        case OPT_COLLAPSE: {
            return Object.assign({}, state, action.data)
        }
        default: {
            return state
        }
    }
}

相應(yīng)的,更改html:

<h2 ng-click="collapse('showSample', showSample)">
       <a href=""><span class="glyphicon glyphicon-menu-down doc-arrow" ng-class="{'doc-arrow-hide': showSample == false}"></span></a>
       <span class="doc-h2-pointer">獲取Sample數(shù)據(jù)</span>
</h2>

大功告成谦疾,完整js代碼如下:

import html from './index.html'
import './index.scss'

let prefix = 'actionApi_'
const OPT_COLLAPSE = prefix + 'OPT_COLLAPSE'

const defaultState = {
    showStatus: true,
    showLog: false,
    showSample: false
}

export function reducer(state = defaultState, action) {
    switch (action.type){
        case OPT_COLLAPSE: {
            return Object.assign({}, state, action.data)
        }
        default: {
            return state
        }
    }
}
export const name = 'action.api'
export const state = {
    url: '/api',
    template: html,
    controllerAs: '$ctrl',
    controller: /*@ngInject*/($scope, $ngRedux, langActions, sessionActions) => {
        let unSubscribe = $ngRedux.connect((state) => {
            return Object.assign({}, state[name])
        }, Object.assign({}, {collapse: (option, selected) => {
            let data = {
                [option]: !selected
            }
            return {
                type: OPT_COLLAPSE, 
                data
            }
            }}))($scope)

        $scope.$on('$destroy', unSubscribe)
    }
}

寫在后面

前文曾經(jīng)說過南蹂,這個效果其實(shí)單純用angular很容易就能實(shí)現(xiàn),這里用redux的方式有點(diǎn)殺雞用牛刀的感覺念恍。

之所以這么大費(fèi)周章碎紊,一方面當(dāng)然是為了學(xué)習(xí)新技術(shù),對action生成函數(shù)樊诺,reducer結(jié)構(gòu)優(yōu)化的整個過程相當(dāng)有意思仗考;另一方面也是出于項(xiàng)目需要,雙向綁定雖然用起來非常爽词爬,但是當(dāng)項(xiàng)目逐漸龐大時,"視圖->模型->視圖"的變化會越來越復(fù)雜秃嗜,直到你最后搞不清楚這些變化的源頭

文檔與參考

redux 中文文檔
angular-redux/ng-redux

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市顿膨,隨后出現(xiàn)的幾起案子锅锨,更是在濱河造成了極大的恐慌,老刑警劉巖恋沃,帶你破解...
    沈念sama閱讀 222,252評論 6 516
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件必搞,死亡現(xiàn)場離奇詭異,居然都是意外死亡囊咏,警方通過查閱死者的電腦和手機(jī)恕洲,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,886評論 3 399
  • 文/潘曉璐 我一進(jìn)店門塔橡,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人霜第,你說我怎么就攤上這事葛家。” “怎么了泌类?”我有些...
    開封第一講書人閱讀 168,814評論 0 361
  • 文/不壞的土叔 我叫張陵癞谒,是天一觀的道長。 經(jīng)常有香客問我刃榨,道長弹砚,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,869評論 1 299
  • 正文 為了忘掉前任枢希,我火速辦了婚禮迅栅,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘晴玖。我一直安慰自己读存,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,888評論 6 398
  • 文/花漫 我一把揭開白布呕屎。 她就那樣靜靜地躺著让簿,像睡著了一般。 火紅的嫁衣襯著肌膚如雪秀睛。 梳的紋絲不亂的頭發(fā)上尔当,一...
    開封第一講書人閱讀 52,475評論 1 312
  • 那天,我揣著相機(jī)與錄音蹂安,去河邊找鬼椭迎。 笑死,一個胖子當(dāng)著我的面吹牛田盈,可吹牛的內(nèi)容都是我干的畜号。 我是一名探鬼主播,決...
    沈念sama閱讀 41,010評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼允瞧,長吁一口氣:“原來是場噩夢啊……” “哼简软!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起述暂,我...
    開封第一講書人閱讀 39,924評論 0 277
  • 序言:老撾萬榮一對情侶失蹤痹升,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后畦韭,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體疼蛾,經(jīng)...
    沈念sama閱讀 46,469評論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,552評論 3 342
  • 正文 我和宋清朗相戀三年艺配,在試婚紗的時候發(fā)現(xiàn)自己被綠了察郁。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片衍慎。...
    茶點(diǎn)故事閱讀 40,680評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖绳锅,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情酝掩,我是刑警寧澤鳞芙,帶...
    沈念sama閱讀 36,362評論 5 351
  • 正文 年R本政府宣布,位于F島的核電站期虾,受9級特大地震影響原朝,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜镶苞,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,037評論 3 335
  • 文/蒙蒙 一喳坠、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧茂蚓,春花似錦壕鹉、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,519評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至牍白,卻和暖如春脊凰,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背茂腥。 一陣腳步聲響...
    開封第一講書人閱讀 33,621評論 1 274
  • 我被黑心中介騙來泰國打工狸涌, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人最岗。 一個月前我還...
    沈念sama閱讀 49,099評論 3 378
  • 正文 我出身青樓帕胆,卻偏偏與公主長得像,于是被迫代替她去往敵國和親般渡。 傳聞我的和親對象是個殘疾皇子惶楼,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,691評論 2 361

推薦閱讀更多精彩內(nèi)容