[Javascript] Broadcaster + Operator + Listener pattern -- 15. Create a Sequence of Broadcasters
JavaScript -- of 15 create pattern sequence Operator
2023-09-14 09:00:46 时间
Many scenarios involve one task happening after another has completed. This lesson walks you through setting up "steps" which trigger one after another to create an asynchronous sequence of tasks.
By adding the done behavior to our broadcasters allow us to perform new operations on it, like create a sequence of broadcasters. This is, fire one broadcaster after the other only if the previous one is done
const sequence = (...broadcasters) => (listener) => { let broadcaster = broadcasters.shift(); let cancel; let sequenceListener = (value) => { if (value === done && broadcasters.length) { broadcaster = broadcasters.shift(); cancel = broadcaster(sequenceListener); return; } listener(value); }; cancel = broadcaster(sequenceListener); return () => { cancel(); }; };
import { addListener, createTimeout, done } from "./broadcasters"; import { hardCode } from "./operators"; const log = console.log; const delayMessage = (message) => hardCode(message)(createTimeout(1000)); const sequence = (...broadcasters) => (listener) => { let broadcaster = broadcasters.shift(); let cancel; let sequenceListener = (value) => { if (value === done && broadcasters.length) { broadcaster = broadcasters.shift(); cancel = broadcaster(sequenceListener); return; } listener(value); }; cancel = broadcaster(sequenceListener); return () => { cancel(); }; }; let cancelMsgs = sequence( delayMessage("Hello"), delayMessage("I"), delayMessage("am"), delayMessage("Zhentian"), delayMessage("Wan") )(log); let inputClick = addListener("#input", "click"); inputClick(cancelMsgs); // broadcasters.js import { curry } from "lodash"; export let done = Symbol("done"); export let createTimeout = curry((time, listener) => { let id = setTimeout(() => { listener(null); listener(done); }, time); return () => { clearTimeout(id); }; }); export let addListener = curry((selector, eventType, listener) => { let element = document.querySelector(selector); element.addEventListener(eventType, listener); return () => { element.removeEventListener(eventType, listener); }; }); export let createInterval = curry((time, listener) => { let i = 0; let id = setInterval(() => { listener(i++); }, time); return () => { clearInterval(id); }; }); //broadcaster = function that accepts a listener export let merge = curry((broadcaster1, broadcaster2, listener) => { let cancel1 = broadcaster1(listener); let cancel2 = broadcaster2(listener); return () => { cancel1(); cancel2(); }; }); export let zip = curry((broadcaster1, broadcaster2, listener) => { let cancelBoth; let buffer1 = []; let cancel1 = broadcaster1((value) => { buffer1.push(value); // console.log(buffer1) if (buffer2.length) { listener([buffer1.shift(), buffer2.shift()]); if (buffer1[0] === done || buffer2[0] === done) { listener(done); cancelBoth(); } } }); let buffer2 = []; let cancel2 = broadcaster2((value) => { buffer2.push(value); if (buffer1.length) { listener([buffer1.shift(), buffer2.shift()]); if (buffer1[0] === done || buffer2[0] === done) { listener(done); cancelBoth(); } } }); cancelBoth = () => { cancel1(); cancel2(); }; return cancelBoth; }); export let forOf = curry((iterable, listener) => { let id = setTimeout(() => { for (let i of iterable) { listener(i); } listener(done); }, 0); return () => { clearTimeout(id); }; }); // operators.js import { curry } from "lodash"; import { done } from "./broadcasters"; let createOperator = curry((operator, broadcaster, listener) => { return operator((behaviorListener) => { return broadcaster((value) => { if (value === done) { listener(done); return; } behaviorListener(value); }); }, listener); }); export let map = (transform) => createOperator((broadcaster, listener) => { return broadcaster((value) => { listener(transform(value)); }); }); export let filter = (predicate) => createOperator((broadcaster, listener) => { return broadcaster((value) => { if (predicate(value)) { listener(value); } }); }); export let split = (splitter) => curry((broadcaster, listener) => { let buffer = []; return broadcaster((value) => { if (value === done) { listener(buffer); buffer = []; listener(done); } if (value == splitter) { listener(buffer); buffer = []; } else { buffer.push(value); } }); }); export let hardCode = (newValue) => createOperator((broadcaster, listener) => { let cancel = broadcaster((value) => { listener(newValue); }); return () => cancel(); }); export let add = (initial) => (broadcaster) => (listener) => { return broadcaster((value) => { listener((initial += value)); }); }; export let startWhen = (whenBroadcaster) => (mainBroadcaster) => (listener) => { let cancelMain; let cancelWhen; cancelWhen = whenBroadcaster((whenValue) => { if (cancelMain) cancelMain(); cancelMain = mainBroadcaster((value) => { if (value === done) { if (whenValue === done) { listener(done); } return; } listener(value); }); }); return () => { cancelMain(); cancelWhen(); }; }; export let stopWhen = (whenBroadcaster) => (mainBroadcaster) => (listener) => { let cancelMain = mainBroadcaster(listener); let cancelWhen = whenBroadcaster((value) => { cancelMain(); }); return () => { cancelMain(); cancelWhen(); }; }; export let targetValue = map((event) => event.target.value); export let mapBroadcaster = (createBroadcaster) => (broadcaster) => ( listener ) => { return broadcaster((value) => { let newBroadcaster = createBroadcaster(value); newBroadcaster(listener); }); }; export let applyOperator = (broadcaster) => mapBroadcaster((operator) => operator(broadcaster)); export let stringConcat = (broadcaster) => (listener) => { let result = ""; return broadcaster((value) => { if (value === done) { listener(result); result = ""; return; } result += value; }); }; export const sequence = (...broadcasters) => (listener) => { let broadcaster = broadcasters.shift(); let cancel; let sequenceListener = (value) => { if (value === done && broadcasters.length) { broadcaster = broadcasters.shift(); cancel = broadcaster(sequenceListener); return; } listener(value); }; cancel = broadcaster(sequenceListener); return () => { cancel(); }; }; export let doneIf = (condition) => (broadcaster) => (listener) => { let cancel; cancel = broadcaster((value) => { if (condition(value)) { listener(done); return; } listener(value); }); return () => { cancel(); }; };
相关文章
- javascript html转换成markdown,如何使用Turndown使用JavaScript将HTML转换为Markdown[通俗易懂]
- JavaScript框架--迈向2023年
- JavaScript学习总结(五)——Javascript中==和===的区别详解编程语言
- javascript setinterval的缺陷详解编程语言
- javascript实现划词标记划词搜索功能修正版
- Javascript中暂停功能的实现代码
- JavaScript初级教程(第三课)
- 比较不错的函数式JavaScript编程指南教程
- Javascript动态伸缩+淡出淡入
- JavaScript报表展示实现代码
- javascript实现面向对象类的功能书写技巧
- JavaScript设计模式富有表现力的Javascript(一)
- javascript一个无懈可击的实例化XMLHttpRequest的方法
- JavaScript高级程序设计扩展--关于动态原型
- JavaScript中数组的排序、乱序和搜索实现代码
- JavaScript动态插入script的基本思路及实现函数
- 利用noesis.Javascript开源组件.Net中执行javascript脚本
- 让jQuery与其他JavaScript库并存避免冲突的方法
- javascript实现十六进制颜色值(HEX)和RGB格式相互转换
- BAT及各大互联网公司2014前端笔试面试题--JavaScript篇
- Javascript基础教程之JavaScript语法