javaquery/api/src/test/java/org/apidesign/bck2brwsr/htmlpage/KnockoutTest.java
author Jaroslav Tulach <jaroslav.tulach@apidesign.org>
Tue, 02 Apr 2013 15:40:08 +0200
branchmodel
changeset 914 81dcd71877d5
parent 909 e51a474fcf79
child 929 b43aaf398748
permissions -rw-r--r--
@OnFunction can be in @Model classes
jaroslav@530
     1
/**
jaroslav@530
     2
 * Back 2 Browser Bytecode Translator
jaroslav@530
     3
 * Copyright (C) 2012 Jaroslav Tulach <jaroslav.tulach@apidesign.org>
jaroslav@530
     4
 *
jaroslav@530
     5
 * This program is free software: you can redistribute it and/or modify
jaroslav@530
     6
 * it under the terms of the GNU General Public License as published by
jaroslav@530
     7
 * the Free Software Foundation, version 2 of the License.
jaroslav@530
     8
 *
jaroslav@530
     9
 * This program is distributed in the hope that it will be useful,
jaroslav@530
    10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
jaroslav@530
    11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
jaroslav@530
    12
 * GNU General Public License for more details.
jaroslav@530
    13
 *
jaroslav@530
    14
 * You should have received a copy of the GNU General Public License
jaroslav@530
    15
 * along with this program. Look for COPYING file in the top folder.
jaroslav@530
    16
 * If not, see http://opensource.org/licenses/GPL-2.0.
jaroslav@530
    17
 */
jaroslav@530
    18
package org.apidesign.bck2brwsr.htmlpage;
jaroslav@530
    19
jaroslav@767
    20
import java.util.List;
jaroslav@763
    21
import org.apidesign.bck2brwsr.core.JavaScriptBody;
jaroslav@530
    22
import org.apidesign.bck2brwsr.htmlpage.api.ComputedProperty;
jaroslav@530
    23
import org.apidesign.bck2brwsr.htmlpage.api.OnEvent;
jaroslav@879
    24
import org.apidesign.bck2brwsr.htmlpage.api.OnFunction;
jaroslav@530
    25
import org.apidesign.bck2brwsr.htmlpage.api.Page;
jaroslav@530
    26
import org.apidesign.bck2brwsr.htmlpage.api.Property;
jaroslav@530
    27
import org.apidesign.bck2brwsr.vmtest.BrwsrTest;
jaroslav@530
    28
import org.apidesign.bck2brwsr.vmtest.HtmlFragment;
jaroslav@530
    29
import org.apidesign.bck2brwsr.vmtest.VMTest;
jaroslav@530
    30
import org.testng.annotations.Factory;
jaroslav@530
    31
jaroslav@530
    32
/**
jaroslav@530
    33
 *
jaroslav@530
    34
 * @author Jaroslav Tulach <jtulach@netbeans.org>
jaroslav@530
    35
 */
jaroslav@530
    36
@Page(xhtml="Knockout.xhtml", className="KnockoutModel", properties={
jaroslav@763
    37
    @Property(name="name", type=String.class),
jaroslav@879
    38
    @Property(name="results", type=String.class, array = true),
jaroslav@906
    39
    @Property(name="callbackCount", type=int.class),
jaroslav@906
    40
    @Property(name="people", type=PersonImpl.class, array = true)
jaroslav@530
    41
}) 
jaroslav@530
    42
public class KnockoutTest {
jaroslav@530
    43
    
jaroslav@530
    44
    @HtmlFragment(
jaroslav@530
    45
        "<h1 data-bind=\"text: helloMessage\">Loading Bck2Brwsr's Hello World...</h1>\n" +
jaroslav@530
    46
        "Your name: <input id='input' data-bind=\"value: name\"></input>\n" +
jaroslav@530
    47
        "<button id=\"hello\">Say Hello!</button>\n"
jaroslav@530
    48
    )
jaroslav@530
    49
    @BrwsrTest public void modifyValueAssertChangeInModel() {
jaroslav@530
    50
        KnockoutModel m = new KnockoutModel();
jaroslav@530
    51
        m.setName("Kukuc");
jaroslav@530
    52
        m.applyBindings();
jaroslav@892
    53
        assert "Kukuc".equals(m.input.getValue()) : "Value is really kukuc: " + m.input.getValue();
jaroslav@892
    54
        m.input.setValue("Jardo");
jaroslav@892
    55
        m.triggerEvent(m.input, OnEvent.CHANGE);
jaroslav@530
    56
        assert "Jardo".equals(m.getName()) : "Name property updated: " + m.getName();
jaroslav@530
    57
    }
jaroslav@530
    58
    
jaroslav@763
    59
    @HtmlFragment(
jaroslav@763
    60
        "<ul id='ul' data-bind='foreach: results'>\n"
jaroslav@879
    61
        + "  <li data-bind='text: $data, click: $root.call'/>\n"
jaroslav@763
    62
        + "</ul>\n"
jaroslav@763
    63
    )
jaroslav@763
    64
    @BrwsrTest public void displayContentOfArray() {
jaroslav@763
    65
        KnockoutModel m = new KnockoutModel();
jaroslav@763
    66
        m.getResults().add("Ahoj");
jaroslav@763
    67
        m.applyBindings();
jaroslav@763
    68
        
jaroslav@763
    69
        int cnt = countChildren("ul");
jaroslav@763
    70
        assert cnt == 1 : "One child, but was " + cnt;
jaroslav@763
    71
        
jaroslav@879
    72
        m.getResults().add("Hi");
jaroslav@763
    73
jaroslav@763
    74
        cnt = countChildren("ul");
jaroslav@763
    75
        assert cnt == 2 : "Two children now, but was " + cnt;
jaroslav@879
    76
        
jaroslav@879
    77
        triggerChildClick("ul", 1);
jaroslav@879
    78
        
jaroslav@879
    79
        assert 1 == m.getCallbackCount() : "One callback " + m.getCallbackCount();
jaroslav@879
    80
        assert "Hi".equals(m.getName()) : "We got callback from 2nd child " + m.getName();
jaroslav@763
    81
    }
jaroslav@763
    82
    
jaroslav@767
    83
    @HtmlFragment(
jaroslav@767
    84
        "<ul id='ul' data-bind='foreach: cmpResults'>\n"
jaroslav@767
    85
        + "  <li><b data-bind='text: $data'></b></li>\n"
jaroslav@767
    86
        + "</ul>\n"
jaroslav@767
    87
    )
jaroslav@767
    88
    @BrwsrTest public void displayContentOfDerivedArray() {
jaroslav@767
    89
        KnockoutModel m = new KnockoutModel();
jaroslav@767
    90
        m.getResults().add("Ahoj");
jaroslav@767
    91
        m.applyBindings();
jaroslav@767
    92
        
jaroslav@767
    93
        int cnt = countChildren("ul");
jaroslav@767
    94
        assert cnt == 1 : "One child, but was " + cnt;
jaroslav@767
    95
        
jaroslav@767
    96
        m.getResults().add("hello");
jaroslav@767
    97
jaroslav@767
    98
        cnt = countChildren("ul");
jaroslav@767
    99
        assert cnt == 2 : "Two children now, but was " + cnt;
jaroslav@767
   100
    }
jaroslav@767
   101
    
jaroslav@906
   102
    @HtmlFragment(
jaroslav@906
   103
        "<ul id='ul' data-bind='foreach: people'>\n"
jaroslav@909
   104
        + "  <li data-bind='text: $data.firstName, click: $root.removePerson'></li>\n"
jaroslav@906
   105
        + "</ul>\n"
jaroslav@906
   106
    )
jaroslav@906
   107
    @BrwsrTest public void displayContentOfArrayOfPeople() {
jaroslav@906
   108
        KnockoutModel m = new KnockoutModel();
jaroslav@908
   109
        
jaroslav@908
   110
        final Person first = new Person();
jaroslav@908
   111
        first.setFirstName("first");
jaroslav@908
   112
        m.getPeople().add(first);
jaroslav@908
   113
        
jaroslav@906
   114
        m.applyBindings();
jaroslav@906
   115
        
jaroslav@906
   116
        int cnt = countChildren("ul");
jaroslav@906
   117
        assert cnt == 1 : "One child, but was " + cnt;
jaroslav@906
   118
        
jaroslav@908
   119
        final Person second = new Person();
jaroslav@908
   120
        second.setFirstName("second");
jaroslav@908
   121
        m.getPeople().add(second);
jaroslav@906
   122
jaroslav@906
   123
        cnt = countChildren("ul");
jaroslav@906
   124
        assert cnt == 2 : "Two children now, but was " + cnt;
jaroslav@906
   125
jaroslav@906
   126
        triggerChildClick("ul", 1);
jaroslav@906
   127
        
jaroslav@906
   128
        assert 1 == m.getCallbackCount() : "One callback " + m.getCallbackCount();
jaroslav@906
   129
jaroslav@906
   130
        cnt = countChildren("ul");
jaroslav@906
   131
        assert cnt == 1 : "Again one child, but was " + cnt;
jaroslav@908
   132
        
jaroslav@908
   133
        String txt = childText("ul", 0);
jaroslav@908
   134
        assert "first".equals(txt) : "Expecting 'first': " + txt;
jaroslav@909
   135
        
jaroslav@909
   136
        first.setFirstName("changed");
jaroslav@909
   137
        
jaroslav@909
   138
        txt = childText("ul", 0);
jaroslav@909
   139
        assert "changed".equals(txt) : "Expecting 'changed': " + txt;
jaroslav@906
   140
    }
jaroslav@914
   141
    
jaroslav@914
   142
    @HtmlFragment(
jaroslav@914
   143
        "<ul id='ul' data-bind='foreach: people'>\n"
jaroslav@914
   144
        + "  <li data-bind='text: $data.firstName, click: changeSex'></li>\n"
jaroslav@914
   145
        + "</ul>\n"
jaroslav@914
   146
    )
jaroslav@914
   147
    @BrwsrTest public void onPersonFunction() {
jaroslav@914
   148
        KnockoutModel m = new KnockoutModel();
jaroslav@914
   149
        
jaroslav@914
   150
        final Person first = new Person();
jaroslav@914
   151
        first.setFirstName("first");
jaroslav@914
   152
        first.setSex(Sex.MALE);
jaroslav@914
   153
        m.getPeople().add(first);
jaroslav@914
   154
        
jaroslav@914
   155
        
jaroslav@914
   156
        m.applyBindings();
jaroslav@914
   157
        
jaroslav@914
   158
        int cnt = countChildren("ul");
jaroslav@914
   159
        assert cnt == 1 : "One child, but was " + cnt;
jaroslav@914
   160
        
jaroslav@914
   161
        
jaroslav@914
   162
        triggerChildClick("ul", 0);
jaroslav@914
   163
        
jaroslav@914
   164
        assert first.getSex() == Sex.FEMALE : "Transverted to female: " + first.getSex();
jaroslav@914
   165
    }
jaroslav@906
   166
     
jaroslav@879
   167
    @OnFunction
jaroslav@879
   168
    static void call(KnockoutModel m, String data) {
jaroslav@879
   169
        m.setName(data);
jaroslav@879
   170
        m.setCallbackCount(m.getCallbackCount() + 1);
jaroslav@879
   171
    }
jaroslav@906
   172
jaroslav@906
   173
    @OnFunction
jaroslav@906
   174
    static void removePerson(KnockoutModel model, Person data) {
jaroslav@906
   175
        model.setCallbackCount(model.getCallbackCount() + 1);
jaroslav@906
   176
        model.getPeople().remove(data);
jaroslav@906
   177
    }
jaroslav@906
   178
    
jaroslav@879
   179
    
jaroslav@530
   180
    @ComputedProperty
jaroslav@530
   181
    static String helloMessage(String name) {
jaroslav@530
   182
        return "Hello " + name + "!";
jaroslav@530
   183
    }
jaroslav@530
   184
    
jaroslav@767
   185
    @ComputedProperty
jaroslav@767
   186
    static List<String> cmpResults(List<String> results) {
jaroslav@767
   187
        return results;
jaroslav@767
   188
    }
jaroslav@767
   189
    
jaroslav@530
   190
    @Factory
jaroslav@530
   191
    public static Object[] create() {
jaroslav@530
   192
        return VMTest.create(KnockoutTest.class);
jaroslav@530
   193
    }
jaroslav@763
   194
    
jaroslav@763
   195
    @JavaScriptBody(args = { "id" }, body = 
jaroslav@763
   196
          "var e = window.document.getElementById(id);\n "
jaroslav@763
   197
        + "if (typeof e === 'undefined') return -2;\n "
jaroslav@763
   198
        + "return e.children.length;\n "
jaroslav@763
   199
    )
jaroslav@763
   200
    private static native int countChildren(String id);
jaroslav@879
   201
jaroslav@879
   202
    @JavaScriptBody(args = { "id", "pos" }, body = 
jaroslav@879
   203
          "var e = window.document.getElementById(id);\n "
jaroslav@879
   204
        + "var ev = window.document.createEvent('MouseEvents');\n "
jaroslav@879
   205
        + "ev.initMouseEvent('click', true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);\n "
jaroslav@879
   206
        + "e.children[pos].dispatchEvent(ev);\n "
jaroslav@879
   207
    )
jaroslav@879
   208
    private static native void triggerChildClick(String id, int pos);
jaroslav@908
   209
jaroslav@908
   210
    @JavaScriptBody(args = { "id", "pos" }, body = 
jaroslav@908
   211
          "var e = window.document.getElementById(id);\n "
jaroslav@908
   212
        + "var t = e.children[pos].innerHTML;\n "
jaroslav@908
   213
        + "return t ? t : null;"
jaroslav@908
   214
    )
jaroslav@908
   215
    private static native String childText(String id, int pos);
jaroslav@530
   216
}