Animacja ASCII Ball in Box

11

Napisz program, który wyświetla ruchomą piłkę ASCII *wewnątrz prostokąta (20 na 10 znaków łącznie z ramką). Piłka nie może przekroczyć prostokąta, np. Odbić się od ściany. Scena musi być czyszczona i przerysowywana co 0,1 sekundy, a piłka musi przesuwać się o 1 znak w kierunku xiy co klatkę. Wygrywa najkrótszy program napisany w dowolnym języku.

Przykładowe dane wyjściowe (ramka 1)

+------------------+
|*                 |
|                  |
|                  |
|                  |
|                  |
|                  |
|                  |
|                  |
+------------------+

Przykładowe dane wyjściowe (ramka 2)

+------------------+
|                  |
| *                |
|                  |
|                  |
|                  |
|                  |
|                  |
|                  |
+------------------+

Przykładowe dane wyjściowe (ramka 8)

+------------------+
|                  |
|                  |
|                  |
|                  |
|                  |
|                  |
|                  |
|       *          |
+------------------+

Przykładowe dane wyjściowe (ramka 9)

+------------------+
|                  |
|                  |
|                  |
|                  |
|                  |
|                  |
|        *         |
|                  |
+------------------+
Patrick Oscity
źródło
4
Czy wydruk 99 znaków nowego wiersza kwalifikuje się jako czyszczenie ekranu?
Ventero
Nie na moim monitorze z 1080x1920rozdzielczością :)
mellamokb

Odpowiedzi:

7

Ruby 1.9, 115 znaków

Logika ruch jest dość podobna do Danko jest odpowiedź .

Ta wersja została przetestowana tylko w systemie Linux.

p=0
loop{u=(?|+?\s*18+"|
")*8
u[165-21*(7-p%14).abs-(17-p%34).abs]=?*
p+=1
puts"\e[2J",r=?++?-*18+?+,u,r
sleep 0.1}
Ventero
źródło
Z Ruby 1.9.3 pod Windows 7, puts"\e[2J"po prostu drukuje ←[2J(i nową linię) na ekranie.
res
@res Zobacz moją edycję dla wersji, która powinna działać w systemie Windows (niestety nie mogę jej przetestować).
Ventero
Używanie clsnie działa dla mnie, ale system ("cls")działa.
res
@res używa `cls`. Dosłowne backtyki.
Mark Reed
@ MarkReed - to właśnie próbowałem napisać .. W każdym razie to nie działa. Windows wydaje się wymagać systemu („cls”).
res
4

PowerShell, 144 znaki

Na podstawie doskonałej odpowiedzi Joeya , wykorzystując fakt, że współrzędne kulkowe są funkcją indeksu klatki (i), więc jeśli masz coś takiego x=n-abs(n-(i mod (2*n))), x zmieni się z 0 na n, z powrotem na 0 itd.

for(){cls
($l="+$('-'*18)+")
7..0|%{$j=$_
"|$(-join(17..0|%{'* '[$j-[Math]::abs(7-$i%14)-or$_-[Math]::abs(17-$i%34)]}))|"}
$l;$i++;sleep -m 100}
Danko Durbić
źródło
Niezłe. Chociaż byłem trochę dumny z mojego, if(-1,18-eq$x){$a*=-1;$x+=2*$a}if(-1,8-eq$y){$b*=-1;$y+=2*$b}który zastąpił cztery ifs wcześniej ;-). Byłem jednak pewien, że musiała istnieć formuła.
Joey,
3

Python 2, 234

Jestem pewien, że można więcej grać w golfa, ale muszę iść, więc oto, co mam sofar. popracuję nad tym później

import os,time
a,b,c,d,e,n='+- |*\n'
w=d+c*18+d+n
s=a+b*18+a+n
x,y=0,0
g,h=17,7
j,k=1,1
while 1:
 if 0>x or x>g:j*=-1;x+=j
 if 0>y or y>h:k*=-1;y+=k
 os.system('cls');print s+w*y+d+c*x+e+c*(g-x)+d+n+w*(h-y)+s;x+=j;y+=k;time.sleep(0.1)

Uwaga: działa w konsoli poleceń systemu Windows. Inne systemy operacyjne mogą używać innych poleceń niż clsdo czyszczenia ekranu, npclear

marynarka
źródło
wykonuje print "\e[H\e[2J"pracę w systemie Windows?
Patrick Oscity,
@padde - Wydaje się, że nie działa, gdy uruchamiam program Ruby w systemie Windows 7 (zobacz mój komentarz do twojego postu).
res
3

Rubin ( 179 174 147)

EDIT pozbył się kilku dodatkowych znaków:

l=?++?-*18+?++?\n
c=?|+?\s*18+?|+?\n
p=22
u=v=1
loop{f=l+c*8+l
f[p]=?*
puts"\e[2J"+f
p+=(u=f[p+u]==' '?u:-u)+21*(v=f[p+21*v]==' '?v:-v)
sleep 0.1}

EDYCJA ogoliła niektóre znaki, teraz 174:

l="+#{'-'*18}+\n";f=l+"|#{' '*18}|\n"*8+l;x=y=u=v=1
loop{f[21*y+x]='*';$><<"\e[2J\e[f"+f;f[21*y+x]=' '
u=f[21*y+x+u]==' '?u:-u;v=f[21*(y+v)+x]==' '?v:-v
x+=u;y+=v;sleep 0.1}

Nie golfowany:

l="+#{'-'*18}+\n"           # top and bottom lines 
f=l+"|#{' '*18}|\n"*8+l     # complete box as string
x=y=u=v=1                   # x,y position; u,v next move
loop {                      #
  f[21*y+x]='*'             # add ball to box
  $> << "\e[2J\e[f"+f       # clear screen and print box with ball
  f[21*y+x]=' '             # remove ball from box
  u=f[21*y+x+u]==' '?u:-u   # next move in x direction
  v=f[21*(y+v)+x]==' '?v:-v # next move in y direction
  x+=u                      # perform move
  y+=v                      # --"--
  sleep 0.1                 # .zZZ...
}                           #
Patrick Oscity
źródło
Nie powinno sleep .1też działać?
Joey,
Nie, nie w 1.9. SyntaxError: (irb):1: no .<digit> floating literal anymore; put 0 before dot. Ale wrócę do tego, jeśli będę tego potrzebować w przyszłości, dzięki!
Patrick Oscity,
Jeśli korzystasz z wersji 1.9, możesz użyć literałów char, aby skrócić kilka rzeczy, np. ?*Zamiast '*'itp.
Joey,
Używając Ruby 1.9.3 pod Win7, drukuje prostokąty jeden po drugim (poniżej), z których każdy ma ←[2J←[f+------------------+pierwszą linię.
res
następnie zmień $> << "\e[2J\e[f"+fna 'cls';$><<f(użyj backticksa cls)
Patrick Oscity,
2

JavaScript (275 283 )

s=Array(19).join(' ');n='\n';z=Array(9).join('|'+s+'|'+n).split(n,8);
x=y=0;a=b=1;t='+'+s.replace(/ /g,'-')+'+';
setInterval(function(){u=z[y];z[y]=u.replace(eval('/( {'+x+'}) /'),'$1*');
$('#o').text(t+n+z.join('\n')+n+t);z[y]=u;x+=a;y+=b;if(!x|x==17)a=-a;if(!y|y==7)b=-b},100)

Demo: http://jsfiddle.net/eKcfu/2/

Napisałem to dość szybko, więc jestem pewien, że jest jeszcze sporo do zrobienia. Sugestie są mile widziane :)

Edycja 1 : Usuń niepotrzebne osobne wywołanie funkcji, osadzone bezpośrednio w setInterval.

mellamokb
źródło
2

Haskell, 212 znaków

import System
main=mapM_ f$s 19`zip`s 9
s n=[2..n]++[n-1,n-2..3]++s n
f p=r"clear">>putStr(unlines[[" |-+*"!!(19#i+2*(9#j)+4*e((i,j)==p))|i<-[1..20]]|j<-[1..10]])>>r"sleep 0.1"
b#n=e$n<2||n>b
e=fromEnum
r=system

Używa bardziej funkcjonalnego podejścia do obliczania współrzędnych, tworząc nieskończoną sekwencję dla każdej współrzędnej osobno, a następnie łącząc je razem (linie 2 i 3). Reszta to rysowanie kodu.

hammar
źródło
2

PowerShell, 184 185 215

Tylko częściowo golfa, ponieważ mój mózg nie działa prawidłowo, gdy jestem chory ...

Jednak kilka fajnych sztuczek.

for($a=$b=1){cls
($l="+$('-'*18)+")
0..7|%{$j=$_
"|$(-join(0..17|%{'* '[$j-$y-or$_-$x]}))|"}
$l
$x+=$a;$y+=$b
if(-1,18-eq$x){$a*=-1;$x+=2*$a}if(-1,8-eq$y){$b*=-1;$y+=2*$b}sleep -m 100}

[Edytuj]: Pętla nad polem jest znacznie krótsza.

Joey
źródło
2

QBasic (QB64), 178 173 bajtów

a$="+------------------+
?a$
for c=1to 8
?"|                  |
next
?a$
do
x=19-abs(17-(i mod 34))
y=9-abs(7-(i mod 14))
locate y,x
?"*
_delay .1
locate y,x
?" "
i=i+1
loop

-5 bajtów dzięki DLosc

pustkowie
źródło
Ładny! Dla nieskończonej pętli, można użyć DO ... LOOPw miejscu WHILE 1 ... WENDi zaoszczędzić 5 bajtów.
DLosc
1

Perl 5, 141 znaków

print"\e[H\e[2J",$h="+"."-"x18 ."+
",(map{"|".$"x$q,(abs$t%14-7)-$_?$":"*",$"x(17-$q),"|
"}0..7),$h,select'','','',0.1while$q=abs$t++%34-17,1

Nie uruchamia się w lewym górnym rogu, jak przykładowe wyjście, ale nie jest to wymagane.

Kevin Reid
źródło
1

Ruby 1.9, 162 znaków

35 znaków wstydzi się odpowiedzi @ Ventero, ale byłem pod wrażeniem, że udało mi się zejść tak daleko, wciąż stosując stosunkowo proste podejście do rzeczywistej logiki. ^ [To dosłowny ESC (1 znak).

x=y=0
v=h=1
s=' '
loop{puts"^[[2J"+b="+#{?-*18}+",*(0..7).map{|i|"|#{i!=y ?s*18:s*x+?*+s*(17-x)}|"},b
y+=v
y+=v*=-1if y<0||y>7
x+=h
x+=h*=-1if x<0||x>17
sleep 0.1}
Mark Reed
źródło
1

R, 233 znaków

s=c("\n+",rep("-",18),"+");for (j in 1:8){cat(s,sep="");cat(rep(c("\n|",rep("",17),"|"),j-1));cat(c("\n|",rep(" ",j-1),"*",rep(" ",18-j),"|"),sep="");cat(rep(c("\n|",rep("",17),"|"),8-j));cat(s,sep="");Sys.sleep(0.1);system("clear")}
Paolo
źródło
1

Kolejny wpis bash - 213 204 znaków

Naprawdę nie jest zdobywcą nagrody, ale i tak było fajnie. Do rysowania używa sekwencji znaków vt100. (kod podany tutaj używa 215 znaków dla czytelności, 2 znaki można usunąć poprzez zmianę znaczenia, np. „*” -> \ *

e(){ printf "\e[$1";}
e 2J;e H
h='+------------------+'
echo $h
for((;++j<9;));do printf '|%18s|\n';done
echo $h
e '3;2H*'
while :;do
e 'D '
((i/17%2))&&e D||e C
((++i/7%2))&&e A||e B
e 'D*'
sleep .1
done
Dan Andreatta
źródło
1

PowerShell, 139 bajtów

Zainspirowany Danko Durbić za odpowiedź .

for(){cls
,"+$('-'*18)+
"*2-join("|$(' '*18)|
"*8-replace"^(\W{$(21*[Math]::abs(7-$i%14)+[Math]::abs(17-$i++%34))}.) ",'$1*')
sleep -m 100}

Ten skrypt używa -replaceoperatora do rysowania *wewnątrz prostokąta.

Mniej golfowy skrypt wyjaśniający, jak to działa:

for(){
    cls
    $y=[Math]::abs(7-$i%14)
    $x=[Math]::abs(17-$i++%34)
    $b="+$('-'*18)+`n"
    $m="|$(' '*18)|`n"*8-replace"^(\W{$(21*$y+$x)}.) ",'$1*'
    ,$b*2-join($m)          # draw it
    sleep -m 100
}
mazzy
źródło
0

Bash 278 300, 296

h="+------------------+"
w="|                  |"
g(){
echo -e "\e[$2;$1H$3"
}
g 1 1 $h
for i in {1..8}
do
echo "$w"
done
echo $h
x=4
y=4
p=1
q=1
for((;;))
do
((x==19&&p==1||x==2&&p==-1))&&p=-$p
((y==9&&q==1||y==2&&q==-1))&&q=-$q
x=$((x+p))
y=$((y+q))
g $x $y \*
sleep .1
g $x $y " "
done

Symbol \ e w linii echo -e "\e[$2;$1H$3"może być wygenerowany przez

echo -e "\x1b"

zastąpić to. Jako binarny 0x1b jest on o 3 znaki krótszy; Liczę tylko 1 dla „\ e”, ponieważ tylko oprogramowanie do układania zmusza mnie do użycia \e.

nieznany użytkownik
źródło
Anonimowy użytkownik zasugerował zmiany, aby usunąć znajdujące się w nich $symbole ((...))i zastąpić x=$(($x+$p))je ((x+=p))i podobnie w poniższym wierszu. (Zasugerowali także użycie \eznaku ucieczki).
Peter Taylor
Sugerowałbym ponadto, że \*może działać jako zamiennik "*".
Peter Taylor
@PeterTaylor: Dzięki anonimowemu użytkownikowi. Włączyłem sugestię nr 1 i używam nr 2 w zmodyfikowanej formie i twojej.
użytkownik nieznany
@userunknown: Repozytorium 1,337musi zostać na stałe zablokowane: P
mellamokb
1
@mellamokb: Nie ma.
użytkownik nieznany
0

TI Basic, 169 167 bajtów

1→X
1→Y
1→S
1→T
While 1
ClrHome
Output(Y,X,"*
S(X>1 and X<20)+(X=1)-(X=20→S
T(Y>1 and Y<10)+(Y=1)-(Y=10→T
X+S→X
Y+T→Y
For(N,1,20,19
For(O,1,10,9
Output(O,N,"+
End
End
For(N,2,19
For(O,1,10,9
Output(O,N,"-
End
End
For(O,2,9
Output(O,1,"|
Output(O,20,"|
End
End

Strasznie wolne, ale działa.

faza
źródło
(X=20)→Smoże być (X=20→Si możesz skorzystać z Ans, aby zaoszczędzić kilka bajtów w końcowym For (pętla. Mam pozytywne linie 8 i 9 można grać w golfa, ale nie jestem pewien, jak w tej chwili.
MI Wright
@MIWright Gdzie mam umieścić Ans?
faza
"| \ Output(O,1,Ans \ Output(O,20,Anspowinno działać.
MI Wright,
@MIWright Ale czy to nie zastępowanie ciągu dwóch znaków ciągiem trzech znaków? I dodanie ciągu dwóch znaków powyżej, co daje +4?
faza
Co masz na myśli? Ansto jednobajtowy token wpisany za pomocą 2nd (-). W rezultacie zaoszczędzisz jeden bajt, ponieważ |token ma dwa bajty na kalkulatorze.
MI Wright,
0

PHP, 196 186 148 bajtów

Usunąłem, unikając przepełnienia liczb całkowitych, aby zaoszczędzić 6 bajtów. Będzie działać przez 29 miliardów lat, zanim nastąpi przelanie; wciąż 6,8 lat w systemie 32-bitowym. Powiedziałbym, że jest to do przyjęcia.

Obliczanie pozycji zamiast jej korygowania znacznie oszczędzało, przygotowując kompletny szablon na raz inną partię.

for(;++$i;$s[-3-21*abs($i%14-7)-abs($i%34-17)]="*",print$f(_,9e5,"
").$s.$z,usleep(1e5))$s=($f=str_pad)($z=$f("+",19,"-")."+
",189,$f("|",19)."|
");
").$z;

Biegnij z -nr . Wymaga PHP 7.1.

podział :

for(;++$i;      # infinite loop:
                    # 2. set asterisk at appropriate position
    $s[-3-21*abs($i%14-7)-abs($i%34-17)]="*";
                    # 3. clear screen: print 900k newlines
    print$f(_,9e5*0+2,"\n")
            .$s.$z, # 4. print field
    usleep(1e5))    # 5. sleep 100000 microseconds
                    # 1. create template
    $s=($f=str_pad)($z=$f("+",19,"-")."+\n",189,$f("|",19)."|\n");
Tytus
źródło