xref: /titanic_54/usr/src/cmd/ast/libshell/common/scripts/termclock.sh (revision 906afcb89d0412cc073b95c2d701a804a8cdb62c)
1*906afcb8SAndy Fiddaman#!/usr/bin/ksh93
2*906afcb8SAndy Fiddaman
3*906afcb8SAndy Fiddaman#
4*906afcb8SAndy Fiddaman# CDDL HEADER START
5*906afcb8SAndy Fiddaman#
6*906afcb8SAndy Fiddaman# The contents of this file are subject to the terms of the
7*906afcb8SAndy Fiddaman# Common Development and Distribution License (the "License").
8*906afcb8SAndy Fiddaman# You may not use this file except in compliance with the License.
9*906afcb8SAndy Fiddaman#
10*906afcb8SAndy Fiddaman# You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
11*906afcb8SAndy Fiddaman# or http://www.opensolaris.org/os/licensing.
12*906afcb8SAndy Fiddaman# See the License for the specific language governing permissions
13*906afcb8SAndy Fiddaman# and limitations under the License.
14*906afcb8SAndy Fiddaman#
15*906afcb8SAndy Fiddaman# When distributing Covered Code, include this CDDL HEADER in each
16*906afcb8SAndy Fiddaman# file and include the License file at usr/src/OPENSOLARIS.LICENSE.
17*906afcb8SAndy Fiddaman# If applicable, add the following below this CDDL HEADER, with the
18*906afcb8SAndy Fiddaman# fields enclosed by brackets "[]" replaced with your own identifying
19*906afcb8SAndy Fiddaman# information: Portions Copyright [yyyy] [name of copyright owner]
20*906afcb8SAndy Fiddaman#
21*906afcb8SAndy Fiddaman# CDDL HEADER END
22*906afcb8SAndy Fiddaman#
23*906afcb8SAndy Fiddaman
24*906afcb8SAndy Fiddaman#
25*906afcb8SAndy Fiddaman# Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved.
26*906afcb8SAndy Fiddaman#
27*906afcb8SAndy Fiddaman
28*906afcb8SAndy Fiddaman#
29*906afcb8SAndy Fiddaman# termclock - a simple analog clock for terminals
30*906afcb8SAndy Fiddaman#
31*906afcb8SAndy Fiddaman
32*906afcb8SAndy Fiddaman# Solaris needs /usr/xpg6/bin:/usr/xpg4/bin because the tools in /usr/bin are not POSIX-conformant
33*906afcb8SAndy Fiddamanexport PATH=/usr/xpg6/bin:/usr/xpg4/bin:/bin:/usr/bin
34*906afcb8SAndy Fiddaman
35*906afcb8SAndy Fiddaman# Make sure all math stuff runs in the "C" locale to avoid problems
36*906afcb8SAndy Fiddaman# with alternative # radix point representations (e.g. ',' instead of
37*906afcb8SAndy Fiddaman# '.' in de_DE.*-locales). This needs to be set _before_ any
38*906afcb8SAndy Fiddaman# floating-point constants are defined in this script).
39*906afcb8SAndy Fiddamanif [[ "${LC_ALL}" != "" ]] ; then
40*906afcb8SAndy Fiddaman    export \
41*906afcb8SAndy Fiddaman        LC_MONETARY="${LC_ALL}" \
42*906afcb8SAndy Fiddaman        LC_MESSAGES="${LC_ALL}" \
43*906afcb8SAndy Fiddaman        LC_COLLATE="${LC_ALL}" \
44*906afcb8SAndy Fiddaman        LC_CTYPE="${LC_ALL}"
45*906afcb8SAndy Fiddaman        unset LC_ALL
46*906afcb8SAndy Fiddamanfi
47*906afcb8SAndy Fiddamanexport LC_NUMERIC=C
48*906afcb8SAndy Fiddaman
49*906afcb8SAndy Fiddamanfunction fatal_error
50*906afcb8SAndy Fiddaman{
51*906afcb8SAndy Fiddaman	print -u2 "${progname}: $*"
52*906afcb8SAndy Fiddaman	exit 1
53*906afcb8SAndy Fiddaman}
54*906afcb8SAndy Fiddaman
55*906afcb8SAndy Fiddaman# cache tput values (to avoid |fork()|'ing a "tput" child every second)
56*906afcb8SAndy Fiddamanfunction tput_cup
57*906afcb8SAndy Fiddaman{
58*906afcb8SAndy Fiddaman	# static variable as cache for "tput_cup"
59*906afcb8SAndy Fiddaman	typeset -S -A tput_cup_cache
60*906afcb8SAndy Fiddaman
61*906afcb8SAndy Fiddaman	integer y="$1" x="$2"
62*906afcb8SAndy Fiddaman	nameref c="tput_cup_cache[\"${y}_${x}\"]"
63*906afcb8SAndy Fiddaman
64*906afcb8SAndy Fiddaman	if [[ "$c" == "" ]] ; then
65*906afcb8SAndy Fiddaman		# fast path for known terminal types
66*906afcb8SAndy Fiddaman		if [[ ${TERM} == ~(Elr)(vt100|vt220|xterm|xterm-color|dtterm) ]] ; then
67*906afcb8SAndy Fiddaman			c="${ printf "\E[%d;%dH" y+1 x+1 ; }"
68*906afcb8SAndy Fiddaman		else
69*906afcb8SAndy Fiddaman			c="${ tput cup $y $x ; }"
70*906afcb8SAndy Fiddaman		fi
71*906afcb8SAndy Fiddaman	fi
72*906afcb8SAndy Fiddaman
73*906afcb8SAndy Fiddaman	print -r -n -- "$c"
74*906afcb8SAndy Fiddaman	return 0
75*906afcb8SAndy Fiddaman}
76*906afcb8SAndy Fiddaman
77*906afcb8SAndy Fiddaman# Get terminal size and put values into a compound variable with the integer
78*906afcb8SAndy Fiddaman# members "columns" and "lines"
79*906afcb8SAndy Fiddamanfunction get_term_size
80*906afcb8SAndy Fiddaman{
81*906afcb8SAndy Fiddaman	nameref rect=$1
82*906afcb8SAndy Fiddaman
83*906afcb8SAndy Fiddaman	rect.columns=${ tput cols ; } || return 1
84*906afcb8SAndy Fiddaman	rect.lines=${ tput lines ; }  || return 1
85*906afcb8SAndy Fiddaman
86*906afcb8SAndy Fiddaman	return 0
87*906afcb8SAndy Fiddaman}
88*906afcb8SAndy Fiddaman
89*906afcb8SAndy Fiddamanfunction draw_clock
90*906afcb8SAndy Fiddaman{
91*906afcb8SAndy Fiddaman	float angle a
92*906afcb8SAndy Fiddaman	float x y
93*906afcb8SAndy Fiddaman	integer i=1
94*906afcb8SAndy Fiddaman
95*906afcb8SAndy Fiddaman	for(( angle=0.0 ; angle < 360. ; angle+=6 )) ; do
96*906afcb8SAndy Fiddaman		((
97*906afcb8SAndy Fiddaman			a=angle/360.*(2*M_PI) ,
98*906afcb8SAndy Fiddaman
99*906afcb8SAndy Fiddaman			x=clock.len_x*cos(a) ,
100*906afcb8SAndy Fiddaman			y=clock.len_y*sin(a)
101*906afcb8SAndy Fiddaman		))
102*906afcb8SAndy Fiddaman
103*906afcb8SAndy Fiddaman		tput_cup $(( y+clock.middle_y )) $(( x+clock.middle_x ))
104*906afcb8SAndy Fiddaman
105*906afcb8SAndy Fiddaman		# add "mark" every 30 degrees
106*906afcb8SAndy Fiddaman		if (( int(angle)%30 == 0 )) ; then
107*906afcb8SAndy Fiddaman			print -r -n "$(((++i)%12+1))"
108*906afcb8SAndy Fiddaman		else
109*906afcb8SAndy Fiddaman			print -r -n "x"
110*906afcb8SAndy Fiddaman		fi
111*906afcb8SAndy Fiddaman	done
112*906afcb8SAndy Fiddaman	return 0
113*906afcb8SAndy Fiddaman}
114*906afcb8SAndy Fiddaman
115*906afcb8SAndy Fiddamanfunction draw_hand
116*906afcb8SAndy Fiddaman{
117*906afcb8SAndy Fiddaman	float   angle="$1" a
118*906afcb8SAndy Fiddaman	typeset ch="$2"
119*906afcb8SAndy Fiddaman	float   length="$3"
120*906afcb8SAndy Fiddaman	float   x y
121*906afcb8SAndy Fiddaman
122*906afcb8SAndy Fiddaman	(( a=angle/360.*(2*M_PI) ))
123*906afcb8SAndy Fiddaman
124*906afcb8SAndy Fiddaman	for (( s=0.0 ; s < 10. ; s+=0.5 )) ; do
125*906afcb8SAndy Fiddaman		((
126*906afcb8SAndy Fiddaman			x=(clock.len_x*(s/10.)*(length/100.))*cos(a) ,
127*906afcb8SAndy Fiddaman			y=(clock.len_y*(s/10.)*(length/100.))*sin(a)
128*906afcb8SAndy Fiddaman		))
129*906afcb8SAndy Fiddaman
130*906afcb8SAndy Fiddaman		tput_cup $(( y+clock.middle_y )) $(( x+clock.middle_x ))
131*906afcb8SAndy Fiddaman		print -r -n -- "${ch}"
132*906afcb8SAndy Fiddaman	done
133*906afcb8SAndy Fiddaman	return 0
134*906afcb8SAndy Fiddaman}
135*906afcb8SAndy Fiddaman
136*906afcb8SAndy Fiddamanfunction draw_clock_hand
137*906afcb8SAndy Fiddaman{
138*906afcb8SAndy Fiddaman	nameref hand=$1
139*906afcb8SAndy Fiddaman	draw_hand $(( 360.*(hand.val/hand.scale)-90. )) "${hand.ch}" ${hand.length}
140*906afcb8SAndy Fiddaman	return 0
141*906afcb8SAndy Fiddaman}
142*906afcb8SAndy Fiddaman
143*906afcb8SAndy Fiddamanfunction clear_clock_hand
144*906afcb8SAndy Fiddaman{
145*906afcb8SAndy Fiddaman	nameref hand=$1
146*906afcb8SAndy Fiddaman	draw_hand $(( 360.*(hand.val/hand.scale)-90. )) " " ${hand.length}
147*906afcb8SAndy Fiddaman	return 0
148*906afcb8SAndy Fiddaman}
149*906afcb8SAndy Fiddaman
150*906afcb8SAndy Fiddamanfunction main_loop
151*906afcb8SAndy Fiddaman{
152*906afcb8SAndy Fiddaman	typeset c
153*906afcb8SAndy Fiddaman
154*906afcb8SAndy Fiddaman	# note: we can't use subshells when writing to the double-buffer file because this
155*906afcb8SAndy Fiddaman	# will render the tput value cache useless
156*906afcb8SAndy Fiddaman	while true ; do
157*906afcb8SAndy Fiddaman		if ${init_screen} ; then
158*906afcb8SAndy Fiddaman			init_screen="false"
159*906afcb8SAndy Fiddaman
160*906afcb8SAndy Fiddaman			get_term_size termsize || fatal_error $"Couldn't get terminal size."
161*906afcb8SAndy Fiddaman
162*906afcb8SAndy Fiddaman			((
163*906afcb8SAndy Fiddaman				clock.middle_x=termsize.columns/2.-.5 ,
164*906afcb8SAndy Fiddaman				clock.middle_y=termsize.lines/2.-.5 ,
165*906afcb8SAndy Fiddaman				clock.len_x=termsize.columns/2-2 ,
166*906afcb8SAndy Fiddaman				clock.len_y=termsize.lines/2-2 ,
167*906afcb8SAndy Fiddaman			))
168*906afcb8SAndy Fiddaman
169*906afcb8SAndy Fiddaman			{
170*906afcb8SAndy Fiddaman				clear
171*906afcb8SAndy Fiddaman				draw_clock
172*906afcb8SAndy Fiddaman			} >&6
173*906afcb8SAndy Fiddaman		fi
174*906afcb8SAndy Fiddaman
175*906afcb8SAndy Fiddaman		{
176*906afcb8SAndy Fiddaman			(( ${ date +"hours.val=%H , minutes.val=%M , seconds.val=%S" ; } ))
177*906afcb8SAndy Fiddaman
178*906afcb8SAndy Fiddaman			# small trick to get a smooth "analog" flair
179*906afcb8SAndy Fiddaman			((
180*906afcb8SAndy Fiddaman				hours.val+=minutes.val/60. ,
181*906afcb8SAndy Fiddaman				minutes.val+=seconds.val/60.
182*906afcb8SAndy Fiddaman			))
183*906afcb8SAndy Fiddaman
184*906afcb8SAndy Fiddaman			draw_clock_hand seconds
185*906afcb8SAndy Fiddaman			draw_clock_hand minutes
186*906afcb8SAndy Fiddaman			draw_clock_hand hours
187*906afcb8SAndy Fiddaman
188*906afcb8SAndy Fiddaman			# move cursor to home position
189*906afcb8SAndy Fiddaman			tput_cup 0 0
190*906afcb8SAndy Fiddaman		} >&6
191*906afcb8SAndy Fiddaman
192*906afcb8SAndy Fiddaman		6<#((0))
193*906afcb8SAndy Fiddaman		cat <&6
194*906afcb8SAndy Fiddaman
195*906afcb8SAndy Fiddaman		redirect 6<&- ;  rm -f "${scratchfile}" ; redirect 6<> "${scratchfile}"
196*906afcb8SAndy Fiddaman
197*906afcb8SAndy Fiddaman		c="" ; read -r -t ${update_interval} -N 1 c
198*906afcb8SAndy Fiddaman		if [[ "$c" != "" ]] ; then
199*906afcb8SAndy Fiddaman			case "$c" in
200*906afcb8SAndy Fiddaman				~(Fi)q | $'\E') return 0 ;;
201*906afcb8SAndy Fiddaman			esac
202*906afcb8SAndy Fiddaman		fi
203*906afcb8SAndy Fiddaman
204*906afcb8SAndy Fiddaman		{
205*906afcb8SAndy Fiddaman			clear_clock_hand hours
206*906afcb8SAndy Fiddaman			clear_clock_hand minutes
207*906afcb8SAndy Fiddaman			clear_clock_hand seconds
208*906afcb8SAndy Fiddaman		} >&6
209*906afcb8SAndy Fiddaman	done
210*906afcb8SAndy Fiddaman}
211*906afcb8SAndy Fiddaman
212*906afcb8SAndy Fiddamanfunction usage
213*906afcb8SAndy Fiddaman{
214*906afcb8SAndy Fiddaman	OPTIND=0
215*906afcb8SAndy Fiddaman	getopts -a "${progname}" "${termclock_usage}" OPT '-?'
216*906afcb8SAndy Fiddaman	exit 2
217*906afcb8SAndy Fiddaman}
218*906afcb8SAndy Fiddaman
219*906afcb8SAndy Fiddaman# program start
220*906afcb8SAndy Fiddamanbuiltin basename
221*906afcb8SAndy Fiddamanbuiltin cat
222*906afcb8SAndy Fiddamanbuiltin date
223*906afcb8SAndy Fiddamanbuiltin mktemp
224*906afcb8SAndy Fiddamanbuiltin rm
225*906afcb8SAndy Fiddaman
226*906afcb8SAndy Fiddamantypeset progname="${ basename "${0}" ; }"
227*906afcb8SAndy Fiddaman
228*906afcb8SAndy Fiddamanfloat -r M_PI=3.14159265358979323846
229*906afcb8SAndy Fiddaman
230*906afcb8SAndy Fiddaman# terminal size rect
231*906afcb8SAndy Fiddamancompound termsize=(
232*906afcb8SAndy Fiddaman	integer columns=-1
233*906afcb8SAndy Fiddaman	integer lines=-1
234*906afcb8SAndy Fiddaman)
235*906afcb8SAndy Fiddaman
236*906afcb8SAndy Fiddamantypeset init_screen="true"
237*906afcb8SAndy Fiddaman
238*906afcb8SAndy Fiddamancompound clock=(
239*906afcb8SAndy Fiddaman	float   middle_x
240*906afcb8SAndy Fiddaman	float   middle_y
241*906afcb8SAndy Fiddaman	integer len_x
242*906afcb8SAndy Fiddaman	integer len_y
243*906afcb8SAndy Fiddaman)
244*906afcb8SAndy Fiddaman
245*906afcb8SAndy Fiddaman
246*906afcb8SAndy Fiddaman# set clock properties
247*906afcb8SAndy Fiddamancompound seconds=(
248*906afcb8SAndy Fiddaman	float val
249*906afcb8SAndy Fiddaman	typeset ch
250*906afcb8SAndy Fiddaman	float   scale
251*906afcb8SAndy Fiddaman	integer length )
252*906afcb8SAndy Fiddamancompound minutes=(
253*906afcb8SAndy Fiddaman	float val
254*906afcb8SAndy Fiddaman	typeset ch
255*906afcb8SAndy Fiddaman	float   scale
256*906afcb8SAndy Fiddaman	integer length )
257*906afcb8SAndy Fiddamancompound hours=(
258*906afcb8SAndy Fiddaman	float val
259*906afcb8SAndy Fiddaman	typeset ch
260*906afcb8SAndy Fiddaman	float   scale
261*906afcb8SAndy Fiddaman	integer length )
262*906afcb8SAndy Fiddaman
263*906afcb8SAndy Fiddamanseconds.length=90 seconds.scale=60 seconds.ch=$"s"
264*906afcb8SAndy Fiddamanminutes.length=75 minutes.scale=60 minutes.ch=$"m"
265*906afcb8SAndy Fiddamanhours.length=50   hours.scale=12   hours.ch=$"h"
266*906afcb8SAndy Fiddaman
267*906afcb8SAndy Fiddamanfloat update_interval=0.9
268*906afcb8SAndy Fiddaman
269*906afcb8SAndy Fiddamantypeset -r termclock_usage=$'+
270*906afcb8SAndy Fiddaman[-?\n@(#)\$Id: termclock (Roland Mainz) 2009-12-02 \$\n]
271*906afcb8SAndy Fiddaman[-author?Roland Mainz <roland.mainz@nrubsig.org>]
272*906afcb8SAndy Fiddaman[-author?David Korn <dgk@research.att.com>]
273*906afcb8SAndy Fiddaman[+NAME?termclock - analog clock for terminals]
274*906afcb8SAndy Fiddaman[+DESCRIPTION?\btermclock\b is an analog clock for terminals.
275*906afcb8SAndy Fiddaman        The termclock program displays the time in analog or digital
276*906afcb8SAndy Fiddaman        form. The time is continuously updated at a frequency which
277*906afcb8SAndy Fiddaman        may be specified by the user.]
278*906afcb8SAndy Fiddaman[u:update?Update interval (defaults to 0.9 seconds).]:[interval]
279*906afcb8SAndy Fiddaman[+SEE ALSO?\bksh93\b(1), \bxclock\b(1)]
280*906afcb8SAndy Fiddaman'
281*906afcb8SAndy Fiddaman
282*906afcb8SAndy Fiddamanwhile getopts -a "${progname}" "${termclock_usage}" OPT ; do
283*906afcb8SAndy Fiddaman#	printmsg "## OPT=|${OPT}|, OPTARG=|${OPTARG}|"
284*906afcb8SAndy Fiddaman	case ${OPT} in
285*906afcb8SAndy Fiddaman		u)    update_interval=${OPTARG} ;;
286*906afcb8SAndy Fiddaman		*)    usage ;;
287*906afcb8SAndy Fiddaman	esac
288*906afcb8SAndy Fiddamandone
289*906afcb8SAndy Fiddamanshift $((OPTIND-1))
290*906afcb8SAndy Fiddaman
291*906afcb8SAndy Fiddaman# prechecks
292*906afcb8SAndy Fiddamanwhich tput >/dev/null   || fatal_error $"tput not found."
293*906afcb8SAndy Fiddaman(( update_interval >= 0. && update_interval <= 7200. )) || fatal_error $"invalid update_interval value."
294*906afcb8SAndy Fiddaman
295*906afcb8SAndy Fiddaman# create temporary file for double-buffering and register an EXIT trap
296*906afcb8SAndy Fiddaman# to remove this file when the shell interpreter exits
297*906afcb8SAndy Fiddamanscratchfile="${ mktemp -t "termclock.ppid${PPID}_pid$$.XXXXXX" ; }"
298*906afcb8SAndy Fiddaman[[ "${scratchfile}" != "" ]] || fatal_error $"Could not create temporary file name."
299*906afcb8SAndy Fiddamantrap 'rm -f "${scratchfile}"' EXIT
300*906afcb8SAndy Fiddamanrm -f "${scratchfile}" ; redirect 6<> "${scratchfile}" || fatal_error $"Could not create temporary file."
301*906afcb8SAndy Fiddaman
302*906afcb8SAndy Fiddaman# register trap to handle window size changes
303*906afcb8SAndy Fiddamantrap 'init_screen="true"' WINCH
304*906afcb8SAndy Fiddaman
305*906afcb8SAndy Fiddamanmain_loop
306*906afcb8SAndy Fiddaman
307*906afcb8SAndy Fiddaman# exiting - put cursor below clock
308*906afcb8SAndy Fiddamantput_cup $((termsize.lines-2)) 0
309*906afcb8SAndy Fiddaman
310*906afcb8SAndy Fiddamanexit 0
311*906afcb8SAndy Fiddaman# EOF.
312