frank-cucumber 0.9.6 → 0.9.7

Sign up to get free protection for your applications and to get access to all the features.
Files changed (50) hide show
  1. data/frank-skeleton/frank_static_resources.bundle/index.html +29 -34
  2. data/frank-skeleton/frank_static_resources.bundle/{index.haml → index.html.haml} +28 -29
  3. data/frank-skeleton/frank_static_resources.bundle/js/accessible_views_view.coffee +41 -0
  4. data/frank-skeleton/frank_static_resources.bundle/js/accessible_views_view.js +46 -0
  5. data/frank-skeleton/frank_static_resources.bundle/js/controller.coffee +129 -0
  6. data/frank-skeleton/frank_static_resources.bundle/js/controller.js +142 -0
  7. data/frank-skeleton/frank_static_resources.bundle/js/details_view.coffee +42 -0
  8. data/frank-skeleton/frank_static_resources.bundle/js/details_view.js +51 -0
  9. data/frank-skeleton/frank_static_resources.bundle/js/dropdown_control.coffee +64 -0
  10. data/frank-skeleton/frank_static_resources.bundle/js/dropdown_control.js +73 -0
  11. data/frank-skeleton/frank_static_resources.bundle/js/ersatz_model.coffee +46 -0
  12. data/frank-skeleton/frank_static_resources.bundle/js/ersatz_model.js +59 -0
  13. data/frank-skeleton/frank_static_resources.bundle/js/ersatz_view.coffee +167 -0
  14. data/frank-skeleton/frank_static_resources.bundle/js/ersatz_view.js +198 -0
  15. data/frank-skeleton/frank_static_resources.bundle/js/experiment_bar_model.coffee +10 -0
  16. data/frank-skeleton/frank_static_resources.bundle/js/experiment_bar_model.js +17 -0
  17. data/frank-skeleton/frank_static_resources.bundle/js/experiment_bar_view.coffee +43 -0
  18. data/frank-skeleton/frank_static_resources.bundle/js/experiment_bar_view.js +60 -0
  19. data/frank-skeleton/frank_static_resources.bundle/js/frank.coffee +96 -0
  20. data/frank-skeleton/frank_static_resources.bundle/js/frank.js +118 -0
  21. data/frank-skeleton/frank_static_resources.bundle/js/lib/backbone.js +1431 -0
  22. data/frank-skeleton/frank_static_resources.bundle/{coffee-script.js → js/lib/coffee-script.js} +0 -0
  23. data/frank-skeleton/frank_static_resources.bundle/{jquery-ui.min.js → js/lib/jquery-ui.min.js} +0 -0
  24. data/frank-skeleton/frank_static_resources.bundle/{jquery.min.js → js/lib/jquery.min.js} +0 -0
  25. data/frank-skeleton/frank_static_resources.bundle/{jquery.treeview.js → js/lib/jquery.treeview.js} +0 -0
  26. data/frank-skeleton/frank_static_resources.bundle/{json2.js → js/lib/json2.js} +0 -0
  27. data/frank-skeleton/frank_static_resources.bundle/js/lib/raphael.js +5815 -0
  28. data/frank-skeleton/frank_static_resources.bundle/js/lib/require.js +2053 -0
  29. data/frank-skeleton/frank_static_resources.bundle/{underscore.js → js/lib/underscore.js} +466 -177
  30. data/frank-skeleton/frank_static_resources.bundle/js/main.coffee +27 -0
  31. data/frank-skeleton/frank_static_resources.bundle/js/main.js +29 -0
  32. data/frank-skeleton/frank_static_resources.bundle/js/tabs_controller.coffee +13 -0
  33. data/frank-skeleton/frank_static_resources.bundle/js/tabs_controller.js +22 -0
  34. data/frank-skeleton/frank_static_resources.bundle/js/toast_controller.coffee +15 -0
  35. data/frank-skeleton/frank_static_resources.bundle/js/toast_controller.js +28 -0
  36. data/frank-skeleton/frank_static_resources.bundle/js/transform_stack.coffee +59 -0
  37. data/frank-skeleton/frank_static_resources.bundle/js/transform_stack.js +78 -0
  38. data/frank-skeleton/frank_static_resources.bundle/js/tree_view.coffee +53 -0
  39. data/frank-skeleton/frank_static_resources.bundle/js/tree_view.js +64 -0
  40. data/frank-skeleton/frank_static_resources.bundle/js/view_heir_model.coffee +37 -0
  41. data/frank-skeleton/frank_static_resources.bundle/js/view_heir_model.js +48 -0
  42. data/frank-skeleton/frank_static_resources.bundle/js/view_model.coffee +39 -0
  43. data/frank-skeleton/frank_static_resources.bundle/js/view_model.js +62 -0
  44. data/frank-skeleton/frank_static_resources.bundle/symbiote.css +116 -84
  45. data/lib/frank-cucumber/frankifier.rb +20 -2
  46. data/lib/frank-cucumber/version.rb +1 -1
  47. metadata +70 -38
  48. data/frank-skeleton/frank_static_resources.bundle/raphael-min.js +0 -7
  49. data/frank-skeleton/frank_static_resources.bundle/symbiote.js +0 -585
  50. data/frank-skeleton/frank_static_resources.bundle/symbiote_ui.coffee +0 -39
@@ -0,0 +1,42 @@
1
+ INTERESTING_PROPERTIES = ['class', 'accessibilityLabel', 'tag', 'alpha', 'isHidden']
2
+ IRREGULAR_PROPERTIES = INTERESTING_PROPERTIES.concat( 'parent', 'subviews', 'children' )
3
+
4
+ renderListItem = ( propertyName, propertyValue, cssClass )->
5
+ propertyValue ?= 'null'
6
+
7
+ if _.isObject( propertyValue )
8
+ propertyValue = JSON.stringify(propertyValue)
9
+
10
+ $("<li>").addClass(cssClass)
11
+ .append(
12
+ $('<div/>').addClass('key').text(propertyName),
13
+ $('<div/>').addClass('value').text(propertyValue)
14
+ )
15
+
16
+ define ->
17
+ DetailsView = Backbone.View.extend
18
+ el: $('#dom-detail')
19
+
20
+ updateModel: (model)->
21
+ @model = model
22
+ @refresh()
23
+
24
+ refresh: ->
25
+ $ul = $('<ul>')
26
+
27
+ for prop in INTERESTING_PROPERTIES
28
+ if @model.has(prop)
29
+ val = @model.get(prop)
30
+ $ul.append( renderListItem( prop, val, 'interesting' ) )
31
+
32
+ for prop in _.keys(@model.attributes).sort()
33
+ continue if _.contains( IRREGULAR_PROPERTIES, prop )
34
+
35
+ val = @model.get(prop)
36
+ $ul.append( renderListItem( prop, val ) )
37
+
38
+ @$el.empty().append($ul)
39
+
40
+
41
+
42
+ DetailsView
@@ -0,0 +1,51 @@
1
+ (function() {
2
+ var INTERESTING_PROPERTIES, IRREGULAR_PROPERTIES, renderListItem;
3
+
4
+ INTERESTING_PROPERTIES = ['class', 'accessibilityLabel', 'tag', 'alpha', 'isHidden'];
5
+
6
+ IRREGULAR_PROPERTIES = INTERESTING_PROPERTIES.concat('parent', 'subviews', 'children');
7
+
8
+ renderListItem = function(propertyName, propertyValue, cssClass) {
9
+ if (propertyValue == null) {
10
+ propertyValue = 'null';
11
+ }
12
+ if (_.isObject(propertyValue)) {
13
+ propertyValue = JSON.stringify(propertyValue);
14
+ }
15
+ return $("<li>").addClass(cssClass).append($('<div/>').addClass('key').text(propertyName), $('<div/>').addClass('value').text(propertyValue));
16
+ };
17
+
18
+ define(function() {
19
+ var DetailsView;
20
+ DetailsView = Backbone.View.extend({
21
+ el: $('#dom-detail'),
22
+ updateModel: function(model) {
23
+ this.model = model;
24
+ return this.refresh();
25
+ },
26
+ refresh: function() {
27
+ var $ul, prop, val, _i, _j, _len, _len1, _ref;
28
+ $ul = $('<ul>');
29
+ for (_i = 0, _len = INTERESTING_PROPERTIES.length; _i < _len; _i++) {
30
+ prop = INTERESTING_PROPERTIES[_i];
31
+ if (this.model.has(prop)) {
32
+ val = this.model.get(prop);
33
+ $ul.append(renderListItem(prop, val, 'interesting'));
34
+ }
35
+ }
36
+ _ref = _.keys(this.model.attributes).sort();
37
+ for (_j = 0, _len1 = _ref.length; _j < _len1; _j++) {
38
+ prop = _ref[_j];
39
+ if (_.contains(IRREGULAR_PROPERTIES, prop)) {
40
+ continue;
41
+ }
42
+ val = this.model.get(prop);
43
+ $ul.append(renderListItem(prop, val));
44
+ }
45
+ return this.$el.empty().append($ul);
46
+ }
47
+ });
48
+ return DetailsView;
49
+ });
50
+
51
+ }).call(this);
@@ -0,0 +1,64 @@
1
+ define ->
2
+
3
+ DropdownModel = Backbone.Model.extend
4
+ select: ->
5
+ @collection.each (model) =>
6
+ model.set( 'selected', model == @ )
7
+ @trigger('option-clicked',@)
8
+
9
+ DropdownCollection = Backbone.Collection.extend
10
+ model: DropdownModel
11
+
12
+ DropdownView = Backbone.View.extend
13
+ tagName: 'div'
14
+ className: 'dropdown'
15
+
16
+ events:
17
+ "click .drop-indicator": 'clickedDrop'
18
+ "click .button": 'clickedItem'
19
+
20
+ initialize: ->
21
+ @collection = new DropdownCollection()
22
+ @collection.on 'all', _.bind(@render,@)
23
+
24
+ # hackery :(
25
+ $('body').on 'click', _.bind(@clickedOutsideDropdown,@)
26
+
27
+ selectedModel: ->
28
+ selected = @collection.find (m)-> m.get('selected')
29
+ selected || @collection.at(0)
30
+ unselectedModels: ->
31
+ _(@collection.reject (m)-> m.get('selected'))
32
+
33
+ createButtonFor: (model)->
34
+ $("""<button>#{model.get('text')}</button>""")
35
+ .on 'click', => @clickedItem(model)
36
+
37
+ render: ->
38
+ @$el
39
+ .empty()
40
+ .append( @createButtonFor(@selectedModel()) )
41
+ .append( $ul = $("<ul>") )
42
+
43
+ @unselectedModels().each (model)=>
44
+ $ul.append( @createButtonFor(model) )
45
+
46
+ @$el.append( """<div class="drop-indicator">v</div>""" )
47
+
48
+ clickedDrop: (e)->
49
+ e.stopPropagation() #prevent clickedOutsideDropdown from firing
50
+ @$('ul').toggleClass('shown')
51
+
52
+ clickedOutsideDropdown: ->
53
+ @$('ul').removeClass('shown')
54
+
55
+ clickedItem: (model)->
56
+ model.select()
57
+
58
+
59
+
60
+
61
+ {
62
+ DropdownModel : DropdownModel
63
+ DropdownView : DropdownView
64
+ }
@@ -0,0 +1,73 @@
1
+ (function() {
2
+
3
+ define(function() {
4
+ var DropdownCollection, DropdownModel, DropdownView;
5
+ DropdownModel = Backbone.Model.extend({
6
+ select: function() {
7
+ var _this = this;
8
+ this.collection.each(function(model) {
9
+ return model.set('selected', model === _this);
10
+ });
11
+ return this.trigger('option-clicked', this);
12
+ }
13
+ });
14
+ DropdownCollection = Backbone.Collection.extend({
15
+ model: DropdownModel
16
+ });
17
+ DropdownView = Backbone.View.extend({
18
+ tagName: 'div',
19
+ className: 'dropdown',
20
+ events: {
21
+ "click .drop-indicator": 'clickedDrop',
22
+ "click .button": 'clickedItem'
23
+ },
24
+ initialize: function() {
25
+ this.collection = new DropdownCollection();
26
+ this.collection.on('all', _.bind(this.render, this));
27
+ return $('body').on('click', _.bind(this.clickedOutsideDropdown, this));
28
+ },
29
+ selectedModel: function() {
30
+ var selected;
31
+ selected = this.collection.find(function(m) {
32
+ return m.get('selected');
33
+ });
34
+ return selected || this.collection.at(0);
35
+ },
36
+ unselectedModels: function() {
37
+ return _(this.collection.reject(function(m) {
38
+ return m.get('selected');
39
+ }));
40
+ },
41
+ createButtonFor: function(model) {
42
+ var _this = this;
43
+ return $("<button>" + (model.get('text')) + "</button>").on('click', function() {
44
+ return _this.clickedItem(model);
45
+ });
46
+ },
47
+ render: function() {
48
+ var $ul,
49
+ _this = this;
50
+ this.$el.empty().append(this.createButtonFor(this.selectedModel())).append($ul = $("<ul>"));
51
+ this.unselectedModels().each(function(model) {
52
+ return $ul.append(_this.createButtonFor(model));
53
+ });
54
+ return this.$el.append("<div class=\"drop-indicator\">v</div>");
55
+ },
56
+ clickedDrop: function(e) {
57
+ e.stopPropagation();
58
+ return this.$('ul').toggleClass('shown');
59
+ },
60
+ clickedOutsideDropdown: function() {
61
+ return this.$('ul').removeClass('shown');
62
+ },
63
+ clickedItem: function(model) {
64
+ return model.select();
65
+ }
66
+ });
67
+ return {
68
+ DropdownModel: DropdownModel,
69
+ DropdownView: DropdownView
70
+ };
71
+ });
72
+
73
+ }).call(this);
@@ -0,0 +1,46 @@
1
+ define ["frank"], (frank)->
2
+
3
+ ErsatzModel = Backbone.Model.extend
4
+ default:
5
+ highlightFrames = []
6
+
7
+ initialize: ->
8
+ @refreshBaseScreenshot()
9
+
10
+ highlightSomeFramesForABit: (frames)->
11
+ @set('highlightFrames',frames)
12
+ @temporaryHighlightTimeout = window.setTimeout( =>
13
+ @set('highlightFrames',[])
14
+ @temporaryHighlightTimeout = undefined
15
+ , 1500 )
16
+
17
+ resetViews: (views,deviceFamily,orientation)->
18
+ @set('allViews',views)
19
+ @set('deviceFamily',deviceFamily)
20
+ @set('orientation',orientation)
21
+ @set('highlightFrames',[])
22
+ @configureAllViews(views)
23
+
24
+ refreshBaseScreenshot: ->
25
+ @set('baseScreenshotUrl',frank.baseScreenshotUrl())
26
+
27
+ toggleAsploded: ->
28
+ isAsploded = !(@get('isAsploded'))
29
+ @set('isAsploded',isAsploded)
30
+ if isAsploded
31
+ @updateAsplodedViews()
32
+
33
+ isAsploded
34
+
35
+ updateAsplodedViews: ()->
36
+ frank.requestSnapshotRefresh().done =>
37
+ @trigger('snapshots-refreshed',@)
38
+
39
+ configureAllViews: (allViews)->
40
+ allViews.on 'change:active', (subject,isActive)=>
41
+ window.clearTimeout(@temporaryHighlightTimeout) if @temporaryHighlightTimeout?
42
+
43
+ if isActive && !@get('isAsploded')
44
+ @set('highlightFrames',[subject.get('accessibilityFrame')])
45
+ else
46
+ @set('highlightFrames',[])
@@ -0,0 +1,59 @@
1
+ (function() {
2
+
3
+ define(["frank"], function(frank) {
4
+ var ErsatzModel, highlightFrames;
5
+ return ErsatzModel = Backbone.Model.extend({
6
+ "default": highlightFrames = [],
7
+ initialize: function() {
8
+ return this.refreshBaseScreenshot();
9
+ },
10
+ highlightSomeFramesForABit: function(frames) {
11
+ var _this = this;
12
+ this.set('highlightFrames', frames);
13
+ return this.temporaryHighlightTimeout = window.setTimeout(function() {
14
+ _this.set('highlightFrames', []);
15
+ return _this.temporaryHighlightTimeout = void 0;
16
+ }, 1500);
17
+ },
18
+ resetViews: function(views, deviceFamily, orientation) {
19
+ this.set('allViews', views);
20
+ this.set('deviceFamily', deviceFamily);
21
+ this.set('orientation', orientation);
22
+ this.set('highlightFrames', []);
23
+ return this.configureAllViews(views);
24
+ },
25
+ refreshBaseScreenshot: function() {
26
+ return this.set('baseScreenshotUrl', frank.baseScreenshotUrl());
27
+ },
28
+ toggleAsploded: function() {
29
+ var isAsploded;
30
+ isAsploded = !(this.get('isAsploded'));
31
+ this.set('isAsploded', isAsploded);
32
+ if (isAsploded) {
33
+ this.updateAsplodedViews();
34
+ }
35
+ return isAsploded;
36
+ },
37
+ updateAsplodedViews: function() {
38
+ var _this = this;
39
+ return frank.requestSnapshotRefresh().done(function() {
40
+ return _this.trigger('snapshots-refreshed', _this);
41
+ });
42
+ },
43
+ configureAllViews: function(allViews) {
44
+ var _this = this;
45
+ return allViews.on('change:active', function(subject, isActive) {
46
+ if (_this.temporaryHighlightTimeout != null) {
47
+ window.clearTimeout(_this.temporaryHighlightTimeout);
48
+ }
49
+ if (isActive && !_this.get('isAsploded')) {
50
+ return _this.set('highlightFrames', [subject.get('accessibilityFrame')]);
51
+ } else {
52
+ return _this.set('highlightFrames', []);
53
+ }
54
+ });
55
+ }
56
+ });
57
+ });
58
+
59
+ }).call(this);
@@ -0,0 +1,167 @@
1
+ ISO_SKEW = 15
2
+ ISO_MAJOR_OFFSET = 50
3
+ ISO_MINOR_OFFSET = 5
4
+ SCREEN_BOUNDS = {
5
+ iphone: { x: 0, y: 0, width: 320, height: 480 }
6
+ ipad: { x: 0, y: 0, width: 768, height: 1024 }
7
+ }
8
+ #SCREENSHOT_URL = symbiote.baseUrlFor( "screenshot" )
9
+
10
+ define ['transform_stack','ersatz_model'], (transformStack,ErsatzModel)->
11
+
12
+ drawStaticBackdropAndReturnTransformer = (paper,deviceFamily,orientation,isoSkew) ->
13
+ paper.clear()
14
+ paper.canvas.setAttribute "width", "100%"
15
+ paper.canvas.setAttribute "height", "100%"
16
+
17
+ isiPhone = ('iphone' == deviceFamily)
18
+
19
+ if isiPhone
20
+ paper.canvas.setAttribute "viewBox", "0 0 380 720"
21
+ rotationPoint = [190,360]
22
+ else
23
+ paper.canvas.setAttribute "viewBox", "0 0 875 1200"
24
+ rotationPoint = [437,600]
25
+
26
+ transformer = transformStack()
27
+
28
+ transformer.skew(0, isoSkew).translate( 6, 6 )
29
+
30
+
31
+ rotation = switch orientation
32
+ when 'landscape_right' then 90
33
+ when 'portrait_upside_down' then 180
34
+ when 'landscape_left' then 270
35
+ else false
36
+
37
+ if rotation
38
+ transformer.rotateAroundPoint(rotation,rotationPoint...)
39
+
40
+ # main outline of device
41
+ if isiPhone
42
+ paper.rect(0, 0, 360, 708, 40).attr(
43
+ fill: "black"
44
+ stroke: "gray"
45
+ "stroke-width": 4
46
+ ).transform transformer.desc()
47
+ else
48
+ paper.rect( 10, 10, 855, 1110, 20 ).attr(
49
+ 'fill': 'black',
50
+ 'stroke': 'gray',
51
+ 'stroke-width': 6
52
+ ).transform transformer.desc()
53
+
54
+
55
+ if isiPhone
56
+ # home button
57
+ transformer.push().translate( 180, 655 )
58
+ paper.circle(0, 0, 34).transform(transformer.desc()).attr( "fill", "90-#303030-#101010" )
59
+
60
+ # square inside home button
61
+ paper.rect(0, 0, 22, 22, 5).attr(
62
+ stroke: "gray"
63
+ "stroke-width": 2
64
+ ).transform transformer.push().translate(-11, -11).descAndPop()
65
+
66
+ transformer.translate(20, 120)
67
+
68
+ else
69
+ transformer.translate(50,50)
70
+
71
+ if( isoSkew > 0 )
72
+ transformer.translate(-ISO_MAJOR_OFFSET, 0)
73
+ transformer
74
+
75
+
76
+ transformFromBaseForViewModel = (baseTransformer,viewModel,withSkew=false)->
77
+ {origin:{x,y}} = viewModel.get('accessibilityFrame')
78
+ baseTransformer.push().translate(x,y)
79
+ if( withSkew )
80
+ baseTransformer.translate(viewModel.get('depth')*-ISO_MINOR_OFFSET, 0)
81
+ baseTransformer.descAndPop()
82
+
83
+ ErsatzViewSnapshotView = Backbone.View.extend
84
+ initialize:->
85
+ @model.on('change:active',_.bind(@updateOpacity,@))
86
+ @render()
87
+
88
+ render: ->
89
+ frame = @model.get('accessibilityFrame')
90
+ @el
91
+ .attr
92
+ transform: transformFromBaseForViewModel( @options.baseTransformer, @model,true )
93
+ src: @model.getSnapshotUrl()
94
+ x: 0
95
+ y: 0
96
+ width: frame.size.width
97
+ height: frame.size.height
98
+ @updateOpacity()
99
+ @el
100
+
101
+ updateOpacity: ->
102
+ opacity = ( if @model.get('active') then 1.0 else 0.05 )
103
+ @el.attr('opacity',opacity)
104
+
105
+
106
+ ErsatzView = Backbone.View.extend
107
+ el: $('#ui-locator-view')
108
+
109
+ initialize: ->
110
+ _.bindAll( @, 'render' )
111
+ @model = new ErsatzModel()
112
+ @highlights = []
113
+ @paper = new Raphael(@.el)
114
+ @model.on 'change:baseScreenshotUrl', _.bind(@refreshBaseScreenshot,@)
115
+ @model.on 'change:isAsploded', _.bind(@render,@)
116
+ @model.on 'snapshots-refreshed', _.bind(@refreshSnapshots,@)
117
+ @model.on 'change:highlightFrames', _.bind(@refreshHighlightFrames,@)
118
+
119
+ render: ->
120
+ @highlights = []
121
+
122
+ isoSkew = (if @model.get('isAsploded') then ISO_SKEW else 0)
123
+ @backdropTransformer = drawStaticBackdropAndReturnTransformer(@paper,@model.get('deviceFamily'),@model.get('orientation'),isoSkew)
124
+ @backdrop = @paper.image()
125
+ @refreshBaseScreenshot()
126
+ if @model.get('isAsploded')
127
+ @backdrop.attr('opacity',0.5)
128
+ @refreshSnapshots()
129
+
130
+ @el
131
+
132
+ screenBounds: -> SCREEN_BOUNDS[@model.get('deviceFamily')]
133
+
134
+
135
+ refreshBaseScreenshot: ->
136
+ newScreenshotUrl = @model.get('baseScreenshotUrl')
137
+ return unless newScreenshotUrl?
138
+
139
+ @backdrop
140
+ .transform(@backdropTransformer.desc())
141
+ .attr( @screenBounds() )
142
+ .attr( 'src', newScreenshotUrl )
143
+ .toFront()
144
+
145
+ refreshSnapshots: ->
146
+ @model.get('allViews').each (viewModel) =>
147
+ snapshotView = new ErsatzViewSnapshotView(
148
+ model: viewModel
149
+ baseTransformer: @backdropTransformer
150
+ el: @paper.image()
151
+ )
152
+
153
+ refreshHighlightFrames: ->
154
+ h.remove() for h in @highlights
155
+ @highlights = []
156
+
157
+ @highlights = _.map @model.get('highlightFrames'), ({origin,size})=>
158
+ @paper.rect().attr(
159
+ fill: "#aaff00"
160
+ opacity: 0.8
161
+ stroke: "black"
162
+ transform: @backdropTransformer.push().translate(origin.x,origin.y).descAndPop()
163
+ x: 0
164
+ y: 0
165
+ width: size.width
166
+ height: size.height
167
+ )