[ALOY-362] Allow controller-less views with some controller logic
| GitHub Issue | n/a | 
|---|---|
| Type | New Feature | 
| Priority | Medium | 
| Status | Closed | 
| Resolution | Fixed | 
| Resolution Date | 2014-05-21T15:18:42.000+0000 | 
| Affected Version/s | n/a | 
| Fix Version/s | Alloy 1.4.0 | 
| Components | Runtime | 
| Labels | notable, qe-testadded | 
| Reporter | Matthew Congrove | 
| Assignee | Tim Poulsen | 
| Created | 2012-11-02T18:37:47.000+0000 | 
| Updated | 2014-05-22T21:52:04.000+0000 | 
Description
	The ability to create views without a controller is quite nice, but it could be taken a step further. During development of a recent application I built numerous controllers with only 1 or 2 lines of code to set properties of elements in the view. For example:
Alloy.createController("MyView", {
	labelText: "Hello, World!",
	imageUrl: "/images/test.png"
}).getView();
$.MyLabel.text = arguments[0].labelText;
$.MyImage.image = arguments[0].imageUrl;
Alloy.createController("MyView", {
	MyLabel: {
		text: "Hello, World!"
	},
	MyImage: {
		image: "/images/test.png"
	}
}).getView();
I like this idea, with a small change to implementation likely. For the sake of keeping the current convention unchanged, I would prefer to use a spceific reserved property where these "by id" property assignments would happen. Something like:
Alloy.createController("MyView", { views: { MyLabel: { text: "Hello, World!" }, MyImage: { image: "/images/test.png" } } }).getView();Alloy.createController("MyView", { '#MyLabel': { text: "Hello, World!" }, '#MyImage': { image: "/images/test.png" } }).getView();Yes, that makes much more sense. I really like the latter as it keeps everything similar across the filetypes (JS -> TSS).
PR: https://github.com/appcelerator/alloy/pull/358 Functional test: Run the app included in the PR. The result should be a white window with two labels: the first with a large gradient background and other properties as set in the index.js controller. Note that the childview.xml contains no properties other than ID. Styles from childview.tss are merged in. There is no childview.js controller file. Properties and values are set in the index.js controller using the new autoView argument.
reopened for alternative implementation
New PR: https://github.com/appcelerator/alloy/pull/376 Functional test as before. In the new implementation, you no longer use the "autoView" object in your createController() call. Instead, pass an argument object whose keys correspond to the IDs of the components in the child view to which the styles will be applied.
$.index.add(Alloy.createController('childview', { "#label": { text: 'I am a label', top: 50 }, "#anotherlabel": { text: 'I am also a label', foo: 'bar' /* a custom/non-existent property */ }, "#someNonExistentId": { text: 'I do not exist'} }).getView());For performance reasons, I'm leaning towards the implementation of https://github.com/appcelerator/alloy/pull/377
$.index.add(Alloy.createController('childview', { alloyAutoView: { "#label": { text: 'I am a label', top: 50 }, "#anotherlabel": { text: 'I am also a label', foo: 'bar' }, "#someNonExistentId": { text: 'I do not exist'} } }).getView());OK, I went with a new dedicated method for controller-less views. This offers the best performance for existing apps and "standard" functionality while making controllerless views available to developers via a specific method. See https://github.com/appcelerator/alloy/pull/376 (merged).
// new method, for instantiating controllerless views $.index.add(Alloy.createControllerlessView('childview', { "#label": { text: 'I am a label', top: 50 } }).getView()); // old method for instantiating controllers is still there $.index.add(Alloy.createController('normalchild', { text: 'Set the old-fashioned way in the normalchild.js file' }).getView());PR merged
re-open for potential api change
The new API moves this functionality to be a method of the controller, which must be invoked after the controller is instantiated. Like this:
var someView = Alloy.createController('someView'); someView.updateViews({ "#myLabel": { text: 'I am a label', top: 50 } }); $.index.add(someView.getView());New PR https://github.com/appcelerator/alloy/pull/404 Merged to 1_4_X branch and cherry-picked to master
Problem on Android was due to case -- sample app used normalChild when the file was named normalchild. The sample app would have failed on an iOS device. I've corrected that. I also made a change that removes an inadvertent SDK dependency (as written, the solution would have required SDK 3.2.0+).
New PR https://github.com/appcelerator/alloy/pull/408 has been merged to 1_4_X and master
Verified fixed and working as expected on Android (Google Nexus Galaxy), iOS (iPhone 5) and MobileWeb TiSDK 3.3.0.v20140522110830 Alloy 1.4.0-alpha4 Appcelerator Studio 3.3.0.201405211748 CLI 3.3.0-alpha5 Closing.