Commit ce363853 by Calen Pennington

Make inline editing save to mongo and then update the preview

parent ff002e35
...@@ -327,7 +327,10 @@ def load_preview_module(request, preview_id, descriptor, instance_state, shared_ ...@@ -327,7 +327,10 @@ def load_preview_module(request, preview_id, descriptor, instance_state, shared_
module, module,
"xmodule_edit.html", "xmodule_edit.html",
{ {
'location': descriptor.location.url(),
'editor_content': descriptor.get_html(), 'editor_content': descriptor.get_html(),
'editor_type': descriptor.js_module_name,
'editor_class': descriptor.__class__.__name__,
# TODO (cpennington): Make descriptors know if they have data that can be editng # TODO (cpennington): Make descriptors know if they have data that can be editng
'editable_data': descriptor.definition.get('data'), 'editable_data': descriptor.definition.get('data'),
'editable_class': 'editable' if descriptor.definition.get('data') else '', 'editable_class': 'editable' if descriptor.definition.get('data') else '',
...@@ -396,7 +399,7 @@ def save_item(request): ...@@ -396,7 +399,7 @@ def save_item(request):
export_to_github(course, "CMS Edit", author_string) export_to_github(course, "CMS Edit", author_string)
descriptor = modulestore().get_item(item_location) descriptor = modulestore().get_item(item_location)
preview_html = get_module_previews(request, descriptor) preview_html = get_module_previews(request, descriptor)[0]
return HttpResponse(json.dumps(preview_html)) return HttpResponse(json.dumps(preview_html))
......
...@@ -5,24 +5,12 @@ class CMS.Models.Module extends Backbone.Model ...@@ -5,24 +5,12 @@ class CMS.Models.Module extends Backbone.Model
children: '' children: ''
metadata: {} metadata: {}
loadModule: (element) -> initialize: (attributes) ->
elt = $(element).find('.xmodule_edit').first() @module = attributes.module
@module = XModule.loadModule(elt) @unset('module')
# find the metadata edit region which should be setup server side, delete attributes.module
# so that we can wire up posting back those changes super(attributes)
@metadata_elt = $(element).find('.metadata_edit')
editUrl: ->
"/edit_item?#{$.param(id: @get('id'))}"
save: (args...) -> save: (args...) ->
@set(data: @module.save()) if @module @set(data: @module.save()) if @module
# cdodge: package up metadata which is separated into a number of input fields
# there's probably a better way to do this, but at least this lets me continue to move onwards
if @metadata_elt
_metadata = {}
# walk through the set of elments which have the 'xmetadata_name' attribute and
# build up a object to pass back to the server on the subsequent POST
_metadata[$(el).data("metadata-name")]=el.value for el in $('[data-metadata-name]', @metadata_elt)
@set(metadata: _metadata)
super(args...) super(args...)
class CMS.Views.ModuleEdit extends Backbone.View class CMS.Views.ModuleEdit extends Backbone.View
tagName: 'section' tagName: 'div'
className: 'edit-pane' className: 'xmodule_edit'
events:
'click .cancel': 'cancel'
'click .module-edit': 'editSubmodule'
'click .save-update': 'save'
initialize: -> initialize: ->
@$el.load @model.editUrl(), => @delegate()
@model.loadModule(@el)
@$component_editor = @$el.find('.component-editor')
# Load preview modules @$metadata = @$component_editor.find('.metadata_edit')
XModule.loadModules('display')
@$children = @$el.find('#sortable') delegate: ->
@enableDrag() id = @$el.data('id')
enableDrag: => events = {}
# Enable dragging things in the #sortable div (if there is one) events["click .component-editor[data-id=#{ id }] .cancel-button"] = 'cancel'
if @$children.length > 0 events["click .component-editor[data-id=#{ id }] .save-button"] = 'save'
@$children.sortable( events["click .component-actions[data-id=#{ id }] .edit-button"] = 'edit'
placeholder: "ui-state-highlight"
update: (event, ui) => @delegateEvents(events)
@model.set(children: @$children.find('.module-edit').map(
(idx, el) -> $(el).data('id') metadata: ->
).toArray()) # cdodge: package up metadata which is separated into a number of input fields
) # there's probably a better way to do this, but at least this lets me continue to move onwards
@$children.disableSelection() _metadata = {}
if @$metadata
# walk through the set of elments which have the 'xmetadata_name' attribute and
# build up a object to pass back to the server on the subsequent POST
_metadata[$(el).data("metadata-name")] = el.value for el in $('[data-metadata-name]', @$metadata)
_metadata
save: (event) => save: (event) =>
event.preventDefault() event.preventDefault()
@model.save().done((previews) => @model.save(
metadata: @metadata()
).done((preview) =>
alert("Your changes have been saved.") alert("Your changes have been saved.")
previews_section = @$el.find('.previews').empty()
$.each(previews, (idx, preview) => new_el = $(preview)
preview_wrapper = $('<section/>', class: 'preview').append preview @$el.replaceWith(new_el)
previews_section.append preview_wrapper @$el = new_el
)
@delegate()
XModule.loadModules('display') @model.module = XModule.loadModule(@$el)
XModule.loadModules(@$el)
).fail( -> ).fail( ->
alert("There was an error saving your changes. Please try again.") alert("There was an error saving your changes. Please try again.")
) )
cancel: (event) -> cancel: (event) ->
event.preventDefault() event.preventDefault()
CMS.popView() @$el.removeClass('editing')
@enableDrag() @$component_editor.slideUp(150)
editSubmodule: (event) -> edit: (event) ->
event.preventDefault() event.preventDefault()
previewType = $(event.target).data('preview-type') @$el.addClass('editing')
moduleType = $(event.target).data('type') @$component_editor.slideDown(150)
CMS.pushView new CMS.Views.ModuleEdit
model: new CMS.Models.Module
id: $(event.target).data('id')
type: if moduleType == 'None' then null else moduleType
previewType: if previewType == 'None' then null else previewType
@enableDrag()
...@@ -6,162 +6,175 @@ var $newComponentStep1; ...@@ -6,162 +6,175 @@ var $newComponentStep1;
var $newComponentStep2; var $newComponentStep2;
$(document).ready(function() { $(document).ready(function() {
$body = $('body'); $body = $('body');
$modal = $('.history-modal'); $modal = $('.history-modal');
$modalCover = $('.modal-cover'); $modalCover = $('.modal-cover');
$newComponentItem = $('.new-component-item'); $newComponentItem = $('.new-component-item');
$newComponentStep1 = $('.new-component-step-1'); $newComponentStep1 = $('.new-component-step-1');
$newComponentStep2 = $('.new-component-step-2'); $newComponentStep2 = $('.new-component-step-2');
$newComponentButton = $('.new-component-button'); $newComponentButton = $('.new-component-button');
$('.expand-collapse-icon').bind('click', toggleSubmodules); $(document).bind('XModule.loaded', function(e, element) {
$('.visibility-options').bind('change', setVisibility); if ($(element).hasClass('.xmodule_display')) {
return
$body.delegate('.xmodule_edit .edit-button', 'click', editComponent); }
$body.delegate('.component-editor .save-button, .component-editor .cancel-button', 'click', closeComponentEditor); var previewType = $(element).data('preview-type');
var moduleType = $(element).data('type');
$newComponentButton.bind('click', showNewComponentForm); new CMS.Views.ModuleEdit({
$newComponentStep1.find('.new-component-type a').bind('click', showNewComponentProperties); el: element,
$newComponentStep2.find('.save-button').bind('click', saveNewComponent); model: new CMS.Models.Module({
$newComponentStep2.find('.cancel-button').bind('click', cancelNewComponent); module: $(element).data('module'),
id: $(element).data('id'),
$('.unit-history ol a').bind('click', showHistoryModal); type: moduleType == 'None' ? null : moduleType,
$modal.bind('click', hideHistoryModal); previewType: previewType == 'None' ? null : previewType,
$modalCover.bind('click', hideHistoryModal); })
});
XModule.loadModules('display'); });
XModule.loadModules()
$('.expand-collapse-icon').bind('click', toggleSubmodules);
$('.visibility-options').bind('change', setVisibility);
$newComponentButton.bind('click', showNewComponentForm);
$newComponentStep1.find('.new-component-type a').bind('click', showNewComponentProperties);
$newComponentStep2.find('.save-button').bind('click', saveNewComponent);
$newComponentStep2.find('.cancel-button').bind('click', cancelNewComponent);
$('.unit-history ol a').bind('click', showHistoryModal);
$modal.bind('click', hideHistoryModal);
$modalCover.bind('click', hideHistoryModal);
}); });
function toggleSubmodules(e) { function toggleSubmodules(e) {
e.preventDefault(); e.preventDefault();
$(this).toggleClass('expand').toggleClass('collapse'); $(this).toggleClass('expand').toggleClass('collapse');
$(this).closest('.branch, .window').toggleClass('collapsed'); $(this).closest('.branch, .window').toggleClass('collapsed');
} }
function setVisibility(e) { function setVisibility(e) {
$(this).find('.checked').removeClass('checked'); $(this).find('.checked').removeClass('checked');
$(e.target).closest('.option').addClass('checked'); $(e.target).closest('.option').addClass('checked');
} }
function editComponent(e) { function editComponent(e) {
e.preventDefault(); e.preventDefault();
$(this).closest('.xmodule_edit').addClass('editing').find('.component-editor').slideDown(150); $(this).closest('.xmodule_edit').addClass('editing').find('.component-editor').slideDown(150);
} }
function closeComponentEditor(e) { function closeComponentEditor(e) {
e.preventDefault(); e.preventDefault();
$(this).closest('.xmodule_edit').removeClass('editing').find('.component-editor').slideUp(150); $(this).closest('.xmodule_edit').removeClass('editing').find('.component-editor').slideUp(150);
} }
function showNewComponentForm(e) { function showNewComponentForm(e) {
e.preventDefault(); e.preventDefault();
$newComponentItem.addClass('adding'); $newComponentItem.addClass('adding');
$(this).slideUp(150); $(this).slideUp(150);
$newComponentStep1.slideDown(150); $newComponentStep1.slideDown(150);
} }
function showNewComponentProperties(e) { function showNewComponentProperties(e) {
e.preventDefault(); e.preventDefault();
var displayName; var displayName;
var componentSource; var componentSource;
var selectionRange; var selectionRange;
var $renderedComponent; var $renderedComponent;
switch($(this).attr('data-type')) { switch($(this).attr('data-type')) {
case 'video': case 'video':
displayName = 'Video'; displayName = 'Video';
componentSource = '<video youtube="1.50:___,1.25:___,1.0:___,0.75:___"/>'; componentSource = '<video youtube="1.50:___,1.25:___,1.0:___,0.75:___"/>';
selectionRange = [21, 24]; selectionRange = [21, 24];
$renderedComponent = $('<div class="rendered-component"><div class="video-unit"><img src="images/video-module.png"></div></div>'); $renderedComponent = $('<div class="rendered-component"><div class="video-unit"><img src="images/video-module.png"></div></div>');
break; break;
case 'textbook': case 'textbook':
displayName = 'Textbook'; displayName = 'Textbook';
componentSource = '<customtag page="___"><impl>book</impl></customtag>'; componentSource = '<customtag page="___"><impl>book</impl></customtag>';
selectionRange = [17, 20]; selectionRange = [17, 20];
$renderedComponent = $('<div class="rendered-component"><p><span class="textbook-icon"></span>More information given in the text.</p></div>'); $renderedComponent = $('<div class="rendered-component"><p><span class="textbook-icon"></span>More information given in the text.</p></div>');
break; break;
case 'slide': case 'slide':
displayName = 'Slide'; displayName = 'Slide';
componentSource = '<customtag page="___"><customtag lecnum="___"><impl>slides</impl></customtag>'; componentSource = '<customtag page="___"><customtag lecnum="___"><impl>slides</impl></customtag>';
selectionRange = [17, 20]; selectionRange = [17, 20];
$renderedComponent = $('<div class="rendered-component"><p><span class="slides-icon"></span>Lecture Slides Handout [Clean] [Annotated]</p></div>'); $renderedComponent = $('<div class="rendered-component"><p><span class="slides-icon"></span>Lecture Slides Handout [Clean] [Annotated]</p></div>');
break; break;
case 'discussion': case 'discussion':
displayName = 'Discussion'; displayName = 'Discussion';
componentSource = '<discussion for="___" id="___" discussion_category="___"/>'; componentSource = '<discussion for="___" id="___" discussion_category="___"/>';
selectionRange = [17, 20]; selectionRange = [17, 20];
$renderedComponent = $('<div class="rendered-component"><div class="discussion-unit"><img src="images/discussion-module.png"></div></div>'); $renderedComponent = $('<div class="rendered-component"><div class="discussion-unit"><img src="images/discussion-module.png"></div></div>');
break; break;
case 'problem': case 'problem':
displayName = 'Problem'; displayName = 'Problem';
componentSource = '<problem>___</problem>'; componentSource = '<problem>___</problem>';
selectionRange = [9, 12]; selectionRange = [9, 12];
$renderedComponent = $('<div class="rendered-component"></div>'); $renderedComponent = $('<div class="rendered-component"></div>');
break; break;
case 'freeform': case 'freeform':
displayName = 'Freeform HTML'; displayName = 'Freeform HTML';
componentSource = ''; componentSource = '';
selectionRange = [0, 0]; selectionRange = [0, 0];
$renderedComponent = $('<div class="rendered-component"></div>'); $renderedComponent = $('<div class="rendered-component"></div>');
break; break;
} }
$newComponentItem.prepend($renderedComponent); $newComponentItem.prepend($renderedComponent);
$renderedComponent.slideDown(250); $renderedComponent.slideDown(250);
$newComponentStep2.find('h5').html('Edit ' + displayName + ' Component'); $newComponentStep2.find('h5').html('Edit ' + displayName + ' Component');
$newComponentStep2.find('textarea').html(componentSource); $newComponentStep2.find('textarea').html(componentSource);
setTimeout(function() { setTimeout(function() {
$newComponentStep2.find('textarea').focus().get(0).setSelectionRange(selectionRange[0], selectionRange[1]); $newComponentStep2.find('textarea').focus().get(0).setSelectionRange(selectionRange[0], selectionRange[1]);
}, 10); }, 10);
$newComponentStep1.slideUp(250); $newComponentStep1.slideUp(250);
$newComponentStep2.slideDown(250); $newComponentStep2.slideDown(250);
} }
function cancelNewComponent(e) { function cancelNewComponent(e) {
e.preventDefault(); e.preventDefault();
$newComponentStep2.slideUp(250); $newComponentStep2.slideUp(250);
$newComponentButton.slideDown(250); $newComponentButton.slideDown(250);
$newComponentItem.removeClass('adding'); $newComponentItem.removeClass('adding');
$newComponentItem.find('.rendered-component').remove(); $newComponentItem.find('.rendered-component').remove();
} }
function saveNewComponent(e) { function saveNewComponent(e) {
e.preventDefault(); e.preventDefault();
var $newComponent = $newComponentItem.clone(); var $newComponent = $newComponentItem.clone();
$newComponent.removeClass('adding').removeClass('new-component-item'); $newComponent.removeClass('adding').removeClass('new-component-item');
$newComponent.find('.new-component-step-2').removeClass('new-component-step-2').addClass('component-editor'); $newComponent.find('.new-component-step-2').removeClass('new-component-step-2').addClass('component-editor');
setTimeout(function() { setTimeout(function() {
$newComponent.find('.component-editor').slideUp(250); $newComponent.find('.component-editor').slideUp(250);
}, 10); }, 10);
$newComponent.append('<div class="component-actions"><a href="#" class="edit-button"><span class="edit-icon white"></span>Edit</a><a href="#" class="delete-button"><span class="delete-icon white"></span>Delete</a> </div><a href="#" class="drag-handle"></a>'); $newComponent.append('<div class="component-actions"><a href="#" class="edit-button"><span class="edit-icon white"></span>Edit</a><a href="#" class="delete-button"><span class="delete-icon white"></span>Delete</a> </div><a href="#" class="drag-handle"></a>');
$newComponent.find('.new-component-step-1').remove(); $newComponent.find('.new-component-step-1').remove();
$newComponent.find('.new-component-button').remove(); $newComponent.find('.new-component-button').remove();
$newComponentStep2.slideUp(250); $newComponentStep2.slideUp(250);
$newComponentButton.slideDown(250); $newComponentButton.slideDown(250);
$newComponentItem.removeClass('adding'); $newComponentItem.removeClass('adding');
$newComponentItem.find('.rendered-component').remove(); $newComponentItem.find('.rendered-component').remove();
$newComponentItem.before($newComponent); $newComponentItem.before($newComponent);
} }
function showHistoryModal(e) { function showHistoryModal(e) {
e.preventDefault(); e.preventDefault();
$modal.show(); $modal.show();
$modalCover.show(); $modalCover.show();
} }
function hideHistoryModal(e) { function hideHistoryModal(e) {
e.preventDefault(); e.preventDefault();
$modal.hide(); $modal.hide();
$modalCover.hide(); $modalCover.hide();
} }
......
...@@ -84,7 +84,7 @@ class @Sequence ...@@ -84,7 +84,7 @@ class @Sequence
@mark_active new_position @mark_active new_position
@$('#seq_content').html @contents.eq(new_position - 1).text() @$('#seq_content').html @contents.eq(new_position - 1).text()
XModule.loadModules('display', @$('#seq_content')) XModule.loadModules(@$('#seq_content'))
MathJax.Hub.Queue(["Typeset", MathJax.Hub, "seq_content"]) # NOTE: Actually redundant. Some other MathJax call also being performed MathJax.Hub.Queue(["Typeset", MathJax.Hub, "seq_content"]) # NOTE: Actually redundant. Some other MathJax call also being performed
window.update_schematics() # For embedded circuit simulator exercises in 6.002x window.update_schematics() # For embedded circuit simulator exercises in 6.002x
......
@XModule = @XModule =
### ###
Load a single module (either an edit module or a display module) Load a single module (either an edit module or a display module)
from the supplied element, which should have a data-type attribute from the supplied element, which should have a data-type attribute
specifying the class to load specifying the class to load
### ###
loadModule: (element) -> loadModule: (element) ->
moduleType = $(element).data('type') moduleType = $(element).data('type')
if moduleType == 'None' if moduleType == 'None'
return return
try try
new window[moduleType](element) $(element).data('module', new window[moduleType](element))
catch error $(document).trigger('XModule.loaded', [element])
console.error "Unable to load #{moduleType}: #{error.message}" if console catch error
console.error "Unable to load #{moduleType}: #{error.message}" if console
### ###
Load all modules on the page of the specified type. Load all modules on the page of the specified type.
If container is provided, only load modules inside that element If container is provided, only load modules inside that element
Type is one of 'display' or 'edit' Type is one of 'display' or 'edit'
### ###
loadModules: (type, container) -> loadModules: (container) ->
selector = ".xmodule_#{type}" selector = ".xmodule_edit, .xmodule_display"
if container? if container?
modules = $(container).find(selector) modules = $(container).find(selector)
else else
modules = $(selector) modules = $(selector)
modules.each (idx, element) -> XModule.loadModule element modules.each (idx, element) -> XModule.loadModule element
<div class="xmodule_edit xmodule_${class_} ${editable_class}" data-type="${module_name}"> <div class="xmodule_edit xmodule_${editor_class} ${editable_class}" data-type="${editor_type}" data-id="${location}">
<%include file="xmodule_display.html"/> <%include file="xmodule_display.html"/>
% if editable_data: % if editable_data:
<div class="component-actions"> <div class="component-actions" data-id="${location}">
<a href="#" class="edit-button"><span class="edit-icon white"></span>Edit</a> <a href="#" class="edit-button"><span class="edit-icon white"></span>Edit</a>
<a href="#" class="delete-button"><span class="delete-icon white"></span>Delete</a> <a href="#" class="delete-button"><span class="delete-icon white"></span>Delete</a>
</div> </div>
<div class="component-editor"> <div class="component-editor" data-id="${location}">
${editor_content} ${editor_content}
<a href="#" class="save-button">Save</a> <a href="#" class="save-button">Save</a>
<a href="#" class="cancel-button">Cancel</a> <a href="#" class="cancel-button">Cancel</a>
......
...@@ -11,7 +11,7 @@ class @Courseware ...@@ -11,7 +11,7 @@ class @Courseware
new Courseware new Courseware
render: -> render: ->
XModule.loadModules('display') XModule.loadModules()
$('.course-content .histogram').each -> $('.course-content .histogram').each ->
id = $(this).attr('id').replace(/histogram_/, '') id = $(this).attr('id').replace(/histogram_/, '')
try try
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment