lbry-desktop/ui/js/actions/content.js

294 lines
7 KiB
JavaScript
Raw Normal View History

2017-04-23 11:56:50 +02:00
import * as types from 'constants/action_types'
import lbry from 'lbry'
2017-04-24 16:17:36 +02:00
import lbryio from 'lbryio'
import lbryuri from 'lbryuri'
2017-05-01 08:39:16 +02:00
import rewards from 'rewards'
import {
selectBalance,
} from 'selectors/wallet'
2017-04-28 17:14:44 +02:00
import {
2017-05-15 05:50:59 +02:00
selectFileInfoForUri,
selectUrisDownloading,
2017-04-28 17:14:44 +02:00
} from 'selectors/file_info'
2017-05-15 18:34:33 +02:00
import {
selectResolvingUris
} from 'selectors/content'
2017-04-28 17:14:44 +02:00
import {
2017-05-15 05:50:59 +02:00
selectCostInfoForUri,
2017-04-28 17:14:44 +02:00
} from 'selectors/cost_info'
import {
selectClaimsByUri,
} from 'selectors/claims'
import {
doOpenModal,
} from 'actions/app'
2017-04-23 11:56:50 +02:00
export function doResolveUri(uri) {
return function(dispatch, getState) {
2017-05-21 16:42:34 +02:00
uri = lbryuri.normalize(uri)
2017-05-15 18:34:33 +02:00
const state = getState()
const alreadyResolving = selectResolvingUris(state).indexOf(uri) !== -1
2017-05-15 18:34:33 +02:00
if (!alreadyResolving) {
dispatch({
2017-05-15 18:34:33 +02:00
type: types.RESOLVE_URI_STARTED,
data: { uri }
})
lbry.resolve({ uri }).then((resolutionInfo) => {
const {
claim,
certificate,
2017-05-15 18:34:33 +02:00
} = resolutionInfo ? resolutionInfo : { claim : null, certificate: null }
dispatch({
type: types.RESOLVE_URI_COMPLETED,
data: {
uri,
claim,
certificate,
}
})
})
2017-05-15 18:34:33 +02:00
}
}
}
export function doCancelResolveUri(uri) {
return function(dispatch, getState) {
lbry.cancelResolve({ uri })
dispatch({
type: types.RESOLVE_URI_CANCELED,
data: { uri }
})
}
}
2017-05-04 05:44:08 +02:00
export function doFetchFeaturedUris() {
2017-04-23 11:56:50 +02:00
return function(dispatch, getState) {
const state = getState()
dispatch({
type: types.FETCH_FEATURED_CONTENT_STARTED,
})
const success = ({ Categories, Uris }) => {
2017-05-04 05:44:08 +02:00
let featuredUris = {}
Categories.forEach((category) => {
if (Uris[category] && Uris[category].length) {
featuredUris[category] = Uris[category]
}
})
2017-04-23 11:56:50 +02:00
dispatch({
type: types.FETCH_FEATURED_CONTENT_COMPLETED,
data: {
categories: Categories,
uris: featuredUris,
2017-04-23 11:56:50 +02:00
}
})
}
const failure = () => {
2017-05-04 05:44:08 +02:00
dispatch({
type: types.FETCH_FEATURED_CONTENT_COMPLETED,
data: {
categories: [],
uris: {}
}
})
2017-04-23 11:56:50 +02:00
}
lbryio.call('discover', 'list', { version: "early-access" } )
.then(success, failure)
}
}
export function doUpdateLoadStatus(uri, outpoint) {
return function(dispatch, getState) {
const state = getState()
lbry.file_list({
outpoint: outpoint,
full_status: true,
}).then(([fileInfo]) => {
if(!fileInfo || fileInfo.written_bytes == 0) {
// download hasn't started yet
setTimeout(() => { dispatch(doUpdateLoadStatus(uri, outpoint)) }, 250)
} else if (fileInfo.completed) {
2017-05-01 08:39:16 +02:00
// TODO this isn't going to get called if they reload the client before
// the download finished
rewards.claimNextPurchaseReward()
dispatch({
type: types.DOWNLOADING_COMPLETED,
data: {
uri,
outpoint,
fileInfo,
}
})
} else {
// ready to play
const {
total_bytes,
written_bytes,
} = fileInfo
const progress = (written_bytes / total_bytes) * 100
dispatch({
type: types.DOWNLOADING_PROGRESSED,
data: {
uri,
outpoint,
fileInfo,
progress,
}
})
setTimeout(() => { dispatch(doUpdateLoadStatus(uri, outpoint)) }, 250)
}
})
}
}
export function doDownloadFile(uri, streamInfo) {
return function(dispatch, getState) {
const state = getState()
lbry.file_list({ outpoint: streamInfo.outpoint, full_status: true }).then(([fileInfo]) => {
dispatch({
type: types.DOWNLOADING_STARTED,
data: {
uri,
outpoint: streamInfo.outpoint,
fileInfo,
}
})
})
lbryio.call('file', 'view', {
uri: uri,
outpoint: streamInfo.outpoint,
claimId: streamInfo.claim_id,
}).catch(() => {})
dispatch(doUpdateLoadStatus(uri, streamInfo.outpoint))
}
}
2017-05-15 05:50:59 +02:00
export function doLoadVideo(uri) {
return function(dispatch, getState) {
const state = getState()
dispatch({
type: types.LOADING_VIDEO_STARTED,
data: {
uri
}
})
lbry.get({ uri }).then(streamInfo => {
2017-05-02 09:07:13 +02:00
const timeout = streamInfo === null ||
typeof streamInfo !== 'object' ||
streamInfo.error == 'Timeout'
if(timeout) {
dispatch({
type: types.LOADING_VIDEO_FAILED,
data: { uri }
})
dispatch(doOpenModal('timedOut'))
} else {
dispatch(doDownloadFile(uri, streamInfo))
}
})
}
}
export function doPurchaseUri(uri) {
return function(dispatch, getState) {
const state = getState()
const balance = selectBalance(state)
2017-05-15 05:50:59 +02:00
const fileInfo = selectFileInfoForUri(state, { uri })
const costInfo = selectCostInfoForUri(state, { uri })
const downloadingByUri = selectUrisDownloading(state)
const alreadyDownloading = !!downloadingByUri[uri]
const { cost } = costInfo
// BUG if you delete a file from the file system system you're going to be
// asked to pay for it again. We need to check if the file is in the blobs
// here and then dispatch doLoadVideo() which will reconstruct it again from
// the blobs. Or perhaps there's another way to see if a file was already
// purchased?
// we already fully downloaded the file. If completed is true but
// writtenBytes is false then we downloaded it before but deleted it again,
// which means it needs to be reconstructed from the blobs by dispatching
// doLoadVideo.
if (fileInfo && fileInfo.completed && !!fileInfo.writtenBytes) {
2017-04-27 09:05:41 +02:00
return Promise.resolve()
}
// we are already downloading the file
if (alreadyDownloading) {
return Promise.resolve()
}
2017-04-28 17:14:44 +02:00
// the file is free or we have partially downloaded it
2017-05-15 05:50:59 +02:00
if (cost <= 0.01 || (fileInfo && fileInfo.download_directory)) {
dispatch(doLoadVideo(uri))
return Promise.resolve()
}
2017-04-27 09:05:41 +02:00
if (cost > balance) {
dispatch(doOpenModal('notEnoughCredits'))
} else {
2017-04-27 09:05:41 +02:00
dispatch(doOpenModal('affirmPurchase'))
}
2017-04-27 09:05:41 +02:00
return Promise.resolve()
}
}
2017-05-13 00:50:51 +02:00
2017-05-15 05:50:59 +02:00
export function doFetchClaimsByChannel(uri) {
2017-05-13 00:50:51 +02:00
return function(dispatch, getState) {
dispatch({
type: types.FETCH_CHANNEL_CLAIMS_STARTED,
data: { uri }
})
lbry.resolve({ uri }).then((resolutionInfo) => {
const {
claims_in_channel,
} = resolutionInfo ? resolutionInfo : { claims_in_channel: [] }
dispatch({
type: types.FETCH_CHANNEL_CLAIMS_COMPLETED,
2017-05-13 00:50:51 +02:00
data: {
uri,
claims: claims_in_channel
}
})
})
}
}
export function doClaimListMine() {
return function(dispatch, getState) {
dispatch({
type: types.CLAIM_LIST_MINE_STARTED
})
lbry.claim_list_mine().then((claims) => {
2017-05-13 00:50:51 +02:00
dispatch({
type: types.CLAIM_LIST_MINE_COMPLETED,
2017-05-13 00:50:51 +02:00
data: {
claims
2017-05-13 00:50:51 +02:00
}
})
})
}
}