2016-04-14 00:28:44 +02:00
|
|
|
const extend = require('xtend')
|
|
|
|
const actions = require('../actions')
|
2016-05-03 23:32:22 +02:00
|
|
|
const txHelper = require('../../lib/tx-helper')
|
2016-04-14 00:28:44 +02:00
|
|
|
|
|
|
|
module.exports = reduceApp
|
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
function reduceApp (state, action) {
|
2016-04-14 00:28:44 +02:00
|
|
|
// clone and defaults
|
2016-04-26 00:18:20 +02:00
|
|
|
const selectedAccount = state.metamask.selectedAccount
|
2016-05-26 01:28:07 +02:00
|
|
|
const pendingTxs = hasPendingTxs(state)
|
|
|
|
let name = 'accounts'
|
|
|
|
if (selectedAccount) {
|
2016-05-26 23:32:45 +02:00
|
|
|
name = 'accountDetail'
|
2016-05-26 01:28:07 +02:00
|
|
|
}
|
|
|
|
if (pendingTxs) {
|
2016-05-26 23:32:45 +02:00
|
|
|
name = 'confTx'
|
2016-05-26 01:28:07 +02:00
|
|
|
}
|
|
|
|
|
2016-04-14 00:28:44 +02:00
|
|
|
var defaultView = {
|
2016-05-26 01:28:07 +02:00
|
|
|
name,
|
2016-04-14 00:28:44 +02:00
|
|
|
detailView: null,
|
2016-04-26 00:18:20 +02:00
|
|
|
context: selectedAccount,
|
2016-04-14 00:28:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// confirm seed words
|
2016-06-03 02:11:12 +02:00
|
|
|
var seedWords = state.metamask.seedWords
|
2016-04-14 00:28:44 +02:00
|
|
|
var seedConfView = {
|
|
|
|
name: 'createVaultComplete',
|
2016-06-03 02:11:12 +02:00
|
|
|
seedWords,
|
2016-04-14 00:28:44 +02:00
|
|
|
}
|
2016-07-21 22:41:10 +02:00
|
|
|
var ethStoreWarning = {
|
|
|
|
name: 'EthStoreWarning',
|
|
|
|
}
|
2016-04-14 00:28:44 +02:00
|
|
|
|
|
|
|
var appState = extend({
|
2016-05-18 21:30:03 +02:00
|
|
|
menuOpen: false,
|
2016-07-21 22:41:10 +02:00
|
|
|
currentView: seedWords ? seedConfView : !state.metamask.isEthConfirmed ? ethStoreWarning : defaultView,
|
2016-05-05 03:08:31 +02:00
|
|
|
accountDetail: {
|
|
|
|
subview: 'transactions',
|
|
|
|
},
|
2016-04-14 00:28:44 +02:00
|
|
|
currentDomain: 'example.com',
|
|
|
|
transForward: true, // Used to render transition direction
|
|
|
|
isLoading: false, // Used to display loading indicator
|
|
|
|
warning: null, // Used to display error text
|
|
|
|
}, state.appState)
|
|
|
|
|
|
|
|
switch (action.type) {
|
|
|
|
|
|
|
|
// intialize
|
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
case actions.SHOW_CREATE_VAULT:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: {
|
|
|
|
name: 'createVault',
|
|
|
|
},
|
|
|
|
transForward: true,
|
|
|
|
warning: null,
|
|
|
|
})
|
2016-04-14 00:28:44 +02:00
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
case actions.SHOW_RESTORE_VAULT:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: {
|
|
|
|
name: 'restoreVault',
|
|
|
|
},
|
|
|
|
transForward: true,
|
|
|
|
})
|
2016-04-14 00:28:44 +02:00
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
case actions.SHOW_INIT_MENU:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: defaultView,
|
|
|
|
transForward: false,
|
|
|
|
})
|
2016-04-14 00:28:44 +02:00
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
case actions.SHOW_CONFIG_PAGE:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: {
|
|
|
|
name: 'config',
|
|
|
|
context: appState.currentView.context,
|
|
|
|
},
|
|
|
|
transForward: action.value,
|
|
|
|
})
|
2016-06-03 01:52:18 +02:00
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
case actions.SHOW_INFO_PAGE:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: {
|
|
|
|
name: 'info',
|
|
|
|
context: appState.currentView.context,
|
|
|
|
},
|
|
|
|
transForward: true,
|
|
|
|
})
|
2016-06-03 01:52:18 +02:00
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
case actions.CREATE_NEW_VAULT_IN_PROGRESS:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: {
|
|
|
|
name: 'createVault',
|
|
|
|
inProgress: true,
|
|
|
|
},
|
|
|
|
transForward: true,
|
|
|
|
isLoading: true,
|
|
|
|
})
|
2016-06-03 01:52:18 +02:00
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
case actions.SHOW_NEW_VAULT_SEED:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: {
|
|
|
|
name: 'createVaultComplete',
|
|
|
|
seedWords: action.value,
|
|
|
|
},
|
|
|
|
transForward: true,
|
|
|
|
isLoading: false,
|
|
|
|
})
|
2016-04-14 00:28:44 +02:00
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
case actions.SHOW_SEND_PAGE:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: {
|
|
|
|
name: 'sendTransaction',
|
|
|
|
context: appState.currentView.context,
|
|
|
|
},
|
|
|
|
transForward: true,
|
|
|
|
warning: null,
|
|
|
|
})
|
|
|
|
|
|
|
|
// unlock
|
2016-05-03 23:32:22 +02:00
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
case actions.UNLOCK_METAMASK:
|
2016-04-14 00:28:44 +02:00
|
|
|
return extend(appState, {
|
2016-06-21 22:18:32 +02:00
|
|
|
detailView: {},
|
|
|
|
transForward: true,
|
|
|
|
isLoading: false,
|
|
|
|
warning: null,
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.LOCK_METAMASK:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: defaultView,
|
2016-04-14 00:28:44 +02:00
|
|
|
transForward: false,
|
2016-06-21 22:18:32 +02:00
|
|
|
warning: null,
|
|
|
|
})
|
|
|
|
|
|
|
|
// reveal seed words
|
|
|
|
|
|
|
|
case actions.REVEAL_SEED_CONFIRMATION:
|
|
|
|
return extend(appState, {
|
2016-04-14 00:28:44 +02:00
|
|
|
currentView: {
|
2016-06-21 22:18:32 +02:00
|
|
|
name: 'reveal-seed-conf',
|
2016-04-14 00:28:44 +02:00
|
|
|
},
|
2016-06-21 22:18:32 +02:00
|
|
|
transForward: true,
|
2016-04-14 00:28:44 +02:00
|
|
|
warning: null,
|
|
|
|
})
|
2016-06-21 22:18:32 +02:00
|
|
|
|
|
|
|
// accounts
|
|
|
|
|
|
|
|
case actions.SET_SELECTED_ACCOUNT:
|
2016-04-14 00:28:44 +02:00
|
|
|
return extend(appState, {
|
2016-06-21 22:18:32 +02:00
|
|
|
activeAddress: action.value,
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.GO_HOME:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: extend(appState.currentView, {
|
|
|
|
name: 'accountDetail',
|
|
|
|
}),
|
|
|
|
accountDetail: {
|
|
|
|
subview: 'transactions',
|
|
|
|
accountExport: 'none',
|
|
|
|
privateKey: '',
|
|
|
|
},
|
2016-04-14 00:28:44 +02:00
|
|
|
transForward: false,
|
2016-04-30 00:53:29 +02:00
|
|
|
warning: null,
|
2016-06-21 22:18:32 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
case actions.SHOW_ACCOUNT_DETAIL:
|
|
|
|
return extend(appState, {
|
2016-04-14 00:28:44 +02:00
|
|
|
currentView: {
|
2016-04-30 00:53:29 +02:00
|
|
|
name: 'accountDetail',
|
2016-06-21 22:56:04 +02:00
|
|
|
context: action.value,
|
2016-04-14 00:28:44 +02:00
|
|
|
},
|
2016-05-05 03:08:31 +02:00
|
|
|
accountDetail: {
|
|
|
|
subview: 'transactions',
|
2016-06-21 22:18:32 +02:00
|
|
|
accountExport: 'none',
|
|
|
|
privateKey: '',
|
2016-05-05 03:08:31 +02:00
|
|
|
},
|
2016-06-21 22:18:32 +02:00
|
|
|
transForward: false,
|
2016-04-14 00:28:44 +02:00
|
|
|
})
|
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
case actions.BACK_TO_ACCOUNT_DETAIL:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: {
|
|
|
|
name: 'accountDetail',
|
|
|
|
context: action.value,
|
|
|
|
},
|
|
|
|
accountDetail: {
|
|
|
|
subview: 'transactions',
|
|
|
|
accountExport: 'none',
|
|
|
|
privateKey: '',
|
|
|
|
},
|
|
|
|
transForward: false,
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.SHOW_ACCOUNTS_PAGE:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: {
|
|
|
|
name: seedWords ? 'createVaultComplete' : 'accounts',
|
|
|
|
seedWords,
|
|
|
|
},
|
|
|
|
transForward: true,
|
|
|
|
isLoading: false,
|
2016-04-14 00:28:44 +02:00
|
|
|
warning: null,
|
2016-06-21 22:18:32 +02:00
|
|
|
scrollToBottom: false,
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.REVEAL_ACCOUNT:
|
|
|
|
return extend(appState, {
|
|
|
|
scrollToBottom: true,
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.SHOW_CONF_TX_PAGE:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: {
|
|
|
|
name: 'confTx',
|
|
|
|
context: 0,
|
|
|
|
},
|
|
|
|
transForward: true,
|
2016-05-26 23:32:45 +02:00
|
|
|
warning: null,
|
2016-06-21 22:18:32 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
case actions.SHOW_CONF_MSG_PAGE:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: {
|
|
|
|
name: 'confTx',
|
|
|
|
context: 0,
|
|
|
|
},
|
|
|
|
transForward: true,
|
2016-04-14 00:28:44 +02:00
|
|
|
warning: null,
|
2016-06-21 22:18:32 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
case actions.COMPLETED_TX:
|
|
|
|
var unconfTxs = state.metamask.unconfTxs
|
|
|
|
var unconfMsgs = state.metamask.unconfMsgs
|
|
|
|
|
|
|
|
var unconfTxList = txHelper(unconfTxs, unconfMsgs)
|
|
|
|
.filter(tx => tx !== tx.id)
|
|
|
|
|
|
|
|
if (unconfTxList && unconfTxList.length > 0) {
|
|
|
|
return extend(appState, {
|
|
|
|
transForward: false,
|
|
|
|
currentView: {
|
|
|
|
name: 'confTx',
|
|
|
|
context: 0,
|
|
|
|
},
|
|
|
|
warning: null,
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
return extend(appState, {
|
|
|
|
transForward: false,
|
|
|
|
warning: null,
|
|
|
|
currentView: {
|
|
|
|
name: 'accountDetail',
|
|
|
|
context: state.metamask.selectedAddress,
|
|
|
|
},
|
|
|
|
accountDetail: {
|
|
|
|
subview: 'transactions',
|
|
|
|
},
|
|
|
|
})
|
2016-04-14 00:28:44 +02:00
|
|
|
}
|
2016-06-21 22:18:32 +02:00
|
|
|
|
|
|
|
case actions.NEXT_TX:
|
|
|
|
return extend(appState, {
|
|
|
|
transForward: true,
|
|
|
|
currentView: {
|
|
|
|
name: 'confTx',
|
|
|
|
context: ++appState.currentView.context,
|
|
|
|
warning: null,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.VIEW_PENDING_TX:
|
|
|
|
const context = indexForPending(state, action.value)
|
|
|
|
return extend(appState, {
|
|
|
|
transForward: true,
|
|
|
|
currentView: {
|
|
|
|
name: 'confTx',
|
|
|
|
context,
|
|
|
|
warning: null,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.PREVIOUS_TX:
|
|
|
|
return extend(appState, {
|
|
|
|
transForward: false,
|
|
|
|
currentView: {
|
|
|
|
name: 'confTx',
|
|
|
|
context: --appState.currentView.context,
|
|
|
|
warning: null,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.TRANSACTION_ERROR:
|
|
|
|
return extend(appState, {
|
|
|
|
currentView: {
|
|
|
|
name: 'confTx',
|
|
|
|
errorMessage: 'There was a problem submitting this transaction.',
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.UNLOCK_FAILED:
|
|
|
|
return extend(appState, {
|
|
|
|
warning: 'Incorrect password. Try again.',
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.SHOW_LOADING:
|
|
|
|
return extend(appState, {
|
|
|
|
isLoading: true,
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.HIDE_LOADING:
|
|
|
|
return extend(appState, {
|
|
|
|
isLoading: false,
|
|
|
|
})
|
|
|
|
|
2016-08-10 22:51:14 +02:00
|
|
|
case actions.SHOW_SUB_LOADING_INDICATION:
|
|
|
|
return extend(appState, {
|
|
|
|
isSubLoading: true,
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.HIDE_SUB_LOADING_INDICATION:
|
|
|
|
return extend(appState, {
|
|
|
|
isSubLoading: false,
|
|
|
|
})
|
2016-06-21 22:18:32 +02:00
|
|
|
case actions.CLEAR_SEED_WORD_CACHE:
|
|
|
|
return extend(appState, {
|
|
|
|
transForward: true,
|
|
|
|
currentView: {},
|
|
|
|
isLoading: false,
|
|
|
|
accountDetail: {
|
|
|
|
subview: 'transactions',
|
|
|
|
accountExport: 'none',
|
|
|
|
privateKey: '',
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.DISPLAY_WARNING:
|
|
|
|
return extend(appState, {
|
|
|
|
warning: action.value,
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.HIDE_WARNING:
|
|
|
|
return extend(appState, {
|
|
|
|
warning: undefined,
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.REQUEST_ACCOUNT_EXPORT:
|
|
|
|
return extend(appState, {
|
|
|
|
transForward: true,
|
|
|
|
currentView: {
|
|
|
|
name: 'accountDetail',
|
|
|
|
context: appState.currentView.context,
|
|
|
|
},
|
|
|
|
accountDetail: {
|
|
|
|
subview: 'export',
|
|
|
|
accountExport: 'requested',
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.EXPORT_ACCOUNT:
|
|
|
|
return extend(appState, {
|
|
|
|
accountDetail: {
|
|
|
|
subview: 'export',
|
|
|
|
accountExport: 'completed',
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.SHOW_PRIVATE_KEY:
|
|
|
|
return extend(appState, {
|
|
|
|
accountDetail: {
|
|
|
|
subview: 'export',
|
|
|
|
accountExport: 'completed',
|
|
|
|
privateKey: action.value,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
2016-08-13 00:41:59 +02:00
|
|
|
case actions.BUY_ETH_VIEW:
|
2016-07-21 22:41:10 +02:00
|
|
|
return extend(appState, {
|
|
|
|
transForward: true,
|
|
|
|
currentView: {
|
2016-08-13 00:41:59 +02:00
|
|
|
name: 'buyEth',
|
2016-07-21 22:41:10 +02:00
|
|
|
context: appState.currentView.context,
|
|
|
|
},
|
2016-08-13 00:41:59 +02:00
|
|
|
buyView: {
|
2016-08-10 22:51:14 +02:00
|
|
|
subview: 'buyForm',
|
|
|
|
amount: '5.00',
|
2016-08-13 00:41:59 +02:00
|
|
|
buyAddress: action.value,
|
2016-08-10 22:51:14 +02:00
|
|
|
formView: {
|
|
|
|
coinbase: true,
|
|
|
|
shapeshift: false,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.UPDATE_BUY_ADDRESS:
|
|
|
|
return extend(appState, {
|
2016-08-13 00:41:59 +02:00
|
|
|
buyView: {
|
2016-08-10 22:51:14 +02:00
|
|
|
subview: 'buyForm',
|
|
|
|
formView: {
|
2016-08-13 00:41:59 +02:00
|
|
|
coinbase: appState.buyView.formView.coinbase,
|
|
|
|
shapeshift: appState.buyView.formView.shapeshift,
|
2016-08-10 22:51:14 +02:00
|
|
|
},
|
|
|
|
buyAddress: action.value,
|
2016-08-13 00:41:59 +02:00
|
|
|
amount: appState.buyView.amount,
|
2016-08-10 22:51:14 +02:00
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.UPDATE_COINBASE_AMOUNT:
|
|
|
|
return extend(appState, {
|
2016-08-13 00:41:59 +02:00
|
|
|
buyView: {
|
2016-08-10 22:51:14 +02:00
|
|
|
subview: 'buyForm',
|
|
|
|
formView: {
|
|
|
|
coinbase: true,
|
|
|
|
shapeshift: false,
|
|
|
|
},
|
2016-08-13 00:41:59 +02:00
|
|
|
buyAddress: appState.buyView.buyAddress,
|
2016-08-10 22:51:14 +02:00
|
|
|
amount: action.value,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.COINBASE_SUBVIEW:
|
|
|
|
return extend(appState, {
|
2016-08-13 00:41:59 +02:00
|
|
|
buyView: {
|
2016-08-10 22:51:14 +02:00
|
|
|
subview: 'buyForm',
|
|
|
|
formView: {
|
|
|
|
coinbase: true,
|
|
|
|
shapeshift: false,
|
|
|
|
},
|
2016-08-13 00:41:59 +02:00
|
|
|
buyAddress: appState.buyView.buyAddress,
|
|
|
|
amount: appState.buyView.amount,
|
2016-08-10 22:51:14 +02:00
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.SHAPESHIFT_SUBVIEW:
|
|
|
|
return extend(appState, {
|
2016-08-13 00:41:59 +02:00
|
|
|
buyView: {
|
2016-08-10 22:51:14 +02:00
|
|
|
subview: 'buyForm',
|
|
|
|
formView: {
|
|
|
|
coinbase: false,
|
|
|
|
shapeshift: true,
|
|
|
|
marketinfo: action.value.marketinfo,
|
|
|
|
coinOptions: action.value.coinOptions,
|
|
|
|
},
|
2016-08-13 00:41:59 +02:00
|
|
|
buyAddress: appState.buyView.buyAddress,
|
|
|
|
amount: appState.buyView.amount,
|
2016-08-10 22:51:14 +02:00
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
case actions.PAIR_UPDATE:
|
|
|
|
return extend(appState, {
|
2016-08-13 00:41:59 +02:00
|
|
|
buyView: {
|
2016-08-10 22:51:14 +02:00
|
|
|
subview: 'buyForm',
|
|
|
|
formView: {
|
|
|
|
coinbase: false,
|
|
|
|
shapeshift: true,
|
|
|
|
marketinfo: action.value.marketinfo,
|
2016-08-13 00:41:59 +02:00
|
|
|
coinOptions: appState.buyView.formView.coinOptions,
|
2016-08-10 22:51:14 +02:00
|
|
|
},
|
2016-08-13 00:41:59 +02:00
|
|
|
buyAddress: appState.buyView.buyAddress,
|
|
|
|
amount: appState.buyView.amount,
|
2016-08-10 22:51:14 +02:00
|
|
|
warning: null,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
2016-08-13 00:41:59 +02:00
|
|
|
case actions.SHOW_QR:
|
2016-08-10 22:51:14 +02:00
|
|
|
return extend(appState, {
|
2016-08-13 00:41:59 +02:00
|
|
|
qrRequested: true,
|
|
|
|
transForward: true,
|
|
|
|
Qr: {
|
|
|
|
message: action.value.message,
|
|
|
|
image: action.value.qr,
|
|
|
|
data: action.value.data,
|
2016-07-21 22:41:10 +02:00
|
|
|
},
|
|
|
|
})
|
2016-06-21 22:18:32 +02:00
|
|
|
default:
|
|
|
|
return appState
|
2016-04-14 00:28:44 +02:00
|
|
|
}
|
|
|
|
}
|
2016-05-26 01:28:07 +02:00
|
|
|
|
|
|
|
function hasPendingTxs (state) {
|
|
|
|
var unconfTxs = state.metamask.unconfTxs
|
|
|
|
var unconfMsgs = state.metamask.unconfMsgs
|
|
|
|
var unconfTxList = txHelper(unconfTxs, unconfMsgs)
|
|
|
|
return unconfTxList.length > 0
|
|
|
|
}
|
2016-05-26 23:32:45 +02:00
|
|
|
|
2016-06-21 22:18:32 +02:00
|
|
|
function indexForPending (state, txId) {
|
2016-05-26 23:32:45 +02:00
|
|
|
var unconfTxs = state.metamask.unconfTxs
|
|
|
|
var unconfMsgs = state.metamask.unconfMsgs
|
|
|
|
var unconfTxList = txHelper(unconfTxs, unconfMsgs)
|
|
|
|
let idx
|
|
|
|
unconfTxList.forEach((tx, i) => {
|
|
|
|
if (tx.id === txId) {
|
|
|
|
idx = i
|
|
|
|
}
|
|
|
|
})
|
|
|
|
return idx
|
|
|
|
}
|
2016-07-21 22:41:10 +02:00
|
|
|
|
|
|
|
|