├── .babelrc
├── .gitignore
├── README.md
├── build
├── reactive-react.es.js
└── reactive-react.umd.js
├── demos
├── components
│ ├── blink.js
│ ├── button.css
│ ├── counter.js
│ ├── crappybird.js
│ ├── index.js
│ ├── sourceswitching.js
│ └── timer.js
├── custom.css
├── eleventy.css
├── index.html
├── index.js
└── time-slicing
│ ├── Charts.js
│ ├── Clock.js
│ ├── README.md
│ ├── index.css
│ ├── index.html
│ └── index.js
├── dependencygraph.svg
├── netlify.toml
├── package.json
├── prototypeAPI.js
├── reactive-react
├── component.js
├── element.js
├── index.js
├── reconciler.js
├── scheduler.js
├── swyxjs.js
└── updateProperties.js
└── rollup.config.js
/.babelrc:
--------------------------------------------------------------------------------
1 | {
2 | "plugins": [
3 | "transform-object-rest-spread",
4 | "transform-class-properties",
5 | [
6 | "transform-react-jsx",
7 | {}
8 | ]
9 | ],
10 | "presets": [
11 | [
12 | "env",
13 | {
14 | "targets": {
15 | "node": "current"
16 | }
17 | }
18 | ]
19 | ]
20 | }
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules
2 | dist
3 | yarn.lock
4 | .cache
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | This is a prototype mockup of a "Reactive" version of React. Do not use unless you are swyx.
2 |
3 | ## Watch the React Rally talk
4 |
5 | Talk video: https://www.youtube.com/watch?v=nyFHR0dDZo0
6 |
7 | TL;DR with writeup: https://www.swyx.io/ReactRally
8 |
9 | ## Description
10 |
11 | **Note: we are aware of the double subscribe bug when a source is switched. didnt have time to figure out the fix before react rally. Blink tag example has a hacky patch for this.**
12 |
13 | In this alternate universe, Observables became a part of Javascript.
14 |
15 | We take a minimal implementation of Observables, zen-observable.
16 |
17 |
18 | # Try it out
19 |
20 | `yarn start` to run the demo locally
21 |
22 | # The `reactive-react` API
23 |
24 | The React API we are targeting looks something like this (see `/demos` for actual examples):
25 |
26 | ```js
27 | class Counter extends Component {
28 | // demonstrate basic counter
29 | initialState = 0
30 | increment = createHandler(e => 1)
31 | decrement = createHandler(e => -1)
32 | source($) {
33 | const source = merge(this.increment.$, this.decrement.$)
34 | const reducer = (acc, n) => acc + n
35 | return {source, reducer}
36 | }
37 | render(state, stateMap) {
38 | const {name = "counter"} = this.props
39 | return
40 | {name}: {state}
41 |
42 |
43 |
44 | }
45 | }
46 |
47 | // taking info from event handler
48 | class Echo extends Component {
49 | handler = createHandler(e => e.target.value)
50 | initialState = 'hello world'
51 | source($) {
52 | const source = this.handler.$
53 | const reducer = (acc, n) => n
54 | return {source, reducer}
55 | }
56 | render(state, prevState) {
57 | return
58 |
59 | {state}
60 |
61 | }
62 | }
63 |
64 | class Timer extends Component {
65 | // demonstrate interval time
66 | initialState = 0
67 | source($) {
68 | const reducer = x => x + 1 // count up
69 | const source = Interval(this.props.ms) // tick every second
70 | // source returns an observable
71 | return scan(source, reducer, 0) // from zero
72 | }
73 | render(state, stateMap) {
74 | return number of seconds elapsed: {state}
75 | }
76 | }
77 |
78 | class Blink extends Component {
79 | // more fun time demo
80 | initialState = true
81 | source($) {
82 | const reducer = x => !x
83 | // tick every ms milliseconds
84 | const source = Interval(this.props.ms)
85 | // source can also return an observable
86 | return scan(source, reducer, true)
87 | }
88 | render(state) {
89 | const style = {display: state ? 'block' : 'none'}
90 | return Bring back the blink tag!
91 | }
92 | }
93 |
94 | class CrappyBird extends Component {
95 | // merging time and counter
96 | initialState = {
97 | input: 50,
98 | target: 50
99 | }
100 | increment = createHandler(e => 3)
101 | source($) {
102 | return this.combineReducer({
103 | input: () => {
104 | const source = merge(this.increment.$, Interval(200,-1))
105 | const reducer = (acc, x) => Math.max(0,acc + x)
106 | return {source, reducer}
107 | },
108 | target: () => {
109 | const source = Interval(200)
110 | const reducer = (acc) => {
111 | const int = acc + Math.random() * 8 - 4
112 | return int - (int-50)/30 // bias toward 50
113 | }
114 | return {source, reducer}
115 | }
116 | })
117 | }
118 | render(state, stateMap) {
119 | const {input, target} = state
120 | return
121 |
122 |
Crappy bird
123 |
Bird:
124 |
Target:
125 |
126 | }
127 | }
128 |
129 | function Counters() {
130 | // demonstrate independent states
131 | return
132 |
133 |
134 |
135 | }
136 | function Source() {
137 | // demonstrate ability to switch sources
138 | return }
140 | right={}
141 | />
142 | }
143 |
144 | class SourceSwitching extends Component {
145 | initialState = true
146 | toggle = createHandler()
147 | source($) {
148 | const source = this.toggle.$
149 | const reducer = x => !x
150 | return {source, reducer}
151 | }
152 | render(state, stateMap) {
153 | return
154 |
155 | {
156 | state ? this.props.left : this.props.right
157 | }
158 |
159 | }
160 | }
161 |
162 | ```
163 |
164 | # Local development
165 |
166 | `yarn run build` and then `npm publish` (but its under my namespace @swyx/reactive-react cos someone else has the generic one)
167 |
168 | Here is the project structure:
169 |
170 | 
171 |
--------------------------------------------------------------------------------
/build/reactive-react.es.js:
--------------------------------------------------------------------------------
1 | import Observable from 'zen-observable';
2 | import { createChangeEmitter } from 'change-emitter';
3 | import { merge } from 'zen-observable/extras';
4 | import h from 'virtual-dom/h';
5 | import VText from 'virtual-dom/vnode/vtext';
6 | import diff from 'virtual-dom/diff';
7 | import patch from 'virtual-dom/patch';
8 | import createElement from 'virtual-dom/create-element';
9 |
10 | var TEXT_ELEMENT = "TEXT ELEMENT";
11 |
12 | function createElement$1(type, config) {
13 | var _ref;
14 |
15 | var props = Object.assign({}, config);
16 |
17 | for (var _len = arguments.length, args = Array(_len > 2 ? _len - 2 : 0), _key = 2; _key < _len; _key++) {
18 | args[_key - 2] = arguments[_key];
19 | }
20 |
21 | var hasChildren = args.length > 0;
22 | var rawChildren = hasChildren ? (_ref = []).concat.apply(_ref, args) : [];
23 | props.children = rawChildren.filter(function (c) {
24 | return c != null && c !== false;
25 | }).map(function (c) {
26 | return c instanceof Object ? c : createTextElement(c);
27 | });
28 | return { type: type, props: props };
29 | }
30 |
31 | function createTextElement(value) {
32 | return createElement$1(TEXT_ELEMENT, { nodeValue: value });
33 | }
34 |
35 | function createHandler(_fn) {
36 | var emitter = createChangeEmitter();
37 | var handler = function handler(x) {
38 | emitter.emit(x);
39 | };
40 | handler.$ = new Observable(function (observer) {
41 | return emitter.listen(function (value) {
42 | observer.next(_fn ? _fn(value) : value);
43 | });
44 | });
45 | return handler;
46 | }
47 |
48 | var NOINIT = Symbol('NO_INITIAL_VALUE');
49 | function scan(obs, cb) {
50 | var seed = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : NOINIT;
51 |
52 | var sub = void 0,
53 | acc = seed,
54 | hasValue = false;
55 | var hasSeed = acc !== NOINIT;
56 | return new Observable(function (observer) {
57 | sub = obs.subscribe(function (value) {
58 | if (observer.closed) return;
59 | var first = !hasValue;
60 | hasValue = true;
61 | if (!first || hasSeed) {
62 | try {
63 | acc = cb(acc, value);
64 | } catch (e) {
65 | return observer.error(e);
66 | }
67 | observer.next(acc);
68 | } else {
69 | acc = value;
70 | }
71 | });
72 | return sub;
73 | });
74 | }
75 |
76 | // Flatten a collection of observables and only output the newest from each
77 |
78 |
79 |
80 |
81 | function startWith(obs, val) {
82 | return new Observable(function (observer) {
83 | observer.next(val); // immediately output this value
84 | var handler = obs.subscribe(function (x) {
85 | return observer.next(x);
86 | });
87 | return function () {
88 | return handler();
89 | };
90 | });
91 | }
92 |
93 | var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; };
94 |
95 | var _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"]) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError("Invalid attempt to destructure non-iterable instance"); } }; }();
96 |
97 | var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
98 |
99 | function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }
100 |
101 | function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
102 |
103 | function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
104 |
105 | // import { reconcile } from "./reconciler";
106 | var Component = function () {
107 | function Component(props) {
108 | _classCallCheck(this, Component);
109 |
110 | this.props = props;
111 | this.state = this.state || {};
112 | }
113 |
114 | // setState(partialState) {
115 | // this.state = Object.assign({}, this.state, partialState);
116 | // updateInstance(this.__internalInstance);
117 | // }
118 |
119 | // class method because it feeds in this.initialState
120 |
121 |
122 | _createClass(Component, [{
123 | key: 'combineReducer',
124 | value: function combineReducer(obj) {
125 | var _this = this;
126 |
127 | var sources = Object.entries(obj).map(function (_ref) {
128 | var _ref2 = _slicedToArray(_ref, 2),
129 | k = _ref2[0],
130 | fn = _ref2[1];
131 |
132 | var subReducer = fn(obj);
133 | // there are two forms of return the subreducer can have
134 | // straight stream form
135 | // or object form where we need to scan it into string
136 | if (subReducer.source && subReducer.reducer) {
137 | // object form
138 | subReducer = scan(subReducer.source, subReducer.reducer || function (_, n) {
139 | return n;
140 | }, _this.initialState[k]);
141 | }
142 | return subReducer.map(function (x) {
143 | return _defineProperty({}, k, x);
144 | }); // map to its particular namespace
145 | });
146 | var source = merge.apply(undefined, _toConsumableArray(sources));
147 | var reducer = function reducer(acc, n) {
148 | return _extends({}, acc, n);
149 | };
150 | return { source: source, reducer: reducer };
151 | }
152 | }]);
153 |
154 | return Component;
155 | }();
156 |
157 | // function updateInstance(internalInstance) {
158 | // const parentDom = internalInstance.dom.parentNode;
159 | // const element = internalInstance.element;
160 | // reconcile(parentDom, internalInstance, element);
161 | // }
162 |
163 | function createPublicInstance(element /*, internalInstance*/) {
164 | var type = element.type,
165 | props = element.props;
166 |
167 | var publicInstance = new type(props);
168 | // publicInstance.__internalInstance = internalInstance;
169 | return publicInstance;
170 | }
171 |
172 | var _slicedToArray$1 = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"]) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError("Invalid attempt to destructure non-iterable instance"); } }; }();
173 |
174 | function _objectWithoutProperties(obj, keys) { var target = {}; for (var i in obj) { if (keys.indexOf(i) >= 0) continue; if (!Object.prototype.hasOwnProperty.call(obj, i)) continue; target[i] = obj[i]; } return target; }
175 |
176 | // import { updateDomProperties } from "./updateProperties";
177 | // import VNode from "virtual-dom/vnode/vnode"
178 | // const circuitBreakerflag = false // set true to enable debugger in infinite loops
179 | // let circuitBreaker = -50
180 | // traverse all children and collect a stream of all sources
181 | // AND render. a bit of duplication, but we get persistent instances which is good
182 | function renderStream(element, instance, state, stateMap) {
183 | // this is a separate function because scope gets messy when being recursive
184 | var isNewStream = false; // assume no stream switching by default
185 | // this is the first ping of data throughout the app
186 | var source = Observable.of(state);
187 | var addToStream = function addToStream(_source) {
188 | // visit each source and merge with source
189 | if (_source) return source = merge(source, _source);
190 | };
191 | var markNewStream = function markNewStream() {
192 | return isNewStream = true;
193 | };
194 | var newInstance = render(source, addToStream, markNewStream)(element, instance, state, stateMap);
195 | return { source: source, instance: newInstance, isNewStream: isNewStream };
196 | }
197 |
198 | /** core render logic */
199 | function render(source, addToStream, markNewStream) {
200 | // this is the nonrecursive part
201 | return function renderWithStream(element, instance, state, stateMap) {
202 | // recursive part
203 | var newInstance = void 0;
204 | var type = element.type,
205 | props = element.props;
206 |
207 |
208 | var isDomElement = typeof type === "string";
209 | // if (circuitBreakerflag && circuitBreaker++ > 0) debugger
210 |
211 | var _props$children = props.children,
212 | children = _props$children === undefined ? [] : _props$children,
213 | rest = _objectWithoutProperties(props, ['children']);
214 |
215 | if (isDomElement) {
216 | var childInstances = children.map(function (el, i) {
217 | // ugly but necessary to allow functional children
218 | // mapping element's children to instance's childInstances
219 | var _childInstances = instance && (instance.childInstance || instance.childInstances[i]);
220 | return renderWithStream( // recursion
221 | el, _childInstances, state, stateMap);
222 | });
223 | var childDoms = childInstances.map(function (childInstance) {
224 | return childInstance.dom;
225 | });
226 | var lcaseProps = {};
227 | Object.entries(rest).forEach(function (_ref) {
228 | var _ref2 = _slicedToArray$1(_ref, 2),
229 | k = _ref2[0],
230 | v = _ref2[1];
231 |
232 | return lcaseProps[formatProps(k)] = v;
233 | });
234 | var dom = type === TEXT_ELEMENT ? new VText(props.nodeValue) : h(type, lcaseProps, childDoms); // equivalent of appendchild
235 | newInstance = { dom: dom, element: element, childInstances: childInstances };
236 | } else {
237 | // component element
238 | var publicInstance = void 0;
239 | // debugger
240 | if (instance && instance.publicInstance && instance.element === element) {
241 | // might have to do more diffing of props
242 | // just reuse old instance if it already exists
243 | publicInstance = instance && instance.publicInstance;
244 | } else {
245 | markNewStream(); // mark as dirty in parent scope; will rerender
246 | publicInstance = createPublicInstance(element);
247 | }
248 | var localState = stateMap.get(publicInstance);
249 | if (localState === undefined) localState = publicInstance.initialState;
250 | publicInstance.state = localState; // for access with this.state
251 | if (Object.keys(rest).length) publicInstance.props = rest; // update with new props // TODO: potentially buggy
252 | // console.log({rest})
253 | if (publicInstance.source) {
254 | var src = publicInstance.source(source);
255 | // there are two forms of Component.source
256 | var src$ = src.reducer && publicInstance.initialState !== undefined ?
257 | // 1. the reducer form
258 | scan(src.source, src.reducer, publicInstance.initialState) :
259 | // 2. and raw stream form
260 | src;
261 | addToStream(src$.map(function (event) {
262 | stateMap.set(publicInstance, event);
263 | return { instance: publicInstance, event: event // tag it to the instance
264 | };
265 | }));
266 | }
267 | var childElement = publicInstance.render ? publicInstance.render(localState, stateMap) : publicInstance;
268 |
269 | var childInstance = renderWithStream(childElement, instance && instance.childInstance, state, stateMap);
270 | var _dom = childInstance.dom;
271 | newInstance = { dom: _dom, element: element, childInstance: childInstance, publicInstance: publicInstance };
272 | }
273 | return newInstance;
274 | };
275 | }
276 |
277 | function formatProps(k) {
278 | if (k.startsWith('on')) return k.toLowerCase();
279 | return k;
280 | }
281 |
282 | var stateMapPointer = new Map();
283 |
284 | var emitter = createChangeEmitter();
285 | // single UI thread; this is the observable that sticks around and swaps out source
286 | var UIthread = new Observable(function (observer) {
287 | emitter.listen(function (x) {
288 | // debugger // success! thread switching!
289 | observer.next(x);
290 | });
291 | });
292 | // mount the vdom on to the dom and
293 | // set up the runtime from sources and
294 | // patch the vdom
295 | // ---
296 | // returns an unsubscribe method you can use to unmount
297 | function mount(rootElement, container) {
298 | // initial, throwaway-ish frame
299 | var _renderStream = renderStream(rootElement, {}, undefined, stateMapPointer),
300 | source = _renderStream.source,
301 | instance = _renderStream.instance;
302 |
303 | var instancePointer = instance;
304 | var rootNode = createElement(instance.dom);
305 | var containerChild = container.firstElementChild;
306 | if (containerChild) {
307 | container.replaceChild(rootNode, containerChild); // hot reloaded mount
308 | } else {
309 | container.appendChild(rootNode); // initial mount
310 | }
311 | var currentSrc$ = null;
312 | var SoS = startWith(UIthread, source); // stream of streams
313 | return SoS.subscribe(function (src$) {
314 | // this is the current sourceStream we are working with
315 | if (currentSrc$) console.log('unsub!') || currentSrc$.unsubscribe(); // unsub from old stream
316 | /**** main */
317 | var source2$ = scan(src$, function (_ref, nextState) {
318 | var instance = _ref.instance,
319 | stateMap = _ref.stateMap;
320 |
321 | var streamOutput = renderStream(rootElement, instance, nextState, stateMap);
322 | if (streamOutput.isNewStream) {
323 | // quick check
324 | var nextSource$ = streamOutput.source;
325 | // debugger
326 | instancePointer = streamOutput.instance;
327 | patch(rootNode, diff(instance.dom, instancePointer.dom)); // render to screen
328 | emitter.emit(nextSource$); // update the UI thread; source will switch
329 | } else {
330 | var nextinstance = streamOutput.instance;
331 | patch(rootNode, diff(instance.dom, nextinstance.dom)); // render to screen
332 | return { instance: nextinstance, stateMap: stateMap };
333 | }
334 | }, { instance: instancePointer, stateMap: stateMapPointer // accumulator
335 | });
336 | /**** end main */
337 | currentSrc$ = source2$.subscribe();
338 | });
339 | }
340 |
341 | var index = {
342 | renderStream: renderStream,
343 | createElement: createElement$1,
344 | createHandler: createHandler,
345 | Component: Component,
346 | mount: mount
347 | };
348 |
349 | export { createElement$1 as createElement, createHandler, Component, renderStream, mount };export default index;
350 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"reactive-react.es.js","sources":["../reactive-react/element.js","../reactive-react/swyxjs.js","../reactive-react/component.js","../reactive-react/reconciler.js","../reactive-react/scheduler.js","../reactive-react/index.js"],"sourcesContent":["import Observable from 'zen-observable'\nimport { createChangeEmitter } from 'change-emitter'\n\nexport const TEXT_ELEMENT = \"TEXT ELEMENT\";\n\nexport function createElement(type, config, ...args) {\n  const props = Object.assign({}, config);\n  const hasChildren = args.length > 0;\n  const rawChildren = hasChildren ? [].concat(...args) : [];\n  props.children = rawChildren\n    .filter(c => c != null && c !== false)\n    .map(c => c instanceof Object ? c : createTextElement(c));\n  return { type, props };\n}\n\nfunction createTextElement(value) {\n  return createElement(TEXT_ELEMENT, { nodeValue: value });\n}\n\nexport function createHandler(_fn) {\n  const emitter = createChangeEmitter()\n  let handler = x => {\n    emitter.emit(x)\n  }\n  handler.$ = new Observable(observer => {\n    return emitter.listen(value => {\n      observer.next(_fn ? _fn(value) : value)\n    }\n    )\n  })\n  return handler\n}","import Observable from 'zen-observable'\nexport { merge, combineLatest, zip } from 'zen-observable/extras'\n\nexport function Interval(tick = 1000, tickData = Symbol('tick')) {\n  return new Observable(observer => {\n    let timer = () => setTimeout(() => {\n      if (typeof tickData === 'function') tickData = tickData()\n      observer.next(tickData);\n      timer()\n      // observer.complete();\n    }, tick);\n    timer()\n  \n    // On unsubscription, cancel the timer\n    return () => clearTimeout(timer);\n\n  })\n}\n\nexport function fromEvent(el, eventType) {\n  return new Observable(observer => {\n    const handler = e => observer.next(e)\n    el.addEventListener(eventType, handler)\n    // on unsub, remove event listener\n    return () => el.removeEventListener(eventType, handler)\n  })\n}\n\nconst NOINIT = Symbol('NO_INITIAL_VALUE')\nexport function scan(obs, cb, seed = NOINIT) {\n  let sub, acc = seed, hasValue = false\n  const hasSeed = acc !== NOINIT\n  return new Observable(observer => {\n    sub = obs.subscribe(value => {\n      if (observer.closed) return\n      let first = !hasValue;\n      hasValue = true\n      if (!first || hasSeed ) {\n        try { acc = cb(acc, value) }\n        catch (e) { return observer.error(e) }\n        observer.next(acc);\n      }\n      else {\n        acc = value\n      }\n    })\n    return sub\n  })\n}\n\n// Flatten a collection of observables and only output the newest from each\nexport function switchLatest(higherObservable) {\n  return new Observable(observer => {\n    let currentObs = null\n    return higherObservable.subscribe({\n      next(obs) {\n        if (currentObs) currentObs.unsubscribe() // unsub and switch\n        currentObs = obs.subscribe(observer.subscribe)\n      },\n      error(e) {\n        observer.error(e) // untested\n      },\n      complete() {\n        // i dont think it should complete?\n        // observer.complete()\n      }\n    })\n  });\n}\n\nexport function mapToConstant(obs, val) {\n  return new Observable(observer => {\n    const handler = obs.subscribe(() => observer.next(val))\n    return handler\n  })\n}\n\nexport function startWith(obs, val) {\n  return new Observable(observer => {\n    observer.next(val) // immediately output this value\n    const handler = obs.subscribe(x => observer.next(x))\n    return () => handler()\n  })\n}","// import { reconcile } from \"./reconciler\";\nimport {Interval, scan, startWith, merge, mapToConstant} from './swyxjs'\n\nexport class Component {\n  constructor(props) {\n    this.props = props;\n    this.state = this.state || {};\n  }\n\n  // setState(partialState) {\n  //   this.state = Object.assign({}, this.state, partialState);\n  //   updateInstance(this.__internalInstance);\n  // }\n\n  // class method because it feeds in this.initialState\n  combineReducer(obj) {\n    const sources = Object.entries(obj).map(([k,fn]) => {\n      let subReducer = fn(obj)\n      // there are two forms of return the subreducer can have\n      // straight stream form\n      // or object form where we need to scan it into string\n      if (subReducer.source && subReducer.reducer) { // object form\n        subReducer = scan(subReducer.source, \n          subReducer.reducer || ((_, n) => n), \n          this.initialState[k]\n        )\n      }\n      return subReducer\n        .map(x => ({[k]: x})) // map to its particular namespace\n    })\n    const source = merge(...sources)\n    const reducer = (acc, n) => ({...acc, ...n})\n    return {source, reducer}\n  }\n}\n\n// function updateInstance(internalInstance) {\n//   const parentDom = internalInstance.dom.parentNode;\n//   const element = internalInstance.element;\n//   reconcile(parentDom, internalInstance, element);\n// }\n\nexport function createPublicInstance(element/*, internalInstance*/) {\n  const { type, props } = element;\n  const publicInstance = new type(props);\n  // publicInstance.__internalInstance = internalInstance;\n  return publicInstance;\n}\n","import Observable from 'zen-observable'\nimport {fromEvent, scan, merge, startWith, switchLatest} from './swyxjs'\n// import { updateDomProperties } from \"./updateProperties\";\nimport { TEXT_ELEMENT } from \"./element\";\nimport { createPublicInstance } from \"./component\";\nimport h from 'virtual-dom/h'\n// import VNode from \"virtual-dom/vnode/vnode\"\nimport VText from \"virtual-dom/vnode/vtext\"\n\n// const circuitBreakerflag = false // set true to enable debugger in infinite loops\n// let circuitBreaker = -50\n// traverse all children and collect a stream of all sources\n// AND render. a bit of duplication, but we get persistent instances which is good\nexport function renderStream(element, instance, state, stateMap) {\n  // this is a separate function because scope gets messy when being recursive\n  let isNewStream = false // assume no stream switching by default\n  // this is the first ping of data throughout the app\n  let source = Observable.of(state) \n  const addToStream = _source => {\n    // visit each source and merge with source\n    if (_source) return source = merge(source, _source)\n  }\n  const markNewStream = () => isNewStream = true\n  const newInstance = render(source, addToStream, markNewStream)(element, instance, state, stateMap)\n  return {source, instance: newInstance, isNewStream}\n}\n\n/** core render logic */\nexport function render(source, addToStream, markNewStream) { // this is the nonrecursive part\n  return function renderWithStream(element, instance, state, stateMap) { // recursive part\n    let newInstance\n    const { type, props } = element\n  \n    const isDomElement = typeof type === \"string\";\n    // if (circuitBreakerflag && circuitBreaker++ > 0) debugger\n    const {children = [], ...rest} = props\n    if (isDomElement) {\n      const childInstances = children.map(\n        (el, i) => {\n          // ugly but necessary to allow functional children\n          // mapping element's children to instance's childInstances\n          const _childInstances = instance && (instance.childInstance || instance.childInstances[i])\n          return renderWithStream(  // recursion\n            el, \n            _childInstances, \n            state, \n            stateMap) \n        }\n      );\n      const childDoms = childInstances.map(childInstance => childInstance.dom);\n      let lcaseProps = {}\n      Object.entries(rest).forEach(([k, v]) => lcaseProps[formatProps(k)] = v)\n      const dom = type === TEXT_ELEMENT\n        ? new VText(props.nodeValue)\n        : h(type, lcaseProps, childDoms); // equivalent of appendchild\n      newInstance = { dom, element, childInstances };\n    } else { // component element\n      let publicInstance \n      // debugger\n      if (instance && instance.publicInstance && instance.element === element) { // might have to do more diffing of props\n        // just reuse old instance if it already exists\n        publicInstance = instance && instance.publicInstance\n      } else {\n        markNewStream() // mark as dirty in parent scope; will rerender\n        publicInstance = createPublicInstance(element);\n      }\n      let localState = stateMap.get(publicInstance)\n      if (localState === undefined) localState = publicInstance.initialState\n      publicInstance.state = localState // for access with this.state\n      if (Object.keys(rest).length) publicInstance.props = rest // update with new props // TODO: potentially buggy\n      // console.log({rest})\n      if (publicInstance.source) {\n        const src = publicInstance.source(source)\n        // there are two forms of Component.source\n        const src$ = src.reducer && publicInstance.initialState !== undefined ? \n            // 1. the reducer form\n            scan(src.source, src.reducer, publicInstance.initialState) : \n            // 2. and raw stream form\n            src\n        addToStream(src$\n          .map(event => {\n            stateMap.set(publicInstance, event)\n            return {instance: publicInstance, event} // tag it to the instance\n          }) \n        );\n      }\n      const childElement = publicInstance.render ? \n          publicInstance.render(localState, stateMap) : \n          publicInstance;\n\n      const childInstance = renderWithStream(childElement, instance && instance.childInstance, state, stateMap)\n      const dom = childInstance.dom\n      newInstance = { dom, element, childInstance, publicInstance }\n    }\n    return newInstance\n  }\n}\n\nfunction formatProps(k) {\n  if (k.startsWith('on')) return k.toLowerCase()\n  return k\n}","import Observable from 'zen-observable'\nimport {fromEvent, scan, merge, startWith, switchLatest} from './swyxjs'\nimport diff from 'virtual-dom/diff';\nimport patch from 'virtual-dom/patch';\nimport createElement from 'virtual-dom/create-element';\nimport { createChangeEmitter } from 'change-emitter'\nimport { renderStream } from './reconciler'\n\nexport const stateMapPointer = new Map()\n\nlet circuitBreaker = -20\n\nconst emitter = createChangeEmitter()\n// single UI thread; this is the observable that sticks around and swaps out source\nconst UIthread = new Observable(observer => {\n  emitter.listen(x => {\n    // debugger // success! thread switching!\n    observer.next(x)\n  })\n})\n// mount the vdom on to the dom and \n// set up the runtime from sources and\n// patch the vdom\n// ---\n// returns an unsubscribe method you can use to unmount\nexport function mount(rootElement, container) {\n  // initial, throwaway-ish frame\n  let {source, instance} = renderStream(rootElement, {}, undefined, stateMapPointer)\n  let instancePointer = instance\n  const rootNode = createElement(instance.dom)\n  const containerChild = container.firstElementChild\n  if (containerChild) {\n    container.replaceChild(rootNode,containerChild) // hot reloaded mount\n  } else {\n    container.appendChild(rootNode) // initial mount\n  }\n  let currentSrc$ = null\n  let SoS = startWith(UIthread, source) // stream of streams\n  return SoS.subscribe(\n    src$ => { // this is the current sourceStream we are working with\n      if (currentSrc$) console.log('unsub!') || currentSrc$.unsubscribe() // unsub from old stream\n      /**** main */\n      const source2$ = scan(\n        src$, \n        ({instance, stateMap}, nextState) => {\n          const streamOutput = renderStream(rootElement, instance, nextState, stateMap)\n          if (streamOutput.isNewStream) { // quick check\n            const nextSource$ = streamOutput.source\n            // debugger\n            instancePointer = streamOutput.instance\n            patch(rootNode, diff(instance.dom, instancePointer.dom)) // render to screen\n            emitter.emit(nextSource$) // update the UI thread; source will switch\n          } else {\n            const nextinstance = streamOutput.instance\n            patch(rootNode, diff(instance.dom, nextinstance.dom)) // render to screen\n            return {instance: nextinstance, stateMap: stateMap}\n          }\n        },\n        {instance: instancePointer, stateMap: stateMapPointer} // accumulator\n      )\n      /**** end main */\n      currentSrc$ = \n        source2$\n          .subscribe()\n    }\n  )\n}\n","import { createElement, createHandler } from \"./element\";\nimport { Component } from \"./component\";\nimport { renderStream } from \"./reconciler\"\nimport { mount } from \"./scheduler\";\n\nexport default {\n  renderStream,\n  createElement,\n  createHandler,\n  Component,\n  mount\n};\n\nexport { createElement, createHandler, Component, \n  renderStream, \n  mount };\n"],"names":["TEXT_ELEMENT","createElement","type","config","props","Object","assign","args","hasChildren","length","rawChildren","concat","children","filter","c","map","createTextElement","value","nodeValue","createHandler","_fn","emitter","createChangeEmitter","handler","emit","x","$","Observable","listen","next","NOINIT","Symbol","scan","obs","cb","seed","sub","acc","hasValue","hasSeed","subscribe","observer","closed","first","e","error","startWith","val","Component","state","obj","sources","entries","k","fn","subReducer","source","reducer","_","n","initialState","merge","createPublicInstance","element","publicInstance","renderStream","instance","stateMap","isNewStream","of","addToStream","_source","markNewStream","newInstance","render","renderWithStream","isDomElement","rest","childInstances","el","i","_childInstances","childInstance","childDoms","dom","lcaseProps","forEach","v","formatProps","VText","h","localState","get","undefined","keys","src","src$","set","event","childElement","startsWith","toLowerCase","stateMapPointer","Map","UIthread","mount","rootElement","container","instancePointer","rootNode","containerChild","firstElementChild","replaceChild","appendChild","currentSrc$","SoS","console","log","unsubscribe","source2$","nextState","streamOutput","nextSource$","diff","nextinstance"],"mappings":"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"}
351 |
--------------------------------------------------------------------------------
/build/reactive-react.umd.js:
--------------------------------------------------------------------------------
1 | (function (global, factory) {
2 | typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('zen-observable'), require('change-emitter'), require('zen-observable/extras'), require('virtual-dom/h'), require('virtual-dom/vnode/vtext'), require('virtual-dom/diff'), require('virtual-dom/patch'), require('virtual-dom/create-element')) :
3 | typeof define === 'function' && define.amd ? define(['exports', 'zen-observable', 'change-emitter', 'zen-observable/extras', 'virtual-dom/h', 'virtual-dom/vnode/vtext', 'virtual-dom/diff', 'virtual-dom/patch', 'virtual-dom/create-element'], factory) :
4 | (factory((global['reactive-react'] = global['reactive-react'] || {}),global.Observable,global.changeEmitter,global.zenObservable_extras,global.h,global.VText,global.diff,global.patch,global.createElement));
5 | }(this, (function (exports,Observable,changeEmitter,zenObservable_extras,h,VText,diff,patch,createElement) { 'use strict';
6 |
7 | Observable = 'default' in Observable ? Observable['default'] : Observable;
8 | h = 'default' in h ? h['default'] : h;
9 | VText = 'default' in VText ? VText['default'] : VText;
10 | diff = 'default' in diff ? diff['default'] : diff;
11 | patch = 'default' in patch ? patch['default'] : patch;
12 | createElement = 'default' in createElement ? createElement['default'] : createElement;
13 |
14 | var TEXT_ELEMENT = "TEXT ELEMENT";
15 |
16 | function createElement$1(type, config) {
17 | var _ref;
18 |
19 | var props = Object.assign({}, config);
20 |
21 | for (var _len = arguments.length, args = Array(_len > 2 ? _len - 2 : 0), _key = 2; _key < _len; _key++) {
22 | args[_key - 2] = arguments[_key];
23 | }
24 |
25 | var hasChildren = args.length > 0;
26 | var rawChildren = hasChildren ? (_ref = []).concat.apply(_ref, args) : [];
27 | props.children = rawChildren.filter(function (c) {
28 | return c != null && c !== false;
29 | }).map(function (c) {
30 | return c instanceof Object ? c : createTextElement(c);
31 | });
32 | return { type: type, props: props };
33 | }
34 |
35 | function createTextElement(value) {
36 | return createElement$1(TEXT_ELEMENT, { nodeValue: value });
37 | }
38 |
39 | function createHandler(_fn) {
40 | var emitter = changeEmitter.createChangeEmitter();
41 | var handler = function handler(x) {
42 | emitter.emit(x);
43 | };
44 | handler.$ = new Observable(function (observer) {
45 | return emitter.listen(function (value) {
46 | observer.next(_fn ? _fn(value) : value);
47 | });
48 | });
49 | return handler;
50 | }
51 |
52 | var NOINIT = Symbol('NO_INITIAL_VALUE');
53 | function scan(obs, cb) {
54 | var seed = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : NOINIT;
55 |
56 | var sub = void 0,
57 | acc = seed,
58 | hasValue = false;
59 | var hasSeed = acc !== NOINIT;
60 | return new Observable(function (observer) {
61 | sub = obs.subscribe(function (value) {
62 | if (observer.closed) return;
63 | var first = !hasValue;
64 | hasValue = true;
65 | if (!first || hasSeed) {
66 | try {
67 | acc = cb(acc, value);
68 | } catch (e) {
69 | return observer.error(e);
70 | }
71 | observer.next(acc);
72 | } else {
73 | acc = value;
74 | }
75 | });
76 | return sub;
77 | });
78 | }
79 |
80 | // Flatten a collection of observables and only output the newest from each
81 |
82 |
83 |
84 |
85 | function startWith(obs, val) {
86 | return new Observable(function (observer) {
87 | observer.next(val); // immediately output this value
88 | var handler = obs.subscribe(function (x) {
89 | return observer.next(x);
90 | });
91 | return function () {
92 | return handler();
93 | };
94 | });
95 | }
96 |
97 | var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; };
98 |
99 | var _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"]) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError("Invalid attempt to destructure non-iterable instance"); } }; }();
100 |
101 | var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
102 |
103 | function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }
104 |
105 | function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
106 |
107 | function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
108 |
109 | // import { reconcile } from "./reconciler";
110 | var Component = function () {
111 | function Component(props) {
112 | _classCallCheck(this, Component);
113 |
114 | this.props = props;
115 | this.state = this.state || {};
116 | }
117 |
118 | // setState(partialState) {
119 | // this.state = Object.assign({}, this.state, partialState);
120 | // updateInstance(this.__internalInstance);
121 | // }
122 |
123 | // class method because it feeds in this.initialState
124 |
125 |
126 | _createClass(Component, [{
127 | key: 'combineReducer',
128 | value: function combineReducer(obj) {
129 | var _this = this;
130 |
131 | var sources = Object.entries(obj).map(function (_ref) {
132 | var _ref2 = _slicedToArray(_ref, 2),
133 | k = _ref2[0],
134 | fn = _ref2[1];
135 |
136 | var subReducer = fn(obj);
137 | // there are two forms of return the subreducer can have
138 | // straight stream form
139 | // or object form where we need to scan it into string
140 | if (subReducer.source && subReducer.reducer) {
141 | // object form
142 | subReducer = scan(subReducer.source, subReducer.reducer || function (_, n) {
143 | return n;
144 | }, _this.initialState[k]);
145 | }
146 | return subReducer.map(function (x) {
147 | return _defineProperty({}, k, x);
148 | }); // map to its particular namespace
149 | });
150 | var source = zenObservable_extras.merge.apply(undefined, _toConsumableArray(sources));
151 | var reducer = function reducer(acc, n) {
152 | return _extends({}, acc, n);
153 | };
154 | return { source: source, reducer: reducer };
155 | }
156 | }]);
157 |
158 | return Component;
159 | }();
160 |
161 | // function updateInstance(internalInstance) {
162 | // const parentDom = internalInstance.dom.parentNode;
163 | // const element = internalInstance.element;
164 | // reconcile(parentDom, internalInstance, element);
165 | // }
166 |
167 | function createPublicInstance(element /*, internalInstance*/) {
168 | var type = element.type,
169 | props = element.props;
170 |
171 | var publicInstance = new type(props);
172 | // publicInstance.__internalInstance = internalInstance;
173 | return publicInstance;
174 | }
175 |
176 | var _slicedToArray$1 = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"]) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError("Invalid attempt to destructure non-iterable instance"); } }; }();
177 |
178 | function _objectWithoutProperties(obj, keys) { var target = {}; for (var i in obj) { if (keys.indexOf(i) >= 0) continue; if (!Object.prototype.hasOwnProperty.call(obj, i)) continue; target[i] = obj[i]; } return target; }
179 |
180 | // import { updateDomProperties } from "./updateProperties";
181 | // import VNode from "virtual-dom/vnode/vnode"
182 | // const circuitBreakerflag = false // set true to enable debugger in infinite loops
183 | // let circuitBreaker = -50
184 | // traverse all children and collect a stream of all sources
185 | // AND render. a bit of duplication, but we get persistent instances which is good
186 | function renderStream(element, instance, state, stateMap) {
187 | // this is a separate function because scope gets messy when being recursive
188 | var isNewStream = false; // assume no stream switching by default
189 | // this is the first ping of data throughout the app
190 | var source = Observable.of(state);
191 | var addToStream = function addToStream(_source) {
192 | // visit each source and merge with source
193 | if (_source) return source = zenObservable_extras.merge(source, _source);
194 | };
195 | var markNewStream = function markNewStream() {
196 | return isNewStream = true;
197 | };
198 | var newInstance = render(source, addToStream, markNewStream)(element, instance, state, stateMap);
199 | return { source: source, instance: newInstance, isNewStream: isNewStream };
200 | }
201 |
202 | /** core render logic */
203 | function render(source, addToStream, markNewStream) {
204 | // this is the nonrecursive part
205 | return function renderWithStream(element, instance, state, stateMap) {
206 | // recursive part
207 | var newInstance = void 0;
208 | var type = element.type,
209 | props = element.props;
210 |
211 |
212 | var isDomElement = typeof type === "string";
213 | // if (circuitBreakerflag && circuitBreaker++ > 0) debugger
214 |
215 | var _props$children = props.children,
216 | children = _props$children === undefined ? [] : _props$children,
217 | rest = _objectWithoutProperties(props, ['children']);
218 |
219 | if (isDomElement) {
220 | var childInstances = children.map(function (el, i) {
221 | // ugly but necessary to allow functional children
222 | // mapping element's children to instance's childInstances
223 | var _childInstances = instance && (instance.childInstance || instance.childInstances[i]);
224 | return renderWithStream( // recursion
225 | el, _childInstances, state, stateMap);
226 | });
227 | var childDoms = childInstances.map(function (childInstance) {
228 | return childInstance.dom;
229 | });
230 | var lcaseProps = {};
231 | Object.entries(rest).forEach(function (_ref) {
232 | var _ref2 = _slicedToArray$1(_ref, 2),
233 | k = _ref2[0],
234 | v = _ref2[1];
235 |
236 | return lcaseProps[formatProps(k)] = v;
237 | });
238 | var dom = type === TEXT_ELEMENT ? new VText(props.nodeValue) : h(type, lcaseProps, childDoms); // equivalent of appendchild
239 | newInstance = { dom: dom, element: element, childInstances: childInstances };
240 | } else {
241 | // component element
242 | var publicInstance = void 0;
243 | // debugger
244 | if (instance && instance.publicInstance && instance.element === element) {
245 | // might have to do more diffing of props
246 | // just reuse old instance if it already exists
247 | publicInstance = instance && instance.publicInstance;
248 | } else {
249 | markNewStream(); // mark as dirty in parent scope; will rerender
250 | publicInstance = createPublicInstance(element);
251 | }
252 | var localState = stateMap.get(publicInstance);
253 | if (localState === undefined) localState = publicInstance.initialState;
254 | publicInstance.state = localState; // for access with this.state
255 | if (Object.keys(rest).length) publicInstance.props = rest; // update with new props // TODO: potentially buggy
256 | // console.log({rest})
257 | if (publicInstance.source) {
258 | var src = publicInstance.source(source);
259 | // there are two forms of Component.source
260 | var src$ = src.reducer && publicInstance.initialState !== undefined ?
261 | // 1. the reducer form
262 | scan(src.source, src.reducer, publicInstance.initialState) :
263 | // 2. and raw stream form
264 | src;
265 | addToStream(src$.map(function (event) {
266 | stateMap.set(publicInstance, event);
267 | return { instance: publicInstance, event: event // tag it to the instance
268 | };
269 | }));
270 | }
271 | var childElement = publicInstance.render ? publicInstance.render(localState, stateMap) : publicInstance;
272 |
273 | var childInstance = renderWithStream(childElement, instance && instance.childInstance, state, stateMap);
274 | var _dom = childInstance.dom;
275 | newInstance = { dom: _dom, element: element, childInstance: childInstance, publicInstance: publicInstance };
276 | }
277 | return newInstance;
278 | };
279 | }
280 |
281 | function formatProps(k) {
282 | if (k.startsWith('on')) return k.toLowerCase();
283 | return k;
284 | }
285 |
286 | var stateMapPointer = new Map();
287 |
288 | var emitter = changeEmitter.createChangeEmitter();
289 | // single UI thread; this is the observable that sticks around and swaps out source
290 | var UIthread = new Observable(function (observer) {
291 | emitter.listen(function (x) {
292 | // debugger // success! thread switching!
293 | observer.next(x);
294 | });
295 | });
296 | // mount the vdom on to the dom and
297 | // set up the runtime from sources and
298 | // patch the vdom
299 | // ---
300 | // returns an unsubscribe method you can use to unmount
301 | function mount(rootElement, container) {
302 | // initial, throwaway-ish frame
303 | var _renderStream = renderStream(rootElement, {}, undefined, stateMapPointer),
304 | source = _renderStream.source,
305 | instance = _renderStream.instance;
306 |
307 | var instancePointer = instance;
308 | var rootNode = createElement(instance.dom);
309 | var containerChild = container.firstElementChild;
310 | if (containerChild) {
311 | container.replaceChild(rootNode, containerChild); // hot reloaded mount
312 | } else {
313 | container.appendChild(rootNode); // initial mount
314 | }
315 | var currentSrc$ = null;
316 | var SoS = startWith(UIthread, source); // stream of streams
317 | return SoS.subscribe(function (src$) {
318 | // this is the current sourceStream we are working with
319 | if (currentSrc$) console.log('unsub!') || currentSrc$.unsubscribe(); // unsub from old stream
320 | /**** main */
321 | var source2$ = scan(src$, function (_ref, nextState) {
322 | var instance = _ref.instance,
323 | stateMap = _ref.stateMap;
324 |
325 | var streamOutput = renderStream(rootElement, instance, nextState, stateMap);
326 | if (streamOutput.isNewStream) {
327 | // quick check
328 | var nextSource$ = streamOutput.source;
329 | // debugger
330 | instancePointer = streamOutput.instance;
331 | patch(rootNode, diff(instance.dom, instancePointer.dom)); // render to screen
332 | emitter.emit(nextSource$); // update the UI thread; source will switch
333 | } else {
334 | var nextinstance = streamOutput.instance;
335 | patch(rootNode, diff(instance.dom, nextinstance.dom)); // render to screen
336 | return { instance: nextinstance, stateMap: stateMap };
337 | }
338 | }, { instance: instancePointer, stateMap: stateMapPointer // accumulator
339 | });
340 | /**** end main */
341 | currentSrc$ = source2$.subscribe();
342 | });
343 | }
344 |
345 | var index = {
346 | renderStream: renderStream,
347 | createElement: createElement$1,
348 | createHandler: createHandler,
349 | Component: Component,
350 | mount: mount
351 | };
352 |
353 | exports['default'] = index;
354 | exports.createElement = createElement$1;
355 | exports.createHandler = createHandler;
356 | exports.Component = Component;
357 | exports.renderStream = renderStream;
358 | exports.mount = mount;
359 |
360 | Object.defineProperty(exports, '__esModule', { value: true });
361 |
362 | })));
363 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"reactive-react.umd.js","sources":["../reactive-react/element.js","../reactive-react/swyxjs.js","../reactive-react/component.js","../reactive-react/reconciler.js","../reactive-react/scheduler.js","../reactive-react/index.js"],"sourcesContent":["import Observable from 'zen-observable'\nimport { createChangeEmitter } from 'change-emitter'\n\nexport const TEXT_ELEMENT = \"TEXT ELEMENT\";\n\nexport function createElement(type, config, ...args) {\n  const props = Object.assign({}, config);\n  const hasChildren = args.length > 0;\n  const rawChildren = hasChildren ? [].concat(...args) : [];\n  props.children = rawChildren\n    .filter(c => c != null && c !== false)\n    .map(c => c instanceof Object ? c : createTextElement(c));\n  return { type, props };\n}\n\nfunction createTextElement(value) {\n  return createElement(TEXT_ELEMENT, { nodeValue: value });\n}\n\nexport function createHandler(_fn) {\n  const emitter = createChangeEmitter()\n  let handler = x => {\n    emitter.emit(x)\n  }\n  handler.$ = new Observable(observer => {\n    return emitter.listen(value => {\n      observer.next(_fn ? _fn(value) : value)\n    }\n    )\n  })\n  return handler\n}","import Observable from 'zen-observable'\nexport { merge, combineLatest, zip } from 'zen-observable/extras'\n\nexport function Interval(tick = 1000, tickData = Symbol('tick')) {\n  return new Observable(observer => {\n    let timer = () => setTimeout(() => {\n      if (typeof tickData === 'function') tickData = tickData()\n      observer.next(tickData);\n      timer()\n      // observer.complete();\n    }, tick);\n    timer()\n  \n    // On unsubscription, cancel the timer\n    return () => clearTimeout(timer);\n\n  })\n}\n\nexport function fromEvent(el, eventType) {\n  return new Observable(observer => {\n    const handler = e => observer.next(e)\n    el.addEventListener(eventType, handler)\n    // on unsub, remove event listener\n    return () => el.removeEventListener(eventType, handler)\n  })\n}\n\nconst NOINIT = Symbol('NO_INITIAL_VALUE')\nexport function scan(obs, cb, seed = NOINIT) {\n  let sub, acc = seed, hasValue = false\n  const hasSeed = acc !== NOINIT\n  return new Observable(observer => {\n    sub = obs.subscribe(value => {\n      if (observer.closed) return\n      let first = !hasValue;\n      hasValue = true\n      if (!first || hasSeed ) {\n        try { acc = cb(acc, value) }\n        catch (e) { return observer.error(e) }\n        observer.next(acc);\n      }\n      else {\n        acc = value\n      }\n    })\n    return sub\n  })\n}\n\n// Flatten a collection of observables and only output the newest from each\nexport function switchLatest(higherObservable) {\n  return new Observable(observer => {\n    let currentObs = null\n    return higherObservable.subscribe({\n      next(obs) {\n        if (currentObs) currentObs.unsubscribe() // unsub and switch\n        currentObs = obs.subscribe(observer.subscribe)\n      },\n      error(e) {\n        observer.error(e) // untested\n      },\n      complete() {\n        // i dont think it should complete?\n        // observer.complete()\n      }\n    })\n  });\n}\n\nexport function mapToConstant(obs, val) {\n  return new Observable(observer => {\n    const handler = obs.subscribe(() => observer.next(val))\n    return handler\n  })\n}\n\nexport function startWith(obs, val) {\n  return new Observable(observer => {\n    observer.next(val) // immediately output this value\n    const handler = obs.subscribe(x => observer.next(x))\n    return () => handler()\n  })\n}","// import { reconcile } from \"./reconciler\";\nimport {Interval, scan, startWith, merge, mapToConstant} from './swyxjs'\n\nexport class Component {\n  constructor(props) {\n    this.props = props;\n    this.state = this.state || {};\n  }\n\n  // setState(partialState) {\n  //   this.state = Object.assign({}, this.state, partialState);\n  //   updateInstance(this.__internalInstance);\n  // }\n\n  // class method because it feeds in this.initialState\n  combineReducer(obj) {\n    const sources = Object.entries(obj).map(([k,fn]) => {\n      let subReducer = fn(obj)\n      // there are two forms of return the subreducer can have\n      // straight stream form\n      // or object form where we need to scan it into string\n      if (subReducer.source && subReducer.reducer) { // object form\n        subReducer = scan(subReducer.source, \n          subReducer.reducer || ((_, n) => n), \n          this.initialState[k]\n        )\n      }\n      return subReducer\n        .map(x => ({[k]: x})) // map to its particular namespace\n    })\n    const source = merge(...sources)\n    const reducer = (acc, n) => ({...acc, ...n})\n    return {source, reducer}\n  }\n}\n\n// function updateInstance(internalInstance) {\n//   const parentDom = internalInstance.dom.parentNode;\n//   const element = internalInstance.element;\n//   reconcile(parentDom, internalInstance, element);\n// }\n\nexport function createPublicInstance(element/*, internalInstance*/) {\n  const { type, props } = element;\n  const publicInstance = new type(props);\n  // publicInstance.__internalInstance = internalInstance;\n  return publicInstance;\n}\n","import Observable from 'zen-observable'\nimport {fromEvent, scan, merge, startWith, switchLatest} from './swyxjs'\n// import { updateDomProperties } from \"./updateProperties\";\nimport { TEXT_ELEMENT } from \"./element\";\nimport { createPublicInstance } from \"./component\";\nimport h from 'virtual-dom/h'\n// import VNode from \"virtual-dom/vnode/vnode\"\nimport VText from \"virtual-dom/vnode/vtext\"\n\n// const circuitBreakerflag = false // set true to enable debugger in infinite loops\n// let circuitBreaker = -50\n// traverse all children and collect a stream of all sources\n// AND render. a bit of duplication, but we get persistent instances which is good\nexport function renderStream(element, instance, state, stateMap) {\n  // this is a separate function because scope gets messy when being recursive\n  let isNewStream = false // assume no stream switching by default\n  // this is the first ping of data throughout the app\n  let source = Observable.of(state) \n  const addToStream = _source => {\n    // visit each source and merge with source\n    if (_source) return source = merge(source, _source)\n  }\n  const markNewStream = () => isNewStream = true\n  const newInstance = render(source, addToStream, markNewStream)(element, instance, state, stateMap)\n  return {source, instance: newInstance, isNewStream}\n}\n\n/** core render logic */\nexport function render(source, addToStream, markNewStream) { // this is the nonrecursive part\n  return function renderWithStream(element, instance, state, stateMap) { // recursive part\n    let newInstance\n    const { type, props } = element\n  \n    const isDomElement = typeof type === \"string\";\n    // if (circuitBreakerflag && circuitBreaker++ > 0) debugger\n    const {children = [], ...rest} = props\n    if (isDomElement) {\n      const childInstances = children.map(\n        (el, i) => {\n          // ugly but necessary to allow functional children\n          // mapping element's children to instance's childInstances\n          const _childInstances = instance && (instance.childInstance || instance.childInstances[i])\n          return renderWithStream(  // recursion\n            el, \n            _childInstances, \n            state, \n            stateMap) \n        }\n      );\n      const childDoms = childInstances.map(childInstance => childInstance.dom);\n      let lcaseProps = {}\n      Object.entries(rest).forEach(([k, v]) => lcaseProps[formatProps(k)] = v)\n      const dom = type === TEXT_ELEMENT\n        ? new VText(props.nodeValue)\n        : h(type, lcaseProps, childDoms); // equivalent of appendchild\n      newInstance = { dom, element, childInstances };\n    } else { // component element\n      let publicInstance \n      // debugger\n      if (instance && instance.publicInstance && instance.element === element) { // might have to do more diffing of props\n        // just reuse old instance if it already exists\n        publicInstance = instance && instance.publicInstance\n      } else {\n        markNewStream() // mark as dirty in parent scope; will rerender\n        publicInstance = createPublicInstance(element);\n      }\n      let localState = stateMap.get(publicInstance)\n      if (localState === undefined) localState = publicInstance.initialState\n      publicInstance.state = localState // for access with this.state\n      if (Object.keys(rest).length) publicInstance.props = rest // update with new props // TODO: potentially buggy\n      // console.log({rest})\n      if (publicInstance.source) {\n        const src = publicInstance.source(source)\n        // there are two forms of Component.source\n        const src$ = src.reducer && publicInstance.initialState !== undefined ? \n            // 1. the reducer form\n            scan(src.source, src.reducer, publicInstance.initialState) : \n            // 2. and raw stream form\n            src\n        addToStream(src$\n          .map(event => {\n            stateMap.set(publicInstance, event)\n            return {instance: publicInstance, event} // tag it to the instance\n          }) \n        );\n      }\n      const childElement = publicInstance.render ? \n          publicInstance.render(localState, stateMap) : \n          publicInstance;\n\n      const childInstance = renderWithStream(childElement, instance && instance.childInstance, state, stateMap)\n      const dom = childInstance.dom\n      newInstance = { dom, element, childInstance, publicInstance }\n    }\n    return newInstance\n  }\n}\n\nfunction formatProps(k) {\n  if (k.startsWith('on')) return k.toLowerCase()\n  return k\n}","import Observable from 'zen-observable'\nimport {fromEvent, scan, merge, startWith, switchLatest} from './swyxjs'\nimport diff from 'virtual-dom/diff';\nimport patch from 'virtual-dom/patch';\nimport createElement from 'virtual-dom/create-element';\nimport { createChangeEmitter } from 'change-emitter'\nimport { renderStream } from './reconciler'\n\nexport const stateMapPointer = new Map()\n\nlet circuitBreaker = -20\n\nconst emitter = createChangeEmitter()\n// single UI thread; this is the observable that sticks around and swaps out source\nconst UIthread = new Observable(observer => {\n  emitter.listen(x => {\n    // debugger // success! thread switching!\n    observer.next(x)\n  })\n})\n// mount the vdom on to the dom and \n// set up the runtime from sources and\n// patch the vdom\n// ---\n// returns an unsubscribe method you can use to unmount\nexport function mount(rootElement, container) {\n  // initial, throwaway-ish frame\n  let {source, instance} = renderStream(rootElement, {}, undefined, stateMapPointer)\n  let instancePointer = instance\n  const rootNode = createElement(instance.dom)\n  const containerChild = container.firstElementChild\n  if (containerChild) {\n    container.replaceChild(rootNode,containerChild) // hot reloaded mount\n  } else {\n    container.appendChild(rootNode) // initial mount\n  }\n  let currentSrc$ = null\n  let SoS = startWith(UIthread, source) // stream of streams\n  return SoS.subscribe(\n    src$ => { // this is the current sourceStream we are working with\n      if (currentSrc$) console.log('unsub!') || currentSrc$.unsubscribe() // unsub from old stream\n      /**** main */\n      const source2$ = scan(\n        src$, \n        ({instance, stateMap}, nextState) => {\n          const streamOutput = renderStream(rootElement, instance, nextState, stateMap)\n          if (streamOutput.isNewStream) { // quick check\n            const nextSource$ = streamOutput.source\n            // debugger\n            instancePointer = streamOutput.instance\n            patch(rootNode, diff(instance.dom, instancePointer.dom)) // render to screen\n            emitter.emit(nextSource$) // update the UI thread; source will switch\n          } else {\n            const nextinstance = streamOutput.instance\n            patch(rootNode, diff(instance.dom, nextinstance.dom)) // render to screen\n            return {instance: nextinstance, stateMap: stateMap}\n          }\n        },\n        {instance: instancePointer, stateMap: stateMapPointer} // accumulator\n      )\n      /**** end main */\n      currentSrc$ = \n        source2$\n          .subscribe()\n    }\n  )\n}\n","import { createElement, createHandler } from \"./element\";\nimport { Component } from \"./component\";\nimport { renderStream } from \"./reconciler\"\nimport { mount } from \"./scheduler\";\n\nexport default {\n  renderStream,\n  createElement,\n  createHandler,\n  Component,\n  mount\n};\n\nexport { createElement, createHandler, Component, \n  renderStream, \n  mount };\n"],"names":["TEXT_ELEMENT","createElement","type","config","props","Object","assign","args","hasChildren","length","rawChildren","concat","children","filter","c","map","createTextElement","value","nodeValue","createHandler","_fn","emitter","createChangeEmitter","handler","emit","x","$","Observable","listen","next","NOINIT","Symbol","scan","obs","cb","seed","sub","acc","hasValue","hasSeed","subscribe","observer","closed","first","e","error","startWith","val","Component","state","obj","sources","entries","k","fn","subReducer","source","reducer","_","n","initialState","merge","createPublicInstance","element","publicInstance","renderStream","instance","stateMap","isNewStream","of","addToStream","_source","markNewStream","newInstance","render","renderWithStream","isDomElement","rest","childInstances","el","i","_childInstances","childInstance","childDoms","dom","lcaseProps","forEach","v","formatProps","VText","h","localState","get","undefined","keys","src","src$","set","event","childElement","startsWith","toLowerCase","stateMapPointer","Map","UIthread","mount","rootElement","container","instancePointer","rootNode","containerChild","firstElementChild","replaceChild","appendChild","currentSrc$","SoS","console","log","unsubscribe","source2$","nextState","streamOutput","nextSource$","diff","nextinstance"],"mappings":";;;;;;;;;;;;;AAGO,IAAMA,eAAe,cAArB;;AAEP,AAAO,SAASC,eAAT,CAAuBC,IAAvB,EAA6BC,MAA7B,EAA8C;;;MAC7CC,QAAQC,OAAOC,MAAP,CAAc,EAAd,EAAkBH,MAAlB,CAAd;;oCAD6CI,IAAM;QAAA;;;MAE7CC,cAAcD,KAAKE,MAAL,GAAc,CAAlC;MACMC,cAAcF,cAAc,YAAGG,MAAH,aAAaJ,IAAb,CAAd,GAAmC,EAAvD;QACMK,QAAN,GAAiBF,YACdG,MADc,CACP;WAAKC,KAAK,IAAL,IAAaA,MAAM,KAAxB;GADO,EAEdC,GAFc,CAEV;WAAKD,aAAaT,MAAb,GAAsBS,CAAtB,GAA0BE,kBAAkBF,CAAlB,CAA/B;GAFU,CAAjB;SAGO,EAAEZ,UAAF,EAAQE,YAAR,EAAP;;;AAGF,SAASY,iBAAT,CAA2BC,KAA3B,EAAkC;SACzBhB,gBAAcD,YAAd,EAA4B,EAAEkB,WAAWD,KAAb,EAA5B,CAAP;;;AAGF,AAAO,SAASE,aAAT,CAAuBC,GAAvB,EAA4B;MAC3BC,UAAUC,mCAAhB;MACIC,UAAU,SAAVA,OAAU,IAAK;YACTC,IAAR,CAAaC,CAAb;GADF;UAGQC,CAAR,GAAY,IAAIC,UAAJ,CAAe,oBAAY;WAC9BN,QAAQO,MAAR,CAAe,iBAAS;eACpBC,IAAT,CAAcT,MAAMA,IAAIH,KAAJ,CAAN,GAAmBA,KAAjC;KADK,CAAP;GADU,CAAZ;SAMOM,OAAP;;;ACFF,IAAMO,SAASC,OAAO,kBAAP,CAAf;AACA,AAAO,SAASC,IAAT,CAAcC,GAAd,EAAmBC,EAAnB,EAAsC;MAAfC,IAAe,uEAARL,MAAQ;;MACvCM,YAAJ;MAASC,MAAMF,IAAf;MAAqBG,WAAW,KAAhC;MACMC,UAAUF,QAAQP,MAAxB;SACO,IAAIH,UAAJ,CAAe,oBAAY;UAC1BM,IAAIO,SAAJ,CAAc,iBAAS;UACvBC,SAASC,MAAb,EAAqB;UACjBC,QAAQ,CAACL,QAAb;iBACW,IAAX;UACI,CAACK,KAAD,IAAUJ,OAAd,EAAwB;YAClB;gBAAQL,GAAGG,GAAH,EAAQpB,KAAR,CAAN;SAAN,CACA,OAAO2B,CAAP,EAAU;iBAASH,SAASI,KAAT,CAAeD,CAAf,CAAP;;iBACHf,IAAT,CAAcQ,GAAd;OAHF,MAKK;cACGpB,KAAN;;KAVE,CAAN;WAaOmB,GAAP;GAdK,CAAP;;;;AAmBF,AAAO;;AAmBP,AAAO;;AAOP,AAAO,SAASU,SAAT,CAAmBb,GAAnB,EAAwBc,GAAxB,EAA6B;SAC3B,IAAIpB,UAAJ,CAAe,oBAAY;aACvBE,IAAT,CAAckB,GAAd,EADgC;QAE1BxB,UAAUU,IAAIO,SAAJ,CAAc;aAAKC,SAASZ,IAAT,CAAcJ,CAAd,CAAL;KAAd,CAAhB;WACO;aAAMF,SAAN;KAAP;GAHK,CAAP;;;;;;;;;;;;;;;;AC7EF,AAEA,IAAayB,SAAb;qBACc5C,KAAZ,EAAmB;;;SACZA,KAAL,GAAaA,KAAb;SACK6C,KAAL,GAAa,KAAKA,KAAL,IAAc,EAA3B;;;;;;;;;;;;;mCASaC,GAZjB,EAYsB;;;UACZC,UAAU9C,OAAO+C,OAAP,CAAeF,GAAf,EAAoBnC,GAApB,CAAwB,gBAAY;;YAAVsC,CAAU;YAARC,EAAQ;;YAC9CC,aAAaD,GAAGJ,GAAH,CAAjB;;;;YAIIK,WAAWC,MAAX,IAAqBD,WAAWE,OAApC,EAA6C;;uBAC9BzB,KAAKuB,WAAWC,MAAhB,EACXD,WAAWE,OAAX,IAAuB,UAACC,CAAD,EAAIC,CAAJ;mBAAUA,CAAV;WADZ,EAEX,MAAKC,YAAL,CAAkBP,CAAlB,CAFW,CAAb;;eAKKE,WACJxC,GADI,CACA;qCAAQsC,CAAR,EAAY5B,CAAZ;SADA,CAAP,CAXkD;OAApC,CAAhB;UAcM+B,SAASK,+DAASV,OAAT,EAAf;UACMM,UAAU,SAAVA,OAAU,CAACpB,GAAD,EAAMsB,CAAN;4BAAiBtB,GAAjB,EAAyBsB,CAAzB;OAAhB;aACO,EAACH,cAAD,EAASC,gBAAT,EAAP;;;;;;;;;;;;;AAUJ,AAAO,SAASK,oBAAT,CAA8BC,OAA9B,yBAA6D;MAC1D7D,IAD0D,GAC1C6D,OAD0C,CAC1D7D,IAD0D;MACpDE,KADoD,GAC1C2D,OAD0C,CACpD3D,KADoD;;MAE5D4D,iBAAiB,IAAI9D,IAAJ,CAASE,KAAT,CAAvB;;SAEO4D,cAAP;;;;;;;AC9CF,AACA,AACA;AACA,AACA,AACA,AACA;AACA,AAEA;;;;AAIA,AAAO,SAASC,YAAT,CAAsBF,OAAtB,EAA+BG,QAA/B,EAAyCjB,KAAzC,EAAgDkB,QAAhD,EAA0D;;MAE3DC,cAAc,KAAlB,CAF+D;;MAI3DZ,SAAS7B,WAAW0C,EAAX,CAAcpB,KAAd,CAAb;MACMqB,cAAc,SAAdA,WAAc,UAAW;;QAEzBC,OAAJ,EAAa,OAAOf,SAASK,2BAAML,MAAN,EAAce,OAAd,CAAhB;GAFf;MAIMC,gBAAgB,SAAhBA,aAAgB;WAAMJ,cAAc,IAApB;GAAtB;MACMK,cAAcC,OAAOlB,MAAP,EAAec,WAAf,EAA4BE,aAA5B,EAA2CT,OAA3C,EAAoDG,QAApD,EAA8DjB,KAA9D,EAAqEkB,QAArE,CAApB;SACO,EAACX,cAAD,EAASU,UAAUO,WAAnB,EAAgCL,wBAAhC,EAAP;;;;AAIF,AAAO,SAASM,MAAT,CAAgBlB,MAAhB,EAAwBc,WAAxB,EAAqCE,aAArC,EAAoD;;SAClD,SAASG,gBAAT,CAA0BZ,OAA1B,EAAmCG,QAAnC,EAA6CjB,KAA7C,EAAoDkB,QAApD,EAA8D;;QAC/DM,oBAAJ;QACQvE,IAF2D,GAE3C6D,OAF2C,CAE3D7D,IAF2D;QAErDE,KAFqD,GAE3C2D,OAF2C,CAErD3D,KAFqD;;;QAI7DwE,eAAe,OAAO1E,IAAP,KAAgB,QAArC;;;0BAEiCE,KANkC,CAM5DQ,QAN4D;QAM5DA,QAN4D,mCAMjD,EANiD;QAM1CiE,IAN0C,4BAMlCzE,KANkC;;QAO/DwE,YAAJ,EAAkB;UACVE,iBAAiBlE,SAASG,GAAT,CACrB,UAACgE,EAAD,EAAKC,CAAL,EAAW;;;YAGHC,kBAAkBf,aAAaA,SAASgB,aAAT,IAA0BhB,SAASY,cAAT,CAAwBE,CAAxB,CAAvC,CAAxB;eACOL;UAAA,EAELM,eAFK,EAGLhC,KAHK,EAILkB,QAJK,CAAP;OALmB,CAAvB;UAYMgB,YAAYL,eAAe/D,GAAf,CAAmB;eAAiBmE,cAAcE,GAA/B;OAAnB,CAAlB;UACIC,aAAa,EAAjB;aACOjC,OAAP,CAAeyB,IAAf,EAAqBS,OAArB,CAA6B;;YAAEjC,CAAF;YAAKkC,CAAL;;eAAYF,WAAWG,YAAYnC,CAAZ,CAAX,IAA6BkC,CAAzC;OAA7B;UACMH,MAAMlF,SAASF,YAAT,GACR,IAAIyF,KAAJ,CAAUrF,MAAMc,SAAhB,CADQ,GAERwE,EAAExF,IAAF,EAAQmF,UAAR,EAAoBF,SAApB,CAFJ,CAhBgB;oBAmBF,EAAEC,QAAF,EAAOrB,gBAAP,EAAgBe,8BAAhB,EAAd;KAnBF,MAoBO;;UACDd,uBAAJ;;UAEIE,YAAYA,SAASF,cAArB,IAAuCE,SAASH,OAAT,KAAqBA,OAAhE,EAAyE;;;yBAEtDG,YAAYA,SAASF,cAAtC;OAFF,MAGO;wBAAA;yBAEYF,qBAAqBC,OAArB,CAAjB;;UAEE4B,aAAaxB,SAASyB,GAAT,CAAa5B,cAAb,CAAjB;UACI2B,eAAeE,SAAnB,EAA8BF,aAAa3B,eAAeJ,YAA5B;qBACfX,KAAf,GAAuB0C,UAAvB,CAZK;UAaDtF,OAAOyF,IAAP,CAAYjB,IAAZ,EAAkBpE,MAAtB,EAA8BuD,eAAe5D,KAAf,GAAuByE,IAAvB,CAbzB;;UAeDb,eAAeR,MAAnB,EAA2B;YACnBuC,MAAM/B,eAAeR,MAAf,CAAsBA,MAAtB,CAAZ;;YAEMwC,OAAOD,IAAItC,OAAJ,IAAeO,eAAeJ,YAAf,KAAgCiC,SAA/C;;aAEJE,IAAIvC,MAAT,EAAiBuC,IAAItC,OAArB,EAA8BO,eAAeJ,YAA7C,CAFS;;WAAb;oBAKYoC,KACTjF,GADS,CACL,iBAAS;mBACHkF,GAAT,CAAajC,cAAb,EAA6BkC,KAA7B;iBACO,EAAChC,UAAUF,cAAX,EAA2BkC,YAA3B;WAAP;SAHQ,CAAZ;;UAOIC,eAAenC,eAAeU,MAAf,GACjBV,eAAeU,MAAf,CAAsBiB,UAAtB,EAAkCxB,QAAlC,CADiB,GAEjBH,cAFJ;;UAIMkB,gBAAgBP,iBAAiBwB,YAAjB,EAA+BjC,YAAYA,SAASgB,aAApD,EAAmEjC,KAAnE,EAA0EkB,QAA1E,CAAtB;UACMiB,OAAMF,cAAcE,GAA1B;oBACc,EAAEA,SAAF,EAAOrB,gBAAP,EAAgBmB,4BAAhB,EAA+BlB,8BAA/B,EAAd;;WAEKS,WAAP;GAjEF;;;AAqEF,SAASe,WAAT,CAAqBnC,CAArB,EAAwB;MAClBA,EAAE+C,UAAF,CAAa,IAAb,CAAJ,EAAwB,OAAO/C,EAAEgD,WAAF,EAAP;SACjBhD,CAAP;;;AC5FK,IAAMiD,kBAAkB,IAAIC,GAAJ,EAAxB;;AAEP,AAEA,IAAMlF,UAAUC,mCAAhB;;AAEA,IAAMkF,WAAW,IAAI7E,UAAJ,CAAe,oBAAY;UAClCC,MAAR,CAAe,aAAK;;aAETC,IAAT,CAAcJ,CAAd;GAFF;CADe,CAAjB;;;;;;AAWA,AAAO,SAASgF,KAAT,CAAeC,WAAf,EAA4BC,SAA5B,EAAuC;;sBAEnB1C,aAAayC,WAAb,EAA0B,EAA1B,EAA8Bb,SAA9B,EAAyCS,eAAzC,CAFmB;MAEvC9C,MAFuC,iBAEvCA,MAFuC;MAE/BU,QAF+B,iBAE/BA,QAF+B;;MAGxC0C,kBAAkB1C,QAAtB;MACM2C,WAAW5G,cAAciE,SAASkB,GAAvB,CAAjB;MACM0B,iBAAiBH,UAAUI,iBAAjC;MACID,cAAJ,EAAoB;cACRE,YAAV,CAAuBH,QAAvB,EAAgCC,cAAhC,EADkB;GAApB,MAEO;cACKG,WAAV,CAAsBJ,QAAtB,EADK;;MAGHK,cAAc,IAAlB;MACIC,MAAMrE,UAAU0D,QAAV,EAAoBhD,MAApB,CAAV,CAZ4C;SAarC2D,IAAI3E,SAAJ,CACL,gBAAQ;;QACF0E,WAAJ,EAAiBE,QAAQC,GAAR,CAAY,QAAZ,KAAyBH,YAAYI,WAAZ,EAAzB,CADX;;QAGAC,WAAWvF,KACfgE,IADe,EAEf,gBAAuBwB,SAAvB,EAAqC;UAAnCtD,QAAmC,QAAnCA,QAAmC;UAAzBC,QAAyB,QAAzBA,QAAyB;;UAC7BsD,eAAexD,aAAayC,WAAb,EAA0BxC,QAA1B,EAAoCsD,SAApC,EAA+CrD,QAA/C,CAArB;UACIsD,aAAarD,WAAjB,EAA8B;;YACtBsD,cAAcD,aAAajE,MAAjC;;0BAEkBiE,aAAavD,QAA/B;cACM2C,QAAN,EAAgBc,KAAKzD,SAASkB,GAAd,EAAmBwB,gBAAgBxB,GAAnC,CAAhB,EAJ4B;gBAKpB5D,IAAR,CAAakG,WAAb,EAL4B;OAA9B,MAMO;YACCE,eAAeH,aAAavD,QAAlC;cACM2C,QAAN,EAAgBc,KAAKzD,SAASkB,GAAd,EAAmBwC,aAAaxC,GAAhC,CAAhB,EAFK;eAGE,EAAClB,UAAU0D,YAAX,EAAyBzD,UAAUA,QAAnC,EAAP;;KAbW,EAgBf,EAACD,UAAU0C,eAAX,EAA4BzC,UAAUmC,eAAtC;KAhBe,CAAjB;;kBAoBEiB,SACG/E,SADH,EADF;GAvBG,CAAP;;;ACjCF,YAAe;4BAAA;gCAAA;8BAAA;sBAAA;;CAAf,CAQA;;;;;;;;;;;"}
364 |
--------------------------------------------------------------------------------
/demos/components/blink.js:
--------------------------------------------------------------------------------
1 | /** @jsx createElement */
2 | import {mount, createElement, Component, createHandler} from '../../reactive-react'
3 | import {Interval, scan, startWith, merge, mapToConstant} from '../../reactive-react/swyxjs'
4 | import Observable from 'zen-observable'
5 |
6 | export default class Blink extends Component {
7 | // more fun time demo
8 | // initialState = true // proper
9 | initialState = 0 // hacky
10 | source($) {
11 | // there is a bug right now where switching sources subscribes to the new source twice
12 | // havent been able to chase it down so i had to do this hacky thing
13 | // i'm sorry :( breaking demos last minute before talk sucks
14 | // const reducer = x => !x // the proper reducer
15 | const reducer = (acc, x) => acc + x // hacky reducer
16 | const source = Interval(500, 1) // tick every second
17 | return {source, reducer}
18 | }
19 | render(state) {
20 | const style = {
21 | visibility: ((state/2 + 1) % 2) ? // hacky
22 | 'visible' :
23 | 'hidden'}
24 | return Bring back the blink tag!
25 | }
26 | }
--------------------------------------------------------------------------------
/demos/components/button.css:
--------------------------------------------------------------------------------
1 |
2 |
3 | /* BonBon Buttons 1.1 by simurai.com
4 |
5 | 1.1 Added unprefixed attributes, :focus style,