【问题标题】:How to create a timer with text mode control in the Tcl / Tk language如何在 Tcl/Tk 语言中创建带有文本模式控制的计时器
【发布时间】:2018-08-28 04:02:53
【问题描述】:

我想要的是类似于下面这个 bash shell 脚本的东西:

壳牌重击

#!/bin/bash
# shell timer
# Note: Do not measure time precisely because there is loss in calculations and other commands
# For a human being is something almost imperceptible, fortunately.
# ------------------------------------------------- -----------------------------
s=00
m=00
h=00

key=""

function _screen() {
clear
# Shows the elapsed time on the terminal screen and plays to the time.txt file always updating
printf "%02d:%02d:%02d" $h $m $s > ~/time.txt
  echo ":: 'p' to pause, 'c' to continue and 's' to exit ::"
}

function _time() {
    _screen
  sleep 1
  s=$((s+1))
  [ $s -eq 60 ] && m=$((m+1)) && s=00
  [ $m -eq 60 ] && h=$((h+1)) && m=00
}

function _pause() {
while :
do
    _screen
    sleep 1
    read key
    [ "$key" = "c" ] && clear && break
done
}

function _main() {

# Put the terminal in special character interpretation mode
stty -echo -icanon min 0

while :
do
    [ "$key" = "s" ] && break   
    [ "$key" = "p" ] && _pause
    _time
    read key
done

# Restores the default mode
stty sane

exit 0
}
_main

也许最明显的是将其转换为 Tcl/Tk。我什至尝试过,但我仍然没有成功。见:

Shell Tclsh

#!/usr/bin/env tclsh
# shell timer
# Note: Do not measure time precisely because there is loss in calculations and other commands
# For a human being is something almost imperceptible, fortunately.
# ------------------------------------------------- -----------------------------
set s 00
set m 00
set h 00

puts -nonewline ""
flush stdout
set key [gets stdin]

proc _screen{ } {
clear


set archive [open [pwd]/time.txt w]

# Shows the elapsed time on the terminal screen and plays to the time.txt file always updating
puts $archive "%02d:%02d:%02d" $h $m $s" 
puts -nonewline ":: 'p' to pause, 'c' to continue and 's' to exit ::"


}

proc _time{ } {
    _screen
  after 1000
  s=[expr s+1]
  if { $s -eq 60 } { m=[expr m+1] } { s=00 }
  if { $m -eq 60 } { h=[expr h+1] } { m=00 }
}

proc _pause{ } {
while { 1 } 
{
    _screen
  after 1000
    $argv key
    if { "$key" = "c" } { break }
  }
}

proc _main{ } {

# Put the terminal in special character interpretation mode
stty -echo -icanon min 0

while { 1 } 
{
    if { "$key" = "s" } { break }
    if { "$key" = "p" } { _pause }
    _time
    $argv key

}

# Restores the default mode
stty sane
close $archive
exit 0
}
after 1000 _main

我仍然致力于并为此努力与引用的示例相同 - bash 脚本。但不排除您可以推广的改进和建议。

我的想法是这样的:

如果这里有人知道并想分享这个想法,请随意。

【问题讨论】:

    标签: linux loops time tcl tk


    【解决方案1】:

    您的 Tcl 代码有几个问题:

    • proc _pause{ } { -- Tcl 对空格非常敏感,因此您需要将过程名称与参数列表分开
    • s=[expr s+1] -- 使用set 设置变量,你需要使用$s 来获取变量valueset s [expr {$s+1}] 或者在这种情况下使用incr 命令incr s
    • if { $s -eq 60 }if { "$key" = "s" } -- 请参阅 expr 手册页以获取正确的运算符。 你想要{$s == 60}{$key eq "s"}
    • stty -echo -icanon min 0 -- stty 是一个外部命令,所以你需要exec stty ...

    这些是主要的语法问题。可以改进您的缩进样式,以便您的代码可读且可维护。


    我认为这是一个有趣的挑战,所以我决定独立于您的代码来实现它。如果您有任何问题,请告诉我:

    #!/usr/bin/env tclsh
    
    set seconds 0
    set running true
    array set status {
        false "(paused)"
        true  "        "
    }
    
    #################################################################
    proc main {} {
        enableRaw
    
        puts "'p' to pause; 'c' to continue; 'q' to quit"
        every 1000 display_time
    
        chan configure stdout -buffering none
        chan configure stdin -blocking no -buffering none
        chan event stdin readable handleStdin
    
        vwait ::forever
    
        disableRaw
        puts ""
    }
    
    # ref https://wiki.tcl.tk/14693
    proc enableRaw {{channel stdin}} {
        exec /bin/stty raw -echo <@$channel
    }
    proc disableRaw {{channel stdin}} {
        exec /bin/stty -raw echo <@$channel
    }
    
    proc every {ms code} {
        after $ms [list every $ms $code]
        uplevel #0 $code
    }
    
    proc display_time {{event ""}} {
        global running seconds
        puts -nonewline "\r [format_time] $::status($running) "
        if {$running && $event eq ""} {incr seconds}
    }
    
    proc format_time {} {
        return [clock format $::seconds -format "%H:%M:%S" -gmt true]
    }
    
    proc handleStdin {} {
        set data [chan read stdin 1]
        switch -- $data {
            P - p {set ::running false; display_time}
            C - c {set ::running true;  display_time unpausing}
            Q - q {set ::forever "now"}
        }
    }
    
    #################################################################
    main
    

    【讨论】:

    • 你运行的是什么版本的 Tcl? puts [info patchlevel] -- 显然是旧版本:为什么?
    • 解决方案:将chan configure替换为fconfigure;将chan event 替换为fileevent;将chan read 替换为read
    • 我运行的是什么版本的 Tcl? Tcl/Tk 8.4 .我怀疑你的crash是同一个版本的Tcl的账号。
    • 这就是stty raw 想要解决的问题,所以你不需要按回车键
    【解决方案2】:

    这是对enableRawdisableRaw 的轻微修改,而不是execstty

    package require term::ansi::ctrl::unix
    
    proc enableRaw {} {
        term::ansi::ctrl::unix::raw
    }
    
    proc diableRaw {} {
        term::ansi::ctrl::unix::cooked
    }
    

    【讨论】:

    • 感谢您提出我的问题。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2021-05-20
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多