@sunny-g/cycle-utils v0.2.4
cycle-utils
utilities and higher-order components for transforming Cycle components, sources and sinks
why
The popular utility library Recompose allows you to write simple React components and independently augment them with additional granular and testable layers of additional functionality.
This library provides simple higher-order component factories and other utilities to simplify Cycle.js components and allow you to write your own higher-order component libraries easily.
installation
npm install --save @sunny-g/cycle-utilsusage
import { mapSources } from '@sunny-g/cycle-utils';
// or
import mapSources from '@sunny-g/cycle-utils/es2015/mapSources';api
A higher-order component (HOC) is a function that takes in a Cycle.js component and returns a Cycle.js component.
This pattern makes HOCs composable and allows us to use Ramda's compose or pipe functions to stitch multiple HOCs together into a single, larger HOC.
The following HOCs and utilities are provided by this library:
- Higher-order components:
- Utilities:
mapSources()
mapSources(
  sourceNames,
  sourceMapper
): HigherOrderComponentHOC that applies the sourceMapper to sources before they've been passed into the BaseComponent.
parameters:
- sourceNames: '*' | string | string[]: Sources you want to transform (- '*'if you want to pass the entire- sourcesobject)
- sourceMapper: (Sources | ...Sources[]) => Sources: Transform function to be applied to specified- sources; the returned- Sourcesare merged into the original- Sources
Example:
// adds a fetched property to the `props` object provided by a fictional `props` source
const withNewProps = mapSources(
  [ 'props', 'HTTP' ], (propsSource, HTTP) => {
    const newProp = HTTP
      .select('newProp')
      .flatten();
    const newPropsSource = combine(propsSource, newProp)
      .map(([ props, newProp ]) => ({
        ...props,
        'newProp': newProp,
      });
    return { props: newPropsSource };
  }
);
const ComponentWithProps = withNewProps(Component);mapSinks()
mapSinks(
  sinkNames,
  sinkMapper
): HigherOrderComponentHOC that applies the sinkMapper to sinks after they've been returned from the BaseComponent.
parameters:
- sinkNames: '*' | string | string[]: Sources you want to transform (- '*'if you want to pass the entire- sinksobject)
- sinkMapper: (Sinks | ...Sinks[]) => Sinks: Transform function to be applied to specified- sinks; the returned- Sinksare merged into the original- Sinks
Example:
// logs all emitted HTTP requests
const logHTTPSink = mapSinks(
  'HTTP', (HTTPSink) => ({
    HTTP: HTTPSink.debug('making an HTTP request'),
  })
);
const ComponentWithLoggedHTTPSink = logHTTPSink(Component);mapSinksWithSources()
mapSourcesAndSink(
  sinkNames,
  sourceNames,
  sinkAndSourceMapper
): HigherOrderComponentHOC to transform a component's sinks with any desired sources after the sinks have been returned from the BaseComponent.
parameters:
- sinkNames: '*' | string | string[]: Sinks you want to transform (- '*'if you want the entire- sinksobject)
- sourceNames: '*' | string | string[]: Sources you want to transform (- '*'if you want the entire- sourcesobject)
- sinkAndSourceMapper: (Sinks | ...Sinks[], Sources | ...Sources[]) => Sinks: Transform function to be applied to specified- sinksand- sources; the returned- Sinksare merged into the original- Sinks
Example:
// logs all emitted sinks values with the current props
const logAllSinks = mapSinksWithSources(
  '*', 'props', (sinks, propsSource) => {
    return Object.keys(sinks)
      .reduce((newSinks, sinkName) => ({
        ...newSinks,
        [sinkName]: sinks[sinkName]
          .compose(sampleCombine(propsSource))
          .debug('new sink emission with current props')
      }), {});
  }
);
const ComponentWithLoggedSinksWithProps = logAllSinks(Component);mapSourcesAndSinks()
mapSourcesAndSink(
  sourceNames,
  sourceMapper,
  sinkNames,
  sinkMapper
): HigherOrderComponentHOC to transform both a component's sources before entering the BaseComponent and a component's sinksafter they've been returned from the BaseComponent.
Uses mapSources and mapSinksWithSources under the hood, so the same requirements of those functions apply.
parameters:
- sourceNames: '*' | string | string[]: Sources you want to transform (- '*'if you want the entire- sourcesobject)
- sourceMapper: (Sources | ...Sources[]) => Sources: Transform function to be applied to specified- sources; the returned- Sourcesare merged into the original- Sources
- sinkNames: '*' | string | string[]: Sources you want to transform (- '*'if you want the entire- sinksobject)
- sinkMapper: (Sinks | ...Sinks[], Sources) => Sinks: Transform function to be applied to specified- sinksas well as the entire- sourcesobject; the returned- Sinksare merged into the original- Sinks
Example:
// TODO: ADD AN EXAMPLE HEREisolate()
isolate(
  config: ((Sources: any) => null | string | {}) | null | string | {}
): HigherOrderComponentHOC version of @cycle/isolate.
parameters:
- config: ((Sources: any) => null | string | {}) | null | string | {}: Either- null, a- string, or an- object, or a function that is given- sourcesand returns- null, a- string, or an- object
Example:
// any component wrapped with randomIsolation will receive a randomly-generated scope
const randomIsolation = isolate(() => Math.random().toString());
const NewComponent = randomIsolation(Component);combineSinks()
combineSinks(
  combiners: { [sinkName: string]: sinkCombiner }
): SinksCombinerUtility to declaratively combine multiple Sinks objects
parameters:
- combiners: { [sinkName: string]: sinkCombiner }: Object of- sinkCombiners for each- sinkNameto combine- each individual sinkCombinerhas the signature(...sink) => sinkand is given eachSinkfrom each passed-inSinksobject and should return a combinedSinkstream
- NOTE: if the sinkCombinerfor a givensinkNameis missing and there are multiplesinksof thatsinkName, thesink's nativemergefunction is applied to the list ofsinks
 
- each individual 
returns:
- SinksCombiner: (...Sinks[]) => Sinks: A function to apply to multiple- Sinksobjects that:- groups all Sinks of the samesinkNameinto an array
- applies each individual sinkCombinerto the destructured array of theSinks
- creates and returns a new Sinksobject from the combinedSinks.
 
- groups all 
Example:
// say we have multiple non-identical children component sinks
// as well as the component's own sinks...
// has only a DOM sink
const mainSinks = { DOM: ... };
// each has both DOM and HTTP sinks
const childOneSinks = childOne(sources);
const childTwoSinks = childTwo(sources);
const sinkCombiner = combineSinks({
  // some drivers only require a merge of their sinks
    // combineSinks merges by default, but is shown here
    // note that there is no placeholder argument for missing sinks (mainSinks has no HTTP sink)
  HTTP: (childOneHTTP, childTwoHTTP) => xs.merge(childOneHTTP, childTwoHTTP),
  // some sinks require custom merging/combining...
  DOM: (mainDOM$, childOneDOM$, childTwoDOM$) => xs
    .combine(mainDOM$, childOneDOM$, childTwoDOM$)
    .map(([ mainDOM, childOneDOM, childTwoDOM ]) =>
      div([
        mainDOM,
        div([
          childOneDOM,
          childTwoDOM,
        ])
      ])
    ),
});
return sinkCombiner(mainSinks, childOneSinks, childTwoSinks);combineCycles()
combineCycles(
  combiners: { [sinkName: string]: sinkCombiner },
  ...BaseComponents: Component[]
): CombinedComponentUtility to declaratively combine multiple Cycle components into a single Cycle component
parameters:
- combiners: { [sinkName: string]: sinkCombiner }: Object of- sinkCombiners for each- sinkNameto combine- each individual sinkCombinerhas the signature(...sinks) => sinkand is given eachSinkfrom each passed-inSinksobject and should return a combinedSinkstream
- NOTE: if the sinkCombinerfor a givensinkNameis missing and there are multiplesinksof thatsinkName, the firstsink's nativemergeoperator is applied to the list ofsinks
 
- each individual 
- ...BaseComponents: Component[]: The desired Cycle.js components to combine into a single component
returns:
- CombinedComponent: (Sources | ...Sources[]) => Sinks: An otherwise normal Cycle.js component that differs from traditional components in that it takes in either:- a single Sourcesobject to be passed into eachBaseComponent
- an list of Sourcesobjects, each one passed into the correspondingBaseComponentby index
 
- a single 
Example:
// assume the same `childOne` and `childTwo` components from the previous example
const ChildrenComponent = combineCycles({
  HTTP: (...httpSinks) => xs.merge(...httpSinks),
  DOM: (childOneDOM$, childTwoDOM$) => xs
    .combine(childOneDOM$, childTwoDOM$)
    .map(([ childOneDOM, childTwoDOM ]) =>
      div([
        childOneDOM,
        childTwoDOM,
      ])
    ),
}, childOne, childTwo);
const childrenSinks = ChildrenComponent(sources);Another useful example is for chaining tasks together in a more linear format:
// the first task, makes a request
const task1 = _ => ({
  HTTP: of({
    url: 'https://google.com',
    category: 'req1',
  }),
});
// the second task
// maps the response of the first task's request into the response sink of the second task
const task2 = ({ HTTP }) => ({
  HTTP: HTTP
    .select('req1')
    .flatten()
    .map(res1 => ({
      url: res1.url,
      category: 'req2'
    })),
});
function main(sources) {
  const Task = combineCycles({}, task1, task2);
  const taskSinks = Task(sources);
  // now, we can use the result of the tasks
  const taskRes = sources.HTTP
    .select('req2')
    .flatten();
  // ... the rest of the component, returns main sinks merged with taskSinks...
}contributing
todo
- ensure typings are comprehensive and correct
- explain contribution process
- add more tests :)
- explain why I wrote this
license
ISC