This is purely just a little bit of fun with awk. I realize this this isn't that constructive so please remove if need be.
Your goal:
Create a one line awk script that generates a diamond shape of any
size. Both the size of the diamond (measured by its middle line) and
the character used ("*" in the examples) should be stored in variables
that can be changed easily. Have fun!
My solution is below - I didn't spend too long on it so I imagine it can be improved a lot more.
Obviously DON'T LOOK HERE if you want to try this yourself from scratch without any clue!:
Thanks pilnet101 . Basically it increases the line length by 2 every line and memorizes the line. Then at maximum length, the loop starts to decrement, rather than increment and only prints the lines memorized in the increment phase.
With regards to the setting of the quotient q and the line n where the maximum line length occurs, and using the name m for maximum line length rather than l which is easily confused with the number 1 ), perhaps this is a bit clearer:
This isn't quite as compact as Scrutinizer's approach (and calculates each line as it goes instead of memorizing lines), but there are some fun games you can play with this one using multi-character strings that produce different artifacts with Scrutinizer's code and with pilnet101's code:
#!/bin/ksh
IAm=${0##*/}
if [ $# -gt 2 ]
then printf 'USAGE: %s [size [character]]
DESCRIPTION: Print a diamond of "character" characters that is "size"
characters wide at the widest point.
OPERANDS: size The size of the diamond. (Default 15).
character The character to print. (Default "*".)
EXAMPLES: In addition to the obvious single character "character"
operands, try commands like:
diamond 20 "<>"
and:
diamond 29 "Happy New Year "\n' "$IAm" >&2
exit 1
fi
awk -v s="${1:-15}" -v c="${2:-*}" '
BEGIN { n = s - ((s + 1) % 2) # # of lines to print.
m = int((s + 1) / 2) # middle line #.
for(i = 1; i <= s; i++) # S is a string of s c characters (the middle
S = S c # and longest line).
w = 2 - s % 2 # The width (i.e. # of c characters to print on
# the current line) which will be 1 or 2 on the
# first line.
for(i = 1; i <= n; i++){# Print each of the n lines...
printf("%*s%.*s\n",(s - w) / 2,N,w,S)
w += i < m ? 2 : -2 # ... and update w for the next line to
# be printed (i.e. 2 more if we have not
# reached the middle line yet; otherwise
# 2 less).
}
}'
H
Hap
Happy
Happy N
Happy New
Happy New Y
Happy New Yea
Happy New Year
Happy New Year Ha
Happy New Year Happ
Happy New Year Happy
Happy New Year Happy Ne
Happy New Year Happy New
Happy New Year Happy New Ye
Happy New Year Happy New Year
Happy New Year Happy New Ye
Happy New Year Happy New
Happy New Year Happy Ne
Happy New Year Happy
Happy New Year Happ
Happy New Year Ha
Happy New Year
Happy New Yea
Happy New Y
Happy New
Happy N
Happy
Hap
H
Compare these outputs to the results you get with the other proposals using the same counts and strings. With single character strings we all produce the same output.
Interesting effect when applied to Don Cragun's text example:
awk -vW=30 -vL="Happy New Year" 'BEGIN{for(i=k=1;i>=1;i+=k) {printf("%*.*s\n",(W/2)+i,i*2-1,L);if(i>=W/2)k=-1}}'
H
Hap
Happy
Happy N
Happy New
Happy New Y
Happy New Yea
Happy New Year
Happy New Year
Happy New Year
Happy New Year
Happy New Year
Happy New Year
Happy New Year
Happy New Year
Happy New Year
Happy New Year
Happy New Year
Happy New Year
Happy New Year
Happy New Year
Happy New Year
Happy New Yea
Happy New Y
Happy New
Happy N
Happy
Hap
H
EDIT: Try this:
awk -vW=30 '-vL=Happy New Year' 'BEGIN{for(i=k=1;i>=1;i+=k) {while(length(L)<W)L=L" "L;printf("%*.*s\n",(W/2)+i,i*2-1,L);if(i>=W/2)k=-1}}'
Thanks for all the great replies guys. I enjoyed playing about with your script Don!
You can turn this into a little 'find the hidden character' type game to. For example, find the hidden $ in a diamond full of S's. Or one of my favorites, find the hidden ";" amongst the ":"'s! (this really gets tricky at larger sizes!) Expanding on my initial code (again a much more elegant solution exists!):
awk '
function _genLine(n,v,r)
{
ch=v
if (r==1) {srand(); RAND=int(1+rand()*n)}
VAR=""
for (i=1;i<=n;i++) {
ch=v
if (i==RAND && r==1) {ch=uniq}
VAR=VAR?VAR sprintf("%s", ch):sprintf("%s", ch)
}
}
BEGIN{
M=":" ; uniq=";" ; L="10"
C=(L%2)?L:L-1
BASE=(L%2)?1:2
SVAR=(L-BASE)/2
srand()
j=1
R=int(1+rand()*C)
while (BASE>0) {
DLINE=VAR
_genLine(BASE,M) ; DLINE=VAR
if (j==R) {_genLine(BASE,M,1) ; DLINE=VAR}
_genLine(SVAR," ") ; SLINE=VAR
printf "%s%s%s\n", SLINE,DLINE,SLINE
if (BASE==L) {f=1}
BASE=(f==1)?BASE-=2:BASE+=2
SVAR=(f==1)?SVAR+=1:SVAR-=1
j++
}
}'