Skip to content

Commit b8d34dc

Browse files
author
fredj
committedAug 30, 2010
Add the FeatureBrowser ux (http://trac.geoext.org/wiki/ux/FeatureBrowser). p=pgiraud, r=elemoine,ahocevar,fredj (closes #288)
git-svn-id: http://svn.geoext.org/extensions/geoext.ux@2293 5b3c568a-86f9-4a59-a47a-7be284f604b4
1 parent dbb0e67 commit b8d34dc

File tree

8 files changed

+432
-0
lines changed

8 files changed

+432
-0
lines changed
 

‎examples/index.html

+1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ <h2>GeoExt.ux examples</h2>
1010
<li><a href="../ux/SimplePrint/examples/SimplePrint.html">SimplePrint Form using GeoExt.data.PrintProvider</a></li>
1111
<li><a href="../ux/Measure/examples/measure.html">Measure tools</a></li>
1212
<li><a href="../ux/FeatureEditing/examples/feature-editor-grid.html">Feature editor grid</a></li>
13+
<li><a href="../ux/FeatureBrowser/examples/featurebrowser.html">Feature browser</a></li>
1314
</ul>
1415
</body>
1516
</html>

‎tests/list-tests.html

+1
Original file line numberDiff line numberDiff line change
@@ -6,4 +6,5 @@
66
<li>../../geoext.ux/ux/Measure/tests/lib/GeoExt.ux/MeasureArea.html</li>
77
<li>../../geoext.ux/ux/Measure/tests/lib/GeoExt.ux/MeasureLength.html</li>
88
<li>../../geoext.ux/ux/FeatureEditing/tests/lib/GeoExt.ux/FeatureEditorGrid.html</li>
9+
<li>../../geoext.ux/ux/FeatureBrowser/tests/lib/GeoExt.ux/FeatureBrowser.html</li>
910
</ul>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
<html>
2+
<head>
3+
<title>FeatureBrowser Example</title>
4+
5+
<link rel="stylesheet" type="text/css" href="http://extjs.cachefly.net/ext-3.2.1/resources/css/ext-all.css" />
6+
<link rel="stylesheet" type="text/css" href="http://extjs.cachefly.net/ext-3.2.1/examples/shared/examples.css" />
7+
8+
<script type="text/javascript" src="http://extjs.cachefly.net/ext-3.2.1/adapter/ext/ext-base.js"></script>
9+
<script type="text/javascript" src="http://extjs.cachefly.net/ext-3.2.1/ext-all.js"></script>
10+
<script src="http://www.openlayers.org/api/2.9.1/OpenLayers.js"></script>
11+
<script type="text/javascript" src="../../../../geoext/lib/GeoExt.js"></script>
12+
13+
<!-- script resources for this ux -->
14+
<script type="text/javascript" src="../lib/GeoExt.ux/FeatureBrowser.js"></script>
15+
16+
<script type="text/javascript" src="featurebrowser.js"></script>
17+
</head>
18+
<body>
19+
<h1>Feature Browser</h1>
20+
<p style="margin-bottom:15px;">See <a href="featurebrowser.js">featurebrowser.js</a> for the source code.</p>
21+
</body>
22+
</html>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
Ext.onReady(function() {
2+
3+
// required for tooltips
4+
Ext.QuickTips.init();
5+
6+
features = [
7+
new OpenLayers.Feature.Vector(null, {
8+
name: 'toto',
9+
age: '20',
10+
photo: 'http://bits.ohloh.net/attachments/30093/GeoExt_med.png',
11+
type: 'employee'
12+
}),
13+
new OpenLayers.Feature.Vector(null, {
14+
foo: 'bar',
15+
dude: 'truite',
16+
type: 'thing'
17+
})
18+
];
19+
20+
var browser = new GeoExt.ux.FeatureBrowser({
21+
title: 'Feature Browser - no template',
22+
renderTo: Ext.getBody(),
23+
width: 200,
24+
height: 200,
25+
features: features,
26+
bodyStyle: "padding: 5px;"
27+
});
28+
29+
var browser = new GeoExt.ux.FeatureBrowser({
30+
title: 'Feature Browser - \'type\' attribute dependent template',
31+
renderTo: Ext.getBody(),
32+
width: 200,
33+
height: 200,
34+
features: features,
35+
tplFeatureAttribute: 'type',
36+
tpl: {
37+
'employee': new Ext.Template(
38+
'<b>{name}</b>({age})<br />',
39+
'<img src="{photo}" />'
40+
)
41+
},
42+
bodyStyle: "padding: 5px;"
43+
});
44+
var browser = new GeoExt.ux.FeatureBrowser({
45+
title: 'Feature Browser - single feature',
46+
renderTo: Ext.getBody(),
47+
width: 200,
48+
height: 200,
49+
features: [features[0]],
50+
bodyStyle: "padding: 5px;"
51+
});
52+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,206 @@
1+
/**
2+
* Copyright (c) 2008-2010 The Open Source Geospatial Foundation
3+
*
4+
* Published under the BSD license.
5+
* See http://svn.geoext.org/core/trunk/geoext/license.txt for the full text
6+
* of the license.
7+
*/
8+
9+
Ext.namespace("GeoExt.ux");
10+
11+
/** api: (define)
12+
* module = GeoExt.ux
13+
* class = FeatureBrowser
14+
* base_link = `Ext.Panel <http://dev.sencha.com/deploy/dev/docs/?class=Ext.Panel>`_
15+
*/
16+
17+
/** api: constructor
18+
* .. class:: FeatureBrowser(config)
19+
*
20+
* Creates a Panel to browse in a features list,
21+
* show attributes for each using templates.
22+
*/
23+
GeoExt.ux.FeatureBrowser = Ext.extend(Ext.Panel, {
24+
25+
/* begin i18n */
26+
/** api: config[counterText]
27+
* ``String`` i18n, The counter message to display (defaults to "{0} of
28+
* {1}). Note that this string is formatted using {0} as a token for index
29+
* and {1} as a token for total. These tokens should be preserved when
30+
* overriding this string if showing those values is desired.
31+
*/
32+
counterText: "{0} of {1} features",
33+
34+
/** api: config[elseTpl]
35+
* ``Ext.Template`` | ``Ext.XTemplate`` Ext.Template or Ext.XTemplate
36+
* to be used for features which don't match any of the tpl keys.
37+
* Will be taken into account only if tpl is an Object. Optional.
38+
*/
39+
elseTpl: null,
40+
41+
/** api: config[tpl]
42+
* ``Ext.Template`` | ``Ext.XTemplate`` | ``Object`` | ``Function``
43+
* Ext.Template or Ext.XTemplate to be applied for each feature with
44+
* feature.attributes as context. If provided as an Object, each key
45+
* may correspond to the value for the tplFeatureAttribute key in the
46+
* feature attributes. The corresponding value has to be a valid template.
47+
* Optional.
48+
*/
49+
tpl: undefined,
50+
51+
/** api: config[tplFeatureAttribute]
52+
* ``String`` | ``Number``
53+
* The attribute which value is to be compared with one of the tpl members
54+
* when provided as an Object. Mandatory is tpl is a Object. Optional.
55+
*/
56+
tplFeatureAttribute: null,
57+
58+
/** api: config[skippedFeatureAttributes]
59+
* ``Array(String)``
60+
* Specifies the feature attributes to skip in the default Ext.Template
61+
* created by this component. Only applies when the ``tpl`` option is
62+
* not set. Optional.
63+
*/
64+
skippedFeatureAttributes: null,
65+
66+
/** api: config[features]
67+
* ``Array`` Array of ``OpenLayers.Feature.Vector`` to build the
68+
* FeatureBrowser with. Required.
69+
*/
70+
features: null,
71+
72+
/** private: method[initComponent]
73+
*/
74+
initComponent: function() {
75+
76+
this.layout = "card";
77+
78+
var nbFeatures = this.features.length;
79+
80+
this.items = [];
81+
var feature, tpl;
82+
for (var i = 0; i < nbFeatures; i++) {
83+
feature = this.features[i];
84+
tpl = this.getTemplateForFeature(feature);
85+
86+
this.items.push(new Ext.BoxComponent({
87+
id: 'card-'+ this.id + i,
88+
html: tpl.apply(feature.attributes)
89+
}));
90+
}
91+
92+
if (nbFeatures > 1) {
93+
this.bbar = [
94+
{xtype: 'tbtext', id: 'counter-' + this.id},
95+
'->',
96+
{
97+
id: 'move-prev' + this.id,
98+
iconCls: "x-tbar-page-prev",
99+
handler: this.navHandler.createDelegate(this,
100+
[-1, nbFeatures, this.features]),
101+
disabled: true,
102+
listeners: {
103+
click: function(button, e) {
104+
e.stopEvent();
105+
}
106+
}
107+
},
108+
{
109+
id: 'move-next' + this.id,
110+
iconCls: "x-tbar-page-next",
111+
handler: this.navHandler.createDelegate(this,
112+
[1, nbFeatures, this.features]),
113+
listeners: {
114+
click: function(button, e) {
115+
e.stopEvent();
116+
}
117+
}
118+
}
119+
];
120+
}
121+
this.activeItem = 0;
122+
123+
GeoExt.ux.FeatureBrowser.superclass.initComponent.apply(this, arguments);
124+
125+
// add custom events
126+
this.addEvents(
127+
128+
/** api: events[featureselected]
129+
* Fires when a feature is displayed in the FeatureBrowser.
130+
* Application may use this to highlight it on the map, for
131+
* example.
132+
*
133+
* Listener arguments:
134+
* * panel - :class:`GeoExt.ux.FeatureBrowser` This panel.
135+
* * feature - ``OpenLayers.Feature.Vector`` The selected feature
136+
*/
137+
'featureselected'
138+
);
139+
this.fireEvent('featureselected', this, this.features[0]);
140+
141+
var counter = Ext.getCmp('counter-' + this.id);
142+
counter && counter.setText(
143+
String.format(this.counterText, 1, nbFeatures)
144+
);
145+
},
146+
147+
/** private: method[navHandler]
148+
* The navigation handler method. Called when navigation buttons
149+
* (next or previous) are clicked
150+
*/
151+
navHandler: function(direction, total, features) {
152+
var lay = this.getLayout();
153+
var i = lay.activeItem.id.split('card-' + this.id)[1];
154+
var next = parseInt(i, 10) + direction;
155+
lay.setActiveItem(next);
156+
this.fireEvent('featureselected', this, features[next]);
157+
Ext.getCmp('move-prev' + this.id).setDisabled(next === 0);
158+
Ext.getCmp('move-next' + this.id).setDisabled(next == total - 1);
159+
160+
var counter = Ext.getCmp('counter-' + this.id);
161+
counter && counter.setText(
162+
String.format(this.counterText, next + 1, total)
163+
);
164+
},
165+
166+
/** private: method[getTemplateForFeature]
167+
* Returns a template for the given feature.
168+
*
169+
* :param feature: ``OpenLayers.Feature.Vector`` The feature to create
170+
* a template with.
171+
*
172+
* :return: ``Ext.Template`` | ``Ext.XTemplate`` The created template.
173+
*/
174+
getTemplateForFeature: function(feature) {
175+
var tpl,
176+
attributes = feature.attributes;
177+
178+
if (this.tpl instanceof Ext.Template) {
179+
tpl = this.tpl;
180+
} else if (typeof this.tpl === 'object') {
181+
tpl = this.tpl[attributes[this.tplFeatureAttribute]] ||
182+
this.elseTpl;
183+
} else if (typeof this.tpl === 'function') {
184+
// currently unsupported
185+
}
186+
187+
// create a default template with key/value pairs
188+
if (!tpl) {
189+
var templateString = '';
190+
for (var k in attributes) {
191+
if (attributes.hasOwnProperty(k) &&
192+
(this.skippedFeatureAttributes == null ||
193+
this.skippedFeatureAttributes.indexOf(k) === -1)) {
194+
195+
templateString += '<div>' +
196+
'<b>' + k + ': </b>' +
197+
'{' + k + '}' +
198+
'</div>';
199+
}
200+
}
201+
tpl = new Ext.Template(templateString);
202+
}
203+
204+
return tpl;
205+
}
206+
});

‎ux/FeatureBrowser/tests/index.html

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
<html>
2+
<head><meta http-equiv="refresh" content="0;url=../../../../geoext/tests/run-tests.html?testlist=../../geoext.ux/ux/FeatureBrowser/tests/list-tests.html"></head>
3+
<body></body>
4+
</html>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,143 @@
1+
<!DOCTYPE html>
2+
<html debug="true">
3+
<head>
4+
<script type="text/javascript" src="http://www.openlayers.org/api/2.9.1/OpenLayers.js"></script>
5+
<script type="text/javascript" src="http://extjs.cachefly.net/ext-3.2.1/adapter/ext/ext-base.js"></script>
6+
<script type="text/javascript" src="http://extjs.cachefly.net/ext-3.2.1/ext-all.js"></script>
7+
<script type="text/javascript" src="../../../../../../geoext/lib/GeoExt.js"></script>
8+
<script type="text/javascript" src="../../../lib/GeoExt.ux/FeatureBrowser.js"></script>
9+
10+
<script type="text/javascript">
11+
var features = [
12+
new OpenLayers.Feature.Vector(null, {
13+
foo: "foo0",
14+
bar: "bar0",
15+
baz: "baz0"
16+
}),
17+
new OpenLayers.Feature.Vector(null, {
18+
foo: "foo1",
19+
bar: "bar1",
20+
baz: "baz1"
21+
})
22+
];
23+
24+
function test_initialize(t) {
25+
t.plan(3);
26+
var browser;
27+
28+
browser = new GeoExt.ux.FeatureBrowser({
29+
features: features
30+
});
31+
t.ok(browser instanceof GeoExt.ux.FeatureBrowser, "Instance created correctly");
32+
t.eq(browser.items.getCount(), 2, "Correct number of items");
33+
browser.destroy();
34+
35+
browser = new GeoExt.ux.FeatureBrowser({
36+
features: [features[0]]
37+
});
38+
t.ok(browser.getBottomToolbar() == null, "No bbar required");
39+
browser.destroy();
40+
}
41+
42+
function test_events(t) {
43+
t.plan(3);
44+
var browser;
45+
46+
var count = 0;
47+
var selectedFeature;
48+
browser = new GeoExt.ux.FeatureBrowser({
49+
renderTo: Ext.getBody(),
50+
features: features,
51+
listeners: {
52+
featureselected: function(panel, feature) {
53+
count++;
54+
selectedFeature = feature;
55+
}
56+
}
57+
});
58+
t.ok(selectedFeature == features[0], "featureselected listener called with correct argument");
59+
60+
var nextBtn = Ext.getCmp('move-next' + browser.id);
61+
nextBtn.handler.call(nextBtn, nextBtn, Ext.EventObject)
62+
t.eq(count, 2, "featureselected event triggered twice");
63+
t.ok(selectedFeature == features[1], "featureselected listener called with correct argument");
64+
browser.destroy();
65+
}
66+
67+
function test_getTemplate(t) {
68+
t.plan(7);
69+
70+
var browser, tpl;
71+
72+
// no template
73+
browser = new GeoExt.ux.FeatureBrowser({
74+
features: features
75+
});
76+
tpl = browser.getTemplateForFeature(features[0]);
77+
t.eq(tpl.apply(features[0].attributes),
78+
'<div><b>foo: </b>foo0</div><div><b>bar: </b>bar0</div><div><b>baz: </b>baz0</div>',
79+
"Template is correct when no template given"
80+
);
81+
82+
// no template, with skippedFeatureAttributes
83+
browser = new GeoExt.ux.FeatureBrowser({
84+
features: features,
85+
skippedFeatureAttributes: ["bar", "baz"]
86+
});
87+
tpl = browser.getTemplateForFeature(features[0]);
88+
t.eq(tpl.apply(features[0].attributes),
89+
'<div><b>foo: </b>foo0</div>',
90+
"Template is correct when no template given and skippedFeatureAttributes provided"
91+
);
92+
93+
// a unique template
94+
browser = new GeoExt.ux.FeatureBrowser({
95+
features: features,
96+
tpl: new Ext.Template('{foo}')
97+
});
98+
tpl = browser.getTemplateForFeature(features[0]);
99+
t.eq(tpl.apply(features[0].attributes), 'foo0',
100+
"Template is correct when a template is given"
101+
);
102+
tpl = browser.getTemplateForFeature(features[1]);
103+
t.eq(tpl.apply(features[1].attributes), 'foo1',
104+
"Template is correct when a template is given"
105+
);
106+
107+
// a hash
108+
browser = new GeoExt.ux.FeatureBrowser({
109+
features: features,
110+
tplFeatureAttribute: 'foo',
111+
tpl: {
112+
'foo0': new Ext.Template('{foo}')
113+
}
114+
});
115+
tpl = browser.getTemplateForFeature(features[0]);
116+
t.eq(tpl.apply(features[0].attributes), 'foo0',
117+
"Template is correct when tpl is a hash and feature matches"
118+
);
119+
tpl = browser.getTemplateForFeature(features[1]);
120+
t.eq(tpl.apply(features[1].attributes),
121+
'<div><b>foo: </b>foo1</div><div><b>bar: </b>bar1</div><div><b>baz: </b>baz1</div>',
122+
"Template is correct when tpl is a hash and feature doesn't match"
123+
);
124+
125+
// a hash and elseTpl defined
126+
browser = new GeoExt.ux.FeatureBrowser({
127+
features: features,
128+
tplFeatureAttribute: 'foo',
129+
elseTpl: new Ext.Template('truite'),
130+
tpl: {
131+
'never': 'something'
132+
}
133+
});
134+
tpl = browser.getTemplateForFeature(features[1]);
135+
t.eq(tpl.apply(features[1].attributes), 'truite',
136+
"Template is correct when tpl is a hash, elseTpl given and feature doesn't match"
137+
);
138+
}
139+
</script>
140+
<body>
141+
<div id="map" style="width:600px;height:400px;"></div>
142+
</body>
143+
</html>
+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
<ul id="testlist">
2+
<li>../../geoext.ux/ux/FeatureBrowser/tests/lib/GeoExt.ux/FeatureBrowser.html</li>
3+
</ul>

0 commit comments

Comments
 (0)
Please sign in to comment.