Watch 'em fall like dominoes
Retina, 87 86 85 bytes
Thanks to Dennis for saving 1 byte.
^.{0,79}$
$0
:`^
<ESC>c
(`/ | \\
__
/\|(?!\\)
//a
(?<!/)\|\\
\\
$
aaaaa
a
aaaa
(a+)+b|a
<empty>
<ESC>
should be replaced by the actual control character (0x1B). <empty>
represents an empty trailing line. You can then run the above code from a single file with the -s
flag.
The code requires a terminal which supports ANSI escape codes. I can't suppress the line feed in Retina's output so I need to clear the entire console with <ESC>c
each time. I've tested the code in bash using Mono to run Retina.
Explanation
^.{0,79}$
$0
We start by appending a space if the input contains fewer than 80 characters. This is so that a /
at the right end does not have to be treated separately.
:`^
<ESC>c
Now we prepend <ESC>c
to the string, which is the ANSI escape code for clearing the terminal. So every time the string is printed it will do so at the top of the terminal. The :`
instructs Retina to print out the result of this substitution, i.e. the initial configuration.
(`/ | \\
__
(`
begins a loop. Since there is no matching )
, the loop is assumed to go until the last stage of the program. Each iteration will simulate one step of the dominoes falling and then "sleep" a bit. This first stage replaces /
and \
next to a space into __
. This automatically handles the / \
case correctly, because matches cannot overlap and are searched for from left to right. So the /<sp>
would be matched and turned into __
such that the \
can't be matched, and we get the correct __\
.
/\|(?!\\)
//a
This turns /|
into //
provided there's no \
next to it. We append an a
such that this new /
doesn't mess with the next stage (which shouldn't "know" about this change yet).
(?<!/)\|\\
\\
The opposite situation: turn |\
into \\
provided there's no /
next to it. We don't need to put an a
here, because we're done with this step of the simulation.
Now the sleeping part...
$
aaaaa
Appends 5 more a
s to the end of the code.
a
aaaa
Turns each a
into 4 a
s, so we get 20 a
s at the end.
(a+)+b|a
<empty>
Now the fun part... we sleep for a bit with the help of catastrophic backtracking. There is an exponential amount of ways to split up a match of (a+)+
between repetitions of the group. Because the b
causes the match to fail, the engine will backtrack and try every single one of those combinations before deciding that (a+)+b
can't match. For the twenty a
s at the end that takes something like half a second.
At the same time, we allow the regex to match a single a
, but only after doing the backtracking. When that matches, we replace it with an empty string, removing all the a
s we inserted for one reason or another from the string.
That leaves printing the string at the end of the loop iteration. Here it comes in handy that I haven't fixed the printing behaviour of loops in Retina yet. Currently, there is only one flag for each stage, which says "print" or "don't print". The default is "don't print" except for the last stage in the program, which defaults to "print". But the stage is looped, so that means it actually prints the current string on each iteration. Normally, that is really annoying, and you almost always need to include an additional empty stage at the end if you only want the final result, but here it lets me save four bytes.
Javascript (ES6), 206 148 129 158 bytes
I'd finally gotten it down to a nicely low point, but it wouldn't clear the console or append an extra space; these problems have been fixed now.
c=console;d=s=>{c.clear(s[79]||(s+=' ')),c.log(s),t=s[R='replace'](/\/ | \\/g,'__')[R](/\/\|/g,'//a')[R](/\|\\/g,'\\\\')[R](/a/g,'');t!=s&&setTimeout(d,99,t)}
Alternate 153 byte version which should work in Node.JS:
d=s=>{s[79]||(s+=' '),console.log("\033c"+s),t=s[R='replace'](/\/ | \\/g,'__')[R](/\/\|/g,'//a')[R](/\|\\/g,'\\\\')[R](/a/g,'');t!=s&&setTimeout(d,99,t)}
IMHO, it's quite fun to play with. Try an HTML version here:
output=document.getElementById("output"),console.log=a=>output.innerHTML=a;
d=s=>{s[79]||(s+=' ');console.log(s),t=s[R='replace'](/\/ | \\/g,'__')[R](/\/\|/g,'//a')[R](/\|\\/g,'\\\\')[R](/a/g,'');t!=s&&setTimeout(d,99,t)}
s=n=>{n=n||15,r='';for(i=0;i<n;i++)r+=(i%2?'|'.repeat(Math.random()*16|0):'\\ /'[Math.random()*3|0].repeat(Math.random()*3+1|0));return r} // randomizer
input=document.getElementById("input");
input.addEventListener("keydown",e=>e.keyCode==13?(e.preventDefault(),d(input.value)):0);
<p>Enter your dominoes here:</p>
<textarea id="input" rows="1" cols="80">|||\ |||\ /|\ /||||||||\ /|||| /|||</textarea>
<button id="random" onclick="input.value=s()">Randomize</button><button id="run" onclick="d(input.value)">Run</button>
<p>Result:</p>
<pre id="output"></pre>
There's probably a good bit more room for golfing. Suggestions welcome!
C#, 335 bytes
Not a great choice of language.
I abused the delay being allowed between 50 and 1000 to select a two-digit number.
New lines and indentation added for clarity:
namespace System.Threading{
class P{
static void Main(string[]z){
var c=@"/|\,/|\,/|,//,|\,\\,/ ,__, \,__".Split(',');
for(string a=z[0].PadRight(80),b="";a!=b;){
Console.Clear();
Console.Write(b=a);
Thread.Sleep(99);
a="";
for(int i,j;(i=a.Length)<80;)
a+=(j=Array.FindIndex(c,d=>b.Substring(i).StartsWith(d)))%2==0
?c[j+1]
:b.Substring(i,1);
}
}
}
}