Make C<undef ~~ 0> and C<undef ~~ ""> not match (like in 5.10.0)
[p5sagit/p5-mst-13.2.git] / t / op / switch.t
index fc88a13..2012c6c 100644 (file)
@@ -3,20 +3,19 @@
 BEGIN {
     chdir 't' if -d 't';
     @INC = '../lib';
+    require './test.pl';
 }
 
 use strict;
 use warnings;
 
-use Test::More tests => 107;
+plan tests => 122;
 
 # The behaviour of the feature pragma should be tested by lib/switch.t
 # using the tests in t/lib/switch/*. This file tests the behaviour of
 # the switch ops themselves.
-              
 
 use feature 'switch';
-no warnings "numeric";
 
 eval { continue };
 like($@, qr/^Can't "continue" outside/, "continue outside");
@@ -58,306 +57,334 @@ sub check_outside1 { is($_, "outside", "\$_ lexically scoped") }
 # Basic string/numeric comparisons and control flow
 
 {    
-    my $ok = 0;
+    my $ok;
     given(3) {
-       when(2) { $ok = 0; }
-       when(3) { $ok = 1; }
-       when(4) { $ok = 0; }
-       default { $ok = 0; }
+       when(2) { $ok = 'two'; }
+       when(3) { $ok = 'three'; }
+       when(4) { $ok = 'four'; }
+       default { $ok = 'd'; }
     }
-    ok($ok, "numeric comparison");
+    is($ok, 'three', "numeric comparison");
 }
 
 {    
-    my $ok = 0;
+    my $ok;
     use integer;
     given(3.14159265) {
-       when(2) { $ok = 0; }
-       when(3) { $ok = 1; }
-       when(4) { $ok = 0; }
-       default { $ok = 0; }
+       when(2) { $ok = 'two'; }
+       when(3) { $ok = 'three'; }
+       when(4) { $ok = 'four'; }
+       default { $ok = 'd'; }
     }
-    ok($ok, "integer comparison");
+    is($ok, 'three', "integer comparison");
 }
 
 {    
-    my ($ok1, $ok2) = (0, 0);
+    my ($ok1, $ok2);
     given(3) {
-       when(3.1)   { $ok1 = 0; }
-       when(3.0)   { $ok1 = 1; continue }
-       when("3.0") { $ok2 = 1; }
-       default     { $ok2 = 0; }
+       when(3.1)   { $ok1 = 'n'; }
+       when(3.0)   { $ok1 = 'y'; continue }
+       when("3.0") { $ok2 = 'y'; }
+       default     { $ok2 = 'n'; }
     }
-    ok($ok1, "more numeric (pt. 1)");
-    ok($ok2, "more numeric (pt. 2)");
+    is($ok1, 'y', "more numeric (pt. 1)");
+    is($ok2, 'y', "more numeric (pt. 2)");
 }
 
 {
-    my $ok = 0;
+    my $ok;
     given("c") {
-       when("b") { $ok = 0; }
-       when("c") { $ok = 1; }
-       when("d") { $ok = 0; }
-       default   { $ok = 0; }
+       when("b") { $ok = 'B'; }
+       when("c") { $ok = 'C'; }
+       when("d") { $ok = 'D'; }
+       default   { $ok = 'def'; }
     }
-    ok($ok, "string comparison");
+    is($ok, 'C', "string comparison");
 }
 
 {
-    my $ok = 0;
+    my $ok;
     given("c") {
-       when("b") { $ok = 0; }
-       when("c") { $ok = 0; continue }
-       when("c") { $ok = 1; }
-       default   { $ok = 0; }
+       when("b") { $ok = 'B'; }
+       when("c") { $ok = 'C'; continue }
+       when("c") { $ok = 'CC'; }
+       default   { $ok = 'D'; }
     }
-    ok($ok, "simple continue");
+    is($ok, 'CC', "simple continue");
 }
 
 # Definedness
 {
     my $ok = 1;
     given (0) { when(undef) {$ok = 0} }
-    ok($ok, "Given(0) when(undef)");
+    is($ok, 1, "Given(0) when(undef)");
 }
 {
     my $undef;
     my $ok = 1;
     given (0) { when($undef) {$ok = 0} }
-    ok($ok, 'Given(0) when($undef)');
+    is($ok, 1, 'Given(0) when($undef)');
 }
 {
     my $undef;
     my $ok = 0;
     given (0) { when($undef++) {$ok = 1} }
-    ok($ok, "Given(0) when($undef++)");
+    is($ok, 1, "Given(0) when($undef++)");
 }
 {
+    no warnings "uninitialized";
     my $ok = 1;
     given (undef) { when(0) {$ok = 0} }
-    ok($ok, "Given(undef) when(0)");
+    is($ok, 1, "Given(undef) when(0)");
 }
 {
+    no warnings "uninitialized";
     my $undef;
     my $ok = 1;
     given ($undef) { when(0) {$ok = 0} }
-    ok($ok, 'Given($undef) when(0)');
+    is($ok, 1, 'Given($undef) when(0)');
 }
 ########
 {
     my $ok = 1;
     given ("") { when(undef) {$ok = 0} }
-    ok($ok, 'Given("") when(undef)');
+    is($ok, 1, 'Given("") when(undef)');
 }
 {
     my $undef;
     my $ok = 1;
     given ("") { when($undef) {$ok = 0} }
-    ok($ok, 'Given("") when($undef)');
+    is($ok, 1, 'Given("") when($undef)');
 }
 {
+    no warnings "uninitialized";
     my $ok = 1;
     given (undef) { when("") {$ok = 0} }
-    ok($ok, 'Given(undef) when("")');
+    is($ok, 1, 'Given(undef) when("")');
 }
 {
+    no warnings "uninitialized";
     my $undef;
     my $ok = 1;
     given ($undef) { when("") {$ok = 0} }
-    ok($ok, 'Given($undef) when("")');
+    is($ok, 1, 'Given($undef) when("")');
 }
 ########
 {
     my $ok = 0;
     given (undef) { when(undef) {$ok = 1} }
-    ok($ok, "Given(undef) when(undef)");
+    is($ok, 1, "Given(undef) when(undef)");
 }
 {
     my $undef;
     my $ok = 0;
     given (undef) { when($undef) {$ok = 1} }
-    ok($ok, 'Given(undef) when($undef)');
+    is($ok, 1, 'Given(undef) when($undef)');
 }
 {
     my $undef;
     my $ok = 0;
     given ($undef) { when(undef) {$ok = 1} }
-    ok($ok, 'Given($undef) when(undef)');
+    is($ok, 1, 'Given($undef) when(undef)');
 }
 {
     my $undef;
     my $ok = 0;
     given ($undef) { when($undef) {$ok = 1} }
-    ok($ok, 'Given($undef) when($undef)');
+    is($ok, 1, 'Given($undef) when($undef)');
 }
 
 
 # Regular expressions
 {
-    my ($ok1, $ok2) = 0;
+    my ($ok1, $ok2);
     given("Hello, world!") {
        when(/lo/)
-           { $ok1 = 1; continue}
+           { $ok1 = 'y'; continue}
        when(/no/)
-           { $ok1 = 0; continue}
+           { $ok1 = 'n'; continue}
        when(/^(Hello,|Goodbye cruel) world[!.?]/)
-           { $ok2 = 1; continue}
+           { $ok2 = 'Y'; continue}
        when(/^(Hello cruel|Goodbye,) world[!.?]/)
-           { $ok2 = 0; continue}
+           { $ok2 = 'n'; continue}
     }
-    ok($ok1, "regex 1");
-    ok($ok2, "regex 2");
+    is($ok1, 'y', "regex 1");
+    is($ok2, 'Y', "regex 2");
 }
 
 # Comparisons
 {
     my $test = "explicit numeric comparison (<)";
     my $twenty_five = 25;
+    my $ok;
     given($twenty_five) {
-       when ($_ < 10) { fail($test) }
-       when ($_ < 20) { fail($test) }
-       when ($_ < 30) { pass($test) }
-       when ($_ < 40) { fail($test) }
-       default        { fail($test) }
+       when ($_ < 10) { $ok = "ten" }
+       when ($_ < 20) { $ok = "twenty" }
+       when ($_ < 30) { $ok = "thirty" }
+       when ($_ < 40) { $ok = "forty" }
+       default        { $ok = "default" }
     }
+    is($ok, "thirty", $test);
 }
 
 {
     use integer;
     my $test = "explicit numeric comparison (integer <)";
     my $twenty_five = 25;
+    my $ok;
     given($twenty_five) {
-       when ($_ < 10) { fail($test) }
-       when ($_ < 20) { fail($test) }
-       when ($_ < 30) { pass($test) }
-       when ($_ < 40) { fail($test) }
-       default        { fail($test) }
+       when ($_ < 10) { $ok = "ten" }
+       when ($_ < 20) { $ok = "twenty" }
+       when ($_ < 30) { $ok = "thirty" }
+       when ($_ < 40) { $ok = "forty" }
+       default        { $ok = "default" }
     }
+    is($ok, "thirty", $test);
 }
 
 {
     my $test = "explicit numeric comparison (<=)";
     my $twenty_five = 25;
+    my $ok;
     given($twenty_five) {
-       when ($_ <= 10) { fail($test) }
-       when ($_ <= 20) { fail($test) }
-       when ($_ <= 30) { pass($test) }
-       when ($_ <= 40) { fail($test) }
-       default         { fail($test) }
+       when ($_ <= 10) { $ok = "ten" }
+       when ($_ <= 20) { $ok = "twenty" }
+       when ($_ <= 30) { $ok = "thirty" }
+       when ($_ <= 40) { $ok = "forty" }
+       default         { $ok = "default" }
     }
+    is($ok, "thirty", $test);
 }
 
 {
     use integer;
     my $test = "explicit numeric comparison (integer <=)";
     my $twenty_five = 25;
+    my $ok;
     given($twenty_five) {
-       when ($_ <= 10) { fail($test) }
-       when ($_ <= 20) { fail($test) }
-       when ($_ <= 30) { pass($test) }
-       when ($_ <= 40) { fail($test) }
-       default         { fail($test) }
+       when ($_ <= 10) { $ok = "ten" }
+       when ($_ <= 20) { $ok = "twenty" }
+       when ($_ <= 30) { $ok = "thirty" }
+       when ($_ <= 40) { $ok = "forty" }
+       default         { $ok = "default" }
     }
+    is($ok, "thirty", $test);
 }
 
 
 {
     my $test = "explicit numeric comparison (>)";
     my $twenty_five = 25;
+    my $ok;
     given($twenty_five) {
-       when ($_ > 40) { fail($test) }
-       when ($_ > 30) { fail($test) }
-       when ($_ > 20) { pass($test) }
-       when ($_ > 10) { fail($test) }
-       default        { fail($test) }
+       when ($_ > 40) { $ok = "forty" }
+       when ($_ > 30) { $ok = "thirty" }
+       when ($_ > 20) { $ok = "twenty" }
+       when ($_ > 10) { $ok = "ten" }
+       default        { $ok = "default" }
     }
+    is($ok, "twenty", $test);
 }
 
 {
     my $test = "explicit numeric comparison (>=)";
     my $twenty_five = 25;
+    my $ok;
     given($twenty_five) {
-       when ($_ >= 40) { fail($test) }
-       when ($_ >= 30) { fail($test) }
-       when ($_ >= 20) { pass($test) }
-       when ($_ >= 10) { fail($test) }
-       default         { fail($test) }
+       when ($_ >= 40) { $ok = "forty" }
+       when ($_ >= 30) { $ok = "thirty" }
+       when ($_ >= 20) { $ok = "twenty" }
+       when ($_ >= 10) { $ok = "ten" }
+       default         { $ok = "default" }
     }
+    is($ok, "twenty", $test);
 }
 
 {
     use integer;
     my $test = "explicit numeric comparison (integer >)";
     my $twenty_five = 25;
+    my $ok;
     given($twenty_five) {
-       when ($_ > 40) { fail($test) }
-       when ($_ > 30) { fail($test) }
-       when ($_ > 20) { pass($test) }
-       when ($_ > 10) { fail($test) }
-       default        { fail($test) }
+       when ($_ > 40) { $ok = "forty" }
+       when ($_ > 30) { $ok = "thirty" }
+       when ($_ > 20) { $ok = "twenty" }
+       when ($_ > 10) { $ok = "ten" }
+       default        { $ok = "default" }
     }
+    is($ok, "twenty", $test);
 }
 
 {
     use integer;
     my $test = "explicit numeric comparison (integer >=)";
     my $twenty_five = 25;
+    my $ok;
     given($twenty_five) {
-       when ($_ >= 40) { fail($test) }
-       when ($_ >= 30) { fail($test) }
-       when ($_ >= 20) { pass($test) }
-       when ($_ >= 10) { fail($test) }
-       default         { fail($test) }
+       when ($_ >= 40) { $ok = "forty" }
+       when ($_ >= 30) { $ok = "thirty" }
+       when ($_ >= 20) { $ok = "twenty" }
+       when ($_ >= 10) { $ok = "ten" }
+       default         { $ok = "default" }
     }
+    is($ok, "twenty", $test);
 }
 
 
 {
     my $test = "explicit string comparison (lt)";
     my $twenty_five = "25";
+    my $ok;
     given($twenty_five) {
-       when ($_ lt "10") { fail($test) }
-       when ($_ lt "20") { fail($test) }
-       when ($_ lt "30") { pass($test) }
-       when ($_ lt "40") { fail($test) }
-       default         { fail($test) }
+       when ($_ lt "10") { $ok = "ten" }
+       when ($_ lt "20") { $ok = "twenty" }
+       when ($_ lt "30") { $ok = "thirty" }
+       when ($_ lt "40") { $ok = "forty" }
+       default           { $ok = "default" }
     }
+    is($ok, "thirty", $test);
 }
 
 {
     my $test = "explicit string comparison (le)";
     my $twenty_five = "25";
+    my $ok;
     given($twenty_five) {
-       when ($_ le "10") { fail($test) }
-       when ($_ le "20") { fail($test) }
-       when ($_ le "30") { pass($test) }
-       when ($_ le "40") { fail($test) }
-       default           { fail($test) }
+       when ($_ le "10") { $ok = "ten" }
+       when ($_ le "20") { $ok = "twenty" }
+       when ($_ le "30") { $ok = "thirty" }
+       when ($_ le "40") { $ok = "forty" }
+       default           { $ok = "default" }
     }
+    is($ok, "thirty", $test);
 }
 
 {
     my $test = "explicit string comparison (gt)";
     my $twenty_five = 25;
+    my $ok;
     given($twenty_five) {
-       when ($_ ge "40") { fail($test) }
-       when ($_ ge "30") { fail($test) }
-       when ($_ ge "20") { pass($test) }
-       when ($_ ge "10") { fail($test) }
-       default           { fail($test) }
+       when ($_ ge "40") { $ok = "forty" }
+       when ($_ ge "30") { $ok = "thirty" }
+       when ($_ ge "20") { $ok = "twenty" }
+       when ($_ ge "10") { $ok = "ten" }
+       default           { $ok = "default" }
     }
+    is($ok, "twenty", $test);
 }
 
 {
     my $test = "explicit string comparison (ge)";
     my $twenty_five = 25;
+    my $ok;
     given($twenty_five) {
-       when ($_ ge "40") { fail($test) }
-       when ($_ ge "30") { fail($test) }
-       when ($_ ge "20") { pass($test) }
-       when ($_ ge "10") { fail($test) }
-       default           { fail($test) }
+       when ($_ ge "40") { $ok = "forty" }
+       when ($_ ge "30") { $ok = "thirty" }
+       when ($_ ge "20") { $ok = "twenty" }
+       when ($_ ge "10") { $ok = "ten" }
+       default           { $ok = "default" }
     }
+    is($ok, "twenty", $test);
 }
 
 # Make sure it still works with a lexical $_:
@@ -365,23 +392,25 @@ sub check_outside1 { is($_, "outside", "\$_ lexically scoped") }
     my $_;
     my $test = "explicit comparison with lexical \$_";
     my $twenty_five = 25;
+    my $ok;
     given($twenty_five) {
-       when ($_ ge "40") { fail($test) }
-       when ($_ ge "30") { fail($test) }
-       when ($_ ge "20") { pass($test) }
-       when ($_ ge "10") { fail($test) }
-       default           { fail($test) }
+       when ($_ ge "40") { $ok = "forty" }
+       when ($_ ge "30") { $ok = "thirty" }
+       when ($_ ge "20") { $ok = "twenty" }
+       when ($_ ge "10") { $ok = "ten" }
+       default           { $ok = "default" }
     }
+    is($ok, "twenty", $test);
 }
 
 # Optimized-away comparisons
 {
-    my $ok = 0;
+    my $ok;
     given(23) {
-       when (2 + 2 == 4) { $ok = 1; continue }
-       when (2 + 2 == 5) { $ok = 0 }
+       when (2 + 2 == 4) { $ok = 'y'; continue }
+       when (2 + 2 == 5) { $ok = 'n' }
     }
-    ok($ok, "Optimized-away comparison");
+    is($ok, 'y', "Optimized-away comparison");
 }
 
 # File tests
@@ -402,11 +431,11 @@ sub check_outside1 { is($_, "outside", "\$_ lexically scoped") }
 }
 
 # Sub and method calls
-sub bar {"bar"}
+sub notfoo {"bar"}
 {
     my $ok = 0;
     given("foo") {
-       when(bar()) {$ok = 1}
+       when(notfoo()) {$ok = 1}
     }
     ok($ok, "Sub call acts as boolean")
 }
@@ -414,7 +443,7 @@ sub bar {"bar"}
 {
     my $ok = 0;
     given("foo") {
-       when(main->bar()) {$ok = 1}
+       when(main->notfoo()) {$ok = 1}
     }
     ok($ok, "Class-method call acts as boolean")
 }
@@ -423,7 +452,7 @@ sub bar {"bar"}
     my $ok = 0;
     my $obj = bless [];
     given("foo") {
-       when($obj->bar()) {$ok = 1}
+       when($obj->notfoo()) {$ok = 1}
     }
     ok($ok, "Object-method call acts as boolean")
 }
@@ -431,6 +460,16 @@ sub bar {"bar"}
 # Other things that should not be smart matched
 {
     my $ok = 0;
+    given(12) {
+        when( /(\d+)/ and ( 1 <= $1 and $1 <= 12 ) ) {
+            $ok = 1;
+        }
+    }
+    ok($ok, "bool not smartmatches");
+}
+
+{
+    my $ok = 0;
     given(0) {
        when(eof(DATA)) {
            $ok = 1;
@@ -474,15 +513,46 @@ sub bar {"bar"}
 }
 
 {
-    my $ok = 1;
-    given(0) {
+    my $n = 0;
+    for my $l qw(a b c d) {
+       given ($l) {
+           when ($_ eq "b" .. $_ eq "c") { $n = 1 }
+           default { $n = 0 }
+       }
+       ok(($n xor $l =~ /[ad]/), 'when(E1..E2) evaluates in boolean context');
+    }
+}
+
+{
+    my $n = 0;
+    for my $l qw(a b c d) {
+       given ($l) {
+           when ($_ eq "b" ... $_ eq "c") { $n = 1 }
+           default { $n = 0 }
+       }
+       ok(($n xor $l =~ /[ad]/), 'when(E1...E2) evaluates in boolean context');
+    }
+}
+
+{
+    my $ok = 0;
+    given("foo") {
        when((1 == $ok) || "foo") {
-           $ok = 0;
+           $ok = 1;
        }
     }
-    ok($ok, '((1 == $ok) || "foo") not smartmatched');
+    ok($ok, '((1 == $ok) || "foo") smartmatched');
 }
 
+{
+    my $ok = 0;
+    given("foo") {
+       when((1 == $ok || undef) // "foo") {
+           $ok = 1;
+       }
+    }
+    ok($ok, '((1 == $ok || undef) // "foo") smartmatched');
+}
 
 # Make sure we aren't invoking the get-magic more than once
 
@@ -512,7 +582,7 @@ sub bar {"bar"}
 my $f = tie my $v, "FetchCounter";
 
 {   my $test_name = "Only one FETCH (in given)";
-    my $ok = 0;
+    my $ok;
     given($v = 23) {
        when(undef) {}
        when(sub{0}->()) {}
@@ -521,12 +591,12 @@ my $f = tie my $v, "FetchCounter";
        when(23) {$ok = 1}
        when(/24/) {$ok = 0}
     }
-    ok($ok, "precheck: $test_name");
+    is($ok, 1, "precheck: $test_name");
     is($f->count(), 1, $test_name);
 }
 
 {   my $test_name = "Only one FETCH (numeric when)";
-    my $ok = 0;
+    my $ok;
     $v = 23;
     is($f->count(), 0, "Sanity check: $test_name");
     given(23) {
@@ -537,12 +607,12 @@ my $f = tie my $v, "FetchCounter";
        when($v) {$ok = 1}
        when(/24/) {$ok = 0}
     }
-    ok($ok, "precheck: $test_name");
+    is($ok, 1, "precheck: $test_name");
     is($f->count(), 1, $test_name);
 }
 
 {   my $test_name = "Only one FETCH (string when)";
-    my $ok = 0;
+    my $ok;
     $v = "23";
     is($f->count(), 0, "Sanity check: $test_name");
     given("23") {
@@ -553,14 +623,15 @@ my $f = tie my $v, "FetchCounter";
        when($v) {$ok = 1}
        when(/24/) {$ok = 0}
     }
-    ok($ok, "precheck: $test_name");
+    is($ok, 1, "precheck: $test_name");
     is($f->count(), 1, $test_name);
 }
 
 {   my $test_name = "Only one FETCH (undef)";
-    my $ok = 0;
+    my $ok;
     $v = undef;
     is($f->count(), 0, "Sanity check: $test_name");
+    no warnings "uninitialized";
     given(my $undef) {
        when(sub{0}->()) {}
        when("21")  {}
@@ -568,7 +639,7 @@ my $f = tie my $v, "FetchCounter";
        when($v)    {$ok = 1}
        when(undef) {$ok = 0}
     }
-    ok($ok, "precheck: $test_name");
+    is($ok, 1, "precheck: $test_name");
     is($f->count(), 1, $test_name);
 }
 
@@ -653,7 +724,7 @@ my $f = tie my $v, "FetchCounter";
                q{Can't "break" in a loop topicalizer});
        }
        when (1) {
-           is($first, 1, "Lecical loop: first");
+           is($first, 1, "Lexical loop: first");
            $first = 0;
            # Implicit break is okay
        }
@@ -663,20 +734,19 @@ my $f = tie my $v, "FetchCounter";
 
 # Code references
 {
-    no warnings "redefine";
     my $called_foo = 0;
-    sub foo {$called_foo = 1}
+    sub foo {$called_foo = 1; "@_" eq "foo"}
     my $called_bar = 0;
-    sub bar {$called_bar = 1}
+    sub bar {$called_bar = 1; "@_" eq "bar"}
     my ($matched_foo, $matched_bar) = (0, 0);
-    given(\&foo) {
+    given("foo") {
        when(\&bar) {$matched_bar = 1}
        when(\&foo) {$matched_foo = 1}
     }
-    is($called_foo, 0,  "Code ref comparison: foo not called");
-    is($called_bar, 0,  "Code ref comparison: bar not called");
-    is($matched_bar, 0, "Code ref didn't match different one");
-    is($matched_foo, 1, "Code ref did match itself");
+    is($called_foo, 1,  "foo() was called");
+    is($called_bar, 1,  "bar() was called");
+    is($matched_bar, 0, "bar didn't match");
+    is($matched_foo, 1, "foo did match");
 }
 
 sub contains_x {
@@ -704,97 +774,196 @@ sub contains_x {
     is($ok2, 1, "Calling sub indirectly (false)");
 }
 
-# Test overloading
-{ package OverloadTest;
+SKIP: {
+    skip "Scalar/Util.pm not yet available", 20
+       unless -r "$INC[0]/Scalar/Util.pm";
+    # Test overloading
+    { package OverloadTest;
+
+      use overload '""' => sub{"string value of obj"};
+      use overload 'eq' => sub{"$_[0]" eq "$_[1]"};
+
+      use overload "~~" => sub {
+         my ($self, $other, $reversed) = @_;
+         if ($reversed) {
+             $self->{left}  = $other;
+             $self->{right} = $self;
+             $self->{reversed} = 1;
+         } else {
+             $self->{left}  = $self;
+             $self->{right} = $other;
+             $self->{reversed} = 0;
+         }
+         $self->{called} = 1;
+         return $self->{retval};
+      };
+    
+      sub new {
+         my ($pkg, $retval) = @_;
+         bless {
+                called => 0,
+                retval => $retval,
+               }, $pkg;
+      }
+  }
+
+    {
+       my $test = "Overloaded obj in given (true)";
+       my $obj = OverloadTest->new(1);
+       my $matched;
+       given($obj) {
+           when ("other arg") {$matched = 1}
+           default {$matched = 0}
+       }
+    
+       is($obj->{called}, 1, "$test: called");
+       ok($matched, "$test: matched");
+    }
 
-    use overload '""' => sub{"string value of obj"};
+    {
+       my $test = "Overloaded obj in given (false)";
+       my $obj = OverloadTest->new(0);
+       my $matched;
+       given($obj) {
+           when ("other arg") {$matched = 1}
+       }
+    
+       is($obj->{called}, 1, "$test: called");
+       ok(!$matched, "$test: not matched");
+    }
 
-    use overload "~~" => sub {
-        my ($self, $other, $reversed) = @_;
-        if ($reversed) {
-           $self->{left}  = $other;
-           $self->{right} = $self;
-           $self->{reversed} = 1;
-        } else {
-           $self->{left}  = $self;
-           $self->{right} = $other;
-           $self->{reversed} = 0;
-        }
-       $self->{called} = 1;
-       return $self->{retval};
-    };
+    {
+       my $test = "Overloaded obj in when (true)";
+       my $obj = OverloadTest->new(1);
+       my $matched;
+       given("topic") {
+           when ($obj) {$matched = 1}
+           default {$matched = 0}
+       }
     
-    sub new {
-       my ($pkg, $retval) = @_;
-       bless {
-           called => 0,
-           retval => $retval,
-       }, $pkg;
+       is($obj->{called},  1, "$test: called");
+       ok($matched, "$test: matched");
+       is($obj->{left}, "topic", "$test: left");
+       is($obj->{right}, "string value of obj", "$test: right");
+       ok($obj->{reversed}, "$test: reversed");
+    }
+
+    {
+       my $test = "Overloaded obj in when (false)";
+       my $obj = OverloadTest->new(0);
+       my $matched;
+       given("topic") {
+           when ($obj) {$matched = 1}
+           default {$matched = 0}
+       }
+    
+       is($obj->{called}, 1, "$test: called");
+       ok(!$matched, "$test: not matched");
+       is($obj->{left}, "topic", "$test: left");
+       is($obj->{right}, "string value of obj", "$test: right");
+       ok($obj->{reversed}, "$test: reversed");
     }
 }
 
+# Postfix when
 {
-    my $test = "Overloaded obj in given (true)";
-    my $obj = OverloadTest->new(1);
-    my $matched;
-    given($obj) {
-       when ("other arg") {$matched = 1}
-       default {$matched = 0}
+    my $ok;
+    given (undef) {
+       $ok = 1 when undef;
     }
-    
-    is($obj->{called},  1, "$test: called");
-    ok($matched, "$test: matched");
-    is($obj->{left}, "string value of obj", "$test: left");
-    is($obj->{right}, "other arg", "$test: right");
-    ok(!$obj->{reversed}, "$test: not reversed");
+    is($ok, 1, "postfix undef");
 }
-
 {
-    my $test = "Overloaded obj in given (false)";
-    my $obj = OverloadTest->new(0);
-    my $matched;
-    given($obj) {
-       when ("other arg") {$matched = 1}
+    my $ok;
+    given (2) {
+       $ok += 1 when 7;
+       $ok += 2 when 9.1685;
+       $ok += 4 when $_ > 4;
+       $ok += 8 when $_ < 2.5;
     }
-    
-    is($obj->{called},  1, "$test: called");
-    ok(!$matched, "$test: not matched");
-    is($obj->{left}, "string value of obj", "$test: left");
-    is($obj->{right}, "other arg", "$test: right");
-    ok(!$obj->{reversed}, "$test: not reversed");
+    is($ok, 8, "postfix numeric");
 }
-
 {
-    my $test = "Overloaded obj in when (true)";
-    my $obj = OverloadTest->new(1);
-    my $matched;
-    given("topic") {
-       when ($obj) {$matched = 1}
-       default {$matched = 0}
+    my $ok;
+    given ("apple") {
+       $ok = 1, continue when $_ eq "apple";
+       $ok += 2;
+       $ok = 0 when "banana";
     }
-    
-    is($obj->{called},  1, "$test: called");
-    ok($matched, "$test: matched");
-    is($obj->{left}, "topic", "$test: left");
-    is($obj->{right}, "string value of obj", "$test: right");
-    ok($obj->{reversed}, "$test: reversed");
+    is($ok, 3, "postfix string");
 }
-
 {
-    my $test = "Overloaded obj in when (false)";
-    my $obj = OverloadTest->new(0);
-    my $matched;
-    given("topic") {
-       when ($obj) {$matched = 1}
-       default {$matched = 0}
+    my $ok;
+    given ("pear") {
+       do { $ok = 1; continue } when /pea/;
+       $ok += 2;
+       $ok = 0 when /pie/;
+       default { $ok += 4 }
+       $ok = 0;
     }
-    
-    is($obj->{called}, 1, "$test: called");
-    ok(!$matched, "$test: not matched");
-    is($obj->{left}, "topic", "$test: left");
-    is($obj->{right}, "string value of obj", "$test: right");
-    ok($obj->{reversed}, "$test: reversed");
+    is($ok, 7, "postfix regex");
+}
+# be_true is defined at the beginning of the file
+{
+    my $x = "what";
+    given(my $x = "foo") {
+       do {
+           is($x, "foo", "scope inside ... when my \$x = ...");
+           continue;
+       } when be_true(my $x = "bar");
+       is($x, "bar", "scope after ... when my \$x = ...");
+    }
+}
+{
+    my $x = 0;
+    given(my $x = 1) {
+       my $x = 2, continue when be_true();
+        is($x, undef, "scope after my \$x = ... when ...");
+    }
+}
+
+# Tests for last and next in when clauses
+my $letter;
+
+$letter = '';
+for ("a".."e") {
+    given ($_) {
+       $letter = $_;
+       when ("b") { last }
+    }
+    $letter = "z";
+}
+is($letter, "b", "last in when");
+
+$letter = '';
+LETTER1: for ("a".."e") {
+    given ($_) {
+       $letter = $_;
+       when ("b") { last LETTER1 }
+    }
+    $letter = "z";
+}
+is($letter, "b", "last LABEL in when");
+
+$letter = '';
+for ("a".."e") {
+    given ($_) {
+       when (/b|d/) { next }
+       $letter .= $_;
+    }
+    $letter .= ',';
+}
+is($letter, "a,c,e,", "next in when");
+
+$letter = '';
+LETTER2: for ("a".."e") {
+    given ($_) {
+       when (/b|d/) { next LETTER2 }
+       $letter .= $_;
+    }
+    $letter .= ',';
 }
+is($letter, "a,c,e,", "next LABEL in when");
 
 # Okay, that'll do for now. The intricacies of the smartmatch
 # semantics are tested in t/op/smartmatch.t