fc77ee76d6
The primary license of the project is changing to: AGPL-3.0-or-later WITH Web-Template-Output-Additional-Permission-3.0-or-later The Additional Permission is designed to permit publicly distributed Javascript code to be relicensed under LGPL-3.0-or-later, but not server-side Javascript code. As such, we've relicensed here static Javscript files under LGPL-3.0-or-later, and those that run as part of build and/or server side under AGPL-3.0-or-later. Note that in future, Javascript files may be updated to be stronger copyleft license with the Additional Permission, particularly if they adapted to run on server side and/or turned into templates. Of course, we'd seek public discussion with the contributor community about such changes. This commit is one of the many steps to relicense the entire codebase. Documentation granting permission for this relicensing (from all past contributors who hold copyrights) is on file with Software Freedom Conservancy, Inc.
166 lines
5.4 KiB
JavaScript
166 lines
5.4 KiB
JavaScript
// License: LGPL-3.0-or-later
|
|
const R = require('ramda')
|
|
const h = require('snabbdom/h')
|
|
const flyd = require('flyd')
|
|
const url$ = require('flyd-url')
|
|
const render = require('ff-core/render')
|
|
const filter = require('flyd/module/filter')
|
|
const snabbdom = require('snabbdom')
|
|
const mergeAll = require('flyd/module/mergeall')
|
|
const sampleOn = require('flyd/module/sampleon')
|
|
const queryString = require('query-string')
|
|
const notification = require('ff-core/notification')
|
|
|
|
const request = require('../../../../common/request')
|
|
const confirm = require('../../../../components/confirmation-modal')
|
|
|
|
const gmail = require('./gmail')
|
|
const actions = require('./supporter-actions')
|
|
const activities = require('./supporter-activities')
|
|
const replyModal = require('./gmail/reply-modal')
|
|
const composeModal = require('./gmail/compose-modal')
|
|
const offsiteDonationForm = require('./offsite-donation-form')
|
|
const supporterNoteForm = require('./supporter-note-form')
|
|
|
|
const flatMap = R.curry(require('flyd/module/flatmap'))
|
|
|
|
const init = _ => {
|
|
var state = {
|
|
clickComposing$: flyd.stream()
|
|
, threadId$: flyd.stream()
|
|
, newNote$: flyd.stream()
|
|
, editNote$: flyd.stream()
|
|
, deleteNote$: flyd.stream()
|
|
}
|
|
|
|
const supporterID$ = R.compose(
|
|
filter(Boolean )
|
|
, flyd.map(url => queryString.parse(url.search).sid)
|
|
)(url$)
|
|
|
|
state.pathPrefix$ = flyd.map(constructPathPrefix, supporterID$)
|
|
|
|
const supporterPath$ = flyd.map(id => `/nonprofits/${app.nonprofit_id}/supporters/${id}`, supporterID$)
|
|
|
|
const supporterResp$ = R.compose(
|
|
flyd.map(x => x.body.data)
|
|
, filter(x => x.status === 200)
|
|
, flatMap(path => request({method: 'get', path}).load)
|
|
)(supporterPath$)
|
|
|
|
state.supporter$ = flyd.merge(supporterResp$, flyd.stream({}))
|
|
|
|
state.composeOrReply$ = flyd.merge(state.clickComposing$, state.threadId$)
|
|
|
|
state.gmail = gmail.init(state)
|
|
|
|
state.offsiteDonationForm = offsiteDonationForm.init(state)
|
|
|
|
state.editNoteData$ = flyd.merge(
|
|
flyd.map(R.always({}), state.newNote$)
|
|
, flyd.map(d => ({id: d.attachment_id, content: d.json_data.content}), state.editNote$))
|
|
|
|
const deleteNoteId$ = flyd.map(d => d.attachment_id, state.deleteNote$)
|
|
|
|
state.noteAjaxMethod$ = mergeAll([
|
|
flyd.map(R.always('post'), state.newNote$)
|
|
, flyd.map(R.always('put'), state.editNote$)
|
|
])
|
|
|
|
state.supporterNoteForm = supporterNoteForm.init(state)
|
|
|
|
state.confirmDelete = confirm.init(deleteNoteId$)
|
|
|
|
const deleteNoteResp$ = flatMap(ajaxDeleteNote(supporterPath$, deleteNoteId$), state.confirmDelete.confirm$)
|
|
|
|
// All streams that we want to trigger a refresh of the supporter timeline
|
|
const fetchActivitiesWith$ = mergeAll([
|
|
state.pathPrefix$
|
|
, state.gmail.saveResult$
|
|
, state.offsiteDonationForm.saved$
|
|
, state.supporterNoteForm.saved$
|
|
, deleteNoteResp$
|
|
])
|
|
|
|
// Stream of activities data, using the pathPrefix$ stream, triggered by fetchActivitiesWith$
|
|
state.activities$ = R.compose(
|
|
R.curryN(2, flatMap)(getActivities)
|
|
, sampleOn(R.__, state.pathPrefix$)
|
|
)(fetchActivitiesWith$)
|
|
|
|
state.activities = activities.init(state)
|
|
|
|
state.modalID$ = mergeAll([
|
|
flyd.map(()=> 'composeGmailModal', state.clickComposing$)
|
|
, flyd.map(()=> 'replyGmailModal', state.threadId$)
|
|
, flyd.map(()=> 'newSupporterNoteModal', state.editNoteData$)
|
|
, flyd.map(()=> null, state.gmail.sendResponse$)
|
|
, flyd.map(()=> null, state.offsiteDonationForm.saved$)
|
|
, flyd.map(()=> null, state.supporterNoteForm.saved$)
|
|
])
|
|
|
|
state.loading$ = mergeAll([
|
|
flyd.map(R.always(true), state.gmail.composeForm.validData$)
|
|
, flyd.map(R.always(true), state.gmail.replyForm.validData$)
|
|
, flyd.map(R.always(false), state.gmail.sendResponse$)
|
|
])
|
|
|
|
|
|
const message$ = mergeAll([
|
|
flyd.map(()=> 'Email sent', state.gmail.sendResponse$)
|
|
, flyd.map(()=> 'Successfully created a new offsite contribution', state.offsiteDonationForm.saved$)
|
|
, flyd.map(()=> `Successfully ${noteMsg(state.noteAjaxMethod$)} supporter note`, state.supporterNoteForm.saved$)
|
|
, flyd.map(()=> 'Successfully deleted supporter note', deleteNoteResp$)
|
|
])
|
|
|
|
state.notification = notification.init({message$})
|
|
|
|
window.state = state
|
|
return state
|
|
}
|
|
|
|
const ajaxDeleteNote = (pathPrefix$, id$) => () => {
|
|
const path = `${pathPrefix$()}/supporter_notes/${id$()}`
|
|
return request({
|
|
method: 'delete'
|
|
, path
|
|
}).load
|
|
}
|
|
|
|
const noteMsg = method$ => {
|
|
if(method$() === 'put') return 'edited'
|
|
if(method$() === 'post') return 'created a new'
|
|
}
|
|
|
|
const getActivities = path =>
|
|
flyd.map(req => req.body, request({path: path + 'activities', method: 'get'}).load)
|
|
|
|
const constructPathPrefix = sid => `/nonprofits/${app.nonprofit_id}/supporters/${sid}/`
|
|
|
|
const view = state => {
|
|
return h('div', [
|
|
actions.view(state)
|
|
, activities.view(state)
|
|
, composeModal(state)
|
|
, notification.view(state.notification)
|
|
, offsiteDonationForm.view(R.merge(state.offsiteDonationForm, {modalID$: state.modalID$}))
|
|
, supporterNoteForm.view(R.merge(state.supporterNoteForm, {modalID$: state.modalID$}))
|
|
, replyModal(state)
|
|
, confirm.view(state.confirmDelete, 'Are you sure you want to delete this note?')
|
|
])
|
|
}
|
|
|
|
var container = document.querySelector('#js-sidePanel')
|
|
|
|
// -- Render to the page
|
|
// render takes state, view function, patch function, and DOM container
|
|
const patch = snabbdom.init([
|
|
require('snabbdom/modules/eventlisteners')
|
|
, require('snabbdom/modules/class')
|
|
, require('snabbdom/modules/props')
|
|
, require('snabbdom/modules/attributes')
|
|
, require('snabbdom/modules/style')
|
|
])
|
|
|
|
render({ patch, container , view, state: init() })
|
|
|