Add update function

Workflow, model and their controllers are gathered.
This commit is contained in:
Shu Muto 2016-11-16 19:41:55 +09:00
parent a6b5d3c11d
commit 8a3c614f22
23 changed files with 645 additions and 284 deletions

View File

@ -26,7 +26,8 @@ import uuid
LOG = logging.getLogger(__name__)
CREATE_ATTRS = ['name']
ATTRIBUTES = ['name', 'description', 'enabled', 'size', 'temperature',
'base', 'flavor', 'topping']
STUB_DATA = {}
@ -68,18 +69,11 @@ def apiclient(request):
def {{cookiecutter.panel}}_create(request, **kwargs):
args = {}
for (key, value) in kwargs.items():
if key in CREATE_ATTRS:
args[str(key)] = str(value)
if key in ATTRIBUTES:
args[str(key)] = value
else:
raise exceptions.BadRequest(
"Key must be in %s" % ",".join(CREATE_ATTRS))
if key == "labels":
labels = {}
vals = value.split(",")
for v in vals:
kv = v.split("=", 1)
labels[kv[0]] = kv[1]
args["labels"] = labels
"Key must be in %s" % ",".join(ATTRIBUTES))
#created = apiclient(request).{{cookiecutter.panel}}s.create(**args)
# create dummy response
@ -92,6 +86,26 @@ def {{cookiecutter.panel}}_create(request, **kwargs):
return created
def {{cookiecutter.panel}}_update(request, id, **kwargs):
args = {}
for (key, value) in kwargs.items():
if key in ATTRIBUTES:
args[str(key)] = value
else:
raise exceptions.BadRequest(
"Key must be in %s" % ",".join(ATTRIBUTES))
#updated = apiclient(request).{{cookiecutter.panel}}.update(id, **args)
# update dummy response
args["uuid"] = id
updated = StubResponse(args)
for k in args:
setattr(updated, k, args[k])
STUB_DATA[updated.uuid] = updated
return updated
def {{cookiecutter.panel}}_delete(request, id):
#deleted = apiclient(request).{{cookiecutter.panel}}s.delete(id)
deleted = STUB_DATA.pop(id)

View File

@ -38,6 +38,17 @@ class {{cookiecutter.panel_func}}(generic.View):
"""Get a specific {{cookiecutter.panel}}"""
return change_to_id(client.{{cookiecutter.panel}}_show(request, id).to_dict())
@rest_utils.ajax(data_required=True)
def post(self, request, id):
"""Update a {{cookiecutter.panel_func}}.
Returns the updated {{cookiecutter.panel_func}} object on success.
"""
{{cookiecutter.panel}} = client.{{cookiecutter.panel}}_update(request, id, **request.DATA)
return rest_utils.CreatedResponse(
'/api/{{cookiecutter.api_module}}/{{cookiecutter.panel}}/%s' % {{cookiecutter.panel}}.uuid,
{{cookiecutter.panel}}.to_dict())
@urls.register
class {{cookiecutter.panel_func}}s(generic.View):
@ -64,12 +75,12 @@ class {{cookiecutter.panel_func}}s(generic.View):
client.{{cookiecutter.panel}}_delete(request, id)
@rest_utils.ajax(data_required=True)
def post(self, request):
def put(self, request):
"""Create a new {{cookiecutter.panel_func}}.
Returns the new {{cookiecutter.panel_func}} object on success.
"""
new_resource = client.{{cookiecutter.panel}}_create(request, **request.DATA)
{{cookiecutter.panel}} = client.{{cookiecutter.panel}}_create(request, **request.DATA)
return rest_utils.CreatedResponse(
'/api/{{cookiecutter.api_module}}/{{cookiecutter.panel}}/%s' % new_resource.uuid,
new_resource.to_dict())
'/api/{{cookiecutter.api_module}}/{{cookiecutter.panel}}/%s' % {{cookiecutter.panel}}.uuid,
{{cookiecutter.panel}}.to_dict())

View File

@ -26,9 +26,10 @@
function API(apiService, toastService, gettext) {
var service = {
create{{cookiecutter.panel_func}}: create{{cookiecutter.panel_func}},
get{{cookiecutter.panel_func}}: get{{cookiecutter.panel_func}},
get{{cookiecutter.panel_func}}s: get{{cookiecutter.panel_func}}s,
create{{cookiecutter.panel_func}}: create{{cookiecutter.panel_func}},
update{{cookiecutter.panel_func}}: update{{cookiecutter.panel_func}},
delete{{cookiecutter.panel_func}}: delete{{cookiecutter.panel_func}},
};
@ -38,14 +39,6 @@
// {{cookiecutter.panel_func}}s //
///////////////
function create{{cookiecutter.panel_func}}(params) {
return apiService.post('/api/{{cookiecutter.api_module}}/{{cookiecutter.panel}}s/', params)
.error(function() {
var msg = gettext('Unable to create the {{cookiecutter.panel_func}} with name: %(name)s');
toastService.add('error', interpolate(msg, { name: params.name }, true));
});
}
function get{{cookiecutter.panel_func}}(id) {
return apiService.get('/api/{{cookiecutter.api_module}}/{{cookiecutter.panel}}s/' + id)
.error(function() {
@ -61,6 +54,22 @@
});
}
function create{{cookiecutter.panel_func}}(params) {
return apiService.put('/api/{{cookiecutter.api_module}}/{{cookiecutter.panel}}s/', params)
.error(function() {
var msg = gettext('Unable to create the {{cookiecutter.panel_func}} with name: %(name)s');
toastService.add('error', interpolate(msg, { name: params.name }, true));
});
}
function update{{cookiecutter.panel_func}}(id, params) {
return apiService.post('/api/{{cookiecutter.api_module}}/{{cookiecutter.panel}}s/' + id, params)
.error(function() {
var msg = gettext('Unable to update the {{cookiecutter.panel_func}} with id: %(id)s');
toastService.add('error', interpolate(msg, { id: params.id }, true));
});
}
function delete{{cookiecutter.panel_func}}(id, suppressError) {
var promise = apiService.delete('/api/{{cookiecutter.api_module}}/{{cookiecutter.panel}}s/', [id]);
return suppressError ? promise : promise.error(function() {

View File

@ -22,13 +22,18 @@
* @description
* Provides all of the actions for {{cookiecutter.panel_func}}s.
*/
angular.module('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.actions', ['horizon.framework', 'horizon.dashboard.{{cookiecutter.panel_group}}'])
.run(register{{cookiecutter.panel_func}}Actions);
angular
.module('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.actions', [
'horizon.framework',
'horizon.dashboard.{{cookiecutter.panel_group}}'
])
.run(register{{cookiecutter.panel_func}}Actions);
register{{cookiecutter.panel_func}}Actions.$inject = [
'horizon.framework.conf.resource-type-registry.service',
'horizon.framework.util.i18n.gettext',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.create.service',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.update.service',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.delete.service',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.resourceType',
];
@ -37,21 +42,12 @@
registry,
gettext,
create{{cookiecutter.panel_func}}Service,
update{{cookiecutter.panel_func}}Service,
delete{{cookiecutter.panel_func}}Service,
resourceType)
{
var {{cookiecutter.panel}}sResourceType = registry.getResourceType(resourceType);
{{cookiecutter.panel}}sResourceType.itemActions
.append({
id: 'delete{{cookiecutter.panel_func}}Action',
service: delete{{cookiecutter.panel_func}}Service,
template: {
type: 'delete',
text: gettext('Delete {{cookiecutter.panel_func}}')
}
});
{{cookiecutter.panel}}sResourceType.batchActions
{{cookiecutter.panel}}sResourceType.globalActions
.append({
id: 'create{{cookiecutter.panel_func}}Action',
service: create{{cookiecutter.panel_func}}Service,
@ -60,6 +56,8 @@
text: gettext('Create {{cookiecutter.panel_func}}')
}
})
{{cookiecutter.panel}}sResourceType.batchActions
.append({
id: 'batchDelete{{cookiecutter.panel_func}}Action',
service: delete{{cookiecutter.panel_func}}Service,
@ -68,6 +66,23 @@
text: gettext('Delete {{cookiecutter.panel_func}}s')
}
});
}
{{cookiecutter.panel}}sResourceType.itemActions
.append({
id: 'update{{cookiecutter.panel_func}}Action',
service: update{{cookiecutter.panel_func}}Service,
template: {
type: 'item',
text: gettext('Update {{cookiecutter.panel_func}}')
}
})
.append({
id: 'delete{{cookiecutter.panel_func}}Action',
service: delete{{cookiecutter.panel_func}}Service,
template: {
type: 'delete',
text: gettext('Delete {{cookiecutter.panel_func}}')
}
});
}
})();

View File

@ -32,14 +32,14 @@
'horizon.framework.util.q.extensions',
'horizon.framework.widgets.modal.wizard-modal.service',
'horizon.framework.widgets.toast.service',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.{{cookiecutter.panel}}Model',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.events',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.resourceType',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.workflow'
];
function createService(
$location, policy, actionResult, gettext, $qExtensions, wizardModalService, toast, model, events, resourceType, workflow
$location, policy, actionResult, gettext, $qExtensions, wizardModalService,
toast, events, resourceType, workflow
) {
var scope;
@ -60,18 +60,16 @@
function initScope($scope) {
scope = $scope;
scope.workflow = workflow;
scope.model = model;
scope.$on('$destroy', function() {
});
}
function perform(selected) {
scope.model.init();
// for creation according to selected item
scope.selected = selected;
return wizardModalService.modal({
scope: scope,
workflow: workflow,
workflow: workflow.init('create'),
submit: submit
}).result;
}
@ -82,7 +80,7 @@
}
function submit(){
return model.create{{cookiecutter.panel_func}}().then(success);
return workflow.save().then(success);
}
function success(response) {

View File

@ -0,0 +1,98 @@
/**
* Licensed under the Apache License, Version 2.0 (the "License"); you may
* not use this file except in compliance with the License. You may obtain
* a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*/
(function() {
'use strict';
/**
* @ngdoc overview
* @name horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.update.service
* @description Service for the {{cookiecutter.panel}} update modal
*/
angular
.module('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s')
.factory('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.update.service', updateService);
updateService.$inject = [
'$location',
'horizon.app.core.openstack-service-api.policy',
'horizon.framework.util.actions.action-result.service',
'horizon.framework.util.i18n.gettext',
'horizon.framework.util.q.extensions',
'horizon.framework.widgets.modal.wizard-modal.service',
'horizon.framework.widgets.toast.service',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.events',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.resourceType',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.workflow'
];
function updateService(
$location, policy, actionResult, gettext, $qExtensions, wizardModalService,
toast, events, resourceType, workflow
) {
var scope;
var message = {
success: gettext('{{cookiecutter.panel_func}} %s was successfully updated.')
};
var service = {
initScope: initScope,
perform: perform,
allowed: allowed
};
return service;
//////////////
function initScope($scope) {
scope = $scope;
scope.workflow = workflow;
scope.$on('$destroy', function() {
});
}
function perform(selected) {
// to use selected item for step controllers
scope.selected = selected;
return wizardModalService.modal({
scope: scope,
workflow: workflow.init('update', selected.id),
submit: submit
}).result;
}
function allowed() {
return $qExtensions.booleanAsPromise(true);
//return policy.ifAllowed({ rules: [['{{cookiecutter.panel}}', 'add_{{cookiecutter.panel}}']] });
}
function submit(){
return workflow.save().then(success);
}
function success(response) {
response.data.id = response.data.uuid;
toast.add('success', interpolate(message.success, [response.data.id]));
var result = actionResult.getActionResult()
.updated(resourceType, response.data.id);
if(result.result.failed.length == 0 && result.result.updated.length > 0){
$location.path('/{{cookiecutter.dashboard}}/{{cookiecutter.panel}}s');
}else{
return result.result;
}
}
}
})();

View File

@ -1,50 +0,0 @@
/**
* Licensed under the Apache License, Version 2.0 (the "License"); you may
* not use this file except in compliance with the License. You may obtain
* a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*/
(function() {
'use strict';
angular
.module('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s')
.factory('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.workflow', {{cookiecutter.panel}}Workflow);
{{cookiecutter.panel}}Workflow.$inject = [
'horizon.dashboard.{{cookiecutter.panel_group}}.basePath',
'horizon.app.core.workflow.factory',
'horizon.framework.util.i18n.gettext'
];
function {{cookiecutter.panel}}Workflow(basePath, dashboardWorkflow, gettext) {
return dashboardWorkflow({
title: gettext('Create {{cookiecutter.panel_func}}'),
steps: [
{
title: gettext('Info'),
templateUrl: basePath + '{{cookiecutter.panel}}s/create/info/info.html',
helpUrl: basePath + '{{cookiecutter.panel}}s/create/info/info.help.html',
formName: '{{cookiecutter.panel}}InfoForm'
}
],
btnText: {
finish: gettext('Create')
},
btnIcon: {
finish: 'fa fa-check'
}
});
}
})();

View File

@ -1,8 +0,0 @@
<div ng-controller="create{{cookiecutter.panel_func}}InfoController as ctrl">
<form name="schemaForm"
sf-schema="ctrl.schema"
sf-form="ctrl.form"
sf-model="ctrl.model">
</form>
</div>

View File

@ -1,146 +0,0 @@
/**
* Licensed under the Apache License, Version 2.0 (the "License"); you may
* not use this file except in compliance with the License. You may obtain
* a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*/
(function() {
'use strict';
/**
* @ngdoc controller
* @name create{{cookiecutter.panel_func}}InfoController
* @ngController
* @description
* Controller for the {{cookiecutter.panel}} info step in create workflow
*/
angular
.module('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s')
.controller('create{{cookiecutter.panel_func}}InfoController', create{{cookiecutter.panel_func}}InfoController);
create{{cookiecutter.panel_func}}InfoController.$inject = [
'$q',
'$scope',
'horizon.dashboard.{{cookiecutter.panel_group}}.basePath',
'horizon.app.core.openstack-service-api.{{cookiecutter.api_module}}',
'horizon.framework.util.i18n.gettext'
];
function create{{cookiecutter.panel_func}}InfoController($q, $scope, basePath, api, gettext) {
var ctrl = this;
var createInfoSchema = {
type: 'object',
properties: {
name: {
title: gettext('Name'),
type: 'string'
},
description: {
title: gettext('Description'),
type: 'string'
},
enabled: {
title: gettext('Enabled'),
type: 'boolean',
default: true
},
size: {
title: gettext('Size'),
type: 'string',
default: 'M'
},
base: {
title: gettext('Base'),
type: 'string'
},
flavor: {
title: gettext('Flavor'),
type: 'string'
},
topping: {
title: gettext('Topping'),
type: 'string'
}
},
required: ['name', 'base']
};
var createInfoForm = [
{
type: 'section',
htmlClass: 'row',
items: [
{
type: 'section',
htmlClass: 'col-sm-12',
items: [
// info tab
{
key: 'name',
placeholder: gettext('Name of the {{cookiecutter.panel}} to create.')
},
{
key: 'description',
type: 'textarea'
},
{
key: 'enabled',
type: 'checkboxes',
titleMap: [
{value: true, name: ''},
]
},
// spec tab
{
key: 'size',
type: 'radiobuttons'
titleMap: [
// get options from API
{value: 'S', name: gettext('Small')},
{value: 'M', name: gettext('Medium')},
{value: 'L', name: gettext('Large')},
{value: 'XL', name: gettext('Extra Large')}
]
},
{
key: 'base',
type: 'select'
titleMap: [
// get options from API
]
},
{
key: 'flavor',
type: 'select'
titleMap: [
// get options from API
]
},
{
key: 'topping',
type: 'checkboxes'
titleMap: [
// get options from API
]
}
]
}
]
}
];
ctrl.schema = createInfoSchema;
ctrl.form = createInfoForm;
ctrl.model = model.spec;
}
})();

View File

@ -22,21 +22,24 @@
* @description
* Provides details features for {{cookiecutter.panel_func}}.
*/
angular.module('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.details',
['horizon.framework.conf', 'horizon.app.core'])
.run(registerDetails);
angular
.module('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.details', [
'horizon.app.core',
'horizon.framework.conf'
])
.run(registerDetails);
registerDetails.$inject = [
'horizon.app.core.openstack-service-api.{{cookiecutter.api_module}}',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.basePath',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.resourceType',
'horizon.app.core.openstack-service-api.{{cookiecutter.api_module}}',
'horizon.framework.conf.resource-type-registry.service'
];
function registerDetails(
api,
basePath,
resourceType,
api,
registry
) {
registry.getResourceType(resourceType)
@ -51,5 +54,4 @@
return api.get{{cookiecutter.panel_func}}(identifier);
}
}
})();

View File

@ -5,6 +5,10 @@
</dl>
</div>
<div class="row rsp-alt-p2">
<dl class="col-md-4">
<dt translate>Topping</dt>
<dd>{$ item.topping | noValue $}</dd>
</dl>
<dl class="col-md-4">
<dt translate>Created</dt>
<dd>{$ item.created_at | date:'short' | noValue $}</dd>

View File

@ -7,7 +7,8 @@
resource-type-name="OS::{{cookiecutter.api_name}}::{{cookiecutter.panel_func}}"
cls="dl-horizontal"
item="ctrl.{{cookiecutter.panel}}"
property-groups="[['name'],
property-groups="[['name', 'description', 'enabled'],
['size', 'temperature', 'base', 'flavor', 'topping'],
['id', 'created_at', 'updated_at']]">
</hz-resource-property-list>
</div>

View File

@ -0,0 +1,7 @@
<div ng-controller="horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.workflow.{{cookiecutter.panel}}InfoController as ctrl">
<form name="schemaForm"
sf-schema="ctrl.schema"
sf-form="ctrl.form"
sf-model="ctrl.model">
</form>
</div>

View File

@ -0,0 +1,104 @@
/**
* Licensed under the Apache License, Version 2.0 (the "License"); you may
* not use this file except in compliance with the License. You may obtain
* a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*/
(function() {
'use strict';
/**
* @ngdoc controller
* @name {{cookiecutter.panel}}InfoController
* @ngController
* @description
* Controller for the {{cookiecutter.panel}} info step in create/update workflow
*/
angular
.module('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s')
.controller(
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.workflow.{{cookiecutter.panel}}InfoController',
{{cookiecutter.panel}}InfoController);
{{cookiecutter.panel}}InfoController.$inject = [
'$q',
'horizon.app.core.openstack-service-api.{{cookiecutter.api_module}}',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.model',
'horizon.framework.util.i18n.gettext'
];
function {{cookiecutter.panel}}InfoController($q, api, model, gettext) {
var ctrl = this;
ctrl.schema = {
type: 'object',
properties: {
name: {
title: gettext('Name'),
type: 'string'
},
description: {
title: gettext('Description'),
type: 'string'
},
enabled: {
title: gettext('Enabled'),
type: 'boolean',
default: true
}
},
required: ['name']
};
ctrl.form = [
{
type: 'section',
htmlClass: 'row',
items: [
{
type: 'section',
htmlClass: 'col-sm-12',
items: [
{
key: 'name',
placeholder: gettext('Name of the {{cookiecutter.panel}} to create.')
},
{
key: 'description',
type: 'textarea'
},
{
key: 'enabled',
type: 'checkbox',
titleMap: [
{value: true, name: ''}
]
}
]
}
]
}
];
ctrl.model = model.spec;
if (model.type === 'update') {
var deferred = api.get{{cookiecutter.panel_func}}(model.spec.id);
deferred.then(onLoad);
function onLoad(response) {
ctrl.model.name = model.spec.name = response.data.name;
ctrl.model.description = model.spec.description = response.data.description;
ctrl.model.enabled = model.spec.enabled = response.data.enabled;
}
}
}
})();

View File

@ -0,0 +1,7 @@
<dl>
<dt translate>Base</dt>
<dd translate>Choose base drink.</dd>
<dt translate>Other options</dt>
<dd translate>Choose favorite options.</dd>
</dl>

View File

@ -0,0 +1,7 @@
<div ng-controller="horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.workflow.{{cookiecutter.panel}}RecipeController as ctrl">
<form name="schemaForm"
sf-schema="ctrl.schema"
sf-form="ctrl.form"
sf-model="ctrl.model">
</form>
</div>

View File

@ -0,0 +1,156 @@
/**
* Licensed under the Apache License, Version 2.0 (the "License"); you may
* not use this file except in compliance with the License. You may obtain
* a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*/
(function() {
'use strict';
/**
* @ngdoc controller
* @name {{cookiecutter.panel}}RecipeController
* @ngController
* @description
* Controller for the {{cookiecutter.panel}} recipe step in create/update workflow
*/
angular
.module('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s')
.controller(
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.workflow.{{cookiecutter.panel}}RecipeController',
{{cookiecutter.panel}}RecipeController);
{{cookiecutter.panel}}RecipeController.$inject = [
'$q',
'horizon.app.core.openstack-service-api.{{cookiecutter.api_module}}',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.model',
'horizon.framework.util.i18n.gettext'
];
function {{cookiecutter.panel}}RecipeController($q, api, model, gettext) {
var ctrl = this;
ctrl.schema = {
type: 'object',
properties: {
size: {
title: gettext('Size'),
type: 'string',
default: 'M'
},
temperature: {
title: gettext('Temperature'),
type: 'string',
default: 'H'
},
base: {
title: gettext('Base'),
type: 'string'
},
flavor: {
title: gettext('Flavor'),
type: 'string'
},
topping: {
title: gettext('Topping'),
type: 'string'
}
},
required: ['base']
};
ctrl.form = [
{
type: 'section',
htmlClass: 'row',
items: [
{
type: 'section',
htmlClass: 'col-sm-12',
items: [
{
key: 'size',
type: 'radiobuttons',
titleMap: [
// if get options from API, call $scope.$broadcast('schemaFormRedraw')
{value: 'S', name: gettext('Small')},
{value: 'M', name: gettext('Medium')},
{value: 'L', name: gettext('Large')},
{value: 'XL', name: gettext('Extra Large')}
]
},
{
key: 'temperature',
type: 'radiobuttons',
titleMap: [
// if get options from API, call $scope.$broadcast('schemaFormRedraw')
{value: 'H', name: gettext('Hot')},
{value: 'I', name: gettext('Ice')}
]
},
{
key: 'base',
type: 'select',
titleMap: [
// if get options from API, call $scope.$broadcast('schemaFormRedraw')
{value: 'blend', name: gettext('House Blend'), group: gettext('Coffee')},
{value: 'mandheling', name: gettext('Mandheling'), group: gettext('Coffee')},
{value: 'colombia', name: gettext('Colombia'), group: gettext('Coffee')},
{value: 'espresso', name: gettext('Espresso'), group: gettext('Coffee')},
{value: 'earl_gray', name: gettext('Earl Gray'), group: gettext('Tea')},
{value: 'darjeeling', name: gettext('Darjeeling'), group: gettext('Tea')},
{value: 'orange_pekoe', name: gettext('Orange Pekoe'), group: gettext('Tea')}
]
},
{
key: 'flavor',
type: 'select',
titleMap: [
// if get options from API, call $scope.$broadcast('schemaFormRedraw')
{value: 'chocolate', name: gettext('Chocolate')},
{value: 'mocha', name: gettext('Mocha')},
{value: 'strawberry', name: gettext('Strawberry')},
{value: 'blueberry', name: gettext('Blueberry')},
{value: 'raspberry', name: gettext('Raspberry')}
]
},
{
key: 'topping',
type: 'checkboxes',
titleMap: [
// if get options from API, call $scope.$broadcast('schemaFormRedraw')
{value: 'clushed_nuts', name: gettext('Clushed Nuts')},
{value: 'whip_cream', name: gettext('Whip Cream')},
{value: 'mixed_serial', name: gettext('Mixed Serial')}
]
}
]
}
]
}
];
ctrl.model = model.spec;
if (model.type === 'update') {
var deferred = api.get{{cookiecutter.panel_func}}(model.spec.id);
deferred.then(onLoad);
function onLoad(response) {
ctrl.model.size = model.spec.size = response.data.size;
ctrl.model.temperature = model.spec.temperature = response.data.temperature;
ctrl.model.base = model.spec.base = response.data.base;
ctrl.model.flavor = model.spec.flavor = response.data.flavor;
ctrl.model.topping = model.spec.topping = response.data.topping;
}
}
}
})();

View File

@ -0,0 +1,88 @@
/**
* Licensed under the Apache License, Version 2.0 (the "License"); you may
* not use this file except in compliance with the License. You may obtain
* a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*/
(function() {
'use strict';
/**
* @ngdoc workflow
* @name horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.workflow
* @description Service for the create/update workflow
*/
angular
.module('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s')
.factory('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.workflow', workflow);
workflow.$inject = [
'horizon.app.core.workflow.factory',
'horizon.dashboard.{{cookiecutter.panel_group}}.basePath',
'horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.model',
'horizon.framework.util.i18n.gettext'
];
function workflow(workflowFactory, basePath, model, gettext) {
var workflow = {
// params
type: '',
// methods
init: init,
save: save
};
function init(type, id) {
var title, btnText;
workflow.type = type;
if (type === 'create') {
title = gettext('Create {{cookiecutter.panel_func}}');
btnText = gettext('Create');
model.init('create');
} else if (type === 'update') {
title = gettext('Update {{cookiecutter.panel_func}}');
btnText = gettext('Update');
model.init('update', id);
}
return workflowFactory({
title: title,
steps: [
{
title: gettext('Info'),
templateUrl: basePath + '{{cookiecutter.panel}}s/workflow/info/info.html',
helpUrl: basePath + '{{cookiecutter.panel}}s/workflow/info/info.help.html',
formName: '{{cookiecutter.panel}}InfoForm'
},
{
title: gettext('Recipe'),
templateUrl: basePath + '{{cookiecutter.panel}}s/workflow/recipe/recipe.html',
helpUrl: basePath + '{{cookiecutter.panel}}s/workflow/recipe/recipe.help.html',
formName: '{{cookiecutter.panel}}RecipeForm'
}
],
btnText: {
finish: btnText
},
btnIcon: {
finish: 'fa fa-check'
}
});
}
function save(){
return model.save();
}
return workflow;
}
})();

View File

@ -15,54 +15,64 @@
(function() {
'use strict';
/**
* @ngdoc model
* @name horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.model
* @description Service for the {{cookiecutter.panel}} model
*/
angular
.module('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s')
.factory('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.{{cookiecutter.panel}}Model', {{cookiecutter.panel}}Model);
.factory('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s.model', model);
{{cookiecutter.panel}}Model.$inject = [
model.$inject = [
'horizon.app.core.openstack-service-api.{{cookiecutter.api_module}}'
];
function {{cookiecutter.panel}}Model(api) {
function model(api) {
var model = {
// params
spec: {},
type: '',
// API methods
// methods
init: init,
create{{cookiecutter.panel_func}}: create{{cookiecutter.panel_func}}
save: save
};
function initSpec() {
function init(type, id) {
model.type = type;
// Reset model
model.spec = {
id: id,
name: null, // text required
description: null, // textarea
enabled: true, // checkbox
size: null, // radio
base: "", // select required
flavor: "", // select
topping: null, // checkboxes
size: 'M', // radio
temperature: 'H', // radio
base: null, // select required
flavor: null, // select
topping: null // checkboxes
};
}
function init() {
// Reset the new {{cookiecutter.panel_func}} spec
initSpec();
}
function create{{cookiecutter.panel_func}}() {
function save() {
var finalSpec = angular.copy(model.spec);
cleanNullProperties(finalSpec);
return api.create{{cookiecutter.panel_func}}(finalSpec);
var id = finalSpec.id;
delete finalSpec.id;
if (model.type === 'create') {
delete finalSpec.id;
return api.create{{cookiecutter.panel_func}}(finalSpec);
} else {
return api.update{{cookiecutter.panel_func}}(id, finalSpec);
}
}
function cleanNullProperties(finalSpec) {
// Initially clean fields that don't have any value.
// Not only "null", blank too.
for (var key in finalSpec) {
if (finalSpec.hasOwnProperty(key) && finalSpec[key] === null
|| finalSpec[key] === "") {
if (finalSpec.hasOwnProperty(key) && finalSpec[key] === null) {
delete finalSpec[key];
}
}

View File

@ -69,6 +69,31 @@
filters: ['noName'],
urlFunction: service.urlFunction
})
.append({
id: 'size',
priority: 1,
filters: ['noValue']
})
.append({
id: 'temperature',
priority: 1,
filters: ['noValue']
})
.append({
id: 'base',
priority: 1,
filters: ['noValue']
})
.append({
id: 'flavor',
priority: 1,
filters: ['noValue']
})
.append({
id: 'topping',
priority: 2,
filters: ['noValue']
})
.append({
id: 'created_at',
priority: 2
@ -84,6 +109,16 @@
'name': 'name',
'singleton': true
})
.append({
'label': gettext('Base'),
'name': 'base',
'singleton': true
})
.append({
'label': gettext('Flavor'),
'name': 'flavor',
'singleton': true
})
.append({
'label': gettext('ID'),
'name': 'id',
@ -95,6 +130,13 @@
return {
id: { label: gettext('ID'), filters: ['noValue'] },
name: { label: gettext('Name'), filters: ['noName'] },
description: { label: gettext('Description'), filters: ['noValue'] },
enabled: { label: gettext('Enabled'), filters: ['yesno'] },
size: { label: gettext('Size'), filters: ['noValue'] },
temperature: { label: gettext('Temperature'), filters: ['noValue'] },
base: { label: gettext('Base'), filters: ['noValue'] },
flavor: { label: gettext('Flavor'), filters: ['noValue'] },
topping: { label: gettext('Topping'), filters: ['noValue'] },
created_at: { label: gettext('Created'), filters: ['simpleDate', 'noValue'] },
updated_at: { label: gettext('Updated'), filters: ['simpleDate', 'noValue'] }
};

View File

@ -1,11 +1,3 @@
describe('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s', function() {
it('should exist', function() {
expect(angular.module('horizon.dashboard.{{cookiecutter.panel_group}}.{{cookiecutter.panel}}s')).toBeDefined();
});
});
})();
/**
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.