* refactor: deprecates ModuleSystem.render_template in favor of the added MakoSystem render_template method. Related changes: * Adds the MakoService to the StudioEditModuleRuntime, PreviewModuleSystem, LmsModuleSystem, and XBlockRuntime * MakoService constructor takes a `namespace_prefix` string, so that the CMS PreviewModuleSystem can render to LMS templates, without needing the special render_from_lms helper method. * ModuleSystem.render_template becomes a read-only property, so the constructor calls and test module systems are updated accordingly. * Adds tests for the MakoService and module system shims. (cherry picked from commit457f959356) * refactor: use MakoService.render_template to remove deprecation warnings from block code. (cherry picked from commit8d62d337f5) * refactor: use MakoService.render_template to remove deprecation warnings from test code. (cherry picked from commit26b43465a4) * test: Adds a test to verify the bug introduced by the previous changes The AuthoringMixin is automatically added to all XBlocks (see settings.XBLOCK_MIXINS), and AuthoringMixin.visibility_view expects the "mako" service. This test verifies the bug by testing the PureXBlock, which does not require the "mako" service, and so fails when the visibility_view is rendered. * fix: AuthoringMixin needs mako service which fixes the visibility_view for XBlocks which don't explicitly require the mako service. Also removes the unneeded class property _services_requested from AuthoringMixin and StudioEditableBlock. This property is better provided by the XBlockMixin class.
275 lines
9.0 KiB
Python
275 lines
9.0 KiB
Python
"""Poll module is ungraded xmodule used by students to
|
|
to do set of polls.
|
|
|
|
On the client side we show:
|
|
If student does not yet anwered - Question with set of choices.
|
|
If student have answered - Question with statistics for each answers.
|
|
"""
|
|
|
|
|
|
import html
|
|
import json
|
|
import logging
|
|
from collections import OrderedDict
|
|
from copy import deepcopy
|
|
|
|
from pkg_resources import resource_string
|
|
from web_fragments.fragment import Fragment
|
|
|
|
from lxml import etree
|
|
from openedx.core.djangolib.markup import Text, HTML
|
|
from xblock.core import XBlock
|
|
from xblock.fields import Boolean, Dict, List, Scope, String # lint-amnesty, pylint: disable=wrong-import-order
|
|
from xmodule.mako_module import MakoTemplateBlockBase
|
|
from xmodule.stringify import stringify_children
|
|
from xmodule.util.xmodule_django import add_webpack_to_fragment
|
|
from xmodule.x_module import (
|
|
HTMLSnippet,
|
|
ResourceTemplates,
|
|
shim_xmodule_js,
|
|
XModuleMixin,
|
|
XModuleDescriptorToXBlockMixin,
|
|
XModuleToXBlockMixin,
|
|
)
|
|
from xmodule.xml_module import XmlMixin
|
|
|
|
log = logging.getLogger(__name__)
|
|
_ = lambda text: text
|
|
|
|
|
|
@XBlock.needs('mako')
|
|
class PollBlock(
|
|
MakoTemplateBlockBase,
|
|
XmlMixin,
|
|
XModuleDescriptorToXBlockMixin,
|
|
XModuleToXBlockMixin,
|
|
HTMLSnippet,
|
|
ResourceTemplates,
|
|
XModuleMixin,
|
|
): # pylint: disable=abstract-method
|
|
"""Poll Module"""
|
|
# Name of poll to use in links to this poll
|
|
display_name = String(
|
|
help=_("The display name for this component."),
|
|
scope=Scope.settings
|
|
)
|
|
|
|
voted = Boolean(
|
|
help=_("Whether this student has voted on the poll"),
|
|
scope=Scope.user_state,
|
|
default=False
|
|
)
|
|
poll_answer = String(
|
|
help=_("Student answer"),
|
|
scope=Scope.user_state,
|
|
default=''
|
|
)
|
|
poll_answers = Dict(
|
|
help=_("Poll answers from all students"),
|
|
scope=Scope.user_state_summary
|
|
)
|
|
|
|
# List of answers, in the form {'id': 'some id', 'text': 'the answer text'}
|
|
answers = List(
|
|
help=_("Poll answers from xml"),
|
|
scope=Scope.content,
|
|
default=[]
|
|
)
|
|
|
|
question = String(
|
|
help=_("Poll question"),
|
|
scope=Scope.content,
|
|
default=''
|
|
)
|
|
|
|
resources_dir = None
|
|
uses_xmodule_styles_setup = True
|
|
|
|
preview_view_js = {
|
|
'js': [
|
|
resource_string(__name__, 'js/src/javascript_loader.js'),
|
|
resource_string(__name__, 'js/src/poll/poll.js'),
|
|
resource_string(__name__, 'js/src/poll/poll_main.js')
|
|
],
|
|
'xmodule_js': resource_string(__name__, 'js/src/xmodule.js'),
|
|
}
|
|
preview_view_css = {
|
|
'scss': [
|
|
resource_string(__name__, 'css/poll/display.scss')
|
|
],
|
|
}
|
|
|
|
# There is no studio_view() for this XBlock but this is needed to make the
|
|
# the static_content command happy.
|
|
studio_view_js = {
|
|
'js': [],
|
|
'xmodule_js': resource_string(__name__, 'js/src/xmodule.js')
|
|
}
|
|
|
|
studio_view_css = {
|
|
'scss': []
|
|
}
|
|
|
|
def handle_ajax(self, dispatch, data): # lint-amnesty, pylint: disable=unused-argument
|
|
"""Ajax handler.
|
|
|
|
Args:
|
|
dispatch: string request slug
|
|
data: dict request data parameters
|
|
|
|
Returns:
|
|
json string
|
|
"""
|
|
if dispatch in self.poll_answers and not self.voted:
|
|
# FIXME: fix this, when xblock will support mutable types.
|
|
# Now we use this hack.
|
|
temp_poll_answers = self.poll_answers
|
|
temp_poll_answers[dispatch] += 1
|
|
self.poll_answers = temp_poll_answers
|
|
|
|
self.voted = True
|
|
self.poll_answer = dispatch
|
|
return json.dumps({'poll_answers': self.poll_answers,
|
|
'total': sum(self.poll_answers.values()),
|
|
'callback': {'objectName': 'Conditional'}
|
|
})
|
|
elif dispatch == 'get_state':
|
|
return json.dumps({'poll_answer': self.poll_answer,
|
|
'poll_answers': self.poll_answers,
|
|
'total': sum(self.poll_answers.values())
|
|
})
|
|
elif dispatch == 'reset_poll' and self.voted and \
|
|
self.xml_attributes.get('reset', 'True').lower() != 'false':
|
|
self.voted = False
|
|
|
|
# FIXME: fix this, when xblock will support mutable types.
|
|
# Now we use this hack.
|
|
temp_poll_answers = self.poll_answers
|
|
temp_poll_answers[self.poll_answer] -= 1
|
|
self.poll_answers = temp_poll_answers
|
|
|
|
self.poll_answer = ''
|
|
return json.dumps({'status': 'success'})
|
|
else: # return error message
|
|
return json.dumps({'error': 'Unknown Command!'})
|
|
|
|
def student_view(self, _context):
|
|
"""
|
|
Renders the student view.
|
|
"""
|
|
fragment = Fragment()
|
|
params = {
|
|
'element_id': self.location.html_id(),
|
|
'element_class': self.location.block_type,
|
|
'ajax_url': self.ajax_url,
|
|
'configuration_json': self.dump_poll(),
|
|
}
|
|
fragment.add_content(self.runtime.service(self, 'mako').render_template('poll.html', params))
|
|
add_webpack_to_fragment(fragment, 'PollBlockPreview')
|
|
shim_xmodule_js(fragment, 'Poll')
|
|
return fragment
|
|
|
|
def dump_poll(self):
|
|
"""Dump poll information.
|
|
|
|
Returns:
|
|
string - Serialize json.
|
|
"""
|
|
# FIXME: hack for resolving caching `default={}` during definition
|
|
# poll_answers field
|
|
if self.poll_answers is None:
|
|
self.poll_answers = {}
|
|
|
|
answers_to_json = OrderedDict()
|
|
|
|
# FIXME: fix this, when xblock support mutable types.
|
|
# Now we use this hack.
|
|
temp_poll_answers = self.poll_answers
|
|
|
|
# Fill self.poll_answers, prepare data for template context.
|
|
for answer in self.answers:
|
|
# Set default count for answer = 0.
|
|
if answer['id'] not in temp_poll_answers:
|
|
temp_poll_answers[answer['id']] = 0
|
|
answers_to_json[answer['id']] = html.escape(answer['text'], quote=False)
|
|
self.poll_answers = temp_poll_answers
|
|
|
|
return json.dumps({
|
|
'answers': answers_to_json,
|
|
'question': html.escape(self.question, quote=False),
|
|
# to show answered poll after reload:
|
|
'poll_answer': self.poll_answer,
|
|
'poll_answers': self.poll_answers if self.voted else {},
|
|
'total': sum(self.poll_answers.values()) if self.voted else 0,
|
|
'reset': str(self.xml_attributes.get('reset', 'true')).lower()
|
|
})
|
|
|
|
_tag_name = 'poll_question'
|
|
_child_tag_name = 'answer'
|
|
|
|
@classmethod
|
|
def definition_from_xml(cls, xml_object, system):
|
|
"""Pull out the data into dictionary.
|
|
|
|
Args:
|
|
xml_object: xml from file.
|
|
system: `system` object.
|
|
|
|
Returns:
|
|
(definition, children) - tuple
|
|
definition - dict:
|
|
{
|
|
'answers': <List of answers>,
|
|
'question': <Question string>
|
|
}
|
|
"""
|
|
# Check for presense of required tags in xml.
|
|
if len(xml_object.xpath(cls._child_tag_name)) == 0:
|
|
raise ValueError("Poll_question definition must include \
|
|
at least one 'answer' tag")
|
|
|
|
xml_object_copy = deepcopy(xml_object)
|
|
answers = []
|
|
for element_answer in xml_object_copy.findall(cls._child_tag_name):
|
|
answer_id = element_answer.get('id', None)
|
|
if answer_id:
|
|
answers.append({
|
|
'id': answer_id,
|
|
'text': stringify_children(element_answer)
|
|
})
|
|
xml_object_copy.remove(element_answer)
|
|
|
|
definition = {
|
|
'answers': answers,
|
|
'question': stringify_children(xml_object_copy)
|
|
}
|
|
children = []
|
|
|
|
return (definition, children)
|
|
|
|
def definition_to_xml(self, resource_fs):
|
|
"""Return an xml element representing to this definition."""
|
|
poll_str = HTML('<{tag_name}>{text}</{tag_name}>').format(
|
|
tag_name=self._tag_name, text=self.question)
|
|
xml_object = etree.fromstring(poll_str)
|
|
xml_object.set('display_name', self.display_name)
|
|
|
|
def add_child(xml_obj, answer): # lint-amnesty, pylint: disable=unused-argument
|
|
# Escape answer text before adding to xml tree.
|
|
answer_text = str(answer['text'])
|
|
child_str = Text('{tag_begin}{text}{tag_end}').format(
|
|
tag_begin=HTML('<{tag_name} id="{id}">').format(
|
|
tag_name=self._child_tag_name,
|
|
id=answer['id']
|
|
),
|
|
text=answer_text,
|
|
tag_end=HTML('</{tag_name}>').format(tag_name=self._child_tag_name)
|
|
)
|
|
child_node = etree.fromstring(child_str)
|
|
xml_object.append(child_node)
|
|
|
|
for answer in self.answers:
|
|
add_child(xml_object, answer)
|
|
|
|
return xml_object
|