-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathUI.hs
492 lines (438 loc) · 21.7 KB
/
UI.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
{-# OPTIONS_GHC -cpp #-}
----------------------------------------------------------------------------------------------------
---- Ñáîð è îòîáðàæåíèå ñòàòèñòèêè ðàáîòû ïðîãðàììû (îáú¸ì îáðàáîòàííûõ äàííûõ, ñêîðîñòü è ò.ä.) ---
----------------------------------------------------------------------------------------------------
#ifdef FREEARC_GUI
module UI (module UI, module UIBase, module GUI) where
import GUI
#else
module UI (module UI, module UIBase, module CUI) where
import CUI
#endif
import Prelude hiding (catch)
import Control.Monad
import Control.Concurrent
import Data.IORef
import Data.Ratio
import Numeric (showFFloat)
import System.CPUTime (getCPUTime)
import System.IO
import System.IO.Unsafe
import System.Time
import Utils
import Errors
import Charsets
import Files
import FileInfo
import Compression (encode_method, showMem, getCompressionMem, getDecompressionMem)
import Options
import UIBase
-- |Îòìåòèòü íà÷àëî âûïîëíåíèÿ ïðîãðàììû
uiStartProgram = do
guiStartProgram
-- |Îòìåòèòü íà÷àëî âûïîëíåíèÿ êîìàíäû
uiStartCommand command = do
pauseAction =: guiPauseAtEnd
ref_command =: command
pause_before_exit =: opt_pause_before_exit command
display_option' =: opt_display command
refStartArchiveTime =:: getClockTime
-- Îòêðûòü ëîãôàéë è âûâåñòè â íåãî âûïîëíÿåìóþ êîìàíäó. Äëèííûå êîììåíòàðèè/ñïèñêè ôàéëîâ/èìåíà ôàéëîâ
-- íå äîëæíû ïîïàäàòü â ëîã-ôàéë, òàê ÷òî ìû îáðåçàåì ñïèñîê è âñå ñòðîêè â í¸ì äî 100 ýëåìåíòîâ
openLogFile (opt_logfile command)
curdir <- getCurrentDirectory
exe <- getExeName
printLog (curdir++">"++takeBaseName exe++" "++unwords(map (takeSome 100 "...")$ hidePasswords$ takeSome 100 ["..."]$ cmd_args command)++"\n")
-- Âûâåäåì âåðñèþ àðõèâàòîðà è èñïîëüçóåìûå äîïîëíèòåëüíûå îïöèè
let addArgs = cmd_additional_args command
once putHeader$ condPrintLine "h" aARC_HEADER
condPrintLine "o" (addArgs &&& "Using additional options: "++unwords(hidePasswords addArgs)++"\n")
myFlushStdout
-- |Îòìåòèòü íà÷àëî âûïîëíåíèÿ ïîäêîìàíäû
uiStartSubCommand command subCommand = do
ref_command =: subCommand
uiArcname =: cmd_arcname command
display_option' =: opt_display subCommand
-- |Îòìåòèòü íà÷àëî îáðàáîòêè î÷åðåäíîãî àðõèâà
uiStartArchive command @ Command {
opt_data_compressor = compressor
, opt_cache = cache
}
method = do
-- Çàïîìíèòü âðåìÿ íà÷àëà îáðàáîòêè àðõèâà è âûïîëíÿåìóþ êîìàíäó
refStartArchiveTime =:: getClockTime
ref_command =: command
display_option' =: opt_display command
uiMessage =: ""
guiStartArchive
-- Îñòàòîê ïðîöåäóðû íå íóæíî âûïîëíÿòü, åñëè ýòî ïîä-êîìàíäà (íàïðèìåð, òåñòèðîâàíèå ïîñëå àðõèâàöèè)
if cmd_subcommand command
then do condPrintLineNeedSeparator "" "\n"
else do
-- Âûâåñòè ñîîáùåíèå òèïà "Testing archive ..."
let cmd = cmd_name command
arcname = cmd_arcname command
uiArcname =: arcname
exist <- fileExist arcname
msg <- i18n (msgStartGUI cmd exist)
condPrintLine "G" $ (format msg arcname)
condPrintLine "a" $ (msgStart cmd exist) ++ arcname
condPrintLine "c" $ (method &&& " using "++encode_method method)
condPrintLine "ac" $ "\n"
when (cmdType cmd == ADD_CMD) $ do
condPrintLineLn "m" $
"Memory for compression "++showMem (getCompressionMem method)
++", decompression " ++showMem (getDecompressionMem method)
++", cache " ++showMem cache
-- Ñîõðàíèòü èíôîðìàöèþ äëÿ ïîñëåäóþùåãî èñïîëüçîâàíèÿ
ref_w0 =:: val warnings
ref_arcExist =: exist
-- |Îòìåòèòü íà÷àëî óïàêîâêè èëè ðàñïàêîâêè äàííûõ
uiStartProcessing filelist archive_total_bytes archive_total_compressed = do
refArchiveProcessingTime =: 0
command <- val ref_command
let cmd = cmd_name command
total_files' = i$ length filelist
total_bytes' = sum (map fiSize filelist)
ui_state = UI_State {
total_files = total_files'
, total_bytes = total_bytes'
, archive_total_bytes = archive_total_bytes
, archive_total_compressed = archive_total_compressed
, datatype = error "internal CUI error: datatype not initialized"
, uiFileinfo = Nothing
, files = 0
, bytes = 0
, cbytes = 0
, dirs = 0
, dir_bytes = 0
, dir_cbytes = 0
, fake_files = 0
, fake_bytes = 0
, fake_cbytes = 0
, algorithmsCount = error "internal CUI error: algorithmsCount not initialized"
, rw_ops = error "internal CUI error: rw_ops not initialized"
, r_bytes = error "internal CUI error: r_bytes not initialized"
, rnum_bytes = error "internal CUI error: rnum_bytes not initialized"
}
ref_ui_state =: ui_state
printLine$ msgDo cmd ++ show_files3 total_files' ++ ", "
++ show_bytes3 total_bytes'
-- Âûâîä ýòîãî "ðàçäåëèòåëÿ" ïîçâîëèò çàòåðåòü íà ýêðàíå ñòðî÷êó ñ òåêóùåé ñòàòèñòèêîé
printLineNeedSeparator $ "\r"++replicate 75 ' '++"\r"
when (opt_indicator command == "1") $ do
myPutStr$ ". Processed "
-- Ñëîæíûé progress indicator ó÷èòûâàåò òàêæå ïðîöåíò îáðàáîòàííûõ ôàéëîâ, ÷òî ïîçâîëÿåò åìó ðàáîòàòü ïëàâíåå
-- Äàííûå ñ ôàéëàìè ñìåøèâàþòñÿ èç ðàñ÷¸òà: ñêîðîñòü óïàêîâêè 1ìá/ñ, âðåìÿ îòêðûòèÿ ôàéëà - 10 ìñåê
let current bytes = do ui_state <- val ref_ui_state
return$ bytes + (bytes_per_sec `div` 100)*i (files ui_state)
total = do ui_state <- val ref_ui_state
return$ total_bytes ui_state + (bytes_per_sec `div` 100)*i (total_files ui_state)
uiStartProgressIndicator INDICATOR_FULL command current total
myFlushStdout
-- |Îòìåòèòü ñòàäèþ âûïîëíåíèÿ ïðîöåññà
uiStage msg = do
syncUI $ do
uiMessage =:: i18n msg
-- |Ñáðîñèòü ñ÷¸ò÷èê ïðîñêàíèðîâàííûõ ôàéëîâ
uiStartScanning = do
files_scanned =: 0
-- |Âûçûâàåòñÿ â õîäå ñêàíèðîâàíèÿ äèñêà, files - ñïèñîê ôàéëîâ, íàéäåííûõ â î÷åðåäíîì êàòàëîãå
uiScanning msg files = do -- Ïîêà ýòî ðàáîòàåò òîëüêî â GUI
#ifdef FREEARC_GUI
failOnTerminated
files_scanned += i(length files)
files_scanned' <- val files_scanned
msg <- i18n msg
uiStage$ format msg (show3 files_scanned')
#endif
return ()
-- |Îòìåòèòü íà÷àëî óïàêîâêè/ðàñïàêîâêè ôàéëîâ
uiStartFiles count = do
syncUI $ do
modifyIORef ref_ui_state $ \ui_state ->
ui_state { datatype = File
, algorithmsCount = count
, rw_ops = replicate count []
, r_bytes = 0
, rnum_bytes = 0
}
-- |Îòìåòèòü íà÷àëî óïàêîâêè/ðàñïàêîâêè êàòàëîãà àðõèâà
uiStartDirectory = do
syncUI $ do
modifyIORef ref_ui_state $ \ui_state ->
ui_state { datatype = Dir
, dirs = dirs ui_state + 1
, algorithmsCount = 0 }
-- |Îòìåòèòü íà÷àëî óïàêîâêè/ðàñïàêîâêè ñëóæåáíîé èíôîðìàöèè àðõèâà
uiStartControlData = do
syncUI $ do
modifyIORef ref_ui_state $ \ui_state ->
ui_state { datatype = CData
, algorithmsCount = 0 }
-- |Îòìåòèòü íà÷àëî óïàêîâêè/ðàñïàêîâêè ôàéëà
uiStartFile fileinfo = do
syncUI $ do
uiMessage =: (fpFullname.fiStoredName) fileinfo ++ (fiIsDir fileinfo &&& "/")
modifyIORef ref_ui_state $ \ui_state ->
ui_state { datatype = File
, uiFileinfo = Just fileinfo
, files = files ui_state + 1}
guiStartFile
-- |Îòêîððåêòèðîâàòü total_bytes â ui_state
uiCorrectTotal files bytes = do
when (files/=0 || bytes/=0) $ do
syncUI $ do
modifyIORef ref_ui_state $ \ui_state ->
ui_state { total_files = total_files ui_state + files
, total_bytes = total_bytes ui_state + bytes }
-- |Îòìåòèòü èìèòàöèþ îáðàáîòêè ôàéëîâ ñîãëàñíî ïðèëàãàåìîìó ñïèñêó
uiFakeFiles filelist compsize = do
let origsize = sum (map fiSize filelist)
syncUI $ do
modifyIORef ref_ui_state $ \ui_state ->
ui_state { datatype = File
, files = (files ui_state) + (i$ length filelist)
, fake_files = (fake_files ui_state) + (i$ length filelist)
, fake_bytes = (fake_bytes ui_state) + origsize
, fake_cbytes = (fake_cbytes ui_state) + compsize
}
uiUnpackedBytes origsize
uiCompressedBytes compsize
uiUpdateProgressIndicator origsize
-- |Îòìåòèòü, ÷òî áûëî îáðàáîòàíî ñòîëüêî-òî áàéò ñæàòûõ äàííûõ (íåâàæíî, ðåçóëüòàò
-- ëè ýòî óïàêîâêè, âõîäíûå äàííûå äëÿ ðàñïàêîâêè èëè ïðîñòî óïàêîâàííûå äàííûå, ïåðåäàííûå
-- èç ñòàðîãî àðõèâà â íîâûé áåç êàêîé-ëèáî ïåðåïàêîâêè)
uiCompressedBytes len = do
syncUI $ do
modifyIORef ref_ui_state $ \ui_state ->
case (datatype ui_state) of
File -> ui_state { cbytes = cbytes ui_state + len }
Dir -> ui_state { dir_cbytes = dir_cbytes ui_state + len }
CData -> ui_state
-- |Îòìåòèòü, ÷òî áûëî îáðàáîòàíî ñòîëüêî-òî áàéò ðàñïàêîâàííûõ äàííûõ (äàæå åñëè ðåàëüíî
-- ýòè áàéòû íèêòî â ãëàçà íå âèäåë, ïîñêîëüêó îíè áûëè ïåðåäàíû â ñæàòîì âèäå èç àðõèâà â àðõèâ)
uiUnpackedBytes len = do
syncUI $ do
modifyIORef ref_ui_state $ \ui_state ->
case (datatype ui_state) of
File -> ui_state { bytes = bytes ui_state + len }
Dir -> ui_state { dir_bytes = dir_bytes ui_state + len }
CData -> ui_state
-- |Îòìåòèòü íà÷àëî óïàêîâêè èëè ðàñïàêîâêè ñîëèä-áëîêà
uiStartDeCompression deCompression = do
x <- getCPUTime
newMVar (x,deCompression,[])
-- |Äîáàâèòü â ñïèñîê âðåìÿ ðàáîòû îäíîãî èç àëãîðèòìîâ â öåïî÷êå
-- (âû÷èñëåííîå â ñèøíîì òðåäå âðåìÿ ðàáîòû óïàêîâùèêà/ðàñïàêîâùèêà)
uiDeCompressionTime times t = do
modifyMVar_ times (\(x,y,ts) -> return (x, y, ts++[t]))
-- |Óïàêîâêà/ðàñïàêîâêà ñîëèä-áëîêà çàâåðøåíà - ïðîñóììèðîâàòü âðåìÿ ðàáîòû âñåõ òðåäîâ
-- èëè èñïîëüçîâàòü wall clock time, åñëè õîòÿ áû îäíî èç âîçâðàù¸ííûõ âðåì¸í == -1
uiFinishDeCompression times = do
(timeStarted, deCompression, results) <- takeMVar times
timeFinished <- getCPUTime
let deCompressionTimes = map snd3 results
refArchiveProcessingTime += {-if (all (>=0) deCompressionTimes) -- Commented out until all compression methods (lzma, grzip) will include timing for all threads
then sum deCompressionTimes
else-} i(timeFinished - timeStarted) / 1e12
let total_times = if (all (>=0) deCompressionTimes)
then " ("++showFFloat (Just 3) (sum deCompressionTimes) ""++" seconds)"
else ""
when (results>[]) $ do
debugLog0$ " Solid block "++deCompression++" results"++total_times
for results $ \(method,time,size) -> do
debugLog0$ " "++method++": "++show3 size++" bytes in "++showFFloat (Just 3) time ""++" seconds"
-- |Îáðàáîòêà î÷åðåäíîãî àðõèâà çàâåðøåíà -> íàïå÷àòàòü ñòàòèñòèêó è âåðíóòü å¸ âûçûâàþùåé ïðîöåäóðå
uiDoneArchive = do
command <- val ref_command
ui_state @ UI_State { total_files = total_files
, total_bytes = total_bytes
, files = files
, bytes = bytes
, cbytes = cbytes
, dirs = dirs
, dir_bytes = dir_bytes
, dir_cbytes = dir_cbytes
, fake_files = fake_files
, fake_bytes = fake_bytes
, fake_cbytes = fake_cbytes } <- val ref_ui_state
let cmd = cmd_name command
uiMessage =: ""
updateAllIndicators
uiDoneProgressIndicator
when (opt_indicator command=="2" && files-fake_files>0) $ do
myPutStrLn ""
printLineNeedSeparator "" -- íóæäà â ðàçäåëèòåëå ïåðåä âûâîäîì ñëåäóþùèõ ñòðîê èñ÷åçëà
-- Ñòàòèñòèêà ñæàòèÿ (íå âûâîäèòñÿ äëÿ ñóá-êîìàíä, ïîñêîëüêó òî÷íî òàêóþ æå ñòàòèñòèêó óæå íàïå÷àòàëà îñíîâíàÿ êîìàíäà)
unless (cmd_subcommand command) $ do
condPrintLineLn "f" $ left_justify 75 $ -- áåç äîïîëíèòåëüíûõ ïðîáåëîâ ìîæåò íå ïåðåïèñàòüñÿ ïîëíîñòüþ ïðåäûäóùàÿ ñòðîêà
msgDone cmd ++ show_files3 files ++ ", " ++ show_ratio cmd bytes cbytes
-- Íàïå÷àòàòü ñòàòèñòèêó ïî êàòàëîãó àðõèâà òîëüêî åñëè îí äîñòàòî÷íî âåëèê
when (dir_bytes>10^4) $ do
condPrintLine "d" $ "Directory " ++ (dirs>1 &&& "has " ++ show3 dirs ++ " chunks, ")
condPrintLineLn "d" $ show_ratio cmd dir_bytes dir_cbytes
-- Èíôîðìàöèÿ î âðåìåíè ðàáîòû è ñêîðîñòè óïàêîâêè/ðàñïàêîâêè
secs <- val refArchiveProcessingTime -- âðåìÿ, çàòðà÷åííîå íåïîñðåäñòâåííî íà óïàêîâêó/ðàñïàêîâêó
real_secs <- return_real_secs -- ïîëíîå âðåìÿ âûïîëíåíèÿ êîìàíäû íàä òåêóùèì àðõèâîì
condPrintLine "t" $ msgStat cmd ++ "time: "++(secs>0 &&& "cpu " ++ showTime secs ++ ", ")
condPrintLine "t" $ "real " ++ showTime real_secs
when (real_secs>=0.01) $ condPrintLine "t" $ ". Speed " ++ showSpeed (bytes-fake_bytes) real_secs
condPrintLineNeedSeparator "rdt" "\n"
myFlushStdout
resetConsoleTitle
return (1,files,bytes,cbytes)
-- |Âûçûâàåòñÿ ïîñëå âñåõ âñïîìîãàòåëüíûõ îïåðàöèé (äîáàâëåíèå recovery info, òåñòèðîâàíèå)
uiDoneArchive2 = do
command <- val ref_command
let cmd = cmd_name command
arcname = cmd_arcname command
w0 <- val ref_w0
w1 <- val warnings
let w = w1-w0 -- number of warnings while processing this archive
unlessM (val operationTerminated) $ do
arcExist <- val ref_arcExist
msg <- i18n (msgFinishGUI cmd arcExist w)
condPrintLine "G" (formatn msg [arcname, show w])
unless (cmd_subcommand command) $ do
condPrintLineNeedSeparator "" "\n\n"
-- |Âûïîëíåíèå ïîäêîìàíäû çàâåðøåíî
uiDoneSubCommand command subCommand results = do
ref_command =: command
display_option' =: opt_display command
-- |Âûïîëíåíèå êîìàíäû çàâåðøåíî, íàïå÷àòàòü ñóììàðíóþ ñòàòèñòèêó ïî âñåì îáðàáîòàííûì àðõèâàì
uiDoneCommand Command{cmd_name=cmd} totals = do
let sum4 (a0,b0,c0,d0) (a,b,c,d) = (a0+a,b0+b,c0+c,d0+d)
(counts, files, bytes, cbytes) = foldl sum4 (0,0,0,0) totals
when (counts>1) $ do
condPrintLine "s" $ "Total: "++show_archives3 counts++", "
++show_files3 files ++", "
++if (cbytes>=0)
then show_ratio cmd bytes cbytes
else show_bytes3 bytes
condPrintLineNeedSeparator "s" "\n\n\n"
-- |Çàâåðøèòü âûïîëíåíèå ïðîãðàììû
uiDoneProgram = do
condPrintLineNeedSeparator "" "\n"
guiDoneProgram
{-# NOINLINE uiStartProgram #-}
{-# NOINLINE uiStartArchive #-}
{-# NOINLINE uiStartProcessing #-}
{-# NOINLINE uiStartFile #-}
{-# NOINLINE uiCorrectTotal #-}
{-# NOINLINE uiUnpackedBytes #-}
{-# NOINLINE uiCompressedBytes #-}
{-# NOINLINE uiDoneArchive #-}
{-# NOINLINE uiDoneCommand #-}
----------------------------------------------------------------------------------------------------
---- Î÷åðåäü îïåðàöèé r/w, ïî êîòîðûì ïðè óïàêîâêå âû÷èñëÿåòñÿ èíäèêàòîð ïðîãðåññà -----------------
----------------------------------------------------------------------------------------------------
-- Äîáàâèòü îïåðàöèþ ÷òåíèÿ/çàïèñè â ãîëîâó ñïèñêà, ñëèâàÿ âìåñòå îïåðàöèè îäíîãî òèïà
add_Read a (UI_Write 0:UI_Read 0:ops) = (UI_Read a:ops) -- èçáàâèòüñÿ îò useless ïàðû r0+w0
add_Read a (UI_Read b:ops) = (UI_Read (a+b):ops)
add_Read a ops = (UI_Read a :ops)
add_Write a (UI_Write b:ops) = (UI_Write(a+b):ops)
add_Write a ops = (UI_Write a :ops)
-- |Àëãîðèòì íîìåð num â öåïî÷êå îáåùàåò çàïèñàòü bytes áàéò, ñîîòâåòñòâóþùèõ ïîñëåäíåìó áëîêó ïðî÷èòàííûõ
-- äàííûõ (ýòà îïåðàöèÿ "îáåùàíèÿ çàïèñè" ïîçâîëÿåò ïîääåðæèâàòü àêêóðàòíûé èíäèêàòîð ïðîãðåññà)
uiQuasiWriteData num bytes = do
-- Ðåàëèçàöèÿ óñòðîåíà òàê, ÷òî ïîñëåäíèì ïðî÷èòàííûì äàííûì ñîïîñòàâëÿþòñÿ bytes çàïèñàííûõ äàííûõ,
-- íî ïðè ýòîì îáùèé ðàçìåð çàïèñàííûõ äàííûõ íå èçìåíÿåòñÿ íè íà éîòó ;)
uiWriteData num bytes
uiReadData num 0
uiWriteData num (-bytes)
-- |Àëãîðèòì íîìåð num â öåïî÷êå çàïèñàë bytes áàéò
uiWriteData num bytes = do
UI_State {algorithmsCount=count, datatype=datatype} <- val ref_ui_state
when (datatype == File) $ do
-- Ñîõðàíèòü â ñïèñîê îïåðàöèé â/â îïåðàöèþ ÷òåíèÿ
when (num>=1 && num<count) $ do
syncUI $ do
ui_state @ UI_State {rw_ops=rw_ops0} <- val ref_ui_state
let rw_ops = updateAt num (add_Write bytes) rw_ops0
return $! length (take 4 (rw_ops!!num)) -- strictify operations list!
ref_ui_state =: ui_state {rw_ops=rw_ops}
-- |Àëãîðèòì íîìåð num â öåïî÷êå ïðî÷èòàë bytes áàéò
uiReadData num bytes = do
UI_State {algorithmsCount=count, datatype=datatype} <- val ref_ui_state
when (datatype == File) $ do
-- Ñîõðàíèòü â ñïèñîê îïåðàöèé â/â îïåðàöèþ çàïèñè
when (num>=1 && num<count) $ do
syncUI $ do
modifyIORef ref_ui_state $ \ui_state @ UI_State {rw_ops=rw_ops} ->
ui_state {rw_ops = updateAt num (add_Read bytes) rw_ops}
-- Îáíîâèòü èíäèêàòîð ïðîãðåññà, åñëè ýòî ïîñëåäíèé àëãîðèòì ñæàòèÿ â öåïî÷êå
when (num>=1 && num==count) $ do
unpBytes <- syncUI $ do
-- Ñîñòîÿíèå äî îáðàáîòêè ýòèõ áàéò
ui_state @ UI_State {r_bytes=r_bytes0, rnum_bytes=rnum_bytes0, rw_ops=rw_ops0} <- val ref_ui_state
-- Ê áàéòàì íà âõîäå àëãîðèòìà num äîáàâëÿåòñÿ bytes áàéò,
-- âûñ÷èòûâàåì êîëè÷åñòâî áàéò íà âõîäå ïåðâîãî àëãîðèòìà åñëè ýòîò áëîê íå ïîõîæ íà ïðîñòî çàãîëîâîê (bytes>16)
let rnum_bytes = rnum_bytes0+bytes
(r_bytes, rw_ops) = if bytes>16
then calc num (reverse rw_ops0) [] rnum_bytes
else (r_bytes0, rw_ops0)
ref_ui_state =: ui_state {r_bytes=r_bytes, rnum_bytes=rnum_bytes, rw_ops=rw_ops}
--for rw_ops $ \x -> print (reverse x)
--print (rnum_bytes0, bytes, r_bytes0, r_bytes-r_bytes0)
-- Âîçâðàùàåì êîëè÷åñòâî áàéò íà âõîäå ïåðâîãî àëãîðèòìà îòíîñèòåëüíî ïðåäûäóùåãî çíà÷åíèÿ ýòîé âåëè÷èíû
return (r_bytes-r_bytes0)
uiUpdateProgressIndicator (toRational unpBytes*9/10)
when (num==1) $ do -- 90% íà ïîñëåäíèé àëãîðèòì â öåïî÷êå è 10% íà ïåðâûé (÷òîáû ñãëàäèòü âûâîä äëÿ external compression and so on)
uiUpdateProgressIndicator (toRational bytes/10)
where
-- Ðåêóðñèâíî ïåðåñ÷èòàòü bytes áàéò íà âõîäå àëãîðèòìà num â êîëè÷åñòâî áàéò íà âõîäå àëãîðèòìà 1
-- Çàîäíî óæ îáíîâèòü î÷åðåäü îïåðàöèé, ïðîñóììèðîâàâ îïåðàöèè ïðåäøåñòâóþøèå òåêóùåé òî÷êå èíòåðåñà
calc 1 _ new_ops bytes = (bytes, []:new_ops)
calc num (old_op:old_ops) new_ops bytes =
-- Ïåðåñ÷èòàòü bytes áàéò íà âûõîäå àëãîðèòìà num-1 â áàéòû íà åãî âõîäå
let (new_bytes, new_op) = go 0 bytes (0,0) (smart_reverse old_op)
in calc (num-1) old_ops (reverse new_op:new_ops) new_bytes
-- Ðåâåðñèðóåì oplist èëè ïðîñòî çàìåíèì åãî íà äâå îïåðàöèè, åñëè â í¸ì áîëüøå 1000 ýëåìåíòîâ
-- (òî åñòü ñêîðåé âñåãî ýòî îïåðàöèè ïåðåä tempfile)
smart_reverse oplist
| length oplist < 1000 = reverse oplist
| otherwise = [UI_Read r, UI_Write w] where (r,w) = go oplist
go (UI_Read r:ops) = mapFst (+r) (go ops)
go (UI_Write w:ops) = mapSnd (+w) (go ops)
go [] = (0,0)
-- Ïåðåñ÷èòûâàåò çàïèñàííûå áàéòû (restW) â ïðî÷èòàííûå (totalR) ñîãëàñíî ïîñëåäîâàòåëüíîñòè îïåðàöèé ââîäà/âûâîäà
go totalR restW (rsum,wsum) ops@(UI_Read r:UI_Write w:rest_ops)
-- Åñëè ñëåäóþùèé êóñîê óïàêîâàííûõ äàííûõ áîëüøå íàøåãî îñòàòêà, òî óâåëè÷èâàåì totalR íà íåãî è äâèæåìñÿ äàëüøå
| w<restW = go (totalR+r) (restW-w) (rsum+r,wsum+w) rest_ops
-- Èíà÷å äåëèì åãî ïðîïîðöèîíàëüíî (r/w * restW) è äîáàâëÿåì ê totalR
| otherwise = (totalR + ((r*restW) `div` max w 1), UI_Read rsum:UI_Write wsum:ops)
-- Âñå ïðî÷èå âàðèàíòû
go totalR _ (rsum,wsum) ops = (totalR, UI_Read rsum:UI_Write wsum:ops)
----------------------------------------------------------------------------------------------------
---- Îáíîâëåíèå èíäèêàòîðà ïðîãðåññà ---------------------------------------------------------------
----------------------------------------------------------------------------------------------------
-- |Èíèöèàëèçèðîâàòü èíäèêàòîð ïðîãðåññà
uiStartProgressIndicator indType command bytes' total' = do
bytes <- bytes' 0; total <- total'
arcname <- val uiArcname
let cmd = cmd_name command
direction = if (cmdType cmd == ADD_CMD) then " => " else " <= "
indicator = select_indicator command total
aProgressIndicatorState =: (indicator, indType, arcname, direction, 0 :: Rational, bytes', total')
indicator_start_real_secs =:: return_real_secs
uiResumeProgressIndicator
-- |Âûâåñòè íà ýêðàí è â çàãîëîâîê îêíà èíäèêàòîð ïðîãðåññà (ñêîëüêî ïðîöåíòîâ äàííûõ óæå îáðàáîòàíî)
uiUpdateProgressIndicator add_b =
when (add_b/=0) $ do
-- Ìàëåíüêàÿ èíäåéñêàÿ õèòðîñòü: ýòà ôóíêöèÿ âûçûâàåòñÿ ÏÅÐÅÄ êàêîé-ëèáî îáðàáîòêîé
-- äàííûõ. Ïðè ýòîì ìû ñ÷èòàåì, ÷òî ïðåäûäóùèå äàííûå ê äàííîìó ìîìåíòó óæå îáðàáîòàíû è
-- ðàïîðòóåì îá ýòîì. Íîâûå æå äàííûå òîëüêî äîáàâëÿþòñÿ ê ñ÷¸ò÷èêó, íî íå âëèÿþò
-- íà âûâîäèìóþ ÑÅÉ×ÀÑ ñòàòèñòèêó. Âîò òàêèå âîò ïðèêîëû â íàøåì ãîðîäêå :)
syncUI $ do
(indicator, indType, arcname, direction, b, bytes', total') <- val aProgressIndicatorState
aProgressIndicatorState =: (indicator, indType, arcname, direction, b+toRational(add_b), bytes', total')
-- |Çàâåðøèòü âûâîä èíäèêàòîðà ïðîãðåññà
uiDoneProgressIndicator = do
uiSuspendProgressIndicator
aProgressIndicatorState =: (NoIndicator, undefined, undefined, undefined, undefined, undefined, undefined)
-- |Îáåðíóòü âûïîëíåíèå êîìàíäû â îòêðûòèå è çàêðûòèå èíäèêàòîðà ïðîãðåññà
uiWithProgressIndicator command arcsize action = do
uiStartProgressIndicator INDICATOR_PERCENTS command return (return arcsize)
ensureCtrlBreak "uiDoneProgressIndicator" uiDoneProgressIndicator action
{-# NOINLINE uiUpdateProgressIndicator #-}