Advertisement
Guest User

multitask

a guest
Nov 23rd, 2015
133
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
text 14.80 KB | None | 0 0
  1. --[[Multitask library
  2. By Konlab (Green)
  3. please check dependencies when removing parts of code!
  4. ]]
  5. --this is a default table for user events
  6. local usrEvents = {
  7. terminate = true,
  8. key = true,
  9. key_up = true,
  10. char = true,
  11. paste = true,
  12. }
  13. --this is the default table for mouse events that need translation
  14. local mouseEvents = {
  15. mouse_click = true,
  16. mouse_drag = true,
  17. mouse_up = true,
  18. mouse_scroll = true,
  19. }
  20. --dependency: none
  21. --Checktypes: checks types in vars and if it's not the types in the table called types errors at lvl with generated error message
  22. local function checkTypes(vars,types,lvl)
  23. lvl = lvl or 3
  24. for k,v in pairs(vars) do
  25. if type(types[k])=="string" and type(v) ~= types[k] then
  26. error(table.concat(types,", ").." expected",lvl)
  27. end
  28. end
  29. end
  30. --dependency: none
  31. --Delegate: returns new delegate
  32. function delegate()
  33. return setmetatable({func = {},
  34. add = function(self,func)
  35. if self and func then
  36. self.func[#self.func+1] = func
  37. else
  38. error("Self, Function expected, got "..type(self)..", "..type(func),2)
  39. end
  40. end,
  41. sub = function(self,func)
  42. if self and func then
  43. for i=1,#self.func do
  44. local v = self.func[i]
  45. if v == func then
  46. table.remove(self.func,i)
  47. end
  48. end
  49. else
  50. error("Self, Function expected, got "..type(self)..", "..type(func),2)
  51. end
  52. end,
  53. },
  54. {__call = function(tbl,...)
  55. if type(tbl.func) == "table" then
  56. local c,r = 0,{}
  57. for k,v in pairs(tbl.func) do
  58. if type(v) == "function" then
  59. c = c+1
  60. r[c] = {pcall(v,...)}
  61. end
  62. end
  63. return c,r
  64. end
  65. end,
  66. __add = function(self,func)
  67. self:add(func)
  68. end,
  69. __sub = function(self,func)
  70. self:sub(func)
  71. end,
  72. })
  73. end
  74. --Event: creates an event from the delegate
  75. function event(del)
  76. return {
  77. add = function(func)
  78. del.func[#del.func+1] = func
  79. end,
  80. sub = function(func)
  81. local limit = #del.func
  82. for i=1,limit do
  83. if del.func[i] == func then
  84. table.remove(del.func,i)
  85. end
  86. end
  87. end,
  88. }
  89. end
  90. --dependecy: checkTypes
  91. --New thread: returns a layer 1 thread table from func
  92. function newThread(func)
  93. checkTypes({func},{"function"})
  94. local t = {
  95. co = coroutine.create(func),
  96. cresume = function(self,...)
  97. return coroutine.resume(self.co,...)
  98. end,
  99. cstatus = function(self)
  100. return coroutine.status(self.co)
  101. end,
  102. }
  103. t.resume = t.cresume --set resume to this layer
  104. t.status = t.cstatus --set status to this layer
  105. return t
  106. end
  107. --dependency newThread
  108. --New CC Thread: layer 2, works with event types
  109. function newCCThread(func)
  110. local t
  111. if type(func) == "table" and func.resume and func.status then
  112. t = func
  113. else
  114. checkTypes({func},{"function"})
  115. t = newThread(func)
  116. end
  117. local parent = {
  118. resume = t.resume,
  119. status = t.status,
  120. }
  121. t.filter = nil
  122. t.ccresume = function(self,event,...)
  123. if event == self.filter or self.filter == nil or event == "terminate" then
  124. local ok,info = parent.resume(self,event,...)
  125. if ok then
  126. self.filter = info
  127. return true,info
  128. else
  129. return false,info
  130. end
  131. end
  132. return nil
  133. end
  134. t.resume = t.ccresume --set resume to this layer
  135. return t
  136. end
  137. --dependency: newCCThread
  138. --New pausable: layer 3, returns pausable thread table that can be continued later, has custom event queueing
  139. function newPausable(func)
  140. local t
  141. if type(func) == "table" and func.resume and func.status then
  142. t = func
  143. else
  144. checkTypes({func},{"function"})
  145. t = newCCThread(func)
  146. end
  147. local parent = {
  148. resume = t.resume,
  149. status = t.status,
  150. }
  151. t.events = {}
  152. t.paused = false
  153. t.presume = function(self,event,...)
  154. if self.paused then
  155. self.events[#self.events + 1] = {event,...}
  156. return true
  157. else
  158. return parent.resume(self,event,...)
  159. end
  160. end
  161. t.clearEvents = function(self)
  162. for i=1,#self.events do
  163. parent.resume(self,unpack(self.events[i]))
  164. end
  165. self.events = {}
  166. end
  167. t.setPause = function(self,pause)
  168. self.paused = pause
  169. if not pause then
  170. self:clearEvents()
  171. end
  172. end
  173. t.resume = t.presume --set resume to this layer
  174. return t
  175. end
  176. --dependency: newPausable
  177. --New Adv Thread: layer 4, adds random things to the thread table, like messages and data
  178. function newAdvThread(func)
  179. local t
  180. if type(func) == "table" and func.resume and func.status then
  181. t = func
  182. else
  183. checkTypes({func},{"function"})
  184. t = newPausable(func)
  185. end
  186. t.msgs = {}
  187. t.data = {}
  188. return t
  189. end
  190. --dependency: newAdvThread
  191. --New Windowed: layer 5, works with windows and redirects
  192. function newWindowed(func,win)
  193. local t
  194. if type(func) == "table" and func.resume and func.status then
  195. t = func
  196. else
  197. checkTypes({func},{"function"})
  198. t = newAdvThread(func)
  199. end
  200. local parent = {
  201. resume = t.resume,
  202. status = t.status,
  203. }
  204. t.used = false
  205. t.win = win
  206. t.term = t.win
  207. t.wresume = function(self,event,...)
  208. local w = term.current()
  209. term.redirect(self.term)
  210. local t = {parent.resume(self,event,...)}
  211. self.term = term.current()
  212. term.redirect(w)
  213. return unpack(t)
  214. end
  215. t.resume = t.wresume --set resume to this layer
  216. return t
  217. end
  218. --In these systems indexes WILL change, please use identifiers instead!
  219. --Please make sure func generates something continueable
  220. --new Basic System generates a layer 1 and 2 thread system with function for converting not compatible types of func
  221. function newBasicSystem(func,...)
  222. local args = {...}
  223. local threads = {}
  224. local thr
  225. for k,v in pairs(args) do
  226. if type(v) == "function" then
  227. thr = func(v)
  228. thr.id = k
  229. threads[#threads+1] = thr
  230. elseif type(v) == "table" and v.ccresume and v.status then
  231. v.id = k
  232. threads[#threads+1] = v
  233. end
  234. end
  235. return {
  236. threads = threads,
  237. anyended = false,
  238. allended = false,
  239. count = #threads,
  240. running = nil,
  241. tEnded = {}, --the t stands for table because ended is a function
  242. maxID = #threads + 1,
  243. onKill = delegate(),
  244. onAdd = delegate(),
  245. kill = function(self,i)
  246. local thr = self.threads[i]
  247. if thr then
  248. self.onKill(i) -- call onKill delegate
  249. self.tEnded[thr.id] = true --add this id to ended list
  250. table.remove(self.threads,i) --kill
  251. self.anyended = true --modify info
  252. local c = #self.threads
  253. self.count = c
  254. if c == 0 then
  255. self.allended = true
  256. end
  257. return true
  258. end
  259. return false
  260. end,
  261. resume = function(self,event)
  262. checkTypes({self,event},{"table","table"})
  263. local limit = #self.threads
  264. if limit == 0 then --check if threads is empty
  265. return false
  266. end
  267. local toKill = {}
  268. local v
  269. for i=1,limit do
  270. v = self.threads[i]
  271. if v then
  272. if v:status() == "suspended" then
  273. self.running = i
  274. local ok,err = v:resume(unpack(event))
  275. self.running = nil
  276. if not ok then error(err,0) end
  277. end
  278. if v:status() == "dead" then
  279. toKill[#toKill + 1] = i
  280. end
  281. end
  282. end
  283. if #toKill > 0 then
  284. for i=1,#toKill do
  285. local k = toKill[i]
  286. self:kill(k)
  287. end
  288. end
  289. return true
  290. end,
  291. add = function(self,thr)
  292. if type(thr) == "function" then
  293. thr = func(thr)
  294. end
  295. checkTypes({self,thr},{"table","table"})
  296. self.onAdd()
  297. local i = #self.threads+1
  298. local id = self.maxID
  299. thr.id = id
  300. self.maxID = id + 1
  301. self.threads[i] = thr
  302. self.count = i
  303. self.allended = false
  304. local ok,err = self.threads[i]:resume{}
  305. if not ok then error(err,0) end
  306. return id
  307. end,
  308. alive = function(self,i)
  309. checkTypes({self,i},{"table","number"})
  310. local thr = self.threads[i]
  311. return thr and thr:status() == "suspended" or false
  312. end,
  313. getID = function(self,i)
  314. local thr = self.threads[i]
  315. return thr and thr.id
  316. end,
  317. findIndex = function(self,id)
  318. local v
  319. for i=1,#self.threads do
  320. v = self.threads[i]
  321. if v and v.id == id then
  322. return i
  323. end
  324. end
  325. return false
  326. end,
  327. ended = function(self,id)
  328. return self.tEnded[id] or false
  329. end,
  330. }
  331. end
  332. --dependency: newBasicSystem and newThread
  333. --new C Thread System creates a thread system with layer 1
  334. function newCThreadSystem(...)
  335. return newBasicSystem(newThread,...)
  336. end
  337. --dependecy: newBasicSystem and newCCThread
  338. --new CC Thread System creates a thread system with layer 2
  339. function newCCThreadSystem(...)
  340. return newBasicSystem(newCCThread,...)
  341. end
  342. --dependancy: newBasicSystem
  343. --new Pausable system wf (with func: function for converting input functions to threads)
  344. function newPausableSystemWF(func,...)
  345. local t = newBasicSystem(func,...)
  346. t.setPause = function(self,i,bool)
  347. checkTypes({self,i,bool},{"table","number","boolean"})
  348. local thr = self.threads[i]
  349. if thr and thr.setPause then
  350. thr:setPause(bool)
  351. else
  352. return nil
  353. end
  354. end
  355. t.getPause = function(self,i)
  356. checkTypes({self,i},{"table","number"})
  357. local thr = self.threads[i]
  358. if thr and type(thr.paused) == "boolean" then
  359. return thr.paused
  360. else
  361. return nil
  362. end
  363. end
  364. t.queueEventAsync = function(self,i,eventdata)
  365. checkTypes({self,i,eventdata},{"table","number","table"})
  366. local thr = self.threads[i]
  367. if thr and thr.events then
  368. thr.events[#thr.events+1] = eventdata
  369. return true
  370. else
  371. return nil
  372. end
  373. end
  374. t.queueEventSync = function(self,i,eventdata)
  375. checkTypes({self,i,eventdata},{"table","number","table"})
  376. local ok = self:queueEventAsync(i,eventdata)
  377. local thr = self.threads[i]
  378. if ok and thr and type(thr.paused) == "boolean" and not thr.paused then
  379. thr:setPause(false) --simulate unpausing so he receives his events
  380. return true
  381. end
  382. end
  383. return t
  384. end
  385. --dependancy: newPausable and newPausableSystemWF
  386. --new Pausable System returns a layer 3 thread system
  387. function newPausableSystem(...)
  388. return newPausableSystemWF(newPausable,...)
  389. end
  390. --Advanced systems have changing IDs becuase you have data to identify threads
  391. --dependancy: newPausableSystemWF
  392. --new Adavnced System wf (see newPausableSystemWF)
  393. function newAdvancedSystemWF(func,...)
  394. local t = newPausableSystemWF(func,...)
  395. t.setData = function(self,i,key,value)
  396. checkTypes({self,i},{"table","number"})
  397. local thr = self.threads[i]
  398. if thr and thr.data then
  399. thr.data[key] = value
  400. return true
  401. end
  402. return false
  403. end
  404. t.getData = function(self,i,key)
  405. checkTypes({self,i},{"table","number"})
  406. local thr = self.threads[i]
  407. if thr and thr.data then
  408. return thr.data[key]
  409. end
  410. return nil
  411. end
  412. t.addMessage = function(self,i,msg)
  413. checkTypes({self,i,msg},{"table","number","table"})
  414. local thr = self.threads[i]
  415. if thr and thr.msgs then
  416. thr.msgs[#thr.msgs+1] = msg
  417. end
  418. end
  419. t.viewMessage = function(self,i)
  420. checkTypes({self,i},{"table","number"})
  421. local thr = self.threads[i]
  422. if thr and thr.msgs then
  423. return thr.msgs[1]
  424. end
  425. return false
  426. end
  427. t.getMessage = function(self,i)
  428. checkTypes({self,i},{"table","number"})
  429. local msg = self:viewMessage(i)
  430. if msg then --We don't need more checks, only remove it
  431. table.remove(self.threads[i].msgs,1)
  432. end
  433. return msg or false
  434. end
  435. t.viewMessages = function(self,i)
  436. checkTypes({self,i},{"table","number"})
  437. local thr = self.threads[i]
  438. if thr and thr.msgs then
  439. local t = {} --just a simple copy function, msgs will not have recursion or metatables, so it's enough
  440. for k,v in pairs(thr.msgs) do
  441. t[k] = v
  442. end
  443. return t
  444. end
  445. return false
  446. end
  447. t.getMessages = function(self,i)
  448. checkTypes({self,i},{"table","number"})
  449. local msgs = self:viewMessages(i)
  450. if msgs then
  451. self.threads[i].msgs = {}
  452. end
  453. return msgs or false
  454. end
  455. return t
  456. end
  457. --dependancy: newAdvThread and newAdvancedSystemWF
  458. --returns a layer 4 thread system
  459. function newAdvancedSystem(...)
  460. return newAdvancedSystemWF(newAdvThread,...)
  461. end
  462. --dependancy: newAdvancedSystemWF
  463. function newWindowedSystemWF(func,ue,me)
  464. ue = ue or usrEvents
  465. me = me or mouseEvents
  466. local t = newAdvancedSystemWF(func)
  467. t.ue = ue --user events
  468. t.me = me --mouse events
  469. t.resume = function(self,event)
  470. checkTypes({self,event},{"table","table"})
  471. local limit = #self.threads
  472. if limit == 0 then --check if threads is empty
  473. return false
  474. end
  475. local toRun = {}
  476. local ename = event[1]
  477. if ename and self.ue[ename] then
  478. if self.threads[self.selected] then
  479. toRun[1] = self.selected
  480. end
  481. elseif ename and self.me[ename] then
  482. local v = self.threads[self.selected]
  483. if v and v.win then
  484. --translate mouse events
  485. local wx,wy = v.win.getPosition()
  486. local ww,wh = v.win.getSize()
  487. local x,y = event[3],event[4]
  488. if x > wx and x < wx + ww and y > wy and y < wy + wh then --event is in the window's bounds
  489. event[3],event[4] = x-wx,y-wy
  490. toRun[1] = self.selected
  491. end
  492. end
  493. else
  494. local v
  495. for i=1,limit do
  496. v = self.threads[i]
  497. if v then
  498. if v:status() == "suspended" then
  499. toRun[#toRun+1] = i
  500. end
  501. end
  502. end
  503. end
  504. local toKill = {}
  505. if #toRun > 1 then
  506. local k,v
  507. for i=1,#toRun do
  508. k = toRun[i]
  509. v = self.threads[k]
  510. if v and v:status() == "suspended" then
  511. self.running = k
  512. local ok,err = v:resume(unpack(event))
  513. self.running = nil
  514. if not ok then error(err,0) end
  515. end
  516. if v and v:status() == "dead" then
  517. toKill[#toKill + 1] = i
  518. end
  519. end
  520. end
  521. for i=1,#toKill do
  522. self:killWindowed(i)
  523. end
  524. return true
  525. end
  526. t.addWindowed = function(self,func,win)
  527. if type(func) == "function" and win then
  528. func = newWindowed(func,win)
  529. end
  530. checkTypes({self,func},{"table","table"})
  531. return self:add(func)
  532. end
  533. t.killWindowed = function(self,i)
  534. local v = self.threads[i]
  535. if v and v.used then
  536. t:kill(i)
  537. else
  538. v.co = coroutine.create(coroutine.yield)
  539. end
  540. end
  541. t.getWindow = function(self,i)
  542. checkTypes({self,i},{"table","number"})
  543. local thr = self.threads[i]
  544. if thr then
  545. return thr.win
  546. end
  547. return false
  548. end
  549. t.selected = nil
  550. t.select = function(self,i)
  551. if self.threads[self.selected] then
  552. self.threads[self.selected].win.setVisible(false)
  553. end
  554. self.selected = i
  555. local thr = self.threads[self.selected]
  556. if thr then
  557. thr.win.setVisible(true)
  558. thr.used = true
  559. end
  560. end
  561. return t
  562. end
  563. --dependancy: newWindowedSystemWF and newWindowed
  564. function newWindowedSystem()
  565. return newWindowedSystemWF(newWindowed)
  566. end
  567.  
  568. --TODO: finally finish windowed and documentation
  569. --TODO: finish and test myshell
  570. --todo: fix name == nil in myshell
  571.  
  572. --TODO: add more comments to code - this won't disappear until a long time :( im lazy
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement